1 /*- 2 * Copyright (c) 2018 Microsemi Corporation. 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 27 /* $FreeBSD$ */ 28 29 /* 30 * Driver for the Microsemi Smart storage controllers 31 */ 32 33 #include "smartpqi_includes.h" 34 #include "smartpqi_prototypes.h" 35 36 /* 37 * Supported devices 38 */ 39 struct pqi_ident 40 { 41 u_int16_t vendor; 42 u_int16_t device; 43 u_int16_t subvendor; 44 u_int16_t subdevice; 45 int hwif; 46 char *desc; 47 } pqi_identifiers[] = { 48 /* (MSCC PM8205 8x12G based) */ 49 {0x9005, 0x028f, 0x103c, 0x600, PQI_HWIF_SRCV, "P408i-p SR Gen10"}, 50 {0x9005, 0x028f, 0x103c, 0x601, PQI_HWIF_SRCV, "P408e-p SR Gen10"}, 51 {0x9005, 0x028f, 0x103c, 0x602, PQI_HWIF_SRCV, "P408i-a SR Gen10"}, 52 {0x9005, 0x028f, 0x103c, 0x603, PQI_HWIF_SRCV, "P408i-c SR Gen10"}, 53 {0x9005, 0x028f, 0x1028, 0x1FE0, PQI_HWIF_SRCV, "SmartRAID 3162-8i/eDell"}, 54 {0x9005, 0x028f, 0x9005, 0x608, PQI_HWIF_SRCV, "SmartRAID 3162-8i/e"}, 55 {0x9005, 0x028f, 0x103c, 0x609, PQI_HWIF_SRCV, "P408i-sb SR G10"}, 56 57 /* (MSCC PM8225 8x12G based) */ 58 {0x9005, 0x028f, 0x103c, 0x650, PQI_HWIF_SRCV, "E208i-p SR Gen10"}, 59 {0x9005, 0x028f, 0x103c, 0x651, PQI_HWIF_SRCV, "E208e-p SR Gen10"}, 60 {0x9005, 0x028f, 0x103c, 0x652, PQI_HWIF_SRCV, "E208i-c SR Gen10"}, 61 {0x9005, 0x028f, 0x103c, 0x654, PQI_HWIF_SRCV, "E208i-a SR Gen10"}, 62 {0x9005, 0x028f, 0x103c, 0x655, PQI_HWIF_SRCV, "P408e-m SR Gen10"}, 63 64 /* (MSCC PM8221 8x12G based) */ 65 {0x9005, 0x028f, 0x103c, 0x700, PQI_HWIF_SRCV, "P204i-c SR Gen10"}, 66 {0x9005, 0x028f, 0x103c, 0x701, PQI_HWIF_SRCV, "P204i-b SR Gen10"}, 67 68 /* (MSCC PM8204 8x12G based) */ 69 {0x9005, 0x028f, 0x9005, 0x800, PQI_HWIF_SRCV, "SmartRAID 3154-8i"}, 70 {0x9005, 0x028f, 0x9005, 0x801, PQI_HWIF_SRCV, "SmartRAID 3152-8i"}, 71 {0x9005, 0x028f, 0x9005, 0x802, PQI_HWIF_SRCV, "SmartRAID 3151-4i"}, 72 {0x9005, 0x028f, 0x9005, 0x803, PQI_HWIF_SRCV, "SmartRAID 3101-4i"}, 73 {0x9005, 0x028f, 0x9005, 0x804, PQI_HWIF_SRCV, "SmartRAID 3154-8e"}, 74 {0x9005, 0x028f, 0x9005, 0x805, PQI_HWIF_SRCV, "SmartRAID 3102-8i"}, 75 {0x9005, 0x028f, 0x9005, 0x806, PQI_HWIF_SRCV, "SmartRAID 3100"}, 76 {0x9005, 0x028f, 0x9005, 0x807, PQI_HWIF_SRCV, "SmartRAID 3162-8i"}, 77 {0x9005, 0x028f, 0x152d, 0x8a22, PQI_HWIF_SRCV, "QS-8204-8i"}, 78 {0x9005, 0x028f, 0x193d, 0xf460, PQI_HWIF_SRCV, "UN RAID P460-M4"}, 79 {0x9005, 0x028f, 0x193d, 0xf461, PQI_HWIF_SRCV, "UN RAID P460-B4"}, 80 {0x9005, 0x028f, 0x1bd4, 0x004b, PQI_HWIF_SRCV, "INSPUR RAID PM8204-2GB"}, 81 {0x9005, 0x028f, 0x1bd4, 0x004c, PQI_HWIF_SRCV, "INSPUR RAID PM8204-4GB"}, 82 83 /* (MSCC PM8222 8x12G based) */ 84 {0x9005, 0x028f, 0x9005, 0x900, PQI_HWIF_SRCV, "SmartHBA 2100-8i"}, 85 {0x9005, 0x028f, 0x9005, 0x901, PQI_HWIF_SRCV, "SmartHBA 2100-4i"}, 86 {0x9005, 0x028f, 0x9005, 0x902, PQI_HWIF_SRCV, "HBA 1100-8i"}, 87 {0x9005, 0x028f, 0x9005, 0x903, PQI_HWIF_SRCV, "HBA 1100-4i"}, 88 {0x9005, 0x028f, 0x9005, 0x904, PQI_HWIF_SRCV, "SmartHBA 2100-8e"}, 89 {0x9005, 0x028f, 0x9005, 0x905, PQI_HWIF_SRCV, "HBA 1100-8e"}, 90 {0x9005, 0x028f, 0x9005, 0x906, PQI_HWIF_SRCV, "SmartHBA 2100-4i4e"}, 91 {0x9005, 0x028f, 0x9005, 0x907, PQI_HWIF_SRCV, "HBA 1100"}, 92 {0x9005, 0x028f, 0x9005, 0x908, PQI_HWIF_SRCV, "SmartHBA 2100"}, 93 {0x9005, 0x028f, 0x9005, 0x90a, PQI_HWIF_SRCV, "SmartHBA 2100A-8i"}, 94 {0x9005, 0x028f, 0x193d, 0x8460, PQI_HWIF_SRCV, "UN HBA H460-M1"}, 95 {0x9005, 0x028f, 0x193d, 0x8461, PQI_HWIF_SRCV, "UN HBA H460-B1"}, 96 {0x9005, 0x028f, 0x1bd4, 0x004a, PQI_HWIF_SRCV, "INSPUR SMART-HBA PM8222-SHBA"}, 97 {0x9005, 0x028f, 0x13fe, 0x8312, PQI_HWIF_SRCV, "MIC-8312BridgeB"}, 98 99 /* (SRCx MSCC FVB 24x12G based) */ 100 {0x9005, 0x028f, 0x103c, 0x1001, PQI_HWIF_SRCV, "MSCC FVB"}, 101 102 /* (MSCC PM8241 24x12G based) */ 103 104 /* (MSCC PM8242 24x12G based) */ 105 {0x9005, 0x028f, 0x152d, 0x8a37, PQI_HWIF_SRCV, "QS-8242-24i"}, 106 {0x9005, 0x028f, 0x9005, 0x1300, PQI_HWIF_SRCV, "HBA 1100-8i8e"}, 107 {0x9005, 0x028f, 0x9005, 0x1301, PQI_HWIF_SRCV, "HBA 1100-24i"}, 108 {0x9005, 0x028f, 0x9005, 0x1302, PQI_HWIF_SRCV, "SmartHBA 2100-8i8e"}, 109 {0x9005, 0x028f, 0x9005, 0x1303, PQI_HWIF_SRCV, "SmartHBA 2100-24i"}, 110 {0x9005, 0x028f, 0x105b, 0x1321, PQI_HWIF_SRCV, "8242-24i"}, 111 {0x9005, 0x028f, 0x1bd4, 0x0045, PQI_HWIF_SRCV, "INSPUR SMART-HBA 8242-24i"}, 112 113 /* (MSCC PM8236 16x12G based) */ 114 {0x9005, 0x028f, 0x152d, 0x8a24, PQI_HWIF_SRCV, "QS-8236-16i"}, 115 {0x9005, 0x028f, 0x9005, 0x1380, PQI_HWIF_SRCV, "SmartRAID 3154-16i"}, 116 {0x9005, 0x028f, 0x1bd4, 0x0046, PQI_HWIF_SRCV, "INSPUR RAID 8236-16i"}, 117 118 /* (MSCC PM8237 24x12G based) */ 119 {0x9005, 0x028f, 0x103c, 0x1100, PQI_HWIF_SRCV, "P816i-a SR Gen10"}, 120 {0x9005, 0x028f, 0x103c, 0x1101, PQI_HWIF_SRCV, "P416ie-m SR G10"}, 121 122 /* (MSCC PM8238 16x12G based) */ 123 {0x9005, 0x028f, 0x152d, 0x8a23, PQI_HWIF_SRCV, "QS-8238-16i"}, 124 {0x9005, 0x028f, 0x9005, 0x1280, PQI_HWIF_SRCV, "HBA 1100-16i"}, 125 {0x9005, 0x028f, 0x9005, 0x1281, PQI_HWIF_SRCV, "HBA 1100-16e"}, 126 {0x9005, 0x028f, 0x105b, 0x1211, PQI_HWIF_SRCV, "8238-16i"}, 127 {0x9005, 0x028f, 0x1bd4, 0x0048, PQI_HWIF_SRCV, "INSPUR SMART-HBA 8238-16i"}, 128 {0x9005, 0x028f, 0x9005, 0x1282, PQI_HWIF_SRCV, "SmartHBA 2100-16i"}, 129 130 /* (MSCC PM8240 24x12G based) */ 131 {0x9005, 0x028f, 0x152d, 0x8a36, PQI_HWIF_SRCV, "QS-8240-24i"}, 132 {0x9005, 0x028f, 0x9005, 0x1200, PQI_HWIF_SRCV, "SmartRAID 3154-24i"}, 133 {0x9005, 0x028f, 0x9005, 0x1201, PQI_HWIF_SRCV, "SmartRAID 3154-8i16e"}, 134 {0x9005, 0x028f, 0x9005, 0x1202, PQI_HWIF_SRCV, "SmartRAID 3154-8i8e"}, 135 {0x9005, 0x028f, 0x1bd4, 0x0047, PQI_HWIF_SRCV, "INSPUR RAID 8240-24i"}, 136 137 {0, 0, 0, 0, 0, 0} 138 }; 139 140 struct pqi_ident 141 pqi_family_identifiers[] = { 142 {0x9005, 0x028f, 0, 0, PQI_HWIF_SRCV, "Smart Array Storage Controller"}, 143 {0, 0, 0, 0, 0, 0} 144 }; 145 146 /* 147 * Function to identify the installed adapter. 148 */ 149 static struct pqi_ident * 150 pqi_find_ident(device_t dev) 151 { 152 struct pqi_ident *m; 153 u_int16_t vendid, devid, sub_vendid, sub_devid; 154 155 vendid = pci_get_vendor(dev); 156 devid = pci_get_device(dev); 157 sub_vendid = pci_get_subvendor(dev); 158 sub_devid = pci_get_subdevice(dev); 159 160 for (m = pqi_identifiers; m->vendor != 0; m++) { 161 if ((m->vendor == vendid) && (m->device == devid) && 162 (m->subvendor == sub_vendid) && 163 (m->subdevice == sub_devid)) { 164 return (m); 165 } 166 } 167 168 for (m = pqi_family_identifiers; m->vendor != 0; m++) { 169 if ((m->vendor == vendid) && (m->device == devid)) { 170 return (m); 171 } 172 } 173 174 return (NULL); 175 } 176 177 /* 178 * Determine whether this is one of our supported adapters. 179 */ 180 static int 181 smartpqi_probe(device_t dev) 182 { 183 struct pqi_ident *id; 184 185 if ((id = pqi_find_ident(dev)) != NULL) { 186 device_set_desc(dev, id->desc); 187 return(BUS_PROBE_VENDOR); 188 } 189 190 return(ENXIO); 191 } 192 193 /* 194 * Store Bus/Device/Function in softs 195 */ 196 void pqisrc_save_controller_info(struct pqisrc_softstate *softs) 197 { 198 device_t dev = softs->os_specific.pqi_dev; 199 200 softs->bus_id = (uint32_t)pci_get_bus(dev); 201 softs->device_id = (uint32_t)pci_get_device(dev); 202 softs->func_id = (uint32_t)pci_get_function(dev); 203 } 204 205 206 /* 207 * Allocate resources for our device, set up the bus interface. 208 * Initialize the PQI related functionality, scan devices, register sim to 209 * upper layer, create management interface device node etc. 210 */ 211 static int 212 smartpqi_attach(device_t dev) 213 { 214 struct pqisrc_softstate *softs = NULL; 215 struct pqi_ident *id = NULL; 216 int error = 0; 217 u_int32_t command = 0, i = 0; 218 int card_index = device_get_unit(dev); 219 rcb_t *rcbp = NULL; 220 221 /* 222 * Initialise softc. 223 */ 224 softs = device_get_softc(dev); 225 226 if (!softs) { 227 printf("Could not get softc\n"); 228 error = EINVAL; 229 goto out; 230 } 231 memset(softs, 0, sizeof(*softs)); 232 softs->os_specific.pqi_dev = dev; 233 234 DBG_FUNC("IN\n"); 235 236 /* assume failure is 'not configured' */ 237 error = ENXIO; 238 239 /* 240 * Verify that the adapter is correctly set up in PCI space. 241 */ 242 pci_enable_busmaster(softs->os_specific.pqi_dev); 243 command = pci_read_config(softs->os_specific.pqi_dev, PCIR_COMMAND, 2); 244 if ((command & PCIM_CMD_MEMEN) == 0) { 245 DBG_ERR("memory window not available command = %d\n", command); 246 error = ENXIO; 247 goto out; 248 } 249 250 /* 251 * Detect the hardware interface version, set up the bus interface 252 * indirection. 253 */ 254 id = pqi_find_ident(dev); 255 softs->os_specific.pqi_hwif = id->hwif; 256 257 switch(softs->os_specific.pqi_hwif) { 258 case PQI_HWIF_SRCV: 259 DBG_INFO("set hardware up for PMC SRCv for %p", softs); 260 break; 261 default: 262 softs->os_specific.pqi_hwif = PQI_HWIF_UNKNOWN; 263 DBG_ERR("unknown hardware type\n"); 264 error = ENXIO; 265 goto out; 266 } 267 268 pqisrc_save_controller_info(softs); 269 270 /* 271 * Allocate the PCI register window. 272 */ 273 softs->os_specific.pqi_regs_rid0 = PCIR_BAR(0); 274 if ((softs->os_specific.pqi_regs_res0 = 275 bus_alloc_resource_any(softs->os_specific.pqi_dev, SYS_RES_MEMORY, 276 &softs->os_specific.pqi_regs_rid0, RF_ACTIVE)) == NULL) { 277 DBG_ERR("couldn't allocate register window 0\n"); 278 /* assume failure is 'out of memory' */ 279 error = ENOMEM; 280 goto out; 281 } 282 283 bus_get_resource_start(softs->os_specific.pqi_dev, SYS_RES_MEMORY, 284 softs->os_specific.pqi_regs_rid0); 285 286 softs->pci_mem_handle.pqi_btag = rman_get_bustag(softs->os_specific.pqi_regs_res0); 287 softs->pci_mem_handle.pqi_bhandle = rman_get_bushandle(softs->os_specific.pqi_regs_res0); 288 /* softs->pci_mem_base_vaddr = (uintptr_t)rman_get_virtual(softs->os_specific.pqi_regs_res0); */ 289 softs->pci_mem_base_vaddr = (char *)rman_get_virtual(softs->os_specific.pqi_regs_res0); 290 291 /* 292 * Allocate the parent bus DMA tag appropriate for our PCI interface. 293 * 294 * Note that some of these controllers are 64-bit capable. 295 */ 296 if (bus_dma_tag_create(bus_get_dma_tag(dev), /* parent */ 297 PAGE_SIZE, 0, /* algnmnt, boundary */ 298 BUS_SPACE_MAXADDR_32BIT,/* lowaddr */ 299 BUS_SPACE_MAXADDR, /* highaddr */ 300 NULL, NULL, /* filter, filterarg */ 301 BUS_SPACE_MAXSIZE_32BIT, /* maxsize */ 302 BUS_SPACE_UNRESTRICTED, /* nsegments */ 303 BUS_SPACE_MAXSIZE_32BIT, /* maxsegsize */ 304 0, /* flags */ 305 NULL, NULL, /* No locking needed */ 306 &softs->os_specific.pqi_parent_dmat)) { 307 DBG_ERR("can't allocate parent DMA tag\n"); 308 /* assume failure is 'out of memory' */ 309 error = ENOMEM; 310 goto dma_out; 311 } 312 313 softs->os_specific.sim_registered = FALSE; 314 softs->os_name = "FreeBSD "; 315 316 /* Initialize the PQI library */ 317 error = pqisrc_init(softs); 318 if (error) { 319 DBG_ERR("Failed to initialize pqi lib error = %d\n", error); 320 error = PQI_STATUS_FAILURE; 321 goto out; 322 } 323 324 mtx_init(&softs->os_specific.cam_lock, "cam_lock", NULL, MTX_DEF); 325 softs->os_specific.mtx_init = TRUE; 326 mtx_init(&softs->os_specific.map_lock, "map_lock", NULL, MTX_DEF); 327 callout_init(&softs->os_specific.wellness_periodic, 1); 328 callout_init(&softs->os_specific.heartbeat_timeout_id, 1); 329 330 /* 331 * Create DMA tag for mapping buffers into controller-addressable space. 332 */ 333 if (bus_dma_tag_create(softs->os_specific.pqi_parent_dmat,/* parent */ 334 1, 0, /* algnmnt, boundary */ 335 BUS_SPACE_MAXADDR_32BIT,/* lowaddr */ 336 BUS_SPACE_MAXADDR, /* highaddr */ 337 NULL, NULL, /* filter, filterarg */ 338 softs->pqi_cap.max_sg_elem*PAGE_SIZE,/*maxsize*/ 339 softs->pqi_cap.max_sg_elem, /* nsegments */ 340 BUS_SPACE_MAXSIZE_32BIT, /* maxsegsize */ 341 BUS_DMA_ALLOCNOW, /* flags */ 342 busdma_lock_mutex, /* lockfunc */ 343 &softs->os_specific.map_lock, /* lockfuncarg*/ 344 &softs->os_specific.pqi_buffer_dmat)) { 345 DBG_ERR("can't allocate buffer DMA tag for pqi_buffer_dmat\n"); 346 return (ENOMEM); 347 } 348 349 rcbp = &softs->rcb[1]; 350 for( i = 1; i <= softs->pqi_cap.max_outstanding_io; i++, rcbp++ ) { 351 if ((error = bus_dmamap_create(softs->os_specific.pqi_buffer_dmat, 0, &rcbp->cm_datamap)) != 0) { 352 DBG_ERR("Cant create datamap for buf @" 353 "rcbp = %p maxio = %d error = %d\n", 354 rcbp, softs->pqi_cap.max_outstanding_io, error); 355 goto dma_out; 356 } 357 } 358 359 os_start_heartbeat_timer((void *)softs); /* Start the heart-beat timer */ 360 callout_reset(&softs->os_specific.wellness_periodic, 120*hz, 361 os_wellness_periodic, softs); 362 /* Register our shutdown handler. */ 363 softs->os_specific.eh = EVENTHANDLER_REGISTER(shutdown_final, 364 smartpqi_shutdown, softs, SHUTDOWN_PRI_DEFAULT); 365 366 error = pqisrc_scan_devices(softs); 367 if (error) { 368 DBG_ERR("Failed to scan lib error = %d\n", error); 369 error = PQI_STATUS_FAILURE; 370 goto out; 371 } 372 373 error = register_sim(softs, card_index); 374 if (error) { 375 DBG_ERR("Failed to register sim index = %d error = %d\n", 376 card_index, error); 377 goto out; 378 } 379 380 smartpqi_target_rescan(softs); 381 382 TASK_INIT(&softs->os_specific.event_task, 0, pqisrc_event_worker,softs); 383 384 error = create_char_dev(softs, card_index); 385 if (error) { 386 DBG_ERR("Failed to register character device index=%d r=%d\n", 387 card_index, error); 388 goto out; 389 } 390 goto out; 391 392 dma_out: 393 if (softs->os_specific.pqi_regs_res0 != NULL) 394 bus_release_resource(softs->os_specific.pqi_dev, SYS_RES_MEMORY, 395 softs->os_specific.pqi_regs_rid0, 396 softs->os_specific.pqi_regs_res0); 397 out: 398 DBG_FUNC("OUT error = %d\n", error); 399 return(error); 400 } 401 402 /* 403 * Deallocate resources for our device. 404 */ 405 static int 406 smartpqi_detach(device_t dev) 407 { 408 struct pqisrc_softstate *softs = NULL; 409 softs = device_get_softc(dev); 410 DBG_FUNC("IN\n"); 411 412 EVENTHANDLER_DEREGISTER(shutdown_final, softs->os_specific.eh); 413 414 /* kill the periodic event */ 415 callout_drain(&softs->os_specific.wellness_periodic); 416 /* Kill the heart beat event */ 417 callout_drain(&softs->os_specific.heartbeat_timeout_id); 418 419 smartpqi_shutdown(softs); 420 destroy_char_dev(softs); 421 pqisrc_uninit(softs); 422 deregister_sim(softs); 423 pci_release_msi(dev); 424 425 DBG_FUNC("OUT\n"); 426 return 0; 427 } 428 429 /* 430 * Bring the controller to a quiescent state, ready for system suspend. 431 */ 432 static int 433 smartpqi_suspend(device_t dev) 434 { 435 struct pqisrc_softstate *softs; 436 softs = device_get_softc(dev); 437 DBG_FUNC("IN\n"); 438 439 DBG_INFO("Suspending the device %p\n", softs); 440 softs->os_specific.pqi_state |= SMART_STATE_SUSPEND; 441 442 DBG_FUNC("OUT\n"); 443 return(0); 444 } 445 446 /* 447 * Bring the controller back to a state ready for operation. 448 */ 449 static int 450 smartpqi_resume(device_t dev) 451 { 452 struct pqisrc_softstate *softs; 453 softs = device_get_softc(dev); 454 DBG_FUNC("IN\n"); 455 456 softs->os_specific.pqi_state &= ~SMART_STATE_SUSPEND; 457 458 DBG_FUNC("OUT\n"); 459 return(0); 460 } 461 462 /* 463 * Do whatever is needed during a system shutdown. 464 */ 465 int 466 smartpqi_shutdown(void *arg) 467 { 468 struct pqisrc_softstate *softs = NULL; 469 int rval = 0; 470 471 DBG_FUNC("IN\n"); 472 473 softs = (struct pqisrc_softstate *)arg; 474 475 rval = pqisrc_flush_cache(softs, PQISRC_SHUTDOWN); 476 if (rval != PQI_STATUS_SUCCESS) { 477 DBG_ERR("Unable to flush adapter cache! rval = %d", rval); 478 } 479 480 DBG_FUNC("OUT\n"); 481 482 return rval; 483 } 484 485 /* 486 * PCI bus interface. 487 */ 488 static device_method_t pqi_methods[] = { 489 /* Device interface */ 490 DEVMETHOD(device_probe, smartpqi_probe), 491 DEVMETHOD(device_attach, smartpqi_attach), 492 DEVMETHOD(device_detach, smartpqi_detach), 493 DEVMETHOD(device_suspend, smartpqi_suspend), 494 DEVMETHOD(device_resume, smartpqi_resume), 495 { 0, 0 } 496 }; 497 498 static devclass_t pqi_devclass; 499 static driver_t smartpqi_pci_driver = { 500 "smartpqi", 501 pqi_methods, 502 sizeof(struct pqisrc_softstate) 503 }; 504 505 DRIVER_MODULE(smartpqi, pci, smartpqi_pci_driver, pqi_devclass, 0, 0); 506 MODULE_DEPEND(smartpqi, pci, 1, 1, 1); 507