1 /*- 2 * Copyright (c) 1999 Jonathan Lemon 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * 2. Redistributions in binary form must reproduce the above copyright 11 * notice, this list of conditions and the following disclaimer in the 12 * documentation and/or other materials provided with the distribution. 13 * 14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 24 * SUCH DAMAGE. 25 * 26 * $FreeBSD$ 27 */ 28 29 /* 30 * Disk driver for Compaq SMART RAID adapters. 31 */ 32 33 #include <sys/param.h> 34 #include <sys/systm.h> 35 #include <sys/malloc.h> 36 #include <sys/kernel.h> 37 38 #include <sys/buf.h> 39 #include <sys/bus.h> 40 #include <sys/conf.h> 41 #include <sys/devicestat.h> 42 #include <sys/disklabel.h> 43 #include <sys/diskslice.h> 44 45 #if NPCI > 0 46 #include <machine/bus_memio.h> 47 #endif 48 #include <machine/bus_pio.h> 49 #include <machine/bus.h> 50 #include <machine/clock.h> 51 #include <sys/rman.h> 52 53 #include <dev/ida/idareg.h> 54 #include <dev/ida/idavar.h> 55 56 /* prototypes */ 57 static void id_drvinit(void); 58 static int idprobe(device_t dev); 59 static int idattach(device_t dev); 60 61 static d_open_t idopen; 62 static d_close_t idclose; 63 static d_strategy_t idstrategy; 64 static d_ioctl_t idioctl; 65 static d_psize_t idsize; 66 67 #define ID_BDEV_MAJOR 29 68 #define ID_CDEV_MAJOR 109 69 70 #define WD_BDEV_MAJOR 0 71 #define WD_CDEV_MAJOR 3 72 73 static struct cdevsw id_cdevsw = { 74 /* open */ idopen, 75 /* close */ idclose, 76 /* read */ physread, 77 /* write */ physwrite, 78 /* ioctl */ idioctl, 79 /* poll */ nopoll, 80 /* mmap */ nommap, 81 /* strategy */ idstrategy, 82 /* name */ "id", 83 /* maj */ ID_CDEV_MAJOR, 84 /* dump */ nodump, 85 /* psize */ idsize, 86 /* flags */ D_DISK, 87 /* bmaj */ ID_BDEV_MAJOR 88 }; 89 static struct cdevsw stolen_cdevsw; 90 91 static devclass_t id_devclass; 92 93 static device_method_t id_methods[] = { 94 DEVMETHOD(device_probe, idprobe), 95 DEVMETHOD(device_attach, idattach), 96 { 0, 0 } 97 }; 98 99 static driver_t id_driver = { 100 "id", 101 id_methods, 102 sizeof(struct id_softc) 103 }; 104 105 static __inline struct id_softc * 106 idgetsoftc(dev_t dev) 107 { 108 int unit; 109 110 unit = dkunit(dev); 111 return ((struct id_softc *)devclass_get_softc(id_devclass, unit)); 112 } 113 114 static int 115 idopen(dev_t dev, int flags, int fmt, struct proc *p) 116 { 117 struct id_softc *drv; 118 struct disklabel label; 119 int error; 120 121 drv = idgetsoftc(dev); 122 if (drv == NULL) 123 return (ENXIO); 124 125 /* XXX block against race where > 1 person is reading label? */ 126 127 bzero(&label, sizeof(label)); 128 label.d_type = DTYPE_SCSI; /* XXX should this be DTYPE_RAID? */ 129 /* 130 strncpy(label.d_typename, ... 131 strncpy(label.d_packname, ... 132 */ 133 label.d_secsize = drv->secsize; 134 label.d_nsectors = drv->sectors; 135 label.d_ntracks = drv->heads; 136 label.d_ncylinders = drv->cylinders; 137 label.d_secpercyl = drv->sectors * drv->heads; 138 label.d_secperunit = drv->secperunit; 139 140 /* Initialize slice tables. */ 141 error = dsopen(dev, fmt, 0, &drv->slices, &label); 142 143 return (error); 144 } 145 146 static int 147 idclose(dev_t dev, int flags, int fmt, struct proc *p) 148 { 149 struct id_softc *drv; 150 151 drv = idgetsoftc(dev); 152 if (drv == NULL) 153 return (ENXIO); 154 dsclose(dev, fmt, drv->slices); 155 return (0); 156 } 157 158 static int 159 idioctl(dev_t dev, u_long cmd, caddr_t addr, int32_t flag, struct proc *p) 160 { 161 struct id_softc *drv; 162 int error; 163 164 drv = idgetsoftc(dev); 165 if (drv == NULL) 166 return (ENXIO); 167 168 error = dsioctl(dev, cmd, addr, flag, &drv->slices); 169 170 if (error == ENOIOCTL) 171 return (ENOTTY); 172 173 return (error); 174 } 175 176 static int 177 idsize(dev_t dev) 178 { 179 struct id_softc *drv; 180 181 drv = idgetsoftc(dev); 182 if (drv == NULL) 183 return (ENXIO); 184 return (dssize(dev, &drv->slices)); 185 } 186 187 /* 188 * Read/write routine for a buffer. Finds the proper unit, range checks 189 * arguments, and schedules the transfer. Does not wait for the transfer 190 * to complete. Multi-page transfers are supported. All I/O requests must 191 * be a multiple of a sector in length. 192 */ 193 static void 194 idstrategy(struct buf *bp) 195 { 196 struct id_softc *drv; 197 int s; 198 199 drv = idgetsoftc(bp->b_dev); 200 if (drv == NULL) { 201 bp->b_error = EINVAL; 202 goto bad; 203 } 204 205 if (dscheck(bp, drv->slices) <= 0) 206 goto done; 207 208 /* 209 * software write protect check 210 */ 211 if (drv->flags & DRV_WRITEPROT && (bp->b_flags & B_READ) == 0) { 212 bp->b_error = EROFS; 213 goto bad; 214 } 215 216 /* 217 * If it's a null transfer, return immediately 218 */ 219 if (bp->b_bcount == 0) 220 goto done; 221 222 bp->b_driver1 = drv; 223 s = splbio(); 224 devstat_start_transaction(&drv->stats); 225 ida_submit_buf(drv->controller, bp); 226 splx(s); 227 return; 228 229 bad: 230 bp->b_flags |= B_ERROR; 231 232 done: 233 /* 234 * Correctly set the buf to indicate a completed transfer 235 */ 236 bp->b_resid = bp->b_bcount; 237 biodone(bp); 238 return; 239 } 240 241 void 242 id_intr(struct buf *bp) 243 { 244 struct id_softc *drv = (struct id_softc *)bp->b_driver1; 245 246 if (bp->b_flags & B_ERROR) 247 bp->b_error = EIO; 248 else 249 bp->b_resid = 0; 250 251 devstat_end_transaction_buf(&drv->stats, bp); 252 biodone(bp); 253 } 254 255 static void 256 id_drvinit(void) 257 { 258 static int devsw_installed = 0; 259 260 if (devsw_installed) 261 return; /* XXX is this needed? */ 262 263 cdevsw_add(&id_cdevsw); 264 stolen_cdevsw = id_cdevsw; 265 stolen_cdevsw.d_maj = WD_CDEV_MAJOR; 266 stolen_cdevsw.d_bmaj = WD_BDEV_MAJOR; 267 cdevsw_add(&stolen_cdevsw); 268 devsw_installed = 1; 269 } 270 271 static int 272 idprobe(device_t dev) 273 { 274 275 device_set_desc(dev, "Compaq Logical Drive"); 276 return (0); 277 } 278 279 static int 280 idattach(device_t dev) 281 { 282 struct ida_drive_info dinfo; 283 struct id_softc *drv; 284 device_t parent; 285 int error; 286 287 id_drvinit(); 288 drv = (struct id_softc *)device_get_softc(dev); 289 parent = device_get_parent(dev); 290 drv->controller = (struct ida_softc *)device_get_softc(parent); 291 drv->unit = device_get_unit(dev); 292 293 error = ida_command(drv->controller, CMD_GET_LOG_DRV_INFO, 294 &dinfo, sizeof(dinfo), drv->unit, DMA_DATA_IN); 295 if (error) { 296 device_printf(dev, "CMD_GET_LOG_DRV_INFO failed\n"); 297 return (ENXIO); 298 } 299 300 drv->cylinders = dinfo.ncylinders; 301 drv->heads = dinfo.nheads; 302 drv->sectors = dinfo.nsectors; 303 drv->secsize = dinfo.secsize; 304 drv->secperunit = dinfo.secperunit; 305 306 /* XXX 307 * other initialization 308 */ 309 device_printf(dev, "%uMB (%u sectors), blocksize=%d\n", 310 drv->secperunit / ((1024 * 1024) / drv->secsize), 311 drv->secperunit, drv->secsize); 312 313 devstat_add_entry(&drv->stats, "id", drv->unit, drv->secsize, 314 DEVSTAT_NO_ORDERED_TAGS, 315 DEVSTAT_TYPE_STORARRAY| DEVSTAT_TYPE_IF_OTHER, 316 DEVSTAT_PRIORITY_ARRAY); 317 318 return (0); 319 } 320 321 DRIVER_MODULE(id, ida, id_driver, id_devclass, 0, 0); 322