4630 clean stale references to ddi_iopb_alloc and ddi_iopb_free 4634 undocument scsi_hba_attach() and ddi_dma_lim(9s)
1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 22 /* 23 * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. 24 */ 25 26 /* 27 * Direct Attached Disk 28 */ 29 30 #include <sys/file.h> 31 #include <sys/scsi/scsi.h> 32 #include <sys/var.h> 33 #include <sys/proc.h> 34 #include <sys/dktp/cm.h> 35 #include <sys/vtoc.h> 36 #include <sys/dkio.h> 37 #include <sys/policy.h> 38 #include <sys/priv.h> 39 40 #include <sys/dktp/dadev.h> 41 #include <sys/dktp/fctypes.h> 42 #include <sys/dktp/flowctrl.h> 43 #include <sys/dktp/tgcom.h> 44 #include <sys/dktp/tgdk.h> 45 #include <sys/dktp/bbh.h> 46 #include <sys/dktp/dadkio.h> 47 #include <sys/dktp/dadk.h> 48 #include <sys/cdio.h> 49 50 /* 51 * Local Function Prototypes 52 */ 53 static void dadk_restart(void *pktp); 54 static void dadk_pktcb(struct cmpkt *pktp); 55 static void dadk_iodone(struct buf *bp); 56 static void dadk_polldone(struct buf *bp); 57 static void dadk_setcap(struct dadk *dadkp); 58 static void dadk_create_errstats(struct dadk *dadkp, int instance); 59 static void dadk_destroy_errstats(struct dadk *dadkp); 60 61 static int dadk_chkerr(struct cmpkt *pktp); 62 static int dadk_ioprep(struct dadk *dadkp, struct cmpkt *pktp); 63 static int dadk_iosetup(struct dadk *dadkp, struct cmpkt *pktp); 64 static int dadk_ioretry(struct cmpkt *pktp, int action); 65 66 static struct cmpkt *dadk_pktprep(struct dadk *dadkp, struct cmpkt *in_pktp, 67 struct buf *bp, void (*cb_func)(struct buf *), int (*func)(caddr_t), 68 caddr_t arg); 69 70 static int dadk_pkt(opaque_t com_data, struct buf *bp, int (*func)(caddr_t), 71 caddr_t arg); 72 static void dadk_transport(opaque_t com_data, struct buf *bp); 73 static int dadk_ctl_ioctl(struct dadk *, uint32_t, uintptr_t, int); 74 75 struct tgcom_objops dadk_com_ops = { 76 nodev, 77 nodev, 78 dadk_pkt, 79 dadk_transport, 80 0, 0 81 }; 82 83 /* 84 * architecture dependent allocation restrictions for dadk_iob_alloc(). For 85 * x86, we'll set dma_attr_addr_hi to dadk_max_phys_addr and dma_attr_sgllen 86 * to dadk_sgl_size during _init(). 87 */ 88 #if defined(__sparc) 89 static ddi_dma_attr_t dadk_alloc_attr = { 90 DMA_ATTR_V0, /* version number */ 91 0x0, /* lowest usable address */ 92 0xFFFFFFFFull, /* high DMA address range */ 93 0xFFFFFFFFull, /* DMA counter register */ 94 1, /* DMA address alignment */ 95 1, /* DMA burstsizes */ 96 1, /* min effective DMA size */ 97 0xFFFFFFFFull, /* max DMA xfer size */ 98 0xFFFFFFFFull, /* segment boundary */ 99 1, /* s/g list length */ 100 512, /* granularity of device */ 101 0, /* DMA transfer flags */ 102 }; 103 #elif defined(__x86) 104 static ddi_dma_attr_t dadk_alloc_attr = { 105 DMA_ATTR_V0, /* version number */ 106 0x0, /* lowest usable address */ 107 0x0, /* high DMA address range [set in _init()] */ 108 0xFFFFull, /* DMA counter register */ 109 512, /* DMA address alignment */ 110 1, /* DMA burstsizes */ 111 1, /* min effective DMA size */ 112 0xFFFFFFFFull, /* max DMA xfer size */ 113 0xFFFFFFFFull, /* segment boundary */ 114 0, /* s/g list length [set in _init()] */ 115 512, /* granularity of device */ 116 0, /* DMA transfer flags */ 117 }; 118 119 uint64_t dadk_max_phys_addr = 0xFFFFFFFFull; 120 int dadk_sgl_size = 0xFF; 121 #endif 122 123 static int dadk_rmb_ioctl(struct dadk *dadkp, int cmd, intptr_t arg, int flags, 124 int silent); 125 static void dadk_rmb_iodone(struct buf *bp); 126 127 static int dadk_dk_buf_setup(struct dadk *dadkp, opaque_t *cmdp, 128 dev_t dev, enum uio_seg dataspace, int rw); 129 static void dadk_dk(struct dadk *dadkp, struct dadkio_rwcmd *scmdp, 130 struct buf *bp); 131 static void dadkmin(struct buf *bp); 132 static int dadk_dk_strategy(struct buf *bp); 133 static void dadk_recorderr(struct cmpkt *pktp, struct dadkio_rwcmd *rwcmdp); 134 135 struct tgdk_objops dadk_ops = { 136 dadk_init, 137 dadk_free, 138 dadk_probe, 139 dadk_attach, 140 dadk_open, 141 dadk_close, 142 dadk_ioctl, 143 dadk_strategy, 144 dadk_setgeom, 145 dadk_getgeom, 146 dadk_iob_alloc, 147 dadk_iob_free, 148 dadk_iob_htoc, 149 dadk_iob_xfer, 150 dadk_dump, 151 dadk_getphygeom, 152 dadk_set_bbhobj, 153 dadk_check_media, 154 dadk_inquiry, 155 dadk_cleanup, 156 0 157 }; 158 159 /* 160 * Local static data 161 */ 162 163 #ifdef DADK_DEBUG 164 #define DENT 0x0001 165 #define DERR 0x0002 166 #define DIO 0x0004 167 #define DGEOM 0x0010 168 #define DSTATE 0x0020 169 static int dadk_debug = DGEOM; 170 171 #endif /* DADK_DEBUG */ 172 173 static int dadk_check_media_time = 3000000; /* 3 Second State Check */ 174 static int dadk_dk_maxphys = 0x80000; 175 176 static char *dadk_cmds[] = { 177 "\000Unknown", /* unknown */ 178 "\001read sector", /* DCMD_READ 1 */ 179 "\002write sector", /* DCMD_WRITE 2 */ 180 "\003format track", /* DCMD_FMTTRK 3 */ 181 "\004format whole drive", /* DCMD_FMTDRV 4 */ 182 "\005recalibrate", /* DCMD_RECAL 5 */ 183 "\006seek sector", /* DCMD_SEEK 6 */ 184 "\007read verify", /* DCMD_RDVER 7 */ 185 "\010read defect list", /* DCMD_GETDEF 8 */ 186 "\011lock door", /* DCMD_LOCK 9 */ 187 "\012unlock door", /* DCMD_UNLOCK 10 */ 188 "\013start motor", /* DCMD_START_MOTOR 11 */ 189 "\014stop motor", /* DCMD_STOP_MOTOR 12 */ 190 "\015eject", /* DCMD_EJECT 13 */ 191 "\016update geometry", /* DCMD_UPDATE_GEOM 14 */ 192 "\017get state", /* DCMD_GET_STATE 15 */ 193 "\020cdrom pause", /* DCMD_PAUSE 16 */ 194 "\021cdrom resume", /* DCMD_RESUME 17 */ 195 "\022cdrom play track index", /* DCMD_PLAYTRKIND 18 */ 196 "\023cdrom play msf", /* DCMD_PLAYMSF 19 */ 197 "\024cdrom sub channel", /* DCMD_SUBCHNL 20 */ 198 "\025cdrom read mode 1", /* DCMD_READMODE1 21 */ 199 "\026cdrom read toc header", /* DCMD_READTOCHDR 22 */ 200 "\027cdrom read toc entry", /* DCMD_READTOCENT 23 */ 201 "\030cdrom read offset", /* DCMD_READOFFSET 24 */ 202 "\031cdrom read mode 2", /* DCMD_READMODE2 25 */ 203 "\032cdrom volume control", /* DCMD_VOLCTRL 26 */ 204 "\033flush cache", /* DCMD_FLUSH_CACHE 27 */ 205 NULL 206 }; 207 208 static char *dadk_sense[] = { 209 "\000Success", /* DERR_SUCCESS */ 210 "\001address mark not found", /* DERR_AMNF */ 211 "\002track 0 not found", /* DERR_TKONF */ 212 "\003aborted command", /* DERR_ABORT */ 213 "\004write fault", /* DERR_DWF */ 214 "\005ID not found", /* DERR_IDNF */ 215 "\006drive busy", /* DERR_BUSY */ 216 "\007uncorrectable data error", /* DERR_UNC */ 217 "\010bad block detected", /* DERR_BBK */ 218 "\011invalid command", /* DERR_INVCDB */ 219 "\012device hard error", /* DERR_HARD */ 220 "\013illegal length indicated", /* DERR_ILI */ 221 "\014end of media", /* DERR_EOM */ 222 "\015media change requested", /* DERR_MCR */ 223 "\016recovered from error", /* DERR_RECOVER */ 224 "\017device not ready", /* DERR_NOTREADY */ 225 "\020medium error", /* DERR_MEDIUM */ 226 "\021hardware error", /* DERR_HW */ 227 "\022illegal request", /* DERR_ILL */ 228 "\023unit attention", /* DERR_UNIT_ATTN */ 229 "\024data protection", /* DERR_DATA_PROT */ 230 "\025miscompare", /* DERR_MISCOMPARE */ 231 "\026ICRC error during UDMA", /* DERR_ICRC */ 232 "\027reserved", /* DERR_RESV */ 233 NULL 234 }; 235 236 static char *dadk_name = "Disk"; 237 238 /* 239 * This is the loadable module wrapper 240 */ 241 #include <sys/modctl.h> 242 243 extern struct mod_ops mod_miscops; 244 245 static struct modlmisc modlmisc = { 246 &mod_miscops, /* Type of module */ 247 "Direct Attached Disk" 248 }; 249 250 static struct modlinkage modlinkage = { 251 MODREV_1, (void *)&modlmisc, NULL 252 }; 253 254 int 255 _init(void) 256 { 257 #ifdef DADK_DEBUG 258 if (dadk_debug & DENT) 259 PRF("dadk_init: call\n"); 260 #endif 261 262 #if defined(__x86) 263 /* set the max physical address for iob allocs on x86 */ 264 dadk_alloc_attr.dma_attr_addr_hi = dadk_max_phys_addr; 265 266 /* 267 * set the sgllen for iob allocs on x86. If this is set less than 268 * the number of pages the buffer will take (taking into account 269 * alignment), it would force the allocator to try and allocate 270 * contiguous pages. 271 */ 272 dadk_alloc_attr.dma_attr_sgllen = dadk_sgl_size; 273 #endif 274 275 return (mod_install(&modlinkage)); 276 } 277 278 int 279 _fini(void) 280 { 281 #ifdef DADK_DEBUG 282 if (dadk_debug & DENT) 283 PRF("dadk_fini: call\n"); 284 #endif 285 286 return (mod_remove(&modlinkage)); 287 } 288 289 int 290 _info(struct modinfo *modinfop) 291 { 292 return (mod_info(&modlinkage, modinfop)); 293 } 294 295 struct tgdk_obj * 296 dadk_create() 297 { 298 struct tgdk_obj *dkobjp; 299 struct dadk *dadkp; 300 301 dkobjp = kmem_zalloc((sizeof (*dkobjp) + sizeof (*dadkp)), KM_NOSLEEP); 302 if (!dkobjp) 303 return (NULL); 304 dadkp = (struct dadk *)(dkobjp+1); 305 306 dkobjp->tg_ops = (struct tgdk_objops *)&dadk_ops; 307 dkobjp->tg_data = (opaque_t)dadkp; 308 dkobjp->tg_ext = &(dkobjp->tg_extblk); 309 dadkp->dad_extp = &(dkobjp->tg_extblk); 310 311 #ifdef DADK_DEBUG 312 if (dadk_debug & DENT) 313 PRF("dadk_create: tgdkobjp= 0x%x dadkp= 0x%x\n", dkobjp, dadkp); 314 #endif 315 return (dkobjp); 316 } 317 318 int 319 dadk_init(opaque_t objp, opaque_t devp, opaque_t flcobjp, opaque_t queobjp, 320 opaque_t bbhobjp, void *lkarg) 321 { 322 struct dadk *dadkp = (struct dadk *)objp; 323 struct scsi_device *sdevp = (struct scsi_device *)devp; 324 325 dadkp->dad_sd = devp; 326 dadkp->dad_ctlobjp = (opaque_t)sdevp->sd_address.a_hba_tran; 327 sdevp->sd_private = (caddr_t)dadkp; 328 329 /* initialize the communication object */ 330 dadkp->dad_com.com_data = (opaque_t)dadkp; 331 dadkp->dad_com.com_ops = &dadk_com_ops; 332 333 dadkp->dad_bbhobjp = bbhobjp; 334 BBH_INIT(bbhobjp); 335 336 dadkp->dad_flcobjp = flcobjp; 337 mutex_init(&dadkp->dad_cmd_mutex, NULL, MUTEX_DRIVER, NULL); 338 dadkp->dad_cmd_count = 0; 339 return (FLC_INIT(flcobjp, &(dadkp->dad_com), queobjp, lkarg)); 340 } 341 342 int 343 dadk_free(struct tgdk_obj *dkobjp) 344 { 345 TGDK_CLEANUP(dkobjp); 346 kmem_free(dkobjp, (sizeof (*dkobjp) + sizeof (struct dadk))); 347 348 return (DDI_SUCCESS); 349 } 350 351 void 352 dadk_cleanup(struct tgdk_obj *dkobjp) 353 { 354 struct dadk *dadkp; 355 356 dadkp = (struct dadk *)(dkobjp->tg_data); 357 if (dadkp->dad_sd) 358 dadkp->dad_sd->sd_private = NULL; 359 if (dadkp->dad_bbhobjp) { 360 BBH_FREE(dadkp->dad_bbhobjp); 361 dadkp->dad_bbhobjp = NULL; 362 } 363 if (dadkp->dad_flcobjp) { 364 FLC_FREE(dadkp->dad_flcobjp); 365 dadkp->dad_flcobjp = NULL; 366 } 367 mutex_destroy(&dadkp->dad_cmd_mutex); 368 } 369 370 /* ARGSUSED */ 371 int 372 dadk_probe(opaque_t objp, int kmsflg) 373 { 374 struct dadk *dadkp = (struct dadk *)objp; 375 struct scsi_device *devp; 376 char name[80]; 377 378 devp = dadkp->dad_sd; 379 if (!devp->sd_inq || (devp->sd_inq->inq_dtype == DTYPE_NOTPRESENT) || 380 (devp->sd_inq->inq_dtype == DTYPE_UNKNOWN)) { 381 return (DDI_PROBE_FAILURE); 382 } 383 384 switch (devp->sd_inq->inq_dtype) { 385 case DTYPE_DIRECT: 386 dadkp->dad_ctype = DKC_DIRECT; 387 dadkp->dad_extp->tg_nodetype = DDI_NT_BLOCK; 388 dadkp->dad_extp->tg_ctype = DKC_DIRECT; 389 break; 390 case DTYPE_RODIRECT: /* eg cdrom */ 391 dadkp->dad_ctype = DKC_CDROM; 392 dadkp->dad_extp->tg_rdonly = 1; 393 dadkp->dad_rdonly = 1; 394 dadkp->dad_cdrom = 1; 395 dadkp->dad_extp->tg_nodetype = DDI_NT_CD; 396 dadkp->dad_extp->tg_ctype = DKC_CDROM; 397 break; 398 case DTYPE_WORM: 399 case DTYPE_OPTICAL: 400 default: 401 return (DDI_PROBE_FAILURE); 402 } 403 404 dadkp->dad_extp->tg_rmb = dadkp->dad_rmb = devp->sd_inq->inq_rmb; 405 406 dadkp->dad_secshf = SCTRSHFT; 407 dadkp->dad_blkshf = 0; 408 409 /* display the device name */ 410 (void) strcpy(name, "Vendor '"); 411 gda_inqfill((caddr_t)devp->sd_inq->inq_vid, 8, &name[strlen(name)]); 412 (void) strcat(name, "' Product '"); 413 gda_inqfill((caddr_t)devp->sd_inq->inq_pid, 16, &name[strlen(name)]); 414 (void) strcat(name, "'"); 415 gda_log(devp->sd_dev, dadk_name, CE_NOTE, "!<%s>\n", name); 416 417 return (DDI_PROBE_SUCCESS); 418 } 419 420 421 /* ARGSUSED */ 422 int 423 dadk_attach(opaque_t objp) 424 { 425 return (DDI_SUCCESS); 426 } 427 428 int 429 dadk_set_bbhobj(opaque_t objp, opaque_t bbhobjp) 430 { 431 struct dadk *dadkp = (struct dadk *)objp; 432 /* free the old bbh object */ 433 if (dadkp->dad_bbhobjp) 434 BBH_FREE(dadkp->dad_bbhobjp); 435 436 /* initialize the new bbh object */ 437 dadkp->dad_bbhobjp = bbhobjp; 438 BBH_INIT(bbhobjp); 439 440 return (DDI_SUCCESS); 441 } 442 443 /* ARGSUSED */ 444 int 445 dadk_open(opaque_t objp, int flag) 446 { 447 struct dadk *dadkp = (struct dadk *)objp; 448 int error; 449 int wce; 450 451 if (!dadkp->dad_rmb) { 452 if (dadkp->dad_phyg.g_cap) { 453 FLC_START_KSTAT(dadkp->dad_flcobjp, "disk", 454 ddi_get_instance(CTL_DIP_DEV(dadkp->dad_ctlobjp))); 455 return (DDI_SUCCESS); 456 } 457 } else { 458 mutex_enter(&dadkp->dad_mutex); 459 dadkp->dad_iostate = DKIO_NONE; 460 cv_broadcast(&dadkp->dad_state_cv); 461 mutex_exit(&dadkp->dad_mutex); 462 463 if (dadk_rmb_ioctl(dadkp, DCMD_START_MOTOR, 0, 0, 464 DADK_SILENT) || 465 dadk_rmb_ioctl(dadkp, DCMD_LOCK, 0, 0, DADK_SILENT) || 466 dadk_rmb_ioctl(dadkp, DCMD_UPDATE_GEOM, 0, 0, 467 DADK_SILENT)) { 468 return (DDI_FAILURE); 469 } 470 471 mutex_enter(&dadkp->dad_mutex); 472 dadkp->dad_iostate = DKIO_INSERTED; 473 cv_broadcast(&dadkp->dad_state_cv); 474 mutex_exit(&dadkp->dad_mutex); 475 } 476 477 /* 478 * get write cache enable state 479 * If there is an error, must assume that write cache 480 * is enabled. 481 * NOTE: Since there is currently no Solaris mechanism to 482 * change the state of the Write Cache Enable feature, 483 * this code just checks the value of the WCE bit 484 * obtained at device init time. If a mechanism 485 * is added to the driver to change WCE, dad_wce 486 * must be updated appropriately. 487 */ 488 error = dadk_ctl_ioctl(dadkp, DIOCTL_GETWCE, 489 (uintptr_t)&wce, FKIOCTL | FNATIVE); 490 mutex_enter(&dadkp->dad_mutex); 491 dadkp->dad_wce = (error != 0) || (wce != 0); 492 mutex_exit(&dadkp->dad_mutex); 493 494 /* logical disk geometry */ 495 (void) dadk_ctl_ioctl(dadkp, DIOCTL_GETGEOM, 496 (uintptr_t)&dadkp->dad_logg, FKIOCTL | FNATIVE); 497 if (dadkp->dad_logg.g_cap == 0) 498 return (DDI_FAILURE); 499 500 /* get physical disk geometry */ 501 (void) dadk_ctl_ioctl(dadkp, DIOCTL_GETPHYGEOM, 502 (uintptr_t)&dadkp->dad_phyg, FKIOCTL | FNATIVE); 503 if (dadkp->dad_phyg.g_cap == 0) 504 return (DDI_FAILURE); 505 506 dadk_setcap(dadkp); 507 508 dadk_create_errstats(dadkp, 509 ddi_get_instance(CTL_DIP_DEV(dadkp->dad_ctlobjp))); 510 511 /* start profiling */ 512 FLC_START_KSTAT(dadkp->dad_flcobjp, "disk", 513 ddi_get_instance(CTL_DIP_DEV(dadkp->dad_ctlobjp))); 514 515 return (DDI_SUCCESS); 516 } 517 518 static void 519 dadk_setcap(struct dadk *dadkp) 520 { 521 int totsize; 522 int i; 523 524 totsize = dadkp->dad_phyg.g_secsiz; 525 526 if (totsize == 0) { 527 if (dadkp->dad_cdrom) { 528 totsize = 2048; 529 } else { 530 totsize = NBPSCTR; 531 } 532 } else { 533 /* Round down sector size to multiple of 512B */ 534 totsize &= ~(NBPSCTR-1); 535 } 536 dadkp->dad_phyg.g_secsiz = totsize; 537 538 /* set sec,block shift factor - (512->0, 1024->1, 2048->2, etc.) */ 539 totsize >>= SCTRSHFT; 540 for (i = 0; totsize != 1; i++, totsize >>= 1) 541 ; 542 dadkp->dad_blkshf = i; 543 dadkp->dad_secshf = i + SCTRSHFT; 544 } 545 546 547 static void 548 dadk_create_errstats(struct dadk *dadkp, int instance) 549 { 550 dadk_errstats_t *dep; 551 char kstatname[KSTAT_STRLEN]; 552 dadk_ioc_string_t dadk_ioc_string; 553 554 if (dadkp->dad_errstats) 555 return; 556 557 (void) sprintf(kstatname, "cmdk%d,error", instance); 558 dadkp->dad_errstats = kstat_create("cmdkerror", instance, 559 kstatname, "device_error", KSTAT_TYPE_NAMED, 560 sizeof (dadk_errstats_t) / sizeof (kstat_named_t), 561 KSTAT_FLAG_PERSISTENT); 562 563 if (!dadkp->dad_errstats) 564 return; 565 566 dep = (dadk_errstats_t *)dadkp->dad_errstats->ks_data; 567 568 kstat_named_init(&dep->dadk_softerrs, 569 "Soft Errors", KSTAT_DATA_UINT32); 570 kstat_named_init(&dep->dadk_harderrs, 571 "Hard Errors", KSTAT_DATA_UINT32); 572 kstat_named_init(&dep->dadk_transerrs, 573 "Transport Errors", KSTAT_DATA_UINT32); 574 kstat_named_init(&dep->dadk_model, 575 "Model", KSTAT_DATA_CHAR); 576 kstat_named_init(&dep->dadk_revision, 577 "Revision", KSTAT_DATA_CHAR); 578 kstat_named_init(&dep->dadk_serial, 579 "Serial No", KSTAT_DATA_CHAR); 580 kstat_named_init(&dep->dadk_capacity, 581 "Size", KSTAT_DATA_ULONGLONG); 582 kstat_named_init(&dep->dadk_rq_media_err, 583 "Media Error", KSTAT_DATA_UINT32); 584 kstat_named_init(&dep->dadk_rq_ntrdy_err, 585 "Device Not Ready", KSTAT_DATA_UINT32); 586 kstat_named_init(&dep->dadk_rq_nodev_err, 587 "No Device", KSTAT_DATA_UINT32); 588 kstat_named_init(&dep->dadk_rq_recov_err, 589 "Recoverable", KSTAT_DATA_UINT32); 590 kstat_named_init(&dep->dadk_rq_illrq_err, 591 "Illegal Request", KSTAT_DATA_UINT32); 592 593 dadkp->dad_errstats->ks_private = dep; 594 dadkp->dad_errstats->ks_update = nulldev; 595 kstat_install(dadkp->dad_errstats); 596 597 /* get model */ 598 dep->dadk_model.value.c[0] = 0; 599 dadk_ioc_string.is_buf = &dep->dadk_model.value.c[0]; 600 dadk_ioc_string.is_size = sizeof (dep->dadk_model.value.c); 601 (void) dadk_ctl_ioctl(dadkp, DIOCTL_GETMODEL, 602 (uintptr_t)&dadk_ioc_string, FKIOCTL | FNATIVE); 603 604 /* get serial */ 605 dep->dadk_serial.value.c[0] = 0; 606 dadk_ioc_string.is_buf = &dep->dadk_serial.value.c[0]; 607 dadk_ioc_string.is_size = sizeof (dep->dadk_serial.value.c); 608 (void) dadk_ctl_ioctl(dadkp, DIOCTL_GETSERIAL, 609 (uintptr_t)&dadk_ioc_string, FKIOCTL | FNATIVE); 610 611 /* Get revision */ 612 dep->dadk_revision.value.c[0] = 0; 613 614 /* Get capacity */ 615 616 dep->dadk_capacity.value.ui64 = 617 (uint64_t)dadkp->dad_logg.g_cap * 618 (uint64_t)dadkp->dad_logg.g_secsiz; 619 } 620 621 622 int 623 dadk_close(opaque_t objp) 624 { 625 struct dadk *dadkp = (struct dadk *)objp; 626 627 if (dadkp->dad_rmb) { 628 (void) dadk_rmb_ioctl(dadkp, DCMD_STOP_MOTOR, 0, 0, 629 DADK_SILENT); 630 (void) dadk_rmb_ioctl(dadkp, DCMD_UNLOCK, 0, 0, DADK_SILENT); 631 } 632 FLC_STOP_KSTAT(dadkp->dad_flcobjp); 633 634 dadk_destroy_errstats(dadkp); 635 636 return (DDI_SUCCESS); 637 } 638 639 static void 640 dadk_destroy_errstats(struct dadk *dadkp) 641 { 642 if (!dadkp->dad_errstats) 643 return; 644 645 kstat_delete(dadkp->dad_errstats); 646 dadkp->dad_errstats = NULL; 647 } 648 649 650 int 651 dadk_strategy(opaque_t objp, struct buf *bp) 652 { 653 struct dadk *dadkp = (struct dadk *)objp; 654 655 if (dadkp->dad_rdonly && !(bp->b_flags & B_READ)) { 656 bioerror(bp, EROFS); 657 return (DDI_FAILURE); 658 } 659 660 if (bp->b_bcount & (dadkp->DAD_SECSIZ-1)) { 661 bioerror(bp, ENXIO); 662 return (DDI_FAILURE); 663 } 664 665 SET_BP_SEC(bp, (LBLK2SEC(GET_BP_SEC(bp), dadkp->dad_blkshf))); 666 mutex_enter(&dadkp->dad_cmd_mutex); 667 dadkp->dad_cmd_count++; 668 mutex_exit(&dadkp->dad_cmd_mutex); 669 FLC_ENQUE(dadkp->dad_flcobjp, bp); 670 671 return (DDI_SUCCESS); 672 } 673 674 int 675 dadk_dump(opaque_t objp, struct buf *bp) 676 { 677 struct dadk *dadkp = (struct dadk *)objp; 678 struct cmpkt *pktp; 679 680 if (dadkp->dad_rdonly) { 681 bioerror(bp, EROFS); 682 return (DDI_FAILURE); 683 } 684 685 if (bp->b_bcount & (dadkp->DAD_SECSIZ-1)) { 686 bioerror(bp, ENXIO); 687 return (DDI_FAILURE); 688 } 689 690 SET_BP_SEC(bp, (LBLK2SEC(GET_BP_SEC(bp), dadkp->dad_blkshf))); 691 692 pktp = dadk_pktprep(dadkp, NULL, bp, dadk_polldone, NULL, NULL); 693 if (!pktp) { 694 cmn_err(CE_WARN, "no resources for dumping"); 695 bioerror(bp, EIO); 696 return (DDI_FAILURE); 697 } 698 pktp->cp_flags |= CPF_NOINTR; 699 700 (void) dadk_ioprep(dadkp, pktp); 701 dadk_transport(dadkp, bp); 702 pktp->cp_byteleft -= pktp->cp_bytexfer; 703 704 while (geterror(bp) == 0 && pktp->cp_byteleft != 0) { 705 (void) dadk_iosetup(dadkp, pktp); 706 dadk_transport(dadkp, bp); 707 pktp->cp_byteleft -= pktp->cp_bytexfer; 708 } 709 710 if (pktp->cp_private) 711 BBH_FREEHANDLE(dadkp->dad_bbhobjp, pktp->cp_private); 712 gda_free(dadkp->dad_ctlobjp, pktp, NULL); 713 return (DDI_SUCCESS); 714 } 715 716 /* ARGSUSED */ 717 int 718 dadk_ioctl(opaque_t objp, dev_t dev, int cmd, intptr_t arg, int flag, 719 cred_t *cred_p, int *rval_p) 720 { 721 struct dadk *dadkp = (struct dadk *)objp; 722 723 switch (cmd) { 724 case DKIOCGETDEF: 725 { 726 struct buf *bp; 727 int err, head; 728 unsigned char *secbuf; 729 STRUCT_DECL(defect_header, adh); 730 731 STRUCT_INIT(adh, flag & FMODELS); 732 733 /* 734 * copyin header .... 735 * yields head number and buffer address 736 */ 737 if (ddi_copyin((caddr_t)arg, STRUCT_BUF(adh), STRUCT_SIZE(adh), 738 flag)) 739 return (EFAULT); 740 head = STRUCT_FGET(adh, head); 741 if (head < 0 || head >= dadkp->dad_phyg.g_head) 742 return (ENXIO); 743 secbuf = kmem_zalloc(NBPSCTR, KM_SLEEP); 744 if (!secbuf) 745 return (ENOMEM); 746 bp = getrbuf(KM_SLEEP); 747 if (!bp) { 748 kmem_free(secbuf, NBPSCTR); 749 return (ENOMEM); 750 } 751 752 bp->b_edev = dev; 753 bp->b_dev = cmpdev(dev); 754 bp->b_flags = B_BUSY; 755 bp->b_resid = 0; 756 bp->b_bcount = NBPSCTR; 757 bp->b_un.b_addr = (caddr_t)secbuf; 758 bp->b_blkno = head; /* I had to put it somwhere! */ 759 bp->b_forw = (struct buf *)dadkp; 760 bp->b_back = (struct buf *)DCMD_GETDEF; 761 762 mutex_enter(&dadkp->dad_cmd_mutex); 763 dadkp->dad_cmd_count++; 764 mutex_exit(&dadkp->dad_cmd_mutex); 765 FLC_ENQUE(dadkp->dad_flcobjp, bp); 766 err = biowait(bp); 767 if (!err) { 768 if (ddi_copyout((caddr_t)secbuf, 769 STRUCT_FGETP(adh, buffer), NBPSCTR, flag)) 770 err = ENXIO; 771 } 772 kmem_free(secbuf, NBPSCTR); 773 freerbuf(bp); 774 return (err); 775 } 776 case DIOCTL_RWCMD: 777 { 778 struct dadkio_rwcmd *rwcmdp; 779 int status, rw; 780 781 /* 782 * copied in by cmdk and, if necessary, converted to the 783 * correct datamodel 784 */ 785 rwcmdp = (struct dadkio_rwcmd *)(intptr_t)arg; 786 787 /* 788 * handle the complex cases here; we pass these 789 * through to the driver, which will queue them and 790 * handle the requests asynchronously. The simpler 791 * cases ,which can return immediately, fail here, and 792 * the request reverts to the dadk_ioctl routine, while 793 * will reroute them directly to the ata driver. 794 */ 795 switch (rwcmdp->cmd) { 796 case DADKIO_RWCMD_READ : 797 /*FALLTHROUGH*/ 798 case DADKIO_RWCMD_WRITE: 799 rw = ((rwcmdp->cmd == DADKIO_RWCMD_WRITE) ? 800 B_WRITE : B_READ); 801 status = dadk_dk_buf_setup(dadkp, 802 (opaque_t)rwcmdp, dev, ((flag &FKIOCTL) ? 803 UIO_SYSSPACE : UIO_USERSPACE), rw); 804 return (status); 805 default: 806 return (EINVAL); 807 } 808 } 809 case DKIOC_UPDATEFW: 810 811 /* 812 * Require PRIV_ALL privilege to invoke DKIOC_UPDATEFW 813 * to protect the firmware update from malicious use 814 */ 815 if (PRIV_POLICY(cred_p, PRIV_ALL, B_FALSE, EPERM, NULL) != 0) 816 return (EPERM); 817 else 818 return (dadk_ctl_ioctl(dadkp, cmd, arg, flag)); 819 820 case DKIOCFLUSHWRITECACHE: 821 { 822 struct buf *bp; 823 int err = 0; 824 struct dk_callback *dkc = (struct dk_callback *)arg; 825 struct cmpkt *pktp; 826 int is_sync = 1; 827 828 mutex_enter(&dadkp->dad_mutex); 829 if (dadkp->dad_noflush || ! dadkp->dad_wce) { 830 err = dadkp->dad_noflush ? ENOTSUP : 0; 831 mutex_exit(&dadkp->dad_mutex); 832 /* 833 * If a callback was requested: a 834 * callback will always be done if the 835 * caller saw the DKIOCFLUSHWRITECACHE 836 * ioctl return 0, and never done if the 837 * caller saw the ioctl return an error. 838 */ 839 if ((flag & FKIOCTL) && dkc != NULL && 840 dkc->dkc_callback != NULL) { 841 (*dkc->dkc_callback)(dkc->dkc_cookie, 842 err); 843 /* 844 * Did callback and reported error. 845 * Since we did a callback, ioctl 846 * should return 0. 847 */ 848 err = 0; 849 } 850 return (err); 851 } 852 mutex_exit(&dadkp->dad_mutex); 853 854 bp = getrbuf(KM_SLEEP); 855 856 bp->b_edev = dev; 857 bp->b_dev = cmpdev(dev); 858 bp->b_flags = B_BUSY; 859 bp->b_resid = 0; 860 bp->b_bcount = 0; 861 SET_BP_SEC(bp, 0); 862 863 if ((flag & FKIOCTL) && dkc != NULL && 864 dkc->dkc_callback != NULL) { 865 struct dk_callback *dkc2 = 866 (struct dk_callback *)kmem_zalloc( 867 sizeof (struct dk_callback), KM_SLEEP); 868 869 bcopy(dkc, dkc2, sizeof (*dkc2)); 870 bp->b_private = dkc2; 871 bp->b_iodone = dadk_flushdone; 872 is_sync = 0; 873 } 874 875 /* 876 * Setup command pkt 877 * dadk_pktprep() can't fail since DDI_DMA_SLEEP set 878 */ 879 pktp = dadk_pktprep(dadkp, NULL, bp, 880 dadk_iodone, DDI_DMA_SLEEP, NULL); 881 882 pktp->cp_time = DADK_FLUSH_CACHE_TIME; 883 884 *((char *)(pktp->cp_cdbp)) = DCMD_FLUSH_CACHE; 885 pktp->cp_byteleft = 0; 886 pktp->cp_private = NULL; 887 pktp->cp_secleft = 0; 888 pktp->cp_srtsec = -1; 889 pktp->cp_bytexfer = 0; 890 891 CTL_IOSETUP(dadkp->dad_ctlobjp, pktp); 892 893 mutex_enter(&dadkp->dad_cmd_mutex); 894 dadkp->dad_cmd_count++; 895 mutex_exit(&dadkp->dad_cmd_mutex); 896 FLC_ENQUE(dadkp->dad_flcobjp, bp); 897 898 if (is_sync) { 899 err = biowait(bp); 900 freerbuf(bp); 901 } 902 return (err); 903 } 904 default: 905 if (!dadkp->dad_rmb) 906 return (dadk_ctl_ioctl(dadkp, cmd, arg, flag)); 907 } 908 909 switch (cmd) { 910 case CDROMSTOP: 911 return (dadk_rmb_ioctl(dadkp, DCMD_STOP_MOTOR, 0, 912 0, DADK_SILENT)); 913 case CDROMSTART: 914 return (dadk_rmb_ioctl(dadkp, DCMD_START_MOTOR, 0, 915 0, DADK_SILENT)); 916 case DKIOCLOCK: 917 return (dadk_rmb_ioctl(dadkp, DCMD_LOCK, 0, 0, DADK_SILENT)); 918 case DKIOCUNLOCK: 919 return (dadk_rmb_ioctl(dadkp, DCMD_UNLOCK, 0, 0, DADK_SILENT)); 920 case DKIOCEJECT: 921 case CDROMEJECT: 922 { 923 int ret; 924 925 if (ret = dadk_rmb_ioctl(dadkp, DCMD_UNLOCK, 0, 0, 926 DADK_SILENT)) { 927 return (ret); 928 } 929 if (ret = dadk_rmb_ioctl(dadkp, DCMD_EJECT, 0, 0, 930 DADK_SILENT)) { 931 return (ret); 932 } 933 mutex_enter(&dadkp->dad_mutex); 934 dadkp->dad_iostate = DKIO_EJECTED; 935 cv_broadcast(&dadkp->dad_state_cv); 936 mutex_exit(&dadkp->dad_mutex); 937 938 return (0); 939 940 } 941 default: 942 return (ENOTTY); 943 /* 944 * cdrom audio commands 945 */ 946 case CDROMPAUSE: 947 cmd = DCMD_PAUSE; 948 break; 949 case CDROMRESUME: 950 cmd = DCMD_RESUME; 951 break; 952 case CDROMPLAYMSF: 953 cmd = DCMD_PLAYMSF; 954 break; 955 case CDROMPLAYTRKIND: 956 cmd = DCMD_PLAYTRKIND; 957 break; 958 case CDROMREADTOCHDR: 959 cmd = DCMD_READTOCHDR; 960 break; 961 case CDROMREADTOCENTRY: 962 cmd = DCMD_READTOCENT; 963 break; 964 case CDROMVOLCTRL: 965 cmd = DCMD_VOLCTRL; 966 break; 967 case CDROMSUBCHNL: 968 cmd = DCMD_SUBCHNL; 969 break; 970 case CDROMREADMODE2: 971 cmd = DCMD_READMODE2; 972 break; 973 case CDROMREADMODE1: 974 cmd = DCMD_READMODE1; 975 break; 976 case CDROMREADOFFSET: 977 cmd = DCMD_READOFFSET; 978 break; 979 } 980 return (dadk_rmb_ioctl(dadkp, cmd, arg, flag, 0)); 981 } 982 983 int 984 dadk_flushdone(struct buf *bp) 985 { 986 struct dk_callback *dkc = bp->b_private; 987 988 ASSERT(dkc != NULL && dkc->dkc_callback != NULL); 989 990 (*dkc->dkc_callback)(dkc->dkc_cookie, geterror(bp)); 991 992 kmem_free(dkc, sizeof (*dkc)); 993 freerbuf(bp); 994 return (0); 995 } 996 997 int 998 dadk_getphygeom(opaque_t objp, struct tgdk_geom *dkgeom_p) 999 { 1000 struct dadk *dadkp = (struct dadk *)objp; 1001 1002 bcopy((caddr_t)&dadkp->dad_phyg, (caddr_t)dkgeom_p, 1003 sizeof (struct tgdk_geom)); 1004 return (DDI_SUCCESS); 1005 } 1006 1007 int 1008 dadk_getgeom(opaque_t objp, struct tgdk_geom *dkgeom_p) 1009 { 1010 struct dadk *dadkp = (struct dadk *)objp; 1011 bcopy((caddr_t)&dadkp->dad_logg, (caddr_t)dkgeom_p, 1012 sizeof (struct tgdk_geom)); 1013 return (DDI_SUCCESS); 1014 } 1015 1016 int 1017 dadk_setgeom(opaque_t objp, struct tgdk_geom *dkgeom_p) 1018 { 1019 struct dadk *dadkp = (struct dadk *)objp; 1020 1021 dadkp->dad_logg.g_cyl = dkgeom_p->g_cyl; 1022 dadkp->dad_logg.g_head = dkgeom_p->g_head; 1023 dadkp->dad_logg.g_sec = dkgeom_p->g_sec; 1024 dadkp->dad_logg.g_cap = dkgeom_p->g_cap; 1025 return (DDI_SUCCESS); 1026 } 1027 1028 1029 tgdk_iob_handle 1030 dadk_iob_alloc(opaque_t objp, daddr_t blkno, ssize_t xfer, int kmsflg) 1031 { 1032 struct dadk *dadkp = (struct dadk *)objp; 1033 struct buf *bp; 1034 struct tgdk_iob *iobp; 1035 size_t rlen; 1036 1037 iobp = kmem_zalloc(sizeof (*iobp), kmsflg); 1038 if (iobp == NULL) 1039 return (NULL); 1040 if ((bp = getrbuf(kmsflg)) == NULL) { 1041 kmem_free(iobp, sizeof (*iobp)); 1042 return (NULL); 1043 } 1044 1045 iobp->b_psec = LBLK2SEC(blkno, dadkp->dad_blkshf); 1046 iobp->b_pbyteoff = (blkno & ((1<<dadkp->dad_blkshf) - 1)) << SCTRSHFT; 1047 iobp->b_pbytecnt = ((iobp->b_pbyteoff + xfer + dadkp->DAD_SECSIZ - 1) 1048 >> dadkp->dad_secshf) << dadkp->dad_secshf; 1049 1050 bp->b_un.b_addr = 0; 1051 /* 1052 * use i_ddi_mem_alloc() for now until we have an interface to allocate 1053 * memory for DMA which doesn't require a DMA handle. ddi_iopb_alloc() 1054 * is obsolete and we want more flexibility in controlling the DMA 1055 * address constraints.. 1056 */ 1057 if (i_ddi_mem_alloc((dadkp->dad_sd)->sd_dev, &dadk_alloc_attr, 1058 (size_t)iobp->b_pbytecnt, ((kmsflg == KM_SLEEP) ? 1 : 0), 0, NULL, 1059 &bp->b_un.b_addr, &rlen, NULL) != DDI_SUCCESS) { 1060 freerbuf(bp); 1061 kmem_free(iobp, sizeof (*iobp)); 1062 return (NULL); 1063 } 1064 iobp->b_flag |= IOB_BPALLOC | IOB_BPBUFALLOC; 1065 iobp->b_bp = bp; 1066 iobp->b_lblk = blkno; 1067 iobp->b_xfer = xfer; 1068 iobp->b_lblk = blkno; 1069 iobp->b_xfer = xfer; 1070 return (iobp); 1071 } 1072 1073 /* ARGSUSED */ 1074 int 1075 dadk_iob_free(opaque_t objp, struct tgdk_iob *iobp) 1076 { 1077 struct buf *bp; 1078 1079 if (iobp) { 1080 if (iobp->b_bp && (iobp->b_flag & IOB_BPALLOC)) { 1081 bp = iobp->b_bp; 1082 if (bp->b_un.b_addr && (iobp->b_flag & IOB_BPBUFALLOC)) 1083 i_ddi_mem_free((caddr_t)bp->b_un.b_addr, NULL); 1084 freerbuf(bp); 1085 } 1086 kmem_free(iobp, sizeof (*iobp)); 1087 } 1088 return (DDI_SUCCESS); 1089 } 1090 1091 /* ARGSUSED */ 1092 caddr_t 1093 dadk_iob_htoc(opaque_t objp, struct tgdk_iob *iobp) 1094 { 1095 return (iobp->b_bp->b_un.b_addr+iobp->b_pbyteoff); 1096 } 1097 1098 1099 caddr_t 1100 dadk_iob_xfer(opaque_t objp, struct tgdk_iob *iobp, int rw) 1101 { 1102 struct dadk *dadkp = (struct dadk *)objp; 1103 struct buf *bp; 1104 int err; 1105 1106 bp = iobp->b_bp; 1107 if (dadkp->dad_rdonly && !(rw & B_READ)) { 1108 bioerror(bp, EROFS); 1109 return (NULL); 1110 } 1111 1112 bp->b_flags |= (B_BUSY | rw); 1113 bp->b_bcount = iobp->b_pbytecnt; 1114 SET_BP_SEC(bp, iobp->b_psec); 1115 bp->av_back = (struct buf *)0; 1116 bp->b_resid = 0; 1117 1118 /* call flow control */ 1119 mutex_enter(&dadkp->dad_cmd_mutex); 1120 dadkp->dad_cmd_count++; 1121 mutex_exit(&dadkp->dad_cmd_mutex); 1122 FLC_ENQUE(dadkp->dad_flcobjp, bp); 1123 err = biowait(bp); 1124 1125 bp->b_bcount = iobp->b_xfer; 1126 bp->b_flags &= ~(B_DONE|B_BUSY); 1127 1128 if (err) 1129 return (NULL); 1130 1131 return (bp->b_un.b_addr+iobp->b_pbyteoff); 1132 } 1133 1134 static void 1135 dadk_transport(opaque_t com_data, struct buf *bp) 1136 { 1137 struct dadk *dadkp = (struct dadk *)com_data; 1138 1139 if (CTL_TRANSPORT(dadkp->dad_ctlobjp, GDA_BP_PKT(bp)) == 1140 CTL_SEND_SUCCESS) 1141 return; 1142 dadk_restart((void*)GDA_BP_PKT(bp)); 1143 } 1144 1145 static int 1146 dadk_pkt(opaque_t com_data, struct buf *bp, int (*func)(caddr_t), caddr_t arg) 1147 { 1148 struct cmpkt *pktp; 1149 struct dadk *dadkp = (struct dadk *)com_data; 1150 1151 if (GDA_BP_PKT(bp)) 1152 return (DDI_SUCCESS); 1153 1154 pktp = dadk_pktprep(dadkp, NULL, bp, dadk_iodone, func, arg); 1155 if (!pktp) 1156 return (DDI_FAILURE); 1157 1158 return (dadk_ioprep(dadkp, pktp)); 1159 } 1160 1161 /* 1162 * Read, Write preparation 1163 */ 1164 static int 1165 dadk_ioprep(struct dadk *dadkp, struct cmpkt *pktp) 1166 { 1167 struct buf *bp; 1168 1169 bp = pktp->cp_bp; 1170 if (bp->b_forw == (struct buf *)dadkp) 1171 *((char *)(pktp->cp_cdbp)) = (char)(intptr_t)bp->b_back; 1172 1173 else if (bp->b_flags & B_READ) 1174 *((char *)(pktp->cp_cdbp)) = DCMD_READ; 1175 else 1176 *((char *)(pktp->cp_cdbp)) = DCMD_WRITE; 1177 pktp->cp_byteleft = bp->b_bcount; 1178 1179 /* setup the bad block list handle */ 1180 pktp->cp_private = BBH_GETHANDLE(dadkp->dad_bbhobjp, bp); 1181 return (dadk_iosetup(dadkp, pktp)); 1182 } 1183 1184 static int 1185 dadk_iosetup(struct dadk *dadkp, struct cmpkt *pktp) 1186 { 1187 struct buf *bp; 1188 bbh_cookie_t bbhckp; 1189 int seccnt; 1190 1191 seccnt = pktp->cp_bytexfer >> dadkp->dad_secshf; 1192 pktp->cp_secleft -= seccnt; 1193 1194 if (pktp->cp_secleft) { 1195 pktp->cp_srtsec += seccnt; 1196 } else { 1197 /* get the first cookie from the bad block list */ 1198 if (!pktp->cp_private) { 1199 bp = pktp->cp_bp; 1200 pktp->cp_srtsec = GET_BP_SEC(bp); 1201 pktp->cp_secleft = (bp->b_bcount >> dadkp->dad_secshf); 1202 } else { 1203 bbhckp = BBH_HTOC(dadkp->dad_bbhobjp, 1204 pktp->cp_private); 1205 pktp->cp_srtsec = BBH_GETCK_SECTOR(dadkp->dad_bbhobjp, 1206 bbhckp); 1207 pktp->cp_secleft = BBH_GETCK_SECLEN(dadkp->dad_bbhobjp, 1208 bbhckp); 1209 } 1210 } 1211 1212 pktp->cp_bytexfer = pktp->cp_secleft << dadkp->dad_secshf; 1213 1214 if (CTL_IOSETUP(dadkp->dad_ctlobjp, pktp)) { 1215 return (DDI_SUCCESS); 1216 } else { 1217 return (DDI_FAILURE); 1218 } 1219 1220 1221 1222 1223 } 1224 1225 static struct cmpkt * 1226 dadk_pktprep(struct dadk *dadkp, struct cmpkt *in_pktp, struct buf *bp, 1227 void (*cb_func)(struct buf *), int (*func)(caddr_t), caddr_t arg) 1228 { 1229 struct cmpkt *pktp; 1230 1231 pktp = gda_pktprep(dadkp->dad_ctlobjp, in_pktp, (opaque_t)bp, func, 1232 arg); 1233 1234 if (pktp) { 1235 pktp->cp_callback = dadk_pktcb; 1236 pktp->cp_time = DADK_IO_TIME; 1237 pktp->cp_flags = 0; 1238 pktp->cp_iodone = cb_func; 1239 pktp->cp_dev_private = (opaque_t)dadkp; 1240 1241 } 1242 1243 return (pktp); 1244 } 1245 1246 1247 static void 1248 dadk_restart(void *vpktp) 1249 { 1250 struct cmpkt *pktp = (struct cmpkt *)vpktp; 1251 1252 if (dadk_ioretry(pktp, QUE_COMMAND) == JUST_RETURN) 1253 return; 1254 pktp->cp_iodone(pktp->cp_bp); 1255 } 1256 1257 static int 1258 dadk_ioretry(struct cmpkt *pktp, int action) 1259 { 1260 struct buf *bp; 1261 struct dadk *dadkp = PKT2DADK(pktp); 1262 1263 switch (action) { 1264 case QUE_COMMAND: 1265 if (pktp->cp_retry++ < DADK_RETRY_COUNT) { 1266 CTL_IOSETUP(dadkp->dad_ctlobjp, pktp); 1267 if (CTL_TRANSPORT(dadkp->dad_ctlobjp, pktp) == 1268 CTL_SEND_SUCCESS) { 1269 return (JUST_RETURN); 1270 } 1271 gda_log(dadkp->dad_sd->sd_dev, dadk_name, 1272 CE_WARN, "transport of command fails\n"); 1273 } else 1274 gda_log(dadkp->dad_sd->sd_dev, 1275 dadk_name, CE_WARN, 1276 "exceeds maximum number of retries\n"); 1277 bioerror(pktp->cp_bp, ENXIO); 1278 /*FALLTHROUGH*/ 1279 case COMMAND_DONE_ERROR: 1280 bp = pktp->cp_bp; 1281 bp->b_resid += pktp->cp_byteleft - pktp->cp_bytexfer + 1282 pktp->cp_resid; 1283 if (geterror(bp) == 0) { 1284 if ((*((char *)(pktp->cp_cdbp)) == DCMD_FLUSH_CACHE) && 1285 (pktp->cp_dev_private == (opaque_t)dadkp) && 1286 ((int)(*(char *)pktp->cp_scbp) == DERR_ABORT)) { 1287 /* 1288 * Flag "unimplemented" responses for 1289 * DCMD_FLUSH_CACHE as ENOTSUP 1290 */ 1291 bioerror(bp, ENOTSUP); 1292 mutex_enter(&dadkp->dad_mutex); 1293 dadkp->dad_noflush = 1; 1294 mutex_exit(&dadkp->dad_mutex); 1295 } else { 1296 bioerror(bp, EIO); 1297 } 1298 } 1299 /*FALLTHROUGH*/ 1300 case COMMAND_DONE: 1301 default: 1302 return (COMMAND_DONE); 1303 } 1304 } 1305 1306 1307 static void 1308 dadk_pktcb(struct cmpkt *pktp) 1309 { 1310 int action; 1311 struct dadkio_rwcmd *rwcmdp; 1312 1313 rwcmdp = (struct dadkio_rwcmd *)pktp->cp_passthru; /* ioctl packet */ 1314 1315 if (pktp->cp_reason == CPS_SUCCESS) { 1316 if (rwcmdp && (rwcmdp != (opaque_t)DADK_SILENT)) 1317 rwcmdp->status.status = DADKIO_STAT_NO_ERROR; 1318 pktp->cp_iodone(pktp->cp_bp); 1319 return; 1320 } 1321 1322 if (rwcmdp && (rwcmdp != (opaque_t)DADK_SILENT)) { 1323 if (pktp->cp_reason == CPS_CHKERR) 1324 dadk_recorderr(pktp, rwcmdp); 1325 dadk_iodone(pktp->cp_bp); 1326 return; 1327 } 1328 1329 if (pktp->cp_reason == CPS_CHKERR) 1330 action = dadk_chkerr(pktp); 1331 else 1332 action = COMMAND_DONE_ERROR; 1333 1334 if (action == JUST_RETURN) 1335 return; 1336 1337 /* 1338 * If we are panicking don't retry the command 1339 * just fail it so we can go down completing all 1340 * of the buffers. 1341 */ 1342 if (ddi_in_panic() && action == QUE_COMMAND) 1343 action = COMMAND_DONE_ERROR; 1344 1345 if (action != COMMAND_DONE) { 1346 if ((dadk_ioretry(pktp, action)) == JUST_RETURN) 1347 return; 1348 } 1349 pktp->cp_iodone(pktp->cp_bp); 1350 } 1351 1352 1353 1354 static struct dadkio_derr dadk_errtab[] = { 1355 {COMMAND_DONE, GDA_INFORMATIONAL}, /* 0 DERR_SUCCESS */ 1356 {QUE_COMMAND, GDA_FATAL}, /* 1 DERR_AMNF */ 1357 {QUE_COMMAND, GDA_FATAL}, /* 2 DERR_TKONF */ 1358 {COMMAND_DONE_ERROR, GDA_INFORMATIONAL}, /* 3 DERR_ABORT */ 1359 {QUE_COMMAND, GDA_RETRYABLE}, /* 4 DERR_DWF */ 1360 {QUE_COMMAND, GDA_FATAL}, /* 5 DERR_IDNF */ 1361 {JUST_RETURN, GDA_INFORMATIONAL}, /* 6 DERR_BUSY */ 1362 {QUE_COMMAND, GDA_FATAL}, /* 7 DERR_UNC */ 1363 {QUE_COMMAND, GDA_RETRYABLE}, /* 8 DERR_BBK */ 1364 {COMMAND_DONE_ERROR, GDA_FATAL}, /* 9 DERR_INVCDB */ 1365 {COMMAND_DONE_ERROR, GDA_FATAL}, /* 10 DERR_HARD */ 1366 {COMMAND_DONE_ERROR, GDA_FATAL}, /* 11 DERR_ILI */ 1367 {COMMAND_DONE_ERROR, GDA_FATAL}, /* 12 DERR_EOM */ 1368 {COMMAND_DONE, GDA_INFORMATIONAL}, /* 13 DERR_MCR */ 1369 {COMMAND_DONE, GDA_INFORMATIONAL}, /* 14 DERR_RECOVER */ 1370 {COMMAND_DONE_ERROR, GDA_FATAL}, /* 15 DERR_NOTREADY */ 1371 {QUE_COMMAND, GDA_RETRYABLE}, /* 16 DERR_MEDIUM */ 1372 {COMMAND_DONE_ERROR, GDA_FATAL}, /* 17 DERR_HW */ 1373 {COMMAND_DONE, GDA_FATAL}, /* 18 DERR_ILL */ 1374 {COMMAND_DONE, GDA_FATAL}, /* 19 DERR_UNIT_ATTN */ 1375 {COMMAND_DONE_ERROR, GDA_FATAL}, /* 20 DERR_DATA_PROT */ 1376 {COMMAND_DONE_ERROR, GDA_FATAL}, /* 21 DERR_MISCOMPARE */ 1377 {QUE_COMMAND, GDA_RETRYABLE}, /* 22 DERR_ICRC */ 1378 {COMMAND_DONE_ERROR, GDA_FATAL}, /* 23 DERR_RESV */ 1379 }; 1380 1381 static int 1382 dadk_chkerr(struct cmpkt *pktp) 1383 { 1384 daddr_t err_blkno; 1385 struct dadk *dadkp = PKT2DADK(pktp); 1386 dadk_errstats_t *dep; 1387 int scb = *(char *)pktp->cp_scbp; 1388 1389 if (scb == DERR_SUCCESS) { 1390 if (pktp->cp_retry != 0 && dadkp->dad_errstats != NULL) { 1391 dep = (dadk_errstats_t *) 1392 dadkp->dad_errstats->ks_data; 1393 dep->dadk_rq_recov_err.value.ui32++; 1394 } 1395 return (COMMAND_DONE); 1396 } 1397 1398 if (pktp->cp_retry) { 1399 err_blkno = pktp->cp_srtsec + ((pktp->cp_bytexfer - 1400 pktp->cp_resid) >> dadkp->dad_secshf); 1401 } else 1402 err_blkno = -1; 1403 1404 if (dadkp->dad_errstats != NULL) { 1405 dep = (dadk_errstats_t *)dadkp->dad_errstats->ks_data; 1406 1407 switch (dadk_errtab[scb].d_severity) { 1408 case GDA_RETRYABLE: 1409 dep->dadk_softerrs.value.ui32++; 1410 break; 1411 1412 case GDA_FATAL: 1413 dep->dadk_harderrs.value.ui32++; 1414 break; 1415 1416 default: 1417 break; 1418 } 1419 1420 switch (scb) { 1421 case DERR_INVCDB: 1422 case DERR_ILI: 1423 case DERR_EOM: 1424 case DERR_HW: 1425 case DERR_ICRC: 1426 dep->dadk_transerrs.value.ui32++; 1427 break; 1428 1429 case DERR_AMNF: 1430 case DERR_TKONF: 1431 case DERR_DWF: 1432 case DERR_BBK: 1433 case DERR_UNC: 1434 case DERR_HARD: 1435 case DERR_MEDIUM: 1436 case DERR_DATA_PROT: 1437 case DERR_MISCOMP: 1438 dep->dadk_rq_media_err.value.ui32++; 1439 break; 1440 1441 case DERR_NOTREADY: 1442 dep->dadk_rq_ntrdy_err.value.ui32++; 1443 break; 1444 1445 case DERR_IDNF: 1446 case DERR_UNIT_ATTN: 1447 dep->dadk_rq_nodev_err.value.ui32++; 1448 break; 1449 1450 case DERR_ILL: 1451 case DERR_RESV: 1452 dep->dadk_rq_illrq_err.value.ui32++; 1453 break; 1454 1455 default: 1456 break; 1457 } 1458 } 1459 1460 /* if attempting to read a sector from a cdrom audio disk */ 1461 if ((dadkp->dad_cdrom) && 1462 (*((char *)(pktp->cp_cdbp)) == DCMD_READ) && 1463 (scb == DERR_ILL)) { 1464 return (COMMAND_DONE); 1465 } 1466 if (pktp->cp_passthru == NULL) { 1467 gda_errmsg(dadkp->dad_sd, pktp, dadk_name, 1468 dadk_errtab[scb].d_severity, pktp->cp_srtsec, 1469 err_blkno, dadk_cmds, dadk_sense); 1470 } 1471 1472 if (scb == DERR_BUSY) { 1473 (void) timeout(dadk_restart, (void *)pktp, DADK_BSY_TIMEOUT); 1474 } 1475 1476 return (dadk_errtab[scb].d_action); 1477 } 1478 1479 static void 1480 dadk_recorderr(struct cmpkt *pktp, struct dadkio_rwcmd *rwcmdp) 1481 { 1482 struct dadk *dadkp; 1483 int scb; 1484 1485 dadkp = PKT2DADK(pktp); 1486 scb = (int)(*(char *)pktp->cp_scbp); 1487 1488 1489 rwcmdp->status.failed_blk = rwcmdp->blkaddr + 1490 ((pktp->cp_bytexfer - pktp->cp_resid) >> dadkp->dad_secshf); 1491 1492 rwcmdp->status.resid = pktp->cp_bp->b_resid + 1493 pktp->cp_byteleft - pktp->cp_bytexfer + pktp->cp_resid; 1494 switch ((int)(* (char *)pktp->cp_scbp)) { 1495 case DERR_AMNF: 1496 case DERR_ABORT: 1497 rwcmdp->status.status = DADKIO_STAT_ILLEGAL_REQUEST; 1498 break; 1499 case DERR_DWF: 1500 case DERR_IDNF: 1501 rwcmdp->status.status = DADKIO_STAT_ILLEGAL_ADDRESS; 1502 break; 1503 case DERR_TKONF: 1504 case DERR_UNC: 1505 case DERR_BBK: 1506 rwcmdp->status.status = DADKIO_STAT_MEDIUM_ERROR; 1507 rwcmdp->status.failed_blk_is_valid = 1; 1508 rwcmdp->status.resid = 0; 1509 break; 1510 case DERR_BUSY: 1511 rwcmdp->status.status = DADKIO_STAT_NOT_READY; 1512 break; 1513 case DERR_INVCDB: 1514 case DERR_HARD: 1515 rwcmdp->status.status = DADKIO_STAT_HARDWARE_ERROR; 1516 break; 1517 case DERR_ICRC: 1518 default: 1519 rwcmdp->status.status = DADKIO_STAT_NOT_SUPPORTED; 1520 } 1521 1522 if (rwcmdp->flags & DADKIO_FLAG_SILENT) 1523 return; 1524 gda_errmsg(dadkp->dad_sd, pktp, dadk_name, dadk_errtab[scb].d_severity, 1525 rwcmdp->blkaddr, rwcmdp->status.failed_blk, 1526 dadk_cmds, dadk_sense); 1527 } 1528 1529 /*ARGSUSED*/ 1530 static void 1531 dadk_polldone(struct buf *bp) 1532 { 1533 struct cmpkt *pktp; 1534 struct dadk *dadkp; 1535 1536 pktp = GDA_BP_PKT(bp); 1537 dadkp = PKT2DADK(pktp); 1538 mutex_enter(&dadkp->dad_cmd_mutex); 1539 dadkp->dad_cmd_count--; 1540 mutex_exit(&dadkp->dad_cmd_mutex); 1541 } 1542 1543 static void 1544 dadk_iodone(struct buf *bp) 1545 { 1546 struct cmpkt *pktp; 1547 struct dadk *dadkp; 1548 1549 pktp = GDA_BP_PKT(bp); 1550 dadkp = PKT2DADK(pktp); 1551 1552 /* check for all iodone */ 1553 pktp->cp_byteleft -= pktp->cp_bytexfer; 1554 if (geterror(bp) == 0 && pktp->cp_byteleft != 0) { 1555 pktp->cp_retry = 0; 1556 (void) dadk_iosetup(dadkp, pktp); 1557 1558 1559 /* transport the next one */ 1560 if (CTL_TRANSPORT(dadkp->dad_ctlobjp, pktp) == CTL_SEND_SUCCESS) 1561 return; 1562 if ((dadk_ioretry(pktp, QUE_COMMAND)) == JUST_RETURN) 1563 return; 1564 } 1565 1566 /* start next one */ 1567 FLC_DEQUE(dadkp->dad_flcobjp, bp); 1568 1569 /* free pkt */ 1570 if (pktp->cp_private) 1571 BBH_FREEHANDLE(dadkp->dad_bbhobjp, pktp->cp_private); 1572 gda_free(dadkp->dad_ctlobjp, pktp, NULL); 1573 mutex_enter(&dadkp->dad_cmd_mutex); 1574 dadkp->dad_cmd_count--; 1575 mutex_exit(&dadkp->dad_cmd_mutex); 1576 biodone(bp); 1577 } 1578 1579 int 1580 dadk_check_media(opaque_t objp, int *state) 1581 { 1582 struct dadk *dadkp = (struct dadk *)objp; 1583 1584 if (!dadkp->dad_rmb) { 1585 return (ENXIO); 1586 } 1587 #ifdef DADK_DEBUG 1588 if (dadk_debug & DSTATE) 1589 PRF("dadk_check_media: user state %x disk state %x\n", 1590 *state, dadkp->dad_iostate); 1591 #endif 1592 /* 1593 * If state already changed just return 1594 */ 1595 if (*state != dadkp->dad_iostate) { 1596 *state = dadkp->dad_iostate; 1597 return (0); 1598 } 1599 1600 /* 1601 * Startup polling on thread state 1602 */ 1603 mutex_enter(&dadkp->dad_mutex); 1604 if (dadkp->dad_thread_cnt == 0) { 1605 /* 1606 * One thread per removable dadk device 1607 */ 1608 (void) thread_create(NULL, 0, dadk_watch_thread, dadkp, 0, &p0, 1609 TS_RUN, v.v_maxsyspri - 2); 1610 } 1611 dadkp->dad_thread_cnt++; 1612 1613 /* 1614 * Wait for state to change 1615 */ 1616 do { 1617 if (cv_wait_sig(&dadkp->dad_state_cv, &dadkp->dad_mutex) == 0) { 1618 dadkp->dad_thread_cnt--; 1619 mutex_exit(&dadkp->dad_mutex); 1620 return (EINTR); 1621 } 1622 } while (*state == dadkp->dad_iostate); 1623 *state = dadkp->dad_iostate; 1624 dadkp->dad_thread_cnt--; 1625 mutex_exit(&dadkp->dad_mutex); 1626 return (0); 1627 } 1628 1629 1630 #define MEDIA_ACCESS_DELAY 2000000 1631 1632 static void 1633 dadk_watch_thread(struct dadk *dadkp) 1634 { 1635 enum dkio_state state; 1636 int interval; 1637 1638 interval = drv_usectohz(dadk_check_media_time); 1639 1640 do { 1641 if (dadk_rmb_ioctl(dadkp, DCMD_GET_STATE, (intptr_t)&state, 0, 1642 DADK_SILENT)) { 1643 /* 1644 * Assume state remained the same 1645 */ 1646 state = dadkp->dad_iostate; 1647 } 1648 1649 /* 1650 * now signal the waiting thread if this is *not* the 1651 * specified state; 1652 * delay the signal if the state is DKIO_INSERTED 1653 * to allow the target to recover 1654 */ 1655 if (state != dadkp->dad_iostate) { 1656 1657 dadkp->dad_iostate = state; 1658 if (state == DKIO_INSERTED) { 1659 /* 1660 * delay the signal to give the drive a chance 1661 * to do what it apparently needs to do 1662 */ 1663 (void) timeout((void(*)(void *))cv_broadcast, 1664 (void *)&dadkp->dad_state_cv, 1665 drv_usectohz((clock_t)MEDIA_ACCESS_DELAY)); 1666 } else { 1667 cv_broadcast(&dadkp->dad_state_cv); 1668 } 1669 } 1670 delay(interval); 1671 } while (dadkp->dad_thread_cnt); 1672 } 1673 1674 int 1675 dadk_inquiry(opaque_t objp, opaque_t *inqpp) 1676 { 1677 struct dadk *dadkp = (struct dadk *)objp; 1678 struct scsi_inquiry **sinqpp = (struct scsi_inquiry **)inqpp; 1679 1680 if (dadkp && dadkp->dad_sd && dadkp->dad_sd->sd_inq) { 1681 *sinqpp = dadkp->dad_sd->sd_inq; 1682 return (DDI_SUCCESS); 1683 } 1684 1685 return (DDI_FAILURE); 1686 } 1687 1688 static int 1689 dadk_rmb_ioctl(struct dadk *dadkp, int cmd, intptr_t arg, int flags, int silent) 1690 1691 { 1692 struct buf *bp; 1693 int err; 1694 struct cmpkt *pktp; 1695 1696 if ((bp = getrbuf(KM_SLEEP)) == NULL) { 1697 return (ENOMEM); 1698 } 1699 pktp = dadk_pktprep(dadkp, NULL, bp, dadk_rmb_iodone, NULL, NULL); 1700 if (!pktp) { 1701 freerbuf(bp); 1702 return (ENOMEM); 1703 } 1704 bp->b_back = (struct buf *)arg; 1705 bp->b_forw = (struct buf *)dadkp->dad_flcobjp; 1706 pktp->cp_passthru = (opaque_t)(intptr_t)silent; 1707 1708 err = dadk_ctl_ioctl(dadkp, cmd, (uintptr_t)pktp, flags); 1709 freerbuf(bp); 1710 gda_free(dadkp->dad_ctlobjp, pktp, NULL); 1711 return (err); 1712 1713 1714 } 1715 1716 static void 1717 dadk_rmb_iodone(struct buf *bp) 1718 { 1719 struct cmpkt *pktp; 1720 struct dadk *dadkp; 1721 1722 pktp = GDA_BP_PKT(bp); 1723 dadkp = PKT2DADK(pktp); 1724 1725 bp->b_flags &= ~(B_DONE|B_BUSY); 1726 1727 /* Start next one */ 1728 FLC_DEQUE(dadkp->dad_flcobjp, bp); 1729 1730 mutex_enter(&dadkp->dad_cmd_mutex); 1731 dadkp->dad_cmd_count--; 1732 mutex_exit(&dadkp->dad_cmd_mutex); 1733 biodone(bp); 1734 } 1735 1736 static int 1737 dadk_dk_buf_setup(struct dadk *dadkp, opaque_t *cmdp, dev_t dev, 1738 enum uio_seg dataspace, int rw) 1739 { 1740 struct dadkio_rwcmd *rwcmdp = (struct dadkio_rwcmd *)cmdp; 1741 struct buf *bp; 1742 struct iovec aiov; 1743 struct uio auio; 1744 struct uio *uio = &auio; 1745 int status; 1746 1747 bp = getrbuf(KM_SLEEP); 1748 1749 bp->av_forw = bp->b_forw = (struct buf *)dadkp; 1750 bp->b_back = (struct buf *)rwcmdp; /* ioctl packet */ 1751 1752 bzero((caddr_t)&auio, sizeof (struct uio)); 1753 bzero((caddr_t)&aiov, sizeof (struct iovec)); 1754 aiov.iov_base = rwcmdp->bufaddr; 1755 aiov.iov_len = rwcmdp->buflen; 1756 uio->uio_iov = &aiov; 1757 1758 uio->uio_iovcnt = 1; 1759 uio->uio_resid = rwcmdp->buflen; 1760 uio->uio_segflg = dataspace; 1761 1762 /* Let physio do the rest... */ 1763 status = physio(dadk_dk_strategy, bp, dev, rw, dadkmin, uio); 1764 1765 freerbuf(bp); 1766 return (status); 1767 1768 } 1769 1770 /* Do not let a user gendisk request get too big or */ 1771 /* else we could use to many resources. */ 1772 1773 static void 1774 dadkmin(struct buf *bp) 1775 { 1776 if (bp->b_bcount > dadk_dk_maxphys) 1777 bp->b_bcount = dadk_dk_maxphys; 1778 } 1779 1780 static int 1781 dadk_dk_strategy(struct buf *bp) 1782 { 1783 dadk_dk((struct dadk *)bp->av_forw, (struct dadkio_rwcmd *)bp->b_back, 1784 bp); 1785 return (0); 1786 } 1787 1788 static void 1789 dadk_dk(struct dadk *dadkp, struct dadkio_rwcmd *rwcmdp, struct buf *bp) 1790 { 1791 struct cmpkt *pktp; 1792 1793 pktp = dadk_pktprep(dadkp, NULL, bp, dadk_iodone, NULL, NULL); 1794 if (!pktp) { 1795 bioerror(bp, ENOMEM); 1796 biodone(bp); 1797 return; 1798 } 1799 1800 pktp->cp_passthru = rwcmdp; 1801 1802 (void) dadk_ioprep(dadkp, pktp); 1803 1804 mutex_enter(&dadkp->dad_cmd_mutex); 1805 dadkp->dad_cmd_count++; 1806 mutex_exit(&dadkp->dad_cmd_mutex); 1807 FLC_ENQUE(dadkp->dad_flcobjp, bp); 1808 } 1809 1810 /* 1811 * There is no existing way to notify cmdk module 1812 * when the command completed, so add this function 1813 * to calculate how many on-going commands. 1814 */ 1815 int 1816 dadk_getcmds(opaque_t objp) 1817 { 1818 struct dadk *dadkp = (struct dadk *)objp; 1819 int count; 1820 1821 mutex_enter(&dadkp->dad_cmd_mutex); 1822 count = dadkp->dad_cmd_count; 1823 mutex_exit(&dadkp->dad_cmd_mutex); 1824 return (count); 1825 } 1826 1827 /* 1828 * this function was used to calc the cmd for CTL_IOCTL 1829 */ 1830 static int 1831 dadk_ctl_ioctl(struct dadk *dadkp, uint32_t cmd, uintptr_t arg, int flag) 1832 { 1833 int error; 1834 mutex_enter(&dadkp->dad_cmd_mutex); 1835 dadkp->dad_cmd_count++; 1836 mutex_exit(&dadkp->dad_cmd_mutex); 1837 error = CTL_IOCTL(dadkp->dad_ctlobjp, cmd, arg, flag); 1838 mutex_enter(&dadkp->dad_cmd_mutex); 1839 dadkp->dad_cmd_count--; 1840 mutex_exit(&dadkp->dad_cmd_mutex); 1841 return (error); 1842 } --- EOF ---