4 * Copyright 2017 Red Hat, Inc.
7 * Fam Zheng <famz@redhat.com>
8 * Paolo Bonzini <pbonzini@redhat.com>
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)
16 #include "qemu/osdep.h"
17 #include "scsi/constants.h"
18 #include "scsi/utils.h"
19 #include "qemu/bswap.h"
21 uint32_t scsi_data_cdb_xfer(uint8_t *buf
)
23 if ((buf
[0] >> 5) == 0 && buf
[4] == 0) {
26 return scsi_cdb_xfer(buf
);
30 uint32_t scsi_cdb_xfer(uint8_t *buf
)
32 switch (buf
[0] >> 5) {
37 return lduw_be_p(&buf
[7]);
39 return ldl_be_p(&buf
[10]) & 0xffffffffULL
;
41 return ldl_be_p(&buf
[6]) & 0xffffffffULL
;
47 uint64_t scsi_cmd_lba(SCSICommand
*cmd
)
49 uint8_t *buf
= cmd
->buf
;
52 switch (buf
[0] >> 5) {
54 lba
= ldl_be_p(&buf
[0]) & 0x1fffff;
59 lba
= ldl_be_p(&buf
[2]) & 0xffffffffULL
;
62 lba
= ldq_be_p(&buf
[2]);
71 int scsi_cdb_length(uint8_t *buf
)
75 switch (buf
[0] >> 5) {
95 SCSISense
scsi_parse_sense_buf(const uint8_t *in_buf
, int in_len
)
101 fixed_in
= (in_buf
[0] & 2) == 0;
104 return SENSE_CODE(IO_ERROR
);
106 sense
.key
= in_buf
[2];
107 sense
.asc
= in_buf
[12];
108 sense
.ascq
= in_buf
[13];
111 return SENSE_CODE(IO_ERROR
);
113 sense
.key
= in_buf
[1];
114 sense
.asc
= in_buf
[2];
115 sense
.ascq
= in_buf
[3];
121 int scsi_build_sense_buf(uint8_t *out_buf
, size_t size
, SCSISense sense
,
125 uint8_t buf
[SCSI_SENSE_LEN
] = { 0 };
132 buf
[13] = sense
.ascq
;
141 len
= MIN(len
, size
);
142 memcpy(out_buf
, buf
, len
);
146 int scsi_build_sense(uint8_t *buf
, SCSISense sense
)
148 return scsi_build_sense_buf(buf
, SCSI_SENSE_LEN
, sense
, true);
152 * Predefined sense codes
155 /* No sense data available */
156 const struct SCSISense sense_code_NO_SENSE
= {
157 .key
= NO_SENSE
, .asc
= 0x00 , .ascq
= 0x00
160 /* LUN not ready, Manual intervention required */
161 const struct SCSISense sense_code_LUN_NOT_READY
= {
162 .key
= NOT_READY
, .asc
= 0x04, .ascq
= 0x03
165 /* LUN not ready, Medium not present */
166 const struct SCSISense sense_code_NO_MEDIUM
= {
167 .key
= NOT_READY
, .asc
= 0x3a, .ascq
= 0x00
170 /* LUN not ready, medium removal prevented */
171 const struct SCSISense sense_code_NOT_READY_REMOVAL_PREVENTED
= {
172 .key
= NOT_READY
, .asc
= 0x53, .ascq
= 0x02
175 /* Hardware error, internal target failure */
176 const struct SCSISense sense_code_TARGET_FAILURE
= {
177 .key
= HARDWARE_ERROR
, .asc
= 0x44, .ascq
= 0x00
180 /* Illegal request, invalid command operation code */
181 const struct SCSISense sense_code_INVALID_OPCODE
= {
182 .key
= ILLEGAL_REQUEST
, .asc
= 0x20, .ascq
= 0x00
185 /* Illegal request, LBA out of range */
186 const struct SCSISense sense_code_LBA_OUT_OF_RANGE
= {
187 .key
= ILLEGAL_REQUEST
, .asc
= 0x21, .ascq
= 0x00
190 /* Illegal request, Invalid field in CDB */
191 const struct SCSISense sense_code_INVALID_FIELD
= {
192 .key
= ILLEGAL_REQUEST
, .asc
= 0x24, .ascq
= 0x00
195 /* Illegal request, Invalid field in parameter list */
196 const struct SCSISense sense_code_INVALID_PARAM
= {
197 .key
= ILLEGAL_REQUEST
, .asc
= 0x26, .ascq
= 0x00
200 /* Illegal request, Invalid value in parameter list */
201 const struct SCSISense sense_code_INVALID_PARAM_VALUE
= {
202 .key
= ILLEGAL_REQUEST
, .asc
= 0x26, .ascq
= 0x01
205 /* Illegal request, Parameter list length error */
206 const struct SCSISense sense_code_INVALID_PARAM_LEN
= {
207 .key
= ILLEGAL_REQUEST
, .asc
= 0x1a, .ascq
= 0x00
210 /* Illegal request, LUN not supported */
211 const struct SCSISense sense_code_LUN_NOT_SUPPORTED
= {
212 .key
= ILLEGAL_REQUEST
, .asc
= 0x25, .ascq
= 0x00
215 /* Illegal request, Saving parameters not supported */
216 const struct SCSISense sense_code_SAVING_PARAMS_NOT_SUPPORTED
= {
217 .key
= ILLEGAL_REQUEST
, .asc
= 0x39, .ascq
= 0x00
220 /* Illegal request, Incompatible medium installed */
221 const struct SCSISense sense_code_INCOMPATIBLE_FORMAT
= {
222 .key
= ILLEGAL_REQUEST
, .asc
= 0x30, .ascq
= 0x00
225 /* Illegal request, medium removal prevented */
226 const struct SCSISense sense_code_ILLEGAL_REQ_REMOVAL_PREVENTED
= {
227 .key
= ILLEGAL_REQUEST
, .asc
= 0x53, .ascq
= 0x02
230 /* Illegal request, Invalid Transfer Tag */
231 const struct SCSISense sense_code_INVALID_TAG
= {
232 .key
= ILLEGAL_REQUEST
, .asc
= 0x4b, .ascq
= 0x01
235 /* Command aborted, I/O process terminated */
236 const struct SCSISense sense_code_IO_ERROR
= {
237 .key
= ABORTED_COMMAND
, .asc
= 0x00, .ascq
= 0x06
240 /* Command aborted, I_T Nexus loss occurred */
241 const struct SCSISense sense_code_I_T_NEXUS_LOSS
= {
242 .key
= ABORTED_COMMAND
, .asc
= 0x29, .ascq
= 0x07
245 /* Command aborted, Logical Unit failure */
246 const struct SCSISense sense_code_LUN_FAILURE
= {
247 .key
= ABORTED_COMMAND
, .asc
= 0x3e, .ascq
= 0x01
250 /* Command aborted, Overlapped Commands Attempted */
251 const struct SCSISense sense_code_OVERLAPPED_COMMANDS
= {
252 .key
= ABORTED_COMMAND
, .asc
= 0x4e, .ascq
= 0x00
255 /* Command aborted, LUN Communication Failure */
256 const struct SCSISense sense_code_LUN_COMM_FAILURE
= {
257 .key
= ABORTED_COMMAND
, .asc
= 0x08, .ascq
= 0x00
260 /* Command aborted, LUN does not respond to selection */
261 const struct SCSISense sense_code_LUN_NOT_RESPONDING
= {
262 .key
= ABORTED_COMMAND
, .asc
= 0x05, .ascq
= 0x00
265 /* Command aborted, Command Timeout during processing */
266 const struct SCSISense sense_code_COMMAND_TIMEOUT
= {
267 .key
= ABORTED_COMMAND
, .asc
= 0x2e, .ascq
= 0x02
270 /* Command aborted, Commands cleared by device server */
271 const struct SCSISense sense_code_COMMAND_ABORTED
= {
272 .key
= ABORTED_COMMAND
, .asc
= 0x2f, .ascq
= 0x02
275 /* Medium Error, Unrecovered read error */
276 const struct SCSISense sense_code_READ_ERROR
= {
277 .key
= MEDIUM_ERROR
, .asc
= 0x11, .ascq
= 0x00
280 /* Not ready, Cause not reportable */
281 const struct SCSISense sense_code_NOT_READY
= {
282 .key
= NOT_READY
, .asc
= 0x04, .ascq
= 0x00
285 /* Unit attention, Capacity data has changed */
286 const struct SCSISense sense_code_CAPACITY_CHANGED
= {
287 .key
= UNIT_ATTENTION
, .asc
= 0x2a, .ascq
= 0x09
290 /* Unit attention, Power on, reset or bus device reset occurred */
291 const struct SCSISense sense_code_RESET
= {
292 .key
= UNIT_ATTENTION
, .asc
= 0x29, .ascq
= 0x00
295 /* Unit attention, SCSI bus reset */
296 const struct SCSISense sense_code_SCSI_BUS_RESET
= {
297 .key
= UNIT_ATTENTION
, .asc
= 0x29, .ascq
= 0x02
300 /* Unit attention, No medium */
301 const struct SCSISense sense_code_UNIT_ATTENTION_NO_MEDIUM
= {
302 .key
= UNIT_ATTENTION
, .asc
= 0x3a, .ascq
= 0x00
305 /* Unit attention, Medium may have changed */
306 const struct SCSISense sense_code_MEDIUM_CHANGED
= {
307 .key
= UNIT_ATTENTION
, .asc
= 0x28, .ascq
= 0x00
310 /* Unit attention, Reported LUNs data has changed */
311 const struct SCSISense sense_code_REPORTED_LUNS_CHANGED
= {
312 .key
= UNIT_ATTENTION
, .asc
= 0x3f, .ascq
= 0x0e
315 /* Unit attention, Device internal reset */
316 const struct SCSISense sense_code_DEVICE_INTERNAL_RESET
= {
317 .key
= UNIT_ATTENTION
, .asc
= 0x29, .ascq
= 0x04
320 /* Data Protection, Write Protected */
321 const struct SCSISense sense_code_WRITE_PROTECTED
= {
322 .key
= DATA_PROTECT
, .asc
= 0x27, .ascq
= 0x00
325 /* Data Protection, Space Allocation Failed Write Protect */
326 const struct SCSISense sense_code_SPACE_ALLOC_FAILED
= {
327 .key
= DATA_PROTECT
, .asc
= 0x27, .ascq
= 0x07
333 * Convert between fixed and descriptor sense buffers
335 int scsi_convert_sense(uint8_t *in_buf
, int in_len
,
336 uint8_t *buf
, int len
, bool fixed
)
342 return scsi_build_sense_buf(buf
, len
, SENSE_CODE(NO_SENSE
), fixed
);
345 fixed_in
= (in_buf
[0] & 2) == 0;
346 if (fixed
== fixed_in
) {
347 memcpy(buf
, in_buf
, MIN(len
, in_len
));
348 return MIN(len
, in_len
);
350 sense
= scsi_parse_sense_buf(in_buf
, in_len
);
351 return scsi_build_sense_buf(buf
, len
, sense
, fixed
);
355 static bool scsi_sense_is_guest_recoverable(int key
, int asc
, int ascq
)
359 case RECOVERED_ERROR
:
361 case ABORTED_COMMAND
:
364 case ILLEGAL_REQUEST
:
372 switch ((asc
<< 8) | ascq
) {
373 case 0x1a00: /* PARAMETER LIST LENGTH ERROR */
374 case 0x2000: /* INVALID OPERATION CODE */
375 case 0x2400: /* INVALID FIELD IN CDB */
376 case 0x2500: /* LOGICAL UNIT NOT SUPPORTED */
377 case 0x2600: /* INVALID FIELD IN PARAMETER LIST */
379 case 0x2104: /* UNALIGNED WRITE COMMAND */
380 case 0x2105: /* WRITE BOUNDARY VIOLATION */
381 case 0x2106: /* ATTEMPT TO READ INVALID DATA */
382 case 0x550e: /* INSUFFICIENT ZONE RESOURCES */
384 case 0x0401: /* NOT READY, IN PROGRESS OF BECOMING READY */
385 case 0x0402: /* NOT READY, INITIALIZING COMMAND REQUIRED */
392 int scsi_sense_to_errno(int key
, int asc
, int ascq
)
396 case RECOVERED_ERROR
:
399 case ABORTED_COMMAND
: /* COMMAND ABORTED */
402 case ILLEGAL_REQUEST
:
409 switch ((asc
<< 8) | ascq
) {
410 case 0x1a00: /* PARAMETER LIST LENGTH ERROR */
411 case 0x2000: /* INVALID OPERATION CODE */
412 case 0x2400: /* INVALID FIELD IN CDB */
413 case 0x2600: /* INVALID FIELD IN PARAMETER LIST */
415 case 0x2100: /* LBA OUT OF RANGE */
416 case 0x2707: /* SPACE ALLOC FAILED */
418 case 0x2500: /* LOGICAL UNIT NOT SUPPORTED */
420 case 0x3a00: /* MEDIUM NOT PRESENT */
421 case 0x3a01: /* MEDIUM NOT PRESENT TRAY CLOSED */
422 case 0x3a02: /* MEDIUM NOT PRESENT TRAY OPEN */
424 case 0x2700: /* WRITE PROTECTED */
426 case 0x0401: /* NOT READY, IN PROGRESS OF BECOMING READY */
428 case 0x0402: /* NOT READY, INITIALIZING COMMAND REQUIRED */
435 int scsi_sense_buf_to_errno(const uint8_t *in_buf
, size_t in_len
)
442 sense
= scsi_parse_sense_buf(in_buf
, in_len
);
443 return scsi_sense_to_errno(sense
.key
, sense
.asc
, sense
.ascq
);
446 bool scsi_sense_buf_is_guest_recoverable(const uint8_t *in_buf
, size_t in_len
)
453 sense
= scsi_parse_sense_buf(in_buf
, in_len
);
454 return scsi_sense_is_guest_recoverable(sense
.key
, sense
.asc
, sense
.ascq
);
457 const char *scsi_command_name(uint8_t cmd
)
459 static const char *names
[] = {
460 [ TEST_UNIT_READY
] = "TEST_UNIT_READY",
461 [ REWIND
] = "REWIND",
462 [ REQUEST_SENSE
] = "REQUEST_SENSE",
463 [ FORMAT_UNIT
] = "FORMAT_UNIT",
464 [ READ_BLOCK_LIMITS
] = "READ_BLOCK_LIMITS",
465 [ REASSIGN_BLOCKS
] = "REASSIGN_BLOCKS/INITIALIZE ELEMENT STATUS",
466 /* LOAD_UNLOAD and INITIALIZE_ELEMENT_STATUS use the same operation code */
467 [ READ_6
] = "READ_6",
468 [ WRITE_6
] = "WRITE_6",
469 [ SET_CAPACITY
] = "SET_CAPACITY",
470 [ READ_REVERSE
] = "READ_REVERSE",
471 [ WRITE_FILEMARKS
] = "WRITE_FILEMARKS",
473 [ INQUIRY
] = "INQUIRY",
474 [ RECOVER_BUFFERED_DATA
] = "RECOVER_BUFFERED_DATA",
475 [ MAINTENANCE_IN
] = "MAINTENANCE_IN",
476 [ MAINTENANCE_OUT
] = "MAINTENANCE_OUT",
477 [ MODE_SELECT
] = "MODE_SELECT",
478 [ RESERVE
] = "RESERVE",
479 [ RELEASE
] = "RELEASE",
482 [ MODE_SENSE
] = "MODE_SENSE",
483 [ START_STOP
] = "START_STOP/LOAD_UNLOAD",
484 /* LOAD_UNLOAD and START_STOP use the same operation code */
485 [ RECEIVE_DIAGNOSTIC
] = "RECEIVE_DIAGNOSTIC",
486 [ SEND_DIAGNOSTIC
] = "SEND_DIAGNOSTIC",
487 [ ALLOW_MEDIUM_REMOVAL
] = "ALLOW_MEDIUM_REMOVAL",
488 [ READ_CAPACITY_10
] = "READ_CAPACITY_10",
489 [ READ_10
] = "READ_10",
490 [ WRITE_10
] = "WRITE_10",
491 [ SEEK_10
] = "SEEK_10/POSITION_TO_ELEMENT",
492 /* SEEK_10 and POSITION_TO_ELEMENT use the same operation code */
493 [ WRITE_VERIFY_10
] = "WRITE_VERIFY_10",
494 [ VERIFY_10
] = "VERIFY_10",
495 [ SEARCH_HIGH
] = "SEARCH_HIGH",
496 [ SEARCH_EQUAL
] = "SEARCH_EQUAL",
497 [ SEARCH_LOW
] = "SEARCH_LOW",
498 [ SET_LIMITS
] = "SET_LIMITS",
499 [ PRE_FETCH
] = "PRE_FETCH/READ_POSITION",
500 /* READ_POSITION and PRE_FETCH use the same operation code */
501 [ SYNCHRONIZE_CACHE
] = "SYNCHRONIZE_CACHE",
502 [ LOCK_UNLOCK_CACHE
] = "LOCK_UNLOCK_CACHE",
503 [ READ_DEFECT_DATA
] = "READ_DEFECT_DATA/INITIALIZE_ELEMENT_STATUS_WITH_RANGE",
504 /* READ_DEFECT_DATA and INITIALIZE_ELEMENT_STATUS_WITH_RANGE use the same operation code */
505 [ MEDIUM_SCAN
] = "MEDIUM_SCAN",
506 [ COMPARE
] = "COMPARE",
507 [ COPY_VERIFY
] = "COPY_VERIFY",
508 [ WRITE_BUFFER
] = "WRITE_BUFFER",
509 [ READ_BUFFER
] = "READ_BUFFER",
510 [ UPDATE_BLOCK
] = "UPDATE_BLOCK",
511 [ READ_LONG_10
] = "READ_LONG_10",
512 [ WRITE_LONG_10
] = "WRITE_LONG_10",
513 [ CHANGE_DEFINITION
] = "CHANGE_DEFINITION",
514 [ WRITE_SAME_10
] = "WRITE_SAME_10",
516 [ READ_TOC
] = "READ_TOC",
517 [ REPORT_DENSITY_SUPPORT
] = "REPORT_DENSITY_SUPPORT",
518 [ SANITIZE
] = "SANITIZE",
519 [ GET_CONFIGURATION
] = "GET_CONFIGURATION",
520 [ LOG_SELECT
] = "LOG_SELECT",
521 [ LOG_SENSE
] = "LOG_SENSE",
522 [ MODE_SELECT_10
] = "MODE_SELECT_10",
523 [ RESERVE_10
] = "RESERVE_10",
524 [ RELEASE_10
] = "RELEASE_10",
525 [ MODE_SENSE_10
] = "MODE_SENSE_10",
526 [ PERSISTENT_RESERVE_IN
] = "PERSISTENT_RESERVE_IN",
527 [ PERSISTENT_RESERVE_OUT
] = "PERSISTENT_RESERVE_OUT",
528 [ WRITE_FILEMARKS_16
] = "WRITE_FILEMARKS_16",
529 [ EXTENDED_COPY
] = "EXTENDED_COPY",
530 [ ATA_PASSTHROUGH_16
] = "ATA_PASSTHROUGH_16",
531 [ ACCESS_CONTROL_IN
] = "ACCESS_CONTROL_IN",
532 [ ACCESS_CONTROL_OUT
] = "ACCESS_CONTROL_OUT",
533 [ READ_16
] = "READ_16",
534 [ COMPARE_AND_WRITE
] = "COMPARE_AND_WRITE",
535 [ WRITE_16
] = "WRITE_16",
536 [ WRITE_VERIFY_16
] = "WRITE_VERIFY_16",
537 [ VERIFY_16
] = "VERIFY_16",
538 [ PRE_FETCH_16
] = "PRE_FETCH_16",
539 [ SYNCHRONIZE_CACHE_16
] = "SPACE_16/SYNCHRONIZE_CACHE_16",
540 /* SPACE_16 and SYNCHRONIZE_CACHE_16 use the same operation code */
541 [ LOCATE_16
] = "LOCATE_16",
542 [ WRITE_SAME_16
] = "ERASE_16/WRITE_SAME_16",
543 /* ERASE_16 and WRITE_SAME_16 use the same operation code */
544 [ SERVICE_ACTION_IN_16
] = "SERVICE_ACTION_IN_16",
545 [ WRITE_LONG_16
] = "WRITE_LONG_16",
546 [ REPORT_LUNS
] = "REPORT_LUNS",
547 [ ATA_PASSTHROUGH_12
] = "BLANK/ATA_PASSTHROUGH_12",
548 [ MOVE_MEDIUM
] = "MOVE_MEDIUM",
549 [ EXCHANGE_MEDIUM
] = "EXCHANGE MEDIUM",
550 [ READ_12
] = "READ_12",
551 [ WRITE_12
] = "WRITE_12",
552 [ ERASE_12
] = "ERASE_12/GET_PERFORMANCE",
553 /* ERASE_12 and GET_PERFORMANCE use the same operation code */
554 [ SERVICE_ACTION_IN_12
] = "SERVICE_ACTION_IN_12",
555 [ WRITE_VERIFY_12
] = "WRITE_VERIFY_12",
556 [ VERIFY_12
] = "VERIFY_12",
557 [ SEARCH_HIGH_12
] = "SEARCH_HIGH_12",
558 [ SEARCH_EQUAL_12
] = "SEARCH_EQUAL_12",
559 [ SEARCH_LOW_12
] = "SEARCH_LOW_12",
560 [ READ_ELEMENT_STATUS
] = "READ_ELEMENT_STATUS",
561 [ SEND_VOLUME_TAG
] = "SEND_VOLUME_TAG/SET_STREAMING",
562 /* SEND_VOLUME_TAG and SET_STREAMING use the same operation code */
563 [ READ_CD
] = "READ_CD",
564 [ READ_DEFECT_DATA_12
] = "READ_DEFECT_DATA_12",
565 [ READ_DVD_STRUCTURE
] = "READ_DVD_STRUCTURE",
566 [ RESERVE_TRACK
] = "RESERVE_TRACK",
567 [ SEND_CUE_SHEET
] = "SEND_CUE_SHEET",
568 [ SEND_DVD_STRUCTURE
] = "SEND_DVD_STRUCTURE",
569 [ SET_CD_SPEED
] = "SET_CD_SPEED",
570 [ SET_READ_AHEAD
] = "SET_READ_AHEAD",
571 [ ALLOW_OVERWRITE
] = "ALLOW_OVERWRITE",
572 [ MECHANISM_STATUS
] = "MECHANISM_STATUS",
573 [ GET_EVENT_STATUS_NOTIFICATION
] = "GET_EVENT_STATUS_NOTIFICATION",
574 [ READ_DISC_INFORMATION
] = "READ_DISC_INFORMATION",
577 if (cmd
>= ARRAY_SIZE(names
) || names
[cmd
] == NULL
) {
583 int scsi_sense_from_errno(int errno_value
, SCSISense
*sense
)
585 switch (errno_value
) {
589 return TASK_SET_FULL
;
591 /* These errno mapping are specific to Linux. For more information:
592 * - scsi_check_sense and scsi_decide_disposition in drivers/scsi/scsi_error.c
593 * - scsi_result_to_blk_status in drivers/scsi/scsi_lib.c
594 * - blk_errors[] in block/blk-core.c
597 return RESERVATION_CONFLICT
;
599 *sense
= SENSE_CODE(READ_ERROR
);
600 return CHECK_CONDITION
;
602 *sense
= SENSE_CODE(TARGET_FAILURE
);
603 return CHECK_CONDITION
;
606 *sense
= SENSE_CODE(NO_MEDIUM
);
607 return CHECK_CONDITION
;
609 *sense
= SENSE_CODE(TARGET_FAILURE
);
610 return CHECK_CONDITION
;
612 *sense
= SENSE_CODE(INVALID_FIELD
);
613 return CHECK_CONDITION
;
615 *sense
= SENSE_CODE(SPACE_ALLOC_FAILED
);
616 return CHECK_CONDITION
;
618 *sense
= SENSE_CODE(IO_ERROR
);
619 return CHECK_CONDITION
;
623 int scsi_sense_from_host_status(uint8_t host_status
,
626 switch (host_status
) {
627 case SCSI_HOST_NO_LUN
:
628 *sense
= SENSE_CODE(LUN_NOT_RESPONDING
);
629 return CHECK_CONDITION
;
632 case SCSI_HOST_TIME_OUT
:
633 *sense
= SENSE_CODE(COMMAND_TIMEOUT
);
634 return CHECK_CONDITION
;
635 case SCSI_HOST_BAD_RESPONSE
:
636 *sense
= SENSE_CODE(LUN_COMM_FAILURE
);
637 return CHECK_CONDITION
;
638 case SCSI_HOST_ABORTED
:
639 *sense
= SENSE_CODE(COMMAND_ABORTED
);
640 return CHECK_CONDITION
;
641 case SCSI_HOST_RESET
:
642 *sense
= SENSE_CODE(RESET
);
643 return CHECK_CONDITION
;
644 case SCSI_HOST_TRANSPORT_DISRUPTED
:
645 *sense
= SENSE_CODE(I_T_NEXUS_LOSS
);
646 return CHECK_CONDITION
;
647 case SCSI_HOST_TARGET_FAILURE
:
648 *sense
= SENSE_CODE(TARGET_FAILURE
);
649 return CHECK_CONDITION
;
650 case SCSI_HOST_RESERVATION_ERROR
:
651 return RESERVATION_CONFLICT
;
652 case SCSI_HOST_ALLOCATION_FAILURE
:
653 *sense
= SENSE_CODE(SPACE_ALLOC_FAILED
);
654 return CHECK_CONDITION
;
655 case SCSI_HOST_MEDIUM_ERROR
:
656 *sense
= SENSE_CODE(READ_ERROR
);
657 return CHECK_CONDITION
;