]> git.ipfire.org Git - ipfire-2.x.git/blobdiff - src/patches/suse-2.6.27.31/patches.drivers/0012-Staging-USB-IP-add-client-driver.patch
Move xen patchset to new version's subdir.
[ipfire-2.x.git] / src / patches / suse-2.6.27.31 / patches.drivers / 0012-Staging-USB-IP-add-client-driver.patch
diff --git a/src/patches/suse-2.6.27.31/patches.drivers/0012-Staging-USB-IP-add-client-driver.patch b/src/patches/suse-2.6.27.31/patches.drivers/0012-Staging-USB-IP-add-client-driver.patch
new file mode 100644 (file)
index 0000000..c872321
--- /dev/null
@@ -0,0 +1,2252 @@
+From 04679b3489e048cd5dae79e050a3afed8e4e42b6 Mon Sep 17 00:00:00 2001
+From: Takahiro Hirofuchi <hirofuchi@users.sourceforge.net>
+Date: Wed, 9 Jul 2008 14:56:51 -0600
+Subject: [PATCH 12/23] Staging: USB/IP: add client driver
+Patch-mainline: 2.6.28
+
+This adds the USB IP client driver
+
+Brian Merrell cleaned up a lot of this code and submitted it for
+inclusion.  Greg also did a lot of cleanup.
+
+Signed-off-by: Brian G. Merrell <bgmerrell@novell.com>
+Cc: Takahiro Hirofuchi <hirofuchi@users.sourceforge.net>
+Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
+---
+ drivers/staging/usbip/Kconfig      |   11 +
+ drivers/staging/usbip/Makefile     |    3 +
+ drivers/staging/usbip/vhci.h       |  142 ++++
+ drivers/staging/usbip/vhci_hcd.c   | 1275 ++++++++++++++++++++++++++++++++++++
+ drivers/staging/usbip/vhci_rx.c    |  251 +++++++
+ drivers/staging/usbip/vhci_sysfs.c |  250 +++++++
+ drivers/staging/usbip/vhci_tx.c    |  239 +++++++
+ 7 files changed, 2171 insertions(+), 0 deletions(-)
+ create mode 100644 drivers/staging/usbip/vhci.h
+ create mode 100644 drivers/staging/usbip/vhci_hcd.c
+ create mode 100644 drivers/staging/usbip/vhci_rx.c
+ create mode 100644 drivers/staging/usbip/vhci_sysfs.c
+ create mode 100644 drivers/staging/usbip/vhci_tx.c
+
+diff --git a/drivers/staging/usbip/Kconfig b/drivers/staging/usbip/Kconfig
+index 37efb5e..c4d68e1 100644
+--- a/drivers/staging/usbip/Kconfig
++++ b/drivers/staging/usbip/Kconfig
+@@ -12,3 +12,14 @@ config USB_IP_COMMON
+         module will be called usbip_common_mod.
+         If unsure, say N.
++
++config USB_IP_VHCI_HCD
++      tristate "USB IP client driver"
++      depends on USB_IP_COMMON
++      default N
++      ---help---
++       This enables the USB IP host controller driver which will
++       run on the client machine.
++
++       To compile this driver as a module, choose M here: the
++       module will be called vhci_hcd.
+diff --git a/drivers/staging/usbip/Makefile b/drivers/staging/usbip/Makefile
+index ce925ca..6ef4c39 100644
+--- a/drivers/staging/usbip/Makefile
++++ b/drivers/staging/usbip/Makefile
+@@ -1,6 +1,9 @@
+ obj-$(CONFIG_USB_IP_COMMON) += usbip_common_mod.o
+ usbip_common_mod-objs := usbip_common.o usbip_event.o
++obj-$(CONFIG_USB_IP_VHCI_HCD) += vhci-hcd.o
++vhci-hcd-objs := vhci_sysfs.o vhci_tx.o vhci_rx.o vhci_hcd.o
++
+ ifeq ($(CONFIG_USB_DEBUG),y)
+       EXTRA_CFLAGS += -DDEBUG
+ endif
+diff --git a/drivers/staging/usbip/vhci.h b/drivers/staging/usbip/vhci.h
+new file mode 100644
+index 0000000..5e37517
+--- /dev/null
++++ b/drivers/staging/usbip/vhci.h
+@@ -0,0 +1,142 @@
++/*
++ * Copyright (C) 2003-2008 Takahiro Hirofuchi
++ *
++ * This is free software; you can redistribute it and/or modify
++ * it under the terms of the GNU General Public License as published by
++ * the Free Software Foundation; either version 2 of the License, or
++ * (at your option) any later version.
++ *
++ * This is distributed in the hope that it will be useful,
++ * but WITHOUT ANY WARRANTY; without even the implied warranty of
++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
++ * GNU General Public License for more details.
++ *
++ * You should have received a copy of the GNU General Public License
++ * along with this program; if not, write to the Free Software
++ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
++ * USA.
++ */
++
++#include <linux/platform_device.h>
++#include "../../usb/core/hcd.h"
++
++
++struct vhci_device {
++      struct usb_device *udev;
++
++      /*
++       * devid specifies a remote usb device uniquely instead
++       * of combination of busnum and devnum.
++       */
++      __u32 devid;
++
++      /* speed of a remote device */
++      enum usb_device_speed speed;
++
++      /*  vhci root-hub port to which this device is attached  */
++      __u32 rhport;
++
++      struct usbip_device ud;
++
++
++      /* lock for the below link lists */
++      spinlock_t priv_lock;
++
++      /* vhci_priv is linked to one of them. */
++      struct list_head priv_tx;
++      struct list_head priv_rx;
++
++      /* vhci_unlink is linked to one of them */
++      struct list_head unlink_tx;
++      struct list_head unlink_rx;
++
++      /* vhci_tx thread sleeps for this queue */
++      wait_queue_head_t waitq_tx;
++};
++
++
++/* urb->hcpriv, use container_of() */
++struct vhci_priv {
++      unsigned long seqnum;
++      struct list_head list;
++
++      struct vhci_device *vdev;
++      struct urb *urb;
++};
++
++
++struct vhci_unlink {
++      /* seqnum of this request */
++      unsigned long seqnum;
++
++      struct list_head list;
++
++      /* seqnum of the unlink target */
++      unsigned long unlink_seqnum;
++};
++
++/*
++ * The number of ports is less than 16 ?
++ * USB_MAXCHILDREN is statically defined to 16 in usb.h.  Its maximum value
++ * would be 31 because the event_bits[1] of struct usb_hub is defined as
++ * unsigned long in hub.h
++ */
++#define VHCI_NPORTS 8
++
++/* for usb_bus.hcpriv */
++struct vhci_hcd {
++      spinlock_t      lock;
++
++      u32     port_status[VHCI_NPORTS];
++
++      unsigned        resuming:1;
++      unsigned long   re_timeout;
++
++      atomic_t seqnum;
++
++      /*
++       * NOTE:
++       * wIndex shows the port number and begins from 1.
++       * But, the index of this array begins from 0.
++       */
++      struct vhci_device vdev[VHCI_NPORTS];
++
++      /* vhci_device which has not been assiged its address yet */
++      int pending_port;
++};
++
++
++extern struct vhci_hcd *the_controller;
++extern struct attribute_group dev_attr_group;
++
++
++/*-------------------------------------------------------------------------*/
++/* prototype declaration */
++
++/* vhci_hcd.c */
++void rh_port_connect(int rhport, enum usb_device_speed speed);
++void rh_port_disconnect(int rhport);
++void vhci_rx_loop(struct usbip_task *ut);
++void vhci_tx_loop(struct usbip_task *ut);
++
++#define hardware              (&the_controller->pdev.dev)
++
++static inline struct vhci_device *port_to_vdev(__u32 port)
++{
++      return &the_controller->vdev[port];
++}
++
++static inline struct vhci_hcd *hcd_to_vhci(struct usb_hcd *hcd)
++{
++      return (struct vhci_hcd *) (hcd->hcd_priv);
++}
++
++static inline struct usb_hcd *vhci_to_hcd(struct vhci_hcd *vhci)
++{
++      return container_of((void *) vhci, struct usb_hcd, hcd_priv);
++}
++
++static inline struct device *vhci_dev(struct vhci_hcd *vhci)
++{
++      return vhci_to_hcd(vhci)->self.controller;
++}
+diff --git a/drivers/staging/usbip/vhci_hcd.c b/drivers/staging/usbip/vhci_hcd.c
+new file mode 100644
+index 0000000..5b5a2e3
+--- /dev/null
++++ b/drivers/staging/usbip/vhci_hcd.c
+@@ -0,0 +1,1275 @@
++/*
++ * Copyright (C) 2003-2008 Takahiro Hirofuchi
++ *
++ * This is free software; you can redistribute it and/or modify
++ * it under the terms of the GNU General Public License as published by
++ * the Free Software Foundation; either version 2 of the License, or
++ * (at your option) any later version.
++ *
++ * This is distributed in the hope that it will be useful,
++ * but WITHOUT ANY WARRANTY; without even the implied warranty of
++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
++ * GNU General Public License for more details.
++ *
++ * You should have received a copy of the GNU General Public License
++ * along with this program; if not, write to the Free Software
++ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
++ * USA.
++ */
++
++
++#include "usbip_common.h"
++#include "vhci.h"
++
++#define DRIVER_VERSION "1.0"
++#define DRIVER_AUTHOR "Takahiro Hirofuchi"
++#define DRIVER_DESC "Virtual Host Controller Interface Driver for USB/IP"
++#define DRIVER_LICENCE "GPL"
++MODULE_AUTHOR(DRIVER_AUTHOR);
++MODULE_DESCRIPTION(DRIVER_DESC);
++MODULE_LICENSE(DRIVER_LICENCE);
++
++
++
++/*
++ * TODO
++ *    - update root hub emulation
++ *    - move the emulation code to userland ?
++ *            porting to other operating systems
++ *            minimize kernel code
++ *    - add suspend/resume code
++ *    - clean up everything
++ */
++
++
++/* See usb gadget dummy hcd */
++
++
++static int vhci_hub_status(struct usb_hcd *hcd, char *buff);
++static int vhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
++              u16 wIndex, char *buff, u16 wLength);
++static int vhci_urb_enqueue(struct usb_hcd *hcd, struct urb *urb,
++                                                      gfp_t mem_flags);
++static int vhci_urb_dequeue(struct usb_hcd *hcd, struct urb *urb, int status);
++static int vhci_start(struct usb_hcd *vhci_hcd);
++static void vhci_stop(struct usb_hcd *hcd);
++static int vhci_get_frame_number(struct usb_hcd *hcd);
++
++static const char driver_name[] = "vhci_hcd";
++static const char driver_desc[] = "USB/IP Virtual Host Contoroller";
++
++struct vhci_hcd *the_controller;
++
++static const char *bit_desc[] = {
++      "CONNECTION",           /*0*/
++      "ENABLE",               /*1*/
++      "SUSPEND",              /*2*/
++      "OVER_CURRENT",         /*3*/
++      "RESET",                /*4*/
++      "R5",           /*5*/
++      "R6",           /*6*/
++      "R7",           /*7*/
++      "POWER",                /*8*/
++      "LOWSPEED",             /*9*/
++      "HIGHSPEED",            /*10*/
++      "PORT_TEST",            /*11*/
++      "INDICATOR",            /*12*/
++      "R13",          /*13*/
++      "R14",          /*14*/
++      "R15",          /*15*/
++      "C_CONNECTION",         /*16*/
++      "C_ENABLE",             /*17*/
++      "C_SUSPEND",            /*18*/
++      "C_OVER_CURRENT",       /*19*/
++      "C_RESET",              /*20*/
++      "R21",          /*21*/
++      "R22",          /*22*/
++      "R23",          /*23*/
++      "R24",          /*24*/
++      "R25",          /*25*/
++      "R26",          /*26*/
++      "R27",          /*27*/
++      "R28",          /*28*/
++      "R29",          /*29*/
++      "R30",          /*30*/
++      "R31",          /*31*/
++};
++
++
++static void dump_port_status(u32 status)
++{
++      int i = 0;
++
++      printk(KERN_DEBUG "status %08x:", status);
++      for (i = 0; i < 32; i++) {
++              if (status & (1 << i))
++                      printk(" %s", bit_desc[i]);
++      }
++
++      printk("\n");
++}
++
++
++
++void rh_port_connect(int rhport, enum usb_device_speed speed)
++{
++      unsigned long   flags;
++
++      dbg_vhci_rh("rh_port_connect %d\n", rhport);
++
++      spin_lock_irqsave(&the_controller->lock, flags);
++
++      the_controller->port_status[rhport] |= USB_PORT_STAT_CONNECTION
++              | (1 << USB_PORT_FEAT_C_CONNECTION);
++
++      switch (speed) {
++      case USB_SPEED_HIGH:
++              the_controller->port_status[rhport] |= USB_PORT_STAT_HIGH_SPEED;
++              break;
++      case USB_SPEED_LOW:
++              the_controller->port_status[rhport] |= USB_PORT_STAT_LOW_SPEED;
++              break;
++      default:
++              break;
++      }
++
++      /* spin_lock(&the_controller->vdev[rhport].ud.lock);
++       * the_controller->vdev[rhport].ud.status = VDEV_CONNECT;
++       * spin_unlock(&the_controller->vdev[rhport].ud.lock); */
++
++      the_controller->pending_port = rhport;
++
++      spin_unlock_irqrestore(&the_controller->lock, flags);
++
++      usb_hcd_poll_rh_status(vhci_to_hcd(the_controller));
++}
++
++void rh_port_disconnect(int rhport)
++{
++      unsigned long flags;
++
++      dbg_vhci_rh("rh_port_disconnect %d\n", rhport);
++
++      spin_lock_irqsave(&the_controller->lock, flags);
++      /* stop_activity(dum, driver); */
++      the_controller->port_status[rhport] &= ~USB_PORT_STAT_CONNECTION;
++      the_controller->port_status[rhport] |=
++                                      (1 << USB_PORT_FEAT_C_CONNECTION);
++
++
++      /* not yet complete the disconnection
++       * spin_lock(&vdev->ud.lock);
++       * vdev->ud.status = VHC_ST_DISCONNECT;
++       * spin_unlock(&vdev->ud.lock); */
++
++      spin_unlock_irqrestore(&the_controller->lock, flags);
++}
++
++
++
++/*----------------------------------------------------------------------*/
++
++#define PORT_C_MASK \
++      ((USB_PORT_STAT_C_CONNECTION \
++        | USB_PORT_STAT_C_ENABLE \
++        | USB_PORT_STAT_C_SUSPEND \
++        | USB_PORT_STAT_C_OVERCURRENT \
++        | USB_PORT_STAT_C_RESET) << 16)
++
++/*
++ * This function is almostly the same as dummy_hcd.c:dummy_hub_status() without
++ * suspend/resume support. But, it is modified to provide multiple ports.
++ *
++ * @buf: a bitmap to show which port status has been changed.
++ *  bit  0: reserved or used for another purpose?
++ *  bit  1: the status of port 0 has been changed.
++ *  bit  2: the status of port 1 has been changed.
++ *  ...
++ *  bit  7: the status of port 6 has been changed.
++ *  bit  8: the status of port 7 has been changed.
++ *  ...
++ *  bit 15: the status of port 14 has been changed.
++ *
++ * So, the maximum number of ports is 31 ( port 0 to port 30) ?
++ *
++ * The return value is the actual transfered length in byte. If nothing has
++ * been changed, return 0. In the case that the number of ports is less than or
++ * equal to 6 (VHCI_NPORTS==7), return 1.
++ *
++ */
++static int vhci_hub_status(struct usb_hcd *hcd, char *buf)
++{
++      struct vhci_hcd *vhci;
++      unsigned long   flags;
++      int             retval = 0;
++
++      /* the enough buffer is allocated according to USB_MAXCHILDREN */
++      unsigned long   *event_bits = (unsigned long *) buf;
++      int             rhport;
++      int             changed = 0;
++
++
++      *event_bits = 0;
++
++      vhci = hcd_to_vhci(hcd);
++
++      spin_lock_irqsave(&vhci->lock, flags);
++      if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) {
++              dbg_vhci_rh("hw accessible flag in on?\n");
++              goto done;
++      }
++
++      /* check pseudo status register for each port */
++      for (rhport = 0; rhport < VHCI_NPORTS; rhport++) {
++              if ((vhci->port_status[rhport] & PORT_C_MASK)) {
++                      /* The status of a port has been changed, */
++                      dbg_vhci_rh("port %d is changed\n", rhport);
++
++                      *event_bits |= 1 << (rhport + 1);
++                      changed = 1;
++              }
++      }
++
++      uinfo("changed %d\n", changed);
++
++      if (hcd->state == HC_STATE_SUSPENDED)
++              usb_hcd_resume_root_hub(hcd);
++
++      if (changed)
++              retval = 1 + (VHCI_NPORTS / 8);
++      else
++              retval = 0;
++
++done:
++      spin_unlock_irqrestore(&vhci->lock, flags);
++      return retval;
++}
++
++/* See hub_configure in hub.c */
++static inline void hub_descriptor(struct usb_hub_descriptor *desc)
++{
++      memset(desc, 0, sizeof(*desc));
++      desc->bDescriptorType = 0x29;
++      desc->bDescLength = 9;
++      desc->wHubCharacteristics = (__force __u16)
++              (__constant_cpu_to_le16(0x0001));
++      desc->bNbrPorts = VHCI_NPORTS;
++      desc->bitmap[0] = 0xff;
++      desc->bitmap[1] = 0xff;
++}
++
++static int vhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
++                          u16 wIndex, char *buf, u16 wLength)
++{
++      struct vhci_hcd *dum;
++      int             retval = 0;
++      unsigned long   flags;
++      int             rhport;
++
++      u32 prev_port_status[VHCI_NPORTS];
++
++      if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags))
++              return -ETIMEDOUT;
++
++      /*
++       * NOTE:
++       * wIndex shows the port number and begins from 1.
++       */
++      dbg_vhci_rh("typeReq %x wValue %x wIndex %x\n", typeReq, wValue,
++                                                              wIndex);
++      if (wIndex > VHCI_NPORTS)
++              printk(KERN_ERR "%s: invalid port number %d\n", __func__, wIndex);
++      rhport = ((__u8)(wIndex & 0x00ff)) - 1;
++
++      dum = hcd_to_vhci(hcd);
++
++      spin_lock_irqsave(&dum->lock, flags);
++
++      /* store old status and compare now and old later */
++      if (dbg_flag_vhci_rh) {
++              int i = 0;
++              for (i = 0; i < VHCI_NPORTS; i++)
++                      prev_port_status[i] = dum->port_status[i];
++      }
++
++      switch (typeReq) {
++      case ClearHubFeature:
++              dbg_vhci_rh(" ClearHubFeature\n");
++              break;
++      case ClearPortFeature:
++              switch (wValue) {
++              case USB_PORT_FEAT_SUSPEND:
++                      if (dum->port_status[rhport] & USB_PORT_STAT_SUSPEND) {
++                              /* 20msec signaling */
++                              dum->resuming = 1;
++                              dum->re_timeout =
++                                      jiffies + msecs_to_jiffies(20);
++                      }
++                      break;
++              case USB_PORT_FEAT_POWER:
++                      dbg_vhci_rh(" ClearPortFeature: USB_PORT_FEAT_POWER\n");
++                      dum->port_status[rhport] = 0;
++                      /* dum->address = 0; */
++                      /* dum->hdev = 0; */
++                      dum->resuming = 0;
++                      break;
++              case USB_PORT_FEAT_C_RESET:
++                      dbg_vhci_rh(" ClearPortFeature: "
++                                      "USB_PORT_FEAT_C_RESET\n");
++                      switch (dum->vdev[rhport].speed) {
++                      case USB_SPEED_HIGH:
++                              dum->port_status[rhport] |=
++                                              USB_PORT_STAT_HIGH_SPEED;
++                              break;
++                      case USB_SPEED_LOW:
++                              dum->port_status[rhport] |=
++                                              USB_PORT_STAT_LOW_SPEED;
++                              break;
++                      default:
++                              break;
++                      }
++              default:
++                      dbg_vhci_rh(" ClearPortFeature: default %x\n", wValue);
++                      dum->port_status[rhport] &= ~(1 << wValue);
++              }
++              break;
++      case GetHubDescriptor:
++              dbg_vhci_rh(" GetHubDescriptor\n");
++              hub_descriptor((struct usb_hub_descriptor *) buf);
++              break;
++      case GetHubStatus:
++              dbg_vhci_rh(" GetHubStatus\n");
++              *(__le32 *) buf = __constant_cpu_to_le32(0);
++              break;
++      case GetPortStatus:
++              dbg_vhci_rh(" GetPortStatus port %x\n", wIndex);
++              if (wIndex > VHCI_NPORTS || wIndex < 1) {
++                      printk(KERN_ERR "%s: invalid port number %d\n",
++                             __func__, wIndex);
++                      retval = -EPIPE;
++              }
++
++              /* we do no care of resume. */
++
++              /* whoever resets or resumes must GetPortStatus to
++               * complete it!!
++               *                                   */
++              if (dum->resuming && time_after(jiffies, dum->re_timeout)) {
++                      printk(KERN_ERR "%s: not yet\n", __func__);
++                      dum->port_status[rhport] |=
++                                      (1 << USB_PORT_FEAT_C_SUSPEND);
++                      dum->port_status[rhport] &=
++                                      ~(1 << USB_PORT_FEAT_SUSPEND);
++                      dum->resuming = 0;
++                      dum->re_timeout = 0;
++                      /* if (dum->driver && dum->driver->resume) {
++                       *      spin_unlock (&dum->lock);
++                       *      dum->driver->resume (&dum->gadget);
++                       *      spin_lock (&dum->lock);
++                       * } */
++              }
++
++              if ((dum->port_status[rhport] & (1 << USB_PORT_FEAT_RESET)) !=
++                              0 && time_after(jiffies, dum->re_timeout)) {
++                      dum->port_status[rhport] |=
++                                              (1 << USB_PORT_FEAT_C_RESET);
++                      dum->port_status[rhport] &=
++                                              ~(1 << USB_PORT_FEAT_RESET);
++                      dum->re_timeout = 0;
++
++                      if (dum->vdev[rhport].ud.status ==
++                                                      VDEV_ST_NOTASSIGNED) {
++                              dbg_vhci_rh(" enable rhport %d (status %u)\n",
++                                              rhport,
++                                              dum->vdev[rhport].ud.status);
++                              dum->port_status[rhport] |=
++                                                      USB_PORT_STAT_ENABLE;
++                      }
++#if 0
++                      if (dum->driver) {
++
++                              dum->port_status[rhport] |=
++                                                      USB_PORT_STAT_ENABLE;
++                              /* give it the best speed we agree on */
++                              dum->gadget.speed = dum->driver->speed;
++                              dum->gadget.ep0->maxpacket = 64;
++                              switch (dum->gadget.speed) {
++                              case USB_SPEED_HIGH:
++                                      dum->port_status[rhport] |=
++                                      USB_PORT_STAT_HIGH_SPEED;
++                                      break;
++                              case USB_SPEED_LOW:
++                                      dum->gadget.ep0->maxpacket = 8;
++                                      dum->port_status[rhport] |=
++                                      USB_PORT_STAT_LOW_SPEED;
++                                      break;
++                              default:
++                                      dum->gadget.speed = USB_SPEED_FULL;
++                                      break;
++                              }
++                      }
++#endif
++
++              }
++              ((u16 *) buf)[0] = cpu_to_le16(dum->port_status[rhport]);
++              ((u16 *) buf)[1] =
++                              cpu_to_le16(dum->port_status[rhport] >> 16);
++
++              dbg_vhci_rh(" GetPortStatus bye %x %x\n", ((u16 *)buf)[0],
++                                                      ((u16 *)buf)[1]);
++              break;
++      case SetHubFeature:
++              dbg_vhci_rh(" SetHubFeature\n");
++              retval = -EPIPE;
++              break;
++      case SetPortFeature:
++              switch (wValue) {
++              case USB_PORT_FEAT_SUSPEND:
++                      dbg_vhci_rh(" SetPortFeature: "
++                                      "USB_PORT_FEAT_SUSPEND\n");
++                      printk(KERN_ERR "%s: not yet\n", __func__);
++#if 0
++                      dum->port_status[rhport] |=
++                                              (1 << USB_PORT_FEAT_SUSPEND);
++                      if (dum->driver->suspend) {
++                              spin_unlock(&dum->lock);
++                              dum->driver->suspend(&dum->gadget);
++                              spin_lock(&dum->lock);
++                      }
++#endif
++                      break;
++              case USB_PORT_FEAT_RESET:
++                      dbg_vhci_rh(" SetPortFeature: USB_PORT_FEAT_RESET\n");
++                      /* if it's already running, disconnect first */
++                      if (dum->port_status[rhport] & USB_PORT_STAT_ENABLE) {
++                              dum->port_status[rhport] &=
++                                              ~(USB_PORT_STAT_ENABLE |
++                                                USB_PORT_STAT_LOW_SPEED |
++                                                USB_PORT_STAT_HIGH_SPEED);
++#if 0
++                              if (dum->driver) {
++                                      dev_dbg(hardware, "disconnect\n");
++                                      stop_activity(dum, dum->driver);
++                              }
++#endif
++
++                              /* FIXME test that code path! */
++                      }
++                      /* 50msec reset signaling */
++                      dum->re_timeout = jiffies + msecs_to_jiffies(50);
++
++                      /* FALLTHROUGH */
++              default:
++                      dbg_vhci_rh(" SetPortFeature: default %d\n", wValue);
++                      dum->port_status[rhport] |= (1 << wValue);
++              }
++              break;
++
++      default:
++              printk(KERN_ERR "%s: default: no such request\n", __func__);
++              /* dev_dbg (hardware,
++               *              "hub control req%04x v%04x i%04x l%d\n",
++               *              typeReq, wValue, wIndex, wLength); */
++
++              /* "protocol stall" on error */
++              retval = -EPIPE;
++      }
++
++      if (dbg_flag_vhci_rh) {
++              printk(KERN_DEBUG "port %d\n", rhport);
++              dump_port_status(prev_port_status[rhport]);
++              dump_port_status(dum->port_status[rhport]);
++      }
++      dbg_vhci_rh(" bye\n");
++
++      spin_unlock_irqrestore(&dum->lock, flags);
++
++      return retval;
++}
++
++
++
++/*----------------------------------------------------------------------*/
++
++static struct vhci_device *get_vdev(struct usb_device *udev)
++{
++      int i;
++
++      if (!udev)
++              return NULL;
++
++      for (i = 0; i < VHCI_NPORTS; i++)
++              if (the_controller->vdev[i].udev == udev)
++                      return port_to_vdev(i);
++
++      return NULL;
++}
++
++static void vhci_tx_urb(struct urb *urb)
++{
++      struct vhci_device *vdev = get_vdev(urb->dev);
++      struct vhci_priv *priv;
++      unsigned long flag;
++
++      if (!vdev) {
++              err("could not get virtual device");
++              /* BUG(); */
++              return;
++      }
++
++      spin_lock_irqsave(&vdev->priv_lock, flag);
++
++      priv = kzalloc(sizeof(struct vhci_priv), GFP_ATOMIC);
++      if (!priv) {
++              dev_err(&urb->dev->dev, "malloc vhci_priv\n");
++              spin_unlock_irqrestore(&vdev->priv_lock, flag);
++              usbip_event_add(&vdev->ud, VDEV_EVENT_ERROR_MALLOC);
++              return;
++      }
++
++      priv->seqnum = atomic_inc_return(&the_controller->seqnum);
++      if (priv->seqnum == 0xffff)
++              uinfo("seqnum max\n");
++
++      priv->vdev = vdev;
++      priv->urb = urb;
++
++      urb->hcpriv = (void *) priv;
++
++
++      list_add_tail(&priv->list, &vdev->priv_tx);
++
++      wake_up(&vdev->waitq_tx);
++      spin_unlock_irqrestore(&vdev->priv_lock, flag);
++}
++
++static int vhci_urb_enqueue(struct usb_hcd *hcd, struct urb *urb,
++                          gfp_t mem_flags)
++{
++      struct device *dev = &urb->dev->dev;
++      int ret = 0;
++      unsigned long flags;
++
++      dbg_vhci_hc("enter, usb_hcd %p urb %p mem_flags %d\n",
++                  hcd, urb, mem_flags);
++
++      /* patch to usb_sg_init() is in 2.5.60 */
++      BUG_ON(!urb->transfer_buffer && urb->transfer_buffer_length);
++
++      spin_lock_irqsave(&the_controller->lock, flags);
++
++      /* check HC is active or not */
++      if (!HC_IS_RUNNING(hcd->state)) {
++              dev_err(dev, "HC is not running\n");
++              spin_unlock_irqrestore(&the_controller->lock, flags);
++              return -ENODEV;
++      }
++
++      if (urb->status != -EINPROGRESS) {
++              dev_err(dev, "URB already unlinked!, status %d\n", urb->status);
++              spin_unlock_irqrestore(&the_controller->lock, flags);
++              return urb->status;
++      }
++
++      ret = usb_hcd_link_urb_to_ep(hcd, urb);
++      if (ret)
++              goto no_need_unlink;
++
++      /*
++       * The enumelation process is as follows;
++       *
++       *  1. Get_Descriptor request to DevAddrs(0) EndPoint(0)
++       *     to get max packet length of default pipe
++       *
++       *  2. Set_Address request to DevAddr(0) EndPoint(0)
++       *
++       */
++
++      if (usb_pipedevice(urb->pipe) == 0) {
++              __u8 type = usb_pipetype(urb->pipe);
++              struct usb_ctrlrequest *ctrlreq =
++                              (struct usb_ctrlrequest *) urb->setup_packet;
++              struct vhci_device *vdev =
++                              port_to_vdev(the_controller->pending_port);
++
++              if (type != PIPE_CONTROL || !ctrlreq) {
++                      dev_err(dev, "invalid request to devnum 0\n");
++                      ret = EINVAL;
++                      goto no_need_xmit;
++              }
++
++              switch (ctrlreq->bRequest) {
++              case USB_REQ_SET_ADDRESS:
++                      /* set_address may come when a device is reset */
++                      dev_info(dev, "SetAddress Request (%d) to port %d\n",
++                               ctrlreq->wValue, vdev->rhport);
++
++                      vdev->udev = urb->dev;
++
++                      spin_lock(&vdev->ud.lock);
++                      vdev->ud.status = VDEV_ST_USED;
++                      spin_unlock(&vdev->ud.lock);
++
++                      if (urb->status == -EINPROGRESS) {
++                              /* This request is successfully completed. */
++                              /* If not -EINPROGRESS, possibly unlinked. */
++                              urb->status = 0;
++                      }
++
++                      goto no_need_xmit;
++
++              case USB_REQ_GET_DESCRIPTOR:
++                      if (ctrlreq->wValue == (USB_DT_DEVICE << 8))
++                              dbg_vhci_hc("Not yet?: "
++                                              "Get_Descriptor to device 0 "
++                                              "(get max pipe size)\n");
++
++                      /* FIXME: reference count? (usb_get_dev()) */
++                      vdev->udev = urb->dev;
++                      goto out;
++
++              default:
++                      /* NOT REACHED */
++                      dev_err(dev, "invalid request to devnum 0 bRequest %u, "
++                              "wValue %u\n", ctrlreq->bRequest,
++                              ctrlreq->wValue);
++                      ret =  -EINVAL;
++                      goto no_need_xmit;
++              }
++
++      }
++
++out:
++      vhci_tx_urb(urb);
++
++      spin_unlock_irqrestore(&the_controller->lock, flags);
++
++      return 0;
++
++no_need_xmit:
++      usb_hcd_unlink_urb_from_ep(hcd, urb);
++no_need_unlink:
++      spin_unlock_irqrestore(&the_controller->lock, flags);
++
++      usb_hcd_giveback_urb(vhci_to_hcd(the_controller), urb, urb->status);
++
++      return 0;
++}
++
++/*
++ * vhci_rx gives back the urb after receiving the reply of the urb.  If an
++ * unlink pdu is sent or not, vhci_rx receives a normal return pdu and gives
++ * back its urb. For the driver unlinking the urb, the content of the urb is
++ * not important, but the calling to its completion handler is important; the
++ * completion of unlinking is notified by the completion handler.
++ *
++ *
++ * CLIENT SIDE
++ *
++ * - When vhci_hcd receives RET_SUBMIT,
++ *
++ *    - case 1a). the urb of the pdu is not unlinking.
++ *            - normal case
++ *            => just give back the urb
++ *
++ *    - case 1b). the urb of the pdu is unlinking.
++ *            - usbip.ko will return a reply of the unlinking request.
++ *            => give back the urb now and go to case 2b).
++ *
++ * - When vhci_hcd receives RET_UNLINK,
++ *
++ *    - case 2a). a submit request is still pending in vhci_hcd.
++ *            - urb was really pending in usbip.ko and urb_unlink_urb() was
++ *              completed there.
++ *            => free a pending submit request
++ *            => notify unlink completeness by giving back the urb
++ *
++ *    - case 2b). a submit request is *not* pending in vhci_hcd.
++ *            - urb was already given back to the core driver.
++ *            => do not give back the urb
++ *
++ *
++ * SERVER SIDE
++ *
++ * - When usbip receives CMD_UNLINK,
++ *
++ *    - case 3a). the urb of the unlink request is now in submission.
++ *            => do usb_unlink_urb().
++ *            => after the unlink is completed, send RET_UNLINK.
++ *
++ *    - case 3b). the urb of the unlink request is not in submission.
++ *            - may be already completed or never be received
++ *            => send RET_UNLINK
++ *
++ */
++static int vhci_urb_dequeue(struct usb_hcd *hcd, struct urb *urb, int status)
++{
++      unsigned long flags;
++      struct vhci_priv *priv;
++      struct vhci_device *vdev;
++
++      uinfo("vhci_hcd: dequeue a urb %p\n", urb);
++
++
++      spin_lock_irqsave(&the_controller->lock, flags);
++
++      priv = urb->hcpriv;
++      if (!priv) {
++              /* URB was never linked! or will be soon given back by
++               * vhci_rx. */
++              spin_unlock_irqrestore(&the_controller->lock, flags);
++              return 0;
++      }
++
++      {
++              int ret = 0;
++              ret = usb_hcd_check_unlink_urb(hcd, urb, status);
++              if (ret) {
++                      spin_unlock_irqrestore(&the_controller->lock, flags);
++                      return 0;
++              }
++      }
++
++       /* send unlink request here? */
++      vdev = priv->vdev;
++
++      if (!vdev->ud.tcp_socket) {
++              /* tcp connection is closed */
++              unsigned long flags2;
++
++              spin_lock_irqsave(&vdev->priv_lock, flags2);
++
++              uinfo("vhci_hcd: device %p seems to be disconnected\n", vdev);
++              list_del(&priv->list);
++              kfree(priv);
++              urb->hcpriv = NULL;
++
++              spin_unlock_irqrestore(&vdev->priv_lock, flags2);
++
++      } else {
++              /* tcp connection is alive */
++              unsigned long flags2;
++              struct vhci_unlink *unlink;
++
++              spin_lock_irqsave(&vdev->priv_lock, flags2);
++
++              /* setup CMD_UNLINK pdu */
++              unlink = kzalloc(sizeof(struct vhci_unlink), GFP_ATOMIC);
++              if (!unlink) {
++                      uerr("malloc vhci_unlink\n");
++                      spin_unlock_irqrestore(&vdev->priv_lock, flags2);
++                      spin_unlock_irqrestore(&the_controller->lock, flags);
++                      usbip_event_add(&vdev->ud, VDEV_EVENT_ERROR_MALLOC);
++                      return -ENOMEM;
++              }
++
++              unlink->seqnum = atomic_inc_return(&the_controller->seqnum);
++              if (unlink->seqnum == 0xffff)
++                      uinfo("seqnum max\n");
++
++              unlink->unlink_seqnum = priv->seqnum;
++
++              uinfo("vhci_hcd: device %p seems to be still connected\n",
++                                                                      vdev);
++
++              /* send cmd_unlink and try to cancel the pending URB in the
++               * peer */
++              list_add_tail(&unlink->list, &vdev->unlink_tx);
++              wake_up(&vdev->waitq_tx);
++
++              spin_unlock_irqrestore(&vdev->priv_lock, flags2);
++      }
++
++
++      /*
++       * If tcp connection is alive, we have sent CMD_UNLINK.
++       * vhci_rx will receive RET_UNLINK and give back the URB.
++       * Otherwise, we give back it here.
++       */
++      if (!vdev->ud.tcp_socket) {
++              /* tcp connection is closed */
++              uinfo("vhci_hcd: vhci_urb_dequeue() gives back urb %p\n", urb);
++
++              usb_hcd_unlink_urb_from_ep(hcd, urb);
++
++              spin_unlock_irqrestore(&the_controller->lock, flags);
++              usb_hcd_giveback_urb(vhci_to_hcd(the_controller), urb,
++                                                              urb->status);
++              spin_lock_irqsave(&the_controller->lock, flags);
++      }
++
++      spin_unlock_irqrestore(&the_controller->lock, flags);
++
++      dbg_vhci_hc("leave\n");
++      return 0;
++}
++
++
++static void vhci_device_unlink_cleanup(struct vhci_device *vdev)
++{
++      struct vhci_unlink *unlink, *tmp;
++
++      spin_lock(&vdev->priv_lock);
++
++      list_for_each_entry_safe(unlink, tmp, &vdev->unlink_tx, list) {
++              list_del(&unlink->list);
++              kfree(unlink);
++      }
++
++      list_for_each_entry_safe(unlink, tmp, &vdev->unlink_rx, list) {
++              list_del(&unlink->list);
++              kfree(unlink);
++      }
++
++      spin_unlock(&vdev->priv_lock);
++}
++
++/*
++ * The important thing is that only one context begins cleanup.
++ * This is why error handling and cleanup become simple.
++ * We do not want to consider race condition as possible.
++ */
++static void vhci_shutdown_connection(struct usbip_device *ud)
++{
++      struct vhci_device *vdev = container_of(ud, struct vhci_device, ud);
++
++      /* need this? see stub_dev.c */
++      if (ud->tcp_socket) {
++              udbg("shutdown tcp_socket %p\n", ud->tcp_socket);
++              kernel_sock_shutdown(ud->tcp_socket, SHUT_RDWR);
++      }
++
++      usbip_stop_threads(&vdev->ud);
++      uinfo("stop threads\n");
++
++      /* active connection is closed */
++      if (vdev->ud.tcp_socket != NULL) {
++              sock_release(vdev->ud.tcp_socket);
++              vdev->ud.tcp_socket = NULL;
++      }
++      uinfo("release socket\n");
++
++      vhci_device_unlink_cleanup(vdev);
++
++      /*
++       * rh_port_disconnect() is a trigger of ...
++       *   usb_disable_device():
++       *      disable all the endpoints for a USB device.
++       *   usb_disable_endpoint():
++       *      disable endpoints. pending urbs are unlinked(dequeued).
++       *
++       * NOTE: After calling rh_port_disconnect(), the USB device drivers of a
++       * deteched device should release used urbs in a cleanup function(i.e.
++       * xxx_disconnect()). Therefore, vhci_hcd does not need to release
++       * pushed urbs and their private data in this function.
++       *
++       * NOTE: vhci_dequeue() must be considered carefully. When shutdowning
++       * a connection, vhci_shutdown_connection() expects vhci_dequeue()
++       * gives back pushed urbs and frees their private data by request of
++       * the cleanup function of a USB driver. When unlinking a urb with an
++       * active connection, vhci_dequeue() does not give back the urb which
++       * is actually given back by vhci_rx after receiving its return pdu.
++       *
++       */
++      rh_port_disconnect(vdev->rhport);
++
++      uinfo("disconnect device\n");
++}
++
++
++static void vhci_device_reset(struct usbip_device *ud)
++{
++      struct vhci_device *vdev = container_of(ud, struct vhci_device, ud);
++
++      spin_lock(&ud->lock);
++
++      vdev->speed  = 0;
++      vdev->devid  = 0;
++
++      ud->tcp_socket = NULL;
++
++      ud->status = VDEV_ST_NULL;
++
++      spin_unlock(&ud->lock);
++}
++
++static void vhci_device_unusable(struct usbip_device *ud)
++{
++      spin_lock(&ud->lock);
++
++      ud->status = VDEV_ST_ERROR;
++
++      spin_unlock(&ud->lock);
++}
++
++static void vhci_device_init(struct vhci_device *vdev)
++{
++      memset(vdev, 0, sizeof(*vdev));
++
++      usbip_task_init(&vdev->ud.tcp_rx, "vhci_rx", vhci_rx_loop);
++      usbip_task_init(&vdev->ud.tcp_tx, "vhci_tx", vhci_tx_loop);
++
++      vdev->ud.side   = USBIP_VHCI;
++      vdev->ud.status = VDEV_ST_NULL;
++      /* vdev->ud.lock   = SPIN_LOCK_UNLOCKED; */
++      spin_lock_init(&vdev->ud.lock);
++
++      INIT_LIST_HEAD(&vdev->priv_rx);
++      INIT_LIST_HEAD(&vdev->priv_tx);
++      INIT_LIST_HEAD(&vdev->unlink_tx);
++      INIT_LIST_HEAD(&vdev->unlink_rx);
++      /* vdev->priv_lock = SPIN_LOCK_UNLOCKED; */
++      spin_lock_init(&vdev->priv_lock);
++
++      init_waitqueue_head(&vdev->waitq_tx);
++
++      vdev->ud.eh_ops.shutdown = vhci_shutdown_connection;
++      vdev->ud.eh_ops.reset = vhci_device_reset;
++      vdev->ud.eh_ops.unusable = vhci_device_unusable;
++
++      usbip_start_eh(&vdev->ud);
++}
++
++
++/*----------------------------------------------------------------------*/
++
++static int vhci_start(struct usb_hcd *hcd)
++{
++      struct vhci_hcd *vhci = hcd_to_vhci(hcd);
++      int rhport;
++      int err = 0;
++
++      dbg_vhci_hc("enter vhci_start\n");
++
++
++      /* initialize private data of usb_hcd */
++
++      for (rhport = 0; rhport < VHCI_NPORTS; rhport++) {
++              struct vhci_device *vdev = &vhci->vdev[rhport];
++              vhci_device_init(vdev);
++              vdev->rhport = rhport;
++      }
++
++      atomic_set(&vhci->seqnum, 0);
++      spin_lock_init(&vhci->lock);
++
++
++
++      hcd->power_budget = 0; /* no limit */
++      hcd->state  = HC_STATE_RUNNING;
++      hcd->uses_new_polling = 1;
++
++
++      /* vhci_hcd is now ready to be controlled through sysfs */
++      err = sysfs_create_group(&vhci_dev(vhci)->kobj, &dev_attr_group);
++      if (err) {
++              uerr("create sysfs files\n");
++              return err;
++      }
++
++      return 0;
++}
++
++static void vhci_stop(struct usb_hcd *hcd)
++{
++      struct vhci_hcd *vhci = hcd_to_vhci(hcd);
++      int rhport = 0;
++
++      dbg_vhci_hc("stop VHCI controller\n");
++
++
++      /* 1. remove the userland interface of vhci_hcd */
++      sysfs_remove_group(&vhci_dev(vhci)->kobj, &dev_attr_group);
++
++      /* 2. shutdown all the ports of vhci_hcd */
++      for (rhport = 0 ; rhport < VHCI_NPORTS; rhport++) {
++              struct vhci_device *vdev = &vhci->vdev[rhport];
++
++              usbip_event_add(&vdev->ud, VDEV_EVENT_REMOVED);
++              usbip_stop_eh(&vdev->ud);
++      }
++
++
++      uinfo("vhci_stop done\n");
++}
++
++/*----------------------------------------------------------------------*/
++
++static int vhci_get_frame_number(struct usb_hcd *hcd)
++{
++      uerr("Not yet implemented\n");
++      return 0;
++}
++
++
++#ifdef CONFIG_PM
++
++/* FIXME: suspend/resume */
++static int vhci_bus_suspend(struct usb_hcd *hcd)
++{
++      struct vhci_hcd *vhci = hcd_to_vhci(hcd);
++
++      dev_dbg(&hcd->self.root_hub->dev, "%s\n", __func__);
++
++      spin_lock_irq(&vhci->lock);
++      /* vhci->rh_state = DUMMY_RH_SUSPENDED;
++       * set_link_state(vhci); */
++      hcd->state = HC_STATE_SUSPENDED;
++      spin_unlock_irq(&vhci->lock);
++
++      return 0;
++}
++
++static int vhci_bus_resume(struct usb_hcd *hcd)
++{
++      struct vhci_hcd *vhci = hcd_to_vhci(hcd);
++      int rc = 0;
++
++      dev_dbg(&hcd->self.root_hub->dev, "%s\n", __func__);
++
++      spin_lock_irq(&vhci->lock);
++      if (!test_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags)) {
++              rc = -ESHUTDOWN;
++      } else {
++              /* vhci->rh_state = DUMMY_RH_RUNNING;
++               * set_link_state(vhci);
++               * if (!list_empty(&vhci->urbp_list))
++               *      mod_timer(&vhci->timer, jiffies); */
++              hcd->state = HC_STATE_RUNNING;
++      }
++      spin_unlock_irq(&vhci->lock);
++      return rc;
++
++      return 0;
++}
++
++#else
++
++#define vhci_bus_suspend      NULL
++#define vhci_bus_resume       NULL
++#endif
++
++
++
++static struct hc_driver vhci_hc_driver = {
++      .description    = driver_name,
++      .product_desc   = driver_desc,
++      .hcd_priv_size  = sizeof(struct vhci_hcd),
++
++      .flags          = HCD_USB2,
++
++      .start          = vhci_start,
++      .stop           = vhci_stop,
++
++      .urb_enqueue    = vhci_urb_enqueue,
++      .urb_dequeue    = vhci_urb_dequeue,
++
++      .get_frame_number = vhci_get_frame_number,
++
++      .hub_status_data = vhci_hub_status,
++      .hub_control    = vhci_hub_control,
++      .bus_suspend    = vhci_bus_suspend,
++      .bus_resume     = vhci_bus_resume,
++};
++
++static int vhci_hcd_probe(struct platform_device *pdev)
++{
++      struct usb_hcd          *hcd;
++      int                     ret;
++
++      uinfo("proving...\n");
++
++      dbg_vhci_hc("name %s id %d\n", pdev->name, pdev->id);
++
++      /* will be removed */
++      if (pdev->dev.dma_mask) {
++              dev_info(&pdev->dev, "vhci_hcd DMA not supported\n");
++              return -EINVAL;
++      }
++
++      /*
++       * Allocate and initialize hcd.
++       * Our private data is also allocated automatically.
++       */
++      hcd = usb_create_hcd(&vhci_hc_driver, &pdev->dev, pdev->dev.bus_id);
++      if (!hcd) {
++              uerr("create hcd failed\n");
++              return -ENOMEM;
++      }
++
++
++      /* this is private data for vhci_hcd */
++      the_controller = hcd_to_vhci(hcd);
++
++      /*
++       * Finish generic HCD structure initialization and register.
++       * Call the driver's reset() and start() routines.
++       */
++      ret = usb_add_hcd(hcd, 0, 0);
++      if (ret != 0) {
++              uerr("usb_add_hcd failed %d\n", ret);
++              usb_put_hcd(hcd);
++              the_controller = NULL;
++              return ret;
++      }
++
++
++      dbg_vhci_hc("bye\n");
++      return 0;
++}
++
++
++static int vhci_hcd_remove(struct platform_device *pdev)
++{
++      struct usb_hcd  *hcd;
++
++      hcd = platform_get_drvdata(pdev);
++      if (!hcd)
++              return 0;
++
++      /*
++       * Disconnects the root hub,
++       * then reverses the effects of usb_add_hcd(),
++       * invoking the HCD's stop() methods.
++       */
++      usb_remove_hcd(hcd);
++      usb_put_hcd(hcd);
++      the_controller = NULL;
++
++
++      return 0;
++}
++
++
++
++#ifdef CONFIG_PM
++
++/* what should happen for USB/IP under suspend/resume? */
++static int vhci_hcd_suspend(struct platform_device *pdev, pm_message_t state)
++{
++      struct usb_hcd *hcd;
++      int rhport = 0;
++      int connected = 0;
++      int ret = 0;
++
++      dev_dbg(&pdev->dev, "%s\n", __func__);
++
++      hcd = platform_get_drvdata(pdev);
++
++      spin_lock(&the_controller->lock);
++
++      for (rhport = 0; rhport < VHCI_NPORTS; rhport++)
++              if (the_controller->port_status[rhport] &
++                                              USB_PORT_STAT_CONNECTION)
++                      connected += 1;
++
++      spin_unlock(&the_controller->lock);
++
++      if (connected > 0) {
++              uinfo("We have %d active connection%s. Do not suspend.\n",
++                              connected, (connected == 1 ? "" : "s"));
++              ret =  -EBUSY;
++      } else {
++              uinfo("suspend vhci_hcd");
++              clear_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags);
++      }
++
++      return ret;
++}
++
++static int vhci_hcd_resume(struct platform_device *pdev)
++{
++      struct usb_hcd *hcd;
++
++      dev_dbg(&pdev->dev, "%s\n", __func__);
++
++      hcd = platform_get_drvdata(pdev);
++      set_bit(HCD_FLAG_HW_ACCESSIBLE, &hcd->flags);
++      usb_hcd_poll_rh_status(hcd);
++
++      return 0;
++}
++
++#else
++
++#define vhci_hcd_suspend      NULL
++#define vhci_hcd_resume               NULL
++
++#endif
++
++
++static struct platform_driver vhci_driver = {
++      .probe  = vhci_hcd_probe,
++      .remove = __devexit_p(vhci_hcd_remove),
++      .suspend = vhci_hcd_suspend,
++      .resume = vhci_hcd_resume,
++      .driver = {
++              .name = (char *) driver_name,
++              .owner = THIS_MODULE,
++      },
++};
++
++/*----------------------------------------------------------------------*/
++
++/*
++ * The VHCI 'device' is 'virtual'; not a real plug&play hardware.
++ * We need to add this virtual device as a platform device arbitrarily:
++ *    1. platform_device_register()
++ */
++static void the_pdev_release(struct device *dev)
++{
++      return;
++}
++
++static struct platform_device the_pdev = {
++      /* should be the same name as driver_name */
++      .name = (char *) driver_name,
++      .id = -1,
++      .dev = {
++              /* .driver = &vhci_driver, */
++              .release = the_pdev_release,
++      },
++};
++
++static int __init vhci_init(void)
++{
++      int ret;
++
++      dbg_vhci_hc("enter\n");
++      if (usb_disabled())
++              return -ENODEV;
++
++      printk(KERN_INFO KBUILD_MODNAME ": %s, %s\n", driver_name,
++             DRIVER_VERSION);
++
++      ret = platform_driver_register(&vhci_driver);
++      if (ret < 0)
++              goto err_driver_register;
++
++      ret = platform_device_register(&the_pdev);
++      if (ret < 0)
++              goto err_platform_device_register;
++
++      dbg_vhci_hc("bye\n");
++      return ret;
++
++      /* error occurred */
++err_platform_device_register:
++      platform_driver_unregister(&vhci_driver);
++
++err_driver_register:
++      dbg_vhci_hc("bye\n");
++      return ret;
++}
++module_init(vhci_init);
++
++static void __exit vhci_cleanup(void)
++{
++      dbg_vhci_hc("enter\n");
++
++      platform_device_unregister(&the_pdev);
++      platform_driver_unregister(&vhci_driver);
++
++      dbg_vhci_hc("bye\n");
++}
++module_exit(vhci_cleanup);
+diff --git a/drivers/staging/usbip/vhci_rx.c b/drivers/staging/usbip/vhci_rx.c
+new file mode 100644
+index 0000000..933ccaf
+--- /dev/null
++++ b/drivers/staging/usbip/vhci_rx.c
+@@ -0,0 +1,251 @@
++/*
++ * Copyright (C) 2003-2008 Takahiro Hirofuchi
++ *
++ * This is free software; you can redistribute it and/or modify
++ * it under the terms of the GNU General Public License as published by
++ * the Free Software Foundation; either version 2 of the License, or
++ * (at your option) any later version.
++ *
++ * This is distributed in the hope that it will be useful,
++ * but WITHOUT ANY WARRANTY; without even the implied warranty of
++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
++ * GNU General Public License for more details.
++ *
++ * You should have received a copy of the GNU General Public License
++ * along with this program; if not, write to the Free Software
++ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
++ * USA.
++ */
++
++#include "usbip_common.h"
++#include "vhci.h"
++
++
++/* get URB from transmitted urb queue */
++static struct urb *pickup_urb_and_free_priv(struct vhci_device *vdev,
++                                          __u32 seqnum)
++{
++      struct vhci_priv *priv, *tmp;
++      struct urb *urb = NULL;
++      int status;
++
++      spin_lock(&vdev->priv_lock);
++
++      list_for_each_entry_safe(priv, tmp, &vdev->priv_rx, list) {
++              if (priv->seqnum == seqnum) {
++                      urb = priv->urb;
++                      status = urb->status;
++
++                      dbg_vhci_rx("find urb %p vurb %p seqnum %u\n",
++                                  urb, priv, seqnum);
++
++                      /* TODO: fix logic here to improve indent situtation */
++                      if (status != -EINPROGRESS) {
++                              if (status == -ENOENT ||
++                                   status == -ECONNRESET)
++                                      dev_info(&urb->dev->dev,
++                                               "urb %p was unlinked "
++                                               "%ssynchronuously.\n", urb,
++                                               status == -ENOENT ? "" : "a");
++                              else
++                                      dev_info(&urb->dev->dev,
++                                               "urb %p may be in a error, "
++                                               "status %d\n", urb, status);
++                      }
++
++                      list_del(&priv->list);
++                      kfree(priv);
++                      urb->hcpriv = NULL;
++
++                      break;
++              }
++      }
++
++      spin_unlock(&vdev->priv_lock);
++
++      return urb;
++}
++
++static void vhci_recv_ret_submit(struct vhci_device *vdev,
++                                              struct usbip_header *pdu)
++{
++      struct usbip_device *ud = &vdev->ud;
++      struct urb *urb;
++
++
++      urb = pickup_urb_and_free_priv(vdev, pdu->base.seqnum);
++
++
++      if (!urb) {
++              uerr("cannot find a urb of seqnum %u\n", pdu->base.seqnum);
++              uinfo("max seqnum %d\n", atomic_read(&the_controller->seqnum));
++              usbip_event_add(ud, VDEV_EVENT_ERROR_TCP);
++              return;
++      }
++
++
++      /* unpack the pdu to a urb */
++      usbip_pack_pdu(pdu, urb, USBIP_RET_SUBMIT, 0);
++
++
++      /* recv transfer buffer */
++      if (usbip_recv_xbuff(ud, urb) < 0)
++              return;
++
++
++      /* recv iso_packet_descriptor */
++      if (usbip_recv_iso(ud, urb) < 0)
++              return;
++
++
++      if (dbg_flag_vhci_rx)
++              usbip_dump_urb(urb);
++
++
++      dbg_vhci_rx("now giveback urb %p\n", urb);
++
++      spin_lock(&the_controller->lock);
++      usb_hcd_unlink_urb_from_ep(vhci_to_hcd(the_controller), urb);
++      spin_unlock(&the_controller->lock);
++
++      usb_hcd_giveback_urb(vhci_to_hcd(the_controller), urb, urb->status);
++
++
++      dbg_vhci_rx("Leave\n");
++
++      return;
++}
++
++
++static struct vhci_unlink *dequeue_pending_unlink(struct vhci_device *vdev,
++              struct usbip_header *pdu)
++{
++      struct vhci_unlink *unlink, *tmp;
++
++      spin_lock(&vdev->priv_lock);
++
++      list_for_each_entry_safe(unlink, tmp, &vdev->unlink_rx, list) {
++              uinfo("unlink->seqnum %lu\n", unlink->seqnum);
++              if (unlink->seqnum == pdu->base.seqnum) {
++                      dbg_vhci_rx("found pending unlink, %lu\n",
++                                                      unlink->seqnum);
++                      list_del(&unlink->list);
++
++                      spin_unlock(&vdev->priv_lock);
++                      return unlink;
++              }
++      }
++
++      spin_unlock(&vdev->priv_lock);
++
++      return NULL;
++}
++
++
++static void vhci_recv_ret_unlink(struct vhci_device *vdev,
++                                              struct usbip_header *pdu)
++{
++      struct vhci_unlink *unlink;
++      struct urb *urb;
++
++      usbip_dump_header(pdu);
++
++      unlink = dequeue_pending_unlink(vdev, pdu);
++      if (!unlink) {
++              uinfo("cannot find the pending unlink %u\n", pdu->base.seqnum);
++              return;
++      }
++
++      urb = pickup_urb_and_free_priv(vdev, unlink->unlink_seqnum);
++      if (!urb) {
++              /*
++               * I get the result of a unlink request. But, it seems that I
++               * already received the result of its submit result and gave
++               * back the URB.
++               */
++              uinfo("the urb (seqnum %d) was already given backed\n",
++                                                      pdu->base.seqnum);
++      } else {
++              dbg_vhci_rx("now giveback urb %p\n", urb);
++
++              /* If unlink is succeed, status is -ECONNRESET */
++              urb->status = pdu->u.ret_unlink.status;
++              uinfo("%d\n", urb->status);
++
++              spin_lock(&the_controller->lock);
++              usb_hcd_unlink_urb_from_ep(vhci_to_hcd(the_controller), urb);
++              spin_unlock(&the_controller->lock);
++
++              usb_hcd_giveback_urb(vhci_to_hcd(the_controller), urb,
++                                                              urb->status);
++      }
++
++      kfree(unlink);
++
++      return;
++}
++
++/* recv a pdu */
++static void vhci_rx_pdu(struct usbip_device *ud)
++{
++      int ret;
++      struct usbip_header pdu;
++      struct vhci_device *vdev = container_of(ud, struct vhci_device, ud);
++
++
++      dbg_vhci_rx("Enter\n");
++
++      memset(&pdu, 0, sizeof(pdu));
++
++
++      /* 1. receive a pdu header */
++      ret = usbip_xmit(0, ud->tcp_socket, (char *) &pdu, sizeof(pdu), 0);
++      if (ret != sizeof(pdu)) {
++              uerr("receiving pdu failed! size is %d, should be %d\n",
++                              ret, sizeof(pdu));
++              usbip_event_add(ud, VDEV_EVENT_ERROR_TCP);
++              return;
++      }
++
++      usbip_header_correct_endian(&pdu, 0);
++
++      if (dbg_flag_vhci_rx)
++              usbip_dump_header(&pdu);
++
++      switch (pdu.base.command) {
++      case USBIP_RET_SUBMIT:
++              vhci_recv_ret_submit(vdev, &pdu);
++              break;
++      case USBIP_RET_UNLINK:
++              vhci_recv_ret_unlink(vdev, &pdu);
++              break;
++      default:
++              /* NOTREACHED */
++              uerr("unknown pdu %u\n", pdu.base.command);
++              usbip_dump_header(&pdu);
++              usbip_event_add(ud, VDEV_EVENT_ERROR_TCP);
++      }
++}
++
++
++/*-------------------------------------------------------------------------*/
++
++void vhci_rx_loop(struct usbip_task *ut)
++{
++      struct usbip_device *ud = container_of(ut, struct usbip_device, tcp_rx);
++
++
++      while (1) {
++              if (signal_pending(current)) {
++                      dbg_vhci_rx("signal catched!\n");
++                      break;
++              }
++
++
++              if (usbip_event_happend(ud))
++                      break;
++
++              vhci_rx_pdu(ud);
++      }
++}
++
+diff --git a/drivers/staging/usbip/vhci_sysfs.c b/drivers/staging/usbip/vhci_sysfs.c
+new file mode 100644
+index 0000000..24c2851
+--- /dev/null
++++ b/drivers/staging/usbip/vhci_sysfs.c
+@@ -0,0 +1,250 @@
++/*
++ * Copyright (C) 2003-2008 Takahiro Hirofuchi
++ *
++ * This is free software; you can redistribute it and/or modify
++ * it under the terms of the GNU General Public License as published by
++ * the Free Software Foundation; either version 2 of the License, or
++ * (at your option) any later version.
++ *
++ * This is distributed in the hope that it will be useful,
++ * but WITHOUT ANY WARRANTY; without even the implied warranty of
++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
++ * GNU General Public License for more details.
++ *
++ * You should have received a copy of the GNU General Public License
++ * along with this program; if not, write to the Free Software
++ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
++ * USA.
++ */
++
++#include "usbip_common.h"
++#include "vhci.h"
++
++#include <linux/in.h>
++
++/* TODO: refine locking ?*/
++
++/* Sysfs entry to show port status */
++static ssize_t show_status(struct device *dev, struct device_attribute *attr,
++                         char *out)
++{
++      char *s = out;
++      int i = 0;
++
++      if (!the_controller || !out)
++              BUG();
++
++      spin_lock(&the_controller->lock);
++
++      /*
++       * output example:
++       * prt sta spd dev socket           local_busid
++       * 000 004 000 000         c5a7bb80 1-2.3
++       * 001 004 000 000         d8cee980 2-3.4
++       *
++       * IP address can be retrieved from a socket pointer address by looking
++       * up /proc/net/{tcp,tcp6}. Also, a userland program may remember a
++       * port number and its peer IP address.
++       */
++      out += sprintf(out, "prt sta spd bus dev socket           "
++                     "local_busid\n");
++
++      for (i = 0; i < VHCI_NPORTS; i++) {
++              struct vhci_device *vdev = port_to_vdev(i);
++
++              spin_lock(&vdev->ud.lock);
++
++              out += sprintf(out, "%03u %03u ", i, vdev->ud.status);
++
++              if (vdev->ud.status == VDEV_ST_USED) {
++                      out += sprintf(out, "%03u %08x ",
++                                      vdev->speed, vdev->devid);
++                      out += sprintf(out, "%16p ", vdev->ud.tcp_socket);
++                      out += sprintf(out, "%s", vdev->udev->dev.bus_id);
++
++              } else
++                      out += sprintf(out, "000 000 000 0000000000000000 0-0");
++
++              out += sprintf(out, "\n");
++
++              spin_unlock(&vdev->ud.lock);
++      }
++
++      spin_unlock(&the_controller->lock);
++
++      return out - s;
++}
++static DEVICE_ATTR(status, S_IRUGO, show_status, NULL);
++
++/* Sysfs entry to shutdown a virtual connection */
++static int vhci_port_disconnect(__u32 rhport)
++{
++      struct vhci_device *vdev;
++
++      dbg_vhci_sysfs("enter\n");
++
++      /* lock */
++      spin_lock(&the_controller->lock);
++
++      vdev = port_to_vdev(rhport);
++
++      spin_lock(&vdev->ud.lock);
++      if (vdev->ud.status == VDEV_ST_NULL) {
++              uerr("not connected %d\n", vdev->ud.status);
++
++              /* unlock */
++              spin_unlock(&vdev->ud.lock);
++              spin_unlock(&the_controller->lock);
++
++              return -EINVAL;
++      }
++
++      /* unlock */
++      spin_unlock(&vdev->ud.lock);
++      spin_unlock(&the_controller->lock);
++
++      usbip_event_add(&vdev->ud, VDEV_EVENT_DOWN);
++
++      return 0;
++}
++
++static ssize_t store_detach(struct device *dev, struct device_attribute *attr,
++                          const char *buf, size_t count)
++{
++      int err;
++      __u32 rhport = 0;
++
++      sscanf(buf, "%u", &rhport);
++
++      /* check rhport */
++      if (rhport >= VHCI_NPORTS) {
++              uerr("invalid port %u\n", rhport);
++              return -EINVAL;
++      }
++
++      err = vhci_port_disconnect(rhport);
++      if (err < 0)
++              return -EINVAL;
++
++      dbg_vhci_sysfs("Leave\n");
++      return count;
++}
++static DEVICE_ATTR(detach, S_IWUSR, NULL, store_detach);
++
++/* Sysfs entry to establish a virtual connection */
++static int valid_args(__u32 rhport, enum usb_device_speed speed)
++{
++      /* check rhport */
++      if ((rhport < 0) || (rhport >= VHCI_NPORTS)) {
++              uerr("port %u\n", rhport);
++              return -EINVAL;
++      }
++
++      /* check speed */
++      switch (speed) {
++      case USB_SPEED_LOW:
++      case USB_SPEED_FULL:
++      case USB_SPEED_HIGH:
++      case USB_SPEED_VARIABLE:
++              break;
++      default:
++              uerr("speed %d\n", speed);
++              return -EINVAL;
++      }
++
++      return 0;
++}
++
++/*
++ * To start a new USB/IP attachment, a userland program needs to setup a TCP
++ * connection and then write its socket descriptor with remote device
++ * information into this sysfs file.
++ *
++ * A remote device is virtually attached to the root-hub port of @rhport with
++ * @speed. @devid is embedded into a request to specify the remote device in a
++ * server host.
++ *
++ * write() returns 0 on success, else negative errno.
++ */
++static ssize_t store_attach(struct device *dev, struct device_attribute *attr,
++                          const char *buf, size_t count)
++{
++      struct vhci_device *vdev;
++      struct socket *socket;
++      int sockfd = 0;
++      __u32 rhport = 0, devid = 0, speed = 0;
++
++      /*
++       * @rhport: port number of vhci_hcd
++       * @sockfd: socket descriptor of an established TCP connection
++       * @devid: unique device identifier in a remote host
++       * @speed: usb device speed in a remote host
++       */
++      sscanf(buf, "%u %u %u %u", &rhport, &sockfd, &devid, &speed);
++
++      dbg_vhci_sysfs("rhport(%u) sockfd(%u) devid(%u) speed(%u)\n",
++                      rhport, sockfd, devid, speed);
++
++
++      /* check received parameters */
++      if (valid_args(rhport, speed) < 0)
++              return -EINVAL;
++
++      /* check sockfd */
++      socket = sockfd_to_socket(sockfd);
++      if (!socket)
++              return  -EINVAL;
++
++      /* now need lock until setting vdev status as used */
++
++      /* begin a lock */
++      spin_lock(&the_controller->lock);
++
++      vdev = port_to_vdev(rhport);
++
++      spin_lock(&vdev->ud.lock);
++
++      if (vdev->ud.status != VDEV_ST_NULL) {
++              /* end of the lock */
++              spin_unlock(&vdev->ud.lock);
++              spin_unlock(&the_controller->lock);
++
++              uerr("port %d already used\n", rhport);
++              return -EINVAL;
++      }
++
++      uinfo("rhport(%u) sockfd(%d) devid(%u) speed(%u)\n",
++                      rhport, sockfd, devid, speed);
++
++      vdev->devid         = devid;
++      vdev->speed         = speed;
++      vdev->ud.tcp_socket = socket;
++      vdev->ud.status     = VDEV_ST_NOTASSIGNED;
++
++      spin_unlock(&vdev->ud.lock);
++      spin_unlock(&the_controller->lock);
++      /* end the lock */
++
++      /*
++       * this function will sleep, so should be out of the lock. but, it's ok
++       * because we already marked vdev as being used. really?
++       */
++      usbip_start_threads(&vdev->ud);
++
++      rh_port_connect(rhport, speed);
++
++      return count;
++}
++static DEVICE_ATTR(attach, S_IWUSR, NULL, store_attach);
++
++static struct attribute *dev_attrs[] = {
++      &dev_attr_status.attr,
++      &dev_attr_detach.attr,
++      &dev_attr_attach.attr,
++      &dev_attr_usbip_debug.attr,
++      NULL,
++};
++
++struct attribute_group dev_attr_group = {
++      .attrs = dev_attrs,
++};
+diff --git a/drivers/staging/usbip/vhci_tx.c b/drivers/staging/usbip/vhci_tx.c
+new file mode 100644
+index 0000000..1f552a9
+--- /dev/null
++++ b/drivers/staging/usbip/vhci_tx.c
+@@ -0,0 +1,239 @@
++/*
++ * Copyright (C) 2003-2008 Takahiro Hirofuchi
++ *
++ * This is free software; you can redistribute it and/or modify
++ * it under the terms of the GNU General Public License as published by
++ * the Free Software Foundation; either version 2 of the License, or
++ * (at your option) any later version.
++ *
++ * This is distributed in the hope that it will be useful,
++ * but WITHOUT ANY WARRANTY; without even the implied warranty of
++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
++ * GNU General Public License for more details.
++ *
++ * You should have received a copy of the GNU General Public License
++ * along with this program; if not, write to the Free Software
++ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
++ * USA.
++ */
++
++#include "usbip_common.h"
++#include "vhci.h"
++
++
++static void setup_cmd_submit_pdu(struct usbip_header *pdup,  struct urb *urb)
++{
++      struct vhci_priv *priv = ((struct vhci_priv *)urb->hcpriv);
++      struct vhci_device *vdev = priv->vdev;
++
++      dbg_vhci_tx("URB, local devnum %u, remote devid %u\n",
++                              usb_pipedevice(urb->pipe), vdev->devid);
++
++      pdup->base.command = USBIP_CMD_SUBMIT;
++      pdup->base.seqnum  = priv->seqnum;
++      pdup->base.devid   = vdev->devid;
++      if (usb_pipein(urb->pipe))
++              pdup->base.direction = USBIP_DIR_IN;
++      else
++              pdup->base.direction = USBIP_DIR_OUT;
++      pdup->base.ep      = usb_pipeendpoint(urb->pipe);
++
++      usbip_pack_pdu(pdup, urb, USBIP_CMD_SUBMIT, 1);
++
++      if (urb->setup_packet)
++              memcpy(pdup->u.cmd_submit.setup, urb->setup_packet, 8);
++}
++
++static struct vhci_priv *dequeue_from_priv_tx(struct vhci_device *vdev)
++{
++      unsigned long flags;
++      struct vhci_priv *priv, *tmp;
++
++      spin_lock_irqsave(&vdev->priv_lock, flags);
++
++      list_for_each_entry_safe(priv, tmp, &vdev->priv_tx, list) {
++              list_move_tail(&priv->list, &vdev->priv_rx);
++              spin_unlock_irqrestore(&vdev->priv_lock, flags);
++              return priv;
++      }
++
++      spin_unlock_irqrestore(&vdev->priv_lock, flags);
++
++      return NULL;
++}
++
++
++
++static int vhci_send_cmd_submit(struct vhci_device *vdev)
++{
++      struct vhci_priv *priv = NULL;
++
++      struct msghdr msg;
++      struct kvec iov[3];
++      size_t txsize;
++
++      size_t total_size = 0;
++
++      while ((priv = dequeue_from_priv_tx(vdev)) != NULL) {
++              int ret;
++              struct urb *urb = priv->urb;
++              struct usbip_header pdu_header;
++              void *iso_buffer = NULL;
++
++              txsize = 0;
++              memset(&pdu_header, 0, sizeof(pdu_header));
++              memset(&msg, 0, sizeof(msg));
++              memset(&iov, 0, sizeof(iov));
++
++              dbg_vhci_tx("setup txdata urb %p\n", urb);
++
++
++              /* 1. setup usbip_header */
++              setup_cmd_submit_pdu(&pdu_header, urb);
++              usbip_header_correct_endian(&pdu_header, 1);
++
++              iov[0].iov_base = &pdu_header;
++              iov[0].iov_len  = sizeof(pdu_header);
++              txsize += sizeof(pdu_header);
++
++              /* 2. setup transfer buffer */
++              if (!usb_pipein(urb->pipe) && urb->transfer_buffer_length > 0) {
++                      iov[1].iov_base = urb->transfer_buffer;
++                      iov[1].iov_len  = urb->transfer_buffer_length;
++                      txsize += urb->transfer_buffer_length;
++              }
++
++              /* 3. setup iso_packet_descriptor */
++              if (usb_pipetype(urb->pipe) == PIPE_ISOCHRONOUS) {
++                      ssize_t len = 0;
++
++                      iso_buffer = usbip_alloc_iso_desc_pdu(urb, &len);
++                      if (!iso_buffer) {
++                              usbip_event_add(&vdev->ud,
++                                              SDEV_EVENT_ERROR_MALLOC);
++                              return -1;
++                      }
++
++                      iov[2].iov_base = iso_buffer;
++                      iov[2].iov_len  = len;
++                      txsize += len;
++              }
++
++              ret = kernel_sendmsg(vdev->ud.tcp_socket, &msg, iov, 3, txsize);
++              if (ret != txsize) {
++                      uerr("sendmsg failed!, retval %d for %zd\n", ret,
++                                                              txsize);
++                      kfree(iso_buffer);
++                      usbip_event_add(&vdev->ud, VDEV_EVENT_ERROR_TCP);
++                      return -1;
++              }
++
++              kfree(iso_buffer);
++              dbg_vhci_tx("send txdata\n");
++
++              total_size += txsize;
++      }
++
++      return total_size;
++}
++
++
++/*-------------------------------------------------------------------------*/
++
++static struct vhci_unlink *dequeue_from_unlink_tx(struct vhci_device *vdev)
++{
++      unsigned long flags;
++      struct vhci_unlink *unlink, *tmp;
++
++      spin_lock_irqsave(&vdev->priv_lock, flags);
++
++      list_for_each_entry_safe(unlink, tmp, &vdev->unlink_tx, list) {
++              list_move_tail(&unlink->list, &vdev->unlink_rx);
++              spin_unlock_irqrestore(&vdev->priv_lock, flags);
++              return unlink;
++      }
++
++      spin_unlock_irqrestore(&vdev->priv_lock, flags);
++
++      return NULL;
++}
++
++static int vhci_send_cmd_unlink(struct vhci_device *vdev)
++{
++      struct vhci_unlink *unlink = NULL;
++
++      struct msghdr msg;
++      struct kvec iov[3];
++      size_t txsize;
++
++      size_t total_size = 0;
++
++      while ((unlink = dequeue_from_unlink_tx(vdev)) != NULL) {
++              int ret;
++              struct usbip_header pdu_header;
++
++              txsize = 0;
++              memset(&pdu_header, 0, sizeof(pdu_header));
++              memset(&msg, 0, sizeof(msg));
++              memset(&iov, 0, sizeof(iov));
++
++              dbg_vhci_tx("setup cmd unlink, %lu \n", unlink->seqnum);
++
++
++              /* 1. setup usbip_header */
++              pdu_header.base.command = USBIP_CMD_UNLINK;
++              pdu_header.base.seqnum  = unlink->seqnum;
++              pdu_header.base.devid   = vdev->devid;
++              pdu_header.base.ep      = 0;
++              pdu_header.u.cmd_unlink.seqnum = unlink->unlink_seqnum;
++
++              usbip_header_correct_endian(&pdu_header, 1);
++
++              iov[0].iov_base = &pdu_header;
++              iov[0].iov_len  = sizeof(pdu_header);
++              txsize += sizeof(pdu_header);
++
++              ret = kernel_sendmsg(vdev->ud.tcp_socket, &msg, iov, 1, txsize);
++              if (ret != txsize) {
++                      uerr("sendmsg failed!, retval %d for %zd\n", ret,
++                                                              txsize);
++                      usbip_event_add(&vdev->ud, VDEV_EVENT_ERROR_TCP);
++                      return -1;
++              }
++
++
++              dbg_vhci_tx("send txdata\n");
++
++              total_size += txsize;
++      }
++
++      return total_size;
++}
++
++
++/*-------------------------------------------------------------------------*/
++
++void vhci_tx_loop(struct usbip_task *ut)
++{
++      struct usbip_device *ud = container_of(ut, struct usbip_device, tcp_tx);
++      struct vhci_device *vdev = container_of(ud, struct vhci_device, ud);
++
++      while (1) {
++              if (signal_pending(current)) {
++                      uinfo("vhci_tx signal catched\n");
++                      break;
++              }
++
++              if (vhci_send_cmd_submit(vdev) < 0)
++                      break;
++
++              if (vhci_send_cmd_unlink(vdev) < 0)
++                      break;
++
++              wait_event_interruptible(vdev->waitq_tx,
++                              (!list_empty(&vdev->priv_tx) ||
++                               !list_empty(&vdev->unlink_tx)));
++
++              dbg_vhci_tx("pending urbs ?, now wake up\n");
++      }
++}
+-- 
+1.6.0.2
+