2
0

utils.c 19 KB

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