| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Generic SCSI Device support | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2007 Bull S.A.S. | 
					
						
							|  |  |  |  * Based on code by Paul Brook | 
					
						
							|  |  |  |  * Based on code by Fabrice Bellard | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Written by Laurent Vivier <Laurent.Vivier@bull.net> | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2011-06-26 12:21:35 +10:00
										 |  |  |  * This code is licensed under the LGPL. | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "qemu-common.h"
 | 
					
						
							| 
									
										
										
										
											2010-02-18 16:24:31 +01:00
										 |  |  | #include "qemu-error.h"
 | 
					
						
							| 
									
										
										
										
											2009-10-30 09:54:00 +01:00
										 |  |  | #include "scsi.h"
 | 
					
						
							| 
									
										
										
										
											2010-08-24 15:22:24 +00:00
										 |  |  | #include "blockdev.h"
 | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-31 14:24:04 +02:00
										 |  |  | #ifdef __linux__
 | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | //#define DEBUG_SCSI
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG_SCSI
 | 
					
						
							| 
									
										
										
										
											2009-05-13 17:53:17 +00:00
										 |  |  | #define DPRINTF(fmt, ...) \
 | 
					
						
							|  |  |  | do { printf("scsi-generic: " fmt , ## __VA_ARGS__); } while (0) | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2009-05-13 17:53:17 +00:00
										 |  |  | #define DPRINTF(fmt, ...) do {} while(0)
 | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-13 17:53:17 +00:00
										 |  |  | #define BADF(fmt, ...) \
 | 
					
						
							|  |  |  | do { fprintf(stderr, "scsi-generic: " fmt , ## __VA_ARGS__); } while (0) | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <sys/types.h>
 | 
					
						
							|  |  |  | #include <sys/stat.h>
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							|  |  |  | #include <scsi/sg.h>
 | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:53 +01:00
										 |  |  | #include "scsi-defs.h"
 | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-09-23 13:38:44 +00:00
										 |  |  | #define SCSI_SENSE_BUF_SIZE 96
 | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-12 14:53:43 +02:00
										 |  |  | #define SG_ERR_DRIVER_TIMEOUT  0x06
 | 
					
						
							|  |  |  | #define SG_ERR_DRIVER_SENSE    0x08
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define SG_ERR_DID_OK          0x00
 | 
					
						
							|  |  |  | #define SG_ERR_DID_NO_CONNECT  0x01
 | 
					
						
							|  |  |  | #define SG_ERR_DID_BUS_BUSY    0x02
 | 
					
						
							|  |  |  | #define SG_ERR_DID_TIME_OUT    0x03
 | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifndef MAX_UINT
 | 
					
						
							|  |  |  | #define MAX_UINT ((unsigned int)-1)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:48 +01:00
										 |  |  | typedef struct SCSIGenericReq { | 
					
						
							|  |  |  |     SCSIRequest req; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     uint8_t *buf; | 
					
						
							|  |  |  |     int buflen; | 
					
						
							|  |  |  |     int len; | 
					
						
							|  |  |  |     sg_io_hdr_t io_header; | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:48 +01:00
										 |  |  | } SCSIGenericReq; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-15 07:24:30 -05:00
										 |  |  | static void scsi_generic_save_request(QEMUFile *f, SCSIRequest *req) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     SCSIGenericReq *r = DO_UPCAST(SCSIGenericReq, req, req); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qemu_put_sbe32s(f, &r->buflen); | 
					
						
							|  |  |  |     if (r->buflen && r->req.cmd.mode == SCSI_XFER_TO_DEV) { | 
					
						
							|  |  |  |         assert(!r->req.sg); | 
					
						
							|  |  |  |         qemu_put_buffer(f, r->buf, r->req.cmd.xfer); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void scsi_generic_load_request(QEMUFile *f, SCSIRequest *req) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     SCSIGenericReq *r = DO_UPCAST(SCSIGenericReq, req, req); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qemu_get_sbe32s(f, &r->buflen); | 
					
						
							|  |  |  |     if (r->buflen && r->req.cmd.mode == SCSI_XFER_TO_DEV) { | 
					
						
							|  |  |  |         assert(!r->req.sg); | 
					
						
							|  |  |  |         qemu_get_buffer(f, r->buf, r->req.cmd.xfer); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-18 16:01:56 +02:00
										 |  |  | static void scsi_free_request(SCSIRequest *req) | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-18 16:01:56 +02:00
										 |  |  |     SCSIGenericReq *r = DO_UPCAST(SCSIGenericReq, req, req); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |     g_free(r->buf); | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Helper function for command completion.  */ | 
					
						
							|  |  |  | static void scsi_command_complete(void *opaque, int ret) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:06 +02:00
										 |  |  |     int status; | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:48 +01:00
										 |  |  |     SCSIGenericReq *r = (SCSIGenericReq *)opaque; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-25 16:53:46 +02:00
										 |  |  |     r->req.aiocb = NULL; | 
					
						
							| 
									
										
										
										
											2011-10-12 14:53:43 +02:00
										 |  |  |     if (r->io_header.driver_status & SG_ERR_DRIVER_SENSE) { | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:07 +02:00
										 |  |  |         r->req.sense_len = r->io_header.sb_len_wr; | 
					
						
							| 
									
										
										
										
											2011-10-12 14:53:43 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-10-17 08:08:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-18 12:53:14 +02:00
										 |  |  |     if (ret != 0) { | 
					
						
							|  |  |  |         switch (ret) { | 
					
						
							| 
									
										
										
										
											2011-04-18 14:29:16 +02:00
										 |  |  |         case -EDOM: | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:06 +02:00
										 |  |  |             status = TASK_SET_FULL; | 
					
						
							| 
									
										
										
										
											2011-04-18 14:29:16 +02:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2011-04-18 12:53:14 +02:00
										 |  |  |         case -ENOMEM: | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:06 +02:00
										 |  |  |             status = CHECK_CONDITION; | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:07 +02:00
										 |  |  |             scsi_req_build_sense(&r->req, SENSE_CODE(TARGET_FAILURE)); | 
					
						
							| 
									
										
										
										
											2011-04-18 12:53:14 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:06 +02:00
										 |  |  |             status = CHECK_CONDITION; | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:07 +02:00
										 |  |  |             scsi_req_build_sense(&r->req, SENSE_CODE(IO_ERROR)); | 
					
						
							| 
									
										
										
										
											2011-04-18 12:53:14 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2011-10-12 14:53:43 +02:00
										 |  |  |         if (r->io_header.host_status == SG_ERR_DID_NO_CONNECT || | 
					
						
							|  |  |  |             r->io_header.host_status == SG_ERR_DID_BUS_BUSY || | 
					
						
							|  |  |  |             r->io_header.host_status == SG_ERR_DID_TIME_OUT || | 
					
						
							|  |  |  |             (r->io_header.driver_status & SG_ERR_DRIVER_TIMEOUT)) { | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:06 +02:00
										 |  |  |             status = BUSY; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |             BADF("Driver Timeout\n"); | 
					
						
							| 
									
										
										
										
											2011-10-12 14:53:43 +02:00
										 |  |  |         } else if (r->io_header.host_status) { | 
					
						
							|  |  |  |             status = CHECK_CONDITION; | 
					
						
							|  |  |  |             scsi_req_build_sense(&r->req, SENSE_CODE(I_T_NEXUS_LOSS)); | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:06 +02:00
										 |  |  |         } else if (r->io_header.status) { | 
					
						
							|  |  |  |             status = r->io_header.status; | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:07 +02:00
										 |  |  |         } else if (r->io_header.driver_status & SG_ERR_DRIVER_SENSE) { | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:06 +02:00
										 |  |  |             status = CHECK_CONDITION; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             status = GOOD; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-10-17 08:08:56 +00:00
										 |  |  |     DPRINTF("Command complete 0x%p tag=0x%x status=%d\n", | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:06 +02:00
										 |  |  |             r, r->req.tag, status); | 
					
						
							| 
									
										
										
										
											2009-11-26 15:34:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:06 +02:00
										 |  |  |     scsi_req_complete(&r->req, status); | 
					
						
							| 
									
										
										
										
											2011-10-25 12:53:35 +02:00
										 |  |  |     if (!r->req.io_canceled) { | 
					
						
							|  |  |  |         scsi_req_unref(&r->req); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Cancel a pending data transfer.  */ | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  | static void scsi_cancel_io(SCSIRequest *req) | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  |     SCSIGenericReq *r = DO_UPCAST(SCSIGenericReq, req, req); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DPRINTF("Cancel tag=0x%x\n", req->tag); | 
					
						
							|  |  |  |     if (r->req.aiocb) { | 
					
						
							|  |  |  |         bdrv_aio_cancel(r->req.aiocb); | 
					
						
							| 
									
										
										
										
											2011-10-25 12:53:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /* This reference was left in by scsi_*_data.  We take ownership of
 | 
					
						
							|  |  |  |          * it independent of whether bdrv_aio_cancel completes the request | 
					
						
							|  |  |  |          * or not.  */ | 
					
						
							|  |  |  |         scsi_req_unref(&r->req); | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  |     r->req.aiocb = NULL; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int execute_command(BlockDriverState *bdrv, | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:48 +01:00
										 |  |  |                            SCSIGenericReq *r, int direction, | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 			   BlockDriverCompletionFunc *complete) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     r->io_header.interface_id = 'S'; | 
					
						
							|  |  |  |     r->io_header.dxfer_direction = direction; | 
					
						
							|  |  |  |     r->io_header.dxferp = r->buf; | 
					
						
							|  |  |  |     r->io_header.dxfer_len = r->buflen; | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:51 +01:00
										 |  |  |     r->io_header.cmdp = r->req.cmd.buf; | 
					
						
							|  |  |  |     r->io_header.cmd_len = r->req.cmd.len; | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:07 +02:00
										 |  |  |     r->io_header.mx_sb_len = sizeof(r->req.sense); | 
					
						
							|  |  |  |     r->io_header.sbp = r->req.sense; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     r->io_header.timeout = MAX_UINT; | 
					
						
							|  |  |  |     r->io_header.usr_ptr = r; | 
					
						
							|  |  |  |     r->io_header.flags |= SG_FLAG_DIRECT_IO; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:48 +01:00
										 |  |  |     r->req.aiocb = bdrv_aio_ioctl(bdrv, SG_IO, &r->io_header, complete, r); | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void scsi_read_complete(void * opaque, int ret) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:48 +01:00
										 |  |  |     SCSIGenericReq *r = (SCSIGenericReq *)opaque; | 
					
						
							| 
									
										
										
										
											2011-10-13 10:36:27 +02:00
										 |  |  |     SCSIDevice *s = r->req.dev; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     int len; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-25 16:53:46 +02:00
										 |  |  |     r->req.aiocb = NULL; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     if (ret) { | 
					
						
							| 
									
										
										
										
											2010-09-01 16:33:21 +02:00
										 |  |  |         DPRINTF("IO error ret %d\n", ret); | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |         scsi_command_complete(r, ret); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     len = r->io_header.dxfer_len - r->io_header.resid; | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:48 +01:00
										 |  |  |     DPRINTF("Data ready tag=0x%x len=%d\n", r->req.tag, len); | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     r->len = -1; | 
					
						
							| 
									
										
										
										
											2011-05-03 14:15:59 +02:00
										 |  |  |     if (len == 0) { | 
					
						
							| 
									
										
										
										
											2008-10-17 08:08:56 +00:00
										 |  |  |         scsi_command_complete(r, 0); | 
					
						
							| 
									
										
										
										
											2011-05-03 14:15:59 +02:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2011-10-13 10:36:27 +02:00
										 |  |  |         /* Snoop READ CAPACITY output to set the blocksize.  */ | 
					
						
							|  |  |  |         if (r->req.cmd.buf[0] == READ_CAPACITY_10) { | 
					
						
							|  |  |  |             s->blocksize = ldl_be_p(&r->buf[4]); | 
					
						
							| 
									
										
										
										
											2011-10-13 10:39:50 +02:00
										 |  |  |             s->max_lba = ldl_be_p(&r->buf[0]); | 
					
						
							| 
									
										
										
										
											2011-10-13 10:36:27 +02:00
										 |  |  |         } else if (r->req.cmd.buf[0] == SERVICE_ACTION_IN_16 && | 
					
						
							|  |  |  |                    (r->req.cmd.buf[1] & 31) == SAI_READ_CAPACITY_16) { | 
					
						
							|  |  |  |             s->blocksize = ldl_be_p(&r->buf[8]); | 
					
						
							| 
									
										
										
										
											2011-10-13 10:39:50 +02:00
										 |  |  |             s->max_lba = ldq_be_p(&r->buf[0]); | 
					
						
							| 
									
										
										
										
											2011-10-13 10:36:27 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         bdrv_set_buffer_alignment(s->conf.bs, s->blocksize); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-18 14:59:13 +02:00
										 |  |  |         scsi_req_data(&r->req, len); | 
					
						
							| 
									
										
										
										
											2011-10-25 12:53:35 +02:00
										 |  |  |         if (!r->req.io_canceled) { | 
					
						
							|  |  |  |             scsi_req_unref(&r->req); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-05-03 14:15:59 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Read more data from scsi device into buffer.  */ | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  | static void scsi_read_data(SCSIRequest *req) | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  |     SCSIGenericReq *r = DO_UPCAST(SCSIGenericReq, req, req); | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |     SCSIDevice *s = r->req.dev; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  |     DPRINTF("scsi_read_data 0x%x\n", req->tag); | 
					
						
							| 
									
										
										
										
											2011-10-25 12:53:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* The request is used as the AIO opaque value, so add a ref.  */ | 
					
						
							|  |  |  |     scsi_req_ref(&r->req); | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     if (r->len == -1) { | 
					
						
							|  |  |  |         scsi_command_complete(r, 0); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |     ret = execute_command(s->conf.bs, r, SG_DXFER_FROM_DEV, scsi_read_complete); | 
					
						
							| 
									
										
										
										
											2011-04-18 12:53:14 +02:00
										 |  |  |     if (ret < 0) { | 
					
						
							|  |  |  |         scsi_command_complete(r, ret); | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void scsi_write_complete(void * opaque, int ret) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:48 +01:00
										 |  |  |     SCSIGenericReq *r = (SCSIGenericReq *)opaque; | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |     SCSIDevice *s = r->req.dev; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     DPRINTF("scsi_write_complete() ret = %d\n", ret); | 
					
						
							| 
									
										
										
										
											2011-05-25 16:53:46 +02:00
										 |  |  |     r->req.aiocb = NULL; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     if (ret) { | 
					
						
							|  |  |  |         DPRINTF("IO error\n"); | 
					
						
							|  |  |  |         scsi_command_complete(r, ret); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:51 +01:00
										 |  |  |     if (r->req.cmd.buf[0] == MODE_SELECT && r->req.cmd.buf[4] == 12 && | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |         s->type == TYPE_TAPE) { | 
					
						
							|  |  |  |         s->blocksize = (r->buf[9] << 16) | (r->buf[10] << 8) | r->buf[11]; | 
					
						
							|  |  |  |         DPRINTF("block size %d\n", s->blocksize); | 
					
						
							| 
									
										
										
										
											2008-10-17 08:08:56 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     scsi_command_complete(r, ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Write data to a scsi device.  Returns nonzero on failure.
 | 
					
						
							|  |  |  |    The transfer may complete asynchronously.  */ | 
					
						
							| 
									
										
										
										
											2011-04-22 09:39:16 +02:00
										 |  |  | static void scsi_write_data(SCSIRequest *req) | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  |     SCSIGenericReq *r = DO_UPCAST(SCSIGenericReq, req, req); | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |     SCSIDevice *s = r->req.dev; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  |     DPRINTF("scsi_write_data 0x%x\n", req->tag); | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     if (r->len == 0) { | 
					
						
							|  |  |  |         r->len = r->buflen; | 
					
						
							| 
									
										
										
										
											2011-04-18 14:59:13 +02:00
										 |  |  |         scsi_req_data(&r->req, r->len); | 
					
						
							| 
									
										
										
										
											2011-04-22 09:39:16 +02:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-25 12:53:35 +02:00
										 |  |  |     /* The request is used as the AIO opaque value, so add a ref.  */ | 
					
						
							|  |  |  |     scsi_req_ref(&r->req); | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |     ret = execute_command(s->conf.bs, r, SG_DXFER_TO_DEV, scsi_write_complete); | 
					
						
							| 
									
										
										
										
											2011-04-18 12:53:14 +02:00
										 |  |  |     if (ret < 0) { | 
					
						
							|  |  |  |         scsi_command_complete(r, ret); | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Return a pointer to the data buffer.  */ | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  | static uint8_t *scsi_get_buf(SCSIRequest *req) | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  |     SCSIGenericReq *r = DO_UPCAST(SCSIGenericReq, req, req); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     return r->buf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Execute a scsi command.  Returns the length of the data expected by the
 | 
					
						
							|  |  |  |    command.  This will be Positive for data transfers from the device | 
					
						
							|  |  |  |    (eg. disk reads), negative for transfers to the device (eg. disk writes), | 
					
						
							|  |  |  |    and zero if the command does not transfer any data.  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  | static int32_t scsi_send_command(SCSIRequest *req, uint8_t *cmd) | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  |     SCSIGenericReq *r = DO_UPCAST(SCSIGenericReq, req, req); | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |     SCSIDevice *s = r->req.dev; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-01 16:33:21 +02:00
										 |  |  |     DPRINTF("Command: lun=%d tag=0x%x len %zd data=0x%02x", lun, tag, | 
					
						
							|  |  |  |             r->req.cmd.xfer, cmd[0]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG_SCSI
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         int i; | 
					
						
							|  |  |  |         for (i = 1; i < r->req.cmd.len; i++) { | 
					
						
							|  |  |  |             printf(" 0x%02x", cmd[i]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         printf("\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:55 +01:00
										 |  |  |     if (r->req.cmd.xfer == 0) { | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |         if (r->buf != NULL) | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |             g_free(r->buf); | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |         r->buflen = 0; | 
					
						
							|  |  |  |         r->buf = NULL; | 
					
						
							| 
									
										
										
										
											2011-10-25 12:53:35 +02:00
										 |  |  |         /* The request is used as the AIO opaque value, so add a ref.  */ | 
					
						
							|  |  |  |         scsi_req_ref(&r->req); | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |         ret = execute_command(s->conf.bs, r, SG_DXFER_NONE, scsi_command_complete); | 
					
						
							| 
									
										
										
										
											2011-04-18 12:53:14 +02:00
										 |  |  |         if (ret < 0) { | 
					
						
							|  |  |  |             scsi_command_complete(r, ret); | 
					
						
							|  |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:55 +01:00
										 |  |  |     if (r->buflen != r->req.cmd.xfer) { | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |         if (r->buf != NULL) | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |             g_free(r->buf); | 
					
						
							|  |  |  |         r->buf = g_malloc(r->req.cmd.xfer); | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:55 +01:00
										 |  |  |         r->buflen = r->req.cmd.xfer; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     memset(r->buf, 0, r->buflen); | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:55 +01:00
										 |  |  |     r->len = r->req.cmd.xfer; | 
					
						
							| 
									
										
										
										
											2009-11-26 15:33:57 +01:00
										 |  |  |     if (r->req.cmd.mode == SCSI_XFER_TO_DEV) { | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |         r->len = 0; | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  |         return -r->req.cmd.xfer; | 
					
						
							| 
									
										
										
										
											2011-04-18 16:01:56 +02:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2011-04-18 12:35:39 +02:00
										 |  |  |         return r->req.cmd.xfer; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-17 08:08:56 +00:00
										 |  |  | static int get_stream_blocksize(BlockDriverState *bdrv) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint8_t cmd[6]; | 
					
						
							|  |  |  |     uint8_t buf[12]; | 
					
						
							|  |  |  |     uint8_t sensebuf[8]; | 
					
						
							|  |  |  |     sg_io_hdr_t io_header; | 
					
						
							|  |  |  |     int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     memset(cmd, 0, sizeof(cmd)); | 
					
						
							|  |  |  |     memset(buf, 0, sizeof(buf)); | 
					
						
							|  |  |  |     cmd[0] = MODE_SENSE; | 
					
						
							|  |  |  |     cmd[4] = sizeof(buf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     memset(&io_header, 0, sizeof(io_header)); | 
					
						
							|  |  |  |     io_header.interface_id = 'S'; | 
					
						
							|  |  |  |     io_header.dxfer_direction = SG_DXFER_FROM_DEV; | 
					
						
							|  |  |  |     io_header.dxfer_len = sizeof(buf); | 
					
						
							|  |  |  |     io_header.dxferp = buf; | 
					
						
							|  |  |  |     io_header.cmdp = cmd; | 
					
						
							|  |  |  |     io_header.cmd_len = sizeof(cmd); | 
					
						
							|  |  |  |     io_header.mx_sb_len = sizeof(sensebuf); | 
					
						
							|  |  |  |     io_header.sbp = sensebuf; | 
					
						
							|  |  |  |     io_header.timeout = 6000; /* XXX */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-28 17:28:41 +00:00
										 |  |  |     ret = bdrv_ioctl(bdrv, SG_IO, &io_header); | 
					
						
							| 
									
										
										
										
											2011-10-12 14:49:48 +02:00
										 |  |  |     if (ret < 0 || io_header.driver_status || io_header.host_status) { | 
					
						
							| 
									
										
										
										
											2008-10-17 08:08:56 +00:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2011-10-12 14:49:48 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-10-17 08:08:56 +00:00
										 |  |  |     return (buf[9] << 16) | (buf[10] << 8) | buf[11]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-06 16:07:33 +02:00
										 |  |  | static void scsi_generic_reset(DeviceState *dev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-15 14:50:08 -06:00
										 |  |  |     SCSIDevice *s = SCSI_DEVICE(dev); | 
					
						
							| 
									
										
										
										
											2010-09-06 16:07:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |     scsi_device_purge_requests(s, SENSE_CODE(RESET)); | 
					
						
							| 
									
										
										
										
											2010-09-06 16:07:33 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  | static void scsi_destroy(SCSIDevice *s) | 
					
						
							| 
									
										
										
										
											2010-09-06 16:07:33 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |     scsi_device_purge_requests(s, SENSE_CODE(NO_SENSE)); | 
					
						
							|  |  |  |     blockdev_mark_auto_del(s->conf.bs); | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  | static int scsi_generic_initfn(SCSIDevice *s) | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     int sg_version; | 
					
						
							|  |  |  |     struct sg_scsi_id scsiid; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |     if (!s->conf.bs) { | 
					
						
							| 
									
										
										
										
											2011-12-21 11:37:57 +01:00
										 |  |  |         error_report("drive property not set"); | 
					
						
							| 
									
										
										
										
											2009-08-31 14:24:04 +02:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-28 17:22:55 +02:00
										 |  |  |     if (bdrv_get_on_error(s->conf.bs, 0) != BLOCKDEV_ON_ERROR_ENOSPC) { | 
					
						
							| 
									
										
										
										
											2010-05-27 20:02:28 +02:00
										 |  |  |         error_report("Device doesn't support drive option werror"); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-09-28 17:22:55 +02:00
										 |  |  |     if (bdrv_get_on_error(s->conf.bs, 1) != BLOCKDEV_ON_ERROR_REPORT) { | 
					
						
							| 
									
										
										
										
											2010-05-27 20:02:28 +02:00
										 |  |  |         error_report("Device doesn't support drive option rerror"); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  |     /* check we are using a driver managing SG_IO (version 3 and after */ | 
					
						
							| 
									
										
											  
											
												ISCSI: Add SCSI passthrough via scsi-generic to libiscsi
Update iscsi to allow passthrough of SG_IO scsi commands when the iscsi
device is forced to be scsi-generic.
Implement both bdrv_ioctl() and bdrv_aio_ioctl() in the iscsi backend,
emulate the SG_IO ioctl and pass the SCSI commands across to the
iscsi target.
This allows end-to-end passthrough of SCSI all the way from the guest,
to qemu, via scsi-generic, then libiscsi all the way to the iscsi target.
To activate this you need to specify that the iscsi lun should be treated
as a scsi-generic device.
Example:
    -device lsi -device scsi-generic,drive=MyISCSI \
    -drive file=iscsi://10.1.1.125/iqn.ronnie.test/1,if=none,id=MyISCSI
Note, you can currently not boot a qemu guest from a scsi device.
Note,
This only works when the host is linux, since the emulation relies on
definitions of SG_IO from the scsi-generic implementation in the
linux kernel.
It should be fairly easy to re-implement some structures similar enough
for non-linux hosts to do the same style of passthrough via a fake
scsi generic layer and libiscsi if need be.
Signed-off-by: Ronnie Sahlberg <ronniesahlberg@gmail.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
											
										 
											2012-05-25 21:59:01 +10:00
										 |  |  |     if (bdrv_ioctl(s->conf.bs, SG_GET_VERSION_NUM, &sg_version) < 0) { | 
					
						
							|  |  |  |         error_report("scsi generic interface not supported"); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (sg_version < 30000) { | 
					
						
							| 
									
										
										
										
											2011-12-21 11:37:57 +01:00
										 |  |  |         error_report("scsi generic interface too old"); | 
					
						
							| 
									
										
										
										
											2009-08-31 14:24:04 +02:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* get LUN of the /dev/sg? */ | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |     if (bdrv_ioctl(s->conf.bs, SG_GET_SCSI_ID, &scsiid)) { | 
					
						
							| 
									
										
										
										
											2011-12-21 11:37:57 +01:00
										 |  |  |         error_report("SG_GET_SCSI_ID ioctl failed"); | 
					
						
							| 
									
										
										
										
											2009-08-31 14:24:04 +02:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* define device state */ | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |     s->type = scsiid.scsi_type; | 
					
						
							|  |  |  |     DPRINTF("device type %d\n", s->type); | 
					
						
							| 
									
										
										
										
											2011-11-18 16:32:02 +01:00
										 |  |  |     if (s->type == TYPE_DISK || s->type == TYPE_ROM) { | 
					
						
							|  |  |  |         add_boot_device_path(s->conf.bootindex, &s->qdev, NULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-13 10:36:27 +02:00
										 |  |  |     switch (s->type) { | 
					
						
							|  |  |  |     case TYPE_TAPE: | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  |         s->blocksize = get_stream_blocksize(s->conf.bs); | 
					
						
							|  |  |  |         if (s->blocksize == -1) { | 
					
						
							|  |  |  |             s->blocksize = 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-10-13 10:36:27 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* Make a guess for block devices, we'll fix it when the guest sends.
 | 
					
						
							|  |  |  |          * READ CAPACITY.  If they don't, they likely would assume these sizes | 
					
						
							|  |  |  |          * anyway. (TODO: they could also send MODE SENSE). | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |     case TYPE_ROM: | 
					
						
							|  |  |  |     case TYPE_WORM: | 
					
						
							|  |  |  |         s->blocksize = 2048; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         s->blocksize = 512; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2008-10-17 08:08:56 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-10-12 12:49:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     DPRINTF("block size %d\n", s->blocksize); | 
					
						
							| 
									
										
										
										
											2009-08-31 14:24:04 +02:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-12 12:54:31 +02:00
										 |  |  | const SCSIReqOps scsi_generic_req_ops = { | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:08 +02:00
										 |  |  |     .size         = sizeof(SCSIGenericReq), | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:09 +02:00
										 |  |  |     .free_req     = scsi_free_request, | 
					
						
							|  |  |  |     .send_command = scsi_send_command, | 
					
						
							|  |  |  |     .read_data    = scsi_read_data, | 
					
						
							|  |  |  |     .write_data   = scsi_write_data, | 
					
						
							|  |  |  |     .cancel_io    = scsi_cancel_io, | 
					
						
							|  |  |  |     .get_buf      = scsi_get_buf, | 
					
						
							| 
									
										
										
										
											2011-12-15 07:24:30 -05:00
										 |  |  |     .load_request = scsi_generic_load_request, | 
					
						
							|  |  |  |     .save_request = scsi_generic_save_request, | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:08 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static SCSIRequest *scsi_new_request(SCSIDevice *d, uint32_t tag, uint32_t lun, | 
					
						
							| 
									
										
										
										
											2011-10-12 12:58:31 +02:00
										 |  |  |                                      uint8_t *buf, void *hba_private) | 
					
						
							| 
									
										
										
										
											2011-08-03 10:49:08 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     SCSIRequest *req; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     req = scsi_req_alloc(&scsi_generic_req_ops, d, tag, lun, hba_private); | 
					
						
							|  |  |  |     return req; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  | static Property scsi_generic_properties[] = { | 
					
						
							| 
									
										
										
										
											2012-08-28 12:46:18 +02:00
										 |  |  |     DEFINE_PROP_DRIVE("drive", SCSIDevice, conf.bs), | 
					
						
							|  |  |  |     DEFINE_PROP_INT32("bootindex", SCSIDevice, conf.bootindex, -1), | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     DEFINE_PROP_END_OF_LIST(), | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-15 14:50:08 -06:00
										 |  |  | static void scsi_generic_class_initfn(ObjectClass *klass, void *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     DeviceClass *dc = DEVICE_CLASS(klass); | 
					
						
							| 
									
										
										
										
											2011-12-15 14:50:08 -06:00
										 |  |  |     SCSIDeviceClass *sc = SCSI_DEVICE_CLASS(klass); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     sc->init         = scsi_generic_initfn; | 
					
						
							|  |  |  |     sc->destroy      = scsi_destroy; | 
					
						
							|  |  |  |     sc->alloc_req    = scsi_new_request; | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     dc->fw_name = "disk"; | 
					
						
							|  |  |  |     dc->desc = "pass through generic scsi device (/dev/sg*)"; | 
					
						
							|  |  |  |     dc->reset = scsi_generic_reset; | 
					
						
							|  |  |  |     dc->props = scsi_generic_properties; | 
					
						
							| 
									
										
										
										
											2011-12-15 07:24:30 -05:00
										 |  |  |     dc->vmsd  = &vmstate_scsi_device; | 
					
						
							| 
									
										
										
										
											2011-12-15 14:50:08 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  | static TypeInfo scsi_generic_info = { | 
					
						
							|  |  |  |     .name          = "scsi-generic", | 
					
						
							|  |  |  |     .parent        = TYPE_SCSI_DEVICE, | 
					
						
							|  |  |  |     .instance_size = sizeof(SCSIDevice), | 
					
						
							|  |  |  |     .class_init    = scsi_generic_class_initfn, | 
					
						
							| 
									
										
										
										
											2009-08-31 14:24:04 +02:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-09 15:20:55 +01:00
										 |  |  | static void scsi_generic_register_types(void) | 
					
						
							| 
									
										
										
										
											2009-08-31 14:24:04 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     type_register_static(&scsi_generic_info); | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-02-09 15:20:55 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | type_init(scsi_generic_register_types) | 
					
						
							| 
									
										
										
										
											2009-08-31 14:24:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-24 16:11:51 +00:00
										 |  |  | #endif /* __linux__ */
 |