qemu with hax to log dma reads & writes jcs.org/2018/11/12/vfio
at master 600 lines 20 kB view raw
1/* 2 * SCSI helpers 3 * 4 * Copyright 2017 Red Hat, Inc. 5 * 6 * Authors: 7 * Fam Zheng <famz@redhat.com> 8 * Paolo Bonzini <pbonzini@redhat.com> 9 * 10 * This program is free software; you can redistribute it and/or modify it 11 * under the terms of the GNU General Public License as published by the Free 12 * Software Foundation; either version 2 of the License, or (at your option) 13 * any later version. 14 */ 15 16#include "qemu/osdep.h" 17#include "scsi/constants.h" 18#include "scsi/utils.h" 19#include "qemu/bswap.h" 20 21uint32_t scsi_data_cdb_xfer(uint8_t *buf) 22{ 23 if ((buf[0] >> 5) == 0 && buf[4] == 0) { 24 return 256; 25 } else { 26 return scsi_cdb_xfer(buf); 27 } 28} 29 30uint32_t scsi_cdb_xfer(uint8_t *buf) 31{ 32 switch (buf[0] >> 5) { 33 case 0: 34 return buf[4]; 35 break; 36 case 1: 37 case 2: 38 return lduw_be_p(&buf[7]); 39 break; 40 case 4: 41 return ldl_be_p(&buf[10]) & 0xffffffffULL; 42 break; 43 case 5: 44 return ldl_be_p(&buf[6]) & 0xffffffffULL; 45 break; 46 default: 47 return -1; 48 } 49} 50 51uint64_t scsi_cmd_lba(SCSICommand *cmd) 52{ 53 uint8_t *buf = cmd->buf; 54 uint64_t lba; 55 56 switch (buf[0] >> 5) { 57 case 0: 58 lba = ldl_be_p(&buf[0]) & 0x1fffff; 59 break; 60 case 1: 61 case 2: 62 case 5: 63 lba = ldl_be_p(&buf[2]) & 0xffffffffULL; 64 break; 65 case 4: 66 lba = ldq_be_p(&buf[2]); 67 break; 68 default: 69 lba = -1; 70 71 } 72 return lba; 73} 74 75int scsi_cdb_length(uint8_t *buf) 76{ 77 int cdb_len; 78 79 switch (buf[0] >> 5) { 80 case 0: 81 cdb_len = 6; 82 break; 83 case 1: 84 case 2: 85 cdb_len = 10; 86 break; 87 case 4: 88 cdb_len = 16; 89 break; 90 case 5: 91 cdb_len = 12; 92 break; 93 default: 94 cdb_len = -1; 95 } 96 return cdb_len; 97} 98 99SCSISense scsi_parse_sense_buf(const uint8_t *in_buf, int in_len) 100{ 101 bool fixed_in; 102 SCSISense sense; 103 104 assert(in_len > 0); 105 fixed_in = (in_buf[0] & 2) == 0; 106 if (fixed_in) { 107 if (in_len < 14) { 108 return SENSE_CODE(IO_ERROR); 109 } 110 sense.key = in_buf[2]; 111 sense.asc = in_buf[12]; 112 sense.ascq = in_buf[13]; 113 } else { 114 if (in_len < 4) { 115 return SENSE_CODE(IO_ERROR); 116 } 117 sense.key = in_buf[1]; 118 sense.asc = in_buf[2]; 119 sense.ascq = in_buf[3]; 120 } 121 122 return sense; 123} 124 125int scsi_build_sense_buf(uint8_t *out_buf, size_t size, SCSISense sense, 126 bool fixed_sense) 127{ 128 int len; 129 uint8_t buf[SCSI_SENSE_LEN] = { 0 }; 130 131 if (fixed_sense) { 132 buf[0] = 0x70; 133 buf[2] = sense.key; 134 buf[7] = 10; 135 buf[12] = sense.asc; 136 buf[13] = sense.ascq; 137 len = 18; 138 } else { 139 buf[0] = 0x72; 140 buf[1] = sense.key; 141 buf[2] = sense.asc; 142 buf[3] = sense.ascq; 143 len = 8; 144 } 145 len = MIN(len, size); 146 memcpy(out_buf, buf, len); 147 return len; 148} 149 150int scsi_build_sense(uint8_t *buf, SCSISense sense) 151{ 152 return scsi_build_sense_buf(buf, SCSI_SENSE_LEN, sense, true); 153} 154 155/* 156 * Predefined sense codes 157 */ 158 159/* No sense data available */ 160const struct SCSISense sense_code_NO_SENSE = { 161 .key = NO_SENSE , .asc = 0x00 , .ascq = 0x00 162}; 163 164/* LUN not ready, Manual intervention required */ 165const struct SCSISense sense_code_LUN_NOT_READY = { 166 .key = NOT_READY, .asc = 0x04, .ascq = 0x03 167}; 168 169/* LUN not ready, Medium not present */ 170const struct SCSISense sense_code_NO_MEDIUM = { 171 .key = NOT_READY, .asc = 0x3a, .ascq = 0x00 172}; 173 174/* LUN not ready, medium removal prevented */ 175const struct SCSISense sense_code_NOT_READY_REMOVAL_PREVENTED = { 176 .key = NOT_READY, .asc = 0x53, .ascq = 0x02 177}; 178 179/* Hardware error, internal target failure */ 180const struct SCSISense sense_code_TARGET_FAILURE = { 181 .key = HARDWARE_ERROR, .asc = 0x44, .ascq = 0x00 182}; 183 184/* Illegal request, invalid command operation code */ 185const struct SCSISense sense_code_INVALID_OPCODE = { 186 .key = ILLEGAL_REQUEST, .asc = 0x20, .ascq = 0x00 187}; 188 189/* Illegal request, LBA out of range */ 190const struct SCSISense sense_code_LBA_OUT_OF_RANGE = { 191 .key = ILLEGAL_REQUEST, .asc = 0x21, .ascq = 0x00 192}; 193 194/* Illegal request, Invalid field in CDB */ 195const struct SCSISense sense_code_INVALID_FIELD = { 196 .key = ILLEGAL_REQUEST, .asc = 0x24, .ascq = 0x00 197}; 198 199/* Illegal request, Invalid field in parameter list */ 200const struct SCSISense sense_code_INVALID_PARAM = { 201 .key = ILLEGAL_REQUEST, .asc = 0x26, .ascq = 0x00 202}; 203 204/* Illegal request, Parameter list length error */ 205const struct SCSISense sense_code_INVALID_PARAM_LEN = { 206 .key = ILLEGAL_REQUEST, .asc = 0x1a, .ascq = 0x00 207}; 208 209/* Illegal request, LUN not supported */ 210const struct SCSISense sense_code_LUN_NOT_SUPPORTED = { 211 .key = ILLEGAL_REQUEST, .asc = 0x25, .ascq = 0x00 212}; 213 214/* Illegal request, Saving parameters not supported */ 215const struct SCSISense sense_code_SAVING_PARAMS_NOT_SUPPORTED = { 216 .key = ILLEGAL_REQUEST, .asc = 0x39, .ascq = 0x00 217}; 218 219/* Illegal request, Incompatible medium installed */ 220const struct SCSISense sense_code_INCOMPATIBLE_FORMAT = { 221 .key = ILLEGAL_REQUEST, .asc = 0x30, .ascq = 0x00 222}; 223 224/* Illegal request, medium removal prevented */ 225const struct SCSISense sense_code_ILLEGAL_REQ_REMOVAL_PREVENTED = { 226 .key = ILLEGAL_REQUEST, .asc = 0x53, .ascq = 0x02 227}; 228 229/* Illegal request, Invalid Transfer Tag */ 230const struct SCSISense sense_code_INVALID_TAG = { 231 .key = ILLEGAL_REQUEST, .asc = 0x4b, .ascq = 0x01 232}; 233 234/* Command aborted, I/O process terminated */ 235const struct SCSISense sense_code_IO_ERROR = { 236 .key = ABORTED_COMMAND, .asc = 0x00, .ascq = 0x06 237}; 238 239/* Command aborted, I_T Nexus loss occurred */ 240const struct SCSISense sense_code_I_T_NEXUS_LOSS = { 241 .key = ABORTED_COMMAND, .asc = 0x29, .ascq = 0x07 242}; 243 244/* Command aborted, Logical Unit failure */ 245const struct SCSISense sense_code_LUN_FAILURE = { 246 .key = ABORTED_COMMAND, .asc = 0x3e, .ascq = 0x01 247}; 248 249/* Command aborted, Overlapped Commands Attempted */ 250const struct SCSISense sense_code_OVERLAPPED_COMMANDS = { 251 .key = ABORTED_COMMAND, .asc = 0x4e, .ascq = 0x00 252}; 253 254/* Command aborted, LUN Communication Failure */ 255const struct SCSISense sense_code_LUN_COMM_FAILURE = { 256 .key = ABORTED_COMMAND, .asc = 0x08, .ascq = 0x00 257}; 258 259/* Medium Error, Unrecovered read error */ 260const struct SCSISense sense_code_READ_ERROR = { 261 .key = MEDIUM_ERROR, .asc = 0x11, .ascq = 0x00 262}; 263 264/* Not ready, Cause not reportable */ 265const struct SCSISense sense_code_NOT_READY = { 266 .key = NOT_READY, .asc = 0x04, .ascq = 0x00 267}; 268 269/* Unit attention, Capacity data has changed */ 270const struct SCSISense sense_code_CAPACITY_CHANGED = { 271 .key = UNIT_ATTENTION, .asc = 0x2a, .ascq = 0x09 272}; 273 274/* Unit attention, Power on, reset or bus device reset occurred */ 275const struct SCSISense sense_code_RESET = { 276 .key = UNIT_ATTENTION, .asc = 0x29, .ascq = 0x00 277}; 278 279/* Unit attention, SCSI bus reset */ 280const struct SCSISense sense_code_SCSI_BUS_RESET = { 281 .key = UNIT_ATTENTION, .asc = 0x29, .ascq = 0x02 282}; 283 284/* Unit attention, No medium */ 285const struct SCSISense sense_code_UNIT_ATTENTION_NO_MEDIUM = { 286 .key = UNIT_ATTENTION, .asc = 0x3a, .ascq = 0x00 287}; 288 289/* Unit attention, Medium may have changed */ 290const struct SCSISense sense_code_MEDIUM_CHANGED = { 291 .key = UNIT_ATTENTION, .asc = 0x28, .ascq = 0x00 292}; 293 294/* Unit attention, Reported LUNs data has changed */ 295const struct SCSISense sense_code_REPORTED_LUNS_CHANGED = { 296 .key = UNIT_ATTENTION, .asc = 0x3f, .ascq = 0x0e 297}; 298 299/* Unit attention, Device internal reset */ 300const struct SCSISense sense_code_DEVICE_INTERNAL_RESET = { 301 .key = UNIT_ATTENTION, .asc = 0x29, .ascq = 0x04 302}; 303 304/* Data Protection, Write Protected */ 305const struct SCSISense sense_code_WRITE_PROTECTED = { 306 .key = DATA_PROTECT, .asc = 0x27, .ascq = 0x00 307}; 308 309/* Data Protection, Space Allocation Failed Write Protect */ 310const struct SCSISense sense_code_SPACE_ALLOC_FAILED = { 311 .key = DATA_PROTECT, .asc = 0x27, .ascq = 0x07 312}; 313 314/* 315 * scsi_convert_sense 316 * 317 * Convert between fixed and descriptor sense buffers 318 */ 319int scsi_convert_sense(uint8_t *in_buf, int in_len, 320 uint8_t *buf, int len, bool fixed) 321{ 322 SCSISense sense; 323 bool fixed_in; 324 325 if (in_len == 0) { 326 return scsi_build_sense_buf(buf, len, SENSE_CODE(NO_SENSE), fixed); 327 } 328 329 fixed_in = (in_buf[0] & 2) == 0; 330 if (fixed == fixed_in) { 331 memcpy(buf, in_buf, MIN(len, in_len)); 332 return MIN(len, in_len); 333 } else { 334 sense = scsi_parse_sense_buf(in_buf, in_len); 335 return scsi_build_sense_buf(buf, len, sense, fixed); 336 } 337} 338 339static bool scsi_sense_is_guest_recoverable(int key, int asc, int ascq) 340{ 341 switch (key) { 342 case NO_SENSE: 343 case RECOVERED_ERROR: 344 case UNIT_ATTENTION: 345 case ABORTED_COMMAND: 346 return true; 347 case NOT_READY: 348 case ILLEGAL_REQUEST: 349 case DATA_PROTECT: 350 /* Parse ASCQ */ 351 break; 352 default: 353 return false; 354 } 355 356 switch ((asc << 8) | ascq) { 357 case 0x1a00: /* PARAMETER LIST LENGTH ERROR */ 358 case 0x2000: /* INVALID OPERATION CODE */ 359 case 0x2400: /* INVALID FIELD IN CDB */ 360 case 0x2500: /* LOGICAL UNIT NOT SUPPORTED */ 361 case 0x2600: /* INVALID FIELD IN PARAMETER LIST */ 362 363 case 0x2104: /* UNALIGNED WRITE COMMAND */ 364 case 0x2105: /* WRITE BOUNDARY VIOLATION */ 365 case 0x2106: /* ATTEMPT TO READ INVALID DATA */ 366 case 0x550e: /* INSUFFICIENT ZONE RESOURCES */ 367 368 case 0x0401: /* NOT READY, IN PROGRESS OF BECOMING READY */ 369 case 0x0402: /* NOT READY, INITIALIZING COMMAND REQUIRED */ 370 return true; 371 default: 372 return false; 373 } 374} 375 376int scsi_sense_to_errno(int key, int asc, int ascq) 377{ 378 switch (key) { 379 case NO_SENSE: 380 case RECOVERED_ERROR: 381 case UNIT_ATTENTION: 382 return EAGAIN; 383 case ABORTED_COMMAND: /* COMMAND ABORTED */ 384 return ECANCELED; 385 case NOT_READY: 386 case ILLEGAL_REQUEST: 387 case DATA_PROTECT: 388 /* Parse ASCQ */ 389 break; 390 default: 391 return EIO; 392 } 393 switch ((asc << 8) | ascq) { 394 case 0x1a00: /* PARAMETER LIST LENGTH ERROR */ 395 case 0x2000: /* INVALID OPERATION CODE */ 396 case 0x2400: /* INVALID FIELD IN CDB */ 397 case 0x2600: /* INVALID FIELD IN PARAMETER LIST */ 398 return EINVAL; 399 case 0x2100: /* LBA OUT OF RANGE */ 400 case 0x2707: /* SPACE ALLOC FAILED */ 401 return ENOSPC; 402 case 0x2500: /* LOGICAL UNIT NOT SUPPORTED */ 403 return ENOTSUP; 404 case 0x3a00: /* MEDIUM NOT PRESENT */ 405 case 0x3a01: /* MEDIUM NOT PRESENT TRAY CLOSED */ 406 case 0x3a02: /* MEDIUM NOT PRESENT TRAY OPEN */ 407 return ENOMEDIUM; 408 case 0x2700: /* WRITE PROTECTED */ 409 return EACCES; 410 case 0x0401: /* NOT READY, IN PROGRESS OF BECOMING READY */ 411 return EINPROGRESS; 412 case 0x0402: /* NOT READY, INITIALIZING COMMAND REQUIRED */ 413 return ENOTCONN; 414 default: 415 return EIO; 416 } 417} 418 419int scsi_sense_buf_to_errno(const uint8_t *in_buf, size_t in_len) 420{ 421 SCSISense sense; 422 if (in_len < 1) { 423 return EIO; 424 } 425 426 sense = scsi_parse_sense_buf(in_buf, in_len); 427 return scsi_sense_to_errno(sense.key, sense.asc, sense.ascq); 428} 429 430bool scsi_sense_buf_is_guest_recoverable(const uint8_t *in_buf, size_t in_len) 431{ 432 SCSISense sense; 433 if (in_len < 1) { 434 return false; 435 } 436 437 sense = scsi_parse_sense_buf(in_buf, in_len); 438 return scsi_sense_is_guest_recoverable(sense.key, sense.asc, sense.ascq); 439} 440 441const char *scsi_command_name(uint8_t cmd) 442{ 443 static const char *names[] = { 444 [ TEST_UNIT_READY ] = "TEST_UNIT_READY", 445 [ REWIND ] = "REWIND", 446 [ REQUEST_SENSE ] = "REQUEST_SENSE", 447 [ FORMAT_UNIT ] = "FORMAT_UNIT", 448 [ READ_BLOCK_LIMITS ] = "READ_BLOCK_LIMITS", 449 [ REASSIGN_BLOCKS ] = "REASSIGN_BLOCKS/INITIALIZE ELEMENT STATUS", 450 /* LOAD_UNLOAD and INITIALIZE_ELEMENT_STATUS use the same operation code */ 451 [ READ_6 ] = "READ_6", 452 [ WRITE_6 ] = "WRITE_6", 453 [ SET_CAPACITY ] = "SET_CAPACITY", 454 [ READ_REVERSE ] = "READ_REVERSE", 455 [ WRITE_FILEMARKS ] = "WRITE_FILEMARKS", 456 [ SPACE ] = "SPACE", 457 [ INQUIRY ] = "INQUIRY", 458 [ RECOVER_BUFFERED_DATA ] = "RECOVER_BUFFERED_DATA", 459 [ MAINTENANCE_IN ] = "MAINTENANCE_IN", 460 [ MAINTENANCE_OUT ] = "MAINTENANCE_OUT", 461 [ MODE_SELECT ] = "MODE_SELECT", 462 [ RESERVE ] = "RESERVE", 463 [ RELEASE ] = "RELEASE", 464 [ COPY ] = "COPY", 465 [ ERASE ] = "ERASE", 466 [ MODE_SENSE ] = "MODE_SENSE", 467 [ START_STOP ] = "START_STOP/LOAD_UNLOAD", 468 /* LOAD_UNLOAD and START_STOP use the same operation code */ 469 [ RECEIVE_DIAGNOSTIC ] = "RECEIVE_DIAGNOSTIC", 470 [ SEND_DIAGNOSTIC ] = "SEND_DIAGNOSTIC", 471 [ ALLOW_MEDIUM_REMOVAL ] = "ALLOW_MEDIUM_REMOVAL", 472 [ READ_CAPACITY_10 ] = "READ_CAPACITY_10", 473 [ READ_10 ] = "READ_10", 474 [ WRITE_10 ] = "WRITE_10", 475 [ SEEK_10 ] = "SEEK_10/POSITION_TO_ELEMENT", 476 /* SEEK_10 and POSITION_TO_ELEMENT use the same operation code */ 477 [ WRITE_VERIFY_10 ] = "WRITE_VERIFY_10", 478 [ VERIFY_10 ] = "VERIFY_10", 479 [ SEARCH_HIGH ] = "SEARCH_HIGH", 480 [ SEARCH_EQUAL ] = "SEARCH_EQUAL", 481 [ SEARCH_LOW ] = "SEARCH_LOW", 482 [ SET_LIMITS ] = "SET_LIMITS", 483 [ PRE_FETCH ] = "PRE_FETCH/READ_POSITION", 484 /* READ_POSITION and PRE_FETCH use the same operation code */ 485 [ SYNCHRONIZE_CACHE ] = "SYNCHRONIZE_CACHE", 486 [ LOCK_UNLOCK_CACHE ] = "LOCK_UNLOCK_CACHE", 487 [ READ_DEFECT_DATA ] = "READ_DEFECT_DATA/INITIALIZE_ELEMENT_STATUS_WITH_RANGE", 488 /* READ_DEFECT_DATA and INITIALIZE_ELEMENT_STATUS_WITH_RANGE use the same operation code */ 489 [ MEDIUM_SCAN ] = "MEDIUM_SCAN", 490 [ COMPARE ] = "COMPARE", 491 [ COPY_VERIFY ] = "COPY_VERIFY", 492 [ WRITE_BUFFER ] = "WRITE_BUFFER", 493 [ READ_BUFFER ] = "READ_BUFFER", 494 [ UPDATE_BLOCK ] = "UPDATE_BLOCK", 495 [ READ_LONG_10 ] = "READ_LONG_10", 496 [ WRITE_LONG_10 ] = "WRITE_LONG_10", 497 [ CHANGE_DEFINITION ] = "CHANGE_DEFINITION", 498 [ WRITE_SAME_10 ] = "WRITE_SAME_10", 499 [ UNMAP ] = "UNMAP", 500 [ READ_TOC ] = "READ_TOC", 501 [ REPORT_DENSITY_SUPPORT ] = "REPORT_DENSITY_SUPPORT", 502 [ SANITIZE ] = "SANITIZE", 503 [ GET_CONFIGURATION ] = "GET_CONFIGURATION", 504 [ LOG_SELECT ] = "LOG_SELECT", 505 [ LOG_SENSE ] = "LOG_SENSE", 506 [ MODE_SELECT_10 ] = "MODE_SELECT_10", 507 [ RESERVE_10 ] = "RESERVE_10", 508 [ RELEASE_10 ] = "RELEASE_10", 509 [ MODE_SENSE_10 ] = "MODE_SENSE_10", 510 [ PERSISTENT_RESERVE_IN ] = "PERSISTENT_RESERVE_IN", 511 [ PERSISTENT_RESERVE_OUT ] = "PERSISTENT_RESERVE_OUT", 512 [ WRITE_FILEMARKS_16 ] = "WRITE_FILEMARKS_16", 513 [ EXTENDED_COPY ] = "EXTENDED_COPY", 514 [ ATA_PASSTHROUGH_16 ] = "ATA_PASSTHROUGH_16", 515 [ ACCESS_CONTROL_IN ] = "ACCESS_CONTROL_IN", 516 [ ACCESS_CONTROL_OUT ] = "ACCESS_CONTROL_OUT", 517 [ READ_16 ] = "READ_16", 518 [ COMPARE_AND_WRITE ] = "COMPARE_AND_WRITE", 519 [ WRITE_16 ] = "WRITE_16", 520 [ WRITE_VERIFY_16 ] = "WRITE_VERIFY_16", 521 [ VERIFY_16 ] = "VERIFY_16", 522 [ PRE_FETCH_16 ] = "PRE_FETCH_16", 523 [ SYNCHRONIZE_CACHE_16 ] = "SPACE_16/SYNCHRONIZE_CACHE_16", 524 /* SPACE_16 and SYNCHRONIZE_CACHE_16 use the same operation code */ 525 [ LOCATE_16 ] = "LOCATE_16", 526 [ WRITE_SAME_16 ] = "ERASE_16/WRITE_SAME_16", 527 /* ERASE_16 and WRITE_SAME_16 use the same operation code */ 528 [ SERVICE_ACTION_IN_16 ] = "SERVICE_ACTION_IN_16", 529 [ WRITE_LONG_16 ] = "WRITE_LONG_16", 530 [ REPORT_LUNS ] = "REPORT_LUNS", 531 [ ATA_PASSTHROUGH_12 ] = "BLANK/ATA_PASSTHROUGH_12", 532 [ MOVE_MEDIUM ] = "MOVE_MEDIUM", 533 [ EXCHANGE_MEDIUM ] = "EXCHANGE MEDIUM", 534 [ READ_12 ] = "READ_12", 535 [ WRITE_12 ] = "WRITE_12", 536 [ ERASE_12 ] = "ERASE_12/GET_PERFORMANCE", 537 /* ERASE_12 and GET_PERFORMANCE use the same operation code */ 538 [ SERVICE_ACTION_IN_12 ] = "SERVICE_ACTION_IN_12", 539 [ WRITE_VERIFY_12 ] = "WRITE_VERIFY_12", 540 [ VERIFY_12 ] = "VERIFY_12", 541 [ SEARCH_HIGH_12 ] = "SEARCH_HIGH_12", 542 [ SEARCH_EQUAL_12 ] = "SEARCH_EQUAL_12", 543 [ SEARCH_LOW_12 ] = "SEARCH_LOW_12", 544 [ READ_ELEMENT_STATUS ] = "READ_ELEMENT_STATUS", 545 [ SEND_VOLUME_TAG ] = "SEND_VOLUME_TAG/SET_STREAMING", 546 /* SEND_VOLUME_TAG and SET_STREAMING use the same operation code */ 547 [ READ_CD ] = "READ_CD", 548 [ READ_DEFECT_DATA_12 ] = "READ_DEFECT_DATA_12", 549 [ READ_DVD_STRUCTURE ] = "READ_DVD_STRUCTURE", 550 [ RESERVE_TRACK ] = "RESERVE_TRACK", 551 [ SEND_CUE_SHEET ] = "SEND_CUE_SHEET", 552 [ SEND_DVD_STRUCTURE ] = "SEND_DVD_STRUCTURE", 553 [ SET_CD_SPEED ] = "SET_CD_SPEED", 554 [ SET_READ_AHEAD ] = "SET_READ_AHEAD", 555 [ ALLOW_OVERWRITE ] = "ALLOW_OVERWRITE", 556 [ MECHANISM_STATUS ] = "MECHANISM_STATUS", 557 [ GET_EVENT_STATUS_NOTIFICATION ] = "GET_EVENT_STATUS_NOTIFICATION", 558 [ READ_DISC_INFORMATION ] = "READ_DISC_INFORMATION", 559 }; 560 561 if (cmd >= ARRAY_SIZE(names) || names[cmd] == NULL) { 562 return "*UNKNOWN*"; 563 } 564 return names[cmd]; 565} 566 567#ifdef CONFIG_LINUX 568int sg_io_sense_from_errno(int errno_value, struct sg_io_hdr *io_hdr, 569 SCSISense *sense) 570{ 571 if (errno_value != 0) { 572 switch (errno_value) { 573 case EDOM: 574 return TASK_SET_FULL; 575 case ENOMEM: 576 *sense = SENSE_CODE(TARGET_FAILURE); 577 return CHECK_CONDITION; 578 default: 579 *sense = SENSE_CODE(IO_ERROR); 580 return CHECK_CONDITION; 581 } 582 } else { 583 if (io_hdr->host_status == SG_ERR_DID_NO_CONNECT || 584 io_hdr->host_status == SG_ERR_DID_BUS_BUSY || 585 io_hdr->host_status == SG_ERR_DID_TIME_OUT || 586 (io_hdr->driver_status & SG_ERR_DRIVER_TIMEOUT)) { 587 return BUSY; 588 } else if (io_hdr->host_status) { 589 *sense = SENSE_CODE(I_T_NEXUS_LOSS); 590 return CHECK_CONDITION; 591 } else if (io_hdr->status) { 592 return io_hdr->status; 593 } else if (io_hdr->driver_status & SG_ERR_DRIVER_SENSE) { 594 return CHECK_CONDITION; 595 } else { 596 return GOOD; 597 } 598 } 599} 600#endif