Re: [Qemu-devel] [PATCH 2/2] virtio-scsi: Implement hotplug support for virtio-scsi

2012-06-25 Thread mc


Quoting Stefan Hajnoczi stefa...@gmail.com:


On Wed, Jun 20, 2012 at 7:47 AM, Cong Meng m...@linux.vnet.ibm.com wrote:

Implement the hotplug() and hot_unplug() interfaces in virtio-scsi,
by signal
the virtio_scsi.ko in guest kernel via event virtual queue.

The counterpart patch of virtio_scsi.ko will be sent soon in another thread.



Signed-off-by: Cong Meng m...@linux.vnet.ibm.com
Signed-off-by: Sen Wang senw...@linux.vnet.ibm.com
---
 hw/virtio-scsi.c |   72
+++--
 1 files changed, 69 insertions(+), 3 deletions(-)


I compared against the virtio-scsi specification and this looks good:
http://ozlabs.org/~rusty/virtio-spec/virtio-0.9.5.pdf

Dropped events and event throttling are not implemented by this patch.
 This means that the guest can miss events if it runs out of event
queue elements.  A scenario that might be able to trigger this is if
multiple LUNs are hotplugged in a single QEMU monitor callback.

Implementing dropped events is easy in hw/virtio-scsi.c.  Keep a bool
or counter of dropped events and report them when the guest kicks us
with a free event element (virtio_scsi_handle_event).


Yes. It's easy to do this in qemu. But I'm not sure what should be done
in virtio-scsi.ko to respond the VIRTIO_SCSI_T_EVENTS_MISSED event.
The spec says poll the logical units for unit attention conditions, or
just a whole bus rescan?



Stefan







Re: [Qemu-devel] [PATCH 2/2] virtio-scsi: Implement hotplug support for virtio-scsi

2012-06-25 Thread Stefan Hajnoczi
On Mon, Jun 25, 2012 at 03:51:13AM -0400, m...@linux.vnet.ibm.com wrote:
 
 Quoting Stefan Hajnoczi stefa...@gmail.com:
 
 On Wed, Jun 20, 2012 at 7:47 AM, Cong Meng m...@linux.vnet.ibm.com wrote:
 Implement the hotplug() and hot_unplug() interfaces in virtio-scsi,
 by signal
 the virtio_scsi.ko in guest kernel via event virtual queue.
 
 The counterpart patch of virtio_scsi.ko will be sent soon in another thread.
 
 Signed-off-by: Cong Meng m...@linux.vnet.ibm.com
 Signed-off-by: Sen Wang senw...@linux.vnet.ibm.com
 ---
  hw/virtio-scsi.c |   72
 +++--
  1 files changed, 69 insertions(+), 3 deletions(-)
 
 I compared against the virtio-scsi specification and this looks good:
 http://ozlabs.org/~rusty/virtio-spec/virtio-0.9.5.pdf
 
 Dropped events and event throttling are not implemented by this patch.
  This means that the guest can miss events if it runs out of event
 queue elements.  A scenario that might be able to trigger this is if
 multiple LUNs are hotplugged in a single QEMU monitor callback.
 
 Implementing dropped events is easy in hw/virtio-scsi.c.  Keep a bool
 or counter of dropped events and report them when the guest kicks us
 with a free event element (virtio_scsi_handle_event).
 
 Yes. It's easy to do this in qemu. But I'm not sure what should be done
 in virtio-scsi.ko to respond the VIRTIO_SCSI_T_EVENTS_MISSED event.
 The spec says poll the logical units for unit attention conditions, or
 just a whole bus rescan?

I'm not sure what the answer is either, maybe you can find an existing
SCSI LLD that does what you need.

Stefan




Re: [Qemu-devel] [PATCH 2/2] virtio-scsi: Implement hotplug support for virtio-scsi

2012-06-21 Thread Stefan Hajnoczi
On Wed, Jun 20, 2012 at 7:47 AM, Cong Meng m...@linux.vnet.ibm.com wrote:
 Implement the hotplug() and hot_unplug() interfaces in virtio-scsi, by signal
 the virtio_scsi.ko in guest kernel via event virtual queue.

 The counterpart patch of virtio_scsi.ko will be sent soon in another thread.

 Signed-off-by: Cong Meng m...@linux.vnet.ibm.com
 Signed-off-by: Sen Wang senw...@linux.vnet.ibm.com
 ---
  hw/virtio-scsi.c |   72 +++--
  1 files changed, 69 insertions(+), 3 deletions(-)

I compared against the virtio-scsi specification and this looks good:
http://ozlabs.org/~rusty/virtio-spec/virtio-0.9.5.pdf

Dropped events and event throttling are not implemented by this patch.
 This means that the guest can miss events if it runs out of event
queue elements.  A scenario that might be able to trigger this is if
multiple LUNs are hotplugged in a single QEMU monitor callback.

Implementing dropped events is easy in hw/virtio-scsi.c.  Keep a bool
or counter of dropped events and report them when the guest kicks us
with a free event element (virtio_scsi_handle_event).

Stefan



[Qemu-devel] [PATCH 2/2] virtio-scsi: Implement hotplug support for virtio-scsi

2012-06-20 Thread Cong Meng
Implement the hotplug() and hot_unplug() interfaces in virtio-scsi, by signal
the virtio_scsi.ko in guest kernel via event virtual queue.

The counterpart patch of virtio_scsi.ko will be sent soon in another thread.

Signed-off-by: Cong Meng m...@linux.vnet.ibm.com
Signed-off-by: Sen Wang senw...@linux.vnet.ibm.com
---
 hw/virtio-scsi.c |   72 +++--
 1 files changed, 69 insertions(+), 3 deletions(-)

diff --git a/hw/virtio-scsi.c b/hw/virtio-scsi.c
index e8328f4..626ec5f 100644
--- a/hw/virtio-scsi.c
+++ b/hw/virtio-scsi.c
@@ -24,6 +24,10 @@
 #define VIRTIO_SCSI_MAX_TARGET  255
 #define VIRTIO_SCSI_MAX_LUN 16383
 
+/* Feature Bits */
+#define VIRTIO_SCSI_F_INOUT0
+#define VIRTIO_SCSI_F_HOTPLUG  1
+
 /* Response codes */
 #define VIRTIO_SCSI_S_OK   0
 #define VIRTIO_SCSI_S_OVERRUN  1
@@ -60,6 +64,11 @@
 #define VIRTIO_SCSI_T_TRANSPORT_RESET  1
 #define VIRTIO_SCSI_T_ASYNC_NOTIFY 2
 
+/* Reasons of transport reset event */ 
+#define VIRTIO_SCSI_EVT_RESET_HARD 0
+#define VIRTIO_SCSI_EVT_RESET_RESCAN   1
+#define VIRTIO_SCSI_EVT_RESET_REMOVED  2
+
 /* SCSI command request, followed by data-out */
 typedef struct {
 uint8_t lun[8];  /* Logical Unit Number */
@@ -206,11 +215,12 @@ static void qemu_sgl_init_external(QEMUSGList *qsgl, 
struct iovec *sg,
 static void virtio_scsi_parse_req(VirtIOSCSI *s, VirtQueue *vq,
   VirtIOSCSIReq *req)
 {
-assert(req-elem.out_num  req-elem.in_num);
+assert(req-elem.in_num);
 req-vq = vq;
 req-dev = s;
 req-sreq = NULL;
-req-req.buf = req-elem.out_sg[0].iov_base;
+if (req-elem.out_num)
+req-req.buf = req-elem.out_sg[0].iov_base;
 req-resp.buf = req-elem.in_sg[0].iov_base;
 
 if (req-elem.out_num  1) {
@@ -405,6 +415,10 @@ static void virtio_scsi_handle_ctrl(VirtIODevice *vdev, 
VirtQueue *vq)
 }
 }
 
+static void virtio_scsi_handle_event(VirtIODevice *vdev, VirtQueue *vq)
+{
+}
+
 static void virtio_scsi_command_complete(SCSIRequest *r, uint32_t status,
  size_t resid)
 {
@@ -541,6 +555,7 @@ static void virtio_scsi_set_config(VirtIODevice *vdev,
 static uint32_t virtio_scsi_get_features(VirtIODevice *vdev,
  uint32_t requested_features)
 {
+requested_features |= (1UL  VIRTIO_SCSI_F_HOTPLUG);
 return requested_features;
 }
 
@@ -568,6 +583,55 @@ static int virtio_scsi_load(QEMUFile *f, void *opaque, int 
version_id)
 return 0;
 }
 
+static void virtio_scsi_push_event(VirtIOSCSI *s, uint32_t target, uint32_t 
lun,
+   uint32_t event, uint32_t reason)
+{
+VirtIOSCSIReq *req;
+VirtIOSCSIEvent *evt;
+
+if ((req = virtio_scsi_pop_req(s, s-event_vq))) {
+int in_size;
+if (req-elem.out_num || req-elem.in_num != 1) {
+virtio_scsi_bad_req();
+}
+
+in_size = req-elem.in_sg[0].iov_len;
+if (in_size  sizeof(VirtIOSCSIEvent)) {
+virtio_scsi_bad_req();
+}
+
+evt = req-resp.event;
+evt-event = event;
+evt-reason = reason;
+evt-lun[0] = 0;
+evt-lun[1] = target;
+evt-lun[2] = (lun  8);
+evt-lun[3] = lun  0xFF;
+virtio_scsi_complete_req(req);
+}
+}
+
+static void virtio_scsi_hotplug(SCSIBus *bus, SCSIDevice *dev)
+{
+VirtIOSCSI *s = container_of(bus, VirtIOSCSI, bus);
+
+if (((s-vdev.guest_features  VIRTIO_SCSI_F_HOTPLUG)  1) 
+(s-vdev.status  VIRTIO_CONFIG_S_DRIVER_OK)) {
+virtio_scsi_push_event(s, dev-id, dev-lun, 
+VIRTIO_SCSI_T_TRANSPORT_RESET, VIRTIO_SCSI_EVT_RESET_RESCAN);
+}
+}
+
+static void virtio_scsi_hot_unplug(SCSIBus *bus, SCSIDevice *dev)
+{
+VirtIOSCSI *s = container_of(bus, VirtIOSCSI, bus);
+
+if ((s-vdev.guest_features  VIRTIO_SCSI_F_HOTPLUG)  1) {
+virtio_scsi_push_event(s, dev-id, dev-lun, 
+VIRTIO_SCSI_T_TRANSPORT_RESET, VIRTIO_SCSI_EVT_RESET_REMOVED);
+}
+}
+
 static struct SCSIBusInfo virtio_scsi_scsi_info = {
 .tcq = true,
 .max_channel = VIRTIO_SCSI_MAX_CHANNEL,
@@ -576,6 +640,8 @@ static struct SCSIBusInfo virtio_scsi_scsi_info = {
 
 .complete = virtio_scsi_command_complete,
 .cancel = virtio_scsi_request_cancelled,
+.hotplug = virtio_scsi_hotplug,
+.hot_unplug = virtio_scsi_hot_unplug,
 .get_sg_list = virtio_scsi_get_sg_list,
 .save_request = virtio_scsi_save_request,
 .load_request = virtio_scsi_load_request,
@@ -604,7 +670,7 @@ VirtIODevice *virtio_scsi_init(DeviceState *dev, 
VirtIOSCSIConf *proxyconf)
 s-ctrl_vq = virtio_add_queue(s-vdev, VIRTIO_SCSI_VQ_SIZE,
virtio_scsi_handle_ctrl);
 s-event_vq = virtio_add_queue(s-vdev, VIRTIO_SCSI_VQ_SIZE,
-