1 /* 2 * Copyright (c) 1997-2007 Kenneth D. Merry 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * 2. Redistributions in binary form must reproduce the above copyright 11 * notice, this list of conditions and the following disclaimer in the 12 * documentation and/or other materials provided with the distribution. 13 * 3. 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 20 * FOR 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/ioctl.h> 33 #include <sys/stdint.h> 34 #include <sys/types.h> 35 #include <sys/stat.h> 36 #include <sys/endian.h> 37 #include <sys/sbuf.h> 38 39 #include <stdio.h> 40 #include <stdlib.h> 41 #include <string.h> 42 #include <unistd.h> 43 #include <inttypes.h> 44 #include <limits.h> 45 #include <fcntl.h> 46 #include <ctype.h> 47 #include <err.h> 48 #include <libutil.h> 49 #ifndef MINIMALISTIC 50 #include <limits.h> 51 #include <inttypes.h> 52 #endif 53 54 #include <cam/cam.h> 55 #include <cam/cam_debug.h> 56 #include <cam/cam_ccb.h> 57 #include <cam/scsi/scsi_all.h> 58 #include <cam/scsi/scsi_da.h> 59 #include <cam/scsi/scsi_pass.h> 60 #include <cam/scsi/scsi_message.h> 61 #include <cam/scsi/smp_all.h> 62 #include <cam/ata/ata_all.h> 63 #include <camlib.h> 64 #include "camcontrol.h" 65 66 typedef enum { 67 CAM_CMD_NONE = 0x00000000, 68 CAM_CMD_DEVLIST = 0x00000001, 69 CAM_CMD_TUR = 0x00000002, 70 CAM_CMD_INQUIRY = 0x00000003, 71 CAM_CMD_STARTSTOP = 0x00000004, 72 CAM_CMD_RESCAN = 0x00000005, 73 CAM_CMD_READ_DEFECTS = 0x00000006, 74 CAM_CMD_MODE_PAGE = 0x00000007, 75 CAM_CMD_SCSI_CMD = 0x00000008, 76 CAM_CMD_DEVTREE = 0x00000009, 77 CAM_CMD_USAGE = 0x0000000a, 78 CAM_CMD_DEBUG = 0x0000000b, 79 CAM_CMD_RESET = 0x0000000c, 80 CAM_CMD_FORMAT = 0x0000000d, 81 CAM_CMD_TAG = 0x0000000e, 82 CAM_CMD_RATE = 0x0000000f, 83 CAM_CMD_DETACH = 0x00000010, 84 CAM_CMD_REPORTLUNS = 0x00000011, 85 CAM_CMD_READCAP = 0x00000012, 86 CAM_CMD_IDENTIFY = 0x00000013, 87 CAM_CMD_IDLE = 0x00000014, 88 CAM_CMD_STANDBY = 0x00000015, 89 CAM_CMD_SLEEP = 0x00000016, 90 CAM_CMD_SMP_CMD = 0x00000017, 91 CAM_CMD_SMP_RG = 0x00000018, 92 CAM_CMD_SMP_PC = 0x00000019, 93 CAM_CMD_SMP_PHYLIST = 0x0000001a, 94 CAM_CMD_SMP_MANINFO = 0x0000001b, 95 CAM_CMD_DOWNLOAD_FW = 0x0000001c, 96 CAM_CMD_SECURITY = 0x0000001d, 97 CAM_CMD_HPA = 0x0000001e, 98 CAM_CMD_SANITIZE = 0x0000001f, 99 CAM_CMD_PERSIST = 0x00000020, 100 CAM_CMD_APM = 0x00000021, 101 CAM_CMD_AAM = 0x00000022, 102 CAM_CMD_ATTRIB = 0x00000023, 103 CAM_CMD_OPCODES = 0x00000024, 104 CAM_CMD_REPROBE = 0x00000025, 105 CAM_CMD_ZONE = 0x00000026, 106 CAM_CMD_EPC = 0x00000027 107 } cam_cmdmask; 108 109 typedef enum { 110 CAM_ARG_NONE = 0x00000000, 111 CAM_ARG_VERBOSE = 0x00000001, 112 CAM_ARG_DEVICE = 0x00000002, 113 CAM_ARG_BUS = 0x00000004, 114 CAM_ARG_TARGET = 0x00000008, 115 CAM_ARG_LUN = 0x00000010, 116 CAM_ARG_EJECT = 0x00000020, 117 CAM_ARG_UNIT = 0x00000040, 118 CAM_ARG_FORMAT_BLOCK = 0x00000080, 119 CAM_ARG_FORMAT_BFI = 0x00000100, 120 CAM_ARG_FORMAT_PHYS = 0x00000200, 121 CAM_ARG_PLIST = 0x00000400, 122 CAM_ARG_GLIST = 0x00000800, 123 CAM_ARG_GET_SERIAL = 0x00001000, 124 CAM_ARG_GET_STDINQ = 0x00002000, 125 CAM_ARG_GET_XFERRATE = 0x00004000, 126 CAM_ARG_INQ_MASK = 0x00007000, 127 CAM_ARG_MODE_EDIT = 0x00008000, 128 CAM_ARG_PAGE_CNTL = 0x00010000, 129 CAM_ARG_TIMEOUT = 0x00020000, 130 CAM_ARG_CMD_IN = 0x00040000, 131 CAM_ARG_CMD_OUT = 0x00080000, 132 CAM_ARG_DBD = 0x00100000, 133 CAM_ARG_ERR_RECOVER = 0x00200000, 134 CAM_ARG_RETRIES = 0x00400000, 135 CAM_ARG_START_UNIT = 0x00800000, 136 CAM_ARG_DEBUG_INFO = 0x01000000, 137 CAM_ARG_DEBUG_TRACE = 0x02000000, 138 CAM_ARG_DEBUG_SUBTRACE = 0x04000000, 139 CAM_ARG_DEBUG_CDB = 0x08000000, 140 CAM_ARG_DEBUG_XPT = 0x10000000, 141 CAM_ARG_DEBUG_PERIPH = 0x20000000, 142 CAM_ARG_DEBUG_PROBE = 0x40000000, 143 } cam_argmask; 144 145 struct camcontrol_opts { 146 const char *optname; 147 uint32_t cmdnum; 148 cam_argmask argnum; 149 const char *subopt; 150 }; 151 152 #ifndef MINIMALISTIC 153 struct ata_res_pass16 { 154 u_int16_t reserved[5]; 155 u_int8_t flags; 156 u_int8_t error; 157 u_int8_t sector_count_exp; 158 u_int8_t sector_count; 159 u_int8_t lba_low_exp; 160 u_int8_t lba_low; 161 u_int8_t lba_mid_exp; 162 u_int8_t lba_mid; 163 u_int8_t lba_high_exp; 164 u_int8_t lba_high; 165 u_int8_t device; 166 u_int8_t status; 167 }; 168 169 struct ata_set_max_pwd 170 { 171 u_int16_t reserved1; 172 u_int8_t password[32]; 173 u_int16_t reserved2[239]; 174 }; 175 176 static const char scsicmd_opts[] = "a:c:dfi:o:r"; 177 static const char readdefect_opts[] = "f:GPqsS:X"; 178 static const char negotiate_opts[] = "acD:M:O:qR:T:UW:"; 179 static const char smprg_opts[] = "l"; 180 static const char smppc_opts[] = "a:A:d:lm:M:o:p:s:S:T:"; 181 static const char smpphylist_opts[] = "lq"; 182 static char pwd_opt; 183 #endif 184 185 static struct camcontrol_opts option_table[] = { 186 #ifndef MINIMALISTIC 187 {"tur", CAM_CMD_TUR, CAM_ARG_NONE, NULL}, 188 {"inquiry", CAM_CMD_INQUIRY, CAM_ARG_NONE, "DSR"}, 189 {"identify", CAM_CMD_IDENTIFY, CAM_ARG_NONE, NULL}, 190 {"start", CAM_CMD_STARTSTOP, CAM_ARG_START_UNIT, NULL}, 191 {"stop", CAM_CMD_STARTSTOP, CAM_ARG_NONE, NULL}, 192 {"load", CAM_CMD_STARTSTOP, CAM_ARG_START_UNIT | CAM_ARG_EJECT, NULL}, 193 {"eject", CAM_CMD_STARTSTOP, CAM_ARG_EJECT, NULL}, 194 {"reportluns", CAM_CMD_REPORTLUNS, CAM_ARG_NONE, "clr:"}, 195 {"readcapacity", CAM_CMD_READCAP, CAM_ARG_NONE, "bhHNqs"}, 196 {"reprobe", CAM_CMD_REPROBE, CAM_ARG_NONE, NULL}, 197 #endif /* MINIMALISTIC */ 198 {"rescan", CAM_CMD_RESCAN, CAM_ARG_NONE, NULL}, 199 {"reset", CAM_CMD_RESET, CAM_ARG_NONE, NULL}, 200 #ifndef MINIMALISTIC 201 {"cmd", CAM_CMD_SCSI_CMD, CAM_ARG_NONE, scsicmd_opts}, 202 {"command", CAM_CMD_SCSI_CMD, CAM_ARG_NONE, scsicmd_opts}, 203 {"smpcmd", CAM_CMD_SMP_CMD, CAM_ARG_NONE, "r:R:"}, 204 {"smprg", CAM_CMD_SMP_RG, CAM_ARG_NONE, smprg_opts}, 205 {"smpreportgeneral", CAM_CMD_SMP_RG, CAM_ARG_NONE, smprg_opts}, 206 {"smppc", CAM_CMD_SMP_PC, CAM_ARG_NONE, smppc_opts}, 207 {"smpphycontrol", CAM_CMD_SMP_PC, CAM_ARG_NONE, smppc_opts}, 208 {"smpplist", CAM_CMD_SMP_PHYLIST, CAM_ARG_NONE, smpphylist_opts}, 209 {"smpphylist", CAM_CMD_SMP_PHYLIST, CAM_ARG_NONE, smpphylist_opts}, 210 {"smpmaninfo", CAM_CMD_SMP_MANINFO, CAM_ARG_NONE, "l"}, 211 {"defects", CAM_CMD_READ_DEFECTS, CAM_ARG_NONE, readdefect_opts}, 212 {"defectlist", CAM_CMD_READ_DEFECTS, CAM_ARG_NONE, readdefect_opts}, 213 #endif /* MINIMALISTIC */ 214 {"devlist", CAM_CMD_DEVTREE, CAM_ARG_NONE, "-b"}, 215 #ifndef MINIMALISTIC 216 {"periphlist", CAM_CMD_DEVLIST, CAM_ARG_NONE, NULL}, 217 {"modepage", CAM_CMD_MODE_PAGE, CAM_ARG_NONE, "bdelm:P:"}, 218 {"tags", CAM_CMD_TAG, CAM_ARG_NONE, "N:q"}, 219 {"negotiate", CAM_CMD_RATE, CAM_ARG_NONE, negotiate_opts}, 220 {"rate", CAM_CMD_RATE, CAM_ARG_NONE, negotiate_opts}, 221 {"debug", CAM_CMD_DEBUG, CAM_ARG_NONE, "IPTSXcp"}, 222 {"format", CAM_CMD_FORMAT, CAM_ARG_NONE, "qrwy"}, 223 {"sanitize", CAM_CMD_SANITIZE, CAM_ARG_NONE, "a:c:IP:qrUwy"}, 224 {"idle", CAM_CMD_IDLE, CAM_ARG_NONE, "t:"}, 225 {"standby", CAM_CMD_STANDBY, CAM_ARG_NONE, "t:"}, 226 {"sleep", CAM_CMD_SLEEP, CAM_ARG_NONE, ""}, 227 {"apm", CAM_CMD_APM, CAM_ARG_NONE, "l:"}, 228 {"aam", CAM_CMD_AAM, CAM_ARG_NONE, "l:"}, 229 {"fwdownload", CAM_CMD_DOWNLOAD_FW, CAM_ARG_NONE, "f:qsy"}, 230 {"security", CAM_CMD_SECURITY, CAM_ARG_NONE, "d:e:fh:k:l:qs:T:U:y"}, 231 {"hpa", CAM_CMD_HPA, CAM_ARG_NONE, "Pflp:qs:U:y"}, 232 {"persist", CAM_CMD_PERSIST, CAM_ARG_NONE, "ai:I:k:K:o:ps:ST:U"}, 233 {"attrib", CAM_CMD_ATTRIB, CAM_ARG_NONE, "a:ce:F:p:r:s:T:w:V:"}, 234 {"opcodes", CAM_CMD_OPCODES, CAM_ARG_NONE, "No:s:T"}, 235 {"zone", CAM_CMD_ZONE, CAM_ARG_NONE, "ac:l:No:P:"}, 236 {"epc", CAM_CMD_EPC, CAM_ARG_NONE, "c:dDeHp:Pr:sS:T:"}, 237 #endif /* MINIMALISTIC */ 238 {"help", CAM_CMD_USAGE, CAM_ARG_NONE, NULL}, 239 {"-?", CAM_CMD_USAGE, CAM_ARG_NONE, NULL}, 240 {"-h", CAM_CMD_USAGE, CAM_ARG_NONE, NULL}, 241 {NULL, 0, 0, NULL} 242 }; 243 244 struct cam_devitem { 245 struct device_match_result dev_match; 246 int num_periphs; 247 struct periph_match_result *periph_matches; 248 struct scsi_vpd_device_id *device_id; 249 int device_id_len; 250 STAILQ_ENTRY(cam_devitem) links; 251 }; 252 253 struct cam_devlist { 254 STAILQ_HEAD(, cam_devitem) dev_queue; 255 path_id_t path_id; 256 }; 257 258 static cam_cmdmask cmdlist; 259 static cam_argmask arglist; 260 261 camcontrol_optret getoption(struct camcontrol_opts *table, char *arg, 262 uint32_t *cmdnum, cam_argmask *argnum, 263 const char **subopt); 264 #ifndef MINIMALISTIC 265 static int getdevlist(struct cam_device *device); 266 #endif /* MINIMALISTIC */ 267 static int getdevtree(int argc, char **argv, char *combinedopt); 268 #ifndef MINIMALISTIC 269 static int testunitready(struct cam_device *device, int retry_count, 270 int timeout, int quiet); 271 static int scsistart(struct cam_device *device, int startstop, int loadeject, 272 int retry_count, int timeout); 273 static int scsiinquiry(struct cam_device *device, int retry_count, int timeout); 274 static int scsiserial(struct cam_device *device, int retry_count, int timeout); 275 #endif /* MINIMALISTIC */ 276 static int parse_btl(char *tstr, path_id_t *bus, target_id_t *target, 277 lun_id_t *lun, cam_argmask *arglst); 278 static int dorescan_or_reset(int argc, char **argv, int rescan); 279 static int rescan_or_reset_bus(path_id_t bus, int rescan); 280 static int scanlun_or_reset_dev(path_id_t bus, target_id_t target, 281 lun_id_t lun, int scan); 282 #ifndef MINIMALISTIC 283 static int readdefects(struct cam_device *device, int argc, char **argv, 284 char *combinedopt, int retry_count, int timeout); 285 static void modepage(struct cam_device *device, int argc, char **argv, 286 char *combinedopt, int retry_count, int timeout); 287 static int scsicmd(struct cam_device *device, int argc, char **argv, 288 char *combinedopt, int retry_count, int timeout); 289 static int smpcmd(struct cam_device *device, int argc, char **argv, 290 char *combinedopt, int retry_count, int timeout); 291 static int smpreportgeneral(struct cam_device *device, int argc, char **argv, 292 char *combinedopt, int retry_count, int timeout); 293 static int smpphycontrol(struct cam_device *device, int argc, char **argv, 294 char *combinedopt, int retry_count, int timeout); 295 static int smpmaninfo(struct cam_device *device, int argc, char **argv, 296 char *combinedopt, int retry_count, int timeout); 297 static int getdevid(struct cam_devitem *item); 298 static int buildbusdevlist(struct cam_devlist *devlist); 299 static void freebusdevlist(struct cam_devlist *devlist); 300 static struct cam_devitem *findsasdevice(struct cam_devlist *devlist, 301 uint64_t sasaddr); 302 static int smpphylist(struct cam_device *device, int argc, char **argv, 303 char *combinedopt, int retry_count, int timeout); 304 static int tagcontrol(struct cam_device *device, int argc, char **argv, 305 char *combinedopt); 306 static void cts_print(struct cam_device *device, 307 struct ccb_trans_settings *cts); 308 static void cpi_print(struct ccb_pathinq *cpi); 309 static int get_cpi(struct cam_device *device, struct ccb_pathinq *cpi); 310 static int get_cgd(struct cam_device *device, struct ccb_getdev *cgd); 311 static int get_print_cts(struct cam_device *device, int user_settings, 312 int quiet, struct ccb_trans_settings *cts); 313 static int ratecontrol(struct cam_device *device, int retry_count, 314 int timeout, int argc, char **argv, char *combinedopt); 315 static int scsiformat(struct cam_device *device, int argc, char **argv, 316 char *combinedopt, int retry_count, int timeout); 317 static int scsisanitize(struct cam_device *device, int argc, char **argv, 318 char *combinedopt, int retry_count, int timeout); 319 static int scsireportluns(struct cam_device *device, int argc, char **argv, 320 char *combinedopt, int retry_count, int timeout); 321 static int scsireadcapacity(struct cam_device *device, int argc, char **argv, 322 char *combinedopt, int retry_count, int timeout); 323 static int atapm(struct cam_device *device, int argc, char **argv, 324 char *combinedopt, int retry_count, int timeout); 325 static int atasecurity(struct cam_device *device, int retry_count, int timeout, 326 int argc, char **argv, char *combinedopt); 327 static int atahpa(struct cam_device *device, int retry_count, int timeout, 328 int argc, char **argv, char *combinedopt); 329 static int scsiprintoneopcode(struct cam_device *device, int req_opcode, 330 int sa_set, int req_sa, uint8_t *buf, 331 uint32_t valid_len); 332 static int scsiprintopcodes(struct cam_device *device, int td_req, uint8_t *buf, 333 uint32_t valid_len); 334 static int scsiopcodes(struct cam_device *device, int argc, char **argv, 335 char *combinedopt, int retry_count, int timeout, 336 int verbose); 337 static int scsireprobe(struct cam_device *device); 338 339 #endif /* MINIMALISTIC */ 340 #ifndef min 341 #define min(a,b) (((a)<(b))?(a):(b)) 342 #endif 343 #ifndef max 344 #define max(a,b) (((a)>(b))?(a):(b)) 345 #endif 346 347 camcontrol_optret 348 getoption(struct camcontrol_opts *table, char *arg, uint32_t *cmdnum, 349 cam_argmask *argnum, const char **subopt) 350 { 351 struct camcontrol_opts *opts; 352 int num_matches = 0; 353 354 for (opts = table; (opts != NULL) && (opts->optname != NULL); 355 opts++) { 356 if (strncmp(opts->optname, arg, strlen(arg)) == 0) { 357 *cmdnum = opts->cmdnum; 358 *argnum = opts->argnum; 359 *subopt = opts->subopt; 360 if (++num_matches > 1) 361 return(CC_OR_AMBIGUOUS); 362 } 363 } 364 365 if (num_matches > 0) 366 return(CC_OR_FOUND); 367 else 368 return(CC_OR_NOT_FOUND); 369 } 370 371 #ifndef MINIMALISTIC 372 static int 373 getdevlist(struct cam_device *device) 374 { 375 union ccb *ccb; 376 char status[32]; 377 int error = 0; 378 379 ccb = cam_getccb(device); 380 381 ccb->ccb_h.func_code = XPT_GDEVLIST; 382 ccb->ccb_h.flags = CAM_DIR_NONE; 383 ccb->ccb_h.retry_count = 1; 384 ccb->cgdl.index = 0; 385 ccb->cgdl.status = CAM_GDEVLIST_MORE_DEVS; 386 while (ccb->cgdl.status == CAM_GDEVLIST_MORE_DEVS) { 387 if (cam_send_ccb(device, ccb) < 0) { 388 perror("error getting device list"); 389 cam_freeccb(ccb); 390 return(1); 391 } 392 393 status[0] = '\0'; 394 395 switch (ccb->cgdl.status) { 396 case CAM_GDEVLIST_MORE_DEVS: 397 strcpy(status, "MORE"); 398 break; 399 case CAM_GDEVLIST_LAST_DEVICE: 400 strcpy(status, "LAST"); 401 break; 402 case CAM_GDEVLIST_LIST_CHANGED: 403 strcpy(status, "CHANGED"); 404 break; 405 case CAM_GDEVLIST_ERROR: 406 strcpy(status, "ERROR"); 407 error = 1; 408 break; 409 } 410 411 fprintf(stdout, "%s%d: generation: %d index: %d status: %s\n", 412 ccb->cgdl.periph_name, 413 ccb->cgdl.unit_number, 414 ccb->cgdl.generation, 415 ccb->cgdl.index, 416 status); 417 418 /* 419 * If the list has changed, we need to start over from the 420 * beginning. 421 */ 422 if (ccb->cgdl.status == CAM_GDEVLIST_LIST_CHANGED) 423 ccb->cgdl.index = 0; 424 } 425 426 cam_freeccb(ccb); 427 428 return(error); 429 } 430 #endif /* MINIMALISTIC */ 431 432 static int 433 getdevtree(int argc, char **argv, char *combinedopt) 434 { 435 union ccb ccb; 436 int bufsize, fd; 437 unsigned int i; 438 int need_close = 0; 439 int error = 0; 440 int skip_device = 0; 441 int busonly = 0; 442 int c; 443 444 while ((c = getopt(argc, argv, combinedopt)) != -1) { 445 switch(c) { 446 case 'b': 447 if ((arglist & CAM_ARG_VERBOSE) == 0) 448 busonly = 1; 449 break; 450 default: 451 break; 452 } 453 } 454 455 if ((fd = open(XPT_DEVICE, O_RDWR)) == -1) { 456 warn("couldn't open %s", XPT_DEVICE); 457 return(1); 458 } 459 460 bzero(&ccb, sizeof(union ccb)); 461 462 ccb.ccb_h.path_id = CAM_XPT_PATH_ID; 463 ccb.ccb_h.target_id = CAM_TARGET_WILDCARD; 464 ccb.ccb_h.target_lun = CAM_LUN_WILDCARD; 465 466 ccb.ccb_h.func_code = XPT_DEV_MATCH; 467 bufsize = sizeof(struct dev_match_result) * 100; 468 ccb.cdm.match_buf_len = bufsize; 469 ccb.cdm.matches = (struct dev_match_result *)malloc(bufsize); 470 if (ccb.cdm.matches == NULL) { 471 warnx("can't malloc memory for matches"); 472 close(fd); 473 return(1); 474 } 475 ccb.cdm.num_matches = 0; 476 477 /* 478 * We fetch all nodes, since we display most of them in the default 479 * case, and all in the verbose case. 480 */ 481 ccb.cdm.num_patterns = 0; 482 ccb.cdm.pattern_buf_len = 0; 483 484 /* 485 * We do the ioctl multiple times if necessary, in case there are 486 * more than 100 nodes in the EDT. 487 */ 488 do { 489 if (ioctl(fd, CAMIOCOMMAND, &ccb) == -1) { 490 warn("error sending CAMIOCOMMAND ioctl"); 491 error = 1; 492 break; 493 } 494 495 if ((ccb.ccb_h.status != CAM_REQ_CMP) 496 || ((ccb.cdm.status != CAM_DEV_MATCH_LAST) 497 && (ccb.cdm.status != CAM_DEV_MATCH_MORE))) { 498 warnx("got CAM error %#x, CDM error %d\n", 499 ccb.ccb_h.status, ccb.cdm.status); 500 error = 1; 501 break; 502 } 503 504 for (i = 0; i < ccb.cdm.num_matches; i++) { 505 switch (ccb.cdm.matches[i].type) { 506 case DEV_MATCH_BUS: { 507 struct bus_match_result *bus_result; 508 509 /* 510 * Only print the bus information if the 511 * user turns on the verbose flag. 512 */ 513 if ((busonly == 0) && 514 (arglist & CAM_ARG_VERBOSE) == 0) 515 break; 516 517 bus_result = 518 &ccb.cdm.matches[i].result.bus_result; 519 520 if (need_close) { 521 fprintf(stdout, ")\n"); 522 need_close = 0; 523 } 524 525 fprintf(stdout, "scbus%d on %s%d bus %d%s\n", 526 bus_result->path_id, 527 bus_result->dev_name, 528 bus_result->unit_number, 529 bus_result->bus_id, 530 (busonly ? "" : ":")); 531 break; 532 } 533 case DEV_MATCH_DEVICE: { 534 struct device_match_result *dev_result; 535 char vendor[16], product[48], revision[16]; 536 char fw[5], tmpstr[256]; 537 538 if (busonly == 1) 539 break; 540 541 dev_result = 542 &ccb.cdm.matches[i].result.device_result; 543 544 if ((dev_result->flags 545 & DEV_RESULT_UNCONFIGURED) 546 && ((arglist & CAM_ARG_VERBOSE) == 0)) { 547 skip_device = 1; 548 break; 549 } else 550 skip_device = 0; 551 552 if (dev_result->protocol == PROTO_SCSI) { 553 cam_strvis(vendor, dev_result->inq_data.vendor, 554 sizeof(dev_result->inq_data.vendor), 555 sizeof(vendor)); 556 cam_strvis(product, 557 dev_result->inq_data.product, 558 sizeof(dev_result->inq_data.product), 559 sizeof(product)); 560 cam_strvis(revision, 561 dev_result->inq_data.revision, 562 sizeof(dev_result->inq_data.revision), 563 sizeof(revision)); 564 sprintf(tmpstr, "<%s %s %s>", vendor, product, 565 revision); 566 } else if (dev_result->protocol == PROTO_ATA || 567 dev_result->protocol == PROTO_SATAPM) { 568 cam_strvis(product, 569 dev_result->ident_data.model, 570 sizeof(dev_result->ident_data.model), 571 sizeof(product)); 572 cam_strvis(revision, 573 dev_result->ident_data.revision, 574 sizeof(dev_result->ident_data.revision), 575 sizeof(revision)); 576 sprintf(tmpstr, "<%s %s>", product, 577 revision); 578 } else if (dev_result->protocol == PROTO_SEMB) { 579 struct sep_identify_data *sid; 580 581 sid = (struct sep_identify_data *) 582 &dev_result->ident_data; 583 cam_strvis(vendor, sid->vendor_id, 584 sizeof(sid->vendor_id), 585 sizeof(vendor)); 586 cam_strvis(product, sid->product_id, 587 sizeof(sid->product_id), 588 sizeof(product)); 589 cam_strvis(revision, sid->product_rev, 590 sizeof(sid->product_rev), 591 sizeof(revision)); 592 cam_strvis(fw, sid->firmware_rev, 593 sizeof(sid->firmware_rev), 594 sizeof(fw)); 595 sprintf(tmpstr, "<%s %s %s %s>", 596 vendor, product, revision, fw); 597 } else { 598 sprintf(tmpstr, "<>"); 599 } 600 if (need_close) { 601 fprintf(stdout, ")\n"); 602 need_close = 0; 603 } 604 605 fprintf(stdout, "%-33s at scbus%d " 606 "target %d lun %jx (", 607 tmpstr, 608 dev_result->path_id, 609 dev_result->target_id, 610 (uintmax_t)dev_result->target_lun); 611 612 need_close = 1; 613 614 break; 615 } 616 case DEV_MATCH_PERIPH: { 617 struct periph_match_result *periph_result; 618 619 periph_result = 620 &ccb.cdm.matches[i].result.periph_result; 621 622 if (busonly || skip_device != 0) 623 break; 624 625 if (need_close > 1) 626 fprintf(stdout, ","); 627 628 fprintf(stdout, "%s%d", 629 periph_result->periph_name, 630 periph_result->unit_number); 631 632 need_close++; 633 break; 634 } 635 default: 636 fprintf(stdout, "unknown match type\n"); 637 break; 638 } 639 } 640 641 } while ((ccb.ccb_h.status == CAM_REQ_CMP) 642 && (ccb.cdm.status == CAM_DEV_MATCH_MORE)); 643 644 if (need_close) 645 fprintf(stdout, ")\n"); 646 647 close(fd); 648 649 return(error); 650 } 651 652 #ifndef MINIMALISTIC 653 static int 654 testunitready(struct cam_device *device, int retry_count, int timeout, 655 int quiet) 656 { 657 int error = 0; 658 union ccb *ccb; 659 660 ccb = cam_getccb(device); 661 662 scsi_test_unit_ready(&ccb->csio, 663 /* retries */ retry_count, 664 /* cbfcnp */ NULL, 665 /* tag_action */ MSG_SIMPLE_Q_TAG, 666 /* sense_len */ SSD_FULL_SIZE, 667 /* timeout */ timeout ? timeout : 5000); 668 669 /* Disable freezing the device queue */ 670 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 671 672 if (arglist & CAM_ARG_ERR_RECOVER) 673 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 674 675 if (cam_send_ccb(device, ccb) < 0) { 676 if (quiet == 0) 677 perror("error sending test unit ready"); 678 679 if (arglist & CAM_ARG_VERBOSE) { 680 cam_error_print(device, ccb, CAM_ESF_ALL, 681 CAM_EPF_ALL, stderr); 682 } 683 684 cam_freeccb(ccb); 685 return(1); 686 } 687 688 if ((ccb->ccb_h.status & CAM_STATUS_MASK) == CAM_REQ_CMP) { 689 if (quiet == 0) 690 fprintf(stdout, "Unit is ready\n"); 691 } else { 692 if (quiet == 0) 693 fprintf(stdout, "Unit is not ready\n"); 694 error = 1; 695 696 if (arglist & CAM_ARG_VERBOSE) { 697 cam_error_print(device, ccb, CAM_ESF_ALL, 698 CAM_EPF_ALL, stderr); 699 } 700 } 701 702 cam_freeccb(ccb); 703 704 return(error); 705 } 706 707 static int 708 scsistart(struct cam_device *device, int startstop, int loadeject, 709 int retry_count, int timeout) 710 { 711 union ccb *ccb; 712 int error = 0; 713 714 ccb = cam_getccb(device); 715 716 /* 717 * If we're stopping, send an ordered tag so the drive in question 718 * will finish any previously queued writes before stopping. If 719 * the device isn't capable of tagged queueing, or if tagged 720 * queueing is turned off, the tag action is a no-op. 721 */ 722 scsi_start_stop(&ccb->csio, 723 /* retries */ retry_count, 724 /* cbfcnp */ NULL, 725 /* tag_action */ startstop ? MSG_SIMPLE_Q_TAG : 726 MSG_ORDERED_Q_TAG, 727 /* start/stop */ startstop, 728 /* load_eject */ loadeject, 729 /* immediate */ 0, 730 /* sense_len */ SSD_FULL_SIZE, 731 /* timeout */ timeout ? timeout : 120000); 732 733 /* Disable freezing the device queue */ 734 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 735 736 if (arglist & CAM_ARG_ERR_RECOVER) 737 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 738 739 if (cam_send_ccb(device, ccb) < 0) { 740 perror("error sending start unit"); 741 742 if (arglist & CAM_ARG_VERBOSE) { 743 cam_error_print(device, ccb, CAM_ESF_ALL, 744 CAM_EPF_ALL, stderr); 745 } 746 747 cam_freeccb(ccb); 748 return(1); 749 } 750 751 if ((ccb->ccb_h.status & CAM_STATUS_MASK) == CAM_REQ_CMP) 752 if (startstop) { 753 fprintf(stdout, "Unit started successfully"); 754 if (loadeject) 755 fprintf(stdout,", Media loaded\n"); 756 else 757 fprintf(stdout,"\n"); 758 } else { 759 fprintf(stdout, "Unit stopped successfully"); 760 if (loadeject) 761 fprintf(stdout, ", Media ejected\n"); 762 else 763 fprintf(stdout, "\n"); 764 } 765 else { 766 error = 1; 767 if (startstop) 768 fprintf(stdout, 769 "Error received from start unit command\n"); 770 else 771 fprintf(stdout, 772 "Error received from stop unit command\n"); 773 774 if (arglist & CAM_ARG_VERBOSE) { 775 cam_error_print(device, ccb, CAM_ESF_ALL, 776 CAM_EPF_ALL, stderr); 777 } 778 } 779 780 cam_freeccb(ccb); 781 782 return(error); 783 } 784 785 int 786 scsidoinquiry(struct cam_device *device, int argc, char **argv, 787 char *combinedopt, int retry_count, int timeout) 788 { 789 int c; 790 int error = 0; 791 792 while ((c = getopt(argc, argv, combinedopt)) != -1) { 793 switch(c) { 794 case 'D': 795 arglist |= CAM_ARG_GET_STDINQ; 796 break; 797 case 'R': 798 arglist |= CAM_ARG_GET_XFERRATE; 799 break; 800 case 'S': 801 arglist |= CAM_ARG_GET_SERIAL; 802 break; 803 default: 804 break; 805 } 806 } 807 808 /* 809 * If the user didn't specify any inquiry options, he wants all of 810 * them. 811 */ 812 if ((arglist & CAM_ARG_INQ_MASK) == 0) 813 arglist |= CAM_ARG_INQ_MASK; 814 815 if (arglist & CAM_ARG_GET_STDINQ) 816 error = scsiinquiry(device, retry_count, timeout); 817 818 if (error != 0) 819 return(error); 820 821 if (arglist & CAM_ARG_GET_SERIAL) 822 scsiserial(device, retry_count, timeout); 823 824 if (arglist & CAM_ARG_GET_XFERRATE) 825 error = camxferrate(device); 826 827 return(error); 828 } 829 830 static int 831 scsiinquiry(struct cam_device *device, int retry_count, int timeout) 832 { 833 union ccb *ccb; 834 struct scsi_inquiry_data *inq_buf; 835 int error = 0; 836 837 ccb = cam_getccb(device); 838 839 if (ccb == NULL) { 840 warnx("couldn't allocate CCB"); 841 return(1); 842 } 843 844 /* cam_getccb cleans up the header, caller has to zero the payload */ 845 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 846 847 inq_buf = (struct scsi_inquiry_data *)malloc( 848 sizeof(struct scsi_inquiry_data)); 849 850 if (inq_buf == NULL) { 851 cam_freeccb(ccb); 852 warnx("can't malloc memory for inquiry\n"); 853 return(1); 854 } 855 bzero(inq_buf, sizeof(*inq_buf)); 856 857 /* 858 * Note that although the size of the inquiry buffer is the full 859 * 256 bytes specified in the SCSI spec, we only tell the device 860 * that we have allocated SHORT_INQUIRY_LENGTH bytes. There are 861 * two reasons for this: 862 * 863 * - The SCSI spec says that when a length field is only 1 byte, 864 * a value of 0 will be interpreted as 256. Therefore 865 * scsi_inquiry() will convert an inq_len (which is passed in as 866 * a u_int32_t, but the field in the CDB is only 1 byte) of 256 867 * to 0. Evidently, very few devices meet the spec in that 868 * regard. Some devices, like many Seagate disks, take the 0 as 869 * 0, and don't return any data. One Pioneer DVD-R drive 870 * returns more data than the command asked for. 871 * 872 * So, since there are numerous devices that just don't work 873 * right with the full inquiry size, we don't send the full size. 874 * 875 * - The second reason not to use the full inquiry data length is 876 * that we don't need it here. The only reason we issue a 877 * standard inquiry is to get the vendor name, device name, 878 * and revision so scsi_print_inquiry() can print them. 879 * 880 * If, at some point in the future, more inquiry data is needed for 881 * some reason, this code should use a procedure similar to the 882 * probe code. i.e., issue a short inquiry, and determine from 883 * the additional length passed back from the device how much 884 * inquiry data the device supports. Once the amount the device 885 * supports is determined, issue an inquiry for that amount and no 886 * more. 887 * 888 * KDM, 2/18/2000 889 */ 890 scsi_inquiry(&ccb->csio, 891 /* retries */ retry_count, 892 /* cbfcnp */ NULL, 893 /* tag_action */ MSG_SIMPLE_Q_TAG, 894 /* inq_buf */ (u_int8_t *)inq_buf, 895 /* inq_len */ SHORT_INQUIRY_LENGTH, 896 /* evpd */ 0, 897 /* page_code */ 0, 898 /* sense_len */ SSD_FULL_SIZE, 899 /* timeout */ timeout ? timeout : 5000); 900 901 /* Disable freezing the device queue */ 902 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 903 904 if (arglist & CAM_ARG_ERR_RECOVER) 905 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 906 907 if (cam_send_ccb(device, ccb) < 0) { 908 perror("error sending SCSI inquiry"); 909 910 if (arglist & CAM_ARG_VERBOSE) { 911 cam_error_print(device, ccb, CAM_ESF_ALL, 912 CAM_EPF_ALL, stderr); 913 } 914 915 cam_freeccb(ccb); 916 return(1); 917 } 918 919 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 920 error = 1; 921 922 if (arglist & CAM_ARG_VERBOSE) { 923 cam_error_print(device, ccb, CAM_ESF_ALL, 924 CAM_EPF_ALL, stderr); 925 } 926 } 927 928 cam_freeccb(ccb); 929 930 if (error != 0) { 931 free(inq_buf); 932 return(error); 933 } 934 935 fprintf(stdout, "%s%d: ", device->device_name, 936 device->dev_unit_num); 937 scsi_print_inquiry(inq_buf); 938 939 free(inq_buf); 940 941 return(0); 942 } 943 944 static int 945 scsiserial(struct cam_device *device, int retry_count, int timeout) 946 { 947 union ccb *ccb; 948 struct scsi_vpd_unit_serial_number *serial_buf; 949 char serial_num[SVPD_SERIAL_NUM_SIZE + 1]; 950 int error = 0; 951 952 ccb = cam_getccb(device); 953 954 if (ccb == NULL) { 955 warnx("couldn't allocate CCB"); 956 return(1); 957 } 958 959 /* cam_getccb cleans up the header, caller has to zero the payload */ 960 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 961 962 serial_buf = (struct scsi_vpd_unit_serial_number *) 963 malloc(sizeof(*serial_buf)); 964 965 if (serial_buf == NULL) { 966 cam_freeccb(ccb); 967 warnx("can't malloc memory for serial number"); 968 return(1); 969 } 970 971 scsi_inquiry(&ccb->csio, 972 /*retries*/ retry_count, 973 /*cbfcnp*/ NULL, 974 /* tag_action */ MSG_SIMPLE_Q_TAG, 975 /* inq_buf */ (u_int8_t *)serial_buf, 976 /* inq_len */ sizeof(*serial_buf), 977 /* evpd */ 1, 978 /* page_code */ SVPD_UNIT_SERIAL_NUMBER, 979 /* sense_len */ SSD_FULL_SIZE, 980 /* timeout */ timeout ? timeout : 5000); 981 982 /* Disable freezing the device queue */ 983 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 984 985 if (arglist & CAM_ARG_ERR_RECOVER) 986 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 987 988 if (cam_send_ccb(device, ccb) < 0) { 989 warn("error getting serial number"); 990 991 if (arglist & CAM_ARG_VERBOSE) { 992 cam_error_print(device, ccb, CAM_ESF_ALL, 993 CAM_EPF_ALL, stderr); 994 } 995 996 cam_freeccb(ccb); 997 free(serial_buf); 998 return(1); 999 } 1000 1001 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 1002 error = 1; 1003 1004 if (arglist & CAM_ARG_VERBOSE) { 1005 cam_error_print(device, ccb, CAM_ESF_ALL, 1006 CAM_EPF_ALL, stderr); 1007 } 1008 } 1009 1010 cam_freeccb(ccb); 1011 1012 if (error != 0) { 1013 free(serial_buf); 1014 return(error); 1015 } 1016 1017 bcopy(serial_buf->serial_num, serial_num, serial_buf->length); 1018 serial_num[serial_buf->length] = '\0'; 1019 1020 if ((arglist & CAM_ARG_GET_STDINQ) 1021 || (arglist & CAM_ARG_GET_XFERRATE)) 1022 fprintf(stdout, "%s%d: Serial Number ", 1023 device->device_name, device->dev_unit_num); 1024 1025 fprintf(stdout, "%.60s\n", serial_num); 1026 1027 free(serial_buf); 1028 1029 return(0); 1030 } 1031 1032 int 1033 camxferrate(struct cam_device *device) 1034 { 1035 struct ccb_pathinq cpi; 1036 u_int32_t freq = 0; 1037 u_int32_t speed = 0; 1038 union ccb *ccb; 1039 u_int mb; 1040 int retval = 0; 1041 1042 if ((retval = get_cpi(device, &cpi)) != 0) 1043 return (1); 1044 1045 ccb = cam_getccb(device); 1046 1047 if (ccb == NULL) { 1048 warnx("couldn't allocate CCB"); 1049 return(1); 1050 } 1051 1052 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->cts); 1053 1054 ccb->ccb_h.func_code = XPT_GET_TRAN_SETTINGS; 1055 ccb->cts.type = CTS_TYPE_CURRENT_SETTINGS; 1056 1057 if (((retval = cam_send_ccb(device, ccb)) < 0) 1058 || ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP)) { 1059 const char error_string[] = "error getting transfer settings"; 1060 1061 if (retval < 0) 1062 warn(error_string); 1063 else 1064 warnx(error_string); 1065 1066 if (arglist & CAM_ARG_VERBOSE) 1067 cam_error_print(device, ccb, CAM_ESF_ALL, 1068 CAM_EPF_ALL, stderr); 1069 1070 retval = 1; 1071 1072 goto xferrate_bailout; 1073 1074 } 1075 1076 speed = cpi.base_transfer_speed; 1077 freq = 0; 1078 if (ccb->cts.transport == XPORT_SPI) { 1079 struct ccb_trans_settings_spi *spi = 1080 &ccb->cts.xport_specific.spi; 1081 1082 if ((spi->valid & CTS_SPI_VALID_SYNC_RATE) != 0) { 1083 freq = scsi_calc_syncsrate(spi->sync_period); 1084 speed = freq; 1085 } 1086 if ((spi->valid & CTS_SPI_VALID_BUS_WIDTH) != 0) { 1087 speed *= (0x01 << spi->bus_width); 1088 } 1089 } else if (ccb->cts.transport == XPORT_FC) { 1090 struct ccb_trans_settings_fc *fc = 1091 &ccb->cts.xport_specific.fc; 1092 1093 if (fc->valid & CTS_FC_VALID_SPEED) 1094 speed = fc->bitrate; 1095 } else if (ccb->cts.transport == XPORT_SAS) { 1096 struct ccb_trans_settings_sas *sas = 1097 &ccb->cts.xport_specific.sas; 1098 1099 if (sas->valid & CTS_SAS_VALID_SPEED) 1100 speed = sas->bitrate; 1101 } else if (ccb->cts.transport == XPORT_ATA) { 1102 struct ccb_trans_settings_pata *pata = 1103 &ccb->cts.xport_specific.ata; 1104 1105 if (pata->valid & CTS_ATA_VALID_MODE) 1106 speed = ata_mode2speed(pata->mode); 1107 } else if (ccb->cts.transport == XPORT_SATA) { 1108 struct ccb_trans_settings_sata *sata = 1109 &ccb->cts.xport_specific.sata; 1110 1111 if (sata->valid & CTS_SATA_VALID_REVISION) 1112 speed = ata_revision2speed(sata->revision); 1113 } 1114 1115 mb = speed / 1000; 1116 if (mb > 0) { 1117 fprintf(stdout, "%s%d: %d.%03dMB/s transfers", 1118 device->device_name, device->dev_unit_num, 1119 mb, speed % 1000); 1120 } else { 1121 fprintf(stdout, "%s%d: %dKB/s transfers", 1122 device->device_name, device->dev_unit_num, 1123 speed); 1124 } 1125 1126 if (ccb->cts.transport == XPORT_SPI) { 1127 struct ccb_trans_settings_spi *spi = 1128 &ccb->cts.xport_specific.spi; 1129 1130 if (((spi->valid & CTS_SPI_VALID_SYNC_OFFSET) != 0) 1131 && (spi->sync_offset != 0)) 1132 fprintf(stdout, " (%d.%03dMHz, offset %d", freq / 1000, 1133 freq % 1000, spi->sync_offset); 1134 1135 if (((spi->valid & CTS_SPI_VALID_BUS_WIDTH) != 0) 1136 && (spi->bus_width > 0)) { 1137 if (((spi->valid & CTS_SPI_VALID_SYNC_OFFSET) != 0) 1138 && (spi->sync_offset != 0)) { 1139 fprintf(stdout, ", "); 1140 } else { 1141 fprintf(stdout, " ("); 1142 } 1143 fprintf(stdout, "%dbit)", 8 * (0x01 << spi->bus_width)); 1144 } else if (((spi->valid & CTS_SPI_VALID_SYNC_OFFSET) != 0) 1145 && (spi->sync_offset != 0)) { 1146 fprintf(stdout, ")"); 1147 } 1148 } else if (ccb->cts.transport == XPORT_ATA) { 1149 struct ccb_trans_settings_pata *pata = 1150 &ccb->cts.xport_specific.ata; 1151 1152 printf(" ("); 1153 if (pata->valid & CTS_ATA_VALID_MODE) 1154 printf("%s, ", ata_mode2string(pata->mode)); 1155 if ((pata->valid & CTS_ATA_VALID_ATAPI) && pata->atapi != 0) 1156 printf("ATAPI %dbytes, ", pata->atapi); 1157 if (pata->valid & CTS_ATA_VALID_BYTECOUNT) 1158 printf("PIO %dbytes", pata->bytecount); 1159 printf(")"); 1160 } else if (ccb->cts.transport == XPORT_SATA) { 1161 struct ccb_trans_settings_sata *sata = 1162 &ccb->cts.xport_specific.sata; 1163 1164 printf(" ("); 1165 if (sata->valid & CTS_SATA_VALID_REVISION) 1166 printf("SATA %d.x, ", sata->revision); 1167 else 1168 printf("SATA, "); 1169 if (sata->valid & CTS_SATA_VALID_MODE) 1170 printf("%s, ", ata_mode2string(sata->mode)); 1171 if ((sata->valid & CTS_SATA_VALID_ATAPI) && sata->atapi != 0) 1172 printf("ATAPI %dbytes, ", sata->atapi); 1173 if (sata->valid & CTS_SATA_VALID_BYTECOUNT) 1174 printf("PIO %dbytes", sata->bytecount); 1175 printf(")"); 1176 } 1177 1178 if (ccb->cts.protocol == PROTO_SCSI) { 1179 struct ccb_trans_settings_scsi *scsi = 1180 &ccb->cts.proto_specific.scsi; 1181 if (scsi->valid & CTS_SCSI_VALID_TQ) { 1182 if (scsi->flags & CTS_SCSI_FLAGS_TAG_ENB) { 1183 fprintf(stdout, ", Command Queueing Enabled"); 1184 } 1185 } 1186 } 1187 1188 fprintf(stdout, "\n"); 1189 1190 xferrate_bailout: 1191 1192 cam_freeccb(ccb); 1193 1194 return(retval); 1195 } 1196 1197 static void 1198 atahpa_print(struct ata_params *parm, u_int64_t hpasize, int header) 1199 { 1200 u_int32_t lbasize = (u_int32_t)parm->lba_size_1 | 1201 ((u_int32_t)parm->lba_size_2 << 16); 1202 1203 u_int64_t lbasize48 = ((u_int64_t)parm->lba_size48_1) | 1204 ((u_int64_t)parm->lba_size48_2 << 16) | 1205 ((u_int64_t)parm->lba_size48_3 << 32) | 1206 ((u_int64_t)parm->lba_size48_4 << 48); 1207 1208 if (header) { 1209 printf("\nFeature " 1210 "Support Enabled Value\n"); 1211 } 1212 1213 printf("Host Protected Area (HPA) "); 1214 if (parm->support.command1 & ATA_SUPPORT_PROTECTED) { 1215 u_int64_t lba = lbasize48 ? lbasize48 : lbasize; 1216 printf("yes %s %ju/%ju\n", (hpasize > lba) ? "yes" : "no ", 1217 lba, hpasize); 1218 1219 printf("HPA - Security "); 1220 if (parm->support.command1 & ATA_SUPPORT_MAXSECURITY) 1221 printf("yes\n"); 1222 else 1223 printf("no\n"); 1224 } else { 1225 printf("no\n"); 1226 } 1227 } 1228 1229 static int 1230 atasata(struct ata_params *parm) 1231 { 1232 1233 1234 if (parm->satacapabilities != 0xffff && 1235 parm->satacapabilities != 0x0000) 1236 return 1; 1237 1238 return 0; 1239 } 1240 1241 static void 1242 atacapprint(struct ata_params *parm) 1243 { 1244 u_int32_t lbasize = (u_int32_t)parm->lba_size_1 | 1245 ((u_int32_t)parm->lba_size_2 << 16); 1246 1247 u_int64_t lbasize48 = ((u_int64_t)parm->lba_size48_1) | 1248 ((u_int64_t)parm->lba_size48_2 << 16) | 1249 ((u_int64_t)parm->lba_size48_3 << 32) | 1250 ((u_int64_t)parm->lba_size48_4 << 48); 1251 1252 printf("\n"); 1253 printf("protocol "); 1254 printf("ATA/ATAPI-%d", ata_version(parm->version_major)); 1255 if (parm->satacapabilities && parm->satacapabilities != 0xffff) { 1256 if (parm->satacapabilities & ATA_SATA_GEN3) 1257 printf(" SATA 3.x\n"); 1258 else if (parm->satacapabilities & ATA_SATA_GEN2) 1259 printf(" SATA 2.x\n"); 1260 else if (parm->satacapabilities & ATA_SATA_GEN1) 1261 printf(" SATA 1.x\n"); 1262 else 1263 printf(" SATA\n"); 1264 } 1265 else 1266 printf("\n"); 1267 printf("device model %.40s\n", parm->model); 1268 printf("firmware revision %.8s\n", parm->revision); 1269 printf("serial number %.20s\n", parm->serial); 1270 if (parm->enabled.extension & ATA_SUPPORT_64BITWWN) { 1271 printf("WWN %04x%04x%04x%04x\n", 1272 parm->wwn[0], parm->wwn[1], parm->wwn[2], parm->wwn[3]); 1273 } 1274 if (parm->enabled.extension & ATA_SUPPORT_MEDIASN) { 1275 printf("media serial number %.30s\n", 1276 parm->media_serial); 1277 } 1278 1279 printf("cylinders %d\n", parm->cylinders); 1280 printf("heads %d\n", parm->heads); 1281 printf("sectors/track %d\n", parm->sectors); 1282 printf("sector size logical %u, physical %lu, offset %lu\n", 1283 ata_logical_sector_size(parm), 1284 (unsigned long)ata_physical_sector_size(parm), 1285 (unsigned long)ata_logical_sector_offset(parm)); 1286 1287 if (parm->config == ATA_PROTO_CFA || 1288 (parm->support.command2 & ATA_SUPPORT_CFA)) 1289 printf("CFA supported\n"); 1290 1291 printf("LBA%ssupported ", 1292 parm->capabilities1 & ATA_SUPPORT_LBA ? " " : " not "); 1293 if (lbasize) 1294 printf("%d sectors\n", lbasize); 1295 else 1296 printf("\n"); 1297 1298 printf("LBA48%ssupported ", 1299 parm->support.command2 & ATA_SUPPORT_ADDRESS48 ? " " : " not "); 1300 if (lbasize48) 1301 printf("%ju sectors\n", (uintmax_t)lbasize48); 1302 else 1303 printf("\n"); 1304 1305 printf("PIO supported PIO"); 1306 switch (ata_max_pmode(parm)) { 1307 case ATA_PIO4: 1308 printf("4"); 1309 break; 1310 case ATA_PIO3: 1311 printf("3"); 1312 break; 1313 case ATA_PIO2: 1314 printf("2"); 1315 break; 1316 case ATA_PIO1: 1317 printf("1"); 1318 break; 1319 default: 1320 printf("0"); 1321 } 1322 if ((parm->capabilities1 & ATA_SUPPORT_IORDY) == 0) 1323 printf(" w/o IORDY"); 1324 printf("\n"); 1325 1326 printf("DMA%ssupported ", 1327 parm->capabilities1 & ATA_SUPPORT_DMA ? " " : " not "); 1328 if (parm->capabilities1 & ATA_SUPPORT_DMA) { 1329 if (parm->mwdmamodes & 0xff) { 1330 printf("WDMA"); 1331 if (parm->mwdmamodes & 0x04) 1332 printf("2"); 1333 else if (parm->mwdmamodes & 0x02) 1334 printf("1"); 1335 else if (parm->mwdmamodes & 0x01) 1336 printf("0"); 1337 printf(" "); 1338 } 1339 if ((parm->atavalid & ATA_FLAG_88) && 1340 (parm->udmamodes & 0xff)) { 1341 printf("UDMA"); 1342 if (parm->udmamodes & 0x40) 1343 printf("6"); 1344 else if (parm->udmamodes & 0x20) 1345 printf("5"); 1346 else if (parm->udmamodes & 0x10) 1347 printf("4"); 1348 else if (parm->udmamodes & 0x08) 1349 printf("3"); 1350 else if (parm->udmamodes & 0x04) 1351 printf("2"); 1352 else if (parm->udmamodes & 0x02) 1353 printf("1"); 1354 else if (parm->udmamodes & 0x01) 1355 printf("0"); 1356 printf(" "); 1357 } 1358 } 1359 printf("\n"); 1360 1361 if (parm->media_rotation_rate == 1) { 1362 printf("media RPM non-rotating\n"); 1363 } else if (parm->media_rotation_rate >= 0x0401 && 1364 parm->media_rotation_rate <= 0xFFFE) { 1365 printf("media RPM %d\n", 1366 parm->media_rotation_rate); 1367 } 1368 1369 printf("\nFeature " 1370 "Support Enabled Value Vendor\n"); 1371 printf("read ahead %s %s\n", 1372 parm->support.command1 & ATA_SUPPORT_LOOKAHEAD ? "yes" : "no", 1373 parm->enabled.command1 & ATA_SUPPORT_LOOKAHEAD ? "yes" : "no"); 1374 printf("write cache %s %s\n", 1375 parm->support.command1 & ATA_SUPPORT_WRITECACHE ? "yes" : "no", 1376 parm->enabled.command1 & ATA_SUPPORT_WRITECACHE ? "yes" : "no"); 1377 printf("flush cache %s %s\n", 1378 parm->support.command2 & ATA_SUPPORT_FLUSHCACHE ? "yes" : "no", 1379 parm->enabled.command2 & ATA_SUPPORT_FLUSHCACHE ? "yes" : "no"); 1380 printf("overlap %s\n", 1381 parm->capabilities1 & ATA_SUPPORT_OVERLAP ? "yes" : "no"); 1382 printf("Tagged Command Queuing (TCQ) %s %s", 1383 parm->support.command2 & ATA_SUPPORT_QUEUED ? "yes" : "no", 1384 parm->enabled.command2 & ATA_SUPPORT_QUEUED ? "yes" : "no"); 1385 if (parm->support.command2 & ATA_SUPPORT_QUEUED) { 1386 printf(" %d tags\n", 1387 ATA_QUEUE_LEN(parm->queue) + 1); 1388 } else 1389 printf("\n"); 1390 printf("Native Command Queuing (NCQ) "); 1391 if (parm->satacapabilities != 0xffff && 1392 (parm->satacapabilities & ATA_SUPPORT_NCQ)) { 1393 printf("yes %d tags\n", 1394 ATA_QUEUE_LEN(parm->queue) + 1); 1395 } else 1396 printf("no\n"); 1397 1398 printf("NCQ Queue Management %s\n", atasata(parm) && 1399 parm->satacapabilities2 & ATA_SUPPORT_NCQ_QMANAGEMENT ? 1400 "yes" : "no"); 1401 printf("NCQ Streaming %s\n", atasata(parm) && 1402 parm->satacapabilities2 & ATA_SUPPORT_NCQ_STREAM ? 1403 "yes" : "no"); 1404 printf("Receive & Send FPDMA Queued %s\n", atasata(parm) && 1405 parm->satacapabilities2 & ATA_SUPPORT_RCVSND_FPDMA_QUEUED ? 1406 "yes" : "no"); 1407 1408 printf("SMART %s %s\n", 1409 parm->support.command1 & ATA_SUPPORT_SMART ? "yes" : "no", 1410 parm->enabled.command1 & ATA_SUPPORT_SMART ? "yes" : "no"); 1411 printf("microcode download %s %s\n", 1412 parm->support.command2 & ATA_SUPPORT_MICROCODE ? "yes" : "no", 1413 parm->enabled.command2 & ATA_SUPPORT_MICROCODE ? "yes" : "no"); 1414 printf("security %s %s\n", 1415 parm->support.command1 & ATA_SUPPORT_SECURITY ? "yes" : "no", 1416 parm->enabled.command1 & ATA_SUPPORT_SECURITY ? "yes" : "no"); 1417 printf("power management %s %s\n", 1418 parm->support.command1 & ATA_SUPPORT_POWERMGT ? "yes" : "no", 1419 parm->enabled.command1 & ATA_SUPPORT_POWERMGT ? "yes" : "no"); 1420 printf("advanced power management %s %s", 1421 parm->support.command2 & ATA_SUPPORT_APM ? "yes" : "no", 1422 parm->enabled.command2 & ATA_SUPPORT_APM ? "yes" : "no"); 1423 if (parm->support.command2 & ATA_SUPPORT_APM) { 1424 printf(" %d/0x%02X\n", 1425 parm->apm_value & 0xff, parm->apm_value & 0xff); 1426 } else 1427 printf("\n"); 1428 printf("automatic acoustic management %s %s", 1429 parm->support.command2 & ATA_SUPPORT_AUTOACOUSTIC ? "yes" :"no", 1430 parm->enabled.command2 & ATA_SUPPORT_AUTOACOUSTIC ? "yes" :"no"); 1431 if (parm->support.command2 & ATA_SUPPORT_AUTOACOUSTIC) { 1432 printf(" %d/0x%02X %d/0x%02X\n", 1433 ATA_ACOUSTIC_CURRENT(parm->acoustic), 1434 ATA_ACOUSTIC_CURRENT(parm->acoustic), 1435 ATA_ACOUSTIC_VENDOR(parm->acoustic), 1436 ATA_ACOUSTIC_VENDOR(parm->acoustic)); 1437 } else 1438 printf("\n"); 1439 printf("media status notification %s %s\n", 1440 parm->support.command2 & ATA_SUPPORT_NOTIFY ? "yes" : "no", 1441 parm->enabled.command2 & ATA_SUPPORT_NOTIFY ? "yes" : "no"); 1442 printf("power-up in Standby %s %s\n", 1443 parm->support.command2 & ATA_SUPPORT_STANDBY ? "yes" : "no", 1444 parm->enabled.command2 & ATA_SUPPORT_STANDBY ? "yes" : "no"); 1445 printf("write-read-verify %s %s", 1446 parm->support2 & ATA_SUPPORT_WRITEREADVERIFY ? "yes" : "no", 1447 parm->enabled2 & ATA_SUPPORT_WRITEREADVERIFY ? "yes" : "no"); 1448 if (parm->support2 & ATA_SUPPORT_WRITEREADVERIFY) { 1449 printf(" %d/0x%x\n", 1450 parm->wrv_mode, parm->wrv_mode); 1451 } else 1452 printf("\n"); 1453 printf("unload %s %s\n", 1454 parm->support.extension & ATA_SUPPORT_UNLOAD ? "yes" : "no", 1455 parm->enabled.extension & ATA_SUPPORT_UNLOAD ? "yes" : "no"); 1456 printf("general purpose logging %s %s\n", 1457 parm->support.extension & ATA_SUPPORT_GENLOG ? "yes" : "no", 1458 parm->enabled.extension & ATA_SUPPORT_GENLOG ? "yes" : "no"); 1459 printf("free-fall %s %s\n", 1460 parm->support2 & ATA_SUPPORT_FREEFALL ? "yes" : "no", 1461 parm->enabled2 & ATA_SUPPORT_FREEFALL ? "yes" : "no"); 1462 printf("Data Set Management (DSM/TRIM) "); 1463 if (parm->support_dsm & ATA_SUPPORT_DSM_TRIM) { 1464 printf("yes\n"); 1465 printf("DSM - max 512byte blocks "); 1466 if (parm->max_dsm_blocks == 0x00) 1467 printf("yes not specified\n"); 1468 else 1469 printf("yes %d\n", 1470 parm->max_dsm_blocks); 1471 1472 printf("DSM - deterministic read "); 1473 if (parm->support3 & ATA_SUPPORT_DRAT) { 1474 if (parm->support3 & ATA_SUPPORT_RZAT) 1475 printf("yes zeroed\n"); 1476 else 1477 printf("yes any value\n"); 1478 } else { 1479 printf("no\n"); 1480 } 1481 } else { 1482 printf("no\n"); 1483 } 1484 } 1485 1486 static int 1487 scsi_cam_pass_16_send(struct cam_device *device, union ccb *ccb, int quiet) 1488 { 1489 struct ata_pass_16 *ata_pass_16; 1490 struct ata_cmd ata_cmd; 1491 1492 ata_pass_16 = (struct ata_pass_16 *)ccb->csio.cdb_io.cdb_bytes; 1493 ata_cmd.command = ata_pass_16->command; 1494 ata_cmd.control = ata_pass_16->control; 1495 ata_cmd.features = ata_pass_16->features; 1496 1497 if (arglist & CAM_ARG_VERBOSE) { 1498 warnx("sending ATA %s via pass_16 with timeout of %u msecs", 1499 ata_op_string(&ata_cmd), 1500 ccb->csio.ccb_h.timeout); 1501 } 1502 1503 /* Disable freezing the device queue */ 1504 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 1505 1506 if (arglist & CAM_ARG_ERR_RECOVER) 1507 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 1508 1509 if (cam_send_ccb(device, ccb) < 0) { 1510 if (quiet != 1 || arglist & CAM_ARG_VERBOSE) { 1511 warn("error sending ATA %s via pass_16", 1512 ata_op_string(&ata_cmd)); 1513 } 1514 1515 if (arglist & CAM_ARG_VERBOSE) { 1516 cam_error_print(device, ccb, CAM_ESF_ALL, 1517 CAM_EPF_ALL, stderr); 1518 } 1519 1520 return (1); 1521 } 1522 1523 if (!(ata_pass_16->flags & AP_FLAG_CHK_COND) && 1524 (ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 1525 if (quiet != 1 || arglist & CAM_ARG_VERBOSE) { 1526 warnx("ATA %s via pass_16 failed", 1527 ata_op_string(&ata_cmd)); 1528 } 1529 if (arglist & CAM_ARG_VERBOSE) { 1530 cam_error_print(device, ccb, CAM_ESF_ALL, 1531 CAM_EPF_ALL, stderr); 1532 } 1533 1534 return (1); 1535 } 1536 1537 return (0); 1538 } 1539 1540 1541 static int 1542 ata_cam_send(struct cam_device *device, union ccb *ccb, int quiet) 1543 { 1544 if (arglist & CAM_ARG_VERBOSE) { 1545 warnx("sending ATA %s with timeout of %u msecs", 1546 ata_op_string(&(ccb->ataio.cmd)), 1547 ccb->ataio.ccb_h.timeout); 1548 } 1549 1550 /* Disable freezing the device queue */ 1551 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 1552 1553 if (arglist & CAM_ARG_ERR_RECOVER) 1554 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 1555 1556 if (cam_send_ccb(device, ccb) < 0) { 1557 if (quiet != 1 || arglist & CAM_ARG_VERBOSE) { 1558 warn("error sending ATA %s", 1559 ata_op_string(&(ccb->ataio.cmd))); 1560 } 1561 1562 if (arglist & CAM_ARG_VERBOSE) { 1563 cam_error_print(device, ccb, CAM_ESF_ALL, 1564 CAM_EPF_ALL, stderr); 1565 } 1566 1567 return (1); 1568 } 1569 1570 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 1571 if (quiet != 1 || arglist & CAM_ARG_VERBOSE) { 1572 warnx("ATA %s failed: %d", 1573 ata_op_string(&(ccb->ataio.cmd)), quiet); 1574 } 1575 1576 if (arglist & CAM_ARG_VERBOSE) { 1577 cam_error_print(device, ccb, CAM_ESF_ALL, 1578 CAM_EPF_ALL, stderr); 1579 } 1580 1581 return (1); 1582 } 1583 1584 return (0); 1585 } 1586 1587 static int 1588 ata_do_pass_16(struct cam_device *device, union ccb *ccb, int retries, 1589 u_int32_t flags, u_int8_t protocol, u_int8_t ata_flags, 1590 u_int8_t tag_action, u_int8_t command, u_int8_t features, 1591 u_int64_t lba, u_int8_t sector_count, u_int8_t *data_ptr, 1592 u_int16_t dxfer_len, int timeout, int quiet) 1593 { 1594 if (data_ptr != NULL) { 1595 ata_flags |= AP_FLAG_BYT_BLOK_BYTES | 1596 AP_FLAG_TLEN_SECT_CNT; 1597 if (flags & CAM_DIR_OUT) 1598 ata_flags |= AP_FLAG_TDIR_TO_DEV; 1599 else 1600 ata_flags |= AP_FLAG_TDIR_FROM_DEV; 1601 } else { 1602 ata_flags |= AP_FLAG_TLEN_NO_DATA; 1603 } 1604 1605 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 1606 1607 scsi_ata_pass_16(&ccb->csio, 1608 retries, 1609 NULL, 1610 flags, 1611 tag_action, 1612 protocol, 1613 ata_flags, 1614 features, 1615 sector_count, 1616 lba, 1617 command, 1618 /*control*/0, 1619 data_ptr, 1620 dxfer_len, 1621 /*sense_len*/SSD_FULL_SIZE, 1622 timeout); 1623 1624 return scsi_cam_pass_16_send(device, ccb, quiet); 1625 } 1626 1627 static int 1628 ata_try_pass_16(struct cam_device *device) 1629 { 1630 struct ccb_pathinq cpi; 1631 1632 if (get_cpi(device, &cpi) != 0) { 1633 warnx("couldn't get CPI"); 1634 return (-1); 1635 } 1636 1637 if (cpi.protocol == PROTO_SCSI) { 1638 /* possibly compatible with pass_16 */ 1639 return (1); 1640 } 1641 1642 /* likely not compatible with pass_16 */ 1643 return (0); 1644 } 1645 1646 static int 1647 ata_do_28bit_cmd(struct cam_device *device, union ccb *ccb, int retries, 1648 u_int32_t flags, u_int8_t protocol, u_int8_t tag_action, 1649 u_int8_t command, u_int8_t features, u_int32_t lba, 1650 u_int8_t sector_count, u_int8_t *data_ptr, u_int16_t dxfer_len, 1651 int timeout, int quiet) 1652 { 1653 1654 1655 switch (ata_try_pass_16(device)) { 1656 case -1: 1657 return (1); 1658 case 1: 1659 /* Try using SCSI Passthrough */ 1660 return ata_do_pass_16(device, ccb, retries, flags, protocol, 1661 0, tag_action, command, features, lba, 1662 sector_count, data_ptr, dxfer_len, 1663 timeout, quiet); 1664 } 1665 1666 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->ataio); 1667 cam_fill_ataio(&ccb->ataio, 1668 retries, 1669 NULL, 1670 flags, 1671 tag_action, 1672 data_ptr, 1673 dxfer_len, 1674 timeout); 1675 1676 ata_28bit_cmd(&ccb->ataio, command, features, lba, sector_count); 1677 return ata_cam_send(device, ccb, quiet); 1678 } 1679 1680 static int 1681 ata_do_cmd(struct cam_device *device, union ccb *ccb, int retries, 1682 u_int32_t flags, u_int8_t protocol, u_int8_t ata_flags, 1683 u_int8_t tag_action, u_int8_t command, u_int8_t features, 1684 u_int64_t lba, u_int8_t sector_count, u_int8_t *data_ptr, 1685 u_int16_t dxfer_len, int timeout, int force48bit) 1686 { 1687 int retval; 1688 1689 retval = ata_try_pass_16(device); 1690 if (retval == -1) 1691 return (1); 1692 1693 if (retval == 1) { 1694 int error; 1695 1696 /* Try using SCSI Passthrough */ 1697 error = ata_do_pass_16(device, ccb, retries, flags, protocol, 1698 ata_flags, tag_action, command, features, 1699 lba, sector_count, data_ptr, dxfer_len, 1700 timeout, 0); 1701 1702 if (ata_flags & AP_FLAG_CHK_COND) { 1703 /* Decode ata_res from sense data */ 1704 struct ata_res_pass16 *res_pass16; 1705 struct ata_res *res; 1706 u_int i; 1707 u_int16_t *ptr; 1708 1709 /* sense_data is 4 byte aligned */ 1710 ptr = (uint16_t*)(uintptr_t)&ccb->csio.sense_data; 1711 for (i = 0; i < sizeof(*res_pass16) / 2; i++) 1712 ptr[i] = le16toh(ptr[i]); 1713 1714 /* sense_data is 4 byte aligned */ 1715 res_pass16 = (struct ata_res_pass16 *)(uintptr_t) 1716 &ccb->csio.sense_data; 1717 res = &ccb->ataio.res; 1718 res->flags = res_pass16->flags; 1719 res->status = res_pass16->status; 1720 res->error = res_pass16->error; 1721 res->lba_low = res_pass16->lba_low; 1722 res->lba_mid = res_pass16->lba_mid; 1723 res->lba_high = res_pass16->lba_high; 1724 res->device = res_pass16->device; 1725 res->lba_low_exp = res_pass16->lba_low_exp; 1726 res->lba_mid_exp = res_pass16->lba_mid_exp; 1727 res->lba_high_exp = res_pass16->lba_high_exp; 1728 res->sector_count = res_pass16->sector_count; 1729 res->sector_count_exp = res_pass16->sector_count_exp; 1730 } 1731 1732 return (error); 1733 } 1734 1735 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->ataio); 1736 cam_fill_ataio(&ccb->ataio, 1737 retries, 1738 NULL, 1739 flags, 1740 tag_action, 1741 data_ptr, 1742 dxfer_len, 1743 timeout); 1744 1745 if (force48bit || lba > ATA_MAX_28BIT_LBA) 1746 ata_48bit_cmd(&ccb->ataio, command, features, lba, sector_count); 1747 else 1748 ata_28bit_cmd(&ccb->ataio, command, features, lba, sector_count); 1749 1750 if (ata_flags & AP_FLAG_CHK_COND) 1751 ccb->ataio.cmd.flags |= CAM_ATAIO_NEEDRESULT; 1752 1753 return ata_cam_send(device, ccb, 0); 1754 } 1755 1756 static void 1757 dump_data(uint16_t *ptr, uint32_t len) 1758 { 1759 u_int i; 1760 1761 for (i = 0; i < len / 2; i++) { 1762 if ((i % 8) == 0) 1763 printf(" %3d: ", i); 1764 printf("%04hx ", ptr[i]); 1765 if ((i % 8) == 7) 1766 printf("\n"); 1767 } 1768 if ((i % 8) != 7) 1769 printf("\n"); 1770 } 1771 1772 static int 1773 atahpa_proc_resp(struct cam_device *device, union ccb *ccb, 1774 int is48bit, u_int64_t *hpasize) 1775 { 1776 struct ata_res *res; 1777 1778 res = &ccb->ataio.res; 1779 if (res->status & ATA_STATUS_ERROR) { 1780 if (arglist & CAM_ARG_VERBOSE) { 1781 cam_error_print(device, ccb, CAM_ESF_ALL, 1782 CAM_EPF_ALL, stderr); 1783 printf("error = 0x%02x, sector_count = 0x%04x, " 1784 "device = 0x%02x, status = 0x%02x\n", 1785 res->error, res->sector_count, 1786 res->device, res->status); 1787 } 1788 1789 if (res->error & ATA_ERROR_ID_NOT_FOUND) { 1790 warnx("Max address has already been set since " 1791 "last power-on or hardware reset"); 1792 } 1793 1794 return (1); 1795 } 1796 1797 if (arglist & CAM_ARG_VERBOSE) { 1798 fprintf(stdout, "%s%d: Raw native max data:\n", 1799 device->device_name, device->dev_unit_num); 1800 /* res is 4 byte aligned */ 1801 dump_data((uint16_t*)(uintptr_t)res, sizeof(struct ata_res)); 1802 1803 printf("error = 0x%02x, sector_count = 0x%04x, device = 0x%02x, " 1804 "status = 0x%02x\n", res->error, res->sector_count, 1805 res->device, res->status); 1806 } 1807 1808 if (hpasize != NULL) { 1809 if (is48bit) { 1810 *hpasize = (((u_int64_t)((res->lba_high_exp << 16) | 1811 (res->lba_mid_exp << 8) | res->lba_low_exp) << 24) | 1812 ((res->lba_high << 16) | (res->lba_mid << 8) | 1813 res->lba_low)) + 1; 1814 } else { 1815 *hpasize = (((res->device & 0x0f) << 24) | 1816 (res->lba_high << 16) | (res->lba_mid << 8) | 1817 res->lba_low) + 1; 1818 } 1819 } 1820 1821 return (0); 1822 } 1823 1824 static int 1825 ata_read_native_max(struct cam_device *device, int retry_count, 1826 u_int32_t timeout, union ccb *ccb, 1827 struct ata_params *parm, u_int64_t *hpasize) 1828 { 1829 int error; 1830 u_int cmd, is48bit; 1831 u_int8_t protocol; 1832 1833 is48bit = parm->support.command2 & ATA_SUPPORT_ADDRESS48; 1834 protocol = AP_PROTO_NON_DATA; 1835 1836 if (is48bit) { 1837 cmd = ATA_READ_NATIVE_MAX_ADDRESS48; 1838 protocol |= AP_EXTEND; 1839 } else { 1840 cmd = ATA_READ_NATIVE_MAX_ADDRESS; 1841 } 1842 1843 error = ata_do_cmd(device, 1844 ccb, 1845 retry_count, 1846 /*flags*/CAM_DIR_NONE, 1847 /*protocol*/protocol, 1848 /*ata_flags*/AP_FLAG_CHK_COND, 1849 /*tag_action*/MSG_SIMPLE_Q_TAG, 1850 /*command*/cmd, 1851 /*features*/0, 1852 /*lba*/0, 1853 /*sector_count*/0, 1854 /*data_ptr*/NULL, 1855 /*dxfer_len*/0, 1856 timeout ? timeout : 1000, 1857 is48bit); 1858 1859 if (error) 1860 return (error); 1861 1862 return atahpa_proc_resp(device, ccb, is48bit, hpasize); 1863 } 1864 1865 static int 1866 atahpa_set_max(struct cam_device *device, int retry_count, 1867 u_int32_t timeout, union ccb *ccb, 1868 int is48bit, u_int64_t maxsize, int persist) 1869 { 1870 int error; 1871 u_int cmd; 1872 u_int8_t protocol; 1873 1874 protocol = AP_PROTO_NON_DATA; 1875 1876 if (is48bit) { 1877 cmd = ATA_SET_MAX_ADDRESS48; 1878 protocol |= AP_EXTEND; 1879 } else { 1880 cmd = ATA_SET_MAX_ADDRESS; 1881 } 1882 1883 /* lba's are zero indexed so the max lba is requested max - 1 */ 1884 if (maxsize) 1885 maxsize--; 1886 1887 error = ata_do_cmd(device, 1888 ccb, 1889 retry_count, 1890 /*flags*/CAM_DIR_NONE, 1891 /*protocol*/protocol, 1892 /*ata_flags*/AP_FLAG_CHK_COND, 1893 /*tag_action*/MSG_SIMPLE_Q_TAG, 1894 /*command*/cmd, 1895 /*features*/ATA_HPA_FEAT_MAX_ADDR, 1896 /*lba*/maxsize, 1897 /*sector_count*/persist, 1898 /*data_ptr*/NULL, 1899 /*dxfer_len*/0, 1900 timeout ? timeout : 1000, 1901 is48bit); 1902 1903 if (error) 1904 return (error); 1905 1906 return atahpa_proc_resp(device, ccb, is48bit, NULL); 1907 } 1908 1909 static int 1910 atahpa_password(struct cam_device *device, int retry_count, 1911 u_int32_t timeout, union ccb *ccb, 1912 int is48bit, struct ata_set_max_pwd *pwd) 1913 { 1914 int error; 1915 u_int cmd; 1916 u_int8_t protocol; 1917 1918 protocol = AP_PROTO_PIO_OUT; 1919 cmd = (is48bit) ? ATA_SET_MAX_ADDRESS48 : ATA_SET_MAX_ADDRESS; 1920 1921 error = ata_do_cmd(device, 1922 ccb, 1923 retry_count, 1924 /*flags*/CAM_DIR_OUT, 1925 /*protocol*/protocol, 1926 /*ata_flags*/AP_FLAG_CHK_COND, 1927 /*tag_action*/MSG_SIMPLE_Q_TAG, 1928 /*command*/cmd, 1929 /*features*/ATA_HPA_FEAT_SET_PWD, 1930 /*lba*/0, 1931 /*sector_count*/0, 1932 /*data_ptr*/(u_int8_t*)pwd, 1933 /*dxfer_len*/sizeof(struct ata_set_max_pwd), 1934 timeout ? timeout : 1000, 1935 is48bit); 1936 1937 if (error) 1938 return (error); 1939 1940 return atahpa_proc_resp(device, ccb, is48bit, NULL); 1941 } 1942 1943 static int 1944 atahpa_lock(struct cam_device *device, int retry_count, 1945 u_int32_t timeout, union ccb *ccb, int is48bit) 1946 { 1947 int error; 1948 u_int cmd; 1949 u_int8_t protocol; 1950 1951 protocol = AP_PROTO_NON_DATA; 1952 cmd = (is48bit) ? ATA_SET_MAX_ADDRESS48 : ATA_SET_MAX_ADDRESS; 1953 1954 error = ata_do_cmd(device, 1955 ccb, 1956 retry_count, 1957 /*flags*/CAM_DIR_NONE, 1958 /*protocol*/protocol, 1959 /*ata_flags*/AP_FLAG_CHK_COND, 1960 /*tag_action*/MSG_SIMPLE_Q_TAG, 1961 /*command*/cmd, 1962 /*features*/ATA_HPA_FEAT_LOCK, 1963 /*lba*/0, 1964 /*sector_count*/0, 1965 /*data_ptr*/NULL, 1966 /*dxfer_len*/0, 1967 timeout ? timeout : 1000, 1968 is48bit); 1969 1970 if (error) 1971 return (error); 1972 1973 return atahpa_proc_resp(device, ccb, is48bit, NULL); 1974 } 1975 1976 static int 1977 atahpa_unlock(struct cam_device *device, int retry_count, 1978 u_int32_t timeout, union ccb *ccb, 1979 int is48bit, struct ata_set_max_pwd *pwd) 1980 { 1981 int error; 1982 u_int cmd; 1983 u_int8_t protocol; 1984 1985 protocol = AP_PROTO_PIO_OUT; 1986 cmd = (is48bit) ? ATA_SET_MAX_ADDRESS48 : ATA_SET_MAX_ADDRESS; 1987 1988 error = ata_do_cmd(device, 1989 ccb, 1990 retry_count, 1991 /*flags*/CAM_DIR_OUT, 1992 /*protocol*/protocol, 1993 /*ata_flags*/AP_FLAG_CHK_COND, 1994 /*tag_action*/MSG_SIMPLE_Q_TAG, 1995 /*command*/cmd, 1996 /*features*/ATA_HPA_FEAT_UNLOCK, 1997 /*lba*/0, 1998 /*sector_count*/0, 1999 /*data_ptr*/(u_int8_t*)pwd, 2000 /*dxfer_len*/sizeof(struct ata_set_max_pwd), 2001 timeout ? timeout : 1000, 2002 is48bit); 2003 2004 if (error) 2005 return (error); 2006 2007 return atahpa_proc_resp(device, ccb, is48bit, NULL); 2008 } 2009 2010 static int 2011 atahpa_freeze_lock(struct cam_device *device, int retry_count, 2012 u_int32_t timeout, union ccb *ccb, int is48bit) 2013 { 2014 int error; 2015 u_int cmd; 2016 u_int8_t protocol; 2017 2018 protocol = AP_PROTO_NON_DATA; 2019 cmd = (is48bit) ? ATA_SET_MAX_ADDRESS48 : ATA_SET_MAX_ADDRESS; 2020 2021 error = ata_do_cmd(device, 2022 ccb, 2023 retry_count, 2024 /*flags*/CAM_DIR_NONE, 2025 /*protocol*/protocol, 2026 /*ata_flags*/AP_FLAG_CHK_COND, 2027 /*tag_action*/MSG_SIMPLE_Q_TAG, 2028 /*command*/cmd, 2029 /*features*/ATA_HPA_FEAT_FREEZE, 2030 /*lba*/0, 2031 /*sector_count*/0, 2032 /*data_ptr*/NULL, 2033 /*dxfer_len*/0, 2034 timeout ? timeout : 1000, 2035 is48bit); 2036 2037 if (error) 2038 return (error); 2039 2040 return atahpa_proc_resp(device, ccb, is48bit, NULL); 2041 } 2042 2043 2044 int 2045 ata_do_identify(struct cam_device *device, int retry_count, int timeout, 2046 union ccb *ccb, struct ata_params** ident_bufp) 2047 { 2048 struct ata_params *ident_buf; 2049 struct ccb_pathinq cpi; 2050 struct ccb_getdev cgd; 2051 u_int i, error; 2052 int16_t *ptr; 2053 u_int8_t command, retry_command; 2054 2055 if (get_cpi(device, &cpi) != 0) { 2056 warnx("couldn't get CPI"); 2057 return (-1); 2058 } 2059 2060 /* Neither PROTO_ATAPI or PROTO_SATAPM are used in cpi.protocol */ 2061 if (cpi.protocol == PROTO_ATA) { 2062 if (get_cgd(device, &cgd) != 0) { 2063 warnx("couldn't get CGD"); 2064 return (-1); 2065 } 2066 2067 command = (cgd.protocol == PROTO_ATA) ? 2068 ATA_ATA_IDENTIFY : ATA_ATAPI_IDENTIFY; 2069 retry_command = 0; 2070 } else { 2071 /* We don't know which for sure so try both */ 2072 command = ATA_ATA_IDENTIFY; 2073 retry_command = ATA_ATAPI_IDENTIFY; 2074 } 2075 2076 ptr = (uint16_t *)calloc(1, sizeof(struct ata_params)); 2077 if (ptr == NULL) { 2078 warnx("can't calloc memory for identify\n"); 2079 return (1); 2080 } 2081 2082 error = ata_do_28bit_cmd(device, 2083 ccb, 2084 /*retries*/retry_count, 2085 /*flags*/CAM_DIR_IN, 2086 /*protocol*/AP_PROTO_PIO_IN, 2087 /*tag_action*/MSG_SIMPLE_Q_TAG, 2088 /*command*/command, 2089 /*features*/0, 2090 /*lba*/0, 2091 /*sector_count*/(u_int8_t)sizeof(struct ata_params), 2092 /*data_ptr*/(u_int8_t *)ptr, 2093 /*dxfer_len*/sizeof(struct ata_params), 2094 /*timeout*/timeout ? timeout : 30 * 1000, 2095 /*quiet*/1); 2096 2097 if (error != 0) { 2098 if (retry_command == 0) { 2099 free(ptr); 2100 return (1); 2101 } 2102 error = ata_do_28bit_cmd(device, 2103 ccb, 2104 /*retries*/retry_count, 2105 /*flags*/CAM_DIR_IN, 2106 /*protocol*/AP_PROTO_PIO_IN, 2107 /*tag_action*/MSG_SIMPLE_Q_TAG, 2108 /*command*/retry_command, 2109 /*features*/0, 2110 /*lba*/0, 2111 /*sector_count*/(u_int8_t) 2112 sizeof(struct ata_params), 2113 /*data_ptr*/(u_int8_t *)ptr, 2114 /*dxfer_len*/sizeof(struct ata_params), 2115 /*timeout*/timeout ? timeout : 30 * 1000, 2116 /*quiet*/0); 2117 2118 if (error != 0) { 2119 free(ptr); 2120 return (1); 2121 } 2122 } 2123 2124 error = 1; 2125 for (i = 0; i < sizeof(struct ata_params) / 2; i++) { 2126 ptr[i] = le16toh(ptr[i]); 2127 if (ptr[i] != 0) 2128 error = 0; 2129 } 2130 2131 if (arglist & CAM_ARG_VERBOSE) { 2132 fprintf(stdout, "%s%d: Raw identify data:\n", 2133 device->device_name, device->dev_unit_num); 2134 dump_data(ptr, sizeof(struct ata_params)); 2135 } 2136 2137 /* check for invalid (all zero) response */ 2138 if (error != 0) { 2139 warnx("Invalid identify response detected"); 2140 free(ptr); 2141 return (error); 2142 } 2143 2144 ident_buf = (struct ata_params *)ptr; 2145 if (strncmp(ident_buf->model, "FX", 2) && 2146 strncmp(ident_buf->model, "NEC", 3) && 2147 strncmp(ident_buf->model, "Pioneer", 7) && 2148 strncmp(ident_buf->model, "SHARP", 5)) { 2149 ata_bswap(ident_buf->model, sizeof(ident_buf->model)); 2150 ata_bswap(ident_buf->revision, sizeof(ident_buf->revision)); 2151 ata_bswap(ident_buf->serial, sizeof(ident_buf->serial)); 2152 ata_bswap(ident_buf->media_serial, sizeof(ident_buf->media_serial)); 2153 } 2154 ata_btrim(ident_buf->model, sizeof(ident_buf->model)); 2155 ata_bpack(ident_buf->model, ident_buf->model, sizeof(ident_buf->model)); 2156 ata_btrim(ident_buf->revision, sizeof(ident_buf->revision)); 2157 ata_bpack(ident_buf->revision, ident_buf->revision, sizeof(ident_buf->revision)); 2158 ata_btrim(ident_buf->serial, sizeof(ident_buf->serial)); 2159 ata_bpack(ident_buf->serial, ident_buf->serial, sizeof(ident_buf->serial)); 2160 ata_btrim(ident_buf->media_serial, sizeof(ident_buf->media_serial)); 2161 ata_bpack(ident_buf->media_serial, ident_buf->media_serial, 2162 sizeof(ident_buf->media_serial)); 2163 2164 *ident_bufp = ident_buf; 2165 2166 return (0); 2167 } 2168 2169 2170 static int 2171 ataidentify(struct cam_device *device, int retry_count, int timeout) 2172 { 2173 union ccb *ccb; 2174 struct ata_params *ident_buf; 2175 u_int64_t hpasize; 2176 2177 if ((ccb = cam_getccb(device)) == NULL) { 2178 warnx("couldn't allocate CCB"); 2179 return (1); 2180 } 2181 2182 if (ata_do_identify(device, retry_count, timeout, ccb, &ident_buf) != 0) { 2183 cam_freeccb(ccb); 2184 return (1); 2185 } 2186 2187 if (ident_buf->support.command1 & ATA_SUPPORT_PROTECTED) { 2188 if (ata_read_native_max(device, retry_count, timeout, ccb, 2189 ident_buf, &hpasize) != 0) { 2190 cam_freeccb(ccb); 2191 return (1); 2192 } 2193 } else { 2194 hpasize = 0; 2195 } 2196 2197 printf("%s%d: ", device->device_name, device->dev_unit_num); 2198 ata_print_ident(ident_buf); 2199 camxferrate(device); 2200 atacapprint(ident_buf); 2201 atahpa_print(ident_buf, hpasize, 0); 2202 2203 free(ident_buf); 2204 cam_freeccb(ccb); 2205 2206 return (0); 2207 } 2208 #endif /* MINIMALISTIC */ 2209 2210 2211 #ifndef MINIMALISTIC 2212 enum { 2213 ATA_SECURITY_ACTION_PRINT, 2214 ATA_SECURITY_ACTION_FREEZE, 2215 ATA_SECURITY_ACTION_UNLOCK, 2216 ATA_SECURITY_ACTION_DISABLE, 2217 ATA_SECURITY_ACTION_ERASE, 2218 ATA_SECURITY_ACTION_ERASE_ENHANCED, 2219 ATA_SECURITY_ACTION_SET_PASSWORD 2220 }; 2221 2222 static void 2223 atasecurity_print_time(u_int16_t tw) 2224 { 2225 2226 if (tw == 0) 2227 printf("unspecified"); 2228 else if (tw >= 255) 2229 printf("> 508 min"); 2230 else 2231 printf("%i min", 2 * tw); 2232 } 2233 2234 static u_int32_t 2235 atasecurity_erase_timeout_msecs(u_int16_t timeout) 2236 { 2237 2238 if (timeout == 0) 2239 return 2 * 3600 * 1000; /* default: two hours */ 2240 else if (timeout > 255) 2241 return (508 + 60) * 60 * 1000; /* spec says > 508 minutes */ 2242 2243 return ((2 * timeout) + 5) * 60 * 1000; /* add a 5min margin */ 2244 } 2245 2246 2247 static void 2248 atasecurity_notify(u_int8_t command, struct ata_security_password *pwd) 2249 { 2250 struct ata_cmd cmd; 2251 2252 bzero(&cmd, sizeof(cmd)); 2253 cmd.command = command; 2254 printf("Issuing %s", ata_op_string(&cmd)); 2255 2256 if (pwd != NULL) { 2257 char pass[sizeof(pwd->password)+1]; 2258 2259 /* pwd->password may not be null terminated */ 2260 pass[sizeof(pwd->password)] = '\0'; 2261 strncpy(pass, pwd->password, sizeof(pwd->password)); 2262 printf(" password='%s', user='%s'", 2263 pass, 2264 (pwd->ctrl & ATA_SECURITY_PASSWORD_MASTER) ? 2265 "master" : "user"); 2266 2267 if (command == ATA_SECURITY_SET_PASSWORD) { 2268 printf(", mode='%s'", 2269 (pwd->ctrl & ATA_SECURITY_LEVEL_MAXIMUM) ? 2270 "maximum" : "high"); 2271 } 2272 } 2273 2274 printf("\n"); 2275 } 2276 2277 static int 2278 atasecurity_freeze(struct cam_device *device, union ccb *ccb, 2279 int retry_count, u_int32_t timeout, int quiet) 2280 { 2281 2282 if (quiet == 0) 2283 atasecurity_notify(ATA_SECURITY_FREEZE_LOCK, NULL); 2284 2285 return ata_do_28bit_cmd(device, 2286 ccb, 2287 retry_count, 2288 /*flags*/CAM_DIR_NONE, 2289 /*protocol*/AP_PROTO_NON_DATA, 2290 /*tag_action*/MSG_SIMPLE_Q_TAG, 2291 /*command*/ATA_SECURITY_FREEZE_LOCK, 2292 /*features*/0, 2293 /*lba*/0, 2294 /*sector_count*/0, 2295 /*data_ptr*/NULL, 2296 /*dxfer_len*/0, 2297 /*timeout*/timeout, 2298 /*quiet*/0); 2299 } 2300 2301 static int 2302 atasecurity_unlock(struct cam_device *device, union ccb *ccb, 2303 int retry_count, u_int32_t timeout, 2304 struct ata_security_password *pwd, int quiet) 2305 { 2306 2307 if (quiet == 0) 2308 atasecurity_notify(ATA_SECURITY_UNLOCK, pwd); 2309 2310 return ata_do_28bit_cmd(device, 2311 ccb, 2312 retry_count, 2313 /*flags*/CAM_DIR_OUT, 2314 /*protocol*/AP_PROTO_PIO_OUT, 2315 /*tag_action*/MSG_SIMPLE_Q_TAG, 2316 /*command*/ATA_SECURITY_UNLOCK, 2317 /*features*/0, 2318 /*lba*/0, 2319 /*sector_count*/0, 2320 /*data_ptr*/(u_int8_t *)pwd, 2321 /*dxfer_len*/sizeof(*pwd), 2322 /*timeout*/timeout, 2323 /*quiet*/0); 2324 } 2325 2326 static int 2327 atasecurity_disable(struct cam_device *device, union ccb *ccb, 2328 int retry_count, u_int32_t timeout, 2329 struct ata_security_password *pwd, int quiet) 2330 { 2331 2332 if (quiet == 0) 2333 atasecurity_notify(ATA_SECURITY_DISABLE_PASSWORD, pwd); 2334 return ata_do_28bit_cmd(device, 2335 ccb, 2336 retry_count, 2337 /*flags*/CAM_DIR_OUT, 2338 /*protocol*/AP_PROTO_PIO_OUT, 2339 /*tag_action*/MSG_SIMPLE_Q_TAG, 2340 /*command*/ATA_SECURITY_DISABLE_PASSWORD, 2341 /*features*/0, 2342 /*lba*/0, 2343 /*sector_count*/0, 2344 /*data_ptr*/(u_int8_t *)pwd, 2345 /*dxfer_len*/sizeof(*pwd), 2346 /*timeout*/timeout, 2347 /*quiet*/0); 2348 } 2349 2350 2351 static int 2352 atasecurity_erase_confirm(struct cam_device *device, 2353 struct ata_params* ident_buf) 2354 { 2355 2356 printf("\nYou are about to ERASE ALL DATA from the following" 2357 " device:\n%s%d,%s%d: ", device->device_name, 2358 device->dev_unit_num, device->given_dev_name, 2359 device->given_unit_number); 2360 ata_print_ident(ident_buf); 2361 2362 for(;;) { 2363 char str[50]; 2364 printf("\nAre you SURE you want to ERASE ALL DATA? (yes/no) "); 2365 2366 if (fgets(str, sizeof(str), stdin) != NULL) { 2367 if (strncasecmp(str, "yes", 3) == 0) { 2368 return (1); 2369 } else if (strncasecmp(str, "no", 2) == 0) { 2370 return (0); 2371 } else { 2372 printf("Please answer \"yes\" or " 2373 "\"no\"\n"); 2374 } 2375 } 2376 } 2377 2378 /* NOTREACHED */ 2379 return (0); 2380 } 2381 2382 static int 2383 atasecurity_erase(struct cam_device *device, union ccb *ccb, 2384 int retry_count, u_int32_t timeout, 2385 u_int32_t erase_timeout, 2386 struct ata_security_password *pwd, int quiet) 2387 { 2388 int error; 2389 2390 if (quiet == 0) 2391 atasecurity_notify(ATA_SECURITY_ERASE_PREPARE, NULL); 2392 2393 error = ata_do_28bit_cmd(device, 2394 ccb, 2395 retry_count, 2396 /*flags*/CAM_DIR_NONE, 2397 /*protocol*/AP_PROTO_NON_DATA, 2398 /*tag_action*/MSG_SIMPLE_Q_TAG, 2399 /*command*/ATA_SECURITY_ERASE_PREPARE, 2400 /*features*/0, 2401 /*lba*/0, 2402 /*sector_count*/0, 2403 /*data_ptr*/NULL, 2404 /*dxfer_len*/0, 2405 /*timeout*/timeout, 2406 /*quiet*/0); 2407 2408 if (error != 0) 2409 return error; 2410 2411 if (quiet == 0) 2412 atasecurity_notify(ATA_SECURITY_ERASE_UNIT, pwd); 2413 2414 error = ata_do_28bit_cmd(device, 2415 ccb, 2416 retry_count, 2417 /*flags*/CAM_DIR_OUT, 2418 /*protocol*/AP_PROTO_PIO_OUT, 2419 /*tag_action*/MSG_SIMPLE_Q_TAG, 2420 /*command*/ATA_SECURITY_ERASE_UNIT, 2421 /*features*/0, 2422 /*lba*/0, 2423 /*sector_count*/0, 2424 /*data_ptr*/(u_int8_t *)pwd, 2425 /*dxfer_len*/sizeof(*pwd), 2426 /*timeout*/erase_timeout, 2427 /*quiet*/0); 2428 2429 if (error == 0 && quiet == 0) 2430 printf("\nErase Complete\n"); 2431 2432 return error; 2433 } 2434 2435 static int 2436 atasecurity_set_password(struct cam_device *device, union ccb *ccb, 2437 int retry_count, u_int32_t timeout, 2438 struct ata_security_password *pwd, int quiet) 2439 { 2440 2441 if (quiet == 0) 2442 atasecurity_notify(ATA_SECURITY_SET_PASSWORD, pwd); 2443 2444 return ata_do_28bit_cmd(device, 2445 ccb, 2446 retry_count, 2447 /*flags*/CAM_DIR_OUT, 2448 /*protocol*/AP_PROTO_PIO_OUT, 2449 /*tag_action*/MSG_SIMPLE_Q_TAG, 2450 /*command*/ATA_SECURITY_SET_PASSWORD, 2451 /*features*/0, 2452 /*lba*/0, 2453 /*sector_count*/0, 2454 /*data_ptr*/(u_int8_t *)pwd, 2455 /*dxfer_len*/sizeof(*pwd), 2456 /*timeout*/timeout, 2457 /*quiet*/0); 2458 } 2459 2460 static void 2461 atasecurity_print(struct ata_params *parm) 2462 { 2463 2464 printf("\nSecurity Option Value\n"); 2465 if (arglist & CAM_ARG_VERBOSE) { 2466 printf("status %04x\n", 2467 parm->security_status); 2468 } 2469 printf("supported %s\n", 2470 parm->security_status & ATA_SECURITY_SUPPORTED ? "yes" : "no"); 2471 if (!(parm->security_status & ATA_SECURITY_SUPPORTED)) 2472 return; 2473 printf("enabled %s\n", 2474 parm->security_status & ATA_SECURITY_ENABLED ? "yes" : "no"); 2475 printf("drive locked %s\n", 2476 parm->security_status & ATA_SECURITY_LOCKED ? "yes" : "no"); 2477 printf("security config frozen %s\n", 2478 parm->security_status & ATA_SECURITY_FROZEN ? "yes" : "no"); 2479 printf("count expired %s\n", 2480 parm->security_status & ATA_SECURITY_COUNT_EXP ? "yes" : "no"); 2481 printf("security level %s\n", 2482 parm->security_status & ATA_SECURITY_LEVEL ? "maximum" : "high"); 2483 printf("enhanced erase supported %s\n", 2484 parm->security_status & ATA_SECURITY_ENH_SUPP ? "yes" : "no"); 2485 printf("erase time "); 2486 atasecurity_print_time(parm->erase_time); 2487 printf("\n"); 2488 printf("enhanced erase time "); 2489 atasecurity_print_time(parm->enhanced_erase_time); 2490 printf("\n"); 2491 printf("master password rev %04x%s\n", 2492 parm->master_passwd_revision, 2493 parm->master_passwd_revision == 0x0000 || 2494 parm->master_passwd_revision == 0xFFFF ? " (unsupported)" : ""); 2495 } 2496 2497 /* 2498 * Validates and copies the password in optarg to the passed buffer. 2499 * If the password in optarg is the same length as the buffer then 2500 * the data will still be copied but no null termination will occur. 2501 */ 2502 static int 2503 ata_getpwd(u_int8_t *passwd, int max, char opt) 2504 { 2505 int len; 2506 2507 len = strlen(optarg); 2508 if (len > max) { 2509 warnx("-%c password is too long", opt); 2510 return (1); 2511 } else if (len == 0) { 2512 warnx("-%c password is missing", opt); 2513 return (1); 2514 } else if (optarg[0] == '-'){ 2515 warnx("-%c password starts with '-' (generic arg?)", opt); 2516 return (1); 2517 } else if (strlen(passwd) != 0 && strcmp(passwd, optarg) != 0) { 2518 warnx("-%c password conflicts with existing password from -%c", 2519 opt, pwd_opt); 2520 return (1); 2521 } 2522 2523 /* Callers pass in a buffer which does NOT need to be terminated */ 2524 strncpy(passwd, optarg, max); 2525 pwd_opt = opt; 2526 2527 return (0); 2528 } 2529 2530 enum { 2531 ATA_HPA_ACTION_PRINT, 2532 ATA_HPA_ACTION_SET_MAX, 2533 ATA_HPA_ACTION_SET_PWD, 2534 ATA_HPA_ACTION_LOCK, 2535 ATA_HPA_ACTION_UNLOCK, 2536 ATA_HPA_ACTION_FREEZE_LOCK 2537 }; 2538 2539 static int 2540 atahpa_set_confirm(struct cam_device *device, struct ata_params* ident_buf, 2541 u_int64_t maxsize, int persist) 2542 { 2543 printf("\nYou are about to configure HPA to limit the user accessible\n" 2544 "sectors to %ju %s on the device:\n%s%d,%s%d: ", maxsize, 2545 persist ? "persistently" : "temporarily", 2546 device->device_name, device->dev_unit_num, 2547 device->given_dev_name, device->given_unit_number); 2548 ata_print_ident(ident_buf); 2549 2550 for(;;) { 2551 char str[50]; 2552 printf("\nAre you SURE you want to configure HPA? (yes/no) "); 2553 2554 if (NULL != fgets(str, sizeof(str), stdin)) { 2555 if (0 == strncasecmp(str, "yes", 3)) { 2556 return (1); 2557 } else if (0 == strncasecmp(str, "no", 2)) { 2558 return (0); 2559 } else { 2560 printf("Please answer \"yes\" or " 2561 "\"no\"\n"); 2562 } 2563 } 2564 } 2565 2566 /* NOTREACHED */ 2567 return (0); 2568 } 2569 2570 static int 2571 atahpa(struct cam_device *device, int retry_count, int timeout, 2572 int argc, char **argv, char *combinedopt) 2573 { 2574 union ccb *ccb; 2575 struct ata_params *ident_buf; 2576 struct ccb_getdev cgd; 2577 struct ata_set_max_pwd pwd; 2578 int error, confirm, quiet, c, action, actions, persist; 2579 int security, is48bit, pwdsize; 2580 u_int64_t hpasize, maxsize; 2581 2582 actions = 0; 2583 confirm = 0; 2584 quiet = 0; 2585 maxsize = 0; 2586 persist = 0; 2587 security = 0; 2588 2589 memset(&pwd, 0, sizeof(pwd)); 2590 2591 /* default action is to print hpa information */ 2592 action = ATA_HPA_ACTION_PRINT; 2593 pwdsize = sizeof(pwd.password); 2594 2595 while ((c = getopt(argc, argv, combinedopt)) != -1) { 2596 switch(c){ 2597 case 's': 2598 action = ATA_HPA_ACTION_SET_MAX; 2599 maxsize = strtoumax(optarg, NULL, 0); 2600 actions++; 2601 break; 2602 2603 case 'p': 2604 if (ata_getpwd(pwd.password, pwdsize, c) != 0) 2605 return (1); 2606 action = ATA_HPA_ACTION_SET_PWD; 2607 security = 1; 2608 actions++; 2609 break; 2610 2611 case 'l': 2612 action = ATA_HPA_ACTION_LOCK; 2613 security = 1; 2614 actions++; 2615 break; 2616 2617 case 'U': 2618 if (ata_getpwd(pwd.password, pwdsize, c) != 0) 2619 return (1); 2620 action = ATA_HPA_ACTION_UNLOCK; 2621 security = 1; 2622 actions++; 2623 break; 2624 2625 case 'f': 2626 action = ATA_HPA_ACTION_FREEZE_LOCK; 2627 security = 1; 2628 actions++; 2629 break; 2630 2631 case 'P': 2632 persist = 1; 2633 break; 2634 2635 case 'y': 2636 confirm++; 2637 break; 2638 2639 case 'q': 2640 quiet++; 2641 break; 2642 } 2643 } 2644 2645 if (actions > 1) { 2646 warnx("too many hpa actions specified"); 2647 return (1); 2648 } 2649 2650 if (get_cgd(device, &cgd) != 0) { 2651 warnx("couldn't get CGD"); 2652 return (1); 2653 } 2654 2655 ccb = cam_getccb(device); 2656 if (ccb == NULL) { 2657 warnx("couldn't allocate CCB"); 2658 return (1); 2659 } 2660 2661 error = ata_do_identify(device, retry_count, timeout, ccb, &ident_buf); 2662 if (error != 0) { 2663 cam_freeccb(ccb); 2664 return (1); 2665 } 2666 2667 if (quiet == 0) { 2668 printf("%s%d: ", device->device_name, device->dev_unit_num); 2669 ata_print_ident(ident_buf); 2670 camxferrate(device); 2671 } 2672 2673 if (action == ATA_HPA_ACTION_PRINT) { 2674 error = ata_read_native_max(device, retry_count, timeout, ccb, 2675 ident_buf, &hpasize); 2676 if (error == 0) 2677 atahpa_print(ident_buf, hpasize, 1); 2678 2679 cam_freeccb(ccb); 2680 free(ident_buf); 2681 return (error); 2682 } 2683 2684 if (!(ident_buf->support.command1 & ATA_SUPPORT_PROTECTED)) { 2685 warnx("HPA is not supported by this device"); 2686 cam_freeccb(ccb); 2687 free(ident_buf); 2688 return (1); 2689 } 2690 2691 if (security && !(ident_buf->support.command1 & ATA_SUPPORT_MAXSECURITY)) { 2692 warnx("HPA Security is not supported by this device"); 2693 cam_freeccb(ccb); 2694 free(ident_buf); 2695 return (1); 2696 } 2697 2698 is48bit = ident_buf->support.command2 & ATA_SUPPORT_ADDRESS48; 2699 2700 /* 2701 * The ATA spec requires: 2702 * 1. Read native max addr is called directly before set max addr 2703 * 2. Read native max addr is NOT called before any other set max call 2704 */ 2705 switch(action) { 2706 case ATA_HPA_ACTION_SET_MAX: 2707 if (confirm == 0 && 2708 atahpa_set_confirm(device, ident_buf, maxsize, 2709 persist) == 0) { 2710 cam_freeccb(ccb); 2711 free(ident_buf); 2712 return (1); 2713 } 2714 2715 error = ata_read_native_max(device, retry_count, timeout, 2716 ccb, ident_buf, &hpasize); 2717 if (error == 0) { 2718 error = atahpa_set_max(device, retry_count, timeout, 2719 ccb, is48bit, maxsize, persist); 2720 if (error == 0) { 2721 /* redo identify to get new lba values */ 2722 error = ata_do_identify(device, retry_count, 2723 timeout, ccb, 2724 &ident_buf); 2725 atahpa_print(ident_buf, hpasize, 1); 2726 } 2727 } 2728 break; 2729 2730 case ATA_HPA_ACTION_SET_PWD: 2731 error = atahpa_password(device, retry_count, timeout, 2732 ccb, is48bit, &pwd); 2733 if (error == 0) 2734 printf("HPA password has been set\n"); 2735 break; 2736 2737 case ATA_HPA_ACTION_LOCK: 2738 error = atahpa_lock(device, retry_count, timeout, 2739 ccb, is48bit); 2740 if (error == 0) 2741 printf("HPA has been locked\n"); 2742 break; 2743 2744 case ATA_HPA_ACTION_UNLOCK: 2745 error = atahpa_unlock(device, retry_count, timeout, 2746 ccb, is48bit, &pwd); 2747 if (error == 0) 2748 printf("HPA has been unlocked\n"); 2749 break; 2750 2751 case ATA_HPA_ACTION_FREEZE_LOCK: 2752 error = atahpa_freeze_lock(device, retry_count, timeout, 2753 ccb, is48bit); 2754 if (error == 0) 2755 printf("HPA has been frozen\n"); 2756 break; 2757 2758 default: 2759 errx(1, "Option currently not supported"); 2760 } 2761 2762 cam_freeccb(ccb); 2763 free(ident_buf); 2764 2765 return (error); 2766 } 2767 2768 static int 2769 atasecurity(struct cam_device *device, int retry_count, int timeout, 2770 int argc, char **argv, char *combinedopt) 2771 { 2772 union ccb *ccb; 2773 struct ata_params *ident_buf; 2774 int error, confirm, quiet, c, action, actions, setpwd; 2775 int security_enabled, erase_timeout, pwdsize; 2776 struct ata_security_password pwd; 2777 2778 actions = 0; 2779 setpwd = 0; 2780 erase_timeout = 0; 2781 confirm = 0; 2782 quiet = 0; 2783 2784 memset(&pwd, 0, sizeof(pwd)); 2785 2786 /* default action is to print security information */ 2787 action = ATA_SECURITY_ACTION_PRINT; 2788 2789 /* user is master by default as its safer that way */ 2790 pwd.ctrl |= ATA_SECURITY_PASSWORD_MASTER; 2791 pwdsize = sizeof(pwd.password); 2792 2793 while ((c = getopt(argc, argv, combinedopt)) != -1) { 2794 switch(c){ 2795 case 'f': 2796 action = ATA_SECURITY_ACTION_FREEZE; 2797 actions++; 2798 break; 2799 2800 case 'U': 2801 if (strcasecmp(optarg, "user") == 0) { 2802 pwd.ctrl |= ATA_SECURITY_PASSWORD_USER; 2803 pwd.ctrl &= ~ATA_SECURITY_PASSWORD_MASTER; 2804 } else if (strcasecmp(optarg, "master") == 0) { 2805 pwd.ctrl |= ATA_SECURITY_PASSWORD_MASTER; 2806 pwd.ctrl &= ~ATA_SECURITY_PASSWORD_USER; 2807 } else { 2808 warnx("-U argument '%s' is invalid (must be " 2809 "'user' or 'master')", optarg); 2810 return (1); 2811 } 2812 break; 2813 2814 case 'l': 2815 if (strcasecmp(optarg, "high") == 0) { 2816 pwd.ctrl |= ATA_SECURITY_LEVEL_HIGH; 2817 pwd.ctrl &= ~ATA_SECURITY_LEVEL_MAXIMUM; 2818 } else if (strcasecmp(optarg, "maximum") == 0) { 2819 pwd.ctrl |= ATA_SECURITY_LEVEL_MAXIMUM; 2820 pwd.ctrl &= ~ATA_SECURITY_LEVEL_HIGH; 2821 } else { 2822 warnx("-l argument '%s' is unknown (must be " 2823 "'high' or 'maximum')", optarg); 2824 return (1); 2825 } 2826 break; 2827 2828 case 'k': 2829 if (ata_getpwd(pwd.password, pwdsize, c) != 0) 2830 return (1); 2831 action = ATA_SECURITY_ACTION_UNLOCK; 2832 actions++; 2833 break; 2834 2835 case 'd': 2836 if (ata_getpwd(pwd.password, pwdsize, c) != 0) 2837 return (1); 2838 action = ATA_SECURITY_ACTION_DISABLE; 2839 actions++; 2840 break; 2841 2842 case 'e': 2843 if (ata_getpwd(pwd.password, pwdsize, c) != 0) 2844 return (1); 2845 action = ATA_SECURITY_ACTION_ERASE; 2846 actions++; 2847 break; 2848 2849 case 'h': 2850 if (ata_getpwd(pwd.password, pwdsize, c) != 0) 2851 return (1); 2852 pwd.ctrl |= ATA_SECURITY_ERASE_ENHANCED; 2853 action = ATA_SECURITY_ACTION_ERASE_ENHANCED; 2854 actions++; 2855 break; 2856 2857 case 's': 2858 if (ata_getpwd(pwd.password, pwdsize, c) != 0) 2859 return (1); 2860 setpwd = 1; 2861 if (action == ATA_SECURITY_ACTION_PRINT) 2862 action = ATA_SECURITY_ACTION_SET_PASSWORD; 2863 /* 2864 * Don't increment action as this can be combined 2865 * with other actions. 2866 */ 2867 break; 2868 2869 case 'y': 2870 confirm++; 2871 break; 2872 2873 case 'q': 2874 quiet++; 2875 break; 2876 2877 case 'T': 2878 erase_timeout = atoi(optarg) * 1000; 2879 break; 2880 } 2881 } 2882 2883 if (actions > 1) { 2884 warnx("too many security actions specified"); 2885 return (1); 2886 } 2887 2888 if ((ccb = cam_getccb(device)) == NULL) { 2889 warnx("couldn't allocate CCB"); 2890 return (1); 2891 } 2892 2893 error = ata_do_identify(device, retry_count, timeout, ccb, &ident_buf); 2894 if (error != 0) { 2895 cam_freeccb(ccb); 2896 return (1); 2897 } 2898 2899 if (quiet == 0) { 2900 printf("%s%d: ", device->device_name, device->dev_unit_num); 2901 ata_print_ident(ident_buf); 2902 camxferrate(device); 2903 } 2904 2905 if (action == ATA_SECURITY_ACTION_PRINT) { 2906 atasecurity_print(ident_buf); 2907 free(ident_buf); 2908 cam_freeccb(ccb); 2909 return (0); 2910 } 2911 2912 if ((ident_buf->support.command1 & ATA_SUPPORT_SECURITY) == 0) { 2913 warnx("Security not supported"); 2914 free(ident_buf); 2915 cam_freeccb(ccb); 2916 return (1); 2917 } 2918 2919 /* default timeout 15 seconds the same as linux hdparm */ 2920 timeout = timeout ? timeout : 15 * 1000; 2921 2922 security_enabled = ident_buf->security_status & ATA_SECURITY_ENABLED; 2923 2924 /* first set the password if requested */ 2925 if (setpwd == 1) { 2926 /* confirm we can erase before setting the password if erasing */ 2927 if (confirm == 0 && 2928 (action == ATA_SECURITY_ACTION_ERASE_ENHANCED || 2929 action == ATA_SECURITY_ACTION_ERASE) && 2930 atasecurity_erase_confirm(device, ident_buf) == 0) { 2931 cam_freeccb(ccb); 2932 free(ident_buf); 2933 return (error); 2934 } 2935 2936 if (pwd.ctrl & ATA_SECURITY_PASSWORD_MASTER) { 2937 pwd.revision = ident_buf->master_passwd_revision; 2938 if (pwd.revision != 0 && pwd.revision != 0xfff && 2939 --pwd.revision == 0) { 2940 pwd.revision = 0xfffe; 2941 } 2942 } 2943 error = atasecurity_set_password(device, ccb, retry_count, 2944 timeout, &pwd, quiet); 2945 if (error != 0) { 2946 cam_freeccb(ccb); 2947 free(ident_buf); 2948 return (error); 2949 } 2950 security_enabled = 1; 2951 } 2952 2953 switch(action) { 2954 case ATA_SECURITY_ACTION_FREEZE: 2955 error = atasecurity_freeze(device, ccb, retry_count, 2956 timeout, quiet); 2957 break; 2958 2959 case ATA_SECURITY_ACTION_UNLOCK: 2960 if (security_enabled) { 2961 if (ident_buf->security_status & ATA_SECURITY_LOCKED) { 2962 error = atasecurity_unlock(device, ccb, 2963 retry_count, timeout, &pwd, quiet); 2964 } else { 2965 warnx("Can't unlock, drive is not locked"); 2966 error = 1; 2967 } 2968 } else { 2969 warnx("Can't unlock, security is disabled"); 2970 error = 1; 2971 } 2972 break; 2973 2974 case ATA_SECURITY_ACTION_DISABLE: 2975 if (security_enabled) { 2976 /* First unlock the drive if its locked */ 2977 if (ident_buf->security_status & ATA_SECURITY_LOCKED) { 2978 error = atasecurity_unlock(device, ccb, 2979 retry_count, 2980 timeout, 2981 &pwd, 2982 quiet); 2983 } 2984 2985 if (error == 0) { 2986 error = atasecurity_disable(device, 2987 ccb, 2988 retry_count, 2989 timeout, 2990 &pwd, 2991 quiet); 2992 } 2993 } else { 2994 warnx("Can't disable security (already disabled)"); 2995 error = 1; 2996 } 2997 break; 2998 2999 case ATA_SECURITY_ACTION_ERASE: 3000 if (security_enabled) { 3001 if (erase_timeout == 0) { 3002 erase_timeout = atasecurity_erase_timeout_msecs( 3003 ident_buf->erase_time); 3004 } 3005 3006 error = atasecurity_erase(device, ccb, retry_count, 3007 timeout, erase_timeout, &pwd, 3008 quiet); 3009 } else { 3010 warnx("Can't secure erase (security is disabled)"); 3011 error = 1; 3012 } 3013 break; 3014 3015 case ATA_SECURITY_ACTION_ERASE_ENHANCED: 3016 if (security_enabled) { 3017 if (ident_buf->security_status & ATA_SECURITY_ENH_SUPP) { 3018 if (erase_timeout == 0) { 3019 erase_timeout = 3020 atasecurity_erase_timeout_msecs( 3021 ident_buf->enhanced_erase_time); 3022 } 3023 3024 error = atasecurity_erase(device, ccb, 3025 retry_count, timeout, 3026 erase_timeout, &pwd, 3027 quiet); 3028 } else { 3029 warnx("Enhanced erase is not supported"); 3030 error = 1; 3031 } 3032 } else { 3033 warnx("Can't secure erase (enhanced), " 3034 "(security is disabled)"); 3035 error = 1; 3036 } 3037 break; 3038 } 3039 3040 cam_freeccb(ccb); 3041 free(ident_buf); 3042 3043 return (error); 3044 } 3045 #endif /* MINIMALISTIC */ 3046 3047 /* 3048 * Parse out a bus, or a bus, target and lun in the following 3049 * format: 3050 * bus 3051 * bus:target 3052 * bus:target:lun 3053 * 3054 * Returns the number of parsed components, or 0. 3055 */ 3056 static int 3057 parse_btl(char *tstr, path_id_t *bus, target_id_t *target, lun_id_t *lun, 3058 cam_argmask *arglst) 3059 { 3060 char *tmpstr; 3061 int convs = 0; 3062 3063 while (isspace(*tstr) && (*tstr != '\0')) 3064 tstr++; 3065 3066 tmpstr = (char *)strtok(tstr, ":"); 3067 if ((tmpstr != NULL) && (*tmpstr != '\0')) { 3068 *bus = strtol(tmpstr, NULL, 0); 3069 *arglst |= CAM_ARG_BUS; 3070 convs++; 3071 tmpstr = (char *)strtok(NULL, ":"); 3072 if ((tmpstr != NULL) && (*tmpstr != '\0')) { 3073 *target = strtol(tmpstr, NULL, 0); 3074 *arglst |= CAM_ARG_TARGET; 3075 convs++; 3076 tmpstr = (char *)strtok(NULL, ":"); 3077 if ((tmpstr != NULL) && (*tmpstr != '\0')) { 3078 *lun = strtol(tmpstr, NULL, 0); 3079 *arglst |= CAM_ARG_LUN; 3080 convs++; 3081 } 3082 } 3083 } 3084 3085 return convs; 3086 } 3087 3088 static int 3089 dorescan_or_reset(int argc, char **argv, int rescan) 3090 { 3091 static const char must[] = 3092 "you must specify \"all\", a bus, or a bus:target:lun to %s"; 3093 int rv, error = 0; 3094 path_id_t bus = CAM_BUS_WILDCARD; 3095 target_id_t target = CAM_TARGET_WILDCARD; 3096 lun_id_t lun = CAM_LUN_WILDCARD; 3097 char *tstr; 3098 3099 if (argc < 3) { 3100 warnx(must, rescan? "rescan" : "reset"); 3101 return(1); 3102 } 3103 3104 tstr = argv[optind]; 3105 while (isspace(*tstr) && (*tstr != '\0')) 3106 tstr++; 3107 if (strncasecmp(tstr, "all", strlen("all")) == 0) 3108 arglist |= CAM_ARG_BUS; 3109 else { 3110 rv = parse_btl(argv[optind], &bus, &target, &lun, &arglist); 3111 if (rv != 1 && rv != 3) { 3112 warnx(must, rescan? "rescan" : "reset"); 3113 return(1); 3114 } 3115 } 3116 3117 if ((arglist & CAM_ARG_BUS) 3118 && (arglist & CAM_ARG_TARGET) 3119 && (arglist & CAM_ARG_LUN)) 3120 error = scanlun_or_reset_dev(bus, target, lun, rescan); 3121 else 3122 error = rescan_or_reset_bus(bus, rescan); 3123 3124 return(error); 3125 } 3126 3127 static int 3128 rescan_or_reset_bus(path_id_t bus, int rescan) 3129 { 3130 union ccb ccb, matchccb; 3131 int fd, retval; 3132 int bufsize; 3133 3134 retval = 0; 3135 3136 if ((fd = open(XPT_DEVICE, O_RDWR)) < 0) { 3137 warnx("error opening transport layer device %s", XPT_DEVICE); 3138 warn("%s", XPT_DEVICE); 3139 return(1); 3140 } 3141 3142 if (bus != CAM_BUS_WILDCARD) { 3143 ccb.ccb_h.func_code = rescan ? XPT_SCAN_BUS : XPT_RESET_BUS; 3144 ccb.ccb_h.path_id = bus; 3145 ccb.ccb_h.target_id = CAM_TARGET_WILDCARD; 3146 ccb.ccb_h.target_lun = CAM_LUN_WILDCARD; 3147 ccb.crcn.flags = CAM_FLAG_NONE; 3148 3149 /* run this at a low priority */ 3150 ccb.ccb_h.pinfo.priority = 5; 3151 3152 if (ioctl(fd, CAMIOCOMMAND, &ccb) == -1) { 3153 warn("CAMIOCOMMAND ioctl failed"); 3154 close(fd); 3155 return(1); 3156 } 3157 3158 if ((ccb.ccb_h.status & CAM_STATUS_MASK) == CAM_REQ_CMP) { 3159 fprintf(stdout, "%s of bus %d was successful\n", 3160 rescan ? "Re-scan" : "Reset", bus); 3161 } else { 3162 fprintf(stdout, "%s of bus %d returned error %#x\n", 3163 rescan ? "Re-scan" : "Reset", bus, 3164 ccb.ccb_h.status & CAM_STATUS_MASK); 3165 retval = 1; 3166 } 3167 3168 close(fd); 3169 return(retval); 3170 3171 } 3172 3173 3174 /* 3175 * The right way to handle this is to modify the xpt so that it can 3176 * handle a wildcarded bus in a rescan or reset CCB. At the moment 3177 * that isn't implemented, so instead we enumerate the busses and 3178 * send the rescan or reset to those busses in the case where the 3179 * given bus is -1 (wildcard). We don't send a rescan or reset 3180 * to the xpt bus; sending a rescan to the xpt bus is effectively a 3181 * no-op, sending a rescan to the xpt bus would result in a status of 3182 * CAM_REQ_INVALID. 3183 */ 3184 CCB_CLEAR_ALL_EXCEPT_HDR(&matchccb.cdm); 3185 matchccb.ccb_h.func_code = XPT_DEV_MATCH; 3186 matchccb.ccb_h.path_id = CAM_BUS_WILDCARD; 3187 bufsize = sizeof(struct dev_match_result) * 20; 3188 matchccb.cdm.match_buf_len = bufsize; 3189 matchccb.cdm.matches=(struct dev_match_result *)malloc(bufsize); 3190 if (matchccb.cdm.matches == NULL) { 3191 warnx("can't malloc memory for matches"); 3192 retval = 1; 3193 goto bailout; 3194 } 3195 matchccb.cdm.num_matches = 0; 3196 3197 matchccb.cdm.num_patterns = 1; 3198 matchccb.cdm.pattern_buf_len = sizeof(struct dev_match_pattern); 3199 3200 matchccb.cdm.patterns = (struct dev_match_pattern *)malloc( 3201 matchccb.cdm.pattern_buf_len); 3202 if (matchccb.cdm.patterns == NULL) { 3203 warnx("can't malloc memory for patterns"); 3204 retval = 1; 3205 goto bailout; 3206 } 3207 matchccb.cdm.patterns[0].type = DEV_MATCH_BUS; 3208 matchccb.cdm.patterns[0].pattern.bus_pattern.flags = BUS_MATCH_ANY; 3209 3210 do { 3211 unsigned int i; 3212 3213 if (ioctl(fd, CAMIOCOMMAND, &matchccb) == -1) { 3214 warn("CAMIOCOMMAND ioctl failed"); 3215 retval = 1; 3216 goto bailout; 3217 } 3218 3219 if ((matchccb.ccb_h.status != CAM_REQ_CMP) 3220 || ((matchccb.cdm.status != CAM_DEV_MATCH_LAST) 3221 && (matchccb.cdm.status != CAM_DEV_MATCH_MORE))) { 3222 warnx("got CAM error %#x, CDM error %d\n", 3223 matchccb.ccb_h.status, matchccb.cdm.status); 3224 retval = 1; 3225 goto bailout; 3226 } 3227 3228 for (i = 0; i < matchccb.cdm.num_matches; i++) { 3229 struct bus_match_result *bus_result; 3230 3231 /* This shouldn't happen. */ 3232 if (matchccb.cdm.matches[i].type != DEV_MATCH_BUS) 3233 continue; 3234 3235 bus_result = &matchccb.cdm.matches[i].result.bus_result; 3236 3237 /* 3238 * We don't want to rescan or reset the xpt bus. 3239 * See above. 3240 */ 3241 if (bus_result->path_id == CAM_XPT_PATH_ID) 3242 continue; 3243 3244 ccb.ccb_h.func_code = rescan ? XPT_SCAN_BUS : 3245 XPT_RESET_BUS; 3246 ccb.ccb_h.path_id = bus_result->path_id; 3247 ccb.ccb_h.target_id = CAM_TARGET_WILDCARD; 3248 ccb.ccb_h.target_lun = CAM_LUN_WILDCARD; 3249 ccb.crcn.flags = CAM_FLAG_NONE; 3250 3251 /* run this at a low priority */ 3252 ccb.ccb_h.pinfo.priority = 5; 3253 3254 if (ioctl(fd, CAMIOCOMMAND, &ccb) == -1) { 3255 warn("CAMIOCOMMAND ioctl failed"); 3256 retval = 1; 3257 goto bailout; 3258 } 3259 3260 if ((ccb.ccb_h.status & CAM_STATUS_MASK) ==CAM_REQ_CMP){ 3261 fprintf(stdout, "%s of bus %d was successful\n", 3262 rescan? "Re-scan" : "Reset", 3263 bus_result->path_id); 3264 } else { 3265 /* 3266 * Don't bail out just yet, maybe the other 3267 * rescan or reset commands will complete 3268 * successfully. 3269 */ 3270 fprintf(stderr, "%s of bus %d returned error " 3271 "%#x\n", rescan? "Re-scan" : "Reset", 3272 bus_result->path_id, 3273 ccb.ccb_h.status & CAM_STATUS_MASK); 3274 retval = 1; 3275 } 3276 } 3277 } while ((matchccb.ccb_h.status == CAM_REQ_CMP) 3278 && (matchccb.cdm.status == CAM_DEV_MATCH_MORE)); 3279 3280 bailout: 3281 3282 if (fd != -1) 3283 close(fd); 3284 3285 if (matchccb.cdm.patterns != NULL) 3286 free(matchccb.cdm.patterns); 3287 if (matchccb.cdm.matches != NULL) 3288 free(matchccb.cdm.matches); 3289 3290 return(retval); 3291 } 3292 3293 static int 3294 scanlun_or_reset_dev(path_id_t bus, target_id_t target, lun_id_t lun, int scan) 3295 { 3296 union ccb ccb; 3297 struct cam_device *device; 3298 int fd; 3299 3300 device = NULL; 3301 3302 if (bus == CAM_BUS_WILDCARD) { 3303 warnx("invalid bus number %d", bus); 3304 return(1); 3305 } 3306 3307 if (target == CAM_TARGET_WILDCARD) { 3308 warnx("invalid target number %d", target); 3309 return(1); 3310 } 3311 3312 if (lun == CAM_LUN_WILDCARD) { 3313 warnx("invalid lun number %jx", (uintmax_t)lun); 3314 return(1); 3315 } 3316 3317 fd = -1; 3318 3319 bzero(&ccb, sizeof(union ccb)); 3320 3321 if (scan) { 3322 if ((fd = open(XPT_DEVICE, O_RDWR)) < 0) { 3323 warnx("error opening transport layer device %s\n", 3324 XPT_DEVICE); 3325 warn("%s", XPT_DEVICE); 3326 return(1); 3327 } 3328 } else { 3329 device = cam_open_btl(bus, target, lun, O_RDWR, NULL); 3330 if (device == NULL) { 3331 warnx("%s", cam_errbuf); 3332 return(1); 3333 } 3334 } 3335 3336 ccb.ccb_h.func_code = (scan)? XPT_SCAN_LUN : XPT_RESET_DEV; 3337 ccb.ccb_h.path_id = bus; 3338 ccb.ccb_h.target_id = target; 3339 ccb.ccb_h.target_lun = lun; 3340 ccb.ccb_h.timeout = 5000; 3341 ccb.crcn.flags = CAM_FLAG_NONE; 3342 3343 /* run this at a low priority */ 3344 ccb.ccb_h.pinfo.priority = 5; 3345 3346 if (scan) { 3347 if (ioctl(fd, CAMIOCOMMAND, &ccb) < 0) { 3348 warn("CAMIOCOMMAND ioctl failed"); 3349 close(fd); 3350 return(1); 3351 } 3352 } else { 3353 if (cam_send_ccb(device, &ccb) < 0) { 3354 warn("error sending XPT_RESET_DEV CCB"); 3355 cam_close_device(device); 3356 return(1); 3357 } 3358 } 3359 3360 if (scan) 3361 close(fd); 3362 else 3363 cam_close_device(device); 3364 3365 /* 3366 * An error code of CAM_BDR_SENT is normal for a BDR request. 3367 */ 3368 if (((ccb.ccb_h.status & CAM_STATUS_MASK) == CAM_REQ_CMP) 3369 || ((!scan) 3370 && ((ccb.ccb_h.status & CAM_STATUS_MASK) == CAM_BDR_SENT))) { 3371 fprintf(stdout, "%s of %d:%d:%jx was successful\n", 3372 scan? "Re-scan" : "Reset", bus, target, (uintmax_t)lun); 3373 return(0); 3374 } else { 3375 fprintf(stdout, "%s of %d:%d:%jx returned error %#x\n", 3376 scan? "Re-scan" : "Reset", bus, target, (uintmax_t)lun, 3377 ccb.ccb_h.status & CAM_STATUS_MASK); 3378 return(1); 3379 } 3380 } 3381 3382 #ifndef MINIMALISTIC 3383 3384 static struct scsi_nv defect_list_type_map[] = { 3385 { "block", SRDD10_BLOCK_FORMAT }, 3386 { "extbfi", SRDD10_EXT_BFI_FORMAT }, 3387 { "extphys", SRDD10_EXT_PHYS_FORMAT }, 3388 { "longblock", SRDD10_LONG_BLOCK_FORMAT }, 3389 { "bfi", SRDD10_BYTES_FROM_INDEX_FORMAT }, 3390 { "phys", SRDD10_PHYSICAL_SECTOR_FORMAT } 3391 }; 3392 3393 static int 3394 readdefects(struct cam_device *device, int argc, char **argv, 3395 char *combinedopt, int retry_count, int timeout) 3396 { 3397 union ccb *ccb = NULL; 3398 struct scsi_read_defect_data_hdr_10 *hdr10 = NULL; 3399 struct scsi_read_defect_data_hdr_12 *hdr12 = NULL; 3400 size_t hdr_size = 0, entry_size = 0; 3401 int use_12byte = 0; 3402 int hex_format = 0; 3403 u_int8_t *defect_list = NULL; 3404 u_int8_t list_format = 0; 3405 int list_type_set = 0; 3406 u_int32_t dlist_length = 0; 3407 u_int32_t returned_length = 0, valid_len = 0; 3408 u_int32_t num_returned = 0, num_valid = 0; 3409 u_int32_t max_possible_size = 0, hdr_max = 0; 3410 u_int32_t starting_offset = 0; 3411 u_int8_t returned_format, returned_type; 3412 unsigned int i; 3413 int summary = 0, quiet = 0; 3414 int c, error = 0; 3415 int lists_specified = 0; 3416 int get_length = 1, first_pass = 1; 3417 int mads = 0; 3418 3419 while ((c = getopt(argc, argv, combinedopt)) != -1) { 3420 switch(c){ 3421 case 'f': 3422 { 3423 scsi_nv_status status; 3424 int entry_num = 0; 3425 3426 status = scsi_get_nv(defect_list_type_map, 3427 sizeof(defect_list_type_map) / 3428 sizeof(defect_list_type_map[0]), optarg, 3429 &entry_num, SCSI_NV_FLAG_IG_CASE); 3430 3431 if (status == SCSI_NV_FOUND) { 3432 list_format = defect_list_type_map[ 3433 entry_num].value; 3434 list_type_set = 1; 3435 } else { 3436 warnx("%s: %s %s option %s", __func__, 3437 (status == SCSI_NV_AMBIGUOUS) ? 3438 "ambiguous" : "invalid", "defect list type", 3439 optarg); 3440 error = 1; 3441 goto defect_bailout; 3442 } 3443 break; 3444 } 3445 case 'G': 3446 arglist |= CAM_ARG_GLIST; 3447 break; 3448 case 'P': 3449 arglist |= CAM_ARG_PLIST; 3450 break; 3451 case 'q': 3452 quiet = 1; 3453 break; 3454 case 's': 3455 summary = 1; 3456 break; 3457 case 'S': { 3458 char *endptr; 3459 3460 starting_offset = strtoul(optarg, &endptr, 0); 3461 if (*endptr != '\0') { 3462 error = 1; 3463 warnx("invalid starting offset %s", optarg); 3464 goto defect_bailout; 3465 } 3466 break; 3467 } 3468 case 'X': 3469 hex_format = 1; 3470 break; 3471 default: 3472 break; 3473 } 3474 } 3475 3476 if (list_type_set == 0) { 3477 error = 1; 3478 warnx("no defect list format specified"); 3479 goto defect_bailout; 3480 } 3481 3482 if (arglist & CAM_ARG_PLIST) { 3483 list_format |= SRDD10_PLIST; 3484 lists_specified++; 3485 } 3486 3487 if (arglist & CAM_ARG_GLIST) { 3488 list_format |= SRDD10_GLIST; 3489 lists_specified++; 3490 } 3491 3492 /* 3493 * This implies a summary, and was the previous behavior. 3494 */ 3495 if (lists_specified == 0) 3496 summary = 1; 3497 3498 ccb = cam_getccb(device); 3499 3500 retry_12byte: 3501 3502 /* 3503 * We start off asking for just the header to determine how much 3504 * defect data is available. Some Hitachi drives return an error 3505 * if you ask for more data than the drive has. Once we know the 3506 * length, we retry the command with the returned length. 3507 */ 3508 if (use_12byte == 0) 3509 dlist_length = sizeof(*hdr10); 3510 else 3511 dlist_length = sizeof(*hdr12); 3512 3513 retry: 3514 if (defect_list != NULL) { 3515 free(defect_list); 3516 defect_list = NULL; 3517 } 3518 defect_list = malloc(dlist_length); 3519 if (defect_list == NULL) { 3520 warnx("can't malloc memory for defect list"); 3521 error = 1; 3522 goto defect_bailout; 3523 } 3524 3525 next_batch: 3526 bzero(defect_list, dlist_length); 3527 3528 /* 3529 * cam_getccb() zeros the CCB header only. So we need to zero the 3530 * payload portion of the ccb. 3531 */ 3532 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 3533 3534 scsi_read_defects(&ccb->csio, 3535 /*retries*/ retry_count, 3536 /*cbfcnp*/ NULL, 3537 /*tag_action*/ MSG_SIMPLE_Q_TAG, 3538 /*list_format*/ list_format, 3539 /*addr_desc_index*/ starting_offset, 3540 /*data_ptr*/ defect_list, 3541 /*dxfer_len*/ dlist_length, 3542 /*minimum_cmd_size*/ use_12byte ? 12 : 0, 3543 /*sense_len*/ SSD_FULL_SIZE, 3544 /*timeout*/ timeout ? timeout : 5000); 3545 3546 /* Disable freezing the device queue */ 3547 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 3548 3549 if (cam_send_ccb(device, ccb) < 0) { 3550 perror("error reading defect list"); 3551 3552 if (arglist & CAM_ARG_VERBOSE) { 3553 cam_error_print(device, ccb, CAM_ESF_ALL, 3554 CAM_EPF_ALL, stderr); 3555 } 3556 3557 error = 1; 3558 goto defect_bailout; 3559 } 3560 3561 valid_len = ccb->csio.dxfer_len - ccb->csio.resid; 3562 3563 if (use_12byte == 0) { 3564 hdr10 = (struct scsi_read_defect_data_hdr_10 *)defect_list; 3565 hdr_size = sizeof(*hdr10); 3566 hdr_max = SRDDH10_MAX_LENGTH; 3567 3568 if (valid_len >= hdr_size) { 3569 returned_length = scsi_2btoul(hdr10->length); 3570 returned_format = hdr10->format; 3571 } else { 3572 returned_length = 0; 3573 returned_format = 0; 3574 } 3575 } else { 3576 hdr12 = (struct scsi_read_defect_data_hdr_12 *)defect_list; 3577 hdr_size = sizeof(*hdr12); 3578 hdr_max = SRDDH12_MAX_LENGTH; 3579 3580 if (valid_len >= hdr_size) { 3581 returned_length = scsi_4btoul(hdr12->length); 3582 returned_format = hdr12->format; 3583 } else { 3584 returned_length = 0; 3585 returned_format = 0; 3586 } 3587 } 3588 3589 returned_type = returned_format & SRDDH10_DLIST_FORMAT_MASK; 3590 switch (returned_type) { 3591 case SRDD10_BLOCK_FORMAT: 3592 entry_size = sizeof(struct scsi_defect_desc_block); 3593 break; 3594 case SRDD10_LONG_BLOCK_FORMAT: 3595 entry_size = sizeof(struct scsi_defect_desc_long_block); 3596 break; 3597 case SRDD10_EXT_PHYS_FORMAT: 3598 case SRDD10_PHYSICAL_SECTOR_FORMAT: 3599 entry_size = sizeof(struct scsi_defect_desc_phys_sector); 3600 break; 3601 case SRDD10_EXT_BFI_FORMAT: 3602 case SRDD10_BYTES_FROM_INDEX_FORMAT: 3603 entry_size = sizeof(struct scsi_defect_desc_bytes_from_index); 3604 break; 3605 default: 3606 warnx("Unknown defect format 0x%x\n", returned_type); 3607 error = 1; 3608 goto defect_bailout; 3609 break; 3610 } 3611 3612 max_possible_size = (hdr_max / entry_size) * entry_size; 3613 num_returned = returned_length / entry_size; 3614 num_valid = min(returned_length, valid_len - hdr_size); 3615 num_valid /= entry_size; 3616 3617 if (get_length != 0) { 3618 get_length = 0; 3619 3620 if ((ccb->ccb_h.status & CAM_STATUS_MASK) == 3621 CAM_SCSI_STATUS_ERROR) { 3622 struct scsi_sense_data *sense; 3623 int error_code, sense_key, asc, ascq; 3624 3625 sense = &ccb->csio.sense_data; 3626 scsi_extract_sense_len(sense, ccb->csio.sense_len - 3627 ccb->csio.sense_resid, &error_code, &sense_key, 3628 &asc, &ascq, /*show_errors*/ 1); 3629 3630 /* 3631 * If the drive is reporting that it just doesn't 3632 * support the defect list format, go ahead and use 3633 * the length it reported. Otherwise, the length 3634 * may not be valid, so use the maximum. 3635 */ 3636 if ((sense_key == SSD_KEY_RECOVERED_ERROR) 3637 && (asc == 0x1c) && (ascq == 0x00) 3638 && (returned_length > 0)) { 3639 if ((use_12byte == 0) 3640 && (returned_length >= max_possible_size)) { 3641 get_length = 1; 3642 use_12byte = 1; 3643 goto retry_12byte; 3644 } 3645 dlist_length = returned_length + hdr_size; 3646 } else if ((sense_key == SSD_KEY_RECOVERED_ERROR) 3647 && (asc == 0x1f) && (ascq == 0x00) 3648 && (returned_length > 0)) { 3649 /* Partial defect list transfer */ 3650 /* 3651 * Hitachi drives return this error 3652 * along with a partial defect list if they 3653 * have more defects than the 10 byte 3654 * command can support. Retry with the 12 3655 * byte command. 3656 */ 3657 if (use_12byte == 0) { 3658 get_length = 1; 3659 use_12byte = 1; 3660 goto retry_12byte; 3661 } 3662 dlist_length = returned_length + hdr_size; 3663 } else if ((sense_key == SSD_KEY_ILLEGAL_REQUEST) 3664 && (asc == 0x24) && (ascq == 0x00)) { 3665 /* Invalid field in CDB */ 3666 /* 3667 * SBC-3 says that if the drive has more 3668 * defects than can be reported with the 3669 * 10 byte command, it should return this 3670 * error and no data. Retry with the 12 3671 * byte command. 3672 */ 3673 if (use_12byte == 0) { 3674 get_length = 1; 3675 use_12byte = 1; 3676 goto retry_12byte; 3677 } 3678 dlist_length = returned_length + hdr_size; 3679 } else { 3680 /* 3681 * If we got a SCSI error and no valid length, 3682 * just use the 10 byte maximum. The 12 3683 * byte maximum is too large. 3684 */ 3685 if (returned_length == 0) 3686 dlist_length = SRDD10_MAX_LENGTH; 3687 else { 3688 if ((use_12byte == 0) 3689 && (returned_length >= 3690 max_possible_size)) { 3691 get_length = 1; 3692 use_12byte = 1; 3693 goto retry_12byte; 3694 } 3695 dlist_length = returned_length + 3696 hdr_size; 3697 } 3698 } 3699 } else if ((ccb->ccb_h.status & CAM_STATUS_MASK) != 3700 CAM_REQ_CMP){ 3701 error = 1; 3702 warnx("Error reading defect header"); 3703 if (arglist & CAM_ARG_VERBOSE) 3704 cam_error_print(device, ccb, CAM_ESF_ALL, 3705 CAM_EPF_ALL, stderr); 3706 goto defect_bailout; 3707 } else { 3708 if ((use_12byte == 0) 3709 && (returned_length >= max_possible_size)) { 3710 get_length = 1; 3711 use_12byte = 1; 3712 goto retry_12byte; 3713 } 3714 dlist_length = returned_length + hdr_size; 3715 } 3716 if (summary != 0) { 3717 fprintf(stdout, "%u", num_returned); 3718 if (quiet == 0) { 3719 fprintf(stdout, " defect%s", 3720 (num_returned != 1) ? "s" : ""); 3721 } 3722 fprintf(stdout, "\n"); 3723 3724 goto defect_bailout; 3725 } 3726 3727 /* 3728 * We always limit the list length to the 10-byte maximum 3729 * length (0xffff). The reason is that some controllers 3730 * can't handle larger I/Os, and we can transfer the entire 3731 * 10 byte list in one shot. For drives that support the 12 3732 * byte read defects command, we'll step through the list 3733 * by specifying a starting offset. For drives that don't 3734 * support the 12 byte command's starting offset, we'll 3735 * just display the first 64K. 3736 */ 3737 dlist_length = min(dlist_length, SRDD10_MAX_LENGTH); 3738 3739 goto retry; 3740 } 3741 3742 3743 if (((ccb->ccb_h.status & CAM_STATUS_MASK) == CAM_SCSI_STATUS_ERROR) 3744 && (ccb->csio.scsi_status == SCSI_STATUS_CHECK_COND) 3745 && ((ccb->ccb_h.status & CAM_AUTOSNS_VALID) != 0)) { 3746 struct scsi_sense_data *sense; 3747 int error_code, sense_key, asc, ascq; 3748 3749 sense = &ccb->csio.sense_data; 3750 scsi_extract_sense_len(sense, ccb->csio.sense_len - 3751 ccb->csio.sense_resid, &error_code, &sense_key, &asc, 3752 &ascq, /*show_errors*/ 1); 3753 3754 /* 3755 * According to the SCSI spec, if the disk doesn't support 3756 * the requested format, it will generally return a sense 3757 * key of RECOVERED ERROR, and an additional sense code 3758 * of "DEFECT LIST NOT FOUND". HGST drives also return 3759 * Primary/Grown defect list not found errors. So just 3760 * check for an ASC of 0x1c. 3761 */ 3762 if ((sense_key == SSD_KEY_RECOVERED_ERROR) 3763 && (asc == 0x1c)) { 3764 const char *format_str; 3765 3766 format_str = scsi_nv_to_str(defect_list_type_map, 3767 sizeof(defect_list_type_map) / 3768 sizeof(defect_list_type_map[0]), 3769 list_format & SRDD10_DLIST_FORMAT_MASK); 3770 warnx("requested defect format %s not available", 3771 format_str ? format_str : "unknown"); 3772 3773 format_str = scsi_nv_to_str(defect_list_type_map, 3774 sizeof(defect_list_type_map) / 3775 sizeof(defect_list_type_map[0]), returned_type); 3776 if (format_str != NULL) { 3777 warnx("Device returned %s format", 3778 format_str); 3779 } else { 3780 error = 1; 3781 warnx("Device returned unknown defect" 3782 " data format %#x", returned_type); 3783 goto defect_bailout; 3784 } 3785 } else { 3786 error = 1; 3787 warnx("Error returned from read defect data command"); 3788 if (arglist & CAM_ARG_VERBOSE) 3789 cam_error_print(device, ccb, CAM_ESF_ALL, 3790 CAM_EPF_ALL, stderr); 3791 goto defect_bailout; 3792 } 3793 } else if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 3794 error = 1; 3795 warnx("Error returned from read defect data command"); 3796 if (arglist & CAM_ARG_VERBOSE) 3797 cam_error_print(device, ccb, CAM_ESF_ALL, 3798 CAM_EPF_ALL, stderr); 3799 goto defect_bailout; 3800 } 3801 3802 if (first_pass != 0) { 3803 fprintf(stderr, "Got %d defect", num_returned); 3804 3805 if ((lists_specified == 0) || (num_returned == 0)) { 3806 fprintf(stderr, "s.\n"); 3807 goto defect_bailout; 3808 } else if (num_returned == 1) 3809 fprintf(stderr, ":\n"); 3810 else 3811 fprintf(stderr, "s:\n"); 3812 3813 first_pass = 0; 3814 } 3815 3816 /* 3817 * XXX KDM I should probably clean up the printout format for the 3818 * disk defects. 3819 */ 3820 switch (returned_type) { 3821 case SRDD10_PHYSICAL_SECTOR_FORMAT: 3822 case SRDD10_EXT_PHYS_FORMAT: 3823 { 3824 struct scsi_defect_desc_phys_sector *dlist; 3825 3826 dlist = (struct scsi_defect_desc_phys_sector *) 3827 (defect_list + hdr_size); 3828 3829 for (i = 0; i < num_valid; i++) { 3830 uint32_t sector; 3831 3832 sector = scsi_4btoul(dlist[i].sector); 3833 if (returned_type == SRDD10_EXT_PHYS_FORMAT) { 3834 mads = (sector & SDD_EXT_PHYS_MADS) ? 3835 0 : 1; 3836 sector &= ~SDD_EXT_PHYS_FLAG_MASK; 3837 } 3838 if (hex_format == 0) 3839 fprintf(stdout, "%d:%d:%d%s", 3840 scsi_3btoul(dlist[i].cylinder), 3841 dlist[i].head, 3842 scsi_4btoul(dlist[i].sector), 3843 mads ? " - " : "\n"); 3844 else 3845 fprintf(stdout, "0x%x:0x%x:0x%x%s", 3846 scsi_3btoul(dlist[i].cylinder), 3847 dlist[i].head, 3848 scsi_4btoul(dlist[i].sector), 3849 mads ? " - " : "\n"); 3850 mads = 0; 3851 } 3852 if (num_valid < num_returned) { 3853 starting_offset += num_valid; 3854 goto next_batch; 3855 } 3856 break; 3857 } 3858 case SRDD10_BYTES_FROM_INDEX_FORMAT: 3859 case SRDD10_EXT_BFI_FORMAT: 3860 { 3861 struct scsi_defect_desc_bytes_from_index *dlist; 3862 3863 dlist = (struct scsi_defect_desc_bytes_from_index *) 3864 (defect_list + hdr_size); 3865 3866 for (i = 0; i < num_valid; i++) { 3867 uint32_t bfi; 3868 3869 bfi = scsi_4btoul(dlist[i].bytes_from_index); 3870 if (returned_type == SRDD10_EXT_BFI_FORMAT) { 3871 mads = (bfi & SDD_EXT_BFI_MADS) ? 1 : 0; 3872 bfi &= ~SDD_EXT_BFI_FLAG_MASK; 3873 } 3874 if (hex_format == 0) 3875 fprintf(stdout, "%d:%d:%d%s", 3876 scsi_3btoul(dlist[i].cylinder), 3877 dlist[i].head, 3878 scsi_4btoul(dlist[i].bytes_from_index), 3879 mads ? " - " : "\n"); 3880 else 3881 fprintf(stdout, "0x%x:0x%x:0x%x%s", 3882 scsi_3btoul(dlist[i].cylinder), 3883 dlist[i].head, 3884 scsi_4btoul(dlist[i].bytes_from_index), 3885 mads ? " - " : "\n"); 3886 3887 mads = 0; 3888 } 3889 if (num_valid < num_returned) { 3890 starting_offset += num_valid; 3891 goto next_batch; 3892 } 3893 break; 3894 } 3895 case SRDDH10_BLOCK_FORMAT: 3896 { 3897 struct scsi_defect_desc_block *dlist; 3898 3899 dlist = (struct scsi_defect_desc_block *) 3900 (defect_list + hdr_size); 3901 3902 for (i = 0; i < num_valid; i++) { 3903 if (hex_format == 0) 3904 fprintf(stdout, "%u\n", 3905 scsi_4btoul(dlist[i].address)); 3906 else 3907 fprintf(stdout, "0x%x\n", 3908 scsi_4btoul(dlist[i].address)); 3909 } 3910 3911 if (num_valid < num_returned) { 3912 starting_offset += num_valid; 3913 goto next_batch; 3914 } 3915 3916 break; 3917 } 3918 case SRDD10_LONG_BLOCK_FORMAT: 3919 { 3920 struct scsi_defect_desc_long_block *dlist; 3921 3922 dlist = (struct scsi_defect_desc_long_block *) 3923 (defect_list + hdr_size); 3924 3925 for (i = 0; i < num_valid; i++) { 3926 if (hex_format == 0) 3927 fprintf(stdout, "%ju\n", 3928 (uintmax_t)scsi_8btou64( 3929 dlist[i].address)); 3930 else 3931 fprintf(stdout, "0x%jx\n", 3932 (uintmax_t)scsi_8btou64( 3933 dlist[i].address)); 3934 } 3935 3936 if (num_valid < num_returned) { 3937 starting_offset += num_valid; 3938 goto next_batch; 3939 } 3940 break; 3941 } 3942 default: 3943 fprintf(stderr, "Unknown defect format 0x%x\n", 3944 returned_type); 3945 error = 1; 3946 break; 3947 } 3948 defect_bailout: 3949 3950 if (defect_list != NULL) 3951 free(defect_list); 3952 3953 if (ccb != NULL) 3954 cam_freeccb(ccb); 3955 3956 return(error); 3957 } 3958 #endif /* MINIMALISTIC */ 3959 3960 #if 0 3961 void 3962 reassignblocks(struct cam_device *device, u_int32_t *blocks, int num_blocks) 3963 { 3964 union ccb *ccb; 3965 3966 ccb = cam_getccb(device); 3967 3968 cam_freeccb(ccb); 3969 } 3970 #endif 3971 3972 #ifndef MINIMALISTIC 3973 void 3974 mode_sense(struct cam_device *device, int mode_page, int page_control, 3975 int dbd, int retry_count, int timeout, u_int8_t *data, int datalen) 3976 { 3977 union ccb *ccb; 3978 int retval; 3979 3980 ccb = cam_getccb(device); 3981 3982 if (ccb == NULL) 3983 errx(1, "mode_sense: couldn't allocate CCB"); 3984 3985 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 3986 3987 scsi_mode_sense(&ccb->csio, 3988 /* retries */ retry_count, 3989 /* cbfcnp */ NULL, 3990 /* tag_action */ MSG_SIMPLE_Q_TAG, 3991 /* dbd */ dbd, 3992 /* page_code */ page_control << 6, 3993 /* page */ mode_page, 3994 /* param_buf */ data, 3995 /* param_len */ datalen, 3996 /* sense_len */ SSD_FULL_SIZE, 3997 /* timeout */ timeout ? timeout : 5000); 3998 3999 if (arglist & CAM_ARG_ERR_RECOVER) 4000 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 4001 4002 /* Disable freezing the device queue */ 4003 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 4004 4005 if (((retval = cam_send_ccb(device, ccb)) < 0) 4006 || ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP)) { 4007 if (arglist & CAM_ARG_VERBOSE) { 4008 cam_error_print(device, ccb, CAM_ESF_ALL, 4009 CAM_EPF_ALL, stderr); 4010 } 4011 cam_freeccb(ccb); 4012 cam_close_device(device); 4013 if (retval < 0) 4014 err(1, "error sending mode sense command"); 4015 else 4016 errx(1, "error sending mode sense command"); 4017 } 4018 4019 cam_freeccb(ccb); 4020 } 4021 4022 void 4023 mode_select(struct cam_device *device, int save_pages, int retry_count, 4024 int timeout, u_int8_t *data, int datalen) 4025 { 4026 union ccb *ccb; 4027 int retval; 4028 4029 ccb = cam_getccb(device); 4030 4031 if (ccb == NULL) 4032 errx(1, "mode_select: couldn't allocate CCB"); 4033 4034 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 4035 4036 scsi_mode_select(&ccb->csio, 4037 /* retries */ retry_count, 4038 /* cbfcnp */ NULL, 4039 /* tag_action */ MSG_SIMPLE_Q_TAG, 4040 /* scsi_page_fmt */ 1, 4041 /* save_pages */ save_pages, 4042 /* param_buf */ data, 4043 /* param_len */ datalen, 4044 /* sense_len */ SSD_FULL_SIZE, 4045 /* timeout */ timeout ? timeout : 5000); 4046 4047 if (arglist & CAM_ARG_ERR_RECOVER) 4048 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 4049 4050 /* Disable freezing the device queue */ 4051 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 4052 4053 if (((retval = cam_send_ccb(device, ccb)) < 0) 4054 || ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP)) { 4055 if (arglist & CAM_ARG_VERBOSE) { 4056 cam_error_print(device, ccb, CAM_ESF_ALL, 4057 CAM_EPF_ALL, stderr); 4058 } 4059 cam_freeccb(ccb); 4060 cam_close_device(device); 4061 4062 if (retval < 0) 4063 err(1, "error sending mode select command"); 4064 else 4065 errx(1, "error sending mode select command"); 4066 4067 } 4068 4069 cam_freeccb(ccb); 4070 } 4071 4072 void 4073 modepage(struct cam_device *device, int argc, char **argv, char *combinedopt, 4074 int retry_count, int timeout) 4075 { 4076 int c, mode_page = -1, page_control = 0; 4077 int binary = 0, list = 0; 4078 4079 while ((c = getopt(argc, argv, combinedopt)) != -1) { 4080 switch(c) { 4081 case 'b': 4082 binary = 1; 4083 break; 4084 case 'd': 4085 arglist |= CAM_ARG_DBD; 4086 break; 4087 case 'e': 4088 arglist |= CAM_ARG_MODE_EDIT; 4089 break; 4090 case 'l': 4091 list = 1; 4092 break; 4093 case 'm': 4094 mode_page = strtol(optarg, NULL, 0); 4095 if (mode_page < 0) 4096 errx(1, "invalid mode page %d", mode_page); 4097 break; 4098 case 'P': 4099 page_control = strtol(optarg, NULL, 0); 4100 if ((page_control < 0) || (page_control > 3)) 4101 errx(1, "invalid page control field %d", 4102 page_control); 4103 arglist |= CAM_ARG_PAGE_CNTL; 4104 break; 4105 default: 4106 break; 4107 } 4108 } 4109 4110 if (mode_page == -1 && list == 0) 4111 errx(1, "you must specify a mode page!"); 4112 4113 if (list) { 4114 mode_list(device, page_control, arglist & CAM_ARG_DBD, 4115 retry_count, timeout); 4116 } else { 4117 mode_edit(device, mode_page, page_control, 4118 arglist & CAM_ARG_DBD, arglist & CAM_ARG_MODE_EDIT, binary, 4119 retry_count, timeout); 4120 } 4121 } 4122 4123 static int 4124 scsicmd(struct cam_device *device, int argc, char **argv, char *combinedopt, 4125 int retry_count, int timeout) 4126 { 4127 union ccb *ccb; 4128 u_int32_t flags = CAM_DIR_NONE; 4129 u_int8_t *data_ptr = NULL; 4130 u_int8_t cdb[20]; 4131 u_int8_t atacmd[12]; 4132 struct get_hook hook; 4133 int c, data_bytes = 0; 4134 int cdb_len = 0; 4135 int atacmd_len = 0; 4136 int dmacmd = 0; 4137 int fpdmacmd = 0; 4138 int need_res = 0; 4139 char *datastr = NULL, *tstr, *resstr = NULL; 4140 int error = 0; 4141 int fd_data = 0, fd_res = 0; 4142 int retval; 4143 4144 ccb = cam_getccb(device); 4145 4146 if (ccb == NULL) { 4147 warnx("scsicmd: error allocating ccb"); 4148 return(1); 4149 } 4150 4151 CCB_CLEAR_ALL_EXCEPT_HDR(ccb); 4152 4153 while ((c = getopt(argc, argv, combinedopt)) != -1) { 4154 switch(c) { 4155 case 'a': 4156 tstr = optarg; 4157 while (isspace(*tstr) && (*tstr != '\0')) 4158 tstr++; 4159 hook.argc = argc - optind; 4160 hook.argv = argv + optind; 4161 hook.got = 0; 4162 atacmd_len = buff_encode_visit(atacmd, sizeof(atacmd), tstr, 4163 iget, &hook); 4164 /* 4165 * Increment optind by the number of arguments the 4166 * encoding routine processed. After each call to 4167 * getopt(3), optind points to the argument that 4168 * getopt should process _next_. In this case, 4169 * that means it points to the first command string 4170 * argument, if there is one. Once we increment 4171 * this, it should point to either the next command 4172 * line argument, or it should be past the end of 4173 * the list. 4174 */ 4175 optind += hook.got; 4176 break; 4177 case 'c': 4178 tstr = optarg; 4179 while (isspace(*tstr) && (*tstr != '\0')) 4180 tstr++; 4181 hook.argc = argc - optind; 4182 hook.argv = argv + optind; 4183 hook.got = 0; 4184 cdb_len = buff_encode_visit(cdb, sizeof(cdb), tstr, 4185 iget, &hook); 4186 /* 4187 * Increment optind by the number of arguments the 4188 * encoding routine processed. After each call to 4189 * getopt(3), optind points to the argument that 4190 * getopt should process _next_. In this case, 4191 * that means it points to the first command string 4192 * argument, if there is one. Once we increment 4193 * this, it should point to either the next command 4194 * line argument, or it should be past the end of 4195 * the list. 4196 */ 4197 optind += hook.got; 4198 break; 4199 case 'd': 4200 dmacmd = 1; 4201 break; 4202 case 'f': 4203 fpdmacmd = 1; 4204 break; 4205 case 'i': 4206 if (arglist & CAM_ARG_CMD_OUT) { 4207 warnx("command must either be " 4208 "read or write, not both"); 4209 error = 1; 4210 goto scsicmd_bailout; 4211 } 4212 arglist |= CAM_ARG_CMD_IN; 4213 flags = CAM_DIR_IN; 4214 data_bytes = strtol(optarg, NULL, 0); 4215 if (data_bytes <= 0) { 4216 warnx("invalid number of input bytes %d", 4217 data_bytes); 4218 error = 1; 4219 goto scsicmd_bailout; 4220 } 4221 hook.argc = argc - optind; 4222 hook.argv = argv + optind; 4223 hook.got = 0; 4224 optind++; 4225 datastr = cget(&hook, NULL); 4226 /* 4227 * If the user supplied "-" instead of a format, he 4228 * wants the data to be written to stdout. 4229 */ 4230 if ((datastr != NULL) 4231 && (datastr[0] == '-')) 4232 fd_data = 1; 4233 4234 data_ptr = (u_int8_t *)malloc(data_bytes); 4235 if (data_ptr == NULL) { 4236 warnx("can't malloc memory for data_ptr"); 4237 error = 1; 4238 goto scsicmd_bailout; 4239 } 4240 break; 4241 case 'o': 4242 if (arglist & CAM_ARG_CMD_IN) { 4243 warnx("command must either be " 4244 "read or write, not both"); 4245 error = 1; 4246 goto scsicmd_bailout; 4247 } 4248 arglist |= CAM_ARG_CMD_OUT; 4249 flags = CAM_DIR_OUT; 4250 data_bytes = strtol(optarg, NULL, 0); 4251 if (data_bytes <= 0) { 4252 warnx("invalid number of output bytes %d", 4253 data_bytes); 4254 error = 1; 4255 goto scsicmd_bailout; 4256 } 4257 hook.argc = argc - optind; 4258 hook.argv = argv + optind; 4259 hook.got = 0; 4260 datastr = cget(&hook, NULL); 4261 data_ptr = (u_int8_t *)malloc(data_bytes); 4262 if (data_ptr == NULL) { 4263 warnx("can't malloc memory for data_ptr"); 4264 error = 1; 4265 goto scsicmd_bailout; 4266 } 4267 bzero(data_ptr, data_bytes); 4268 /* 4269 * If the user supplied "-" instead of a format, he 4270 * wants the data to be read from stdin. 4271 */ 4272 if ((datastr != NULL) 4273 && (datastr[0] == '-')) 4274 fd_data = 1; 4275 else 4276 buff_encode_visit(data_ptr, data_bytes, datastr, 4277 iget, &hook); 4278 optind += hook.got; 4279 break; 4280 case 'r': 4281 need_res = 1; 4282 hook.argc = argc - optind; 4283 hook.argv = argv + optind; 4284 hook.got = 0; 4285 resstr = cget(&hook, NULL); 4286 if ((resstr != NULL) && (resstr[0] == '-')) 4287 fd_res = 1; 4288 optind += hook.got; 4289 break; 4290 default: 4291 break; 4292 } 4293 } 4294 4295 /* 4296 * If fd_data is set, and we're writing to the device, we need to 4297 * read the data the user wants written from stdin. 4298 */ 4299 if ((fd_data == 1) && (arglist & CAM_ARG_CMD_OUT)) { 4300 ssize_t amt_read; 4301 int amt_to_read = data_bytes; 4302 u_int8_t *buf_ptr = data_ptr; 4303 4304 for (amt_read = 0; amt_to_read > 0; 4305 amt_read = read(STDIN_FILENO, buf_ptr, amt_to_read)) { 4306 if (amt_read == -1) { 4307 warn("error reading data from stdin"); 4308 error = 1; 4309 goto scsicmd_bailout; 4310 } 4311 amt_to_read -= amt_read; 4312 buf_ptr += amt_read; 4313 } 4314 } 4315 4316 if (arglist & CAM_ARG_ERR_RECOVER) 4317 flags |= CAM_PASS_ERR_RECOVER; 4318 4319 /* Disable freezing the device queue */ 4320 flags |= CAM_DEV_QFRZDIS; 4321 4322 if (cdb_len) { 4323 /* 4324 * This is taken from the SCSI-3 draft spec. 4325 * (T10/1157D revision 0.3) 4326 * The top 3 bits of an opcode are the group code. 4327 * The next 5 bits are the command code. 4328 * Group 0: six byte commands 4329 * Group 1: ten byte commands 4330 * Group 2: ten byte commands 4331 * Group 3: reserved 4332 * Group 4: sixteen byte commands 4333 * Group 5: twelve byte commands 4334 * Group 6: vendor specific 4335 * Group 7: vendor specific 4336 */ 4337 switch((cdb[0] >> 5) & 0x7) { 4338 case 0: 4339 cdb_len = 6; 4340 break; 4341 case 1: 4342 case 2: 4343 cdb_len = 10; 4344 break; 4345 case 3: 4346 case 6: 4347 case 7: 4348 /* computed by buff_encode_visit */ 4349 break; 4350 case 4: 4351 cdb_len = 16; 4352 break; 4353 case 5: 4354 cdb_len = 12; 4355 break; 4356 } 4357 4358 /* 4359 * We should probably use csio_build_visit or something like that 4360 * here, but it's easier to encode arguments as you go. The 4361 * alternative would be skipping the CDB argument and then encoding 4362 * it here, since we've got the data buffer argument by now. 4363 */ 4364 bcopy(cdb, &ccb->csio.cdb_io.cdb_bytes, cdb_len); 4365 4366 cam_fill_csio(&ccb->csio, 4367 /*retries*/ retry_count, 4368 /*cbfcnp*/ NULL, 4369 /*flags*/ flags, 4370 /*tag_action*/ MSG_SIMPLE_Q_TAG, 4371 /*data_ptr*/ data_ptr, 4372 /*dxfer_len*/ data_bytes, 4373 /*sense_len*/ SSD_FULL_SIZE, 4374 /*cdb_len*/ cdb_len, 4375 /*timeout*/ timeout ? timeout : 5000); 4376 } else { 4377 atacmd_len = 12; 4378 bcopy(atacmd, &ccb->ataio.cmd.command, atacmd_len); 4379 if (need_res) 4380 ccb->ataio.cmd.flags |= CAM_ATAIO_NEEDRESULT; 4381 if (dmacmd) 4382 ccb->ataio.cmd.flags |= CAM_ATAIO_DMA; 4383 if (fpdmacmd) 4384 ccb->ataio.cmd.flags |= CAM_ATAIO_FPDMA; 4385 4386 cam_fill_ataio(&ccb->ataio, 4387 /*retries*/ retry_count, 4388 /*cbfcnp*/ NULL, 4389 /*flags*/ flags, 4390 /*tag_action*/ 0, 4391 /*data_ptr*/ data_ptr, 4392 /*dxfer_len*/ data_bytes, 4393 /*timeout*/ timeout ? timeout : 5000); 4394 } 4395 4396 if (((retval = cam_send_ccb(device, ccb)) < 0) 4397 || ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP)) { 4398 const char warnstr[] = "error sending command"; 4399 4400 if (retval < 0) 4401 warn(warnstr); 4402 else 4403 warnx(warnstr); 4404 4405 if (arglist & CAM_ARG_VERBOSE) { 4406 cam_error_print(device, ccb, CAM_ESF_ALL, 4407 CAM_EPF_ALL, stderr); 4408 } 4409 4410 error = 1; 4411 goto scsicmd_bailout; 4412 } 4413 4414 if (atacmd_len && need_res) { 4415 if (fd_res == 0) { 4416 buff_decode_visit(&ccb->ataio.res.status, 11, resstr, 4417 arg_put, NULL); 4418 fprintf(stdout, "\n"); 4419 } else { 4420 fprintf(stdout, 4421 "%02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X\n", 4422 ccb->ataio.res.status, 4423 ccb->ataio.res.error, 4424 ccb->ataio.res.lba_low, 4425 ccb->ataio.res.lba_mid, 4426 ccb->ataio.res.lba_high, 4427 ccb->ataio.res.device, 4428 ccb->ataio.res.lba_low_exp, 4429 ccb->ataio.res.lba_mid_exp, 4430 ccb->ataio.res.lba_high_exp, 4431 ccb->ataio.res.sector_count, 4432 ccb->ataio.res.sector_count_exp); 4433 fflush(stdout); 4434 } 4435 } 4436 4437 if (((ccb->ccb_h.status & CAM_STATUS_MASK) == CAM_REQ_CMP) 4438 && (arglist & CAM_ARG_CMD_IN) 4439 && (data_bytes > 0)) { 4440 if (fd_data == 0) { 4441 buff_decode_visit(data_ptr, data_bytes, datastr, 4442 arg_put, NULL); 4443 fprintf(stdout, "\n"); 4444 } else { 4445 ssize_t amt_written; 4446 int amt_to_write = data_bytes; 4447 u_int8_t *buf_ptr = data_ptr; 4448 4449 for (amt_written = 0; (amt_to_write > 0) && 4450 (amt_written =write(1, buf_ptr,amt_to_write))> 0;){ 4451 amt_to_write -= amt_written; 4452 buf_ptr += amt_written; 4453 } 4454 if (amt_written == -1) { 4455 warn("error writing data to stdout"); 4456 error = 1; 4457 goto scsicmd_bailout; 4458 } else if ((amt_written == 0) 4459 && (amt_to_write > 0)) { 4460 warnx("only wrote %u bytes out of %u", 4461 data_bytes - amt_to_write, data_bytes); 4462 } 4463 } 4464 } 4465 4466 scsicmd_bailout: 4467 4468 if ((data_bytes > 0) && (data_ptr != NULL)) 4469 free(data_ptr); 4470 4471 cam_freeccb(ccb); 4472 4473 return(error); 4474 } 4475 4476 static int 4477 camdebug(int argc, char **argv, char *combinedopt) 4478 { 4479 int c, fd; 4480 path_id_t bus = CAM_BUS_WILDCARD; 4481 target_id_t target = CAM_TARGET_WILDCARD; 4482 lun_id_t lun = CAM_LUN_WILDCARD; 4483 char *tstr, *tmpstr = NULL; 4484 union ccb ccb; 4485 int error = 0; 4486 4487 bzero(&ccb, sizeof(union ccb)); 4488 4489 while ((c = getopt(argc, argv, combinedopt)) != -1) { 4490 switch(c) { 4491 case 'I': 4492 arglist |= CAM_ARG_DEBUG_INFO; 4493 ccb.cdbg.flags |= CAM_DEBUG_INFO; 4494 break; 4495 case 'P': 4496 arglist |= CAM_ARG_DEBUG_PERIPH; 4497 ccb.cdbg.flags |= CAM_DEBUG_PERIPH; 4498 break; 4499 case 'S': 4500 arglist |= CAM_ARG_DEBUG_SUBTRACE; 4501 ccb.cdbg.flags |= CAM_DEBUG_SUBTRACE; 4502 break; 4503 case 'T': 4504 arglist |= CAM_ARG_DEBUG_TRACE; 4505 ccb.cdbg.flags |= CAM_DEBUG_TRACE; 4506 break; 4507 case 'X': 4508 arglist |= CAM_ARG_DEBUG_XPT; 4509 ccb.cdbg.flags |= CAM_DEBUG_XPT; 4510 break; 4511 case 'c': 4512 arglist |= CAM_ARG_DEBUG_CDB; 4513 ccb.cdbg.flags |= CAM_DEBUG_CDB; 4514 break; 4515 case 'p': 4516 arglist |= CAM_ARG_DEBUG_PROBE; 4517 ccb.cdbg.flags |= CAM_DEBUG_PROBE; 4518 break; 4519 default: 4520 break; 4521 } 4522 } 4523 4524 if ((fd = open(XPT_DEVICE, O_RDWR)) < 0) { 4525 warnx("error opening transport layer device %s", XPT_DEVICE); 4526 warn("%s", XPT_DEVICE); 4527 return(1); 4528 } 4529 argc -= optind; 4530 argv += optind; 4531 4532 if (argc <= 0) { 4533 warnx("you must specify \"off\", \"all\" or a bus,"); 4534 warnx("bus:target, or bus:target:lun"); 4535 close(fd); 4536 return(1); 4537 } 4538 4539 tstr = *argv; 4540 4541 while (isspace(*tstr) && (*tstr != '\0')) 4542 tstr++; 4543 4544 if (strncmp(tstr, "off", 3) == 0) { 4545 ccb.cdbg.flags = CAM_DEBUG_NONE; 4546 arglist &= ~(CAM_ARG_DEBUG_INFO|CAM_ARG_DEBUG_PERIPH| 4547 CAM_ARG_DEBUG_TRACE|CAM_ARG_DEBUG_SUBTRACE| 4548 CAM_ARG_DEBUG_XPT|CAM_ARG_DEBUG_PROBE); 4549 } else if (strncmp(tstr, "all", 3) != 0) { 4550 tmpstr = (char *)strtok(tstr, ":"); 4551 if ((tmpstr != NULL) && (*tmpstr != '\0')){ 4552 bus = strtol(tmpstr, NULL, 0); 4553 arglist |= CAM_ARG_BUS; 4554 tmpstr = (char *)strtok(NULL, ":"); 4555 if ((tmpstr != NULL) && (*tmpstr != '\0')){ 4556 target = strtol(tmpstr, NULL, 0); 4557 arglist |= CAM_ARG_TARGET; 4558 tmpstr = (char *)strtok(NULL, ":"); 4559 if ((tmpstr != NULL) && (*tmpstr != '\0')){ 4560 lun = strtol(tmpstr, NULL, 0); 4561 arglist |= CAM_ARG_LUN; 4562 } 4563 } 4564 } else { 4565 error = 1; 4566 warnx("you must specify \"all\", \"off\", or a bus,"); 4567 warnx("bus:target, or bus:target:lun to debug"); 4568 } 4569 } 4570 4571 if (error == 0) { 4572 4573 ccb.ccb_h.func_code = XPT_DEBUG; 4574 ccb.ccb_h.path_id = bus; 4575 ccb.ccb_h.target_id = target; 4576 ccb.ccb_h.target_lun = lun; 4577 4578 if (ioctl(fd, CAMIOCOMMAND, &ccb) == -1) { 4579 warn("CAMIOCOMMAND ioctl failed"); 4580 error = 1; 4581 } 4582 4583 if (error == 0) { 4584 if ((ccb.ccb_h.status & CAM_STATUS_MASK) == 4585 CAM_FUNC_NOTAVAIL) { 4586 warnx("CAM debugging not available"); 4587 warnx("you need to put options CAMDEBUG in" 4588 " your kernel config file!"); 4589 error = 1; 4590 } else if ((ccb.ccb_h.status & CAM_STATUS_MASK) != 4591 CAM_REQ_CMP) { 4592 warnx("XPT_DEBUG CCB failed with status %#x", 4593 ccb.ccb_h.status); 4594 error = 1; 4595 } else { 4596 if (ccb.cdbg.flags == CAM_DEBUG_NONE) { 4597 fprintf(stderr, 4598 "Debugging turned off\n"); 4599 } else { 4600 fprintf(stderr, 4601 "Debugging enabled for " 4602 "%d:%d:%jx\n", 4603 bus, target, (uintmax_t)lun); 4604 } 4605 } 4606 } 4607 close(fd); 4608 } 4609 4610 return(error); 4611 } 4612 4613 static int 4614 tagcontrol(struct cam_device *device, int argc, char **argv, 4615 char *combinedopt) 4616 { 4617 int c; 4618 union ccb *ccb; 4619 int numtags = -1; 4620 int retval = 0; 4621 int quiet = 0; 4622 char pathstr[1024]; 4623 4624 ccb = cam_getccb(device); 4625 4626 if (ccb == NULL) { 4627 warnx("tagcontrol: error allocating ccb"); 4628 return(1); 4629 } 4630 4631 while ((c = getopt(argc, argv, combinedopt)) != -1) { 4632 switch(c) { 4633 case 'N': 4634 numtags = strtol(optarg, NULL, 0); 4635 if (numtags < 0) { 4636 warnx("tag count %d is < 0", numtags); 4637 retval = 1; 4638 goto tagcontrol_bailout; 4639 } 4640 break; 4641 case 'q': 4642 quiet++; 4643 break; 4644 default: 4645 break; 4646 } 4647 } 4648 4649 cam_path_string(device, pathstr, sizeof(pathstr)); 4650 4651 if (numtags >= 0) { 4652 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->crs); 4653 ccb->ccb_h.func_code = XPT_REL_SIMQ; 4654 ccb->ccb_h.flags = CAM_DEV_QFREEZE; 4655 ccb->crs.release_flags = RELSIM_ADJUST_OPENINGS; 4656 ccb->crs.openings = numtags; 4657 4658 4659 if (cam_send_ccb(device, ccb) < 0) { 4660 perror("error sending XPT_REL_SIMQ CCB"); 4661 retval = 1; 4662 goto tagcontrol_bailout; 4663 } 4664 4665 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 4666 warnx("XPT_REL_SIMQ CCB failed"); 4667 cam_error_print(device, ccb, CAM_ESF_ALL, 4668 CAM_EPF_ALL, stderr); 4669 retval = 1; 4670 goto tagcontrol_bailout; 4671 } 4672 4673 4674 if (quiet == 0) 4675 fprintf(stdout, "%stagged openings now %d\n", 4676 pathstr, ccb->crs.openings); 4677 } 4678 4679 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->cgds); 4680 4681 ccb->ccb_h.func_code = XPT_GDEV_STATS; 4682 4683 if (cam_send_ccb(device, ccb) < 0) { 4684 perror("error sending XPT_GDEV_STATS CCB"); 4685 retval = 1; 4686 goto tagcontrol_bailout; 4687 } 4688 4689 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 4690 warnx("XPT_GDEV_STATS CCB failed"); 4691 cam_error_print(device, ccb, CAM_ESF_ALL, 4692 CAM_EPF_ALL, stderr); 4693 retval = 1; 4694 goto tagcontrol_bailout; 4695 } 4696 4697 if (arglist & CAM_ARG_VERBOSE) { 4698 fprintf(stdout, "%s", pathstr); 4699 fprintf(stdout, "dev_openings %d\n", ccb->cgds.dev_openings); 4700 fprintf(stdout, "%s", pathstr); 4701 fprintf(stdout, "dev_active %d\n", ccb->cgds.dev_active); 4702 fprintf(stdout, "%s", pathstr); 4703 fprintf(stdout, "allocated %d\n", ccb->cgds.allocated); 4704 fprintf(stdout, "%s", pathstr); 4705 fprintf(stdout, "queued %d\n", ccb->cgds.queued); 4706 fprintf(stdout, "%s", pathstr); 4707 fprintf(stdout, "held %d\n", ccb->cgds.held); 4708 fprintf(stdout, "%s", pathstr); 4709 fprintf(stdout, "mintags %d\n", ccb->cgds.mintags); 4710 fprintf(stdout, "%s", pathstr); 4711 fprintf(stdout, "maxtags %d\n", ccb->cgds.maxtags); 4712 } else { 4713 if (quiet == 0) { 4714 fprintf(stdout, "%s", pathstr); 4715 fprintf(stdout, "device openings: "); 4716 } 4717 fprintf(stdout, "%d\n", ccb->cgds.dev_openings + 4718 ccb->cgds.dev_active); 4719 } 4720 4721 tagcontrol_bailout: 4722 4723 cam_freeccb(ccb); 4724 return(retval); 4725 } 4726 4727 static void 4728 cts_print(struct cam_device *device, struct ccb_trans_settings *cts) 4729 { 4730 char pathstr[1024]; 4731 4732 cam_path_string(device, pathstr, sizeof(pathstr)); 4733 4734 if (cts->transport == XPORT_SPI) { 4735 struct ccb_trans_settings_spi *spi = 4736 &cts->xport_specific.spi; 4737 4738 if ((spi->valid & CTS_SPI_VALID_SYNC_RATE) != 0) { 4739 4740 fprintf(stdout, "%ssync parameter: %d\n", pathstr, 4741 spi->sync_period); 4742 4743 if (spi->sync_offset != 0) { 4744 u_int freq; 4745 4746 freq = scsi_calc_syncsrate(spi->sync_period); 4747 fprintf(stdout, "%sfrequency: %d.%03dMHz\n", 4748 pathstr, freq / 1000, freq % 1000); 4749 } 4750 } 4751 4752 if (spi->valid & CTS_SPI_VALID_SYNC_OFFSET) { 4753 fprintf(stdout, "%soffset: %d\n", pathstr, 4754 spi->sync_offset); 4755 } 4756 4757 if (spi->valid & CTS_SPI_VALID_BUS_WIDTH) { 4758 fprintf(stdout, "%sbus width: %d bits\n", pathstr, 4759 (0x01 << spi->bus_width) * 8); 4760 } 4761 4762 if (spi->valid & CTS_SPI_VALID_DISC) { 4763 fprintf(stdout, "%sdisconnection is %s\n", pathstr, 4764 (spi->flags & CTS_SPI_FLAGS_DISC_ENB) ? 4765 "enabled" : "disabled"); 4766 } 4767 } 4768 if (cts->transport == XPORT_FC) { 4769 struct ccb_trans_settings_fc *fc = 4770 &cts->xport_specific.fc; 4771 4772 if (fc->valid & CTS_FC_VALID_WWNN) 4773 fprintf(stdout, "%sWWNN: 0x%llx\n", pathstr, 4774 (long long) fc->wwnn); 4775 if (fc->valid & CTS_FC_VALID_WWPN) 4776 fprintf(stdout, "%sWWPN: 0x%llx\n", pathstr, 4777 (long long) fc->wwpn); 4778 if (fc->valid & CTS_FC_VALID_PORT) 4779 fprintf(stdout, "%sPortID: 0x%x\n", pathstr, fc->port); 4780 if (fc->valid & CTS_FC_VALID_SPEED) 4781 fprintf(stdout, "%stransfer speed: %d.%03dMB/s\n", 4782 pathstr, fc->bitrate / 1000, fc->bitrate % 1000); 4783 } 4784 if (cts->transport == XPORT_SAS) { 4785 struct ccb_trans_settings_sas *sas = 4786 &cts->xport_specific.sas; 4787 4788 if (sas->valid & CTS_SAS_VALID_SPEED) 4789 fprintf(stdout, "%stransfer speed: %d.%03dMB/s\n", 4790 pathstr, sas->bitrate / 1000, sas->bitrate % 1000); 4791 } 4792 if (cts->transport == XPORT_ATA) { 4793 struct ccb_trans_settings_pata *pata = 4794 &cts->xport_specific.ata; 4795 4796 if ((pata->valid & CTS_ATA_VALID_MODE) != 0) { 4797 fprintf(stdout, "%sATA mode: %s\n", pathstr, 4798 ata_mode2string(pata->mode)); 4799 } 4800 if ((pata->valid & CTS_ATA_VALID_ATAPI) != 0) { 4801 fprintf(stdout, "%sATAPI packet length: %d\n", pathstr, 4802 pata->atapi); 4803 } 4804 if ((pata->valid & CTS_ATA_VALID_BYTECOUNT) != 0) { 4805 fprintf(stdout, "%sPIO transaction length: %d\n", 4806 pathstr, pata->bytecount); 4807 } 4808 } 4809 if (cts->transport == XPORT_SATA) { 4810 struct ccb_trans_settings_sata *sata = 4811 &cts->xport_specific.sata; 4812 4813 if ((sata->valid & CTS_SATA_VALID_REVISION) != 0) { 4814 fprintf(stdout, "%sSATA revision: %d.x\n", pathstr, 4815 sata->revision); 4816 } 4817 if ((sata->valid & CTS_SATA_VALID_MODE) != 0) { 4818 fprintf(stdout, "%sATA mode: %s\n", pathstr, 4819 ata_mode2string(sata->mode)); 4820 } 4821 if ((sata->valid & CTS_SATA_VALID_ATAPI) != 0) { 4822 fprintf(stdout, "%sATAPI packet length: %d\n", pathstr, 4823 sata->atapi); 4824 } 4825 if ((sata->valid & CTS_SATA_VALID_BYTECOUNT) != 0) { 4826 fprintf(stdout, "%sPIO transaction length: %d\n", 4827 pathstr, sata->bytecount); 4828 } 4829 if ((sata->valid & CTS_SATA_VALID_PM) != 0) { 4830 fprintf(stdout, "%sPMP presence: %d\n", pathstr, 4831 sata->pm_present); 4832 } 4833 if ((sata->valid & CTS_SATA_VALID_TAGS) != 0) { 4834 fprintf(stdout, "%sNumber of tags: %d\n", pathstr, 4835 sata->tags); 4836 } 4837 if ((sata->valid & CTS_SATA_VALID_CAPS) != 0) { 4838 fprintf(stdout, "%sSATA capabilities: %08x\n", pathstr, 4839 sata->caps); 4840 } 4841 } 4842 if (cts->protocol == PROTO_ATA) { 4843 struct ccb_trans_settings_ata *ata= 4844 &cts->proto_specific.ata; 4845 4846 if (ata->valid & CTS_ATA_VALID_TQ) { 4847 fprintf(stdout, "%stagged queueing: %s\n", pathstr, 4848 (ata->flags & CTS_ATA_FLAGS_TAG_ENB) ? 4849 "enabled" : "disabled"); 4850 } 4851 } 4852 if (cts->protocol == PROTO_SCSI) { 4853 struct ccb_trans_settings_scsi *scsi= 4854 &cts->proto_specific.scsi; 4855 4856 if (scsi->valid & CTS_SCSI_VALID_TQ) { 4857 fprintf(stdout, "%stagged queueing: %s\n", pathstr, 4858 (scsi->flags & CTS_SCSI_FLAGS_TAG_ENB) ? 4859 "enabled" : "disabled"); 4860 } 4861 } 4862 4863 } 4864 4865 /* 4866 * Get a path inquiry CCB for the specified device. 4867 */ 4868 static int 4869 get_cpi(struct cam_device *device, struct ccb_pathinq *cpi) 4870 { 4871 union ccb *ccb; 4872 int retval = 0; 4873 4874 ccb = cam_getccb(device); 4875 if (ccb == NULL) { 4876 warnx("get_cpi: couldn't allocate CCB"); 4877 return(1); 4878 } 4879 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->cpi); 4880 ccb->ccb_h.func_code = XPT_PATH_INQ; 4881 if (cam_send_ccb(device, ccb) < 0) { 4882 warn("get_cpi: error sending Path Inquiry CCB"); 4883 if (arglist & CAM_ARG_VERBOSE) 4884 cam_error_print(device, ccb, CAM_ESF_ALL, 4885 CAM_EPF_ALL, stderr); 4886 retval = 1; 4887 goto get_cpi_bailout; 4888 } 4889 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 4890 if (arglist & CAM_ARG_VERBOSE) 4891 cam_error_print(device, ccb, CAM_ESF_ALL, 4892 CAM_EPF_ALL, stderr); 4893 retval = 1; 4894 goto get_cpi_bailout; 4895 } 4896 bcopy(&ccb->cpi, cpi, sizeof(struct ccb_pathinq)); 4897 4898 get_cpi_bailout: 4899 cam_freeccb(ccb); 4900 return(retval); 4901 } 4902 4903 /* 4904 * Get a get device CCB for the specified device. 4905 */ 4906 static int 4907 get_cgd(struct cam_device *device, struct ccb_getdev *cgd) 4908 { 4909 union ccb *ccb; 4910 int retval = 0; 4911 4912 ccb = cam_getccb(device); 4913 if (ccb == NULL) { 4914 warnx("get_cgd: couldn't allocate CCB"); 4915 return(1); 4916 } 4917 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->cgd); 4918 ccb->ccb_h.func_code = XPT_GDEV_TYPE; 4919 if (cam_send_ccb(device, ccb) < 0) { 4920 warn("get_cgd: error sending Path Inquiry CCB"); 4921 if (arglist & CAM_ARG_VERBOSE) 4922 cam_error_print(device, ccb, CAM_ESF_ALL, 4923 CAM_EPF_ALL, stderr); 4924 retval = 1; 4925 goto get_cgd_bailout; 4926 } 4927 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 4928 if (arglist & CAM_ARG_VERBOSE) 4929 cam_error_print(device, ccb, CAM_ESF_ALL, 4930 CAM_EPF_ALL, stderr); 4931 retval = 1; 4932 goto get_cgd_bailout; 4933 } 4934 bcopy(&ccb->cgd, cgd, sizeof(struct ccb_getdev)); 4935 4936 get_cgd_bailout: 4937 cam_freeccb(ccb); 4938 return(retval); 4939 } 4940 4941 /* 4942 * Returns 1 if the device has the VPD page, 0 if it does not, and -1 on an 4943 * error. 4944 */ 4945 int 4946 dev_has_vpd_page(struct cam_device *dev, uint8_t page_id, int retry_count, 4947 int timeout, int verbosemode) 4948 { 4949 union ccb *ccb = NULL; 4950 struct scsi_vpd_supported_page_list sup_pages; 4951 int i; 4952 int retval = 0; 4953 4954 ccb = cam_getccb(dev); 4955 if (ccb == NULL) { 4956 warn("Unable to allocate CCB"); 4957 retval = -1; 4958 goto bailout; 4959 } 4960 4961 /* cam_getccb cleans up the header, caller has to zero the payload */ 4962 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 4963 4964 bzero(&sup_pages, sizeof(sup_pages)); 4965 4966 scsi_inquiry(&ccb->csio, 4967 /*retries*/ retry_count, 4968 /*cbfcnp*/ NULL, 4969 /* tag_action */ MSG_SIMPLE_Q_TAG, 4970 /* inq_buf */ (u_int8_t *)&sup_pages, 4971 /* inq_len */ sizeof(sup_pages), 4972 /* evpd */ 1, 4973 /* page_code */ SVPD_SUPPORTED_PAGE_LIST, 4974 /* sense_len */ SSD_FULL_SIZE, 4975 /* timeout */ timeout ? timeout : 5000); 4976 4977 /* Disable freezing the device queue */ 4978 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 4979 4980 if (retry_count != 0) 4981 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 4982 4983 if (cam_send_ccb(dev, ccb) < 0) { 4984 cam_freeccb(ccb); 4985 ccb = NULL; 4986 retval = -1; 4987 goto bailout; 4988 } 4989 4990 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 4991 if (verbosemode != 0) 4992 cam_error_print(dev, ccb, CAM_ESF_ALL, 4993 CAM_EPF_ALL, stderr); 4994 retval = -1; 4995 goto bailout; 4996 } 4997 4998 for (i = 0; i < sup_pages.length; i++) { 4999 if (sup_pages.list[i] == page_id) { 5000 retval = 1; 5001 goto bailout; 5002 } 5003 } 5004 bailout: 5005 if (ccb != NULL) 5006 cam_freeccb(ccb); 5007 5008 return (retval); 5009 } 5010 5011 /* 5012 * devtype is filled in with the type of device. 5013 * Returns 0 for success, non-zero for failure. 5014 */ 5015 int 5016 get_device_type(struct cam_device *dev, int retry_count, int timeout, 5017 int verbosemode, camcontrol_devtype *devtype) 5018 { 5019 struct ccb_getdev cgd; 5020 int retval = 0; 5021 5022 retval = get_cgd(dev, &cgd); 5023 if (retval != 0) 5024 goto bailout; 5025 5026 switch (cgd.protocol) { 5027 case PROTO_SCSI: 5028 break; 5029 case PROTO_ATA: 5030 case PROTO_ATAPI: 5031 case PROTO_SATAPM: 5032 *devtype = CC_DT_ATA; 5033 goto bailout; 5034 break; /*NOTREACHED*/ 5035 default: 5036 *devtype = CC_DT_UNKNOWN; 5037 goto bailout; 5038 break; /*NOTREACHED*/ 5039 } 5040 5041 /* 5042 * Check for the ATA Information VPD page (0x89). If this is an 5043 * ATA device behind a SCSI to ATA translation layer, this VPD page 5044 * should be present. 5045 * 5046 * If that VPD page isn't present, or we get an error back from the 5047 * INQUIRY command, we'll just treat it as a normal SCSI device. 5048 */ 5049 retval = dev_has_vpd_page(dev, SVPD_ATA_INFORMATION, retry_count, 5050 timeout, verbosemode); 5051 if (retval == 1) 5052 *devtype = CC_DT_ATA_BEHIND_SCSI; 5053 else 5054 *devtype = CC_DT_SCSI; 5055 5056 retval = 0; 5057 5058 bailout: 5059 return (retval); 5060 } 5061 5062 int 5063 build_ata_cmd(union ccb *ccb, uint32_t retry_count, uint32_t flags, 5064 uint8_t tag_action, uint8_t protocol, uint8_t ata_flags, uint16_t features, 5065 uint16_t sector_count, uint64_t lba, uint8_t command, uint32_t auxiliary, 5066 uint8_t *data_ptr, uint32_t dxfer_len, uint8_t *cdb_storage, 5067 size_t cdb_storage_len, uint8_t sense_len, uint32_t timeout, 5068 int is48bit, camcontrol_devtype devtype) 5069 { 5070 int retval = 0; 5071 5072 if (devtype == CC_DT_ATA) { 5073 cam_fill_ataio(&ccb->ataio, 5074 /*retries*/ retry_count, 5075 /*cbfcnp*/ NULL, 5076 /*flags*/ flags, 5077 /*tag_action*/ tag_action, 5078 /*data_ptr*/ data_ptr, 5079 /*dxfer_len*/ dxfer_len, 5080 /*timeout*/ timeout); 5081 if (is48bit || lba > ATA_MAX_28BIT_LBA) 5082 ata_48bit_cmd(&ccb->ataio, command, features, lba, 5083 sector_count); 5084 else 5085 ata_28bit_cmd(&ccb->ataio, command, features, lba, 5086 sector_count); 5087 5088 if (auxiliary != 0) { 5089 ccb->ataio.ata_flags |= ATA_FLAG_AUX; 5090 ccb->ataio.aux = auxiliary; 5091 } 5092 5093 if (ata_flags & AP_FLAG_CHK_COND) 5094 ccb->ataio.cmd.flags |= CAM_ATAIO_NEEDRESULT; 5095 5096 if ((protocol & AP_PROTO_MASK) == AP_PROTO_DMA) 5097 ccb->ataio.cmd.flags |= CAM_ATAIO_DMA; 5098 else if ((protocol & AP_PROTO_MASK) == AP_PROTO_FPDMA) 5099 ccb->ataio.cmd.flags |= CAM_ATAIO_FPDMA; 5100 } else { 5101 if (is48bit || lba > ATA_MAX_28BIT_LBA) 5102 protocol |= AP_EXTEND; 5103 5104 retval = scsi_ata_pass(&ccb->csio, 5105 /*retries*/ retry_count, 5106 /*cbfcnp*/ NULL, 5107 /*flags*/ flags, 5108 /*tag_action*/ tag_action, 5109 /*protocol*/ protocol, 5110 /*ata_flags*/ ata_flags, 5111 /*features*/ features, 5112 /*sector_count*/ sector_count, 5113 /*lba*/ lba, 5114 /*command*/ command, 5115 /*device*/ 0, 5116 /*icc*/ 0, 5117 /*auxiliary*/ auxiliary, 5118 /*control*/ 0, 5119 /*data_ptr*/ data_ptr, 5120 /*dxfer_len*/ dxfer_len, 5121 /*cdb_storage*/ cdb_storage, 5122 /*cdb_storage_len*/ cdb_storage_len, 5123 /*minimum_cmd_size*/ 0, 5124 /*sense_len*/ sense_len, 5125 /*timeout*/ timeout); 5126 } 5127 5128 return (retval); 5129 } 5130 5131 int 5132 get_ata_status(struct cam_device *dev, union ccb *ccb, uint8_t *error, 5133 uint16_t *count, uint64_t *lba, uint8_t *device, uint8_t *status) 5134 { 5135 int retval = 0; 5136 5137 switch (ccb->ccb_h.func_code) { 5138 case XPT_SCSI_IO: { 5139 uint8_t opcode; 5140 int error_code = 0, sense_key = 0, asc = 0, ascq = 0; 5141 5142 /* 5143 * In this case, we have SCSI ATA PASS-THROUGH command, 12 5144 * or 16 byte, and need to see what 5145 */ 5146 if (ccb->ccb_h.flags & CAM_CDB_POINTER) 5147 opcode = ccb->csio.cdb_io.cdb_ptr[0]; 5148 else 5149 opcode = ccb->csio.cdb_io.cdb_bytes[0]; 5150 if ((opcode != ATA_PASS_12) 5151 && (opcode != ATA_PASS_16)) { 5152 retval = 1; 5153 warnx("%s: unsupported opcode %02x", __func__, opcode); 5154 goto bailout; 5155 } 5156 5157 retval = scsi_extract_sense_ccb(ccb, &error_code, &sense_key, 5158 &asc, &ascq); 5159 /* Note: the _ccb() variant returns 0 for an error */ 5160 if (retval == 0) { 5161 retval = 1; 5162 goto bailout; 5163 } else 5164 retval = 0; 5165 5166 switch (error_code) { 5167 case SSD_DESC_CURRENT_ERROR: 5168 case SSD_DESC_DEFERRED_ERROR: { 5169 struct scsi_sense_data_desc *sense; 5170 struct scsi_sense_ata_ret_desc *desc; 5171 uint8_t *desc_ptr; 5172 5173 sense = (struct scsi_sense_data_desc *) 5174 &ccb->csio.sense_data; 5175 5176 desc_ptr = scsi_find_desc(sense, ccb->csio.sense_len - 5177 ccb->csio.sense_resid, SSD_DESC_ATA); 5178 if (desc_ptr == NULL) { 5179 cam_error_print(dev, ccb, CAM_ESF_ALL, 5180 CAM_EPF_ALL, stderr); 5181 retval = 1; 5182 goto bailout; 5183 } 5184 desc = (struct scsi_sense_ata_ret_desc *)desc_ptr; 5185 5186 *error = desc->error; 5187 *count = (desc->count_15_8 << 8) | 5188 desc->count_7_0; 5189 *lba = ((uint64_t)desc->lba_47_40 << 40) | 5190 ((uint64_t)desc->lba_39_32 << 32) | 5191 (desc->lba_31_24 << 24) | 5192 (desc->lba_23_16 << 16) | 5193 (desc->lba_15_8 << 8) | 5194 desc->lba_7_0; 5195 *device = desc->device; 5196 *status = desc->status; 5197 5198 /* 5199 * If the extend bit isn't set, the result is for a 5200 * 12-byte ATA PASS-THROUGH command or a 16 or 32 byte 5201 * command without the extend bit set. This means 5202 * that the device is supposed to return 28-bit 5203 * status. The count field is only 8 bits, and the 5204 * LBA field is only 8 bits. 5205 */ 5206 if ((desc->flags & SSD_DESC_ATA_FLAG_EXTEND) == 0){ 5207 *count &= 0xff; 5208 *lba &= 0x0fffffff; 5209 } 5210 break; 5211 } 5212 case SSD_CURRENT_ERROR: 5213 case SSD_DEFERRED_ERROR: { 5214 #if 0 5215 struct scsi_sense_data_fixed *sense; 5216 #endif 5217 /* 5218 * XXX KDM need to support fixed sense data. 5219 */ 5220 warnx("%s: Fixed sense data not supported yet", 5221 __func__); 5222 retval = 1; 5223 goto bailout; 5224 break; /*NOTREACHED*/ 5225 } 5226 default: 5227 retval = 1; 5228 goto bailout; 5229 break; 5230 } 5231 5232 break; 5233 } 5234 case XPT_ATA_IO: { 5235 struct ata_res *res; 5236 5237 /* 5238 * In this case, we have an ATA command, and we need to 5239 * fill in the requested values from the result register 5240 * set. 5241 */ 5242 res = &ccb->ataio.res; 5243 *error = res->error; 5244 *status = res->status; 5245 *device = res->device; 5246 *count = res->sector_count; 5247 *lba = (res->lba_high << 16) | 5248 (res->lba_mid << 8) | 5249 (res->lba_low); 5250 if (res->flags & CAM_ATAIO_48BIT) { 5251 *count |= (res->sector_count_exp << 8); 5252 *lba |= (res->lba_low_exp << 24) | 5253 ((uint64_t)res->lba_mid_exp << 32) | 5254 ((uint64_t)res->lba_high_exp << 40); 5255 } else { 5256 *lba |= (res->device & 0xf) << 24; 5257 } 5258 break; 5259 } 5260 default: 5261 retval = 1; 5262 break; 5263 } 5264 bailout: 5265 return (retval); 5266 } 5267 5268 static void 5269 cpi_print(struct ccb_pathinq *cpi) 5270 { 5271 char adapter_str[1024]; 5272 int i; 5273 5274 snprintf(adapter_str, sizeof(adapter_str), 5275 "%s%d:", cpi->dev_name, cpi->unit_number); 5276 5277 fprintf(stdout, "%s SIM/HBA version: %d\n", adapter_str, 5278 cpi->version_num); 5279 5280 for (i = 1; i < 0xff; i = i << 1) { 5281 const char *str; 5282 5283 if ((i & cpi->hba_inquiry) == 0) 5284 continue; 5285 5286 fprintf(stdout, "%s supports ", adapter_str); 5287 5288 switch(i) { 5289 case PI_MDP_ABLE: 5290 str = "MDP message"; 5291 break; 5292 case PI_WIDE_32: 5293 str = "32 bit wide SCSI"; 5294 break; 5295 case PI_WIDE_16: 5296 str = "16 bit wide SCSI"; 5297 break; 5298 case PI_SDTR_ABLE: 5299 str = "SDTR message"; 5300 break; 5301 case PI_LINKED_CDB: 5302 str = "linked CDBs"; 5303 break; 5304 case PI_TAG_ABLE: 5305 str = "tag queue messages"; 5306 break; 5307 case PI_SOFT_RST: 5308 str = "soft reset alternative"; 5309 break; 5310 case PI_SATAPM: 5311 str = "SATA Port Multiplier"; 5312 break; 5313 default: 5314 str = "unknown PI bit set"; 5315 break; 5316 } 5317 fprintf(stdout, "%s\n", str); 5318 } 5319 5320 for (i = 1; i < 0xff; i = i << 1) { 5321 const char *str; 5322 5323 if ((i & cpi->hba_misc) == 0) 5324 continue; 5325 5326 fprintf(stdout, "%s ", adapter_str); 5327 5328 switch(i) { 5329 case PIM_SCANHILO: 5330 str = "bus scans from high ID to low ID"; 5331 break; 5332 case PIM_NOREMOVE: 5333 str = "removable devices not included in scan"; 5334 break; 5335 case PIM_NOINITIATOR: 5336 str = "initiator role not supported"; 5337 break; 5338 case PIM_NOBUSRESET: 5339 str = "user has disabled initial BUS RESET or" 5340 " controller is in target/mixed mode"; 5341 break; 5342 case PIM_NO_6_BYTE: 5343 str = "do not send 6-byte commands"; 5344 break; 5345 case PIM_SEQSCAN: 5346 str = "scan bus sequentially"; 5347 break; 5348 default: 5349 str = "unknown PIM bit set"; 5350 break; 5351 } 5352 fprintf(stdout, "%s\n", str); 5353 } 5354 5355 for (i = 1; i < 0xff; i = i << 1) { 5356 const char *str; 5357 5358 if ((i & cpi->target_sprt) == 0) 5359 continue; 5360 5361 fprintf(stdout, "%s supports ", adapter_str); 5362 switch(i) { 5363 case PIT_PROCESSOR: 5364 str = "target mode processor mode"; 5365 break; 5366 case PIT_PHASE: 5367 str = "target mode phase cog. mode"; 5368 break; 5369 case PIT_DISCONNECT: 5370 str = "disconnects in target mode"; 5371 break; 5372 case PIT_TERM_IO: 5373 str = "terminate I/O message in target mode"; 5374 break; 5375 case PIT_GRP_6: 5376 str = "group 6 commands in target mode"; 5377 break; 5378 case PIT_GRP_7: 5379 str = "group 7 commands in target mode"; 5380 break; 5381 default: 5382 str = "unknown PIT bit set"; 5383 break; 5384 } 5385 5386 fprintf(stdout, "%s\n", str); 5387 } 5388 fprintf(stdout, "%s HBA engine count: %d\n", adapter_str, 5389 cpi->hba_eng_cnt); 5390 fprintf(stdout, "%s maximum target: %d\n", adapter_str, 5391 cpi->max_target); 5392 fprintf(stdout, "%s maximum LUN: %d\n", adapter_str, 5393 cpi->max_lun); 5394 fprintf(stdout, "%s highest path ID in subsystem: %d\n", 5395 adapter_str, cpi->hpath_id); 5396 fprintf(stdout, "%s initiator ID: %d\n", adapter_str, 5397 cpi->initiator_id); 5398 fprintf(stdout, "%s SIM vendor: %s\n", adapter_str, cpi->sim_vid); 5399 fprintf(stdout, "%s HBA vendor: %s\n", adapter_str, cpi->hba_vid); 5400 fprintf(stdout, "%s HBA vendor ID: 0x%04x\n", 5401 adapter_str, cpi->hba_vendor); 5402 fprintf(stdout, "%s HBA device ID: 0x%04x\n", 5403 adapter_str, cpi->hba_device); 5404 fprintf(stdout, "%s HBA subvendor ID: 0x%04x\n", 5405 adapter_str, cpi->hba_subvendor); 5406 fprintf(stdout, "%s HBA subdevice ID: 0x%04x\n", 5407 adapter_str, cpi->hba_subdevice); 5408 fprintf(stdout, "%s bus ID: %d\n", adapter_str, cpi->bus_id); 5409 fprintf(stdout, "%s base transfer speed: ", adapter_str); 5410 if (cpi->base_transfer_speed > 1000) 5411 fprintf(stdout, "%d.%03dMB/sec\n", 5412 cpi->base_transfer_speed / 1000, 5413 cpi->base_transfer_speed % 1000); 5414 else 5415 fprintf(stdout, "%dKB/sec\n", 5416 (cpi->base_transfer_speed % 1000) * 1000); 5417 fprintf(stdout, "%s maximum transfer size: %u bytes\n", 5418 adapter_str, cpi->maxio); 5419 } 5420 5421 static int 5422 get_print_cts(struct cam_device *device, int user_settings, int quiet, 5423 struct ccb_trans_settings *cts) 5424 { 5425 int retval; 5426 union ccb *ccb; 5427 5428 retval = 0; 5429 ccb = cam_getccb(device); 5430 5431 if (ccb == NULL) { 5432 warnx("get_print_cts: error allocating ccb"); 5433 return(1); 5434 } 5435 5436 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->cts); 5437 5438 ccb->ccb_h.func_code = XPT_GET_TRAN_SETTINGS; 5439 5440 if (user_settings == 0) 5441 ccb->cts.type = CTS_TYPE_CURRENT_SETTINGS; 5442 else 5443 ccb->cts.type = CTS_TYPE_USER_SETTINGS; 5444 5445 if (cam_send_ccb(device, ccb) < 0) { 5446 perror("error sending XPT_GET_TRAN_SETTINGS CCB"); 5447 if (arglist & CAM_ARG_VERBOSE) 5448 cam_error_print(device, ccb, CAM_ESF_ALL, 5449 CAM_EPF_ALL, stderr); 5450 retval = 1; 5451 goto get_print_cts_bailout; 5452 } 5453 5454 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 5455 warnx("XPT_GET_TRANS_SETTINGS CCB failed"); 5456 if (arglist & CAM_ARG_VERBOSE) 5457 cam_error_print(device, ccb, CAM_ESF_ALL, 5458 CAM_EPF_ALL, stderr); 5459 retval = 1; 5460 goto get_print_cts_bailout; 5461 } 5462 5463 if (quiet == 0) 5464 cts_print(device, &ccb->cts); 5465 5466 if (cts != NULL) 5467 bcopy(&ccb->cts, cts, sizeof(struct ccb_trans_settings)); 5468 5469 get_print_cts_bailout: 5470 5471 cam_freeccb(ccb); 5472 5473 return(retval); 5474 } 5475 5476 static int 5477 ratecontrol(struct cam_device *device, int retry_count, int timeout, 5478 int argc, char **argv, char *combinedopt) 5479 { 5480 int c; 5481 union ccb *ccb; 5482 int user_settings = 0; 5483 int retval = 0; 5484 int disc_enable = -1, tag_enable = -1; 5485 int mode = -1; 5486 int offset = -1; 5487 double syncrate = -1; 5488 int bus_width = -1; 5489 int quiet = 0; 5490 int change_settings = 0, send_tur = 0; 5491 struct ccb_pathinq cpi; 5492 5493 ccb = cam_getccb(device); 5494 if (ccb == NULL) { 5495 warnx("ratecontrol: error allocating ccb"); 5496 return(1); 5497 } 5498 while ((c = getopt(argc, argv, combinedopt)) != -1) { 5499 switch(c){ 5500 case 'a': 5501 send_tur = 1; 5502 break; 5503 case 'c': 5504 user_settings = 0; 5505 break; 5506 case 'D': 5507 if (strncasecmp(optarg, "enable", 6) == 0) 5508 disc_enable = 1; 5509 else if (strncasecmp(optarg, "disable", 7) == 0) 5510 disc_enable = 0; 5511 else { 5512 warnx("-D argument \"%s\" is unknown", optarg); 5513 retval = 1; 5514 goto ratecontrol_bailout; 5515 } 5516 change_settings = 1; 5517 break; 5518 case 'M': 5519 mode = ata_string2mode(optarg); 5520 if (mode < 0) { 5521 warnx("unknown mode '%s'", optarg); 5522 retval = 1; 5523 goto ratecontrol_bailout; 5524 } 5525 change_settings = 1; 5526 break; 5527 case 'O': 5528 offset = strtol(optarg, NULL, 0); 5529 if (offset < 0) { 5530 warnx("offset value %d is < 0", offset); 5531 retval = 1; 5532 goto ratecontrol_bailout; 5533 } 5534 change_settings = 1; 5535 break; 5536 case 'q': 5537 quiet++; 5538 break; 5539 case 'R': 5540 syncrate = atof(optarg); 5541 if (syncrate < 0) { 5542 warnx("sync rate %f is < 0", syncrate); 5543 retval = 1; 5544 goto ratecontrol_bailout; 5545 } 5546 change_settings = 1; 5547 break; 5548 case 'T': 5549 if (strncasecmp(optarg, "enable", 6) == 0) 5550 tag_enable = 1; 5551 else if (strncasecmp(optarg, "disable", 7) == 0) 5552 tag_enable = 0; 5553 else { 5554 warnx("-T argument \"%s\" is unknown", optarg); 5555 retval = 1; 5556 goto ratecontrol_bailout; 5557 } 5558 change_settings = 1; 5559 break; 5560 case 'U': 5561 user_settings = 1; 5562 break; 5563 case 'W': 5564 bus_width = strtol(optarg, NULL, 0); 5565 if (bus_width < 0) { 5566 warnx("bus width %d is < 0", bus_width); 5567 retval = 1; 5568 goto ratecontrol_bailout; 5569 } 5570 change_settings = 1; 5571 break; 5572 default: 5573 break; 5574 } 5575 } 5576 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->cpi); 5577 /* 5578 * Grab path inquiry information, so we can determine whether 5579 * or not the initiator is capable of the things that the user 5580 * requests. 5581 */ 5582 ccb->ccb_h.func_code = XPT_PATH_INQ; 5583 if (cam_send_ccb(device, ccb) < 0) { 5584 perror("error sending XPT_PATH_INQ CCB"); 5585 if (arglist & CAM_ARG_VERBOSE) { 5586 cam_error_print(device, ccb, CAM_ESF_ALL, 5587 CAM_EPF_ALL, stderr); 5588 } 5589 retval = 1; 5590 goto ratecontrol_bailout; 5591 } 5592 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 5593 warnx("XPT_PATH_INQ CCB failed"); 5594 if (arglist & CAM_ARG_VERBOSE) { 5595 cam_error_print(device, ccb, CAM_ESF_ALL, 5596 CAM_EPF_ALL, stderr); 5597 } 5598 retval = 1; 5599 goto ratecontrol_bailout; 5600 } 5601 bcopy(&ccb->cpi, &cpi, sizeof(struct ccb_pathinq)); 5602 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->cts); 5603 if (quiet == 0) { 5604 fprintf(stdout, "%s parameters:\n", 5605 user_settings ? "User" : "Current"); 5606 } 5607 retval = get_print_cts(device, user_settings, quiet, &ccb->cts); 5608 if (retval != 0) 5609 goto ratecontrol_bailout; 5610 5611 if (arglist & CAM_ARG_VERBOSE) 5612 cpi_print(&cpi); 5613 5614 if (change_settings) { 5615 int didsettings = 0; 5616 struct ccb_trans_settings_spi *spi = NULL; 5617 struct ccb_trans_settings_pata *pata = NULL; 5618 struct ccb_trans_settings_sata *sata = NULL; 5619 struct ccb_trans_settings_ata *ata = NULL; 5620 struct ccb_trans_settings_scsi *scsi = NULL; 5621 5622 if (ccb->cts.transport == XPORT_SPI) 5623 spi = &ccb->cts.xport_specific.spi; 5624 if (ccb->cts.transport == XPORT_ATA) 5625 pata = &ccb->cts.xport_specific.ata; 5626 if (ccb->cts.transport == XPORT_SATA) 5627 sata = &ccb->cts.xport_specific.sata; 5628 if (ccb->cts.protocol == PROTO_ATA) 5629 ata = &ccb->cts.proto_specific.ata; 5630 if (ccb->cts.protocol == PROTO_SCSI) 5631 scsi = &ccb->cts.proto_specific.scsi; 5632 ccb->cts.xport_specific.valid = 0; 5633 ccb->cts.proto_specific.valid = 0; 5634 if (spi && disc_enable != -1) { 5635 spi->valid |= CTS_SPI_VALID_DISC; 5636 if (disc_enable == 0) 5637 spi->flags &= ~CTS_SPI_FLAGS_DISC_ENB; 5638 else 5639 spi->flags |= CTS_SPI_FLAGS_DISC_ENB; 5640 didsettings++; 5641 } 5642 if (tag_enable != -1) { 5643 if ((cpi.hba_inquiry & PI_TAG_ABLE) == 0) { 5644 warnx("HBA does not support tagged queueing, " 5645 "so you cannot modify tag settings"); 5646 retval = 1; 5647 goto ratecontrol_bailout; 5648 } 5649 if (ata) { 5650 ata->valid |= CTS_SCSI_VALID_TQ; 5651 if (tag_enable == 0) 5652 ata->flags &= ~CTS_ATA_FLAGS_TAG_ENB; 5653 else 5654 ata->flags |= CTS_ATA_FLAGS_TAG_ENB; 5655 didsettings++; 5656 } else if (scsi) { 5657 scsi->valid |= CTS_SCSI_VALID_TQ; 5658 if (tag_enable == 0) 5659 scsi->flags &= ~CTS_SCSI_FLAGS_TAG_ENB; 5660 else 5661 scsi->flags |= CTS_SCSI_FLAGS_TAG_ENB; 5662 didsettings++; 5663 } 5664 } 5665 if (spi && offset != -1) { 5666 if ((cpi.hba_inquiry & PI_SDTR_ABLE) == 0) { 5667 warnx("HBA is not capable of changing offset"); 5668 retval = 1; 5669 goto ratecontrol_bailout; 5670 } 5671 spi->valid |= CTS_SPI_VALID_SYNC_OFFSET; 5672 spi->sync_offset = offset; 5673 didsettings++; 5674 } 5675 if (spi && syncrate != -1) { 5676 int prelim_sync_period; 5677 5678 if ((cpi.hba_inquiry & PI_SDTR_ABLE) == 0) { 5679 warnx("HBA is not capable of changing " 5680 "transfer rates"); 5681 retval = 1; 5682 goto ratecontrol_bailout; 5683 } 5684 spi->valid |= CTS_SPI_VALID_SYNC_RATE; 5685 /* 5686 * The sync rate the user gives us is in MHz. 5687 * We need to translate it into KHz for this 5688 * calculation. 5689 */ 5690 syncrate *= 1000; 5691 /* 5692 * Next, we calculate a "preliminary" sync period 5693 * in tenths of a nanosecond. 5694 */ 5695 if (syncrate == 0) 5696 prelim_sync_period = 0; 5697 else 5698 prelim_sync_period = 10000000 / syncrate; 5699 spi->sync_period = 5700 scsi_calc_syncparam(prelim_sync_period); 5701 didsettings++; 5702 } 5703 if (sata && syncrate != -1) { 5704 if ((cpi.hba_inquiry & PI_SDTR_ABLE) == 0) { 5705 warnx("HBA is not capable of changing " 5706 "transfer rates"); 5707 retval = 1; 5708 goto ratecontrol_bailout; 5709 } 5710 if (!user_settings) { 5711 warnx("You can modify only user rate " 5712 "settings for SATA"); 5713 retval = 1; 5714 goto ratecontrol_bailout; 5715 } 5716 sata->revision = ata_speed2revision(syncrate * 100); 5717 if (sata->revision < 0) { 5718 warnx("Invalid rate %f", syncrate); 5719 retval = 1; 5720 goto ratecontrol_bailout; 5721 } 5722 sata->valid |= CTS_SATA_VALID_REVISION; 5723 didsettings++; 5724 } 5725 if ((pata || sata) && mode != -1) { 5726 if ((cpi.hba_inquiry & PI_SDTR_ABLE) == 0) { 5727 warnx("HBA is not capable of changing " 5728 "transfer rates"); 5729 retval = 1; 5730 goto ratecontrol_bailout; 5731 } 5732 if (!user_settings) { 5733 warnx("You can modify only user mode " 5734 "settings for ATA/SATA"); 5735 retval = 1; 5736 goto ratecontrol_bailout; 5737 } 5738 if (pata) { 5739 pata->mode = mode; 5740 pata->valid |= CTS_ATA_VALID_MODE; 5741 } else { 5742 sata->mode = mode; 5743 sata->valid |= CTS_SATA_VALID_MODE; 5744 } 5745 didsettings++; 5746 } 5747 /* 5748 * The bus_width argument goes like this: 5749 * 0 == 8 bit 5750 * 1 == 16 bit 5751 * 2 == 32 bit 5752 * Therefore, if you shift the number of bits given on the 5753 * command line right by 4, you should get the correct 5754 * number. 5755 */ 5756 if (spi && bus_width != -1) { 5757 /* 5758 * We might as well validate things here with a 5759 * decipherable error message, rather than what 5760 * will probably be an indecipherable error message 5761 * by the time it gets back to us. 5762 */ 5763 if ((bus_width == 16) 5764 && ((cpi.hba_inquiry & PI_WIDE_16) == 0)) { 5765 warnx("HBA does not support 16 bit bus width"); 5766 retval = 1; 5767 goto ratecontrol_bailout; 5768 } else if ((bus_width == 32) 5769 && ((cpi.hba_inquiry & PI_WIDE_32) == 0)) { 5770 warnx("HBA does not support 32 bit bus width"); 5771 retval = 1; 5772 goto ratecontrol_bailout; 5773 } else if ((bus_width != 8) 5774 && (bus_width != 16) 5775 && (bus_width != 32)) { 5776 warnx("Invalid bus width %d", bus_width); 5777 retval = 1; 5778 goto ratecontrol_bailout; 5779 } 5780 spi->valid |= CTS_SPI_VALID_BUS_WIDTH; 5781 spi->bus_width = bus_width >> 4; 5782 didsettings++; 5783 } 5784 if (didsettings == 0) { 5785 goto ratecontrol_bailout; 5786 } 5787 ccb->ccb_h.func_code = XPT_SET_TRAN_SETTINGS; 5788 if (cam_send_ccb(device, ccb) < 0) { 5789 perror("error sending XPT_SET_TRAN_SETTINGS CCB"); 5790 if (arglist & CAM_ARG_VERBOSE) { 5791 cam_error_print(device, ccb, CAM_ESF_ALL, 5792 CAM_EPF_ALL, stderr); 5793 } 5794 retval = 1; 5795 goto ratecontrol_bailout; 5796 } 5797 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 5798 warnx("XPT_SET_TRANS_SETTINGS CCB failed"); 5799 if (arglist & CAM_ARG_VERBOSE) { 5800 cam_error_print(device, ccb, CAM_ESF_ALL, 5801 CAM_EPF_ALL, stderr); 5802 } 5803 retval = 1; 5804 goto ratecontrol_bailout; 5805 } 5806 } 5807 if (send_tur) { 5808 retval = testunitready(device, retry_count, timeout, 5809 (arglist & CAM_ARG_VERBOSE) ? 0 : 1); 5810 /* 5811 * If the TUR didn't succeed, just bail. 5812 */ 5813 if (retval != 0) { 5814 if (quiet == 0) 5815 fprintf(stderr, "Test Unit Ready failed\n"); 5816 goto ratecontrol_bailout; 5817 } 5818 } 5819 if ((change_settings || send_tur) && !quiet && 5820 (ccb->cts.transport == XPORT_ATA || 5821 ccb->cts.transport == XPORT_SATA || send_tur)) { 5822 fprintf(stdout, "New parameters:\n"); 5823 retval = get_print_cts(device, user_settings, 0, NULL); 5824 } 5825 5826 ratecontrol_bailout: 5827 cam_freeccb(ccb); 5828 return(retval); 5829 } 5830 5831 static int 5832 scsiformat(struct cam_device *device, int argc, char **argv, 5833 char *combinedopt, int retry_count, int timeout) 5834 { 5835 union ccb *ccb; 5836 int c; 5837 int ycount = 0, quiet = 0; 5838 int error = 0, retval = 0; 5839 int use_timeout = 10800 * 1000; 5840 int immediate = 1; 5841 struct format_defect_list_header fh; 5842 u_int8_t *data_ptr = NULL; 5843 u_int32_t dxfer_len = 0; 5844 u_int8_t byte2 = 0; 5845 int num_warnings = 0; 5846 int reportonly = 0; 5847 5848 ccb = cam_getccb(device); 5849 5850 if (ccb == NULL) { 5851 warnx("scsiformat: error allocating ccb"); 5852 return(1); 5853 } 5854 5855 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 5856 5857 while ((c = getopt(argc, argv, combinedopt)) != -1) { 5858 switch(c) { 5859 case 'q': 5860 quiet++; 5861 break; 5862 case 'r': 5863 reportonly = 1; 5864 break; 5865 case 'w': 5866 immediate = 0; 5867 break; 5868 case 'y': 5869 ycount++; 5870 break; 5871 } 5872 } 5873 5874 if (reportonly) 5875 goto doreport; 5876 5877 if (quiet == 0) { 5878 fprintf(stdout, "You are about to REMOVE ALL DATA from the " 5879 "following device:\n"); 5880 5881 error = scsidoinquiry(device, argc, argv, combinedopt, 5882 retry_count, timeout); 5883 5884 if (error != 0) { 5885 warnx("scsiformat: error sending inquiry"); 5886 goto scsiformat_bailout; 5887 } 5888 } 5889 5890 if (ycount == 0) { 5891 if (!get_confirmation()) { 5892 error = 1; 5893 goto scsiformat_bailout; 5894 } 5895 } 5896 5897 if (timeout != 0) 5898 use_timeout = timeout; 5899 5900 if (quiet == 0) { 5901 fprintf(stdout, "Current format timeout is %d seconds\n", 5902 use_timeout / 1000); 5903 } 5904 5905 /* 5906 * If the user hasn't disabled questions and didn't specify a 5907 * timeout on the command line, ask them if they want the current 5908 * timeout. 5909 */ 5910 if ((ycount == 0) 5911 && (timeout == 0)) { 5912 char str[1024]; 5913 int new_timeout = 0; 5914 5915 fprintf(stdout, "Enter new timeout in seconds or press\n" 5916 "return to keep the current timeout [%d] ", 5917 use_timeout / 1000); 5918 5919 if (fgets(str, sizeof(str), stdin) != NULL) { 5920 if (str[0] != '\0') 5921 new_timeout = atoi(str); 5922 } 5923 5924 if (new_timeout != 0) { 5925 use_timeout = new_timeout * 1000; 5926 fprintf(stdout, "Using new timeout value %d\n", 5927 use_timeout / 1000); 5928 } 5929 } 5930 5931 /* 5932 * Keep this outside the if block below to silence any unused 5933 * variable warnings. 5934 */ 5935 bzero(&fh, sizeof(fh)); 5936 5937 /* 5938 * If we're in immediate mode, we've got to include the format 5939 * header 5940 */ 5941 if (immediate != 0) { 5942 fh.byte2 = FU_DLH_IMMED; 5943 data_ptr = (u_int8_t *)&fh; 5944 dxfer_len = sizeof(fh); 5945 byte2 = FU_FMT_DATA; 5946 } else if (quiet == 0) { 5947 fprintf(stdout, "Formatting..."); 5948 fflush(stdout); 5949 } 5950 5951 scsi_format_unit(&ccb->csio, 5952 /* retries */ retry_count, 5953 /* cbfcnp */ NULL, 5954 /* tag_action */ MSG_SIMPLE_Q_TAG, 5955 /* byte2 */ byte2, 5956 /* ileave */ 0, 5957 /* data_ptr */ data_ptr, 5958 /* dxfer_len */ dxfer_len, 5959 /* sense_len */ SSD_FULL_SIZE, 5960 /* timeout */ use_timeout); 5961 5962 /* Disable freezing the device queue */ 5963 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 5964 5965 if (arglist & CAM_ARG_ERR_RECOVER) 5966 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 5967 5968 if (((retval = cam_send_ccb(device, ccb)) < 0) 5969 || ((immediate == 0) 5970 && ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP))) { 5971 const char errstr[] = "error sending format command"; 5972 5973 if (retval < 0) 5974 warn(errstr); 5975 else 5976 warnx(errstr); 5977 5978 if (arglist & CAM_ARG_VERBOSE) { 5979 cam_error_print(device, ccb, CAM_ESF_ALL, 5980 CAM_EPF_ALL, stderr); 5981 } 5982 error = 1; 5983 goto scsiformat_bailout; 5984 } 5985 5986 /* 5987 * If we ran in non-immediate mode, we already checked for errors 5988 * above and printed out any necessary information. If we're in 5989 * immediate mode, we need to loop through and get status 5990 * information periodically. 5991 */ 5992 if (immediate == 0) { 5993 if (quiet == 0) { 5994 fprintf(stdout, "Format Complete\n"); 5995 } 5996 goto scsiformat_bailout; 5997 } 5998 5999 doreport: 6000 do { 6001 cam_status status; 6002 6003 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 6004 6005 /* 6006 * There's really no need to do error recovery or 6007 * retries here, since we're just going to sit in a 6008 * loop and wait for the device to finish formatting. 6009 */ 6010 scsi_test_unit_ready(&ccb->csio, 6011 /* retries */ 0, 6012 /* cbfcnp */ NULL, 6013 /* tag_action */ MSG_SIMPLE_Q_TAG, 6014 /* sense_len */ SSD_FULL_SIZE, 6015 /* timeout */ 5000); 6016 6017 /* Disable freezing the device queue */ 6018 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 6019 6020 retval = cam_send_ccb(device, ccb); 6021 6022 /* 6023 * If we get an error from the ioctl, bail out. SCSI 6024 * errors are expected. 6025 */ 6026 if (retval < 0) { 6027 warn("error sending CAMIOCOMMAND ioctl"); 6028 if (arglist & CAM_ARG_VERBOSE) { 6029 cam_error_print(device, ccb, CAM_ESF_ALL, 6030 CAM_EPF_ALL, stderr); 6031 } 6032 error = 1; 6033 goto scsiformat_bailout; 6034 } 6035 6036 status = ccb->ccb_h.status & CAM_STATUS_MASK; 6037 6038 if ((status != CAM_REQ_CMP) 6039 && (status == CAM_SCSI_STATUS_ERROR) 6040 && ((ccb->ccb_h.status & CAM_AUTOSNS_VALID) != 0)) { 6041 struct scsi_sense_data *sense; 6042 int error_code, sense_key, asc, ascq; 6043 6044 sense = &ccb->csio.sense_data; 6045 scsi_extract_sense_len(sense, ccb->csio.sense_len - 6046 ccb->csio.sense_resid, &error_code, &sense_key, 6047 &asc, &ascq, /*show_errors*/ 1); 6048 6049 /* 6050 * According to the SCSI-2 and SCSI-3 specs, a 6051 * drive that is in the middle of a format should 6052 * return NOT READY with an ASC of "logical unit 6053 * not ready, format in progress". The sense key 6054 * specific bytes will then be a progress indicator. 6055 */ 6056 if ((sense_key == SSD_KEY_NOT_READY) 6057 && (asc == 0x04) && (ascq == 0x04)) { 6058 uint8_t sks[3]; 6059 6060 if ((scsi_get_sks(sense, ccb->csio.sense_len - 6061 ccb->csio.sense_resid, sks) == 0) 6062 && (quiet == 0)) { 6063 int val; 6064 u_int64_t percentage; 6065 6066 val = scsi_2btoul(&sks[1]); 6067 percentage = 10000 * val; 6068 6069 fprintf(stdout, 6070 "\rFormatting: %ju.%02u %% " 6071 "(%d/%d) done", 6072 (uintmax_t)(percentage / 6073 (0x10000 * 100)), 6074 (unsigned)((percentage / 6075 0x10000) % 100), 6076 val, 0x10000); 6077 fflush(stdout); 6078 } else if ((quiet == 0) 6079 && (++num_warnings <= 1)) { 6080 warnx("Unexpected SCSI Sense Key " 6081 "Specific value returned " 6082 "during format:"); 6083 scsi_sense_print(device, &ccb->csio, 6084 stderr); 6085 warnx("Unable to print status " 6086 "information, but format will " 6087 "proceed."); 6088 warnx("will exit when format is " 6089 "complete"); 6090 } 6091 sleep(1); 6092 } else { 6093 warnx("Unexpected SCSI error during format"); 6094 cam_error_print(device, ccb, CAM_ESF_ALL, 6095 CAM_EPF_ALL, stderr); 6096 error = 1; 6097 goto scsiformat_bailout; 6098 } 6099 6100 } else if (status != CAM_REQ_CMP) { 6101 warnx("Unexpected CAM status %#x", status); 6102 if (arglist & CAM_ARG_VERBOSE) 6103 cam_error_print(device, ccb, CAM_ESF_ALL, 6104 CAM_EPF_ALL, stderr); 6105 error = 1; 6106 goto scsiformat_bailout; 6107 } 6108 6109 } while((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP); 6110 6111 if (quiet == 0) 6112 fprintf(stdout, "\nFormat Complete\n"); 6113 6114 scsiformat_bailout: 6115 6116 cam_freeccb(ccb); 6117 6118 return(error); 6119 } 6120 6121 static int 6122 scsisanitize(struct cam_device *device, int argc, char **argv, 6123 char *combinedopt, int retry_count, int timeout) 6124 { 6125 union ccb *ccb; 6126 u_int8_t action = 0; 6127 int c; 6128 int ycount = 0, quiet = 0; 6129 int error = 0, retval = 0; 6130 int use_timeout = 10800 * 1000; 6131 int immediate = 1; 6132 int invert = 0; 6133 int passes = 0; 6134 int ause = 0; 6135 int fd = -1; 6136 const char *pattern = NULL; 6137 u_int8_t *data_ptr = NULL; 6138 u_int32_t dxfer_len = 0; 6139 u_int8_t byte2 = 0; 6140 int num_warnings = 0; 6141 int reportonly = 0; 6142 6143 ccb = cam_getccb(device); 6144 6145 if (ccb == NULL) { 6146 warnx("scsisanitize: error allocating ccb"); 6147 return(1); 6148 } 6149 6150 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 6151 6152 while ((c = getopt(argc, argv, combinedopt)) != -1) { 6153 switch(c) { 6154 case 'a': 6155 if (strcasecmp(optarg, "overwrite") == 0) 6156 action = SSZ_SERVICE_ACTION_OVERWRITE; 6157 else if (strcasecmp(optarg, "block") == 0) 6158 action = SSZ_SERVICE_ACTION_BLOCK_ERASE; 6159 else if (strcasecmp(optarg, "crypto") == 0) 6160 action = SSZ_SERVICE_ACTION_CRYPTO_ERASE; 6161 else if (strcasecmp(optarg, "exitfailure") == 0) 6162 action = SSZ_SERVICE_ACTION_EXIT_MODE_FAILURE; 6163 else { 6164 warnx("invalid service operation \"%s\"", 6165 optarg); 6166 error = 1; 6167 goto scsisanitize_bailout; 6168 } 6169 break; 6170 case 'c': 6171 passes = strtol(optarg, NULL, 0); 6172 if (passes < 1 || passes > 31) { 6173 warnx("invalid passes value %d", passes); 6174 error = 1; 6175 goto scsisanitize_bailout; 6176 } 6177 break; 6178 case 'I': 6179 invert = 1; 6180 break; 6181 case 'P': 6182 pattern = optarg; 6183 break; 6184 case 'q': 6185 quiet++; 6186 break; 6187 case 'U': 6188 ause = 1; 6189 break; 6190 case 'r': 6191 reportonly = 1; 6192 break; 6193 case 'w': 6194 immediate = 0; 6195 break; 6196 case 'y': 6197 ycount++; 6198 break; 6199 } 6200 } 6201 6202 if (reportonly) 6203 goto doreport; 6204 6205 if (action == 0) { 6206 warnx("an action is required"); 6207 error = 1; 6208 goto scsisanitize_bailout; 6209 } else if (action == SSZ_SERVICE_ACTION_OVERWRITE) { 6210 struct scsi_sanitize_parameter_list *pl; 6211 struct stat sb; 6212 ssize_t sz, amt; 6213 6214 if (pattern == NULL) { 6215 warnx("overwrite action requires -P argument"); 6216 error = 1; 6217 goto scsisanitize_bailout; 6218 } 6219 fd = open(pattern, O_RDONLY); 6220 if (fd < 0) { 6221 warn("cannot open pattern file %s", pattern); 6222 error = 1; 6223 goto scsisanitize_bailout; 6224 } 6225 if (fstat(fd, &sb) < 0) { 6226 warn("cannot stat pattern file %s", pattern); 6227 error = 1; 6228 goto scsisanitize_bailout; 6229 } 6230 sz = sb.st_size; 6231 if (sz > SSZPL_MAX_PATTERN_LENGTH) { 6232 warnx("pattern file size exceeds maximum value %d", 6233 SSZPL_MAX_PATTERN_LENGTH); 6234 error = 1; 6235 goto scsisanitize_bailout; 6236 } 6237 dxfer_len = sizeof(*pl) + sz; 6238 data_ptr = calloc(1, dxfer_len); 6239 if (data_ptr == NULL) { 6240 warnx("cannot allocate parameter list buffer"); 6241 error = 1; 6242 goto scsisanitize_bailout; 6243 } 6244 6245 amt = read(fd, data_ptr + sizeof(*pl), sz); 6246 if (amt < 0) { 6247 warn("cannot read pattern file"); 6248 error = 1; 6249 goto scsisanitize_bailout; 6250 } else if (amt != sz) { 6251 warnx("short pattern file read"); 6252 error = 1; 6253 goto scsisanitize_bailout; 6254 } 6255 6256 pl = (struct scsi_sanitize_parameter_list *)data_ptr; 6257 if (passes == 0) 6258 pl->byte1 = 1; 6259 else 6260 pl->byte1 = passes; 6261 if (invert != 0) 6262 pl->byte1 |= SSZPL_INVERT; 6263 scsi_ulto2b(sz, pl->length); 6264 } else { 6265 const char *arg; 6266 6267 if (passes != 0) 6268 arg = "-c"; 6269 else if (invert != 0) 6270 arg = "-I"; 6271 else if (pattern != NULL) 6272 arg = "-P"; 6273 else 6274 arg = NULL; 6275 if (arg != NULL) { 6276 warnx("%s argument only valid with overwrite " 6277 "operation", arg); 6278 error = 1; 6279 goto scsisanitize_bailout; 6280 } 6281 } 6282 6283 if (quiet == 0) { 6284 fprintf(stdout, "You are about to REMOVE ALL DATA from the " 6285 "following device:\n"); 6286 6287 error = scsidoinquiry(device, argc, argv, combinedopt, 6288 retry_count, timeout); 6289 6290 if (error != 0) { 6291 warnx("scsisanitize: error sending inquiry"); 6292 goto scsisanitize_bailout; 6293 } 6294 } 6295 6296 if (ycount == 0) { 6297 if (!get_confirmation()) { 6298 error = 1; 6299 goto scsisanitize_bailout; 6300 } 6301 } 6302 6303 if (timeout != 0) 6304 use_timeout = timeout; 6305 6306 if (quiet == 0) { 6307 fprintf(stdout, "Current sanitize timeout is %d seconds\n", 6308 use_timeout / 1000); 6309 } 6310 6311 /* 6312 * If the user hasn't disabled questions and didn't specify a 6313 * timeout on the command line, ask them if they want the current 6314 * timeout. 6315 */ 6316 if ((ycount == 0) 6317 && (timeout == 0)) { 6318 char str[1024]; 6319 int new_timeout = 0; 6320 6321 fprintf(stdout, "Enter new timeout in seconds or press\n" 6322 "return to keep the current timeout [%d] ", 6323 use_timeout / 1000); 6324 6325 if (fgets(str, sizeof(str), stdin) != NULL) { 6326 if (str[0] != '\0') 6327 new_timeout = atoi(str); 6328 } 6329 6330 if (new_timeout != 0) { 6331 use_timeout = new_timeout * 1000; 6332 fprintf(stdout, "Using new timeout value %d\n", 6333 use_timeout / 1000); 6334 } 6335 } 6336 6337 byte2 = action; 6338 if (ause != 0) 6339 byte2 |= SSZ_UNRESTRICTED_EXIT; 6340 if (immediate != 0) 6341 byte2 |= SSZ_IMMED; 6342 6343 scsi_sanitize(&ccb->csio, 6344 /* retries */ retry_count, 6345 /* cbfcnp */ NULL, 6346 /* tag_action */ MSG_SIMPLE_Q_TAG, 6347 /* byte2 */ byte2, 6348 /* control */ 0, 6349 /* data_ptr */ data_ptr, 6350 /* dxfer_len */ dxfer_len, 6351 /* sense_len */ SSD_FULL_SIZE, 6352 /* timeout */ use_timeout); 6353 6354 /* Disable freezing the device queue */ 6355 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 6356 6357 if (arglist & CAM_ARG_ERR_RECOVER) 6358 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 6359 6360 if (cam_send_ccb(device, ccb) < 0) { 6361 warn("error sending sanitize command"); 6362 error = 1; 6363 goto scsisanitize_bailout; 6364 } 6365 6366 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 6367 struct scsi_sense_data *sense; 6368 int error_code, sense_key, asc, ascq; 6369 6370 if ((ccb->ccb_h.status & CAM_STATUS_MASK) == 6371 CAM_SCSI_STATUS_ERROR) { 6372 sense = &ccb->csio.sense_data; 6373 scsi_extract_sense_len(sense, ccb->csio.sense_len - 6374 ccb->csio.sense_resid, &error_code, &sense_key, 6375 &asc, &ascq, /*show_errors*/ 1); 6376 6377 if (sense_key == SSD_KEY_ILLEGAL_REQUEST && 6378 asc == 0x20 && ascq == 0x00) 6379 warnx("sanitize is not supported by " 6380 "this device"); 6381 else 6382 warnx("error sanitizing this device"); 6383 } else 6384 warnx("error sanitizing this device"); 6385 6386 if (arglist & CAM_ARG_VERBOSE) { 6387 cam_error_print(device, ccb, CAM_ESF_ALL, 6388 CAM_EPF_ALL, stderr); 6389 } 6390 error = 1; 6391 goto scsisanitize_bailout; 6392 } 6393 6394 /* 6395 * If we ran in non-immediate mode, we already checked for errors 6396 * above and printed out any necessary information. If we're in 6397 * immediate mode, we need to loop through and get status 6398 * information periodically. 6399 */ 6400 if (immediate == 0) { 6401 if (quiet == 0) { 6402 fprintf(stdout, "Sanitize Complete\n"); 6403 } 6404 goto scsisanitize_bailout; 6405 } 6406 6407 doreport: 6408 do { 6409 cam_status status; 6410 6411 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 6412 6413 /* 6414 * There's really no need to do error recovery or 6415 * retries here, since we're just going to sit in a 6416 * loop and wait for the device to finish sanitizing. 6417 */ 6418 scsi_test_unit_ready(&ccb->csio, 6419 /* retries */ 0, 6420 /* cbfcnp */ NULL, 6421 /* tag_action */ MSG_SIMPLE_Q_TAG, 6422 /* sense_len */ SSD_FULL_SIZE, 6423 /* timeout */ 5000); 6424 6425 /* Disable freezing the device queue */ 6426 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 6427 6428 retval = cam_send_ccb(device, ccb); 6429 6430 /* 6431 * If we get an error from the ioctl, bail out. SCSI 6432 * errors are expected. 6433 */ 6434 if (retval < 0) { 6435 warn("error sending CAMIOCOMMAND ioctl"); 6436 if (arglist & CAM_ARG_VERBOSE) { 6437 cam_error_print(device, ccb, CAM_ESF_ALL, 6438 CAM_EPF_ALL, stderr); 6439 } 6440 error = 1; 6441 goto scsisanitize_bailout; 6442 } 6443 6444 status = ccb->ccb_h.status & CAM_STATUS_MASK; 6445 6446 if ((status != CAM_REQ_CMP) 6447 && (status == CAM_SCSI_STATUS_ERROR) 6448 && ((ccb->ccb_h.status & CAM_AUTOSNS_VALID) != 0)) { 6449 struct scsi_sense_data *sense; 6450 int error_code, sense_key, asc, ascq; 6451 6452 sense = &ccb->csio.sense_data; 6453 scsi_extract_sense_len(sense, ccb->csio.sense_len - 6454 ccb->csio.sense_resid, &error_code, &sense_key, 6455 &asc, &ascq, /*show_errors*/ 1); 6456 6457 /* 6458 * According to the SCSI-3 spec, a drive that is in the 6459 * middle of a sanitize should return NOT READY with an 6460 * ASC of "logical unit not ready, sanitize in 6461 * progress". The sense key specific bytes will then 6462 * be a progress indicator. 6463 */ 6464 if ((sense_key == SSD_KEY_NOT_READY) 6465 && (asc == 0x04) && (ascq == 0x1b)) { 6466 uint8_t sks[3]; 6467 6468 if ((scsi_get_sks(sense, ccb->csio.sense_len - 6469 ccb->csio.sense_resid, sks) == 0) 6470 && (quiet == 0)) { 6471 int val; 6472 u_int64_t percentage; 6473 6474 val = scsi_2btoul(&sks[1]); 6475 percentage = 10000 * val; 6476 6477 fprintf(stdout, 6478 "\rSanitizing: %ju.%02u %% " 6479 "(%d/%d) done", 6480 (uintmax_t)(percentage / 6481 (0x10000 * 100)), 6482 (unsigned)((percentage / 6483 0x10000) % 100), 6484 val, 0x10000); 6485 fflush(stdout); 6486 } else if ((quiet == 0) 6487 && (++num_warnings <= 1)) { 6488 warnx("Unexpected SCSI Sense Key " 6489 "Specific value returned " 6490 "during sanitize:"); 6491 scsi_sense_print(device, &ccb->csio, 6492 stderr); 6493 warnx("Unable to print status " 6494 "information, but sanitze will " 6495 "proceed."); 6496 warnx("will exit when sanitize is " 6497 "complete"); 6498 } 6499 sleep(1); 6500 } else { 6501 warnx("Unexpected SCSI error during sanitize"); 6502 cam_error_print(device, ccb, CAM_ESF_ALL, 6503 CAM_EPF_ALL, stderr); 6504 error = 1; 6505 goto scsisanitize_bailout; 6506 } 6507 6508 } else if (status != CAM_REQ_CMP) { 6509 warnx("Unexpected CAM status %#x", status); 6510 if (arglist & CAM_ARG_VERBOSE) 6511 cam_error_print(device, ccb, CAM_ESF_ALL, 6512 CAM_EPF_ALL, stderr); 6513 error = 1; 6514 goto scsisanitize_bailout; 6515 } 6516 } while((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP); 6517 6518 if (quiet == 0) 6519 fprintf(stdout, "\nSanitize Complete\n"); 6520 6521 scsisanitize_bailout: 6522 if (fd >= 0) 6523 close(fd); 6524 if (data_ptr != NULL) 6525 free(data_ptr); 6526 cam_freeccb(ccb); 6527 6528 return(error); 6529 } 6530 6531 static int 6532 scsireportluns(struct cam_device *device, int argc, char **argv, 6533 char *combinedopt, int retry_count, int timeout) 6534 { 6535 union ccb *ccb; 6536 int c, countonly, lunsonly; 6537 struct scsi_report_luns_data *lundata; 6538 int alloc_len; 6539 uint8_t report_type; 6540 uint32_t list_len, i, j; 6541 int retval; 6542 6543 retval = 0; 6544 lundata = NULL; 6545 report_type = RPL_REPORT_DEFAULT; 6546 ccb = cam_getccb(device); 6547 6548 if (ccb == NULL) { 6549 warnx("%s: error allocating ccb", __func__); 6550 return (1); 6551 } 6552 6553 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 6554 6555 countonly = 0; 6556 lunsonly = 0; 6557 6558 while ((c = getopt(argc, argv, combinedopt)) != -1) { 6559 switch (c) { 6560 case 'c': 6561 countonly++; 6562 break; 6563 case 'l': 6564 lunsonly++; 6565 break; 6566 case 'r': 6567 if (strcasecmp(optarg, "default") == 0) 6568 report_type = RPL_REPORT_DEFAULT; 6569 else if (strcasecmp(optarg, "wellknown") == 0) 6570 report_type = RPL_REPORT_WELLKNOWN; 6571 else if (strcasecmp(optarg, "all") == 0) 6572 report_type = RPL_REPORT_ALL; 6573 else { 6574 warnx("%s: invalid report type \"%s\"", 6575 __func__, optarg); 6576 retval = 1; 6577 goto bailout; 6578 } 6579 break; 6580 default: 6581 break; 6582 } 6583 } 6584 6585 if ((countonly != 0) 6586 && (lunsonly != 0)) { 6587 warnx("%s: you can only specify one of -c or -l", __func__); 6588 retval = 1; 6589 goto bailout; 6590 } 6591 /* 6592 * According to SPC-4, the allocation length must be at least 16 6593 * bytes -- enough for the header and one LUN. 6594 */ 6595 alloc_len = sizeof(*lundata) + 8; 6596 6597 retry: 6598 6599 lundata = malloc(alloc_len); 6600 6601 if (lundata == NULL) { 6602 warn("%s: error mallocing %d bytes", __func__, alloc_len); 6603 retval = 1; 6604 goto bailout; 6605 } 6606 6607 scsi_report_luns(&ccb->csio, 6608 /*retries*/ retry_count, 6609 /*cbfcnp*/ NULL, 6610 /*tag_action*/ MSG_SIMPLE_Q_TAG, 6611 /*select_report*/ report_type, 6612 /*rpl_buf*/ lundata, 6613 /*alloc_len*/ alloc_len, 6614 /*sense_len*/ SSD_FULL_SIZE, 6615 /*timeout*/ timeout ? timeout : 5000); 6616 6617 /* Disable freezing the device queue */ 6618 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 6619 6620 if (arglist & CAM_ARG_ERR_RECOVER) 6621 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 6622 6623 if (cam_send_ccb(device, ccb) < 0) { 6624 warn("error sending REPORT LUNS command"); 6625 6626 if (arglist & CAM_ARG_VERBOSE) 6627 cam_error_print(device, ccb, CAM_ESF_ALL, 6628 CAM_EPF_ALL, stderr); 6629 6630 retval = 1; 6631 goto bailout; 6632 } 6633 6634 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 6635 cam_error_print(device, ccb, CAM_ESF_ALL, CAM_EPF_ALL, stderr); 6636 retval = 1; 6637 goto bailout; 6638 } 6639 6640 6641 list_len = scsi_4btoul(lundata->length); 6642 6643 /* 6644 * If we need to list the LUNs, and our allocation 6645 * length was too short, reallocate and retry. 6646 */ 6647 if ((countonly == 0) 6648 && (list_len > (alloc_len - sizeof(*lundata)))) { 6649 alloc_len = list_len + sizeof(*lundata); 6650 free(lundata); 6651 goto retry; 6652 } 6653 6654 if (lunsonly == 0) 6655 fprintf(stdout, "%u LUN%s found\n", list_len / 8, 6656 ((list_len / 8) > 1) ? "s" : ""); 6657 6658 if (countonly != 0) 6659 goto bailout; 6660 6661 for (i = 0; i < (list_len / 8); i++) { 6662 int no_more; 6663 6664 no_more = 0; 6665 for (j = 0; j < sizeof(lundata->luns[i].lundata); j += 2) { 6666 if (j != 0) 6667 fprintf(stdout, ","); 6668 switch (lundata->luns[i].lundata[j] & 6669 RPL_LUNDATA_ATYP_MASK) { 6670 case RPL_LUNDATA_ATYP_PERIPH: 6671 if ((lundata->luns[i].lundata[j] & 6672 RPL_LUNDATA_PERIPH_BUS_MASK) != 0) 6673 fprintf(stdout, "%d:", 6674 lundata->luns[i].lundata[j] & 6675 RPL_LUNDATA_PERIPH_BUS_MASK); 6676 else if ((j == 0) 6677 && ((lundata->luns[i].lundata[j+2] & 6678 RPL_LUNDATA_PERIPH_BUS_MASK) == 0)) 6679 no_more = 1; 6680 6681 fprintf(stdout, "%d", 6682 lundata->luns[i].lundata[j+1]); 6683 break; 6684 case RPL_LUNDATA_ATYP_FLAT: { 6685 uint8_t tmplun[2]; 6686 tmplun[0] = lundata->luns[i].lundata[j] & 6687 RPL_LUNDATA_FLAT_LUN_MASK; 6688 tmplun[1] = lundata->luns[i].lundata[j+1]; 6689 6690 fprintf(stdout, "%d", scsi_2btoul(tmplun)); 6691 no_more = 1; 6692 break; 6693 } 6694 case RPL_LUNDATA_ATYP_LUN: 6695 fprintf(stdout, "%d:%d:%d", 6696 (lundata->luns[i].lundata[j+1] & 6697 RPL_LUNDATA_LUN_BUS_MASK) >> 5, 6698 lundata->luns[i].lundata[j] & 6699 RPL_LUNDATA_LUN_TARG_MASK, 6700 lundata->luns[i].lundata[j+1] & 6701 RPL_LUNDATA_LUN_LUN_MASK); 6702 break; 6703 case RPL_LUNDATA_ATYP_EXTLUN: { 6704 int field_len_code, eam_code; 6705 6706 eam_code = lundata->luns[i].lundata[j] & 6707 RPL_LUNDATA_EXT_EAM_MASK; 6708 field_len_code = (lundata->luns[i].lundata[j] & 6709 RPL_LUNDATA_EXT_LEN_MASK) >> 4; 6710 6711 if ((eam_code == RPL_LUNDATA_EXT_EAM_WK) 6712 && (field_len_code == 0x00)) { 6713 fprintf(stdout, "%d", 6714 lundata->luns[i].lundata[j+1]); 6715 } else if ((eam_code == 6716 RPL_LUNDATA_EXT_EAM_NOT_SPEC) 6717 && (field_len_code == 0x03)) { 6718 uint8_t tmp_lun[8]; 6719 6720 /* 6721 * This format takes up all 8 bytes. 6722 * If we aren't starting at offset 0, 6723 * that's a bug. 6724 */ 6725 if (j != 0) { 6726 fprintf(stdout, "Invalid " 6727 "offset %d for " 6728 "Extended LUN not " 6729 "specified format", j); 6730 no_more = 1; 6731 break; 6732 } 6733 bzero(tmp_lun, sizeof(tmp_lun)); 6734 bcopy(&lundata->luns[i].lundata[j+1], 6735 &tmp_lun[1], sizeof(tmp_lun) - 1); 6736 fprintf(stdout, "%#jx", 6737 (intmax_t)scsi_8btou64(tmp_lun)); 6738 no_more = 1; 6739 } else { 6740 fprintf(stderr, "Unknown Extended LUN" 6741 "Address method %#x, length " 6742 "code %#x", eam_code, 6743 field_len_code); 6744 no_more = 1; 6745 } 6746 break; 6747 } 6748 default: 6749 fprintf(stderr, "Unknown LUN address method " 6750 "%#x\n", lundata->luns[i].lundata[0] & 6751 RPL_LUNDATA_ATYP_MASK); 6752 break; 6753 } 6754 /* 6755 * For the flat addressing method, there are no 6756 * other levels after it. 6757 */ 6758 if (no_more != 0) 6759 break; 6760 } 6761 fprintf(stdout, "\n"); 6762 } 6763 6764 bailout: 6765 6766 cam_freeccb(ccb); 6767 6768 free(lundata); 6769 6770 return (retval); 6771 } 6772 6773 static int 6774 scsireadcapacity(struct cam_device *device, int argc, char **argv, 6775 char *combinedopt, int retry_count, int timeout) 6776 { 6777 union ccb *ccb; 6778 int blocksizeonly, humanize, numblocks, quiet, sizeonly, baseten; 6779 struct scsi_read_capacity_data rcap; 6780 struct scsi_read_capacity_data_long rcaplong; 6781 uint64_t maxsector; 6782 uint32_t block_len; 6783 int retval; 6784 int c; 6785 6786 blocksizeonly = 0; 6787 humanize = 0; 6788 numblocks = 0; 6789 quiet = 0; 6790 sizeonly = 0; 6791 baseten = 0; 6792 retval = 0; 6793 6794 ccb = cam_getccb(device); 6795 6796 if (ccb == NULL) { 6797 warnx("%s: error allocating ccb", __func__); 6798 return (1); 6799 } 6800 6801 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 6802 6803 while ((c = getopt(argc, argv, combinedopt)) != -1) { 6804 switch (c) { 6805 case 'b': 6806 blocksizeonly++; 6807 break; 6808 case 'h': 6809 humanize++; 6810 baseten = 0; 6811 break; 6812 case 'H': 6813 humanize++; 6814 baseten++; 6815 break; 6816 case 'N': 6817 numblocks++; 6818 break; 6819 case 'q': 6820 quiet++; 6821 break; 6822 case 's': 6823 sizeonly++; 6824 break; 6825 default: 6826 break; 6827 } 6828 } 6829 6830 if ((blocksizeonly != 0) 6831 && (numblocks != 0)) { 6832 warnx("%s: you can only specify one of -b or -N", __func__); 6833 retval = 1; 6834 goto bailout; 6835 } 6836 6837 if ((blocksizeonly != 0) 6838 && (sizeonly != 0)) { 6839 warnx("%s: you can only specify one of -b or -s", __func__); 6840 retval = 1; 6841 goto bailout; 6842 } 6843 6844 if ((humanize != 0) 6845 && (quiet != 0)) { 6846 warnx("%s: you can only specify one of -h/-H or -q", __func__); 6847 retval = 1; 6848 goto bailout; 6849 } 6850 6851 if ((humanize != 0) 6852 && (blocksizeonly != 0)) { 6853 warnx("%s: you can only specify one of -h/-H or -b", __func__); 6854 retval = 1; 6855 goto bailout; 6856 } 6857 6858 scsi_read_capacity(&ccb->csio, 6859 /*retries*/ retry_count, 6860 /*cbfcnp*/ NULL, 6861 /*tag_action*/ MSG_SIMPLE_Q_TAG, 6862 &rcap, 6863 SSD_FULL_SIZE, 6864 /*timeout*/ timeout ? timeout : 5000); 6865 6866 /* Disable freezing the device queue */ 6867 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 6868 6869 if (arglist & CAM_ARG_ERR_RECOVER) 6870 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 6871 6872 if (cam_send_ccb(device, ccb) < 0) { 6873 warn("error sending READ CAPACITY command"); 6874 6875 if (arglist & CAM_ARG_VERBOSE) 6876 cam_error_print(device, ccb, CAM_ESF_ALL, 6877 CAM_EPF_ALL, stderr); 6878 6879 retval = 1; 6880 goto bailout; 6881 } 6882 6883 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 6884 cam_error_print(device, ccb, CAM_ESF_ALL, CAM_EPF_ALL, stderr); 6885 retval = 1; 6886 goto bailout; 6887 } 6888 6889 maxsector = scsi_4btoul(rcap.addr); 6890 block_len = scsi_4btoul(rcap.length); 6891 6892 /* 6893 * A last block of 2^32-1 means that the true capacity is over 2TB, 6894 * and we need to issue the long READ CAPACITY to get the real 6895 * capacity. Otherwise, we're all set. 6896 */ 6897 if (maxsector != 0xffffffff) 6898 goto do_print; 6899 6900 scsi_read_capacity_16(&ccb->csio, 6901 /*retries*/ retry_count, 6902 /*cbfcnp*/ NULL, 6903 /*tag_action*/ MSG_SIMPLE_Q_TAG, 6904 /*lba*/ 0, 6905 /*reladdr*/ 0, 6906 /*pmi*/ 0, 6907 /*rcap_buf*/ (uint8_t *)&rcaplong, 6908 /*rcap_buf_len*/ sizeof(rcaplong), 6909 /*sense_len*/ SSD_FULL_SIZE, 6910 /*timeout*/ timeout ? timeout : 5000); 6911 6912 /* Disable freezing the device queue */ 6913 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 6914 6915 if (arglist & CAM_ARG_ERR_RECOVER) 6916 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 6917 6918 if (cam_send_ccb(device, ccb) < 0) { 6919 warn("error sending READ CAPACITY (16) command"); 6920 6921 if (arglist & CAM_ARG_VERBOSE) 6922 cam_error_print(device, ccb, CAM_ESF_ALL, 6923 CAM_EPF_ALL, stderr); 6924 6925 retval = 1; 6926 goto bailout; 6927 } 6928 6929 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 6930 cam_error_print(device, ccb, CAM_ESF_ALL, CAM_EPF_ALL, stderr); 6931 retval = 1; 6932 goto bailout; 6933 } 6934 6935 maxsector = scsi_8btou64(rcaplong.addr); 6936 block_len = scsi_4btoul(rcaplong.length); 6937 6938 do_print: 6939 if (blocksizeonly == 0) { 6940 /* 6941 * Humanize implies !quiet, and also implies numblocks. 6942 */ 6943 if (humanize != 0) { 6944 char tmpstr[6]; 6945 int64_t tmpbytes; 6946 int ret; 6947 6948 tmpbytes = (maxsector + 1) * block_len; 6949 ret = humanize_number(tmpstr, sizeof(tmpstr), 6950 tmpbytes, "", HN_AUTOSCALE, 6951 HN_B | HN_DECIMAL | 6952 ((baseten != 0) ? 6953 HN_DIVISOR_1000 : 0)); 6954 if (ret == -1) { 6955 warnx("%s: humanize_number failed!", __func__); 6956 retval = 1; 6957 goto bailout; 6958 } 6959 fprintf(stdout, "Device Size: %s%s", tmpstr, 6960 (sizeonly == 0) ? ", " : "\n"); 6961 } else if (numblocks != 0) { 6962 fprintf(stdout, "%s%ju%s", (quiet == 0) ? 6963 "Blocks: " : "", (uintmax_t)maxsector + 1, 6964 (sizeonly == 0) ? ", " : "\n"); 6965 } else { 6966 fprintf(stdout, "%s%ju%s", (quiet == 0) ? 6967 "Last Block: " : "", (uintmax_t)maxsector, 6968 (sizeonly == 0) ? ", " : "\n"); 6969 } 6970 } 6971 if (sizeonly == 0) 6972 fprintf(stdout, "%s%u%s\n", (quiet == 0) ? 6973 "Block Length: " : "", block_len, (quiet == 0) ? 6974 " bytes" : ""); 6975 bailout: 6976 cam_freeccb(ccb); 6977 6978 return (retval); 6979 } 6980 6981 static int 6982 smpcmd(struct cam_device *device, int argc, char **argv, char *combinedopt, 6983 int retry_count, int timeout) 6984 { 6985 int c, error = 0; 6986 union ccb *ccb; 6987 uint8_t *smp_request = NULL, *smp_response = NULL; 6988 int request_size = 0, response_size = 0; 6989 int fd_request = 0, fd_response = 0; 6990 char *datastr = NULL; 6991 struct get_hook hook; 6992 int retval; 6993 int flags = 0; 6994 6995 /* 6996 * Note that at the moment we don't support sending SMP CCBs to 6997 * devices that aren't probed by CAM. 6998 */ 6999 ccb = cam_getccb(device); 7000 if (ccb == NULL) { 7001 warnx("%s: error allocating CCB", __func__); 7002 return (1); 7003 } 7004 7005 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->smpio); 7006 7007 while ((c = getopt(argc, argv, combinedopt)) != -1) { 7008 switch (c) { 7009 case 'R': 7010 arglist |= CAM_ARG_CMD_IN; 7011 response_size = strtol(optarg, NULL, 0); 7012 if (response_size <= 0) { 7013 warnx("invalid number of response bytes %d", 7014 response_size); 7015 error = 1; 7016 goto smpcmd_bailout; 7017 } 7018 hook.argc = argc - optind; 7019 hook.argv = argv + optind; 7020 hook.got = 0; 7021 optind++; 7022 datastr = cget(&hook, NULL); 7023 /* 7024 * If the user supplied "-" instead of a format, he 7025 * wants the data to be written to stdout. 7026 */ 7027 if ((datastr != NULL) 7028 && (datastr[0] == '-')) 7029 fd_response = 1; 7030 7031 smp_response = (u_int8_t *)malloc(response_size); 7032 if (smp_response == NULL) { 7033 warn("can't malloc memory for SMP response"); 7034 error = 1; 7035 goto smpcmd_bailout; 7036 } 7037 break; 7038 case 'r': 7039 arglist |= CAM_ARG_CMD_OUT; 7040 request_size = strtol(optarg, NULL, 0); 7041 if (request_size <= 0) { 7042 warnx("invalid number of request bytes %d", 7043 request_size); 7044 error = 1; 7045 goto smpcmd_bailout; 7046 } 7047 hook.argc = argc - optind; 7048 hook.argv = argv + optind; 7049 hook.got = 0; 7050 datastr = cget(&hook, NULL); 7051 smp_request = (u_int8_t *)malloc(request_size); 7052 if (smp_request == NULL) { 7053 warn("can't malloc memory for SMP request"); 7054 error = 1; 7055 goto smpcmd_bailout; 7056 } 7057 bzero(smp_request, request_size); 7058 /* 7059 * If the user supplied "-" instead of a format, he 7060 * wants the data to be read from stdin. 7061 */ 7062 if ((datastr != NULL) 7063 && (datastr[0] == '-')) 7064 fd_request = 1; 7065 else 7066 buff_encode_visit(smp_request, request_size, 7067 datastr, 7068 iget, &hook); 7069 optind += hook.got; 7070 break; 7071 default: 7072 break; 7073 } 7074 } 7075 7076 /* 7077 * If fd_data is set, and we're writing to the device, we need to 7078 * read the data the user wants written from stdin. 7079 */ 7080 if ((fd_request == 1) && (arglist & CAM_ARG_CMD_OUT)) { 7081 ssize_t amt_read; 7082 int amt_to_read = request_size; 7083 u_int8_t *buf_ptr = smp_request; 7084 7085 for (amt_read = 0; amt_to_read > 0; 7086 amt_read = read(STDIN_FILENO, buf_ptr, amt_to_read)) { 7087 if (amt_read == -1) { 7088 warn("error reading data from stdin"); 7089 error = 1; 7090 goto smpcmd_bailout; 7091 } 7092 amt_to_read -= amt_read; 7093 buf_ptr += amt_read; 7094 } 7095 } 7096 7097 if (((arglist & CAM_ARG_CMD_IN) == 0) 7098 || ((arglist & CAM_ARG_CMD_OUT) == 0)) { 7099 warnx("%s: need both the request (-r) and response (-R) " 7100 "arguments", __func__); 7101 error = 1; 7102 goto smpcmd_bailout; 7103 } 7104 7105 flags |= CAM_DEV_QFRZDIS; 7106 7107 cam_fill_smpio(&ccb->smpio, 7108 /*retries*/ retry_count, 7109 /*cbfcnp*/ NULL, 7110 /*flags*/ flags, 7111 /*smp_request*/ smp_request, 7112 /*smp_request_len*/ request_size, 7113 /*smp_response*/ smp_response, 7114 /*smp_response_len*/ response_size, 7115 /*timeout*/ timeout ? timeout : 5000); 7116 7117 ccb->smpio.flags = SMP_FLAG_NONE; 7118 7119 if (((retval = cam_send_ccb(device, ccb)) < 0) 7120 || ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP)) { 7121 const char warnstr[] = "error sending command"; 7122 7123 if (retval < 0) 7124 warn(warnstr); 7125 else 7126 warnx(warnstr); 7127 7128 if (arglist & CAM_ARG_VERBOSE) { 7129 cam_error_print(device, ccb, CAM_ESF_ALL, 7130 CAM_EPF_ALL, stderr); 7131 } 7132 } 7133 7134 if (((ccb->ccb_h.status & CAM_STATUS_MASK) == CAM_REQ_CMP) 7135 && (response_size > 0)) { 7136 if (fd_response == 0) { 7137 buff_decode_visit(smp_response, response_size, 7138 datastr, arg_put, NULL); 7139 fprintf(stdout, "\n"); 7140 } else { 7141 ssize_t amt_written; 7142 int amt_to_write = response_size; 7143 u_int8_t *buf_ptr = smp_response; 7144 7145 for (amt_written = 0; (amt_to_write > 0) && 7146 (amt_written = write(STDOUT_FILENO, buf_ptr, 7147 amt_to_write)) > 0;){ 7148 amt_to_write -= amt_written; 7149 buf_ptr += amt_written; 7150 } 7151 if (amt_written == -1) { 7152 warn("error writing data to stdout"); 7153 error = 1; 7154 goto smpcmd_bailout; 7155 } else if ((amt_written == 0) 7156 && (amt_to_write > 0)) { 7157 warnx("only wrote %u bytes out of %u", 7158 response_size - amt_to_write, 7159 response_size); 7160 } 7161 } 7162 } 7163 smpcmd_bailout: 7164 if (ccb != NULL) 7165 cam_freeccb(ccb); 7166 7167 if (smp_request != NULL) 7168 free(smp_request); 7169 7170 if (smp_response != NULL) 7171 free(smp_response); 7172 7173 return (error); 7174 } 7175 7176 static int 7177 smpreportgeneral(struct cam_device *device, int argc, char **argv, 7178 char *combinedopt, int retry_count, int timeout) 7179 { 7180 union ccb *ccb; 7181 struct smp_report_general_request *request = NULL; 7182 struct smp_report_general_response *response = NULL; 7183 struct sbuf *sb = NULL; 7184 int error = 0; 7185 int c, long_response = 0; 7186 int retval; 7187 7188 /* 7189 * Note that at the moment we don't support sending SMP CCBs to 7190 * devices that aren't probed by CAM. 7191 */ 7192 ccb = cam_getccb(device); 7193 if (ccb == NULL) { 7194 warnx("%s: error allocating CCB", __func__); 7195 return (1); 7196 } 7197 7198 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->smpio); 7199 7200 while ((c = getopt(argc, argv, combinedopt)) != -1) { 7201 switch (c) { 7202 case 'l': 7203 long_response = 1; 7204 break; 7205 default: 7206 break; 7207 } 7208 } 7209 request = malloc(sizeof(*request)); 7210 if (request == NULL) { 7211 warn("%s: unable to allocate %zd bytes", __func__, 7212 sizeof(*request)); 7213 error = 1; 7214 goto bailout; 7215 } 7216 7217 response = malloc(sizeof(*response)); 7218 if (response == NULL) { 7219 warn("%s: unable to allocate %zd bytes", __func__, 7220 sizeof(*response)); 7221 error = 1; 7222 goto bailout; 7223 } 7224 7225 try_long: 7226 smp_report_general(&ccb->smpio, 7227 retry_count, 7228 /*cbfcnp*/ NULL, 7229 request, 7230 /*request_len*/ sizeof(*request), 7231 (uint8_t *)response, 7232 /*response_len*/ sizeof(*response), 7233 /*long_response*/ long_response, 7234 timeout); 7235 7236 if (((retval = cam_send_ccb(device, ccb)) < 0) 7237 || ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP)) { 7238 const char warnstr[] = "error sending command"; 7239 7240 if (retval < 0) 7241 warn(warnstr); 7242 else 7243 warnx(warnstr); 7244 7245 if (arglist & CAM_ARG_VERBOSE) { 7246 cam_error_print(device, ccb, CAM_ESF_ALL, 7247 CAM_EPF_ALL, stderr); 7248 } 7249 error = 1; 7250 goto bailout; 7251 } 7252 7253 /* 7254 * If the device supports the long response bit, try again and see 7255 * if we can get all of the data. 7256 */ 7257 if ((response->long_response & SMP_RG_LONG_RESPONSE) 7258 && (long_response == 0)) { 7259 ccb->ccb_h.status = CAM_REQ_INPROG; 7260 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->smpio); 7261 long_response = 1; 7262 goto try_long; 7263 } 7264 7265 /* 7266 * XXX KDM detect and decode SMP errors here. 7267 */ 7268 sb = sbuf_new_auto(); 7269 if (sb == NULL) { 7270 warnx("%s: error allocating sbuf", __func__); 7271 goto bailout; 7272 } 7273 7274 smp_report_general_sbuf(response, sizeof(*response), sb); 7275 7276 if (sbuf_finish(sb) != 0) { 7277 warnx("%s: sbuf_finish", __func__); 7278 goto bailout; 7279 } 7280 7281 printf("%s", sbuf_data(sb)); 7282 7283 bailout: 7284 if (ccb != NULL) 7285 cam_freeccb(ccb); 7286 7287 if (request != NULL) 7288 free(request); 7289 7290 if (response != NULL) 7291 free(response); 7292 7293 if (sb != NULL) 7294 sbuf_delete(sb); 7295 7296 return (error); 7297 } 7298 7299 static struct camcontrol_opts phy_ops[] = { 7300 {"nop", SMP_PC_PHY_OP_NOP, CAM_ARG_NONE, NULL}, 7301 {"linkreset", SMP_PC_PHY_OP_LINK_RESET, CAM_ARG_NONE, NULL}, 7302 {"hardreset", SMP_PC_PHY_OP_HARD_RESET, CAM_ARG_NONE, NULL}, 7303 {"disable", SMP_PC_PHY_OP_DISABLE, CAM_ARG_NONE, NULL}, 7304 {"clearerrlog", SMP_PC_PHY_OP_CLEAR_ERR_LOG, CAM_ARG_NONE, NULL}, 7305 {"clearaffiliation", SMP_PC_PHY_OP_CLEAR_AFFILIATON, CAM_ARG_NONE,NULL}, 7306 {"sataportsel", SMP_PC_PHY_OP_TRANS_SATA_PSS, CAM_ARG_NONE, NULL}, 7307 {"clearitnl", SMP_PC_PHY_OP_CLEAR_STP_ITN_LS, CAM_ARG_NONE, NULL}, 7308 {"setdevname", SMP_PC_PHY_OP_SET_ATT_DEV_NAME, CAM_ARG_NONE, NULL}, 7309 {NULL, 0, 0, NULL} 7310 }; 7311 7312 static int 7313 smpphycontrol(struct cam_device *device, int argc, char **argv, 7314 char *combinedopt, int retry_count, int timeout) 7315 { 7316 union ccb *ccb; 7317 struct smp_phy_control_request *request = NULL; 7318 struct smp_phy_control_response *response = NULL; 7319 int long_response = 0; 7320 int retval = 0; 7321 int phy = -1; 7322 uint32_t phy_operation = SMP_PC_PHY_OP_NOP; 7323 int phy_op_set = 0; 7324 uint64_t attached_dev_name = 0; 7325 int dev_name_set = 0; 7326 uint32_t min_plr = 0, max_plr = 0; 7327 uint32_t pp_timeout_val = 0; 7328 int slumber_partial = 0; 7329 int set_pp_timeout_val = 0; 7330 int c; 7331 7332 /* 7333 * Note that at the moment we don't support sending SMP CCBs to 7334 * devices that aren't probed by CAM. 7335 */ 7336 ccb = cam_getccb(device); 7337 if (ccb == NULL) { 7338 warnx("%s: error allocating CCB", __func__); 7339 return (1); 7340 } 7341 7342 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->smpio); 7343 7344 while ((c = getopt(argc, argv, combinedopt)) != -1) { 7345 switch (c) { 7346 case 'a': 7347 case 'A': 7348 case 's': 7349 case 'S': { 7350 int enable = -1; 7351 7352 if (strcasecmp(optarg, "enable") == 0) 7353 enable = 1; 7354 else if (strcasecmp(optarg, "disable") == 0) 7355 enable = 2; 7356 else { 7357 warnx("%s: Invalid argument %s", __func__, 7358 optarg); 7359 retval = 1; 7360 goto bailout; 7361 } 7362 switch (c) { 7363 case 's': 7364 slumber_partial |= enable << 7365 SMP_PC_SAS_SLUMBER_SHIFT; 7366 break; 7367 case 'S': 7368 slumber_partial |= enable << 7369 SMP_PC_SAS_PARTIAL_SHIFT; 7370 break; 7371 case 'a': 7372 slumber_partial |= enable << 7373 SMP_PC_SATA_SLUMBER_SHIFT; 7374 break; 7375 case 'A': 7376 slumber_partial |= enable << 7377 SMP_PC_SATA_PARTIAL_SHIFT; 7378 break; 7379 default: 7380 warnx("%s: programmer error", __func__); 7381 retval = 1; 7382 goto bailout; 7383 break; /*NOTREACHED*/ 7384 } 7385 break; 7386 } 7387 case 'd': 7388 attached_dev_name = (uintmax_t)strtoumax(optarg, 7389 NULL,0); 7390 dev_name_set = 1; 7391 break; 7392 case 'l': 7393 long_response = 1; 7394 break; 7395 case 'm': 7396 /* 7397 * We don't do extensive checking here, so this 7398 * will continue to work when new speeds come out. 7399 */ 7400 min_plr = strtoul(optarg, NULL, 0); 7401 if ((min_plr == 0) 7402 || (min_plr > 0xf)) { 7403 warnx("%s: invalid link rate %x", 7404 __func__, min_plr); 7405 retval = 1; 7406 goto bailout; 7407 } 7408 break; 7409 case 'M': 7410 /* 7411 * We don't do extensive checking here, so this 7412 * will continue to work when new speeds come out. 7413 */ 7414 max_plr = strtoul(optarg, NULL, 0); 7415 if ((max_plr == 0) 7416 || (max_plr > 0xf)) { 7417 warnx("%s: invalid link rate %x", 7418 __func__, max_plr); 7419 retval = 1; 7420 goto bailout; 7421 } 7422 break; 7423 case 'o': { 7424 camcontrol_optret optreturn; 7425 cam_argmask argnums; 7426 const char *subopt; 7427 7428 if (phy_op_set != 0) { 7429 warnx("%s: only one phy operation argument " 7430 "(-o) allowed", __func__); 7431 retval = 1; 7432 goto bailout; 7433 } 7434 7435 phy_op_set = 1; 7436 7437 /* 7438 * Allow the user to specify the phy operation 7439 * numerically, as well as with a name. This will 7440 * future-proof it a bit, so options that are added 7441 * in future specs can be used. 7442 */ 7443 if (isdigit(optarg[0])) { 7444 phy_operation = strtoul(optarg, NULL, 0); 7445 if ((phy_operation == 0) 7446 || (phy_operation > 0xff)) { 7447 warnx("%s: invalid phy operation %#x", 7448 __func__, phy_operation); 7449 retval = 1; 7450 goto bailout; 7451 } 7452 break; 7453 } 7454 optreturn = getoption(phy_ops, optarg, &phy_operation, 7455 &argnums, &subopt); 7456 7457 if (optreturn == CC_OR_AMBIGUOUS) { 7458 warnx("%s: ambiguous option %s", __func__, 7459 optarg); 7460 usage(0); 7461 retval = 1; 7462 goto bailout; 7463 } else if (optreturn == CC_OR_NOT_FOUND) { 7464 warnx("%s: option %s not found", __func__, 7465 optarg); 7466 usage(0); 7467 retval = 1; 7468 goto bailout; 7469 } 7470 break; 7471 } 7472 case 'p': 7473 phy = atoi(optarg); 7474 break; 7475 case 'T': 7476 pp_timeout_val = strtoul(optarg, NULL, 0); 7477 if (pp_timeout_val > 15) { 7478 warnx("%s: invalid partial pathway timeout " 7479 "value %u, need a value less than 16", 7480 __func__, pp_timeout_val); 7481 retval = 1; 7482 goto bailout; 7483 } 7484 set_pp_timeout_val = 1; 7485 break; 7486 default: 7487 break; 7488 } 7489 } 7490 7491 if (phy == -1) { 7492 warnx("%s: a PHY (-p phy) argument is required",__func__); 7493 retval = 1; 7494 goto bailout; 7495 } 7496 7497 if (((dev_name_set != 0) 7498 && (phy_operation != SMP_PC_PHY_OP_SET_ATT_DEV_NAME)) 7499 || ((phy_operation == SMP_PC_PHY_OP_SET_ATT_DEV_NAME) 7500 && (dev_name_set == 0))) { 7501 warnx("%s: -d name and -o setdevname arguments both " 7502 "required to set device name", __func__); 7503 retval = 1; 7504 goto bailout; 7505 } 7506 7507 request = malloc(sizeof(*request)); 7508 if (request == NULL) { 7509 warn("%s: unable to allocate %zd bytes", __func__, 7510 sizeof(*request)); 7511 retval = 1; 7512 goto bailout; 7513 } 7514 7515 response = malloc(sizeof(*response)); 7516 if (response == NULL) { 7517 warn("%s: unable to allocate %zd bytes", __func__, 7518 sizeof(*request)); 7519 retval = 1; 7520 goto bailout; 7521 } 7522 7523 smp_phy_control(&ccb->smpio, 7524 retry_count, 7525 /*cbfcnp*/ NULL, 7526 request, 7527 sizeof(*request), 7528 (uint8_t *)response, 7529 sizeof(*response), 7530 long_response, 7531 /*expected_exp_change_count*/ 0, 7532 phy, 7533 phy_operation, 7534 (set_pp_timeout_val != 0) ? 1 : 0, 7535 attached_dev_name, 7536 min_plr, 7537 max_plr, 7538 slumber_partial, 7539 pp_timeout_val, 7540 timeout); 7541 7542 if (((retval = cam_send_ccb(device, ccb)) < 0) 7543 || ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP)) { 7544 const char warnstr[] = "error sending command"; 7545 7546 if (retval < 0) 7547 warn(warnstr); 7548 else 7549 warnx(warnstr); 7550 7551 if (arglist & CAM_ARG_VERBOSE) { 7552 /* 7553 * Use CAM_EPF_NORMAL so we only get one line of 7554 * SMP command decoding. 7555 */ 7556 cam_error_print(device, ccb, CAM_ESF_ALL, 7557 CAM_EPF_NORMAL, stderr); 7558 } 7559 retval = 1; 7560 goto bailout; 7561 } 7562 7563 /* XXX KDM print out something here for success? */ 7564 bailout: 7565 if (ccb != NULL) 7566 cam_freeccb(ccb); 7567 7568 if (request != NULL) 7569 free(request); 7570 7571 if (response != NULL) 7572 free(response); 7573 7574 return (retval); 7575 } 7576 7577 static int 7578 smpmaninfo(struct cam_device *device, int argc, char **argv, 7579 char *combinedopt, int retry_count, int timeout) 7580 { 7581 union ccb *ccb; 7582 struct smp_report_manuf_info_request request; 7583 struct smp_report_manuf_info_response response; 7584 struct sbuf *sb = NULL; 7585 int long_response = 0; 7586 int retval = 0; 7587 int c; 7588 7589 /* 7590 * Note that at the moment we don't support sending SMP CCBs to 7591 * devices that aren't probed by CAM. 7592 */ 7593 ccb = cam_getccb(device); 7594 if (ccb == NULL) { 7595 warnx("%s: error allocating CCB", __func__); 7596 return (1); 7597 } 7598 7599 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->smpio); 7600 7601 while ((c = getopt(argc, argv, combinedopt)) != -1) { 7602 switch (c) { 7603 case 'l': 7604 long_response = 1; 7605 break; 7606 default: 7607 break; 7608 } 7609 } 7610 bzero(&request, sizeof(request)); 7611 bzero(&response, sizeof(response)); 7612 7613 smp_report_manuf_info(&ccb->smpio, 7614 retry_count, 7615 /*cbfcnp*/ NULL, 7616 &request, 7617 sizeof(request), 7618 (uint8_t *)&response, 7619 sizeof(response), 7620 long_response, 7621 timeout); 7622 7623 if (((retval = cam_send_ccb(device, ccb)) < 0) 7624 || ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP)) { 7625 const char warnstr[] = "error sending command"; 7626 7627 if (retval < 0) 7628 warn(warnstr); 7629 else 7630 warnx(warnstr); 7631 7632 if (arglist & CAM_ARG_VERBOSE) { 7633 cam_error_print(device, ccb, CAM_ESF_ALL, 7634 CAM_EPF_ALL, stderr); 7635 } 7636 retval = 1; 7637 goto bailout; 7638 } 7639 7640 sb = sbuf_new_auto(); 7641 if (sb == NULL) { 7642 warnx("%s: error allocating sbuf", __func__); 7643 goto bailout; 7644 } 7645 7646 smp_report_manuf_info_sbuf(&response, sizeof(response), sb); 7647 7648 if (sbuf_finish(sb) != 0) { 7649 warnx("%s: sbuf_finish", __func__); 7650 goto bailout; 7651 } 7652 7653 printf("%s", sbuf_data(sb)); 7654 7655 bailout: 7656 7657 if (ccb != NULL) 7658 cam_freeccb(ccb); 7659 7660 if (sb != NULL) 7661 sbuf_delete(sb); 7662 7663 return (retval); 7664 } 7665 7666 static int 7667 getdevid(struct cam_devitem *item) 7668 { 7669 int retval = 0; 7670 union ccb *ccb = NULL; 7671 7672 struct cam_device *dev; 7673 7674 dev = cam_open_btl(item->dev_match.path_id, 7675 item->dev_match.target_id, 7676 item->dev_match.target_lun, O_RDWR, NULL); 7677 7678 if (dev == NULL) { 7679 warnx("%s", cam_errbuf); 7680 retval = 1; 7681 goto bailout; 7682 } 7683 7684 item->device_id_len = 0; 7685 7686 ccb = cam_getccb(dev); 7687 if (ccb == NULL) { 7688 warnx("%s: error allocating CCB", __func__); 7689 retval = 1; 7690 goto bailout; 7691 } 7692 7693 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->cdai); 7694 7695 /* 7696 * On the first try, we just probe for the size of the data, and 7697 * then allocate that much memory and try again. 7698 */ 7699 retry: 7700 ccb->ccb_h.func_code = XPT_DEV_ADVINFO; 7701 ccb->ccb_h.flags = CAM_DIR_IN; 7702 ccb->cdai.flags = CDAI_FLAG_NONE; 7703 ccb->cdai.buftype = CDAI_TYPE_SCSI_DEVID; 7704 ccb->cdai.bufsiz = item->device_id_len; 7705 if (item->device_id_len != 0) 7706 ccb->cdai.buf = (uint8_t *)item->device_id; 7707 7708 if (cam_send_ccb(dev, ccb) < 0) { 7709 warn("%s: error sending XPT_GDEV_ADVINFO CCB", __func__); 7710 retval = 1; 7711 goto bailout; 7712 } 7713 7714 if (ccb->ccb_h.status != CAM_REQ_CMP) { 7715 warnx("%s: CAM status %#x", __func__, ccb->ccb_h.status); 7716 retval = 1; 7717 goto bailout; 7718 } 7719 7720 if (item->device_id_len == 0) { 7721 /* 7722 * This is our first time through. Allocate the buffer, 7723 * and then go back to get the data. 7724 */ 7725 if (ccb->cdai.provsiz == 0) { 7726 warnx("%s: invalid .provsiz field returned with " 7727 "XPT_GDEV_ADVINFO CCB", __func__); 7728 retval = 1; 7729 goto bailout; 7730 } 7731 item->device_id_len = ccb->cdai.provsiz; 7732 item->device_id = malloc(item->device_id_len); 7733 if (item->device_id == NULL) { 7734 warn("%s: unable to allocate %d bytes", __func__, 7735 item->device_id_len); 7736 retval = 1; 7737 goto bailout; 7738 } 7739 ccb->ccb_h.status = CAM_REQ_INPROG; 7740 goto retry; 7741 } 7742 7743 bailout: 7744 if (dev != NULL) 7745 cam_close_device(dev); 7746 7747 if (ccb != NULL) 7748 cam_freeccb(ccb); 7749 7750 return (retval); 7751 } 7752 7753 /* 7754 * XXX KDM merge this code with getdevtree()? 7755 */ 7756 static int 7757 buildbusdevlist(struct cam_devlist *devlist) 7758 { 7759 union ccb ccb; 7760 int bufsize, fd = -1; 7761 struct dev_match_pattern *patterns; 7762 struct cam_devitem *item = NULL; 7763 int skip_device = 0; 7764 int retval = 0; 7765 7766 if ((fd = open(XPT_DEVICE, O_RDWR)) == -1) { 7767 warn("couldn't open %s", XPT_DEVICE); 7768 return(1); 7769 } 7770 7771 bzero(&ccb, sizeof(union ccb)); 7772 7773 ccb.ccb_h.path_id = CAM_XPT_PATH_ID; 7774 ccb.ccb_h.target_id = CAM_TARGET_WILDCARD; 7775 ccb.ccb_h.target_lun = CAM_LUN_WILDCARD; 7776 7777 ccb.ccb_h.func_code = XPT_DEV_MATCH; 7778 bufsize = sizeof(struct dev_match_result) * 100; 7779 ccb.cdm.match_buf_len = bufsize; 7780 ccb.cdm.matches = (struct dev_match_result *)malloc(bufsize); 7781 if (ccb.cdm.matches == NULL) { 7782 warnx("can't malloc memory for matches"); 7783 close(fd); 7784 return(1); 7785 } 7786 ccb.cdm.num_matches = 0; 7787 ccb.cdm.num_patterns = 2; 7788 ccb.cdm.pattern_buf_len = sizeof(struct dev_match_pattern) * 7789 ccb.cdm.num_patterns; 7790 7791 patterns = (struct dev_match_pattern *)malloc(ccb.cdm.pattern_buf_len); 7792 if (patterns == NULL) { 7793 warnx("can't malloc memory for patterns"); 7794 retval = 1; 7795 goto bailout; 7796 } 7797 7798 ccb.cdm.patterns = patterns; 7799 bzero(patterns, ccb.cdm.pattern_buf_len); 7800 7801 patterns[0].type = DEV_MATCH_DEVICE; 7802 patterns[0].pattern.device_pattern.flags = DEV_MATCH_PATH; 7803 patterns[0].pattern.device_pattern.path_id = devlist->path_id; 7804 patterns[1].type = DEV_MATCH_PERIPH; 7805 patterns[1].pattern.periph_pattern.flags = PERIPH_MATCH_PATH; 7806 patterns[1].pattern.periph_pattern.path_id = devlist->path_id; 7807 7808 /* 7809 * We do the ioctl multiple times if necessary, in case there are 7810 * more than 100 nodes in the EDT. 7811 */ 7812 do { 7813 unsigned int i; 7814 7815 if (ioctl(fd, CAMIOCOMMAND, &ccb) == -1) { 7816 warn("error sending CAMIOCOMMAND ioctl"); 7817 retval = 1; 7818 goto bailout; 7819 } 7820 7821 if ((ccb.ccb_h.status != CAM_REQ_CMP) 7822 || ((ccb.cdm.status != CAM_DEV_MATCH_LAST) 7823 && (ccb.cdm.status != CAM_DEV_MATCH_MORE))) { 7824 warnx("got CAM error %#x, CDM error %d\n", 7825 ccb.ccb_h.status, ccb.cdm.status); 7826 retval = 1; 7827 goto bailout; 7828 } 7829 7830 for (i = 0; i < ccb.cdm.num_matches; i++) { 7831 switch (ccb.cdm.matches[i].type) { 7832 case DEV_MATCH_DEVICE: { 7833 struct device_match_result *dev_result; 7834 7835 dev_result = 7836 &ccb.cdm.matches[i].result.device_result; 7837 7838 if (dev_result->flags & 7839 DEV_RESULT_UNCONFIGURED) { 7840 skip_device = 1; 7841 break; 7842 } else 7843 skip_device = 0; 7844 7845 item = malloc(sizeof(*item)); 7846 if (item == NULL) { 7847 warn("%s: unable to allocate %zd bytes", 7848 __func__, sizeof(*item)); 7849 retval = 1; 7850 goto bailout; 7851 } 7852 bzero(item, sizeof(*item)); 7853 bcopy(dev_result, &item->dev_match, 7854 sizeof(*dev_result)); 7855 STAILQ_INSERT_TAIL(&devlist->dev_queue, item, 7856 links); 7857 7858 if (getdevid(item) != 0) { 7859 retval = 1; 7860 goto bailout; 7861 } 7862 break; 7863 } 7864 case DEV_MATCH_PERIPH: { 7865 struct periph_match_result *periph_result; 7866 7867 periph_result = 7868 &ccb.cdm.matches[i].result.periph_result; 7869 7870 if (skip_device != 0) 7871 break; 7872 item->num_periphs++; 7873 item->periph_matches = realloc( 7874 item->periph_matches, 7875 item->num_periphs * 7876 sizeof(struct periph_match_result)); 7877 if (item->periph_matches == NULL) { 7878 warn("%s: error allocating periph " 7879 "list", __func__); 7880 retval = 1; 7881 goto bailout; 7882 } 7883 bcopy(periph_result, &item->periph_matches[ 7884 item->num_periphs - 1], 7885 sizeof(*periph_result)); 7886 break; 7887 } 7888 default: 7889 fprintf(stderr, "%s: unexpected match " 7890 "type %d\n", __func__, 7891 ccb.cdm.matches[i].type); 7892 retval = 1; 7893 goto bailout; 7894 break; /*NOTREACHED*/ 7895 } 7896 } 7897 } while ((ccb.ccb_h.status == CAM_REQ_CMP) 7898 && (ccb.cdm.status == CAM_DEV_MATCH_MORE)); 7899 bailout: 7900 7901 if (fd != -1) 7902 close(fd); 7903 7904 free(patterns); 7905 7906 free(ccb.cdm.matches); 7907 7908 if (retval != 0) 7909 freebusdevlist(devlist); 7910 7911 return (retval); 7912 } 7913 7914 static void 7915 freebusdevlist(struct cam_devlist *devlist) 7916 { 7917 struct cam_devitem *item, *item2; 7918 7919 STAILQ_FOREACH_SAFE(item, &devlist->dev_queue, links, item2) { 7920 STAILQ_REMOVE(&devlist->dev_queue, item, cam_devitem, 7921 links); 7922 free(item->device_id); 7923 free(item->periph_matches); 7924 free(item); 7925 } 7926 } 7927 7928 static struct cam_devitem * 7929 findsasdevice(struct cam_devlist *devlist, uint64_t sasaddr) 7930 { 7931 struct cam_devitem *item; 7932 7933 STAILQ_FOREACH(item, &devlist->dev_queue, links) { 7934 struct scsi_vpd_id_descriptor *idd; 7935 7936 /* 7937 * XXX KDM look for LUN IDs as well? 7938 */ 7939 idd = scsi_get_devid(item->device_id, 7940 item->device_id_len, 7941 scsi_devid_is_sas_target); 7942 if (idd == NULL) 7943 continue; 7944 7945 if (scsi_8btou64(idd->identifier) == sasaddr) 7946 return (item); 7947 } 7948 7949 return (NULL); 7950 } 7951 7952 static int 7953 smpphylist(struct cam_device *device, int argc, char **argv, 7954 char *combinedopt, int retry_count, int timeout) 7955 { 7956 struct smp_report_general_request *rgrequest = NULL; 7957 struct smp_report_general_response *rgresponse = NULL; 7958 struct smp_discover_request *disrequest = NULL; 7959 struct smp_discover_response *disresponse = NULL; 7960 struct cam_devlist devlist; 7961 union ccb *ccb; 7962 int long_response = 0; 7963 int num_phys = 0; 7964 int quiet = 0; 7965 int retval; 7966 int i, c; 7967 7968 /* 7969 * Note that at the moment we don't support sending SMP CCBs to 7970 * devices that aren't probed by CAM. 7971 */ 7972 ccb = cam_getccb(device); 7973 if (ccb == NULL) { 7974 warnx("%s: error allocating CCB", __func__); 7975 return (1); 7976 } 7977 7978 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->smpio); 7979 STAILQ_INIT(&devlist.dev_queue); 7980 7981 rgrequest = malloc(sizeof(*rgrequest)); 7982 if (rgrequest == NULL) { 7983 warn("%s: unable to allocate %zd bytes", __func__, 7984 sizeof(*rgrequest)); 7985 retval = 1; 7986 goto bailout; 7987 } 7988 7989 rgresponse = malloc(sizeof(*rgresponse)); 7990 if (rgresponse == NULL) { 7991 warn("%s: unable to allocate %zd bytes", __func__, 7992 sizeof(*rgresponse)); 7993 retval = 1; 7994 goto bailout; 7995 } 7996 7997 while ((c = getopt(argc, argv, combinedopt)) != -1) { 7998 switch (c) { 7999 case 'l': 8000 long_response = 1; 8001 break; 8002 case 'q': 8003 quiet = 1; 8004 break; 8005 default: 8006 break; 8007 } 8008 } 8009 8010 smp_report_general(&ccb->smpio, 8011 retry_count, 8012 /*cbfcnp*/ NULL, 8013 rgrequest, 8014 /*request_len*/ sizeof(*rgrequest), 8015 (uint8_t *)rgresponse, 8016 /*response_len*/ sizeof(*rgresponse), 8017 /*long_response*/ long_response, 8018 timeout); 8019 8020 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 8021 8022 if (((retval = cam_send_ccb(device, ccb)) < 0) 8023 || ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP)) { 8024 const char warnstr[] = "error sending command"; 8025 8026 if (retval < 0) 8027 warn(warnstr); 8028 else 8029 warnx(warnstr); 8030 8031 if (arglist & CAM_ARG_VERBOSE) { 8032 cam_error_print(device, ccb, CAM_ESF_ALL, 8033 CAM_EPF_ALL, stderr); 8034 } 8035 retval = 1; 8036 goto bailout; 8037 } 8038 8039 num_phys = rgresponse->num_phys; 8040 8041 if (num_phys == 0) { 8042 if (quiet == 0) 8043 fprintf(stdout, "%s: No Phys reported\n", __func__); 8044 retval = 1; 8045 goto bailout; 8046 } 8047 8048 devlist.path_id = device->path_id; 8049 8050 retval = buildbusdevlist(&devlist); 8051 if (retval != 0) 8052 goto bailout; 8053 8054 if (quiet == 0) { 8055 fprintf(stdout, "%d PHYs:\n", num_phys); 8056 fprintf(stdout, "PHY Attached SAS Address\n"); 8057 } 8058 8059 disrequest = malloc(sizeof(*disrequest)); 8060 if (disrequest == NULL) { 8061 warn("%s: unable to allocate %zd bytes", __func__, 8062 sizeof(*disrequest)); 8063 retval = 1; 8064 goto bailout; 8065 } 8066 8067 disresponse = malloc(sizeof(*disresponse)); 8068 if (disresponse == NULL) { 8069 warn("%s: unable to allocate %zd bytes", __func__, 8070 sizeof(*disresponse)); 8071 retval = 1; 8072 goto bailout; 8073 } 8074 8075 for (i = 0; i < num_phys; i++) { 8076 struct cam_devitem *item; 8077 struct device_match_result *dev_match; 8078 char vendor[16], product[48], revision[16]; 8079 char tmpstr[256]; 8080 int j; 8081 8082 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->smpio); 8083 8084 ccb->ccb_h.status = CAM_REQ_INPROG; 8085 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 8086 8087 smp_discover(&ccb->smpio, 8088 retry_count, 8089 /*cbfcnp*/ NULL, 8090 disrequest, 8091 sizeof(*disrequest), 8092 (uint8_t *)disresponse, 8093 sizeof(*disresponse), 8094 long_response, 8095 /*ignore_zone_group*/ 0, 8096 /*phy*/ i, 8097 timeout); 8098 8099 if (((retval = cam_send_ccb(device, ccb)) < 0) 8100 || (((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) 8101 && (disresponse->function_result != SMP_FR_PHY_VACANT))) { 8102 const char warnstr[] = "error sending command"; 8103 8104 if (retval < 0) 8105 warn(warnstr); 8106 else 8107 warnx(warnstr); 8108 8109 if (arglist & CAM_ARG_VERBOSE) { 8110 cam_error_print(device, ccb, CAM_ESF_ALL, 8111 CAM_EPF_ALL, stderr); 8112 } 8113 retval = 1; 8114 goto bailout; 8115 } 8116 8117 if (disresponse->function_result == SMP_FR_PHY_VACANT) { 8118 if (quiet == 0) 8119 fprintf(stdout, "%3d <vacant>\n", i); 8120 continue; 8121 } 8122 8123 if (disresponse->attached_device == SMP_DIS_AD_TYPE_NONE) { 8124 item = NULL; 8125 } else { 8126 item = findsasdevice(&devlist, 8127 scsi_8btou64(disresponse->attached_sas_address)); 8128 } 8129 8130 if ((quiet == 0) 8131 || (item != NULL)) { 8132 fprintf(stdout, "%3d 0x%016jx", i, 8133 (uintmax_t)scsi_8btou64( 8134 disresponse->attached_sas_address)); 8135 if (item == NULL) { 8136 fprintf(stdout, "\n"); 8137 continue; 8138 } 8139 } else if (quiet != 0) 8140 continue; 8141 8142 dev_match = &item->dev_match; 8143 8144 if (dev_match->protocol == PROTO_SCSI) { 8145 cam_strvis(vendor, dev_match->inq_data.vendor, 8146 sizeof(dev_match->inq_data.vendor), 8147 sizeof(vendor)); 8148 cam_strvis(product, dev_match->inq_data.product, 8149 sizeof(dev_match->inq_data.product), 8150 sizeof(product)); 8151 cam_strvis(revision, dev_match->inq_data.revision, 8152 sizeof(dev_match->inq_data.revision), 8153 sizeof(revision)); 8154 sprintf(tmpstr, "<%s %s %s>", vendor, product, 8155 revision); 8156 } else if ((dev_match->protocol == PROTO_ATA) 8157 || (dev_match->protocol == PROTO_SATAPM)) { 8158 cam_strvis(product, dev_match->ident_data.model, 8159 sizeof(dev_match->ident_data.model), 8160 sizeof(product)); 8161 cam_strvis(revision, dev_match->ident_data.revision, 8162 sizeof(dev_match->ident_data.revision), 8163 sizeof(revision)); 8164 sprintf(tmpstr, "<%s %s>", product, revision); 8165 } else { 8166 sprintf(tmpstr, "<>"); 8167 } 8168 fprintf(stdout, " %-33s ", tmpstr); 8169 8170 /* 8171 * If we have 0 periphs, that's a bug... 8172 */ 8173 if (item->num_periphs == 0) { 8174 fprintf(stdout, "\n"); 8175 continue; 8176 } 8177 8178 fprintf(stdout, "("); 8179 for (j = 0; j < item->num_periphs; j++) { 8180 if (j > 0) 8181 fprintf(stdout, ","); 8182 8183 fprintf(stdout, "%s%d", 8184 item->periph_matches[j].periph_name, 8185 item->periph_matches[j].unit_number); 8186 8187 } 8188 fprintf(stdout, ")\n"); 8189 } 8190 bailout: 8191 if (ccb != NULL) 8192 cam_freeccb(ccb); 8193 8194 free(rgrequest); 8195 8196 free(rgresponse); 8197 8198 free(disrequest); 8199 8200 free(disresponse); 8201 8202 freebusdevlist(&devlist); 8203 8204 return (retval); 8205 } 8206 8207 static int 8208 atapm(struct cam_device *device, int argc, char **argv, 8209 char *combinedopt, int retry_count, int timeout) 8210 { 8211 union ccb *ccb; 8212 int retval = 0; 8213 int t = -1; 8214 int c; 8215 u_char cmd, sc; 8216 8217 ccb = cam_getccb(device); 8218 8219 if (ccb == NULL) { 8220 warnx("%s: error allocating ccb", __func__); 8221 return (1); 8222 } 8223 8224 while ((c = getopt(argc, argv, combinedopt)) != -1) { 8225 switch (c) { 8226 case 't': 8227 t = atoi(optarg); 8228 break; 8229 default: 8230 break; 8231 } 8232 } 8233 if (strcmp(argv[1], "idle") == 0) { 8234 if (t == -1) 8235 cmd = ATA_IDLE_IMMEDIATE; 8236 else 8237 cmd = ATA_IDLE_CMD; 8238 } else if (strcmp(argv[1], "standby") == 0) { 8239 if (t == -1) 8240 cmd = ATA_STANDBY_IMMEDIATE; 8241 else 8242 cmd = ATA_STANDBY_CMD; 8243 } else { 8244 cmd = ATA_SLEEP; 8245 t = -1; 8246 } 8247 8248 if (t < 0) 8249 sc = 0; 8250 else if (t <= (240 * 5)) 8251 sc = (t + 4) / 5; 8252 else if (t <= (252 * 5)) 8253 /* special encoding for 21 minutes */ 8254 sc = 252; 8255 else if (t <= (11 * 30 * 60)) 8256 sc = (t - 1) / (30 * 60) + 241; 8257 else 8258 sc = 253; 8259 8260 retval = ata_do_28bit_cmd(device, 8261 ccb, 8262 /*retries*/retry_count, 8263 /*flags*/CAM_DIR_NONE, 8264 /*protocol*/AP_PROTO_NON_DATA, 8265 /*tag_action*/MSG_SIMPLE_Q_TAG, 8266 /*command*/cmd, 8267 /*features*/0, 8268 /*lba*/0, 8269 /*sector_count*/sc, 8270 /*data_ptr*/NULL, 8271 /*dxfer_len*/0, 8272 /*timeout*/timeout ? timeout : 30 * 1000, 8273 /*quiet*/1); 8274 8275 cam_freeccb(ccb); 8276 return (retval); 8277 } 8278 8279 static int 8280 ataaxm(struct cam_device *device, int argc, char **argv, 8281 char *combinedopt, int retry_count, int timeout) 8282 { 8283 union ccb *ccb; 8284 int retval = 0; 8285 int l = -1; 8286 int c; 8287 u_char cmd, sc; 8288 8289 ccb = cam_getccb(device); 8290 8291 if (ccb == NULL) { 8292 warnx("%s: error allocating ccb", __func__); 8293 return (1); 8294 } 8295 8296 while ((c = getopt(argc, argv, combinedopt)) != -1) { 8297 switch (c) { 8298 case 'l': 8299 l = atoi(optarg); 8300 break; 8301 default: 8302 break; 8303 } 8304 } 8305 sc = 0; 8306 if (strcmp(argv[1], "apm") == 0) { 8307 if (l == -1) 8308 cmd = 0x85; 8309 else { 8310 cmd = 0x05; 8311 sc = l; 8312 } 8313 } else /* aam */ { 8314 if (l == -1) 8315 cmd = 0xC2; 8316 else { 8317 cmd = 0x42; 8318 sc = l; 8319 } 8320 } 8321 8322 retval = ata_do_28bit_cmd(device, 8323 ccb, 8324 /*retries*/retry_count, 8325 /*flags*/CAM_DIR_NONE, 8326 /*protocol*/AP_PROTO_NON_DATA, 8327 /*tag_action*/MSG_SIMPLE_Q_TAG, 8328 /*command*/ATA_SETFEATURES, 8329 /*features*/cmd, 8330 /*lba*/0, 8331 /*sector_count*/sc, 8332 /*data_ptr*/NULL, 8333 /*dxfer_len*/0, 8334 /*timeout*/timeout ? timeout : 30 * 1000, 8335 /*quiet*/1); 8336 8337 cam_freeccb(ccb); 8338 return (retval); 8339 } 8340 8341 int 8342 scsigetopcodes(struct cam_device *device, int opcode_set, int opcode, 8343 int show_sa_errors, int sa_set, int service_action, 8344 int timeout_desc, int retry_count, int timeout, int verbosemode, 8345 uint32_t *fill_len, uint8_t **data_ptr) 8346 { 8347 union ccb *ccb = NULL; 8348 uint8_t *buf = NULL; 8349 uint32_t alloc_len = 0, num_opcodes; 8350 uint32_t valid_len = 0; 8351 uint32_t avail_len = 0; 8352 struct scsi_report_supported_opcodes_all *all_hdr; 8353 struct scsi_report_supported_opcodes_one *one; 8354 int options = 0; 8355 int retval = 0; 8356 8357 /* 8358 * Make it clear that we haven't yet allocated or filled anything. 8359 */ 8360 *fill_len = 0; 8361 *data_ptr = NULL; 8362 8363 ccb = cam_getccb(device); 8364 if (ccb == NULL) { 8365 warnx("couldn't allocate CCB"); 8366 retval = 1; 8367 goto bailout; 8368 } 8369 8370 /* cam_getccb cleans up the header, caller has to zero the payload */ 8371 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 8372 8373 if (opcode_set != 0) { 8374 options |= RSO_OPTIONS_OC; 8375 num_opcodes = 1; 8376 alloc_len = sizeof(*one) + CAM_MAX_CDBLEN; 8377 } else { 8378 num_opcodes = 256; 8379 alloc_len = sizeof(*all_hdr) + (num_opcodes * 8380 sizeof(struct scsi_report_supported_opcodes_descr)); 8381 } 8382 8383 if (timeout_desc != 0) { 8384 options |= RSO_RCTD; 8385 alloc_len += num_opcodes * 8386 sizeof(struct scsi_report_supported_opcodes_timeout); 8387 } 8388 8389 if (sa_set != 0) { 8390 options |= RSO_OPTIONS_OC_SA; 8391 if (show_sa_errors != 0) 8392 options &= ~RSO_OPTIONS_OC; 8393 } 8394 8395 retry_alloc: 8396 if (buf != NULL) { 8397 free(buf); 8398 buf = NULL; 8399 } 8400 8401 buf = malloc(alloc_len); 8402 if (buf == NULL) { 8403 warn("Unable to allocate %u bytes", alloc_len); 8404 retval = 1; 8405 goto bailout; 8406 } 8407 bzero(buf, alloc_len); 8408 8409 scsi_report_supported_opcodes(&ccb->csio, 8410 /*retries*/ retry_count, 8411 /*cbfcnp*/ NULL, 8412 /*tag_action*/ MSG_SIMPLE_Q_TAG, 8413 /*options*/ options, 8414 /*req_opcode*/ opcode, 8415 /*req_service_action*/ service_action, 8416 /*data_ptr*/ buf, 8417 /*dxfer_len*/ alloc_len, 8418 /*sense_len*/ SSD_FULL_SIZE, 8419 /*timeout*/ timeout ? timeout : 10000); 8420 8421 ccb->ccb_h.flags |= CAM_DEV_QFRZDIS; 8422 8423 if (retry_count != 0) 8424 ccb->ccb_h.flags |= CAM_PASS_ERR_RECOVER; 8425 8426 if (cam_send_ccb(device, ccb) < 0) { 8427 perror("error sending REPORT SUPPORTED OPERATION CODES"); 8428 retval = 1; 8429 goto bailout; 8430 } 8431 8432 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 8433 if (verbosemode != 0) 8434 cam_error_print(device, ccb, CAM_ESF_ALL, 8435 CAM_EPF_ALL, stderr); 8436 8437 retval = 1; 8438 goto bailout; 8439 } 8440 8441 valid_len = ccb->csio.dxfer_len - ccb->csio.resid; 8442 8443 if (((options & RSO_OPTIONS_MASK) == RSO_OPTIONS_ALL) 8444 && (valid_len >= sizeof(*all_hdr))) { 8445 all_hdr = (struct scsi_report_supported_opcodes_all *)buf; 8446 avail_len = scsi_4btoul(all_hdr->length) + sizeof(*all_hdr); 8447 } else if (((options & RSO_OPTIONS_MASK) != RSO_OPTIONS_ALL) 8448 && (valid_len >= sizeof(*one))) { 8449 uint32_t cdb_length; 8450 8451 one = (struct scsi_report_supported_opcodes_one *)buf; 8452 cdb_length = scsi_2btoul(one->cdb_length); 8453 avail_len = sizeof(*one) + cdb_length; 8454 if (one->support & RSO_ONE_CTDP) { 8455 struct scsi_report_supported_opcodes_timeout *td; 8456 8457 td = (struct scsi_report_supported_opcodes_timeout *) 8458 &buf[avail_len]; 8459 if (valid_len >= (avail_len + sizeof(td->length))) { 8460 avail_len += scsi_2btoul(td->length) + 8461 sizeof(td->length); 8462 } else { 8463 avail_len += sizeof(*td); 8464 } 8465 } 8466 } 8467 8468 /* 8469 * avail_len could be zero if we didn't get enough data back from 8470 * thet target to determine 8471 */ 8472 if ((avail_len != 0) 8473 && (avail_len > valid_len)) { 8474 alloc_len = avail_len; 8475 goto retry_alloc; 8476 } 8477 8478 *fill_len = valid_len; 8479 *data_ptr = buf; 8480 bailout: 8481 if (retval != 0) 8482 free(buf); 8483 8484 cam_freeccb(ccb); 8485 8486 return (retval); 8487 } 8488 8489 static int 8490 scsiprintoneopcode(struct cam_device *device, int req_opcode, int sa_set, 8491 int req_sa, uint8_t *buf, uint32_t valid_len) 8492 { 8493 struct scsi_report_supported_opcodes_one *one; 8494 struct scsi_report_supported_opcodes_timeout *td; 8495 uint32_t cdb_len = 0, td_len = 0; 8496 const char *op_desc = NULL; 8497 unsigned int i; 8498 int retval = 0; 8499 8500 one = (struct scsi_report_supported_opcodes_one *)buf; 8501 8502 /* 8503 * If we don't have the full single opcode descriptor, no point in 8504 * continuing. 8505 */ 8506 if (valid_len < __offsetof(struct scsi_report_supported_opcodes_one, 8507 cdb_length)) { 8508 warnx("Only %u bytes returned, not enough to verify support", 8509 valid_len); 8510 retval = 1; 8511 goto bailout; 8512 } 8513 8514 op_desc = scsi_op_desc(req_opcode, &device->inq_data); 8515 8516 printf("%s (0x%02x)", op_desc != NULL ? op_desc : "UNKNOWN", 8517 req_opcode); 8518 if (sa_set != 0) 8519 printf(", SA 0x%x", req_sa); 8520 printf(": "); 8521 8522 switch (one->support & RSO_ONE_SUP_MASK) { 8523 case RSO_ONE_SUP_UNAVAIL: 8524 printf("No command support information currently available\n"); 8525 break; 8526 case RSO_ONE_SUP_NOT_SUP: 8527 printf("Command not supported\n"); 8528 retval = 1; 8529 goto bailout; 8530 break; /*NOTREACHED*/ 8531 case RSO_ONE_SUP_AVAIL: 8532 printf("Command is supported, complies with a SCSI standard\n"); 8533 break; 8534 case RSO_ONE_SUP_VENDOR: 8535 printf("Command is supported, vendor-specific " 8536 "implementation\n"); 8537 break; 8538 default: 8539 printf("Unknown command support flags 0x%#x\n", 8540 one->support & RSO_ONE_SUP_MASK); 8541 break; 8542 } 8543 8544 /* 8545 * If we don't have the CDB length, it isn't exactly an error, the 8546 * command probably isn't supported. 8547 */ 8548 if (valid_len < __offsetof(struct scsi_report_supported_opcodes_one, 8549 cdb_usage)) 8550 goto bailout; 8551 8552 cdb_len = scsi_2btoul(one->cdb_length); 8553 8554 /* 8555 * If our valid data doesn't include the full reported length, 8556 * return. The caller should have detected this and adjusted his 8557 * allocation length to get all of the available data. 8558 */ 8559 if (valid_len < sizeof(*one) + cdb_len) { 8560 retval = 1; 8561 goto bailout; 8562 } 8563 8564 /* 8565 * If all we have is the opcode, there is no point in printing out 8566 * the usage bitmap. 8567 */ 8568 if (cdb_len <= 1) { 8569 retval = 1; 8570 goto bailout; 8571 } 8572 8573 printf("CDB usage bitmap:"); 8574 for (i = 0; i < cdb_len; i++) { 8575 printf(" %02x", one->cdb_usage[i]); 8576 } 8577 printf("\n"); 8578 8579 /* 8580 * If we don't have a timeout descriptor, we're done. 8581 */ 8582 if ((one->support & RSO_ONE_CTDP) == 0) 8583 goto bailout; 8584 8585 /* 8586 * If we don't have enough valid length to include the timeout 8587 * descriptor length, we're done. 8588 */ 8589 if (valid_len < (sizeof(*one) + cdb_len + sizeof(td->length))) 8590 goto bailout; 8591 8592 td = (struct scsi_report_supported_opcodes_timeout *) 8593 &buf[sizeof(*one) + cdb_len]; 8594 td_len = scsi_2btoul(td->length); 8595 td_len += sizeof(td->length); 8596 8597 /* 8598 * If we don't have the full timeout descriptor, we're done. 8599 */ 8600 if (td_len < sizeof(*td)) 8601 goto bailout; 8602 8603 /* 8604 * If we don't have enough valid length to contain the full timeout 8605 * descriptor, we're done. 8606 */ 8607 if (valid_len < (sizeof(*one) + cdb_len + td_len)) 8608 goto bailout; 8609 8610 printf("Timeout information:\n"); 8611 printf("Command-specific: 0x%02x\n", td->cmd_specific); 8612 printf("Nominal timeout: %u seconds\n", 8613 scsi_4btoul(td->nominal_time)); 8614 printf("Recommended timeout: %u seconds\n", 8615 scsi_4btoul(td->recommended_time)); 8616 8617 bailout: 8618 return (retval); 8619 } 8620 8621 static int 8622 scsiprintopcodes(struct cam_device *device, int td_req, uint8_t *buf, 8623 uint32_t valid_len) 8624 { 8625 struct scsi_report_supported_opcodes_all *hdr; 8626 struct scsi_report_supported_opcodes_descr *desc; 8627 uint32_t avail_len = 0, used_len = 0; 8628 uint8_t *cur_ptr; 8629 int retval = 0; 8630 8631 if (valid_len < sizeof(*hdr)) { 8632 warnx("%s: not enough returned data (%u bytes) opcode list", 8633 __func__, valid_len); 8634 retval = 1; 8635 goto bailout; 8636 } 8637 hdr = (struct scsi_report_supported_opcodes_all *)buf; 8638 avail_len = scsi_4btoul(hdr->length); 8639 avail_len += sizeof(hdr->length); 8640 /* 8641 * Take the lesser of the amount of data the drive claims is 8642 * available, and the amount of data the HBA says was returned. 8643 */ 8644 avail_len = MIN(avail_len, valid_len); 8645 8646 used_len = sizeof(hdr->length); 8647 8648 printf("%-6s %4s %8s ", 8649 "Opcode", "SA", "CDB len" ); 8650 8651 if (td_req != 0) 8652 printf("%5s %6s %6s ", "CS", "Nom", "Rec"); 8653 printf(" Description\n"); 8654 8655 while ((avail_len - used_len) > sizeof(*desc)) { 8656 struct scsi_report_supported_opcodes_timeout *td; 8657 uint32_t td_len; 8658 const char *op_desc = NULL; 8659 8660 cur_ptr = &buf[used_len]; 8661 desc = (struct scsi_report_supported_opcodes_descr *)cur_ptr; 8662 8663 op_desc = scsi_op_desc(desc->opcode, &device->inq_data); 8664 if (op_desc == NULL) 8665 op_desc = "UNKNOWN"; 8666 8667 printf("0x%02x %#4x %8u ", desc->opcode, 8668 scsi_2btoul(desc->service_action), 8669 scsi_2btoul(desc->cdb_length)); 8670 8671 used_len += sizeof(*desc); 8672 8673 if ((desc->flags & RSO_CTDP) == 0) { 8674 printf(" %s\n", op_desc); 8675 continue; 8676 } 8677 8678 /* 8679 * If we don't have enough space to fit a timeout 8680 * descriptor, then we're done. 8681 */ 8682 if (avail_len - used_len < sizeof(*td)) { 8683 used_len = avail_len; 8684 printf(" %s\n", op_desc); 8685 continue; 8686 } 8687 cur_ptr = &buf[used_len]; 8688 td = (struct scsi_report_supported_opcodes_timeout *)cur_ptr; 8689 td_len = scsi_2btoul(td->length); 8690 td_len += sizeof(td->length); 8691 8692 used_len += td_len; 8693 /* 8694 * If the given timeout descriptor length is less than what 8695 * we understand, skip it. 8696 */ 8697 if (td_len < sizeof(*td)) { 8698 printf(" %s\n", op_desc); 8699 continue; 8700 } 8701 8702 printf(" 0x%02x %6u %6u %s\n", td->cmd_specific, 8703 scsi_4btoul(td->nominal_time), 8704 scsi_4btoul(td->recommended_time), op_desc); 8705 } 8706 bailout: 8707 return (retval); 8708 } 8709 8710 static int 8711 scsiopcodes(struct cam_device *device, int argc, char **argv, 8712 char *combinedopt, int retry_count, int timeout, int verbosemode) 8713 { 8714 int c; 8715 uint32_t opcode = 0, service_action = 0; 8716 int td_set = 0, opcode_set = 0, sa_set = 0; 8717 int show_sa_errors = 1; 8718 uint32_t valid_len = 0; 8719 uint8_t *buf = NULL; 8720 char *endptr; 8721 int retval = 0; 8722 8723 while ((c = getopt(argc, argv, combinedopt)) != -1) { 8724 switch (c) { 8725 case 'N': 8726 show_sa_errors = 0; 8727 break; 8728 case 'o': 8729 opcode = strtoul(optarg, &endptr, 0); 8730 if (*endptr != '\0') { 8731 warnx("Invalid opcode \"%s\", must be a number", 8732 optarg); 8733 retval = 1; 8734 goto bailout; 8735 } 8736 if (opcode > 0xff) { 8737 warnx("Invalid opcode 0x%#x, must be between" 8738 "0 and 0xff inclusive", opcode); 8739 retval = 1; 8740 goto bailout; 8741 } 8742 opcode_set = 1; 8743 break; 8744 case 's': 8745 service_action = strtoul(optarg, &endptr, 0); 8746 if (*endptr != '\0') { 8747 warnx("Invalid service action \"%s\", must " 8748 "be a number", optarg); 8749 retval = 1; 8750 goto bailout; 8751 } 8752 if (service_action > 0xffff) { 8753 warnx("Invalid service action 0x%#x, must " 8754 "be between 0 and 0xffff inclusive", 8755 service_action); 8756 retval = 1; 8757 } 8758 sa_set = 1; 8759 break; 8760 case 'T': 8761 td_set = 1; 8762 break; 8763 default: 8764 break; 8765 } 8766 } 8767 8768 if ((sa_set != 0) 8769 && (opcode_set == 0)) { 8770 warnx("You must specify an opcode with -o if a service " 8771 "action is given"); 8772 retval = 1; 8773 goto bailout; 8774 } 8775 retval = scsigetopcodes(device, opcode_set, opcode, show_sa_errors, 8776 sa_set, service_action, td_set, retry_count, 8777 timeout, verbosemode, &valid_len, &buf); 8778 if (retval != 0) 8779 goto bailout; 8780 8781 if ((opcode_set != 0) 8782 || (sa_set != 0)) { 8783 retval = scsiprintoneopcode(device, opcode, sa_set, 8784 service_action, buf, valid_len); 8785 } else { 8786 retval = scsiprintopcodes(device, td_set, buf, valid_len); 8787 } 8788 8789 bailout: 8790 free(buf); 8791 8792 return (retval); 8793 } 8794 8795 #endif /* MINIMALISTIC */ 8796 8797 static int 8798 scsireprobe(struct cam_device *device) 8799 { 8800 union ccb *ccb; 8801 int retval = 0; 8802 8803 ccb = cam_getccb(device); 8804 8805 if (ccb == NULL) { 8806 warnx("%s: error allocating ccb", __func__); 8807 return (1); 8808 } 8809 8810 CCB_CLEAR_ALL_EXCEPT_HDR(&ccb->csio); 8811 8812 ccb->ccb_h.func_code = XPT_REPROBE_LUN; 8813 8814 if (cam_send_ccb(device, ccb) < 0) { 8815 warn("error sending XPT_REPROBE_LUN CCB"); 8816 retval = 1; 8817 goto bailout; 8818 } 8819 8820 if ((ccb->ccb_h.status & CAM_STATUS_MASK) != CAM_REQ_CMP) { 8821 cam_error_print(device, ccb, CAM_ESF_ALL, CAM_EPF_ALL, stderr); 8822 retval = 1; 8823 goto bailout; 8824 } 8825 8826 bailout: 8827 cam_freeccb(ccb); 8828 8829 return (retval); 8830 } 8831 8832 void 8833 usage(int printlong) 8834 { 8835 8836 fprintf(printlong ? stdout : stderr, 8837 "usage: camcontrol <command> [device id][generic args][command args]\n" 8838 " camcontrol devlist [-b] [-v]\n" 8839 #ifndef MINIMALISTIC 8840 " camcontrol periphlist [dev_id][-n dev_name] [-u unit]\n" 8841 " camcontrol tur [dev_id][generic args]\n" 8842 " camcontrol inquiry [dev_id][generic args] [-D] [-S] [-R]\n" 8843 " camcontrol identify [dev_id][generic args] [-v]\n" 8844 " camcontrol reportluns [dev_id][generic args] [-c] [-l] [-r report]\n" 8845 " camcontrol readcap [dev_id][generic args] [-b] [-h] [-H] [-N]\n" 8846 " [-q] [-s]\n" 8847 " camcontrol start [dev_id][generic args]\n" 8848 " camcontrol stop [dev_id][generic args]\n" 8849 " camcontrol load [dev_id][generic args]\n" 8850 " camcontrol eject [dev_id][generic args]\n" 8851 " camcontrol reprobe [dev_id][generic args]\n" 8852 #endif /* MINIMALISTIC */ 8853 " camcontrol rescan <all | bus[:target:lun]>\n" 8854 " camcontrol reset <all | bus[:target:lun]>\n" 8855 #ifndef MINIMALISTIC 8856 " camcontrol defects [dev_id][generic args] <-f format> [-P][-G]\n" 8857 " [-q][-s][-S offset][-X]\n" 8858 " camcontrol modepage [dev_id][generic args] <-m page | -l>\n" 8859 " [-P pagectl][-e | -b][-d]\n" 8860 " camcontrol cmd [dev_id][generic args]\n" 8861 " <-a cmd [args] | -c cmd [args]>\n" 8862 " [-d] [-f] [-i len fmt|-o len fmt [args]] [-r fmt]\n" 8863 " camcontrol smpcmd [dev_id][generic args]\n" 8864 " <-r len fmt [args]> <-R len fmt [args]>\n" 8865 " camcontrol smprg [dev_id][generic args][-l]\n" 8866 " camcontrol smppc [dev_id][generic args] <-p phy> [-l]\n" 8867 " [-o operation][-d name][-m rate][-M rate]\n" 8868 " [-T pp_timeout][-a enable|disable]\n" 8869 " [-A enable|disable][-s enable|disable]\n" 8870 " [-S enable|disable]\n" 8871 " camcontrol smpphylist [dev_id][generic args][-l][-q]\n" 8872 " camcontrol smpmaninfo [dev_id][generic args][-l]\n" 8873 " camcontrol debug [-I][-P][-T][-S][-X][-c]\n" 8874 " <all|bus[:target[:lun]]|off>\n" 8875 " camcontrol tags [dev_id][generic args] [-N tags] [-q] [-v]\n" 8876 " camcontrol negotiate [dev_id][generic args] [-a][-c]\n" 8877 " [-D <enable|disable>][-M mode][-O offset]\n" 8878 " [-q][-R syncrate][-v][-T <enable|disable>]\n" 8879 " [-U][-W bus_width]\n" 8880 " camcontrol format [dev_id][generic args][-q][-r][-w][-y]\n" 8881 " camcontrol sanitize [dev_id][generic args]\n" 8882 " [-a overwrite|block|crypto|exitfailure]\n" 8883 " [-c passes][-I][-P pattern][-q][-U][-r][-w]\n" 8884 " [-y]\n" 8885 " camcontrol idle [dev_id][generic args][-t time]\n" 8886 " camcontrol standby [dev_id][generic args][-t time]\n" 8887 " camcontrol sleep [dev_id][generic args]\n" 8888 " camcontrol apm [dev_id][generic args][-l level]\n" 8889 " camcontrol aam [dev_id][generic args][-l level]\n" 8890 " camcontrol fwdownload [dev_id][generic args] <-f fw_image> [-q]\n" 8891 " [-s][-y]\n" 8892 " camcontrol security [dev_id][generic args]\n" 8893 " <-d pwd | -e pwd | -f | -h pwd | -k pwd>\n" 8894 " [-l <high|maximum>] [-q] [-s pwd] [-T timeout]\n" 8895 " [-U <user|master>] [-y]\n" 8896 " camcontrol hpa [dev_id][generic args] [-f] [-l] [-P] [-p pwd]\n" 8897 " [-q] [-s max_sectors] [-U pwd] [-y]\n" 8898 " camcontrol persist [dev_id][generic args] <-i action|-o action>\n" 8899 " [-a][-I tid][-k key][-K sa_key][-p][-R rtp]\n" 8900 " [-s scope][-S][-T type][-U]\n" 8901 " camcontrol attrib [dev_id][generic args] <-r action|-w attr>\n" 8902 " [-a attr_num][-c][-e elem][-F form1,form1]\n" 8903 " [-p part][-s start][-T type][-V vol]\n" 8904 " camcontrol opcodes [dev_id][generic args][-o opcode][-s SA]\n" 8905 " [-N][-T]\n" 8906 " camcontrol zone [dev_id][generic args]<-c cmd> [-a] [-l LBA]\n" 8907 " [-o rep_opts] [-P print_opts]\n" 8908 " camcontrol epc [dev_id][generic_args]<-c cmd> [-d] [-D] [-e]\n" 8909 " [-H] [-p power_cond] [-P] [-r rst_src] [-s]\n" 8910 " [-S power_src] [-T timer]\n" 8911 #endif /* MINIMALISTIC */ 8912 " camcontrol help\n"); 8913 if (!printlong) 8914 return; 8915 #ifndef MINIMALISTIC 8916 fprintf(stdout, 8917 "Specify one of the following options:\n" 8918 "devlist list all CAM devices\n" 8919 "periphlist list all CAM peripheral drivers attached to a device\n" 8920 "tur send a test unit ready to the named device\n" 8921 "inquiry send a SCSI inquiry command to the named device\n" 8922 "identify send a ATA identify command to the named device\n" 8923 "reportluns send a SCSI report luns command to the device\n" 8924 "readcap send a SCSI read capacity command to the device\n" 8925 "start send a Start Unit command to the device\n" 8926 "stop send a Stop Unit command to the device\n" 8927 "load send a Start Unit command to the device with the load bit set\n" 8928 "eject send a Stop Unit command to the device with the eject bit set\n" 8929 "reprobe update capacity information of the given device\n" 8930 "rescan rescan all busses, the given bus, or bus:target:lun\n" 8931 "reset reset all busses, the given bus, or bus:target:lun\n" 8932 "defects read the defect list of the specified device\n" 8933 "modepage display or edit (-e) the given mode page\n" 8934 "cmd send the given SCSI command, may need -i or -o as well\n" 8935 "smpcmd send the given SMP command, requires -o and -i\n" 8936 "smprg send the SMP Report General command\n" 8937 "smppc send the SMP PHY Control command, requires -p\n" 8938 "smpphylist display phys attached to a SAS expander\n" 8939 "smpmaninfo send the SMP Report Manufacturer Info command\n" 8940 "debug turn debugging on/off for a bus, target, or lun, or all devices\n" 8941 "tags report or set the number of transaction slots for a device\n" 8942 "negotiate report or set device negotiation parameters\n" 8943 "format send the SCSI FORMAT UNIT command to the named device\n" 8944 "sanitize send the SCSI SANITIZE command to the named device\n" 8945 "idle send the ATA IDLE command to the named device\n" 8946 "standby send the ATA STANDBY command to the named device\n" 8947 "sleep send the ATA SLEEP command to the named device\n" 8948 "fwdownload program firmware of the named device with the given image\n" 8949 "security report or send ATA security commands to the named device\n" 8950 "persist send the SCSI PERSISTENT RESERVE IN or OUT commands\n" 8951 "attrib send the SCSI READ or WRITE ATTRIBUTE commands\n" 8952 "opcodes send the SCSI REPORT SUPPORTED OPCODES command\n" 8953 "zone manage Zoned Block (Shingled) devices\n" 8954 "epc send ATA Extended Power Conditions commands\n" 8955 "help this message\n" 8956 "Device Identifiers:\n" 8957 "bus:target specify the bus and target, lun defaults to 0\n" 8958 "bus:target:lun specify the bus, target and lun\n" 8959 "deviceUNIT specify the device name, like \"da4\" or \"cd2\"\n" 8960 "Generic arguments:\n" 8961 "-v be verbose, print out sense information\n" 8962 "-t timeout command timeout in seconds, overrides default timeout\n" 8963 "-n dev_name specify device name, e.g. \"da\", \"cd\"\n" 8964 "-u unit specify unit number, e.g. \"0\", \"5\"\n" 8965 "-E have the kernel attempt to perform SCSI error recovery\n" 8966 "-C count specify the SCSI command retry count (needs -E to work)\n" 8967 "modepage arguments:\n" 8968 "-l list all available mode pages\n" 8969 "-m page specify the mode page to view or edit\n" 8970 "-e edit the specified mode page\n" 8971 "-b force view to binary mode\n" 8972 "-d disable block descriptors for mode sense\n" 8973 "-P pgctl page control field 0-3\n" 8974 "defects arguments:\n" 8975 "-f format specify defect list format (block, bfi or phys)\n" 8976 "-G get the grown defect list\n" 8977 "-P get the permanent defect list\n" 8978 "inquiry arguments:\n" 8979 "-D get the standard inquiry data\n" 8980 "-S get the serial number\n" 8981 "-R get the transfer rate, etc.\n" 8982 "reportluns arguments:\n" 8983 "-c only report a count of available LUNs\n" 8984 "-l only print out luns, and not a count\n" 8985 "-r <reporttype> specify \"default\", \"wellknown\" or \"all\"\n" 8986 "readcap arguments\n" 8987 "-b only report the blocksize\n" 8988 "-h human readable device size, base 2\n" 8989 "-H human readable device size, base 10\n" 8990 "-N print the number of blocks instead of last block\n" 8991 "-q quiet, print numbers only\n" 8992 "-s only report the last block/device size\n" 8993 "cmd arguments:\n" 8994 "-c cdb [args] specify the SCSI CDB\n" 8995 "-i len fmt specify input data and input data format\n" 8996 "-o len fmt [args] specify output data and output data fmt\n" 8997 "smpcmd arguments:\n" 8998 "-r len fmt [args] specify the SMP command to be sent\n" 8999 "-R len fmt [args] specify SMP response format\n" 9000 "smprg arguments:\n" 9001 "-l specify the long response format\n" 9002 "smppc arguments:\n" 9003 "-p phy specify the PHY to operate on\n" 9004 "-l specify the long request/response format\n" 9005 "-o operation specify the phy control operation\n" 9006 "-d name set the attached device name\n" 9007 "-m rate set the minimum physical link rate\n" 9008 "-M rate set the maximum physical link rate\n" 9009 "-T pp_timeout set the partial pathway timeout value\n" 9010 "-a enable|disable enable or disable SATA slumber\n" 9011 "-A enable|disable enable or disable SATA partial phy power\n" 9012 "-s enable|disable enable or disable SAS slumber\n" 9013 "-S enable|disable enable or disable SAS partial phy power\n" 9014 "smpphylist arguments:\n" 9015 "-l specify the long response format\n" 9016 "-q only print phys with attached devices\n" 9017 "smpmaninfo arguments:\n" 9018 "-l specify the long response format\n" 9019 "debug arguments:\n" 9020 "-I CAM_DEBUG_INFO -- scsi commands, errors, data\n" 9021 "-T CAM_DEBUG_TRACE -- routine flow tracking\n" 9022 "-S CAM_DEBUG_SUBTRACE -- internal routine command flow\n" 9023 "-c CAM_DEBUG_CDB -- print out SCSI CDBs only\n" 9024 "tags arguments:\n" 9025 "-N tags specify the number of tags to use for this device\n" 9026 "-q be quiet, don't report the number of tags\n" 9027 "-v report a number of tag-related parameters\n" 9028 "negotiate arguments:\n" 9029 "-a send a test unit ready after negotiation\n" 9030 "-c report/set current negotiation settings\n" 9031 "-D <arg> \"enable\" or \"disable\" disconnection\n" 9032 "-M mode set ATA mode\n" 9033 "-O offset set command delay offset\n" 9034 "-q be quiet, don't report anything\n" 9035 "-R syncrate synchronization rate in MHz\n" 9036 "-T <arg> \"enable\" or \"disable\" tagged queueing\n" 9037 "-U report/set user negotiation settings\n" 9038 "-W bus_width set the bus width in bits (8, 16 or 32)\n" 9039 "-v also print a Path Inquiry CCB for the controller\n" 9040 "format arguments:\n" 9041 "-q be quiet, don't print status messages\n" 9042 "-r run in report only mode\n" 9043 "-w don't send immediate format command\n" 9044 "-y don't ask any questions\n" 9045 "sanitize arguments:\n" 9046 "-a operation operation mode: overwrite, block, crypto or exitfailure\n" 9047 "-c passes overwrite passes to perform (1 to 31)\n" 9048 "-I invert overwrite pattern after each pass\n" 9049 "-P pattern path to overwrite pattern file\n" 9050 "-q be quiet, don't print status messages\n" 9051 "-r run in report only mode\n" 9052 "-U run operation in unrestricted completion exit mode\n" 9053 "-w don't send immediate sanitize command\n" 9054 "-y don't ask any questions\n" 9055 "idle/standby arguments:\n" 9056 "-t <arg> number of seconds before respective state.\n" 9057 "fwdownload arguments:\n" 9058 "-f fw_image path to firmware image file\n" 9059 "-q don't print informational messages, only errors\n" 9060 "-s run in simulation mode\n" 9061 "-v print info for every firmware segment sent to device\n" 9062 "-y don't ask any questions\n" 9063 "security arguments:\n" 9064 "-d pwd disable security using the given password for the selected\n" 9065 " user\n" 9066 "-e pwd erase the device using the given pwd for the selected user\n" 9067 "-f freeze the security configuration of the specified device\n" 9068 "-h pwd enhanced erase the device using the given pwd for the\n" 9069 " selected user\n" 9070 "-k pwd unlock the device using the given pwd for the selected\n" 9071 " user\n" 9072 "-l <high|maximum> specifies which security level to set: high or maximum\n" 9073 "-q be quiet, do not print any status messages\n" 9074 "-s pwd password the device (enable security) using the given\n" 9075 " pwd for the selected user\n" 9076 "-T timeout overrides the timeout (seconds) used for erase operation\n" 9077 "-U <user|master> specifies which user to set: user or master\n" 9078 "-y don't ask any questions\n" 9079 "hpa arguments:\n" 9080 "-f freeze the HPA configuration of the device\n" 9081 "-l lock the HPA configuration of the device\n" 9082 "-P make the HPA max sectors persist\n" 9083 "-p pwd Set the HPA configuration password required for unlock\n" 9084 " calls\n" 9085 "-q be quiet, do not print any status messages\n" 9086 "-s sectors configures the maximum user accessible sectors of the\n" 9087 " device\n" 9088 "-U pwd unlock the HPA configuration of the device\n" 9089 "-y don't ask any questions\n" 9090 "persist arguments:\n" 9091 "-i action specify read_keys, read_reservation, report_cap, or\n" 9092 " read_full_status\n" 9093 "-o action specify register, register_ignore, reserve, release,\n" 9094 " clear, preempt, preempt_abort, register_move, replace_lost\n" 9095 "-a set the All Target Ports (ALL_TG_PT) bit\n" 9096 "-I tid specify a Transport ID, e.g.: sas,0x1234567812345678\n" 9097 "-k key specify the Reservation Key\n" 9098 "-K sa_key specify the Service Action Reservation Key\n" 9099 "-p set the Activate Persist Through Power Loss bit\n" 9100 "-R rtp specify the Relative Target Port\n" 9101 "-s scope specify the scope: lun, extent, element or a number\n" 9102 "-S specify Transport ID for register, requires -I\n" 9103 "-T res_type specify the reservation type: read_shared, wr_ex, rd_ex,\n" 9104 " ex_ac, wr_ex_ro, ex_ac_ro, wr_ex_ar, ex_ac_ar\n" 9105 "-U unregister the current initiator for register_move\n" 9106 "attrib arguments:\n" 9107 "-r action specify attr_values, attr_list, lv_list, part_list, or\n" 9108 " supp_attr\n" 9109 "-w attr specify an attribute to write, one -w argument per attr\n" 9110 "-a attr_num only display this attribute number\n" 9111 "-c get cached attributes\n" 9112 "-e elem_addr request attributes for the given element in a changer\n" 9113 "-F form1,form2 output format, comma separated list: text_esc, text_raw,\n" 9114 " nonascii_esc, nonascii_trim, nonascii_raw, field_all,\n" 9115 " field_none, field_desc, field_num, field_size, field_rw\n" 9116 "-p partition request attributes for the given partition\n" 9117 "-s start_attr request attributes starting at the given number\n" 9118 "-T elem_type specify the element type (used with -e)\n" 9119 "-V logical_vol specify the logical volume ID\n" 9120 "opcodes arguments:\n" 9121 "-o opcode specify the individual opcode to list\n" 9122 "-s service_action specify the service action for the opcode\n" 9123 "-N do not return SCSI error for unsupported SA\n" 9124 "-T request nominal and recommended timeout values\n" 9125 "zone arguments:\n" 9126 "-c cmd required: rz, open, close, finish, or rwp\n" 9127 "-a apply the action to all zones\n" 9128 "-l LBA specify the zone starting LBA\n" 9129 "-o rep_opts report zones options: all, empty, imp_open, exp_open,\n" 9130 " closed, full, ro, offline, reset, nonseq, nonwp\n" 9131 "-P print_opt report zones printing: normal, summary, script\n" 9132 "epc arguments:\n" 9133 "-c cmd required: restore, goto, timer, state, enable, disable,\n" 9134 " source, status, list\n" 9135 "-d disable power mode (timer, state)\n" 9136 "-D delayed entry (goto)\n" 9137 "-e enable power mode (timer, state)\n" 9138 "-H hold power mode (goto)\n" 9139 "-p power_cond Idle_a, Idle_b, Idle_c, Standby_y, Standby_z (timer,\n" 9140 " state, goto)\n" 9141 "-P only display power mode (status)\n" 9142 "-r rst_src restore settings from: default, saved (restore)\n" 9143 "-s save mode (timer, state, restore)\n" 9144 "-S power_src set power source: battery, nonbattery (source)\n" 9145 "-T timer set timer, seconds, .1 sec resolution (timer)\n" 9146 ); 9147 #endif /* MINIMALISTIC */ 9148 } 9149 9150 int 9151 main(int argc, char **argv) 9152 { 9153 int c; 9154 char *device = NULL; 9155 int unit = 0; 9156 struct cam_device *cam_dev = NULL; 9157 int timeout = 0, retry_count = 1; 9158 camcontrol_optret optreturn; 9159 char *tstr; 9160 const char *mainopt = "C:En:t:u:v"; 9161 const char *subopt = NULL; 9162 char combinedopt[256]; 9163 int error = 0, optstart = 2; 9164 int devopen = 1; 9165 #ifndef MINIMALISTIC 9166 path_id_t bus; 9167 target_id_t target; 9168 lun_id_t lun; 9169 #endif /* MINIMALISTIC */ 9170 9171 cmdlist = CAM_CMD_NONE; 9172 arglist = CAM_ARG_NONE; 9173 9174 if (argc < 2) { 9175 usage(0); 9176 exit(1); 9177 } 9178 9179 /* 9180 * Get the base option. 9181 */ 9182 optreturn = getoption(option_table,argv[1], &cmdlist, &arglist,&subopt); 9183 9184 if (optreturn == CC_OR_AMBIGUOUS) { 9185 warnx("ambiguous option %s", argv[1]); 9186 usage(0); 9187 exit(1); 9188 } else if (optreturn == CC_OR_NOT_FOUND) { 9189 warnx("option %s not found", argv[1]); 9190 usage(0); 9191 exit(1); 9192 } 9193 9194 /* 9195 * Ahh, getopt(3) is a pain. 9196 * 9197 * This is a gross hack. There really aren't many other good 9198 * options (excuse the pun) for parsing options in a situation like 9199 * this. getopt is kinda braindead, so you end up having to run 9200 * through the options twice, and give each invocation of getopt 9201 * the option string for the other invocation. 9202 * 9203 * You would think that you could just have two groups of options. 9204 * The first group would get parsed by the first invocation of 9205 * getopt, and the second group would get parsed by the second 9206 * invocation of getopt. It doesn't quite work out that way. When 9207 * the first invocation of getopt finishes, it leaves optind pointing 9208 * to the argument _after_ the first argument in the second group. 9209 * So when the second invocation of getopt comes around, it doesn't 9210 * recognize the first argument it gets and then bails out. 9211 * 9212 * A nice alternative would be to have a flag for getopt that says 9213 * "just keep parsing arguments even when you encounter an unknown 9214 * argument", but there isn't one. So there's no real clean way to 9215 * easily parse two sets of arguments without having one invocation 9216 * of getopt know about the other. 9217 * 9218 * Without this hack, the first invocation of getopt would work as 9219 * long as the generic arguments are first, but the second invocation 9220 * (in the subfunction) would fail in one of two ways. In the case 9221 * where you don't set optreset, it would fail because optind may be 9222 * pointing to the argument after the one it should be pointing at. 9223 * In the case where you do set optreset, and reset optind, it would 9224 * fail because getopt would run into the first set of options, which 9225 * it doesn't understand. 9226 * 9227 * All of this would "sort of" work if you could somehow figure out 9228 * whether optind had been incremented one option too far. The 9229 * mechanics of that, however, are more daunting than just giving 9230 * both invocations all of the expect options for either invocation. 9231 * 9232 * Needless to say, I wouldn't mind if someone invented a better 9233 * (non-GPL!) command line parsing interface than getopt. I 9234 * wouldn't mind if someone added more knobs to getopt to make it 9235 * work better. Who knows, I may talk myself into doing it someday, 9236 * if the standards weenies let me. As it is, it just leads to 9237 * hackery like this and causes people to avoid it in some cases. 9238 * 9239 * KDM, September 8th, 1998 9240 */ 9241 if (subopt != NULL) 9242 sprintf(combinedopt, "%s%s", mainopt, subopt); 9243 else 9244 sprintf(combinedopt, "%s", mainopt); 9245 9246 /* 9247 * For these options we do not parse optional device arguments and 9248 * we do not open a passthrough device. 9249 */ 9250 if ((cmdlist == CAM_CMD_RESCAN) 9251 || (cmdlist == CAM_CMD_RESET) 9252 || (cmdlist == CAM_CMD_DEVTREE) 9253 || (cmdlist == CAM_CMD_USAGE) 9254 || (cmdlist == CAM_CMD_DEBUG)) 9255 devopen = 0; 9256 9257 #ifndef MINIMALISTIC 9258 if ((devopen == 1) 9259 && (argc > 2 && argv[2][0] != '-')) { 9260 char name[30]; 9261 int rv; 9262 9263 if (isdigit(argv[2][0])) { 9264 /* device specified as bus:target[:lun] */ 9265 rv = parse_btl(argv[2], &bus, &target, &lun, &arglist); 9266 if (rv < 2) 9267 errx(1, "numeric device specification must " 9268 "be either bus:target, or " 9269 "bus:target:lun"); 9270 /* default to 0 if lun was not specified */ 9271 if ((arglist & CAM_ARG_LUN) == 0) { 9272 lun = 0; 9273 arglist |= CAM_ARG_LUN; 9274 } 9275 optstart++; 9276 } else { 9277 if (cam_get_device(argv[2], name, sizeof name, &unit) 9278 == -1) 9279 errx(1, "%s", cam_errbuf); 9280 device = strdup(name); 9281 arglist |= CAM_ARG_DEVICE | CAM_ARG_UNIT; 9282 optstart++; 9283 } 9284 } 9285 #endif /* MINIMALISTIC */ 9286 /* 9287 * Start getopt processing at argv[2/3], since we've already 9288 * accepted argv[1..2] as the command name, and as a possible 9289 * device name. 9290 */ 9291 optind = optstart; 9292 9293 /* 9294 * Now we run through the argument list looking for generic 9295 * options, and ignoring options that possibly belong to 9296 * subfunctions. 9297 */ 9298 while ((c = getopt(argc, argv, combinedopt))!= -1){ 9299 switch(c) { 9300 case 'C': 9301 retry_count = strtol(optarg, NULL, 0); 9302 if (retry_count < 0) 9303 errx(1, "retry count %d is < 0", 9304 retry_count); 9305 arglist |= CAM_ARG_RETRIES; 9306 break; 9307 case 'E': 9308 arglist |= CAM_ARG_ERR_RECOVER; 9309 break; 9310 case 'n': 9311 arglist |= CAM_ARG_DEVICE; 9312 tstr = optarg; 9313 while (isspace(*tstr) && (*tstr != '\0')) 9314 tstr++; 9315 device = (char *)strdup(tstr); 9316 break; 9317 case 't': 9318 timeout = strtol(optarg, NULL, 0); 9319 if (timeout < 0) 9320 errx(1, "invalid timeout %d", timeout); 9321 /* Convert the timeout from seconds to ms */ 9322 timeout *= 1000; 9323 arglist |= CAM_ARG_TIMEOUT; 9324 break; 9325 case 'u': 9326 arglist |= CAM_ARG_UNIT; 9327 unit = strtol(optarg, NULL, 0); 9328 break; 9329 case 'v': 9330 arglist |= CAM_ARG_VERBOSE; 9331 break; 9332 default: 9333 break; 9334 } 9335 } 9336 9337 #ifndef MINIMALISTIC 9338 /* 9339 * For most commands we'll want to open the passthrough device 9340 * associated with the specified device. In the case of the rescan 9341 * commands, we don't use a passthrough device at all, just the 9342 * transport layer device. 9343 */ 9344 if (devopen == 1) { 9345 if (((arglist & (CAM_ARG_BUS|CAM_ARG_TARGET)) == 0) 9346 && (((arglist & CAM_ARG_DEVICE) == 0) 9347 || ((arglist & CAM_ARG_UNIT) == 0))) { 9348 errx(1, "subcommand \"%s\" requires a valid device " 9349 "identifier", argv[1]); 9350 } 9351 9352 if ((cam_dev = ((arglist & (CAM_ARG_BUS | CAM_ARG_TARGET))? 9353 cam_open_btl(bus, target, lun, O_RDWR, NULL) : 9354 cam_open_spec_device(device,unit,O_RDWR,NULL))) 9355 == NULL) 9356 errx(1,"%s", cam_errbuf); 9357 } 9358 #endif /* MINIMALISTIC */ 9359 9360 /* 9361 * Reset optind to 2, and reset getopt, so these routines can parse 9362 * the arguments again. 9363 */ 9364 optind = optstart; 9365 optreset = 1; 9366 9367 switch(cmdlist) { 9368 #ifndef MINIMALISTIC 9369 case CAM_CMD_DEVLIST: 9370 error = getdevlist(cam_dev); 9371 break; 9372 case CAM_CMD_HPA: 9373 error = atahpa(cam_dev, retry_count, timeout, 9374 argc, argv, combinedopt); 9375 break; 9376 #endif /* MINIMALISTIC */ 9377 case CAM_CMD_DEVTREE: 9378 error = getdevtree(argc, argv, combinedopt); 9379 break; 9380 #ifndef MINIMALISTIC 9381 case CAM_CMD_TUR: 9382 error = testunitready(cam_dev, retry_count, timeout, 0); 9383 break; 9384 case CAM_CMD_INQUIRY: 9385 error = scsidoinquiry(cam_dev, argc, argv, combinedopt, 9386 retry_count, timeout); 9387 break; 9388 case CAM_CMD_IDENTIFY: 9389 error = ataidentify(cam_dev, retry_count, timeout); 9390 break; 9391 case CAM_CMD_STARTSTOP: 9392 error = scsistart(cam_dev, arglist & CAM_ARG_START_UNIT, 9393 arglist & CAM_ARG_EJECT, retry_count, 9394 timeout); 9395 break; 9396 #endif /* MINIMALISTIC */ 9397 case CAM_CMD_RESCAN: 9398 error = dorescan_or_reset(argc, argv, 1); 9399 break; 9400 case CAM_CMD_RESET: 9401 error = dorescan_or_reset(argc, argv, 0); 9402 break; 9403 #ifndef MINIMALISTIC 9404 case CAM_CMD_READ_DEFECTS: 9405 error = readdefects(cam_dev, argc, argv, combinedopt, 9406 retry_count, timeout); 9407 break; 9408 case CAM_CMD_MODE_PAGE: 9409 modepage(cam_dev, argc, argv, combinedopt, 9410 retry_count, timeout); 9411 break; 9412 case CAM_CMD_SCSI_CMD: 9413 error = scsicmd(cam_dev, argc, argv, combinedopt, 9414 retry_count, timeout); 9415 break; 9416 case CAM_CMD_SMP_CMD: 9417 error = smpcmd(cam_dev, argc, argv, combinedopt, 9418 retry_count, timeout); 9419 break; 9420 case CAM_CMD_SMP_RG: 9421 error = smpreportgeneral(cam_dev, argc, argv, 9422 combinedopt, retry_count, 9423 timeout); 9424 break; 9425 case CAM_CMD_SMP_PC: 9426 error = smpphycontrol(cam_dev, argc, argv, combinedopt, 9427 retry_count, timeout); 9428 break; 9429 case CAM_CMD_SMP_PHYLIST: 9430 error = smpphylist(cam_dev, argc, argv, combinedopt, 9431 retry_count, timeout); 9432 break; 9433 case CAM_CMD_SMP_MANINFO: 9434 error = smpmaninfo(cam_dev, argc, argv, combinedopt, 9435 retry_count, timeout); 9436 break; 9437 case CAM_CMD_DEBUG: 9438 error = camdebug(argc, argv, combinedopt); 9439 break; 9440 case CAM_CMD_TAG: 9441 error = tagcontrol(cam_dev, argc, argv, combinedopt); 9442 break; 9443 case CAM_CMD_RATE: 9444 error = ratecontrol(cam_dev, retry_count, timeout, 9445 argc, argv, combinedopt); 9446 break; 9447 case CAM_CMD_FORMAT: 9448 error = scsiformat(cam_dev, argc, argv, 9449 combinedopt, retry_count, timeout); 9450 break; 9451 case CAM_CMD_REPORTLUNS: 9452 error = scsireportluns(cam_dev, argc, argv, 9453 combinedopt, retry_count, 9454 timeout); 9455 break; 9456 case CAM_CMD_READCAP: 9457 error = scsireadcapacity(cam_dev, argc, argv, 9458 combinedopt, retry_count, 9459 timeout); 9460 break; 9461 case CAM_CMD_IDLE: 9462 case CAM_CMD_STANDBY: 9463 case CAM_CMD_SLEEP: 9464 error = atapm(cam_dev, argc, argv, 9465 combinedopt, retry_count, timeout); 9466 break; 9467 case CAM_CMD_APM: 9468 case CAM_CMD_AAM: 9469 error = ataaxm(cam_dev, argc, argv, 9470 combinedopt, retry_count, timeout); 9471 break; 9472 case CAM_CMD_SECURITY: 9473 error = atasecurity(cam_dev, retry_count, timeout, 9474 argc, argv, combinedopt); 9475 break; 9476 case CAM_CMD_DOWNLOAD_FW: 9477 error = fwdownload(cam_dev, argc, argv, combinedopt, 9478 arglist & CAM_ARG_VERBOSE, retry_count, timeout); 9479 break; 9480 case CAM_CMD_SANITIZE: 9481 error = scsisanitize(cam_dev, argc, argv, 9482 combinedopt, retry_count, timeout); 9483 break; 9484 case CAM_CMD_PERSIST: 9485 error = scsipersist(cam_dev, argc, argv, combinedopt, 9486 retry_count, timeout, arglist & CAM_ARG_VERBOSE, 9487 arglist & CAM_ARG_ERR_RECOVER); 9488 break; 9489 case CAM_CMD_ATTRIB: 9490 error = scsiattrib(cam_dev, argc, argv, combinedopt, 9491 retry_count, timeout, arglist & CAM_ARG_VERBOSE, 9492 arglist & CAM_ARG_ERR_RECOVER); 9493 break; 9494 case CAM_CMD_OPCODES: 9495 error = scsiopcodes(cam_dev, argc, argv, combinedopt, 9496 retry_count, timeout, arglist & CAM_ARG_VERBOSE); 9497 break; 9498 case CAM_CMD_REPROBE: 9499 error = scsireprobe(cam_dev); 9500 break; 9501 case CAM_CMD_ZONE: 9502 error = zone(cam_dev, argc, argv, combinedopt, 9503 retry_count, timeout, arglist & CAM_ARG_VERBOSE); 9504 break; 9505 case CAM_CMD_EPC: 9506 error = epc(cam_dev, argc, argv, combinedopt, 9507 retry_count, timeout, arglist & CAM_ARG_VERBOSE); 9508 break; 9509 #endif /* MINIMALISTIC */ 9510 case CAM_CMD_USAGE: 9511 usage(1); 9512 break; 9513 default: 9514 usage(0); 9515 error = 1; 9516 break; 9517 } 9518 9519 if (cam_dev != NULL) 9520 cam_close_device(cam_dev); 9521 9522 exit(error); 9523 } 9524