Commit 22864256db67a1f85b7f09436612bd9633474dac
1 parent
71fb2348
Correct SCSI error reporting (Laurent Vivier)
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@5455 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
1 changed file
with
27 additions
and
15 deletions
hw/scsi-disk.c
... | ... | @@ -34,6 +34,9 @@ do { fprintf(stderr, "scsi-disk: " fmt , ##args); } while (0) |
34 | 34 | #define SENSE_HARDWARE_ERROR 4 |
35 | 35 | #define SENSE_ILLEGAL_REQUEST 5 |
36 | 36 | |
37 | +#define STATUS_GOOD 0 | |
38 | +#define STATUS_CHECK_CONDITION 2 | |
39 | + | |
37 | 40 | #define SCSI_DMA_BUF_SIZE 131072 |
38 | 41 | |
39 | 42 | typedef struct SCSIRequest { |
... | ... | @@ -124,15 +127,15 @@ static SCSIRequest *scsi_find_request(SCSIDeviceState *s, uint32_t tag) |
124 | 127 | } |
125 | 128 | |
126 | 129 | /* Helper function for command completion. */ |
127 | -static void scsi_command_complete(SCSIRequest *r, int sense) | |
130 | +static void scsi_command_complete(SCSIRequest *r, int status, int sense) | |
128 | 131 | { |
129 | 132 | SCSIDeviceState *s = r->dev; |
130 | 133 | uint32_t tag; |
131 | - DPRINTF("Command complete tag=0x%x sense=%d\n", r->tag, sense); | |
134 | + DPRINTF("Command complete tag=0x%x status=%d sense=%d\n", r->tag, status, sense); | |
132 | 135 | s->sense = sense; |
133 | 136 | tag = r->tag; |
134 | 137 | scsi_remove_request(r); |
135 | - s->completion(s->opaque, SCSI_REASON_DONE, tag, sense); | |
138 | + s->completion(s->opaque, SCSI_REASON_DONE, tag, status); | |
136 | 139 | } |
137 | 140 | |
138 | 141 | /* Cancel a pending data transfer. */ |
... | ... | @@ -157,7 +160,8 @@ static void scsi_read_complete(void * opaque, int ret) |
157 | 160 | |
158 | 161 | if (ret) { |
159 | 162 | DPRINTF("IO error\n"); |
160 | - scsi_command_complete(r, SENSE_HARDWARE_ERROR); | |
163 | + s->completion(s->opaque, SCSI_REASON_DATA, r->tag, 0); | |
164 | + scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_NO_SENSE); | |
161 | 165 | return; |
162 | 166 | } |
163 | 167 | DPRINTF("Data ready tag=0x%x len=%d\n", r->tag, r->buf_len); |
... | ... | @@ -176,7 +180,7 @@ static void scsi_read_data(SCSIDevice *d, uint32_t tag) |
176 | 180 | if (!r) { |
177 | 181 | BADF("Bad read tag 0x%x\n", tag); |
178 | 182 | /* ??? This is the wrong error. */ |
179 | - scsi_command_complete(r, SENSE_HARDWARE_ERROR); | |
183 | + scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_HARDWARE_ERROR); | |
180 | 184 | return; |
181 | 185 | } |
182 | 186 | if (r->sector_count == (uint32_t)-1) { |
... | ... | @@ -187,7 +191,7 @@ static void scsi_read_data(SCSIDevice *d, uint32_t tag) |
187 | 191 | } |
188 | 192 | DPRINTF("Read sector_count=%d\n", r->sector_count); |
189 | 193 | if (r->sector_count == 0) { |
190 | - scsi_command_complete(r, SENSE_NO_SENSE); | |
194 | + scsi_command_complete(r, STATUS_GOOD, SENSE_NO_SENSE); | |
191 | 195 | return; |
192 | 196 | } |
193 | 197 | |
... | ... | @@ -199,7 +203,7 @@ static void scsi_read_data(SCSIDevice *d, uint32_t tag) |
199 | 203 | r->aiocb = bdrv_aio_read(s->bdrv, r->sector, r->dma_buf, n, |
200 | 204 | scsi_read_complete, r); |
201 | 205 | if (r->aiocb == NULL) |
202 | - scsi_command_complete(r, SENSE_HARDWARE_ERROR); | |
206 | + scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_HARDWARE_ERROR); | |
203 | 207 | r->sector += n; |
204 | 208 | r->sector_count -= n; |
205 | 209 | } |
... | ... | @@ -217,7 +221,7 @@ static void scsi_write_complete(void * opaque, int ret) |
217 | 221 | |
218 | 222 | r->aiocb = NULL; |
219 | 223 | if (r->sector_count == 0) { |
220 | - scsi_command_complete(r, SENSE_NO_SENSE); | |
224 | + scsi_command_complete(r, STATUS_GOOD, SENSE_NO_SENSE); | |
221 | 225 | } else { |
222 | 226 | len = r->sector_count * 512; |
223 | 227 | if (len > SCSI_DMA_BUF_SIZE) { |
... | ... | @@ -241,7 +245,7 @@ static int scsi_write_data(SCSIDevice *d, uint32_t tag) |
241 | 245 | r = scsi_find_request(s, tag); |
242 | 246 | if (!r) { |
243 | 247 | BADF("Bad write tag 0x%x\n", tag); |
244 | - scsi_command_complete(r, SENSE_HARDWARE_ERROR); | |
248 | + scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_HARDWARE_ERROR); | |
245 | 249 | return 1; |
246 | 250 | } |
247 | 251 | if (r->aiocb) |
... | ... | @@ -251,7 +255,8 @@ static int scsi_write_data(SCSIDevice *d, uint32_t tag) |
251 | 255 | r->aiocb = bdrv_aio_write(s->bdrv, r->sector, r->dma_buf, n, |
252 | 256 | scsi_write_complete, r); |
253 | 257 | if (r->aiocb == NULL) |
254 | - scsi_command_complete(r, SENSE_HARDWARE_ERROR); | |
258 | + scsi_command_complete(r, STATUS_CHECK_CONDITION, | |
259 | + SENSE_HARDWARE_ERROR); | |
255 | 260 | r->sector += n; |
256 | 261 | r->sector_count -= n; |
257 | 262 | } else { |
... | ... | @@ -344,7 +349,8 @@ static int32_t scsi_send_command(SCSIDevice *d, uint32_t tag, |
344 | 349 | if (lun || buf[1] >> 5) { |
345 | 350 | /* Only LUN 0 supported. */ |
346 | 351 | DPRINTF("Unimplemented LUN %d\n", lun ? lun : buf[1] >> 5); |
347 | - goto fail; | |
352 | + if (command != 0x03 && command != 0x12) /* REQUEST SENSE and INQUIRY */ | |
353 | + goto fail; | |
348 | 354 | } |
349 | 355 | switch (command) { |
350 | 356 | case 0x0: |
... | ... | @@ -487,7 +493,10 @@ static int32_t scsi_send_command(SCSIDevice *d, uint32_t tag, |
487 | 493 | } |
488 | 494 | } |
489 | 495 | memset(outbuf, 0, 36); |
490 | - if (bdrv_get_type_hint(s->bdrv) == BDRV_TYPE_CDROM) { | |
496 | + | |
497 | + if (lun || buf[1] >> 5) { | |
498 | + outbuf[0] = 0x7f; /* LUN not supported */ | |
499 | + } else if (bdrv_get_type_hint(s->bdrv) == BDRV_TYPE_CDROM) { | |
491 | 500 | outbuf[0] = 5; |
492 | 501 | outbuf[1] = 0x80; |
493 | 502 | memcpy(&outbuf[16], "QEMU CD-ROM ", 16); |
... | ... | @@ -670,7 +679,7 @@ static int32_t scsi_send_command(SCSIDevice *d, uint32_t tag, |
670 | 679 | outbuf[7] = 0; |
671 | 680 | r->buf_len = 8; |
672 | 681 | } else { |
673 | - scsi_command_complete(r, SENSE_NOT_READY); | |
682 | + scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_NOT_READY); | |
674 | 683 | return 0; |
675 | 684 | } |
676 | 685 | break; |
... | ... | @@ -754,14 +763,17 @@ static int32_t scsi_send_command(SCSIDevice *d, uint32_t tag, |
754 | 763 | outbuf[3] = 8; |
755 | 764 | r->buf_len = 16; |
756 | 765 | break; |
766 | + case 0x2f: | |
767 | + DPRINTF("Verify (sector %d, count %d)\n", lba, len); | |
768 | + break; | |
757 | 769 | default: |
758 | 770 | DPRINTF("Unknown SCSI command (%2.2x)\n", buf[0]); |
759 | 771 | fail: |
760 | - scsi_command_complete(r, SENSE_ILLEGAL_REQUEST); | |
772 | + scsi_command_complete(r, STATUS_CHECK_CONDITION, SENSE_ILLEGAL_REQUEST); | |
761 | 773 | return 0; |
762 | 774 | } |
763 | 775 | if (r->sector_count == 0 && r->buf_len == 0) { |
764 | - scsi_command_complete(r, SENSE_NO_SENSE); | |
776 | + scsi_command_complete(r, STATUS_GOOD, SENSE_NO_SENSE); | |
765 | 777 | } |
766 | 778 | len = r->sector_count * 512 + r->buf_len; |
767 | 779 | if (is_write) { | ... | ... |