1 /*- 2 * Implementation of SCSI Direct Access Peripheral driver for CAM. 3 * 4 * Copyright (c) 1997 Justin T. Gibbs. 5 * All rights reserved. 6 * 7 * Redistribution and use in source and binary forms, with or without 8 * modification, are permitted provided that the following conditions 9 * are met: 10 * 1. Redistributions of source code must retain the above copyright 11 * notice, this list of conditions, and the following disclaimer, 12 * without modification, immediately at the beginning of the file. 13 * 2. The name of the author may not be used to endorse or promote products 14 * derived from this software without specific prior written permission. 15 * 16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 17 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR 20 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 22 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 24 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 25 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26 * SUCH DAMAGE. 27 */ 28 29 #include <sys/cdefs.h> 30 __FBSDID("$FreeBSD$"); 31 32 #include <sys/param.h> 33 34 #ifdef _KERNEL 35 #include <sys/systm.h> 36 #include <sys/kernel.h> 37 #include <sys/bio.h> 38 #include <sys/sysctl.h> 39 #include <sys/taskqueue.h> 40 #include <sys/lock.h> 41 #include <sys/mutex.h> 42 #include <sys/conf.h> 43 #include <sys/devicestat.h> 44 #include <sys/eventhandler.h> 45 #include <sys/malloc.h> 46 #include <sys/cons.h> 47 #include <geom/geom.h> 48 #include <geom/geom_disk.h> 49 #endif /* _KERNEL */ 50 51 #ifndef _KERNEL 52 #include <stdio.h> 53 #include <string.h> 54 #endif /* _KERNEL */ 55 56 #include <cam/cam.h> 57 #include <cam/cam_ccb.h> 58 #include <cam/cam_periph.h> 59 #include <cam/cam_xpt_periph.h> 60 #include <cam/cam_sim.h> 61 62 #include <cam/scsi/scsi_message.h> 63 64 #ifndef _KERNEL 65 #include <cam/scsi/scsi_da.h> 66 #endif /* !_KERNEL */ 67 68 #ifdef _KERNEL 69 typedef enum { 70 DA_STATE_PROBE, 71 DA_STATE_PROBE2, 72 DA_STATE_NORMAL 73 } da_state; 74 75 typedef enum { 76 DA_FLAG_PACK_INVALID = 0x001, 77 DA_FLAG_NEW_PACK = 0x002, 78 DA_FLAG_PACK_LOCKED = 0x004, 79 DA_FLAG_PACK_REMOVABLE = 0x008, 80 DA_FLAG_TAGGED_QUEUING = 0x010, 81 DA_FLAG_NEED_OTAG = 0x020, 82 DA_FLAG_WENT_IDLE = 0x040, 83 DA_FLAG_RETRY_UA = 0x080, 84 DA_FLAG_OPEN = 0x100, 85 DA_FLAG_SCTX_INIT = 0x200 86 } da_flags; 87 88 typedef enum { 89 DA_Q_NONE = 0x00, 90 DA_Q_NO_SYNC_CACHE = 0x01, 91 DA_Q_NO_6_BYTE = 0x02, 92 DA_Q_NO_PREVENT = 0x04, 93 DA_Q_4K = 0x08 94 } da_quirks; 95 96 typedef enum { 97 DA_CCB_PROBE = 0x01, 98 DA_CCB_PROBE2 = 0x02, 99 DA_CCB_BUFFER_IO = 0x03, 100 DA_CCB_WAITING = 0x04, 101 DA_CCB_DUMP = 0x05, 102 DA_CCB_TYPE_MASK = 0x0F, 103 DA_CCB_RETRY_UA = 0x10 104 } da_ccb_state; 105 106 /* Offsets into our private area for storing information */ 107 #define ccb_state ppriv_field0 108 #define ccb_bp ppriv_ptr1 109 110 struct disk_params { 111 u_int8_t heads; 112 u_int32_t cylinders; 113 u_int8_t secs_per_track; 114 u_int32_t secsize; /* Number of bytes/sector */ 115 u_int64_t sectors; /* total number sectors */ 116 u_int stripesize; 117 u_int stripeoffset; 118 }; 119 120 struct da_softc { 121 struct bio_queue_head bio_queue; 122 SLIST_ENTRY(da_softc) links; 123 LIST_HEAD(, ccb_hdr) pending_ccbs; 124 da_state state; 125 da_flags flags; 126 da_quirks quirks; 127 int minimum_cmd_size; 128 int ordered_tag_count; 129 int outstanding_cmds; 130 struct disk_params params; 131 struct disk *disk; 132 union ccb saved_ccb; 133 struct task sysctl_task; 134 struct sysctl_ctx_list sysctl_ctx; 135 struct sysctl_oid *sysctl_tree; 136 struct callout sendordered_c; 137 uint64_t wwpn; 138 }; 139 140 struct da_quirk_entry { 141 struct scsi_inquiry_pattern inq_pat; 142 da_quirks quirks; 143 }; 144 145 static const char quantum[] = "QUANTUM"; 146 static const char microp[] = "MICROP"; 147 148 static struct da_quirk_entry da_quirk_table[] = 149 { 150 /* SPI, FC devices */ 151 { 152 /* 153 * Fujitsu M2513A MO drives. 154 * Tested devices: M2513A2 firmware versions 1200 & 1300. 155 * (dip switch selects whether T_DIRECT or T_OPTICAL device) 156 * Reported by: W.Scholten <whs@xs4all.nl> 157 */ 158 {T_DIRECT, SIP_MEDIA_REMOVABLE, "FUJITSU", "M2513A", "*"}, 159 /*quirks*/ DA_Q_NO_SYNC_CACHE 160 }, 161 { 162 /* See above. */ 163 {T_OPTICAL, SIP_MEDIA_REMOVABLE, "FUJITSU", "M2513A", "*"}, 164 /*quirks*/ DA_Q_NO_SYNC_CACHE 165 }, 166 { 167 /* 168 * This particular Fujitsu drive doesn't like the 169 * synchronize cache command. 170 * Reported by: Tom Jackson <toj@gorilla.net> 171 */ 172 {T_DIRECT, SIP_MEDIA_FIXED, "FUJITSU", "M2954*", "*"}, 173 /*quirks*/ DA_Q_NO_SYNC_CACHE 174 }, 175 { 176 /* 177 * This drive doesn't like the synchronize cache command 178 * either. Reported by: Matthew Jacob <mjacob@feral.com> 179 * in NetBSD PR kern/6027, August 24, 1998. 180 */ 181 {T_DIRECT, SIP_MEDIA_FIXED, microp, "2217*", "*"}, 182 /*quirks*/ DA_Q_NO_SYNC_CACHE 183 }, 184 { 185 /* 186 * This drive doesn't like the synchronize cache command 187 * either. Reported by: Hellmuth Michaelis (hm@kts.org) 188 * (PR 8882). 189 */ 190 {T_DIRECT, SIP_MEDIA_FIXED, microp, "2112*", "*"}, 191 /*quirks*/ DA_Q_NO_SYNC_CACHE 192 }, 193 { 194 /* 195 * Doesn't like the synchronize cache command. 196 * Reported by: Blaz Zupan <blaz@gold.amis.net> 197 */ 198 {T_DIRECT, SIP_MEDIA_FIXED, "NEC", "D3847*", "*"}, 199 /*quirks*/ DA_Q_NO_SYNC_CACHE 200 }, 201 { 202 /* 203 * Doesn't like the synchronize cache command. 204 * Reported by: Blaz Zupan <blaz@gold.amis.net> 205 */ 206 {T_DIRECT, SIP_MEDIA_FIXED, quantum, "MAVERICK 540S", "*"}, 207 /*quirks*/ DA_Q_NO_SYNC_CACHE 208 }, 209 { 210 /* 211 * Doesn't like the synchronize cache command. 212 */ 213 {T_DIRECT, SIP_MEDIA_FIXED, quantum, "LPS525S", "*"}, 214 /*quirks*/ DA_Q_NO_SYNC_CACHE 215 }, 216 { 217 /* 218 * Doesn't like the synchronize cache command. 219 * Reported by: walter@pelissero.de 220 */ 221 {T_DIRECT, SIP_MEDIA_FIXED, quantum, "LPS540S", "*"}, 222 /*quirks*/ DA_Q_NO_SYNC_CACHE 223 }, 224 { 225 /* 226 * Doesn't work correctly with 6 byte reads/writes. 227 * Returns illegal request, and points to byte 9 of the 228 * 6-byte CDB. 229 * Reported by: Adam McDougall <bsdx@spawnet.com> 230 */ 231 {T_DIRECT, SIP_MEDIA_FIXED, quantum, "VIKING 4*", "*"}, 232 /*quirks*/ DA_Q_NO_6_BYTE 233 }, 234 { 235 /* See above. */ 236 {T_DIRECT, SIP_MEDIA_FIXED, quantum, "VIKING 2*", "*"}, 237 /*quirks*/ DA_Q_NO_6_BYTE 238 }, 239 { 240 /* 241 * Doesn't like the synchronize cache command. 242 * Reported by: walter@pelissero.de 243 */ 244 {T_DIRECT, SIP_MEDIA_FIXED, "CONNER", "CP3500*", "*"}, 245 /*quirks*/ DA_Q_NO_SYNC_CACHE 246 }, 247 { 248 /* 249 * The CISS RAID controllers do not support SYNC_CACHE 250 */ 251 {T_DIRECT, SIP_MEDIA_FIXED, "COMPAQ", "RAID*", "*"}, 252 /*quirks*/ DA_Q_NO_SYNC_CACHE 253 }, 254 /* USB mass storage devices supported by umass(4) */ 255 { 256 /* 257 * EXATELECOM (Sigmatel) i-Bead 100/105 USB Flash MP3 Player 258 * PR: kern/51675 259 */ 260 {T_DIRECT, SIP_MEDIA_REMOVABLE, "EXATEL", "i-BEAD10*", "*"}, 261 /*quirks*/ DA_Q_NO_SYNC_CACHE 262 }, 263 { 264 /* 265 * Power Quotient Int. (PQI) USB flash key 266 * PR: kern/53067 267 */ 268 {T_DIRECT, SIP_MEDIA_REMOVABLE, "Generic*", "USB Flash Disk*", 269 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 270 }, 271 { 272 /* 273 * Creative Nomad MUVO mp3 player (USB) 274 * PR: kern/53094 275 */ 276 {T_DIRECT, SIP_MEDIA_REMOVABLE, "CREATIVE", "NOMAD_MUVO", "*"}, 277 /*quirks*/ DA_Q_NO_SYNC_CACHE|DA_Q_NO_PREVENT 278 }, 279 { 280 /* 281 * Jungsoft NEXDISK USB flash key 282 * PR: kern/54737 283 */ 284 {T_DIRECT, SIP_MEDIA_REMOVABLE, "JUNGSOFT", "NEXDISK*", "*"}, 285 /*quirks*/ DA_Q_NO_SYNC_CACHE 286 }, 287 { 288 /* 289 * FreeDik USB Mini Data Drive 290 * PR: kern/54786 291 */ 292 {T_DIRECT, SIP_MEDIA_REMOVABLE, "FreeDik*", "Mini Data Drive", 293 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 294 }, 295 { 296 /* 297 * Sigmatel USB Flash MP3 Player 298 * PR: kern/57046 299 */ 300 {T_DIRECT, SIP_MEDIA_REMOVABLE, "SigmaTel", "MSCN", "*"}, 301 /*quirks*/ DA_Q_NO_SYNC_CACHE|DA_Q_NO_PREVENT 302 }, 303 { 304 /* 305 * Neuros USB Digital Audio Computer 306 * PR: kern/63645 307 */ 308 {T_DIRECT, SIP_MEDIA_REMOVABLE, "NEUROS", "dig. audio comp.", 309 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 310 }, 311 { 312 /* 313 * SEAGRAND NP-900 MP3 Player 314 * PR: kern/64563 315 */ 316 {T_DIRECT, SIP_MEDIA_REMOVABLE, "SEAGRAND", "NP-900*", "*"}, 317 /*quirks*/ DA_Q_NO_SYNC_CACHE|DA_Q_NO_PREVENT 318 }, 319 { 320 /* 321 * iRiver iFP MP3 player (with UMS Firmware) 322 * PR: kern/54881, i386/63941, kern/66124 323 */ 324 {T_DIRECT, SIP_MEDIA_REMOVABLE, "iRiver", "iFP*", "*"}, 325 /*quirks*/ DA_Q_NO_SYNC_CACHE 326 }, 327 { 328 /* 329 * Frontier Labs NEX IA+ Digital Audio Player, rev 1.10/0.01 330 * PR: kern/70158 331 */ 332 {T_DIRECT, SIP_MEDIA_REMOVABLE, "FL" , "Nex*", "*"}, 333 /*quirks*/ DA_Q_NO_SYNC_CACHE 334 }, 335 { 336 /* 337 * ZICPlay USB MP3 Player with FM 338 * PR: kern/75057 339 */ 340 {T_DIRECT, SIP_MEDIA_REMOVABLE, "ACTIONS*" , "USB DISK*", "*"}, 341 /*quirks*/ DA_Q_NO_SYNC_CACHE 342 }, 343 { 344 /* 345 * TEAC USB floppy mechanisms 346 */ 347 {T_DIRECT, SIP_MEDIA_REMOVABLE, "TEAC" , "FD-05*", "*"}, 348 /*quirks*/ DA_Q_NO_SYNC_CACHE 349 }, 350 { 351 /* 352 * Kingston DataTraveler II+ USB Pen-Drive. 353 * Reported by: Pawel Jakub Dawidek <pjd@FreeBSD.org> 354 */ 355 {T_DIRECT, SIP_MEDIA_REMOVABLE, "Kingston" , "DataTraveler II+", 356 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 357 }, 358 { 359 /* 360 * Motorola E398 Mobile Phone (TransFlash memory card). 361 * Reported by: Wojciech A. Koszek <dunstan@FreeBSD.czest.pl> 362 * PR: usb/89889 363 */ 364 {T_DIRECT, SIP_MEDIA_REMOVABLE, "Motorola" , "Motorola Phone", 365 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 366 }, 367 { 368 /* 369 * Qware BeatZkey! Pro 370 * PR: usb/79164 371 */ 372 {T_DIRECT, SIP_MEDIA_REMOVABLE, "GENERIC", "USB DISK DEVICE", 373 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 374 }, 375 { 376 /* 377 * Time DPA20B 1GB MP3 Player 378 * PR: usb/81846 379 */ 380 {T_DIRECT, SIP_MEDIA_REMOVABLE, "USB2.0*", "(FS) FLASH DISK*", 381 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 382 }, 383 { 384 /* 385 * Samsung USB key 128Mb 386 * PR: usb/90081 387 */ 388 {T_DIRECT, SIP_MEDIA_REMOVABLE, "USB-DISK", "FreeDik-FlashUsb", 389 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 390 }, 391 { 392 /* 393 * Kingston DataTraveler 2.0 USB Flash memory. 394 * PR: usb/89196 395 */ 396 {T_DIRECT, SIP_MEDIA_REMOVABLE, "Kingston", "DataTraveler 2.0", 397 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 398 }, 399 { 400 /* 401 * Creative MUVO Slim mp3 player (USB) 402 * PR: usb/86131 403 */ 404 {T_DIRECT, SIP_MEDIA_REMOVABLE, "CREATIVE", "MuVo Slim", 405 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE|DA_Q_NO_PREVENT 406 }, 407 { 408 /* 409 * United MP5512 Portable MP3 Player (2-in-1 USB DISK/MP3) 410 * PR: usb/80487 411 */ 412 {T_DIRECT, SIP_MEDIA_REMOVABLE, "Generic*", "MUSIC DISK", 413 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 414 }, 415 { 416 /* 417 * SanDisk Micro Cruzer 128MB 418 * PR: usb/75970 419 */ 420 {T_DIRECT, SIP_MEDIA_REMOVABLE, "SanDisk" , "Micro Cruzer", 421 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 422 }, 423 { 424 /* 425 * TOSHIBA TransMemory USB sticks 426 * PR: kern/94660 427 */ 428 {T_DIRECT, SIP_MEDIA_REMOVABLE, "TOSHIBA", "TransMemory", 429 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 430 }, 431 { 432 /* 433 * PNY USB Flash keys 434 * PR: usb/75578, usb/72344, usb/65436 435 */ 436 {T_DIRECT, SIP_MEDIA_REMOVABLE, "*" , "USB DISK*", 437 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 438 }, 439 { 440 /* 441 * Genesys 6-in-1 Card Reader 442 * PR: usb/94647 443 */ 444 {T_DIRECT, SIP_MEDIA_REMOVABLE, "Generic*", "STORAGE DEVICE*", 445 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 446 }, 447 { 448 /* 449 * Rekam Digital CAMERA 450 * PR: usb/98713 451 */ 452 {T_DIRECT, SIP_MEDIA_REMOVABLE, "CAMERA*", "4MP-9J6*", 453 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 454 }, 455 { 456 /* 457 * iRiver H10 MP3 player 458 * PR: usb/102547 459 */ 460 {T_DIRECT, SIP_MEDIA_REMOVABLE, "iriver", "H10*", 461 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 462 }, 463 { 464 /* 465 * iRiver U10 MP3 player 466 * PR: usb/92306 467 */ 468 {T_DIRECT, SIP_MEDIA_REMOVABLE, "iriver", "U10*", 469 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 470 }, 471 { 472 /* 473 * X-Micro Flash Disk 474 * PR: usb/96901 475 */ 476 {T_DIRECT, SIP_MEDIA_REMOVABLE, "X-Micro", "Flash Disk", 477 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 478 }, 479 { 480 /* 481 * EasyMP3 EM732X USB 2.0 Flash MP3 Player 482 * PR: usb/96546 483 */ 484 {T_DIRECT, SIP_MEDIA_REMOVABLE, "EM732X", "MP3 Player*", 485 "1.00"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 486 }, 487 { 488 /* 489 * Denver MP3 player 490 * PR: usb/107101 491 */ 492 {T_DIRECT, SIP_MEDIA_REMOVABLE, "DENVER", "MP3 PLAYER", 493 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 494 }, 495 { 496 /* 497 * Philips USB Key Audio KEY013 498 * PR: usb/68412 499 */ 500 {T_DIRECT, SIP_MEDIA_REMOVABLE, "PHILIPS", "Key*", "*"}, 501 /*quirks*/ DA_Q_NO_SYNC_CACHE | DA_Q_NO_PREVENT 502 }, 503 { 504 /* 505 * JNC MP3 Player 506 * PR: usb/94439 507 */ 508 {T_DIRECT, SIP_MEDIA_REMOVABLE, "JNC*" , "MP3 Player*", 509 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 510 }, 511 { 512 /* 513 * SAMSUNG MP0402H 514 * PR: usb/108427 515 */ 516 {T_DIRECT, SIP_MEDIA_FIXED, "SAMSUNG", "MP0402H", "*"}, 517 /*quirks*/ DA_Q_NO_SYNC_CACHE 518 }, 519 { 520 /* 521 * I/O Magic USB flash - Giga Bank 522 * PR: usb/108810 523 */ 524 {T_DIRECT, SIP_MEDIA_FIXED, "GS-Magic", "stor*", "*"}, 525 /*quirks*/ DA_Q_NO_SYNC_CACHE 526 }, 527 { 528 /* 529 * JoyFly 128mb USB Flash Drive 530 * PR: 96133 531 */ 532 {T_DIRECT, SIP_MEDIA_REMOVABLE, "USB 2.0", "Flash Disk*", 533 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 534 }, 535 { 536 /* 537 * ChipsBnk usb stick 538 * PR: 103702 539 */ 540 {T_DIRECT, SIP_MEDIA_REMOVABLE, "ChipsBnk", "USB*", 541 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 542 }, 543 { 544 /* 545 * Storcase (Kingston) InfoStation IFS FC2/SATA-R 201A 546 * PR: 129858 547 */ 548 {T_DIRECT, SIP_MEDIA_FIXED, "IFS", "FC2/SATA-R*", 549 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 550 }, 551 { 552 /* 553 * Samsung YP-U3 mp3-player 554 * PR: 125398 555 */ 556 {T_DIRECT, SIP_MEDIA_REMOVABLE, "Samsung", "YP-U3", 557 "*"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 558 }, 559 { 560 {T_DIRECT, SIP_MEDIA_REMOVABLE, "Netac", "OnlyDisk*", 561 "2000"}, /*quirks*/ DA_Q_NO_SYNC_CACHE 562 }, 563 { 564 /* 565 * Sony Cyber-Shot DSC cameras 566 * PR: usb/137035 567 */ 568 {T_DIRECT, SIP_MEDIA_REMOVABLE, "Sony", "Sony DSC", "*"}, 569 /*quirks*/ DA_Q_NO_SYNC_CACHE | DA_Q_NO_PREVENT 570 }, 571 /* ATA/SATA devices over SAS/USB/... */ 572 { 573 /* Hitachi Advanced Format (4k) drives */ 574 { T_DIRECT, SIP_MEDIA_FIXED, "Hitachi", "H??????????E3*", "*" }, 575 /*quirks*/DA_Q_4K 576 }, 577 { 578 /* Samsung Advanced Format (4k) drives */ 579 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "SAMSUNG HD155UI*", "*" }, 580 /*quirks*/DA_Q_4K 581 }, 582 { 583 /* Samsung Advanced Format (4k) drives */ 584 { T_DIRECT, SIP_MEDIA_FIXED, "SAMSUNG", "HD155UI*", "*" }, 585 /*quirks*/DA_Q_4K 586 }, 587 { 588 /* Samsung Advanced Format (4k) drives */ 589 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "SAMSUNG HD204UI*", "*" }, 590 /*quirks*/DA_Q_4K 591 }, 592 { 593 /* Samsung Advanced Format (4k) drives */ 594 { T_DIRECT, SIP_MEDIA_FIXED, "SAMSUNG", "HD204UI*", "*" }, 595 /*quirks*/DA_Q_4K 596 }, 597 { 598 /* Seagate Barracuda Green Advanced Format (4k) drives */ 599 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "ST????DL*", "*" }, 600 /*quirks*/DA_Q_4K 601 }, 602 { 603 /* Seagate Barracuda Green Advanced Format (4k) drives */ 604 { T_DIRECT, SIP_MEDIA_FIXED, "ST????DL", "*", "*" }, 605 /*quirks*/DA_Q_4K 606 }, 607 { 608 /* Seagate Barracuda Green Advanced Format (4k) drives */ 609 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "ST???DM*", "*" }, 610 /*quirks*/DA_Q_4K 611 }, 612 { 613 /* Seagate Barracuda Green Advanced Format (4k) drives */ 614 { T_DIRECT, SIP_MEDIA_FIXED, "ST???DM*", "*", "*" }, 615 /*quirks*/DA_Q_4K 616 }, 617 { 618 /* Seagate Barracuda Green Advanced Format (4k) drives */ 619 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "ST????DM*", "*" }, 620 /*quirks*/DA_Q_4K 621 }, 622 { 623 /* Seagate Barracuda Green Advanced Format (4k) drives */ 624 { T_DIRECT, SIP_MEDIA_FIXED, "ST????DM", "*", "*" }, 625 /*quirks*/DA_Q_4K 626 }, 627 { 628 /* Seagate Momentus Advanced Format (4k) drives */ 629 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "ST9500423AS*", "*" }, 630 /*quirks*/DA_Q_4K 631 }, 632 { 633 /* Seagate Momentus Advanced Format (4k) drives */ 634 { T_DIRECT, SIP_MEDIA_FIXED, "ST950042", "3AS*", "*" }, 635 /*quirks*/DA_Q_4K 636 }, 637 { 638 /* Seagate Momentus Advanced Format (4k) drives */ 639 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "ST9500424AS*", "*" }, 640 /*quirks*/DA_Q_4K 641 }, 642 { 643 /* Seagate Momentus Advanced Format (4k) drives */ 644 { T_DIRECT, SIP_MEDIA_FIXED, "ST950042", "4AS*", "*" }, 645 /*quirks*/DA_Q_4K 646 }, 647 { 648 /* Seagate Momentus Advanced Format (4k) drives */ 649 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "ST9640423AS*", "*" }, 650 /*quirks*/DA_Q_4K 651 }, 652 { 653 /* Seagate Momentus Advanced Format (4k) drives */ 654 { T_DIRECT, SIP_MEDIA_FIXED, "ST964042", "3AS*", "*" }, 655 /*quirks*/DA_Q_4K 656 }, 657 { 658 /* Seagate Momentus Advanced Format (4k) drives */ 659 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "ST9640424AS*", "*" }, 660 /*quirks*/DA_Q_4K 661 }, 662 { 663 /* Seagate Momentus Advanced Format (4k) drives */ 664 { T_DIRECT, SIP_MEDIA_FIXED, "ST964042", "4AS*", "*" }, 665 /*quirks*/DA_Q_4K 666 }, 667 { 668 /* Seagate Momentus Advanced Format (4k) drives */ 669 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "ST9750420AS*", "*" }, 670 /*quirks*/DA_Q_4K 671 }, 672 { 673 /* Seagate Momentus Advanced Format (4k) drives */ 674 { T_DIRECT, SIP_MEDIA_FIXED, "ST975042", "0AS*", "*" }, 675 /*quirks*/DA_Q_4K 676 }, 677 { 678 /* Seagate Momentus Advanced Format (4k) drives */ 679 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "ST9750422AS*", "*" }, 680 /*quirks*/DA_Q_4K 681 }, 682 { 683 /* Seagate Momentus Advanced Format (4k) drives */ 684 { T_DIRECT, SIP_MEDIA_FIXED, "ST975042", "2AS*", "*" }, 685 /*quirks*/DA_Q_4K 686 }, 687 { 688 /* Seagate Momentus Advanced Format (4k) drives */ 689 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "ST9750423AS*", "*" }, 690 /*quirks*/DA_Q_4K 691 }, 692 { 693 /* Seagate Momentus Advanced Format (4k) drives */ 694 { T_DIRECT, SIP_MEDIA_FIXED, "ST975042", "3AS*", "*" }, 695 /*quirks*/DA_Q_4K 696 }, 697 { 698 /* Seagate Momentus Thin Advanced Format (4k) drives */ 699 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "ST???LT*", "*" }, 700 /*quirks*/DA_Q_4K 701 }, 702 { 703 /* Seagate Momentus Thin Advanced Format (4k) drives */ 704 { T_DIRECT, SIP_MEDIA_FIXED, "ST???LT*", "*", "*" }, 705 /*quirks*/DA_Q_4K 706 }, 707 { 708 /* WDC Caviar Green Advanced Format (4k) drives */ 709 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "WDC WD????RS*", "*" }, 710 /*quirks*/DA_Q_4K 711 }, 712 { 713 /* WDC Caviar Green Advanced Format (4k) drives */ 714 { T_DIRECT, SIP_MEDIA_FIXED, "WDC WD??", "??RS*", "*" }, 715 /*quirks*/DA_Q_4K 716 }, 717 { 718 /* WDC Caviar Green Advanced Format (4k) drives */ 719 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "WDC WD????RX*", "*" }, 720 /*quirks*/DA_Q_4K 721 }, 722 { 723 /* WDC Caviar Green Advanced Format (4k) drives */ 724 { T_DIRECT, SIP_MEDIA_FIXED, "WDC WD??", "??RX*", "*" }, 725 /*quirks*/DA_Q_4K 726 }, 727 { 728 /* WDC Caviar Green Advanced Format (4k) drives */ 729 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "WDC WD??????RS*", "*" }, 730 /*quirks*/DA_Q_4K 731 }, 732 { 733 /* WDC Caviar Green Advanced Format (4k) drives */ 734 { T_DIRECT, SIP_MEDIA_FIXED, "WDC WD??", "????RS*", "*" }, 735 /*quirks*/DA_Q_4K 736 }, 737 { 738 /* WDC Caviar Green Advanced Format (4k) drives */ 739 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "WDC WD??????RX*", "*" }, 740 /*quirks*/DA_Q_4K 741 }, 742 { 743 /* WDC Caviar Green Advanced Format (4k) drives */ 744 { T_DIRECT, SIP_MEDIA_FIXED, "WDC WD??", "????RX*", "*" }, 745 /*quirks*/DA_Q_4K 746 }, 747 { 748 /* WDC Scorpio Black Advanced Format (4k) drives */ 749 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "WDC WD???PKT*", "*" }, 750 /*quirks*/DA_Q_4K 751 }, 752 { 753 /* WDC Scorpio Black Advanced Format (4k) drives */ 754 { T_DIRECT, SIP_MEDIA_FIXED, "WDC WD??", "?PKT*", "*" }, 755 /*quirks*/DA_Q_4K 756 }, 757 { 758 /* WDC Scorpio Black Advanced Format (4k) drives */ 759 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "WDC WD?????PKT*", "*" }, 760 /*quirks*/DA_Q_4K 761 }, 762 { 763 /* WDC Scorpio Black Advanced Format (4k) drives */ 764 { T_DIRECT, SIP_MEDIA_FIXED, "WDC WD??", "???PKT*", "*" }, 765 /*quirks*/DA_Q_4K 766 }, 767 { 768 /* WDC Scorpio Blue Advanced Format (4k) drives */ 769 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "WDC WD???PVT*", "*" }, 770 /*quirks*/DA_Q_4K 771 }, 772 { 773 /* WDC Scorpio Blue Advanced Format (4k) drives */ 774 { T_DIRECT, SIP_MEDIA_FIXED, "WDC WD??", "?PVT*", "*" }, 775 /*quirks*/DA_Q_4K 776 }, 777 { 778 /* WDC Scorpio Blue Advanced Format (4k) drives */ 779 { T_DIRECT, SIP_MEDIA_FIXED, "ATA", "WDC WD?????PVT*", "*" }, 780 /*quirks*/DA_Q_4K 781 }, 782 { 783 /* WDC Scorpio Blue Advanced Format (4k) drives */ 784 { T_DIRECT, SIP_MEDIA_FIXED, "WDC WD??", "???PVT*", "*" }, 785 /*quirks*/DA_Q_4K 786 }, 787 }; 788 789 static disk_strategy_t dastrategy; 790 static dumper_t dadump; 791 static periph_init_t dainit; 792 static void daasync(void *callback_arg, u_int32_t code, 793 struct cam_path *path, void *arg); 794 static void dasysctlinit(void *context, int pending); 795 static int dacmdsizesysctl(SYSCTL_HANDLER_ARGS); 796 static periph_ctor_t daregister; 797 static periph_dtor_t dacleanup; 798 static periph_start_t dastart; 799 static periph_oninv_t daoninvalidate; 800 static void dadone(struct cam_periph *periph, 801 union ccb *done_ccb); 802 static int daerror(union ccb *ccb, u_int32_t cam_flags, 803 u_int32_t sense_flags); 804 static void daprevent(struct cam_periph *periph, int action); 805 static int dagetcapacity(struct cam_periph *periph); 806 static void dasetgeom(struct cam_periph *periph, uint32_t block_len, 807 uint64_t maxsector, u_int lbppbe, u_int lalba); 808 static timeout_t dasendorderedtag; 809 static void dashutdown(void *arg, int howto); 810 811 #ifndef DA_DEFAULT_TIMEOUT 812 #define DA_DEFAULT_TIMEOUT 60 /* Timeout in seconds */ 813 #endif 814 815 #ifndef DA_DEFAULT_RETRY 816 #define DA_DEFAULT_RETRY 4 817 #endif 818 819 #ifndef DA_DEFAULT_SEND_ORDERED 820 #define DA_DEFAULT_SEND_ORDERED 1 821 #endif 822 823 824 static int da_retry_count = DA_DEFAULT_RETRY; 825 static int da_default_timeout = DA_DEFAULT_TIMEOUT; 826 static int da_send_ordered = DA_DEFAULT_SEND_ORDERED; 827 828 static SYSCTL_NODE(_kern_cam, OID_AUTO, da, CTLFLAG_RD, 0, 829 "CAM Direct Access Disk driver"); 830 SYSCTL_INT(_kern_cam_da, OID_AUTO, retry_count, CTLFLAG_RW, 831 &da_retry_count, 0, "Normal I/O retry count"); 832 TUNABLE_INT("kern.cam.da.retry_count", &da_retry_count); 833 SYSCTL_INT(_kern_cam_da, OID_AUTO, default_timeout, CTLFLAG_RW, 834 &da_default_timeout, 0, "Normal I/O timeout (in seconds)"); 835 TUNABLE_INT("kern.cam.da.default_timeout", &da_default_timeout); 836 SYSCTL_INT(_kern_cam_da, OID_AUTO, da_send_ordered, CTLFLAG_RW, 837 &da_send_ordered, 0, "Send Ordered Tags"); 838 TUNABLE_INT("kern.cam.da.da_send_ordered", &da_send_ordered); 839 840 /* 841 * DA_ORDEREDTAG_INTERVAL determines how often, relative 842 * to the default timeout, we check to see whether an ordered 843 * tagged transaction is appropriate to prevent simple tag 844 * starvation. Since we'd like to ensure that there is at least 845 * 1/2 of the timeout length left for a starved transaction to 846 * complete after we've sent an ordered tag, we must poll at least 847 * four times in every timeout period. This takes care of the worst 848 * case where a starved transaction starts during an interval that 849 * meets the requirement "don't send an ordered tag" test so it takes 850 * us two intervals to determine that a tag must be sent. 851 */ 852 #ifndef DA_ORDEREDTAG_INTERVAL 853 #define DA_ORDEREDTAG_INTERVAL 4 854 #endif 855 856 static struct periph_driver dadriver = 857 { 858 dainit, "da", 859 TAILQ_HEAD_INITIALIZER(dadriver.units), /* generation */ 0 860 }; 861 862 PERIPHDRIVER_DECLARE(da, dadriver); 863 864 static MALLOC_DEFINE(M_SCSIDA, "scsi_da", "scsi_da buffers"); 865 866 static int 867 daopen(struct disk *dp) 868 { 869 struct cam_periph *periph; 870 struct da_softc *softc; 871 int unit; 872 int error; 873 874 periph = (struct cam_periph *)dp->d_drv1; 875 if (periph == NULL) { 876 return (ENXIO); 877 } 878 879 if (cam_periph_acquire(periph) != CAM_REQ_CMP) { 880 return(ENXIO); 881 } 882 883 cam_periph_lock(periph); 884 if ((error = cam_periph_hold(periph, PRIBIO|PCATCH)) != 0) { 885 cam_periph_unlock(periph); 886 cam_periph_release(periph); 887 return (error); 888 } 889 890 unit = periph->unit_number; 891 softc = (struct da_softc *)periph->softc; 892 softc->flags |= DA_FLAG_OPEN; 893 894 CAM_DEBUG(periph->path, CAM_DEBUG_TRACE, 895 ("daopen: disk=%s%d (unit %d)\n", dp->d_name, dp->d_unit, 896 unit)); 897 898 if ((softc->flags & DA_FLAG_PACK_INVALID) != 0) { 899 /* Invalidate our pack information. */ 900 softc->flags &= ~DA_FLAG_PACK_INVALID; 901 } 902 903 error = dagetcapacity(periph); 904 905 if (error == 0) { 906 907 softc->disk->d_sectorsize = softc->params.secsize; 908 softc->disk->d_mediasize = softc->params.secsize * (off_t)softc->params.sectors; 909 softc->disk->d_stripesize = softc->params.stripesize; 910 softc->disk->d_stripeoffset = softc->params.stripeoffset; 911 /* XXX: these are not actually "firmware" values, so they may be wrong */ 912 softc->disk->d_fwsectors = softc->params.secs_per_track; 913 softc->disk->d_fwheads = softc->params.heads; 914 softc->disk->d_devstat->block_size = softc->params.secsize; 915 softc->disk->d_devstat->flags &= ~DEVSTAT_BS_UNAVAILABLE; 916 917 if ((softc->flags & DA_FLAG_PACK_REMOVABLE) != 0 && 918 (softc->quirks & DA_Q_NO_PREVENT) == 0) 919 daprevent(periph, PR_PREVENT); 920 } else 921 softc->flags &= ~DA_FLAG_OPEN; 922 923 cam_periph_unhold(periph); 924 cam_periph_unlock(periph); 925 926 if (error != 0) { 927 cam_periph_release(periph); 928 } 929 return (error); 930 } 931 932 static int 933 daclose(struct disk *dp) 934 { 935 struct cam_periph *periph; 936 struct da_softc *softc; 937 int error; 938 939 periph = (struct cam_periph *)dp->d_drv1; 940 if (periph == NULL) 941 return (ENXIO); 942 943 cam_periph_lock(periph); 944 if ((error = cam_periph_hold(periph, PRIBIO)) != 0) { 945 cam_periph_unlock(periph); 946 cam_periph_release(periph); 947 return (error); 948 } 949 950 softc = (struct da_softc *)periph->softc; 951 952 if ((softc->quirks & DA_Q_NO_SYNC_CACHE) == 0 953 && (softc->flags & DA_FLAG_PACK_INVALID) == 0) { 954 union ccb *ccb; 955 956 ccb = cam_periph_getccb(periph, CAM_PRIORITY_NORMAL); 957 958 scsi_synchronize_cache(&ccb->csio, 959 /*retries*/1, 960 /*cbfcnp*/dadone, 961 MSG_SIMPLE_Q_TAG, 962 /*begin_lba*/0,/* Cover the whole disk */ 963 /*lb_count*/0, 964 SSD_FULL_SIZE, 965 5 * 60 * 1000); 966 967 cam_periph_runccb(ccb, /*error_routine*/NULL, /*cam_flags*/0, 968 /*sense_flags*/SF_RETRY_UA, 969 softc->disk->d_devstat); 970 971 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 972 if ((ccb->ccb_h.status & CAM_STATUS_MASK) == 973 CAM_SCSI_STATUS_ERROR) { 974 int asc, ascq; 975 int sense_key, error_code; 976 977 scsi_extract_sense_len(&ccb->csio.sense_data, 978 ccb->csio.sense_len - ccb->csio.sense_resid, 979 &error_code, &sense_key, &asc, &ascq, 980 /*show_errors*/ 1); 981 if (sense_key != SSD_KEY_ILLEGAL_REQUEST) 982 scsi_sense_print(&ccb->csio); 983 } else { 984 xpt_print(periph->path, "Synchronize cache " 985 "failed, status == 0x%x, scsi status == " 986 "0x%x\n", ccb->csio.ccb_h.status, 987 ccb->csio.scsi_status); 988 } 989 } 990 991 if ((ccb->ccb_h.status & CAM_DEV_QFRZN) != 0) 992 cam_release_devq(ccb->ccb_h.path, 993 /*relsim_flags*/0, 994 /*reduction*/0, 995 /*timeout*/0, 996 /*getcount_only*/0); 997 998 xpt_release_ccb(ccb); 999 1000 } 1001 1002 if ((softc->flags & DA_FLAG_PACK_REMOVABLE) != 0) { 1003 if ((softc->quirks & DA_Q_NO_PREVENT) == 0) 1004 daprevent(periph, PR_ALLOW); 1005 /* 1006 * If we've got removeable media, mark the blocksize as 1007 * unavailable, since it could change when new media is 1008 * inserted. 1009 */ 1010 softc->disk->d_devstat->flags |= DEVSTAT_BS_UNAVAILABLE; 1011 } 1012 1013 softc->flags &= ~DA_FLAG_OPEN; 1014 cam_periph_unhold(periph); 1015 cam_periph_unlock(periph); 1016 cam_periph_release(periph); 1017 return (0); 1018 } 1019 1020 /* 1021 * Actually translate the requested transfer into one the physical driver 1022 * can understand. The transfer is described by a buf and will include 1023 * only one physical transfer. 1024 */ 1025 static void 1026 dastrategy(struct bio *bp) 1027 { 1028 struct cam_periph *periph; 1029 struct da_softc *softc; 1030 1031 periph = (struct cam_periph *)bp->bio_disk->d_drv1; 1032 if (periph == NULL) { 1033 biofinish(bp, NULL, ENXIO); 1034 return; 1035 } 1036 softc = (struct da_softc *)periph->softc; 1037 1038 cam_periph_lock(periph); 1039 1040 /* 1041 * If the device has been made invalid, error out 1042 */ 1043 if ((softc->flags & DA_FLAG_PACK_INVALID)) { 1044 cam_periph_unlock(periph); 1045 biofinish(bp, NULL, ENXIO); 1046 return; 1047 } 1048 1049 /* 1050 * Place it in the queue of disk activities for this disk 1051 */ 1052 bioq_disksort(&softc->bio_queue, bp); 1053 1054 /* 1055 * Schedule ourselves for performing the work. 1056 */ 1057 xpt_schedule(periph, CAM_PRIORITY_NORMAL); 1058 cam_periph_unlock(periph); 1059 1060 return; 1061 } 1062 1063 static int 1064 dadump(void *arg, void *virtual, vm_offset_t physical, off_t offset, size_t length) 1065 { 1066 struct cam_periph *periph; 1067 struct da_softc *softc; 1068 u_int secsize; 1069 struct ccb_scsiio csio; 1070 struct disk *dp; 1071 1072 dp = arg; 1073 periph = dp->d_drv1; 1074 if (periph == NULL) 1075 return (ENXIO); 1076 softc = (struct da_softc *)periph->softc; 1077 cam_periph_lock(periph); 1078 secsize = softc->params.secsize; 1079 1080 if ((softc->flags & DA_FLAG_PACK_INVALID) != 0) { 1081 cam_periph_unlock(periph); 1082 return (ENXIO); 1083 } 1084 1085 if (length > 0) { 1086 xpt_setup_ccb(&csio.ccb_h, periph->path, CAM_PRIORITY_NORMAL); 1087 csio.ccb_h.ccb_state = DA_CCB_DUMP; 1088 scsi_read_write(&csio, 1089 /*retries*/1, 1090 dadone, 1091 MSG_ORDERED_Q_TAG, 1092 /*read*/FALSE, 1093 /*byte2*/0, 1094 /*minimum_cmd_size*/ softc->minimum_cmd_size, 1095 offset / secsize, 1096 length / secsize, 1097 /*data_ptr*/(u_int8_t *) virtual, 1098 /*dxfer_len*/length, 1099 /*sense_len*/SSD_FULL_SIZE, 1100 DA_DEFAULT_TIMEOUT * 1000); 1101 xpt_polled_action((union ccb *)&csio); 1102 1103 if ((csio.ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 1104 printf("Aborting dump due to I/O error.\n"); 1105 if ((csio.ccb_h.status & CAM_STATUS_MASK) == 1106 CAM_SCSI_STATUS_ERROR) 1107 scsi_sense_print(&csio); 1108 else 1109 printf("status == 0x%x, scsi status == 0x%x\n", 1110 csio.ccb_h.status, csio.scsi_status); 1111 return(EIO); 1112 } 1113 cam_periph_unlock(periph); 1114 return(0); 1115 } 1116 1117 /* 1118 * Sync the disk cache contents to the physical media. 1119 */ 1120 if ((softc->quirks & DA_Q_NO_SYNC_CACHE) == 0) { 1121 1122 xpt_setup_ccb(&csio.ccb_h, periph->path, CAM_PRIORITY_NORMAL); 1123 csio.ccb_h.ccb_state = DA_CCB_DUMP; 1124 scsi_synchronize_cache(&csio, 1125 /*retries*/1, 1126 /*cbfcnp*/dadone, 1127 MSG_SIMPLE_Q_TAG, 1128 /*begin_lba*/0,/* Cover the whole disk */ 1129 /*lb_count*/0, 1130 SSD_FULL_SIZE, 1131 5 * 60 * 1000); 1132 xpt_polled_action((union ccb *)&csio); 1133 1134 if ((csio.ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 1135 if ((csio.ccb_h.status & CAM_STATUS_MASK) == 1136 CAM_SCSI_STATUS_ERROR) { 1137 int asc, ascq; 1138 int sense_key, error_code; 1139 1140 scsi_extract_sense_len(&csio.sense_data, 1141 csio.sense_len - csio.sense_resid, 1142 &error_code, &sense_key, &asc, &ascq, 1143 /*show_errors*/ 1); 1144 if (sense_key != SSD_KEY_ILLEGAL_REQUEST) 1145 scsi_sense_print(&csio); 1146 } else { 1147 xpt_print(periph->path, "Synchronize cache " 1148 "failed, status == 0x%x, scsi status == " 1149 "0x%x\n", csio.ccb_h.status, 1150 csio.scsi_status); 1151 } 1152 } 1153 } 1154 cam_periph_unlock(periph); 1155 return (0); 1156 } 1157 1158 static int 1159 dagetattr(struct bio *bp) 1160 { 1161 int ret = -1; 1162 struct cam_periph *periph; 1163 1164 if (bp->bio_disk == NULL || bp->bio_disk->d_drv1 == NULL) 1165 return ENXIO; 1166 periph = (struct cam_periph *)bp->bio_disk->d_drv1; 1167 if (periph->path == NULL) 1168 return ENXIO; 1169 1170 ret = xpt_getattr(bp->bio_data, bp->bio_length, bp->bio_attribute, 1171 periph->path); 1172 if (ret == 0) 1173 bp->bio_completed = bp->bio_length; 1174 return ret; 1175 } 1176 1177 static void 1178 dainit(void) 1179 { 1180 cam_status status; 1181 1182 /* 1183 * Install a global async callback. This callback will 1184 * receive async callbacks like "new device found". 1185 */ 1186 status = xpt_register_async(AC_FOUND_DEVICE, daasync, NULL, NULL); 1187 1188 if (status != CAM_REQ_CMP) { 1189 printf("da: Failed to attach master async callback " 1190 "due to status 0x%x!\n", status); 1191 } else if (da_send_ordered) { 1192 1193 /* Register our shutdown event handler */ 1194 if ((EVENTHANDLER_REGISTER(shutdown_post_sync, dashutdown, 1195 NULL, SHUTDOWN_PRI_DEFAULT)) == NULL) 1196 printf("dainit: shutdown event registration failed!\n"); 1197 } 1198 } 1199 1200 static void 1201 daoninvalidate(struct cam_periph *periph) 1202 { 1203 struct da_softc *softc; 1204 1205 softc = (struct da_softc *)periph->softc; 1206 1207 /* 1208 * De-register any async callbacks. 1209 */ 1210 xpt_register_async(0, daasync, periph, periph->path); 1211 1212 softc->flags |= DA_FLAG_PACK_INVALID; 1213 1214 /* 1215 * Return all queued I/O with ENXIO. 1216 * XXX Handle any transactions queued to the card 1217 * with XPT_ABORT_CCB. 1218 */ 1219 bioq_flush(&softc->bio_queue, NULL, ENXIO); 1220 1221 disk_gone(softc->disk); 1222 xpt_print(periph->path, "lost device - %d outstanding\n", 1223 softc->outstanding_cmds); 1224 } 1225 1226 static void 1227 dacleanup(struct cam_periph *periph) 1228 { 1229 struct da_softc *softc; 1230 1231 softc = (struct da_softc *)periph->softc; 1232 1233 xpt_print(periph->path, "removing device entry\n"); 1234 cam_periph_unlock(periph); 1235 1236 /* 1237 * If we can't free the sysctl tree, oh well... 1238 */ 1239 if ((softc->flags & DA_FLAG_SCTX_INIT) != 0 1240 && sysctl_ctx_free(&softc->sysctl_ctx) != 0) { 1241 xpt_print(periph->path, "can't remove sysctl context\n"); 1242 } 1243 1244 disk_destroy(softc->disk); 1245 callout_drain(&softc->sendordered_c); 1246 free(softc, M_DEVBUF); 1247 cam_periph_lock(periph); 1248 } 1249 1250 static void 1251 daasync(void *callback_arg, u_int32_t code, 1252 struct cam_path *path, void *arg) 1253 { 1254 struct cam_periph *periph; 1255 1256 periph = (struct cam_periph *)callback_arg; 1257 switch (code) { 1258 case AC_FOUND_DEVICE: 1259 { 1260 struct ccb_getdev *cgd; 1261 cam_status status; 1262 1263 cgd = (struct ccb_getdev *)arg; 1264 if (cgd == NULL) 1265 break; 1266 1267 if (cgd->protocol != PROTO_SCSI) 1268 break; 1269 1270 if (SID_TYPE(&cgd->inq_data) != T_DIRECT 1271 && SID_TYPE(&cgd->inq_data) != T_RBC 1272 && SID_TYPE(&cgd->inq_data) != T_OPTICAL) 1273 break; 1274 1275 /* 1276 * Allocate a peripheral instance for 1277 * this device and start the probe 1278 * process. 1279 */ 1280 status = cam_periph_alloc(daregister, daoninvalidate, 1281 dacleanup, dastart, 1282 "da", CAM_PERIPH_BIO, 1283 cgd->ccb_h.path, daasync, 1284 AC_FOUND_DEVICE, cgd); 1285 1286 if (status != CAM_REQ_CMP 1287 && status != CAM_REQ_INPROG) 1288 printf("daasync: Unable to attach to new device " 1289 "due to status 0x%x\n", status); 1290 return; 1291 } 1292 case AC_ADVINFO_CHANGED: 1293 { 1294 uintptr_t buftype; 1295 1296 buftype = (uintptr_t)arg; 1297 if (buftype == CDAI_TYPE_PHYS_PATH) { 1298 struct da_softc *softc; 1299 1300 softc = periph->softc; 1301 disk_attr_changed(softc->disk, "GEOM::physpath", 1302 M_NOWAIT); 1303 } 1304 break; 1305 } 1306 case AC_SENT_BDR: 1307 case AC_BUS_RESET: 1308 { 1309 struct da_softc *softc; 1310 struct ccb_hdr *ccbh; 1311 1312 softc = (struct da_softc *)periph->softc; 1313 /* 1314 * Don't fail on the expected unit attention 1315 * that will occur. 1316 */ 1317 softc->flags |= DA_FLAG_RETRY_UA; 1318 LIST_FOREACH(ccbh, &softc->pending_ccbs, periph_links.le) 1319 ccbh->ccb_state |= DA_CCB_RETRY_UA; 1320 break; 1321 } 1322 default: 1323 break; 1324 } 1325 cam_periph_async(periph, code, path, arg); 1326 } 1327 1328 static void 1329 dasysctlinit(void *context, int pending) 1330 { 1331 struct cam_periph *periph; 1332 struct da_softc *softc; 1333 char tmpstr[80], tmpstr2[80]; 1334 struct ccb_trans_settings cts; 1335 1336 periph = (struct cam_periph *)context; 1337 /* 1338 * periph was held for us when this task was enqueued 1339 */ 1340 if (periph->flags & CAM_PERIPH_INVALID) { 1341 cam_periph_release(periph); 1342 return; 1343 } 1344 1345 softc = (struct da_softc *)periph->softc; 1346 snprintf(tmpstr, sizeof(tmpstr), "CAM DA unit %d", periph->unit_number); 1347 snprintf(tmpstr2, sizeof(tmpstr2), "%d", periph->unit_number); 1348 1349 sysctl_ctx_init(&softc->sysctl_ctx); 1350 softc->flags |= DA_FLAG_SCTX_INIT; 1351 softc->sysctl_tree = SYSCTL_ADD_NODE(&softc->sysctl_ctx, 1352 SYSCTL_STATIC_CHILDREN(_kern_cam_da), OID_AUTO, tmpstr2, 1353 CTLFLAG_RD, 0, tmpstr); 1354 if (softc->sysctl_tree == NULL) { 1355 printf("dasysctlinit: unable to allocate sysctl tree\n"); 1356 cam_periph_release(periph); 1357 return; 1358 } 1359 1360 /* 1361 * Now register the sysctl handler, so the user can change the value on 1362 * the fly. 1363 */ 1364 SYSCTL_ADD_PROC(&softc->sysctl_ctx, SYSCTL_CHILDREN(softc->sysctl_tree), 1365 OID_AUTO, "minimum_cmd_size", CTLTYPE_INT | CTLFLAG_RW, 1366 &softc->minimum_cmd_size, 0, dacmdsizesysctl, "I", 1367 "Minimum CDB size"); 1368 1369 /* 1370 * Add some addressing info. 1371 */ 1372 memset(&cts, 0, sizeof (cts)); 1373 xpt_setup_ccb(&cts.ccb_h, periph->path, /*priority*/1); 1374 cts.ccb_h.func_code = XPT_GET_TRAN_SETTINGS; 1375 cts.type = CTS_TYPE_CURRENT_SETTINGS; 1376 cam_periph_lock(periph); 1377 xpt_action((union ccb *)&cts); 1378 cam_periph_unlock(periph); 1379 if (cts.ccb_h.status != CAM_REQ_CMP) { 1380 cam_periph_release(periph); 1381 return; 1382 } 1383 if (cts.protocol == PROTO_SCSI && cts.transport == XPORT_FC) { 1384 struct ccb_trans_settings_fc *fc = &cts.xport_specific.fc; 1385 if (fc->valid & CTS_FC_VALID_WWPN) { 1386 softc->wwpn = fc->wwpn; 1387 SYSCTL_ADD_UQUAD(&softc->sysctl_ctx, 1388 SYSCTL_CHILDREN(softc->sysctl_tree), 1389 OID_AUTO, "wwpn", CTLFLAG_RD, 1390 &softc->wwpn, "World Wide Port Name"); 1391 } 1392 } 1393 cam_periph_release(periph); 1394 } 1395 1396 static int 1397 dacmdsizesysctl(SYSCTL_HANDLER_ARGS) 1398 { 1399 int error, value; 1400 1401 value = *(int *)arg1; 1402 1403 error = sysctl_handle_int(oidp, &value, 0, req); 1404 1405 if ((error != 0) 1406 || (req->newptr == NULL)) 1407 return (error); 1408 1409 /* 1410 * Acceptable values here are 6, 10, 12 or 16. 1411 */ 1412 if (value < 6) 1413 value = 6; 1414 else if ((value > 6) 1415 && (value <= 10)) 1416 value = 10; 1417 else if ((value > 10) 1418 && (value <= 12)) 1419 value = 12; 1420 else if (value > 12) 1421 value = 16; 1422 1423 *(int *)arg1 = value; 1424 1425 return (0); 1426 } 1427 1428 static cam_status 1429 daregister(struct cam_periph *periph, void *arg) 1430 { 1431 struct da_softc *softc; 1432 struct ccb_pathinq cpi; 1433 struct ccb_getdev *cgd; 1434 char tmpstr[80]; 1435 caddr_t match; 1436 1437 cgd = (struct ccb_getdev *)arg; 1438 if (periph == NULL) { 1439 printf("daregister: periph was NULL!!\n"); 1440 return(CAM_REQ_CMP_ERR); 1441 } 1442 1443 if (cgd == NULL) { 1444 printf("daregister: no getdev CCB, can't register device\n"); 1445 return(CAM_REQ_CMP_ERR); 1446 } 1447 1448 softc = (struct da_softc *)malloc(sizeof(*softc), M_DEVBUF, 1449 M_NOWAIT|M_ZERO); 1450 1451 if (softc == NULL) { 1452 printf("daregister: Unable to probe new device. " 1453 "Unable to allocate softc\n"); 1454 return(CAM_REQ_CMP_ERR); 1455 } 1456 1457 LIST_INIT(&softc->pending_ccbs); 1458 softc->state = DA_STATE_PROBE; 1459 bioq_init(&softc->bio_queue); 1460 if (SID_IS_REMOVABLE(&cgd->inq_data)) 1461 softc->flags |= DA_FLAG_PACK_REMOVABLE; 1462 if ((cgd->inq_data.flags & SID_CmdQue) != 0) 1463 softc->flags |= DA_FLAG_TAGGED_QUEUING; 1464 1465 periph->softc = softc; 1466 1467 /* 1468 * See if this device has any quirks. 1469 */ 1470 match = cam_quirkmatch((caddr_t)&cgd->inq_data, 1471 (caddr_t)da_quirk_table, 1472 sizeof(da_quirk_table)/sizeof(*da_quirk_table), 1473 sizeof(*da_quirk_table), scsi_inquiry_match); 1474 1475 if (match != NULL) 1476 softc->quirks = ((struct da_quirk_entry *)match)->quirks; 1477 else 1478 softc->quirks = DA_Q_NONE; 1479 1480 /* Check if the SIM does not want 6 byte commands */ 1481 bzero(&cpi, sizeof(cpi)); 1482 xpt_setup_ccb(&cpi.ccb_h, periph->path, CAM_PRIORITY_NORMAL); 1483 cpi.ccb_h.func_code = XPT_PATH_INQ; 1484 xpt_action((union ccb *)&cpi); 1485 if (cpi.ccb_h.status == CAM_REQ_CMP && (cpi.hba_misc & PIM_NO_6_BYTE)) 1486 softc->quirks |= DA_Q_NO_6_BYTE; 1487 1488 TASK_INIT(&softc->sysctl_task, 0, dasysctlinit, periph); 1489 1490 /* 1491 * Take an exclusive refcount on the periph while dastart is called 1492 * to finish the probe. The reference will be dropped in dadone at 1493 * the end of probe. 1494 */ 1495 (void)cam_periph_hold(periph, PRIBIO); 1496 1497 /* 1498 * Schedule a periodic event to occasionally send an 1499 * ordered tag to a device. 1500 */ 1501 callout_init_mtx(&softc->sendordered_c, periph->sim->mtx, 0); 1502 callout_reset(&softc->sendordered_c, 1503 (DA_DEFAULT_TIMEOUT * hz) / DA_ORDEREDTAG_INTERVAL, 1504 dasendorderedtag, softc); 1505 1506 mtx_unlock(periph->sim->mtx); 1507 /* 1508 * RBC devices don't have to support READ(6), only READ(10). 1509 */ 1510 if (softc->quirks & DA_Q_NO_6_BYTE || SID_TYPE(&cgd->inq_data) == T_RBC) 1511 softc->minimum_cmd_size = 10; 1512 else 1513 softc->minimum_cmd_size = 6; 1514 1515 /* 1516 * Load the user's default, if any. 1517 */ 1518 snprintf(tmpstr, sizeof(tmpstr), "kern.cam.da.%d.minimum_cmd_size", 1519 periph->unit_number); 1520 TUNABLE_INT_FETCH(tmpstr, &softc->minimum_cmd_size); 1521 1522 /* 1523 * 6, 10, 12 and 16 are the currently permissible values. 1524 */ 1525 if (softc->minimum_cmd_size < 6) 1526 softc->minimum_cmd_size = 6; 1527 else if ((softc->minimum_cmd_size > 6) 1528 && (softc->minimum_cmd_size <= 10)) 1529 softc->minimum_cmd_size = 10; 1530 else if ((softc->minimum_cmd_size > 10) 1531 && (softc->minimum_cmd_size <= 12)) 1532 softc->minimum_cmd_size = 12; 1533 else if (softc->minimum_cmd_size > 12) 1534 softc->minimum_cmd_size = 16; 1535 1536 /* 1537 * Register this media as a disk. 1538 */ 1539 softc->disk = disk_alloc(); 1540 softc->disk->d_devstat = devstat_new_entry(periph->periph_name, 1541 periph->unit_number, 0, 1542 DEVSTAT_BS_UNAVAILABLE, 1543 SID_TYPE(&cgd->inq_data) | 1544 XPORT_DEVSTAT_TYPE(cpi.transport), 1545 DEVSTAT_PRIORITY_DISK); 1546 softc->disk->d_open = daopen; 1547 softc->disk->d_close = daclose; 1548 softc->disk->d_strategy = dastrategy; 1549 softc->disk->d_dump = dadump; 1550 softc->disk->d_getattr = dagetattr; 1551 softc->disk->d_name = "da"; 1552 softc->disk->d_drv1 = periph; 1553 if (cpi.maxio == 0) 1554 softc->disk->d_maxsize = DFLTPHYS; /* traditional default */ 1555 else if (cpi.maxio > MAXPHYS) 1556 softc->disk->d_maxsize = MAXPHYS; /* for safety */ 1557 else 1558 softc->disk->d_maxsize = cpi.maxio; 1559 softc->disk->d_unit = periph->unit_number; 1560 softc->disk->d_flags = 0; 1561 if ((softc->quirks & DA_Q_NO_SYNC_CACHE) == 0) 1562 softc->disk->d_flags |= DISKFLAG_CANFLUSHCACHE; 1563 cam_strvis(softc->disk->d_descr, cgd->inq_data.vendor, 1564 sizeof(cgd->inq_data.vendor), sizeof(softc->disk->d_descr)); 1565 strlcat(softc->disk->d_descr, " ", sizeof(softc->disk->d_descr)); 1566 cam_strvis(&softc->disk->d_descr[strlen(softc->disk->d_descr)], 1567 cgd->inq_data.product, sizeof(cgd->inq_data.product), 1568 sizeof(softc->disk->d_descr) - strlen(softc->disk->d_descr)); 1569 softc->disk->d_hba_vendor = cpi.hba_vendor; 1570 softc->disk->d_hba_device = cpi.hba_device; 1571 softc->disk->d_hba_subvendor = cpi.hba_subvendor; 1572 softc->disk->d_hba_subdevice = cpi.hba_subdevice; 1573 disk_create(softc->disk, DISK_VERSION); 1574 mtx_lock(periph->sim->mtx); 1575 1576 /* 1577 * Add async callbacks for events of interest. 1578 * I don't bother checking if this fails as, 1579 * in most cases, the system will function just 1580 * fine without them and the only alternative 1581 * would be to not attach the device on failure. 1582 */ 1583 xpt_register_async(AC_SENT_BDR | AC_BUS_RESET 1584 | AC_LOST_DEVICE | AC_ADVINFO_CHANGED, 1585 daasync, periph, periph->path); 1586 1587 /* 1588 * Emit an attribute changed notification just in case 1589 * physical path information arrived before our async 1590 * event handler was registered, but after anyone attaching 1591 * to our disk device polled it. 1592 */ 1593 disk_attr_changed(softc->disk, "GEOM::physpath", M_NOWAIT); 1594 1595 xpt_schedule(periph, CAM_PRIORITY_DEV); 1596 1597 return(CAM_REQ_CMP); 1598 } 1599 1600 static void 1601 dastart(struct cam_periph *periph, union ccb *start_ccb) 1602 { 1603 struct da_softc *softc; 1604 1605 softc = (struct da_softc *)periph->softc; 1606 1607 switch (softc->state) { 1608 case DA_STATE_NORMAL: 1609 { 1610 /* Pull a buffer from the queue and get going on it */ 1611 struct bio *bp; 1612 1613 /* 1614 * See if there is a buf with work for us to do.. 1615 */ 1616 bp = bioq_first(&softc->bio_queue); 1617 if (periph->immediate_priority <= periph->pinfo.priority) { 1618 CAM_DEBUG_PRINT(CAM_DEBUG_SUBTRACE, 1619 ("queuing for immediate ccb\n")); 1620 start_ccb->ccb_h.ccb_state = DA_CCB_WAITING; 1621 SLIST_INSERT_HEAD(&periph->ccb_list, &start_ccb->ccb_h, 1622 periph_links.sle); 1623 periph->immediate_priority = CAM_PRIORITY_NONE; 1624 wakeup(&periph->ccb_list); 1625 } else if (bp == NULL) { 1626 xpt_release_ccb(start_ccb); 1627 } else { 1628 u_int8_t tag_code; 1629 1630 bioq_remove(&softc->bio_queue, bp); 1631 1632 if ((bp->bio_flags & BIO_ORDERED) != 0 1633 || (softc->flags & DA_FLAG_NEED_OTAG) != 0) { 1634 softc->flags &= ~DA_FLAG_NEED_OTAG; 1635 softc->ordered_tag_count++; 1636 tag_code = MSG_ORDERED_Q_TAG; 1637 } else { 1638 tag_code = MSG_SIMPLE_Q_TAG; 1639 } 1640 switch (bp->bio_cmd) { 1641 case BIO_READ: 1642 case BIO_WRITE: 1643 scsi_read_write(&start_ccb->csio, 1644 /*retries*/da_retry_count, 1645 /*cbfcnp*/dadone, 1646 /*tag_action*/tag_code, 1647 /*read_op*/bp->bio_cmd 1648 == BIO_READ, 1649 /*byte2*/0, 1650 softc->minimum_cmd_size, 1651 /*lba*/bp->bio_pblkno, 1652 /*block_count*/bp->bio_bcount / 1653 softc->params.secsize, 1654 /*data_ptr*/ bp->bio_data, 1655 /*dxfer_len*/ bp->bio_bcount, 1656 /*sense_len*/SSD_FULL_SIZE, 1657 da_default_timeout * 1000); 1658 break; 1659 case BIO_FLUSH: 1660 /* 1661 * BIO_FLUSH doesn't currently communicate 1662 * range data, so we synchronize the cache 1663 * over the whole disk. We also force 1664 * ordered tag semantics the flush applies 1665 * to all previously queued I/O. 1666 */ 1667 scsi_synchronize_cache(&start_ccb->csio, 1668 /*retries*/1, 1669 /*cbfcnp*/dadone, 1670 MSG_ORDERED_Q_TAG, 1671 /*begin_lba*/0, 1672 /*lb_count*/0, 1673 SSD_FULL_SIZE, 1674 da_default_timeout*1000); 1675 break; 1676 } 1677 start_ccb->ccb_h.ccb_state = DA_CCB_BUFFER_IO; 1678 1679 /* 1680 * Block out any asyncronous callbacks 1681 * while we touch the pending ccb list. 1682 */ 1683 LIST_INSERT_HEAD(&softc->pending_ccbs, 1684 &start_ccb->ccb_h, periph_links.le); 1685 softc->outstanding_cmds++; 1686 1687 /* We expect a unit attention from this device */ 1688 if ((softc->flags & DA_FLAG_RETRY_UA) != 0) { 1689 start_ccb->ccb_h.ccb_state |= DA_CCB_RETRY_UA; 1690 softc->flags &= ~DA_FLAG_RETRY_UA; 1691 } 1692 1693 start_ccb->ccb_h.ccb_bp = bp; 1694 bp = bioq_first(&softc->bio_queue); 1695 1696 xpt_action(start_ccb); 1697 } 1698 1699 if (bp != NULL) { 1700 /* Have more work to do, so ensure we stay scheduled */ 1701 xpt_schedule(periph, CAM_PRIORITY_NORMAL); 1702 } 1703 break; 1704 } 1705 case DA_STATE_PROBE: 1706 { 1707 struct ccb_scsiio *csio; 1708 struct scsi_read_capacity_data *rcap; 1709 1710 rcap = (struct scsi_read_capacity_data *) 1711 malloc(sizeof(*rcap), M_SCSIDA, M_NOWAIT|M_ZERO); 1712 if (rcap == NULL) { 1713 printf("dastart: Couldn't malloc read_capacity data\n"); 1714 /* da_free_periph??? */ 1715 break; 1716 } 1717 csio = &start_ccb->csio; 1718 scsi_read_capacity(csio, 1719 /*retries*/4, 1720 dadone, 1721 MSG_SIMPLE_Q_TAG, 1722 rcap, 1723 SSD_FULL_SIZE, 1724 /*timeout*/5000); 1725 start_ccb->ccb_h.ccb_bp = NULL; 1726 start_ccb->ccb_h.ccb_state = DA_CCB_PROBE; 1727 xpt_action(start_ccb); 1728 break; 1729 } 1730 case DA_STATE_PROBE2: 1731 { 1732 struct ccb_scsiio *csio; 1733 struct scsi_read_capacity_data_long *rcaplong; 1734 1735 rcaplong = (struct scsi_read_capacity_data_long *) 1736 malloc(sizeof(*rcaplong), M_SCSIDA, M_NOWAIT|M_ZERO); 1737 if (rcaplong == NULL) { 1738 printf("dastart: Couldn't malloc read_capacity data\n"); 1739 /* da_free_periph??? */ 1740 break; 1741 } 1742 csio = &start_ccb->csio; 1743 scsi_read_capacity_16(csio, 1744 /*retries*/ 4, 1745 /*cbfcnp*/ dadone, 1746 /*tag_action*/ MSG_SIMPLE_Q_TAG, 1747 /*lba*/ 0, 1748 /*reladr*/ 0, 1749 /*pmi*/ 0, 1750 rcaplong, 1751 /*sense_len*/ SSD_FULL_SIZE, 1752 /*timeout*/ 60000); 1753 start_ccb->ccb_h.ccb_bp = NULL; 1754 start_ccb->ccb_h.ccb_state = DA_CCB_PROBE2; 1755 xpt_action(start_ccb); 1756 break; 1757 } 1758 } 1759 } 1760 1761 static int 1762 cmd6workaround(union ccb *ccb) 1763 { 1764 struct scsi_rw_6 cmd6; 1765 struct scsi_rw_10 *cmd10; 1766 struct da_softc *softc; 1767 u_int8_t *cdb; 1768 int frozen; 1769 1770 cdb = ccb->csio.cdb_io.cdb_bytes; 1771 1772 /* Translation only possible if CDB is an array and cmd is R/W6 */ 1773 if ((ccb->ccb_h.flags & CAM_CDB_POINTER) != 0 || 1774 (*cdb != READ_6 && *cdb != WRITE_6)) 1775 return 0; 1776 1777 xpt_print(ccb->ccb_h.path, "READ(6)/WRITE(6) not supported, " 1778 "increasing minimum_cmd_size to 10.\n"); 1779 softc = (struct da_softc *)xpt_path_periph(ccb->ccb_h.path)->softc; 1780 softc->minimum_cmd_size = 10; 1781 1782 bcopy(cdb, &cmd6, sizeof(struct scsi_rw_6)); 1783 cmd10 = (struct scsi_rw_10 *)cdb; 1784 cmd10->opcode = (cmd6.opcode == READ_6) ? READ_10 : WRITE_10; 1785 cmd10->byte2 = 0; 1786 scsi_ulto4b(scsi_3btoul(cmd6.addr), cmd10->addr); 1787 cmd10->reserved = 0; 1788 scsi_ulto2b(cmd6.length, cmd10->length); 1789 cmd10->control = cmd6.control; 1790 ccb->csio.cdb_len = sizeof(*cmd10); 1791 1792 /* Requeue request, unfreezing queue if necessary */ 1793 frozen = (ccb->ccb_h.status & CAM_DEV_QFRZN) != 0; 1794 ccb->ccb_h.status = CAM_REQUEUE_REQ; 1795 xpt_action(ccb); 1796 if (frozen) { 1797 cam_release_devq(ccb->ccb_h.path, 1798 /*relsim_flags*/0, 1799 /*reduction*/0, 1800 /*timeout*/0, 1801 /*getcount_only*/0); 1802 } 1803 return (ERESTART); 1804 } 1805 1806 static void 1807 dadone(struct cam_periph *periph, union ccb *done_ccb) 1808 { 1809 struct da_softc *softc; 1810 struct ccb_scsiio *csio; 1811 u_int32_t priority; 1812 1813 softc = (struct da_softc *)periph->softc; 1814 priority = done_ccb->ccb_h.pinfo.priority; 1815 csio = &done_ccb->csio; 1816 switch (csio->ccb_h.ccb_state & DA_CCB_TYPE_MASK) { 1817 case DA_CCB_BUFFER_IO: 1818 { 1819 struct bio *bp; 1820 1821 bp = (struct bio *)done_ccb->ccb_h.ccb_bp; 1822 if ((done_ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 1823 int error; 1824 int sf; 1825 1826 if ((csio->ccb_h.ccb_state & DA_CCB_RETRY_UA) != 0) 1827 sf = SF_RETRY_UA; 1828 else 1829 sf = 0; 1830 1831 error = daerror(done_ccb, CAM_RETRY_SELTO, sf); 1832 if (error == ERESTART) { 1833 /* 1834 * A retry was scheuled, so 1835 * just return. 1836 */ 1837 return; 1838 } 1839 if (error != 0) { 1840 int queued_error; 1841 1842 /* 1843 * return all queued I/O with EIO, so that 1844 * the client can retry these I/Os in the 1845 * proper order should it attempt to recover. 1846 */ 1847 queued_error = EIO; 1848 1849 if (error == ENXIO 1850 && (softc->flags & DA_FLAG_PACK_INVALID)== 0) { 1851 /* 1852 * Catastrophic error. Mark our pack as 1853 * invalid. 1854 */ 1855 /* 1856 * XXX See if this is really a media 1857 * XXX change first? 1858 */ 1859 xpt_print(periph->path, 1860 "Invalidating pack\n"); 1861 softc->flags |= DA_FLAG_PACK_INVALID; 1862 queued_error = ENXIO; 1863 } 1864 bioq_flush(&softc->bio_queue, NULL, 1865 queued_error); 1866 bp->bio_error = error; 1867 bp->bio_resid = bp->bio_bcount; 1868 bp->bio_flags |= BIO_ERROR; 1869 } else { 1870 bp->bio_resid = csio->resid; 1871 bp->bio_error = 0; 1872 if (bp->bio_resid != 0) 1873 bp->bio_flags |= BIO_ERROR; 1874 } 1875 if ((done_ccb->ccb_h.status & CAM_DEV_QFRZN) != 0) 1876 cam_release_devq(done_ccb->ccb_h.path, 1877 /*relsim_flags*/0, 1878 /*reduction*/0, 1879 /*timeout*/0, 1880 /*getcount_only*/0); 1881 } else { 1882 if ((done_ccb->ccb_h.status & CAM_DEV_QFRZN) != 0) 1883 panic("REQ_CMP with QFRZN"); 1884 bp->bio_resid = csio->resid; 1885 if (csio->resid > 0) 1886 bp->bio_flags |= BIO_ERROR; 1887 } 1888 1889 /* 1890 * Block out any asyncronous callbacks 1891 * while we touch the pending ccb list. 1892 */ 1893 LIST_REMOVE(&done_ccb->ccb_h, periph_links.le); 1894 softc->outstanding_cmds--; 1895 if (softc->outstanding_cmds == 0) 1896 softc->flags |= DA_FLAG_WENT_IDLE; 1897 1898 if ((softc->flags & DA_FLAG_PACK_INVALID) != 0) { 1899 xpt_print(periph->path, "oustanding %d\n", 1900 softc->outstanding_cmds); 1901 } 1902 1903 biodone(bp); 1904 break; 1905 } 1906 case DA_CCB_PROBE: 1907 case DA_CCB_PROBE2: 1908 { 1909 struct scsi_read_capacity_data *rdcap; 1910 struct scsi_read_capacity_data_long *rcaplong; 1911 char announce_buf[80]; 1912 1913 rdcap = NULL; 1914 rcaplong = NULL; 1915 if (softc->state == DA_STATE_PROBE) 1916 rdcap =(struct scsi_read_capacity_data *)csio->data_ptr; 1917 else 1918 rcaplong = (struct scsi_read_capacity_data_long *) 1919 csio->data_ptr; 1920 1921 if ((csio->ccb_h.status & CAM_STATUS_MASK) == CAM_REQ_CMP) { 1922 struct disk_params *dp; 1923 uint32_t block_size; 1924 uint64_t maxsector; 1925 1926 if (softc->state == DA_STATE_PROBE) { 1927 block_size = scsi_4btoul(rdcap->length); 1928 maxsector = scsi_4btoul(rdcap->addr); 1929 1930 /* 1931 * According to SBC-2, if the standard 10 1932 * byte READ CAPACITY command returns 2^32, 1933 * we should issue the 16 byte version of 1934 * the command, since the device in question 1935 * has more sectors than can be represented 1936 * with the short version of the command. 1937 */ 1938 if (maxsector == 0xffffffff) { 1939 softc->state = DA_STATE_PROBE2; 1940 free(rdcap, M_SCSIDA); 1941 xpt_release_ccb(done_ccb); 1942 xpt_schedule(periph, priority); 1943 return; 1944 } 1945 } else { 1946 block_size = scsi_4btoul(rcaplong->length); 1947 maxsector = scsi_8btou64(rcaplong->addr); 1948 } 1949 1950 /* 1951 * Because GEOM code just will panic us if we 1952 * give them an 'illegal' value we'll avoid that 1953 * here. 1954 */ 1955 if (block_size == 0 && maxsector == 0) { 1956 snprintf(announce_buf, sizeof(announce_buf), 1957 "0MB (no media?)"); 1958 } else if (block_size >= MAXPHYS || block_size == 0) { 1959 xpt_print(periph->path, 1960 "unsupportable block size %ju\n", 1961 (uintmax_t) block_size); 1962 announce_buf[0] = '\0'; 1963 cam_periph_invalidate(periph); 1964 } else { 1965 dasetgeom(periph, block_size, maxsector, 0, 0); 1966 dp = &softc->params; 1967 snprintf(announce_buf, sizeof(announce_buf), 1968 "%juMB (%ju %u byte sectors: %dH %dS/T " 1969 "%dC)", (uintmax_t) 1970 (((uintmax_t)dp->secsize * 1971 dp->sectors) / (1024*1024)), 1972 (uintmax_t)dp->sectors, 1973 dp->secsize, dp->heads, 1974 dp->secs_per_track, dp->cylinders); 1975 } 1976 } else { 1977 int error; 1978 1979 announce_buf[0] = '\0'; 1980 1981 /* 1982 * Retry any UNIT ATTENTION type errors. They 1983 * are expected at boot. 1984 */ 1985 error = daerror(done_ccb, CAM_RETRY_SELTO, 1986 SF_RETRY_UA|SF_NO_PRINT); 1987 if (error == ERESTART) { 1988 /* 1989 * A retry was scheuled, so 1990 * just return. 1991 */ 1992 return; 1993 } else if (error != 0) { 1994 struct scsi_sense_data *sense; 1995 int asc, ascq; 1996 int sense_key, error_code; 1997 int have_sense; 1998 cam_status status; 1999 struct ccb_getdev cgd; 2000 2001 /* Don't wedge this device's queue */ 2002 status = done_ccb->ccb_h.status; 2003 if ((status & CAM_DEV_QFRZN) != 0) 2004 cam_release_devq(done_ccb->ccb_h.path, 2005 /*relsim_flags*/0, 2006 /*reduction*/0, 2007 /*timeout*/0, 2008 /*getcount_only*/0); 2009 2010 2011 xpt_setup_ccb(&cgd.ccb_h, 2012 done_ccb->ccb_h.path, 2013 CAM_PRIORITY_NORMAL); 2014 cgd.ccb_h.func_code = XPT_GDEV_TYPE; 2015 xpt_action((union ccb *)&cgd); 2016 2017 if (((csio->ccb_h.flags & CAM_SENSE_PHYS) != 0) 2018 || ((csio->ccb_h.flags & CAM_SENSE_PTR) != 0) 2019 || ((status & CAM_AUTOSNS_VALID) == 0)) 2020 have_sense = FALSE; 2021 else 2022 have_sense = TRUE; 2023 2024 if (have_sense) { 2025 sense = &csio->sense_data; 2026 scsi_extract_sense_len(sense, 2027 csio->sense_len - csio->sense_resid, 2028 &error_code, &sense_key, &asc, 2029 &ascq, /*show_errors*/ 1); 2030 } 2031 /* 2032 * Attach to anything that claims to be a 2033 * direct access or optical disk device, 2034 * as long as it doesn't return a "Logical 2035 * unit not supported" (0x25) error. 2036 */ 2037 if ((have_sense) && (asc != 0x25) 2038 && (error_code == SSD_CURRENT_ERROR)) { 2039 const char *sense_key_desc; 2040 const char *asc_desc; 2041 2042 scsi_sense_desc(sense_key, asc, ascq, 2043 &cgd.inq_data, 2044 &sense_key_desc, 2045 &asc_desc); 2046 snprintf(announce_buf, 2047 sizeof(announce_buf), 2048 "Attempt to query device " 2049 "size failed: %s, %s", 2050 sense_key_desc, 2051 asc_desc); 2052 } else { 2053 if (have_sense) 2054 scsi_sense_print( 2055 &done_ccb->csio); 2056 else { 2057 xpt_print(periph->path, 2058 "got CAM status %#x\n", 2059 done_ccb->ccb_h.status); 2060 } 2061 2062 xpt_print(periph->path, "fatal error, " 2063 "failed to attach to device\n"); 2064 2065 /* 2066 * Free up resources. 2067 */ 2068 cam_periph_invalidate(periph); 2069 } 2070 } 2071 } 2072 free(csio->data_ptr, M_SCSIDA); 2073 if (announce_buf[0] != '\0') { 2074 xpt_announce_periph(periph, announce_buf); 2075 /* 2076 * Create our sysctl variables, now that we know 2077 * we have successfully attached. 2078 */ 2079 (void) cam_periph_acquire(periph); /* increase the refcount */ 2080 taskqueue_enqueue(taskqueue_thread,&softc->sysctl_task); 2081 } 2082 softc->state = DA_STATE_NORMAL; 2083 /* 2084 * Since our peripheral may be invalidated by an error 2085 * above or an external event, we must release our CCB 2086 * before releasing the probe lock on the peripheral. 2087 * The peripheral will only go away once the last lock 2088 * is removed, and we need it around for the CCB release 2089 * operation. 2090 */ 2091 xpt_release_ccb(done_ccb); 2092 cam_periph_unhold(periph); 2093 return; 2094 } 2095 case DA_CCB_WAITING: 2096 { 2097 /* Caller will release the CCB */ 2098 wakeup(&done_ccb->ccb_h.cbfcnp); 2099 return; 2100 } 2101 case DA_CCB_DUMP: 2102 /* No-op. We're polling */ 2103 return; 2104 default: 2105 break; 2106 } 2107 xpt_release_ccb(done_ccb); 2108 } 2109 2110 static int 2111 daerror(union ccb *ccb, u_int32_t cam_flags, u_int32_t sense_flags) 2112 { 2113 struct da_softc *softc; 2114 struct cam_periph *periph; 2115 int error; 2116 2117 periph = xpt_path_periph(ccb->ccb_h.path); 2118 softc = (struct da_softc *)periph->softc; 2119 2120 /* 2121 * Automatically detect devices that do not support 2122 * READ(6)/WRITE(6) and upgrade to using 10 byte cdbs. 2123 */ 2124 error = 0; 2125 if ((ccb->ccb_h.status & CAM_STATUS_MASK) == CAM_REQ_INVALID) { 2126 error = cmd6workaround(ccb); 2127 } else if (((ccb->ccb_h.status & CAM_STATUS_MASK) == 2128 CAM_SCSI_STATUS_ERROR) 2129 && (ccb->ccb_h.status & CAM_AUTOSNS_VALID) 2130 && (ccb->csio.scsi_status == SCSI_STATUS_CHECK_COND) 2131 && ((ccb->ccb_h.flags & CAM_SENSE_PHYS) == 0) 2132 && ((ccb->ccb_h.flags & CAM_SENSE_PTR) == 0)) { 2133 int sense_key, error_code, asc, ascq; 2134 2135 scsi_extract_sense(&ccb->csio.sense_data, 2136 &error_code, &sense_key, &asc, &ascq); 2137 if (sense_key == SSD_KEY_ILLEGAL_REQUEST) 2138 error = cmd6workaround(ccb); 2139 } 2140 if (error == ERESTART) 2141 return (ERESTART); 2142 2143 /* 2144 * XXX 2145 * Until we have a better way of doing pack validation, 2146 * don't treat UAs as errors. 2147 */ 2148 sense_flags |= SF_RETRY_UA; 2149 return(cam_periph_error(ccb, cam_flags, sense_flags, 2150 &softc->saved_ccb)); 2151 } 2152 2153 static void 2154 daprevent(struct cam_periph *periph, int action) 2155 { 2156 struct da_softc *softc; 2157 union ccb *ccb; 2158 int error; 2159 2160 softc = (struct da_softc *)periph->softc; 2161 2162 if (((action == PR_ALLOW) 2163 && (softc->flags & DA_FLAG_PACK_LOCKED) == 0) 2164 || ((action == PR_PREVENT) 2165 && (softc->flags & DA_FLAG_PACK_LOCKED) != 0)) { 2166 return; 2167 } 2168 2169 ccb = cam_periph_getccb(periph, CAM_PRIORITY_NORMAL); 2170 2171 scsi_prevent(&ccb->csio, 2172 /*retries*/1, 2173 /*cbcfp*/dadone, 2174 MSG_SIMPLE_Q_TAG, 2175 action, 2176 SSD_FULL_SIZE, 2177 5000); 2178 2179 error = cam_periph_runccb(ccb, /*error_routine*/NULL, CAM_RETRY_SELTO, 2180 SF_RETRY_UA, softc->disk->d_devstat); 2181 2182 if (error == 0) { 2183 if (action == PR_ALLOW) 2184 softc->flags &= ~DA_FLAG_PACK_LOCKED; 2185 else 2186 softc->flags |= DA_FLAG_PACK_LOCKED; 2187 } 2188 2189 xpt_release_ccb(ccb); 2190 } 2191 2192 static int 2193 dagetcapacity(struct cam_periph *periph) 2194 { 2195 struct da_softc *softc; 2196 union ccb *ccb; 2197 struct scsi_read_capacity_data *rcap; 2198 struct scsi_read_capacity_data_long *rcaplong; 2199 uint32_t block_len; 2200 uint64_t maxsector; 2201 int error; 2202 u_int32_t sense_flags; 2203 2204 softc = (struct da_softc *)periph->softc; 2205 block_len = 0; 2206 maxsector = 0; 2207 error = 0; 2208 sense_flags = SF_RETRY_UA; 2209 if (softc->flags & DA_FLAG_PACK_REMOVABLE) 2210 sense_flags |= SF_NO_PRINT; 2211 2212 /* Do a read capacity */ 2213 rcap = (struct scsi_read_capacity_data *)malloc(sizeof(*rcaplong), 2214 M_SCSIDA, 2215 M_NOWAIT); 2216 if (rcap == NULL) 2217 return (ENOMEM); 2218 2219 ccb = cam_periph_getccb(periph, CAM_PRIORITY_NORMAL); 2220 scsi_read_capacity(&ccb->csio, 2221 /*retries*/4, 2222 /*cbfncp*/dadone, 2223 MSG_SIMPLE_Q_TAG, 2224 rcap, 2225 SSD_FULL_SIZE, 2226 /*timeout*/60000); 2227 ccb->ccb_h.ccb_bp = NULL; 2228 2229 error = cam_periph_runccb(ccb, daerror, 2230 /*cam_flags*/CAM_RETRY_SELTO, 2231 sense_flags, 2232 softc->disk->d_devstat); 2233 2234 if ((ccb->ccb_h.status & CAM_DEV_QFRZN) != 0) 2235 cam_release_devq(ccb->ccb_h.path, 2236 /*relsim_flags*/0, 2237 /*reduction*/0, 2238 /*timeout*/0, 2239 /*getcount_only*/0); 2240 2241 if (error == 0) { 2242 block_len = scsi_4btoul(rcap->length); 2243 maxsector = scsi_4btoul(rcap->addr); 2244 2245 if (maxsector != 0xffffffff) 2246 goto done; 2247 } else 2248 goto done; 2249 2250 rcaplong = (struct scsi_read_capacity_data_long *)rcap; 2251 2252 scsi_read_capacity_16(&ccb->csio, 2253 /*retries*/ 4, 2254 /*cbfcnp*/ dadone, 2255 /*tag_action*/ MSG_SIMPLE_Q_TAG, 2256 /*lba*/ 0, 2257 /*reladr*/ 0, 2258 /*pmi*/ 0, 2259 rcaplong, 2260 /*sense_len*/ SSD_FULL_SIZE, 2261 /*timeout*/ 60000); 2262 ccb->ccb_h.ccb_bp = NULL; 2263 2264 error = cam_periph_runccb(ccb, daerror, 2265 /*cam_flags*/CAM_RETRY_SELTO, 2266 sense_flags, 2267 softc->disk->d_devstat); 2268 2269 if ((ccb->ccb_h.status & CAM_DEV_QFRZN) != 0) 2270 cam_release_devq(ccb->ccb_h.path, 2271 /*relsim_flags*/0, 2272 /*reduction*/0, 2273 /*timeout*/0, 2274 /*getcount_only*/0); 2275 2276 if (error == 0) { 2277 block_len = scsi_4btoul(rcaplong->length); 2278 maxsector = scsi_8btou64(rcaplong->addr); 2279 } 2280 2281 done: 2282 2283 if (error == 0) { 2284 if (block_len >= MAXPHYS || block_len == 0) { 2285 xpt_print(periph->path, 2286 "unsupportable block size %ju\n", 2287 (uintmax_t) block_len); 2288 error = EINVAL; 2289 } else 2290 dasetgeom(periph, block_len, maxsector, 0, 0); 2291 } 2292 2293 xpt_release_ccb(ccb); 2294 2295 free(rcap, M_SCSIDA); 2296 2297 return (error); 2298 } 2299 2300 static void 2301 dasetgeom(struct cam_periph *periph, uint32_t block_len, uint64_t maxsector, 2302 u_int lbppbe, u_int lalba) 2303 { 2304 struct ccb_calc_geometry ccg; 2305 struct da_softc *softc; 2306 struct disk_params *dp; 2307 2308 softc = (struct da_softc *)periph->softc; 2309 2310 dp = &softc->params; 2311 dp->secsize = block_len; 2312 dp->sectors = maxsector + 1; 2313 if (lbppbe > 0) { 2314 dp->stripesize = block_len << lbppbe; 2315 dp->stripeoffset = dp->stripesize - block_len * lalba; 2316 } else if (softc->quirks & DA_Q_4K) { 2317 dp->stripesize = 4096; 2318 dp->stripeoffset = 0; 2319 } else { 2320 dp->stripesize = 0; 2321 dp->stripeoffset = 0; 2322 } 2323 /* 2324 * Have the controller provide us with a geometry 2325 * for this disk. The only time the geometry 2326 * matters is when we boot and the controller 2327 * is the only one knowledgeable enough to come 2328 * up with something that will make this a bootable 2329 * device. 2330 */ 2331 xpt_setup_ccb(&ccg.ccb_h, periph->path, CAM_PRIORITY_NORMAL); 2332 ccg.ccb_h.func_code = XPT_CALC_GEOMETRY; 2333 ccg.block_size = dp->secsize; 2334 ccg.volume_size = dp->sectors; 2335 ccg.heads = 0; 2336 ccg.secs_per_track = 0; 2337 ccg.cylinders = 0; 2338 xpt_action((union ccb*)&ccg); 2339 if ((ccg.ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 2340 /* 2341 * We don't know what went wrong here- but just pick 2342 * a geometry so we don't have nasty things like divide 2343 * by zero. 2344 */ 2345 dp->heads = 255; 2346 dp->secs_per_track = 255; 2347 dp->cylinders = dp->sectors / (255 * 255); 2348 if (dp->cylinders == 0) { 2349 dp->cylinders = 1; 2350 } 2351 } else { 2352 dp->heads = ccg.heads; 2353 dp->secs_per_track = ccg.secs_per_track; 2354 dp->cylinders = ccg.cylinders; 2355 } 2356 } 2357 2358 static void 2359 dasendorderedtag(void *arg) 2360 { 2361 struct da_softc *softc = arg; 2362 2363 if (da_send_ordered) { 2364 if ((softc->ordered_tag_count == 0) 2365 && ((softc->flags & DA_FLAG_WENT_IDLE) == 0)) { 2366 softc->flags |= DA_FLAG_NEED_OTAG; 2367 } 2368 if (softc->outstanding_cmds > 0) 2369 softc->flags &= ~DA_FLAG_WENT_IDLE; 2370 2371 softc->ordered_tag_count = 0; 2372 } 2373 /* Queue us up again */ 2374 callout_reset(&softc->sendordered_c, 2375 (DA_DEFAULT_TIMEOUT * hz) / DA_ORDEREDTAG_INTERVAL, 2376 dasendorderedtag, softc); 2377 } 2378 2379 /* 2380 * Step through all DA peripheral drivers, and if the device is still open, 2381 * sync the disk cache to physical media. 2382 */ 2383 static void 2384 dashutdown(void * arg, int howto) 2385 { 2386 struct cam_periph *periph; 2387 struct da_softc *softc; 2388 2389 TAILQ_FOREACH(periph, &dadriver.units, unit_links) { 2390 union ccb ccb; 2391 2392 cam_periph_lock(periph); 2393 softc = (struct da_softc *)periph->softc; 2394 2395 /* 2396 * We only sync the cache if the drive is still open, and 2397 * if the drive is capable of it.. 2398 */ 2399 if (((softc->flags & DA_FLAG_OPEN) == 0) 2400 || (softc->quirks & DA_Q_NO_SYNC_CACHE)) { 2401 cam_periph_unlock(periph); 2402 continue; 2403 } 2404 2405 xpt_setup_ccb(&ccb.ccb_h, periph->path, CAM_PRIORITY_NORMAL); 2406 2407 ccb.ccb_h.ccb_state = DA_CCB_DUMP; 2408 scsi_synchronize_cache(&ccb.csio, 2409 /*retries*/1, 2410 /*cbfcnp*/dadone, 2411 MSG_SIMPLE_Q_TAG, 2412 /*begin_lba*/0, /* whole disk */ 2413 /*lb_count*/0, 2414 SSD_FULL_SIZE, 2415 60 * 60 * 1000); 2416 2417 xpt_polled_action(&ccb); 2418 2419 if ((ccb.ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 2420 if (((ccb.ccb_h.status & CAM_STATUS_MASK) == 2421 CAM_SCSI_STATUS_ERROR) 2422 && (ccb.csio.scsi_status == SCSI_STATUS_CHECK_COND)){ 2423 int error_code, sense_key, asc, ascq; 2424 2425 scsi_extract_sense(&ccb.csio.sense_data, 2426 &error_code, &sense_key, 2427 &asc, &ascq); 2428 2429 if (sense_key != SSD_KEY_ILLEGAL_REQUEST) 2430 scsi_sense_print(&ccb.csio); 2431 } else { 2432 xpt_print(periph->path, "Synchronize " 2433 "cache failed, status == 0x%x, scsi status " 2434 "== 0x%x\n", ccb.ccb_h.status, 2435 ccb.csio.scsi_status); 2436 } 2437 } 2438 2439 if ((ccb.ccb_h.status & CAM_DEV_QFRZN) != 0) 2440 cam_release_devq(ccb.ccb_h.path, 2441 /*relsim_flags*/0, 2442 /*reduction*/0, 2443 /*timeout*/0, 2444 /*getcount_only*/0); 2445 cam_periph_unlock(periph); 2446 } 2447 } 2448 2449 #else /* !_KERNEL */ 2450 2451 /* 2452 * XXX This is only left out of the kernel build to silence warnings. If, 2453 * for some reason this function is used in the kernel, the ifdefs should 2454 * be moved so it is included both in the kernel and userland. 2455 */ 2456 void 2457 scsi_format_unit(struct ccb_scsiio *csio, u_int32_t retries, 2458 void (*cbfcnp)(struct cam_periph *, union ccb *), 2459 u_int8_t tag_action, u_int8_t byte2, u_int16_t ileave, 2460 u_int8_t *data_ptr, u_int32_t dxfer_len, u_int8_t sense_len, 2461 u_int32_t timeout) 2462 { 2463 struct scsi_format_unit *scsi_cmd; 2464 2465 scsi_cmd = (struct scsi_format_unit *)&csio->cdb_io.cdb_bytes; 2466 scsi_cmd->opcode = FORMAT_UNIT; 2467 scsi_cmd->byte2 = byte2; 2468 scsi_ulto2b(ileave, scsi_cmd->interleave); 2469 2470 cam_fill_csio(csio, 2471 retries, 2472 cbfcnp, 2473 /*flags*/ (dxfer_len > 0) ? CAM_DIR_OUT : CAM_DIR_NONE, 2474 tag_action, 2475 data_ptr, 2476 dxfer_len, 2477 sense_len, 2478 sizeof(*scsi_cmd), 2479 timeout); 2480 } 2481 2482 #endif /* _KERNEL */ 2483