From 526b8fec2aff8961453ee40c0d7b19bdb30da0bb Mon Sep 17 00:00:00 2001
From: NIIBE Yutaka <gniibe@fsij.org>
Date: Mon, 26 Jun 2017 19:32:43 +0900
Subject: Added USBIP driver for emulation.

---
 AUTHORS         |    3 +
 ChangeLog       |    8 +
 mcu/usb-usbip.c | 2204 +++++++++++++++++++++++++++++++++++++++++++++++++++++++
 3 files changed, 2215 insertions(+)
 create mode 100644 mcu/usb-usbip.c

diff --git a/AUTHORS b/AUTHORS
index a2ef4d1..e12bd99 100644
--- a/AUTHORS
+++ b/AUTHORS
@@ -42,6 +42,9 @@ NIIBE Yutaka:
 
 	board-fs-bb48.h
 
+    Under contract of g10 Code GmbH, wrote:
+        mcu/usb-usbip.c
+
 Paul Fertser:
     Added Blue Pill support.
 	board/board-blue-pill.h
diff --git a/ChangeLog b/ChangeLog
index cce33a2..a93af2b 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,11 @@
+2017-06-26  NIIBE Yutaka  <gniibe@fsij.org>
+
+	* mcu/usb-usbip.c: New.
+
+	* rules.mk: Support EMULATION.
+
+	* entry.c (main): Support GNU_LINUX_EMULATION.
+
 2017-06-23  NIIBE Yutaka  <gniibe@fsij.org>
 
 	* chopstx-gnu-linux.c, chopstx-gnu-linux.h: New.
diff --git a/mcu/usb-usbip.c b/mcu/usb-usbip.c
new file mode 100644
index 0000000..ad19d43
--- /dev/null
+++ b/mcu/usb-usbip.c
@@ -0,0 +1,2204 @@
+/*
+ * usb-usbip.c - USB Device Emulation (server side) by USBIP
+ *
+ * Copyright (C) 2017 g10 Code GmbH
+ * Author: NIIBE Yutaka <gniibe@fsij.org>
+ *
+ * This file is a part of Chopstx, a thread library for embedded.
+ *
+ * Chopstx 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 3 of the License, or
+ * (at your option) any later version.
+ *
+ * Chopstx 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, see <http://www.gnu.org/licenses/>.
+ *
+ */
+
+/*
+ FIXME:
+   RESET handling
+   USB Shutdown
+   Use reply structure of its own
+ */
+
+#include <pthread.h>
+
+#include <unistd.h>
+
+#include <stdint.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <errno.h>
+
+#include <arpa/inet.h>
+#include <sys/types.h>
+#include <sys/socket.h>
+
+#include <signal.h>
+#include <sys/eventfd.h>
+#include <poll.h>
+
+#include <usb_lld.h>
+#include <usb_lld_common.h>
+
+#include <alloca.h>
+
+static pthread_t tid_main;
+static pthread_t tid_usbip;
+
+#define USBIP_PORT 3240
+
+#define CMD_REQ_LIST   0x01118005
+#define CMD_REQ_ATTACH 0x01118003
+#define CMD_URB_SUBMIT 0x00000001
+#define CMD_URB_UNLINK 0x00000002
+
+#define REP_URB_SUBMIT 0x00000003
+#define REP_URB_UNLINK 0x00000004
+
+struct usbip_msg_head {
+  uint32_t cmd;
+  uint32_t seq;
+};
+
+struct usbip_usb_device {
+  char path[256];
+  char busid[32];
+
+  uint32_t busnum;
+  uint32_t devnum;
+  uint32_t speed;
+
+  uint16_t idVendor;
+  uint16_t idProduct;
+  uint16_t bcdDevice;
+
+  uint8_t bDeviceClass;
+  uint8_t bDeviceSubClass;
+  uint8_t bDeviceProtocol;
+
+  uint8_t bConfigurationValue;
+  uint8_t bNumConfigurations;
+  uint8_t bNumInterfaces;
+} __attribute__((packed));
+
+struct urb {
+  struct urb *next;
+  struct urb *prev;
+
+  uint16_t remain;
+  char *data_p;
+
+  pthread_t tid;
+  uint32_t seq;
+  uint32_t devid;
+  uint32_t dir;
+  uint32_t ep;
+  uint32_t len;
+  uint8_t setup[8];
+  char data[0];
+};
+
+/*
+ * Only support a single device.
+ */
+static struct usbip_usb_device usbip_usb_device;
+
+static struct urb *issue_get_desc (void);
+
+#define MY_BUS_ID "1-1\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000\000"
+
+#define USBIP_DIR_OUT 0
+#define USBIP_DIR_IN  1
+
+static void
+refresh_usb_device (void)
+{
+  struct urb *urb = issue_get_desc ();
+  char *desc = urb->data;
+
+  memset (usbip_usb_device.path, 0, 256);
+  strcpy (usbip_usb_device.path,
+	  "/sys/devices/pci0000:00/0000:00:01.1/usb1/1-1");
+  memcpy (usbip_usb_device.busid, MY_BUS_ID, 32);
+
+  usbip_usb_device.busnum = 0;
+  usbip_usb_device.devnum = 0;
+  usbip_usb_device.speed =  htonl (2); /* Full speed.  */
+
+  /* USB descriptors are little endian.  USBIP is network order.  */
+
+  /* 0: size=18 (or follows more desc) */
+  /* 1: DEVICE_DESCRIPTOR */
+  /* 2: bcdUSB: ignore or use for speed? */
+  /* 4: bDeviceClass */
+  /* 5: bDeviceSubClass */
+  /* 6: bDeviceProtocol */
+  /* 7: bMaxPacketSize: ignore */
+  /* 8: idVendor */
+  /* 10: idProduct */
+  /* 12: bcdDevice */
+  /* 14: iManufacturer: ignore */
+  /* 15: iProduct: ignore */
+  /* 16: iSerialNumber: ignore */
+  /* 17: bNumConfigurations */
+  /* ... */
+  usbip_usb_device.idVendor = htons (((desc[9] << 8)|desc[8]));
+  usbip_usb_device.idProduct = htons (((desc[11] << 8)|desc[10]));
+  usbip_usb_device.bcdDevice = htons (((desc[13] << 8)|desc[12]));
+
+  usbip_usb_device.bDeviceClass = desc[4];
+  usbip_usb_device.bDeviceSubClass = desc[5];
+  usbip_usb_device.bDeviceProtocol = desc[6];
+
+  usbip_usb_device.bConfigurationValue = 0;
+  usbip_usb_device.bNumConfigurations = desc[17];
+  usbip_usb_device.bNumInterfaces = 0;
+  free (urb);
+}
+
+#define USBIP_REPLY_HEADER_SIZE 8
+#define DEVICE_INFO_SIZE        (256+32+12+6+6)
+#define INTERFACE_INFO_SIZE     4
+#define DEVICE_LIST_SIZE        (DEVICE_INFO_SIZE*1)
+
+#define USBIP_REPLY_DEVICE_LIST "\x01\x11\x00\x05"
+#define USBIP_REPLY_ATTACH      "\x01\x11\x00\x03"
+
+#define NETWORK_UINT32_ZERO     "\x00\x00\x00\x00"
+#define NETWORK_UINT32_ONE      "\x00\x00\x00\x01"
+#define NETWORK_UINT32_TWO      "\x00\x00\x00\x02"
+#define NETWORK_UINT16_ZERO      "\x00\x00"
+#define NETWORK_UINT16_ONE_ONE   "\x01\x01"
+
+enum {
+  USB_INTR_NONE = 0,
+  USB_INTR_SETUP,
+  USB_INTR_DATA_TRANSFER,
+  USB_INTR_RESET,
+  USB_INTR_SUSPEND,
+};
+
+struct usb_controller {
+  pthread_mutex_t mutex;
+  pthread_cond_t cond;
+  uint8_t intr;
+  uint8_t dir;
+  uint8_t ep_num;
+};
+
+static struct usb_controller usbc;
+
+static void
+notify_device (uint8_t intr, uint8_t ep_num, uint8_t dir)
+{
+  extern sigset_t ss_cur;
+
+  pthread_mutex_lock (&usbc.mutex);
+  if (usbc.intr)
+    pthread_cond_wait (&usbc.cond, &usbc.mutex);
+  usbc.intr = intr;
+  usbc.dir = (dir == USBIP_DIR_IN);
+  usbc.ep_num = ep_num;
+  fprintf (stderr, "sigmask: %08llx\n", *(long long *)&ss_cur);
+  pthread_kill (tid_main, SIGUSR1);
+  pthread_mutex_unlock (&usbc.mutex);
+}
+
+
+static const char *
+list_devices (size_t *len_p)
+{
+  refresh_usb_device ();
+  *len_p = sizeof (usbip_usb_device);
+  return (const char *)&usbip_usb_device;
+}
+
+static const char *
+attach_device (char busid[32], size_t *len_p)
+{
+  *len_p = 0;
+  if (memcmp (busid, MY_BUS_ID, 32) != 0) 
+    return NULL;
+
+  //  notify_device (USB_INTR_RESET, 0, 0);
+
+  refresh_usb_device ();
+  *len_p = sizeof (usbip_usb_device);
+  return (const char *)&usbip_usb_device;
+}
+
+#define URB_DATA_SIZE 65535
+
+struct usbip_msg_ctl {
+  uint32_t devid;
+  uint32_t dir;
+  uint32_t ep;
+  uint32_t status;
+  uint32_t len;
+  uint32_t rsvd[2];
+  uint32_t err_cnt;
+};
+
+static pthread_mutex_t urb_mutex;
+static struct urb *urb_list;
+
+static uint8_t usb_setup[8];
+
+static int control_setup_transaction (struct urb *urb);
+static int control_write_data_transaction (char *buf, uint16_t count);
+static int control_write_status_transaction (void);
+static int control_read_data_transaction (char *buf, uint16_t count);
+static int control_read_status_transaction (void);
+
+enum {
+  USB_STATE_DISABLED = 0,
+  USB_STATE_STALL,
+  USB_STATE_NAK,
+  USB_STATE_READY,
+};
+
+struct usb_control {
+  pthread_mutex_t mutex;
+  int eventfd;
+
+  /* Device side: state, buf, len */
+  uint8_t state;
+  uint8_t *buf;
+  uint16_t len;
+
+  /* Host controller side: urb */
+  struct urb *urb;
+};
+
+static struct usb_control usbc_ep_in[7];
+static struct usb_control usbc_ep_out[7];
+#define usbc_ep0 usbc_ep_out[0]
+/*
+ * usbc_ep_in[0] not used.
+ */
+
+static int write_data_transaction (struct usb_control *usbc_p,
+				   int ep_num, char *buf, uint16_t count);
+static int read_data_transaction (struct usb_control *usbc_p,
+				  int ep_num, char *buf, uint16_t count);
+
+static int
+hc_handle_control_urb (struct urb *urb)
+{
+  int r;
+  uint16_t count;
+  uint16_t remain = urb->len;
+  uint64_t l;
+
+  puts ("hcu 0");
+
+  usbc_ep0.urb = urb;
+  r = control_setup_transaction (urb);
+  if (r < 0)
+    goto error;
+
+  puts ("hcu 1");
+  if (urb->dir == USBIP_DIR_OUT)
+    {				/* Output from host to device.  */
+      printf ("hcu: %d\n", r);
+      while (r == 0)
+	{
+	  if (remain > 64)
+	    count = 64;
+	  else
+	    count = remain;
+
+	  read (usbc_ep0.eventfd, &l, sizeof (l));
+	  r = control_write_data_transaction (urb->data_p, count);
+	  if (r < 0)
+	    break;
+
+	  urb->data_p += count;
+	  remain -= count;
+	  if (count < 64)
+	    break;
+	}
+      if (r >= 0)
+	{
+	  read (usbc_ep0.eventfd, &l, sizeof (l));
+	  r = control_write_status_transaction ();
+	}
+    }
+  else
+    {				/* Input from device to host.  */
+      puts ("hcu 2");
+      while (1)
+	{
+	  if (remain > 64)
+	    count = 64;
+	  else
+	    count = remain;
+
+	  read (usbc_ep0.eventfd, &l, sizeof (l));
+	  r = control_read_data_transaction (urb->data_p, count);
+	  if (r < 0)
+	    break;
+
+	  puts ("hcu 3");
+	  remain -= r;
+	  urb->data_p += r;
+	  if (r < 64)
+	    break;
+	}
+      puts ("hcu 4");
+      if (r >= 0)
+	{
+	  puts ("hcu 5");
+	  read (usbc_ep0.eventfd, &l, sizeof (l));
+	  r = control_read_status_transaction ();
+	  if (r >= 0)
+	    r = remain;
+	}
+      puts ("hcu 6");
+    }
+
+  if (r < 0)
+    {
+    error:
+      fprintf (stderr, "hcu 7 %d\n", r);
+      /* recovery.  */
+      usbc_ep0.state = USB_STATE_READY;
+    }
+  else
+    /* Wait until the device is ready to accept the SETUP token.  */
+    read (usbc_ep0.eventfd, &l, sizeof (l));
+
+  if (urb->dir == USBIP_DIR_IN)
+    {
+      if (r >= 0)	      /* R>0 means buf remained as unused.  */
+	urb->len -= r;
+      else
+	urb->len = 0;
+      r = 0;
+    }
+  else
+    urb->len = 0;
+
+  printf ("hu-next: %d (%d)\n", urb->len, urb->seq);
+  usbc_ep0.urb = NULL;
+  return r;
+}
+
+static void usbip_finish_urb (struct urb *urb, int r);
+
+static void
+usbip_handle_control_urb (struct urb *urb)
+{
+  int r = 0;
+  r = hc_handle_control_urb (urb);
+  usbip_finish_urb (urb, r);
+}
+
+static int
+usbip_handle_data_urb (struct urb *urb)
+{
+  int r;
+  struct usb_control *usbc_p;
+
+  if (urb->dir == USBIP_DIR_OUT)
+    /* Output from host to device.  */
+    usbc_p = &usbc_ep_out[urb->ep];
+  else
+    /* Input from device to host.  */
+    usbc_p = &usbc_ep_in[urb->ep];
+
+  pthread_mutex_lock (&usbc_p->mutex);
+  if (usbc_p->state == USB_STATE_DISABLED
+      || usbc_p->state == USB_STATE_STALL)
+    r = -EAGAIN;
+  else /* nak or ready */
+    {
+      if (usbc_p->urb == NULL)
+	usbc_p->urb = urb;
+      r = 0;
+    }
+  pthread_mutex_unlock (&usbc_p->mutex);
+  return r;
+}
+
+static int fd;
+static pthread_mutex_t fd_mutex;
+
+static void unlink_urb (struct urb *urb);
+
+static void
+usbip_finish_urb (struct urb *urb, int r)
+{
+  struct usbip_msg_head msg;
+  struct usbip_msg_ctl msg_ctl;
+  const char zeros[8] = { 0, 0, 0, 0, 0, 0, 0, 0 };
+
+  msg.cmd = htonl (REP_URB_SUBMIT);
+  msg.seq = htonl (urb->seq);
+
+  msg_ctl.devid = 0;
+  msg_ctl.dir = 0;
+  msg_ctl.ep = 0;
+  msg_ctl.len = htonl (urb->len);/*???*/
+  msg_ctl.rsvd[0] = msg_ctl.rsvd[1] = 0;
+  msg_ctl.err_cnt = 0;
+
+  fprintf (stderr, "ufu: %d (%d)\n", r, urb->seq);
+
+  if (r < 0)
+    msg_ctl.status = htonl (r);
+  else
+    msg_ctl.status = 0;
+
+  pthread_mutex_lock (&fd_mutex);
+  if ((size_t)send (fd, &msg, sizeof (msg), 0) != sizeof (msg))
+    {
+      perror ("reply send");
+    }
+
+  if ((size_t)send (fd, &msg_ctl, sizeof (msg_ctl), 0) != sizeof (msg_ctl))
+    {
+      perror ("reply send");
+    }
+
+  if ((size_t)send (fd, zeros, sizeof (zeros), 0) != sizeof (zeros))
+    {
+      perror ("reply send");
+    }
+
+  if (urb->dir == USBIP_DIR_IN && urb->len)
+    {
+      if (send (fd, urb->data, urb->len, 0) != urb->len)
+	{
+	  perror ("reply send");
+	}
+    }
+  pthread_mutex_unlock (&fd_mutex);
+
+  unlink_urb (urb);
+  free (urb);
+}
+
+
+static int
+hc_handle_data_urb  (struct usb_control *usbc_p)
+{
+  int r;
+  uint16_t count;
+  struct urb *urb = usbc_p->urb;
+
+  puts ("hc_hdu 0");
+  if (urb->remain > 64)
+    count = 64;
+  else
+    count = urb->remain;
+
+  if (urb->dir == USBIP_DIR_OUT)
+    {				/* Output from host to device.  */
+      puts ("hc_hdu 1");
+      r = write_data_transaction (usbc_p, urb->ep, urb->data_p, count);
+      if (r < 0)
+	return r;
+
+      urb->data_p += count;
+      urb->remain -= count;
+
+      if (count < 64)
+	{
+	  size_t len = urb->len - urb->remain;
+
+	  fprintf (stderr, "->data: %lu\n", len);
+	  // successfully finished
+	  if (len)
+	    {
+	      char *s = alloca (len + 1);
+	      memcpy (s, urb->data, len);
+	      s[len] = 0;
+	      fprintf (stderr, "   : %s\n", s);
+	    }
+	  return 0;
+	}
+
+      return 1;
+    }
+  else
+    {				/* Input from device to host.  */
+      puts ("hc_hdu 2");
+      r = read_data_transaction (usbc_p, urb->ep, urb->data_p, count);
+      if (r < 0)
+	return r;
+
+      urb->remain -= r;
+      urb->data_p += r;
+      if (r < 64)
+	{
+	  size_t len = urb->len - urb->remain;
+
+	  fprintf (stderr, "<-data: %lu\n", len);
+	  // successfully finished
+	  if (len)
+	    {
+	      char *s = alloca (len + 1);
+	      memcpy (s, urb->data, len);
+	      s[len] = 0;
+	      fprintf (stderr, "   : %s\n", s);
+	    }
+	  urb->len = len;
+	  return 0;
+	}
+
+      return 1;
+    }
+}
+
+#define USB_REQ_GET_DESCRIPTOR		0x06
+
+static struct urb *
+issue_get_desc (void)
+{
+  struct urb *urb;
+
+  urb = malloc (sizeof (struct urb) + 64);
+
+  urb->next = urb->prev = urb;
+
+  urb->setup[0] = 0x80;		         /* Type: GET, Standard, DEVICE */
+  urb->setup[1] = USB_REQ_GET_DESCRIPTOR; /* Request */
+  urb->setup[2] = 0;			 /* Value L: desc_index */
+  urb->setup[3] = 1;			 /* Value H: desc_type */
+  urb->setup[4] = 0;              	 /* Index */
+  urb->setup[5] = 0;
+  urb->setup[6] = 64;		         /* Length */
+  urb->setup[7] = 0;
+  urb->data_p = urb->data;
+  urb->seq = 0;
+  urb->devid = 0;
+  urb->dir = USBIP_DIR_IN;
+  urb->ep = 0;
+  urb->remain = urb->len = 64;
+  hc_handle_control_urb (urb);
+  return urb;
+}
+
+
+static void
+unlink_urb (struct urb *urb)
+{
+  if (urb_list == urb)
+    {
+      if (urb->next == urb)
+	urb_list = NULL;
+      else
+	urb_list = urb->next;
+    }
+
+  urb->next->prev = urb->prev;
+  urb->prev->next = urb->next;
+}
+
+static void
+usbip_handle_urb (uint32_t seq)
+{
+  int r = 0;
+  struct usbip_msg_head msg;
+  struct usbip_msg_ctl msg_ctl;
+  struct urb *urb = NULL;
+  const char zeros[8] = { 0, 0, 0, 0, 0, 0, 0, 0 };
+
+  if (recv (fd, (char *)&msg_ctl, sizeof (msg_ctl), 0) != sizeof (msg_ctl))
+    {
+      perror ("msg recv ctl");
+      r = -EINVAL;
+      goto leave;
+    }
+
+  if (ntohl (msg_ctl.len) > URB_DATA_SIZE)
+    {
+      perror ("msg len too long");
+      r = -EINVAL;
+      goto leave;
+    }
+
+  urb = malloc (sizeof (struct urb) + ntohl (msg_ctl.len));
+  if (urb == NULL)
+    {
+      perror ("URB alloc");
+      exit (1);
+    }
+
+  pthread_mutex_lock (&urb_mutex);
+  if (urb_list == NULL)
+    {
+      urb_list = urb;
+      urb->next = urb->prev = urb;
+    }
+  else
+    {
+      urb->next = urb_list;
+      urb->prev = urb_list->prev;
+      urb_list->prev->next = urb;
+      urb_list->prev = urb;
+      urb_list = urb;
+    }
+  pthread_mutex_unlock (&urb_mutex);
+
+  urb->tid = 0;
+  urb->seq = seq;
+  urb->devid = ntohl (msg_ctl.devid);
+  urb->dir = ntohl (msg_ctl.dir);
+  urb->ep = ntohl (msg_ctl.ep);
+  urb->remain = urb->len = ntohl (msg_ctl.len);
+  urb->data_p = urb->data;
+
+  printf ("URB: dir=%s, ep=%d, len=%d\n", urb->dir==USBIP_DIR_IN? "IN": "OUT",
+	  urb->ep, urb->len);
+
+  if (recv (fd, (char *)urb->setup, sizeof (urb->setup), 0) != sizeof (urb->setup))
+    {
+      perror ("msg recv setup");
+      r = -EINVAL;
+      goto leave;
+    }
+
+  if (urb->ep == 0)
+    printf ("URB: %02x %02x %02x %02x %02x %02x %02x %02x\n",
+	    urb->setup[0], urb->setup[1], urb->setup[2], urb->setup[3],
+	    urb->setup[4], urb->setup[5], urb->setup[6], urb->setup[7]);
+
+  if (urb->dir == USBIP_DIR_OUT && urb->len)
+    {
+      if (recv (fd, urb->data, urb->len, 0) != urb->len)
+	{
+	  perror ("msg recv data");
+	  r = -EINVAL;
+	  goto leave;
+	}
+    }
+
+  if (urb->ep == 0)
+    {
+      usbip_handle_control_urb (urb);
+      return;
+    }
+  else
+    {
+      r = usbip_handle_data_urb (urb);
+      if (r == 0)
+	return;
+    }
+
+  r = -r;
+
+ leave:
+  msg.cmd = htonl (REP_URB_SUBMIT);
+  msg.seq = htonl (urb->seq);
+
+  msg_ctl.devid = 0;
+  msg_ctl.dir = 0;
+  msg_ctl.ep = 0;
+  msg_ctl.status = htonl (r);
+  msg_ctl.len = 0;
+  msg_ctl.rsvd[0] = msg_ctl.rsvd[1] = 0;
+  msg_ctl.err_cnt = 0;
+  
+  pthread_mutex_lock (&fd_mutex);
+  if ((size_t)send (fd, &msg, sizeof (msg), 0) != sizeof (msg))
+    {
+      perror ("reply send");
+    }
+
+  if ((size_t)send (fd, &msg_ctl, sizeof (msg_ctl), 0) != sizeof (msg_ctl))
+    {
+      perror ("reply send");
+    }
+
+  if ((size_t)send (fd, zeros, sizeof (zeros), 0) != sizeof (zeros))
+    {
+      perror ("reply send");
+    }
+
+  pthread_mutex_unlock (&fd_mutex);
+  if (urb)
+    {
+      pthread_mutex_lock (&urb_mutex);
+      unlink_urb (urb);
+      pthread_mutex_unlock (&urb_mutex);
+      free (urb);
+    }
+}
+
+
+static void
+usbip_send_reply (char *reply, int ok)
+{
+  char buf[8];
+  char *p = buf;
+
+  memcpy (p, reply, 4);
+  p += 4;
+  if (ok)
+    memcpy (p, NETWORK_UINT32_ZERO, 4);
+  else
+    memcpy (p, NETWORK_UINT32_ONE, 4);
+  p += 4;
+
+  if ((size_t)send (fd, buf, 8, 0) != 8)
+    {
+      perror ("reply send");
+    }
+}
+
+static int attached = 0;
+
+
+static void
+unlink_urb_ep (struct urb *urb)
+{
+  struct usb_control *usbc_p;
+
+  if (urb->dir == USBIP_DIR_OUT)
+    usbc_p = &usbc_ep_out[urb->ep];
+  else
+    usbc_p = &usbc_ep_in[urb->ep];
+
+  pthread_mutex_lock (&usbc_p->mutex);
+  if (usbc_p->urb == urb)
+    usbc_p->urb = NULL;
+  pthread_mutex_unlock (&usbc_p->mutex);
+/* FIXME: rescan the list and register??? */
+}
+
+static int
+usbip_process_cmd (void)
+{
+  struct usbip_msg_head msg;
+
+  if (recv (fd, (char *)&msg, sizeof (msg), 0) != sizeof (msg))
+    {
+      perror ("msg recv");
+      return -1;
+    }
+
+  msg.cmd = ntohl (msg.cmd);
+  msg.seq = ntohl (msg.seq);
+
+  if (msg.cmd == CMD_REQ_LIST)
+    {
+      const char *device_list;
+      size_t device_list_size;
+
+      printf ("Device List\n");
+      if (attached)
+	{
+	  fprintf (stderr, "REQ list while attached\n");
+	  return -1;
+	}
+
+      device_list = list_devices (&device_list_size);
+
+      pthread_mutex_lock (&fd_mutex);
+      usbip_send_reply (USBIP_REPLY_DEVICE_LIST, !!device_list);
+
+      if (send (fd, NETWORK_UINT32_ONE, 4, 0) == 4
+	  && (size_t)send (fd, device_list, device_list_size, 0) == device_list_size)
+	pthread_mutex_unlock (&fd_mutex);
+      else
+	{
+	  pthread_mutex_unlock (&fd_mutex);
+	  perror ("list send");
+	  return -1;
+	}
+
+      close (fd);
+      return 1;
+    }
+  else if (msg.cmd == CMD_REQ_ATTACH)
+    {
+      const char *attach;
+      size_t attach_size;
+      char busid[32];
+
+      printf ("Attach device\n");
+      if (attached)
+	{
+	  fprintf (stderr, "REQ attach while attached\n");
+	  return -1;
+	}
+	      
+      if (recv (fd, busid, 32, 0) != 32)
+	{
+	  perror ("attach recv");
+	  return -1;
+	}
+
+      attach = attach_device (busid, &attach_size);
+
+      pthread_mutex_lock (&fd_mutex);
+      usbip_send_reply (USBIP_REPLY_ATTACH, !!attach);
+      if (attach
+	  && (size_t)send (fd, attach, attach_size, 0) == attach_size)
+	{
+	  printf ("Attach device!\n");
+	  attached = 1;
+	  pthread_mutex_unlock (&fd_mutex);
+	}
+      else
+	{
+	  pthread_mutex_unlock (&fd_mutex);
+	  perror ("attach send");
+	  return -1;
+	}
+    }
+  else if (msg.cmd == CMD_URB_SUBMIT)
+    {
+      if (!attached)
+	{
+	  fprintf (stderr, "SUBMIT while not attached\n");
+	  return -1;
+	}
+
+      printf ("URB SUBMIT! %d\n", msg.seq);
+      usbip_handle_urb (msg.seq);
+    }
+  else if (msg.cmd == CMD_URB_UNLINK)
+    {
+      struct usbip_msg_ctl msg_ctl;
+      const char zeros[8] = { 0, 0, 0, 0, 0, 0, 0, 0 };
+      uint32_t seq;
+      struct urb *urb;
+      char buf[8];
+      int found = 0;
+
+      if (!attached)
+	{
+	  fprintf (stderr, "UNLINK while not attached\n");
+	  return -1;
+	}
+
+      if (recv (fd, (char *)&msg_ctl, sizeof (msg_ctl), 0) != sizeof (msg_ctl))
+	{
+	  perror ("msg recv ctl");
+	  return -1;
+	}
+
+      if (recv (fd, buf, sizeof (buf), 0) != sizeof (buf))
+	{
+	  perror ("msg recv setup");
+	  return -1;
+	}
+
+      seq = ntohl (msg_ctl.status);
+
+      pthread_mutex_lock (&urb_mutex);
+      if ((urb = urb_list))
+	{
+	  do
+	    if (urb->seq == seq)
+	      {
+		found = 1;
+		break;
+	      }
+	    else
+	      urb = urb->next;
+	  while (urb != urb_list);
+
+	  if (found)
+	    {
+	      unlink_urb (urb);
+	      unlink_urb_ep (urb);
+	      free (urb);
+	    }
+	}
+      pthread_mutex_unlock (&urb_mutex);
+
+      msg.cmd = htonl (REP_URB_UNLINK);
+      msg.seq = htonl (msg.seq);
+
+      msg_ctl.devid = 0;
+      msg_ctl.dir = 0;
+      msg_ctl.ep = 0;
+      if (found)
+	msg_ctl.status = htonl(-ECONNRESET);
+      else
+	msg_ctl.status = 0;
+
+      printf ("URB UNLINK! %d: %s\n", seq, found?"o":"x");
+
+      pthread_mutex_lock (&fd_mutex);
+      if ((size_t)send (fd, &msg, sizeof (msg), 0) != sizeof (msg))
+	{
+	  perror ("reply send");
+	}
+
+      if ((size_t)send (fd, &msg_ctl, sizeof (msg_ctl), 0) != sizeof (msg_ctl))
+	{
+	  perror ("reply send");
+	}
+
+      if ((size_t)send (fd, zeros, sizeof (zeros), 0) != sizeof (zeros))
+	{
+	  perror ("reply send");
+	}
+      pthread_mutex_unlock (&fd_mutex);
+    }
+  else
+    {
+      fprintf (stderr, "Unknown command %08x, disconnecting.\n", msg.cmd);
+      return -1;
+    }
+
+  return 0;
+}
+
+static void
+usbip_ep_ready (struct usb_control *usbc_p)
+{
+  uint64_t l;
+  int r;
+
+  if (!usbc_p->urb)
+    {
+      puts ("???usbip_ep_ready");
+      return;
+    }
+
+  read (usbc_p->eventfd, &l, sizeof (l));
+  pthread_mutex_lock (&usbc_p->mutex);
+  r = hc_handle_data_urb (usbc_p);
+  if (r <= 0)
+    {
+      struct urb *urb;
+      int found = 0;
+
+      urb = usbc_p->urb->prev;
+      do
+	{
+	  if (urb->ep == usbc_p->urb->ep
+	      && urb->dir == usbc_p->urb->dir)
+	    {
+	      if (urb != usbc_p->urb)
+		found = 1;
+	      break;
+	    }
+	  if (urb == urb_list)
+	    break;
+	  urb = urb->prev;
+	}
+      while (1);
+
+      usbip_finish_urb (usbc_p->urb, r);
+
+      if (found)
+	usbc_p->urb = urb;
+      else
+	usbc_p->urb = NULL;
+    }
+  pthread_mutex_unlock (&usbc_p->mutex);
+}
+
+/*
+ * In the USBIP protocol, client sends URB (USB Request Block) to this
+ * server.
+ *
+ * This server acts/emulates as a USB host controller, and transforms
+ * URB into packets to device, transforms packets from device into
+ * URB.
+ */
+static void *
+usbip_run_server (void *arg)
+{
+  int sock;
+  struct sockaddr_in v4addr;
+  const int one = 1;
+  struct pollfd pollfds[16];
+  int i;
+
+  pthread_mutex_init (&usbc.mutex, NULL);
+  pthread_cond_init (&usbc.cond, NULL);
+
+  pthread_mutex_init (&fd_mutex, NULL);
+  pthread_mutex_init (&urb_mutex, NULL);
+
+  (void)arg;
+  if ((sock = socket (PF_INET, SOCK_STREAM, 0)) < 0)
+    {
+      perror ("socket");
+      exit (1);
+    }
+
+  if (setsockopt (sock, SOL_SOCKET, SO_REUSEADDR,
+		  (const char*)&one, sizeof (int)) < 0)
+    perror ("WARN: setsockopt");
+
+  memset (&v4addr, 0, sizeof (v4addr));
+  v4addr.sin_family = AF_INET;
+  v4addr.sin_addr.s_addr = htonl (INADDR_LOOPBACK);
+  v4addr.sin_port = htons (USBIP_PORT);
+
+  if (bind (sock, (const struct sockaddr *)&v4addr, sizeof (v4addr)) < 0)
+    {
+      perror ("bind");
+      exit (1);
+    }
+
+  /* We only accept a connection from a single client.  */
+  if (listen (sock, 1) < 0)
+    {
+      perror ("listen");
+      exit (1);
+    }
+
+  /* FIXME: We don't use this pollstruct as EP0 is handled
+     synchronously.  */
+  pollfds[1].fd = usbc_ep0.eventfd;
+  pollfds[1].events = POLLIN;
+  pollfds[1].revents = 0;
+
+  for (i = 1; i < 8; i++)
+    {
+      pollfds[i*2].fd = usbc_ep_in[i].eventfd;
+      pollfds[i*2].revents = 0;
+      pollfds[i*2+1].fd = usbc_ep_out[i].eventfd;
+      pollfds[i*2+1].revents = 0;
+    }
+
+ again:
+  /* We don't care who is connecting.  */
+  if ((fd = accept (sock, NULL, NULL)) < 0)
+    {
+      perror ("accept");
+      exit (1);
+    }
+
+  pollfds[0].fd = fd;
+  pollfds[0].events = POLLIN;
+  pollfds[0].revents = 0;
+
+  for (;;)
+    {
+      int r;
+
+      for (i = 1; i < 8; i++)
+	{
+	  if (usbc_ep_in[i].urb)
+	    pollfds[i*2].events = POLLIN;
+	  else
+	    pollfds[i*2].events = 0;
+
+	  if (usbc_ep_out[i].urb)
+	    pollfds[i*2+1].events = POLLIN;
+	  else
+	    pollfds[i*2+1].events = 0;
+
+	  pollfds[i*2].revents = 0;
+	  pollfds[i*2+1].revents = 0;
+	}
+
+      if (poll (pollfds, 16, -1) < 0)
+	{
+	  if (errno == EINTR)
+	    goto again;
+
+	  perror ("poll");
+	  exit (1);
+	}
+
+      if ((pollfds[0].revents & POLLNVAL)
+	  || (pollfds[0].revents & POLLERR))
+	{
+	  fprintf (stderr, "Error on USBIP client socket: %d\n", pollfds[0].revents);
+	  exit (1);
+	}
+      if ((pollfds[0].revents & POLLIN))
+	{
+	  r = usbip_process_cmd ();
+	  if (r < 0)
+	    break;
+	  if (r)
+	    goto again;
+	}
+
+      for (i = 1; i < 8; i++)
+	{
+	  if ((pollfds[i*2].revents & POLLNVAL)
+	      || (pollfds[i*2].revents & POLLERR))
+	    {
+	      fprintf (stderr, "Error on ENDP_IN eventfd: %d\n", i);
+	      exit (1);
+	    }
+	  if ((pollfds[i*2].revents & POLLIN))
+	    {
+	      puts ("poll in read");
+	      usbip_ep_ready (&usbc_ep_in[i]);
+	    }
+
+	  if ((pollfds[i*2+1].revents & POLLNVAL)
+	      || (pollfds[i*2+1].revents & POLLERR))
+	    {
+	      fprintf (stderr, "Error on ENDP_OUT eventfd: %d\n", i);
+	      exit (1);
+	    }
+	  if ((pollfds[i*2+1].revents & POLLIN))
+	    {
+	      puts ("poll out read");
+	      usbip_ep_ready (&usbc_ep_out[i]);
+	    }
+	}
+    }
+
+  close (fd);
+  close (sock);
+  exit (0);
+  return NULL;
+}
+
+/** Control setup transaction.
+ *
+ * -->[SETUP]-->[DATA0]-->{ACK}-> Success
+ *                      \-------> Error
+ */
+static int
+control_setup_transaction (struct urb *urb)
+{
+  int r;
+
+  pthread_mutex_lock (&usbc_ep0.mutex);
+  if (usbc_ep0.state == USB_STATE_READY)
+    {
+      if (urb->dir == USBIP_DIR_OUT
+	  && urb->setup[6] == 0 && urb->setup[7] == 0)
+	/* Control Write Transfer with no data stage.  */
+	r = 1;
+      else
+	r = 0;
+
+      usbc_ep0.state = USB_STATE_NAK;
+      memcpy (usb_setup, urb->setup, sizeof (usb_setup));
+      notify_device (USB_INTR_SETUP, 0, urb->dir);
+    }
+  else if (usbc_ep0.state == USB_STATE_NAK)
+    /* something wrong. */
+    r = -EAGAIN;
+  else
+    {
+      fprintf (stderr, "cst error %d\n", usbc_ep0.state);
+      r = -EPIPE;
+    }
+  pthread_mutex_unlock (&usbc_ep0.mutex);
+  return r;
+}
+
+/** Control WRITE transaction.
+ *
+ * -->[OUT]-->[DATA0]---->{ACK}---> Success
+ *                     \--{NAK}---> again
+ *                     \--{STALL}-> Error
+ *                     \----------> Error
+ */
+static int
+control_write_data_transaction (char *buf, uint16_t count)
+{
+  int r;
+
+  pthread_mutex_lock (&usbc_ep0.mutex);
+  if (usbc_ep0.state == USB_STATE_READY)
+    {
+      if (usbc_ep0.len < count)
+	{
+	  fprintf (stderr, "*** usbc_ep0.len < count");
+	  r = -EPIPE;
+	  usbc_ep0.state = USB_STATE_STALL;
+	}
+      else
+	{
+	  r = 0;
+	  usbc_ep0.state = USB_STATE_NAK;
+	  memcpy (usbc_ep0.buf, buf, count);
+	  usbc_ep0.len = count;
+	  notify_device (USB_INTR_DATA_TRANSFER, 0, USBIP_DIR_OUT);
+	}
+    }
+  else if (usbc_ep0.state == USB_STATE_NAK)
+    /* something wrong. */
+    r = -EAGAIN;
+  else
+    r = -EPIPE;
+  pthread_mutex_unlock (&usbc_ep0.mutex);
+  return r;
+}
+
+/** Status transaction for control WRITE.
+ *            zero-len
+ * -->[IN]--->{DATA0}->[ACK]---> Success
+ *         \--{NAK}---> again
+ *         \--{STALL}-> Error
+ *         \----------> Error
+ */
+static int
+control_write_status_transaction (void)
+{
+  int r;
+
+  pthread_mutex_lock (&usbc_ep0.mutex);
+  if (usbc_ep0.state == USB_STATE_READY)
+    {
+      puts ("control_write_status_transaction");
+      if (usbc_ep0.len != 0)
+	fprintf (stderr, "*** ACK length %d\n", usbc_ep0.len);
+      usbc_ep0.state = USB_STATE_NAK;
+      notify_device (USB_INTR_DATA_TRANSFER, 0, USBIP_DIR_IN);
+      r = 0;
+    }
+  else if (usbc_ep0.state == USB_STATE_NAK)
+    r = -EAGAIN;
+  else
+    r = -EPIPE;
+  pthread_mutex_unlock (&usbc_ep0.mutex);
+  return r;
+}
+
+/** Control READ transaction.
+ *
+ * -->[IN]-->{DATAx}---->[ACK]---> Success
+ *         |          \---------> Error
+ *         \-{STALL}------------> Error
+ *         \-{NAK}--------------> again
+ */
+static int
+control_read_data_transaction (char *buf, uint16_t count)
+{
+  int r;
+
+  pthread_mutex_lock (&usbc_ep0.mutex);
+  if (usbc_ep0.state == USB_STATE_READY)
+    {
+      if (usbc_ep0.len > count)
+	fprintf (stderr, "*** length %d\n", usbc_ep0.len);
+      else
+	count = usbc_ep0.len;
+
+      memcpy (buf, usbc_ep0.buf, count);
+      usbc_ep0.state = USB_STATE_NAK;
+      notify_device (USB_INTR_DATA_TRANSFER, 0, USBIP_DIR_IN);
+      r = count;
+    }
+  else if (usbc_ep0.state == USB_STATE_NAK)
+    r = -EAGAIN;
+  else
+    r = -EPIPE;
+  pthread_mutex_unlock (&usbc_ep0.mutex);
+  return r;
+}
+
+/* Status transaction for control READ.
+ *            zero-len
+ * -->[OUT]--->[DATA0]--->{ACK}---> Success
+ *                     \--{NAK}---> again
+ *                     \--{STALL}-> Error
+ *                     \----------> Error
+ */
+static int
+control_read_status_transaction (void)
+{
+  int r;
+
+  pthread_mutex_lock (&usbc_ep0.mutex);
+  if (usbc_ep0.state == USB_STATE_READY)
+    {
+      usbc_ep0.len = 0;
+      usbc_ep0.state = USB_STATE_NAK;
+      notify_device (USB_INTR_DATA_TRANSFER, 0, USBIP_DIR_OUT);
+      r = 0;
+    }
+  else if (usbc_ep0.state == USB_STATE_NAK)
+    r = -EAGAIN;
+  else
+    r = -EPIPE;
+  pthread_mutex_unlock (&usbc_ep0.mutex);
+  return r;
+}
+
+#define USB_URB_TIMEOUT 1000000 /* nanosecond */
+
+/* WRITE transaction
+ * -->[OUT]--->[DATAx]--->{ACK}---> Success
+ *                     \--{NAK}---> again
+ *                     \--{STALL}-> Error
+ *                     \----------> Error
+ */
+static int
+write_data_transaction (struct usb_control *usbc_p,
+			int ep_num, char *buf, uint16_t count)
+{
+  if (usbc_p->len < count)
+    {
+      usbc_p->state = USB_STATE_STALL;
+      return -EPIPE;
+    }
+
+  usbc_p->state = USB_STATE_NAK;
+  memcpy (usbc_p->buf, buf, count);
+  usbc_p->len = count;
+  notify_device (USB_INTR_DATA_TRANSFER, ep_num, USBIP_DIR_OUT);
+  return count;
+}
+
+/** READ transaction.
+ *
+ * -->[IN]-->{DATAx}---->[ACK]---> Success
+ *         |          \---------> Error
+ *         \-{STALL}------------> Error
+ *         \-{NAK}--------------> again
+ */
+static int
+read_data_transaction (struct usb_control *usbc_p,
+		       int ep_num, char *buf, uint16_t count)
+{
+  if (usbc_p->len > count)
+    {
+      fprintf (stderr, "*** length %d\n", usbc_p->len);
+      usbc_p->state = USB_STATE_STALL;
+      return -EPIPE;
+    }
+
+  usbc_p->state = USB_STATE_NAK;
+  memcpy (buf, usbc_p->buf, usbc_p->len);
+  notify_device (USB_INTR_DATA_TRANSFER, ep_num, USBIP_DIR_IN);
+  return usbc_p->len;
+}
+
+void chx_handle_intr (uint32_t irq_num);
+
+#define INTR_REQ_USB SIGUSR1
+
+static void
+usb_intr (int signum, siginfo_t *siginfo, void *arg)
+{
+  extern void chx_sigmask (ucontext_t *uc);
+
+  ucontext_t *uc = arg;
+  (void)signum;
+  (void)siginfo;
+  chx_handle_intr (INTR_REQ_USB);
+  chx_sigmask (uc);
+}
+
+
+void
+usb_lld_init (struct usb_dev *dev, uint8_t feature)
+{
+  int r;
+  sigset_t sigset;
+  struct sigaction act;
+  uint8_t ep;
+
+  sigemptyset (&sigset);
+  sigaddset (&sigset, SIGUSR1);
+  sigaddset (&sigset, SIGALRM);
+
+  /*
+   * Launch the thread for USBIP.  This maps to usb_lld_sys_init where
+   * we initialize USB controller on MCU.
+   */
+  tid_main = pthread_self ();
+
+  pthread_sigmask (SIG_BLOCK, &sigset, NULL);
+
+  pthread_mutex_init (&usbc_ep0.mutex, NULL);
+  usbc_ep0.urb = NULL;
+  usbc_ep0.eventfd = eventfd (0, EFD_CLOEXEC);
+  if (usbc_ep0.eventfd < 0)
+    {
+      perror ("eventfd");
+      exit (1);
+    }
+
+  for (ep = 1; ep < 8; ep++)
+    {
+      pthread_mutex_init (&usbc_ep_in[ep].mutex, NULL);
+      pthread_mutex_init (&usbc_ep_out[ep].mutex, NULL);
+
+      usbc_ep_in[ep].urb = NULL;
+      usbc_ep_in[ep].eventfd = eventfd (0, EFD_CLOEXEC);
+      if (usbc_ep_in[ep].eventfd < 0)
+	{
+	  perror ("eventfd");
+	  exit (1);
+	}
+
+      usbc_ep_out[ep].urb = NULL;
+      usbc_ep_out[ep].eventfd = eventfd (0, EFD_CLOEXEC);
+      if (usbc_ep_out[ep].eventfd < 0)
+	{
+	  perror ("eventfd");
+	  exit (1);
+	}
+    }
+
+  r = pthread_create (&tid_usbip, NULL, usbip_run_server, NULL);
+  if (r)
+    {
+      fprintf (stderr, "usb_lld_init: %s\n", strerror (r));
+      exit (1);
+    }
+
+  act.sa_sigaction = usb_intr;
+  sigfillset (&act.sa_mask);
+  act.sa_flags = SA_SIGINFO|SA_RESTART;
+  sigaction (SIGUSR1, &act, NULL);
+
+  pthread_sigmask (SIG_UNBLOCK, &sigset, NULL);
+
+  dev->configuration = 0;
+  dev->feature = feature;
+  dev->state = WAIT_SETUP;
+
+  usbc_ep0.state = USB_STATE_READY;
+}
+
+
+void
+usb_lld_prepare_shutdown (void)
+{
+}
+
+void
+usb_lld_shutdown (void)
+{
+  /* 
+   * Stop USBIP server thread.
+   */
+  pthread_cancel (tid_usbip);
+  pthread_join (tid_usbip, NULL);
+  /* FIXME: Cancel all URB in the list here.  */
+}
+
+#define USB_MAKE_EV(event) (event<<24)
+#define USB_MAKE_TXRX(ep_num,txrx,len) ((txrx? (1<<23):0)|(ep_num<<16)|len)
+
+static int handle_setup0 (struct usb_dev *dev);
+static int usb_handle_transfer (struct usb_dev *dev, uint8_t dir, uint8_t ep_num);
+
+int
+usb_lld_event_handler (struct usb_dev *dev)
+{
+  uint8_t intr;
+  uint8_t dir;
+  uint8_t ep_num;
+
+  pthread_mutex_lock (&usbc.mutex);
+  intr = usbc.intr;
+  dir = usbc.dir;
+  ep_num = usbc.ep_num;
+  usbc.intr = USB_INTR_NONE;
+  pthread_cond_signal (&usbc.cond);
+  pthread_mutex_unlock (&usbc.mutex);
+
+  if (intr == USB_INTR_RESET)
+    return USB_MAKE_EV (USB_EVENT_DEVICE_RESET);
+  else if (intr == USB_INTR_SETUP)
+    return USB_MAKE_EV (handle_setup0 (dev));
+  else if (intr == USB_INTR_DATA_TRANSFER)
+    return usb_handle_transfer (dev, dir, ep_num);
+
+  return USB_EVENT_OK;
+}
+
+static void handle_datastage_out (struct usb_dev *dev)
+{
+  struct ctrl_data *data_p = &dev->ctrl_data;
+  uint32_t len;
+  const uint64_t l = 1;
+
+  pthread_mutex_lock (&usbc_ep0.mutex);
+  len = usbc_ep0.len;
+  data_p->len -= len;
+  data_p->addr += len;
+
+  len = data_p->len;
+  if (len > USB_MAX_PACKET_SIZE)
+    len = USB_MAX_PACKET_SIZE;
+
+  if (dev->ctrl_data.len == 0)
+    {
+      dev->state = WAIT_STATUS_IN;
+      usbc_ep0.buf = usb_setup;
+      usbc_ep0.len = 0;
+      usbc_ep0.state = USB_STATE_READY;
+    }
+  else
+    {
+      dev->state = OUT_DATA;
+      usbc_ep0.buf = data_p->addr;
+      usbc_ep0.len = len;
+      usbc_ep0.state = USB_STATE_READY;
+    }
+
+  write (usbc_ep0.eventfd, &l, sizeof (l));
+  pthread_mutex_unlock (&usbc_ep0.mutex);
+}
+
+static void handle_datastage_in (struct usb_dev *dev)
+{
+  struct ctrl_data *data_p = &dev->ctrl_data;
+  uint32_t len = USB_MAX_PACKET_SIZE;
+  const uint64_t l = 1;
+
+  if ((data_p->len == 0) && (dev->state == LAST_IN_DATA))
+    {
+      pthread_mutex_lock (&usbc_ep0.mutex);
+
+      if (data_p->require_zlp)
+	{
+	  data_p->require_zlp = 0;
+
+	  /* No more data to send.  Send empty packet */
+	  usbc_ep0.buf = usb_setup;
+	  usbc_ep0.len = 0;
+	  usbc_ep0.state = USB_STATE_READY;
+	}
+      else
+	{
+	  /* No more data to send, proceed to receive OUT acknowledge.  */
+	  dev->state = WAIT_STATUS_OUT;
+	  usbc_ep0.buf = usb_setup;
+	  usbc_ep0.len = 0;
+	  usbc_ep0.state = USB_STATE_READY;
+	}
+
+      write (usbc_ep0.eventfd, &l, sizeof (l));
+      pthread_mutex_unlock (&usbc_ep0.mutex);
+      return;
+    }
+
+  dev->state = (data_p->len <= len) ? LAST_IN_DATA : IN_DATA;
+
+  if (len > data_p->len)
+    len = data_p->len;
+
+  pthread_mutex_lock (&usbc_ep0.mutex);
+  usbc_ep0.buf = data_p->addr;
+  usbc_ep0.len = len;
+  usbc_ep0.state = USB_STATE_READY;
+  data_p->len -= len;
+  data_p->addr += len;
+  write (usbc_ep0.eventfd, &l, sizeof (l));
+  pthread_mutex_unlock (&usbc_ep0.mutex);
+}
+
+typedef int (*HANDLER) (struct usb_dev *dev);
+
+static int std_none (struct usb_dev *dev)
+{
+  (void)dev;
+  return -1;
+}
+
+static uint16_t status_info;
+
+static int std_get_status (struct usb_dev *dev)
+{
+  struct device_req *arg = &dev->dev_req;
+  uint8_t rcp = (arg->type & RECIPIENT);
+
+  if (arg->value != 0 || arg->len != 2 || (arg->index >> 8) != 0
+      || USB_SETUP_SET (arg->type))
+    return -1;
+
+  if (rcp == DEVICE_RECIPIENT)
+    {
+      if (arg->index == 0)
+	{
+	  /* Get Device Status */
+	  uint8_t feature = dev->feature;
+
+	  /* Remote Wakeup enabled */
+	  if ((feature & (1 << 5)))
+	    status_info |= 2;
+	  else
+	    status_info &= ~2;
+
+	  /* Bus-powered */
+	  if ((feature & (1 << 6)))
+	    status_info |= 1;
+	  else /* Self-powered */
+	    status_info &= ~1;
+
+	  return usb_lld_ctrl_send (dev, &status_info, 2);
+	}
+    }
+  else if (rcp == INTERFACE_RECIPIENT)
+    {
+      if (dev->configuration == 0)
+	return -1;
+
+      return USB_EVENT_GET_STATUS_INTERFACE;
+    }
+  else if (rcp == ENDPOINT_RECIPIENT)
+    {
+      uint8_t ep_num = (arg->index & 0x0f);
+
+      if ((arg->index & 0x70) || ep_num == ENDP0)
+	return -1;
+
+      if ((arg->index & 0x80))
+	{
+	  if (usbc_ep_in[ep_num].state == USB_STATE_DISABLED)
+	    return -1;
+	  else if (usbc_ep_in[ep_num].state == USB_STATE_STALL)
+	    status_info |= 1;
+	}
+      else
+	{
+	  if (usbc_ep_out[ep_num].state == USB_STATE_DISABLED)
+	    return -1;
+	  else if (usbc_ep_out[ep_num].state == USB_STATE_STALL)
+	    status_info |= 1;
+	}
+
+      return usb_lld_ctrl_send (dev, &status_info, 2);
+    }
+
+  return -1;
+}
+
+static int std_clear_feature (struct usb_dev *dev)
+{
+  struct device_req *arg = &dev->dev_req;
+  uint8_t rcp = arg->type & RECIPIENT;
+
+  if (USB_SETUP_GET (arg->type))
+    return -1;
+
+  if (rcp == DEVICE_RECIPIENT)
+    {
+      if (arg->len != 0 || arg->index != 0)
+	return -1;
+
+      if (arg->value == FEATURE_DEVICE_REMOTE_WAKEUP)
+	{
+	  dev->feature &= ~(1 << 5);
+	  return USB_EVENT_CLEAR_FEATURE_DEVICE;
+	}
+    }
+  else if (rcp == ENDPOINT_RECIPIENT)
+    {
+      uint8_t ep_num = (arg->index & 0x0f);
+
+      if (dev->configuration == 0)
+	return -1;
+
+      if (arg->len != 0 || (arg->index >> 8) != 0
+	  || arg->value != FEATURE_ENDPOINT_HALT || ep_num == ENDP0)
+	return -1;
+
+      if ((arg->index & 0x80))
+	{
+	  if (usbc_ep_in[ep_num].state == USB_STATE_DISABLED)
+	    return -1;
+
+	  usbc_ep_in[ep_num].state = USB_STATE_NAK;
+	}
+      else
+	{
+	  if (usbc_ep_out[ep_num].state == USB_STATE_DISABLED)
+	    return -1;
+
+	  usbc_ep_out[ep_num].state = USB_STATE_NAK;
+	}
+
+      return USB_EVENT_CLEAR_FEATURE_ENDPOINT;
+    }
+
+  return -1;
+}
+
+static int std_set_feature (struct usb_dev *dev)
+{
+  struct device_req *arg = &dev->dev_req;
+  uint8_t rcp = arg->type & RECIPIENT;
+
+  if (USB_SETUP_GET (arg->type))
+    return -1;
+
+  if (rcp == DEVICE_RECIPIENT)
+    {
+      if (arg->len != 0 || arg->index != 0)
+	return -1;
+
+      if (arg->value == FEATURE_DEVICE_REMOTE_WAKEUP)
+	{
+	  dev->feature |= 1 << 5;
+	  return USB_EVENT_SET_FEATURE_DEVICE;
+	}
+    }
+  else if (rcp == ENDPOINT_RECIPIENT)
+    {
+      uint8_t ep_num = (arg->index & 0x0f);
+
+      if (dev->configuration == 0)
+	return -1;
+
+      if (arg->len != 0 || (arg->index >> 8) != 0
+	  || arg->value != FEATURE_ENDPOINT_HALT || ep_num == ENDP0)
+	return -1;
+
+      if ((arg->index & 0x80))
+	{
+	  if (usbc_ep_in[ep_num].state == USB_STATE_DISABLED)
+	    return -1;
+
+	  usbc_ep_in[ep_num].state = USB_STATE_STALL;
+	}
+      else
+	{
+	  if (usbc_ep_out[ep_num].state == USB_STATE_DISABLED)
+	    return -1;
+
+	  usbc_ep_out[ep_num].state = USB_STATE_STALL;
+	}
+
+      return USB_EVENT_SET_FEATURE_ENDPOINT;
+    }
+
+  return -1;
+}
+
+static int std_set_address (struct usb_dev *dev)
+{
+  struct device_req *arg = &dev->dev_req;
+  uint8_t rcp = arg->type & RECIPIENT;
+
+  if (USB_SETUP_GET (arg->type))
+    return -1;
+
+  if (rcp == DEVICE_RECIPIENT && arg->len == 0 && arg->value <= 127
+      && arg->index == 0 && dev->configuration == 0)
+    return usb_lld_ctrl_ack (dev);
+
+  return -1;
+}
+
+static int std_get_descriptor (struct usb_dev *dev)
+{
+  struct device_req *arg = &dev->dev_req;
+  if (USB_SETUP_SET (arg->type))
+    return -1;
+
+  return USB_EVENT_GET_DESCRIPTOR;
+}
+
+static int std_get_configuration (struct usb_dev *dev)
+{
+  struct device_req *arg = &dev->dev_req;
+  uint8_t rcp = arg->type & RECIPIENT;
+
+  if (USB_SETUP_SET (arg->type))
+    return -1;
+
+  if (arg->value != 0 || arg->index != 0 || arg->len != 1)
+    return -1;
+
+  if (rcp == DEVICE_RECIPIENT)
+    return usb_lld_ctrl_send (dev, &dev->configuration, 1);
+
+  return -1;
+}
+
+static int std_set_configuration (struct usb_dev *dev)
+{
+  struct device_req *arg = &dev->dev_req;
+  uint8_t rcp = arg->type & RECIPIENT;
+
+  if (USB_SETUP_GET (arg->type))
+    return -1;
+
+  if (arg->index != 0 || arg->len != 0)
+    return -1;
+
+  if (rcp == DEVICE_RECIPIENT)
+    return USB_EVENT_SET_CONFIGURATION;
+
+  return -1;
+}
+
+static int std_get_interface (struct usb_dev *dev)
+{
+  struct device_req *arg = &dev->dev_req;
+  uint8_t rcp = arg->type & RECIPIENT;
+
+  if (USB_SETUP_SET (arg->type))
+    return -1;
+
+  if (arg->value != 0 || (arg->index >> 8) != 0 || arg->len != 1)
+    return -1;
+
+  if (dev->configuration == 0)
+    return -1;
+
+  if (rcp == INTERFACE_RECIPIENT)
+    return USB_EVENT_GET_INTERFACE;
+
+  return -1;
+}
+
+static int std_set_interface (struct usb_dev *dev)
+{
+  struct device_req *arg = &dev->dev_req;
+  uint8_t rcp = arg->type & RECIPIENT;
+
+  if (USB_SETUP_GET (arg->type) || rcp != INTERFACE_RECIPIENT
+      || arg->len != 0 || (arg->index >> 8) != 0
+      || (arg->value >> 8) != 0 || dev->configuration == 0)
+    return -1;
+
+  return USB_EVENT_SET_INTERFACE;
+}
+
+
+static int
+handle_setup0 (struct usb_dev *dev)
+{
+  uint8_t req_no;
+  HANDLER handler;
+
+  dev->dev_req.type = usb_setup[0];
+  dev->dev_req.request = req_no = usb_setup[1];
+  dev->dev_req.value = (usb_setup[3] << 8) + usb_setup[2];
+  dev->dev_req.index = (usb_setup[5] << 8) + usb_setup[4];
+  dev->dev_req.len = (usb_setup[7] << 8) + usb_setup[6];
+
+  dev->ctrl_data.addr = NULL;
+  dev->ctrl_data.len = 0;
+  dev->ctrl_data.require_zlp = 0;
+
+  if ((dev->dev_req.type & REQUEST_TYPE) == STANDARD_REQUEST)
+    {
+      int r;
+
+      switch (req_no)
+	{
+	case 0: handler = std_get_status;  break;
+	case 1: handler = std_clear_feature;  break;
+	case 3: handler = std_set_feature;  break;
+	case 5: handler = std_set_address;  break;
+	case 6: handler = std_get_descriptor;  break;
+	case 8: handler = std_get_configuration;  break;
+	case 9: handler = std_set_configuration;  break;
+	case 10: handler = std_get_interface;  break;
+	case 11: handler = std_set_interface;  break;
+	default: handler = std_none;  break;
+	}
+
+      if ((r = (*handler) (dev)) < 0)
+	{
+	  usb_lld_ctrl_error (dev);
+	  return USB_EVENT_OK;
+	}
+      else
+	return r;
+    }
+  else
+    return USB_EVENT_CTRL_REQUEST;
+}
+
+static int handle_in0 (struct usb_dev *dev)
+{
+  int r = 0;
+  const uint64_t l = 1;
+
+  if (dev->state == IN_DATA || dev->state == LAST_IN_DATA)
+    handle_datastage_in (dev);
+  else if (dev->state == WAIT_STATUS_IN)
+    {
+      if ((dev->dev_req.request == SET_ADDRESS) &&
+	  ((dev->dev_req.type & (REQUEST_TYPE | RECIPIENT))
+	   == (STANDARD_REQUEST | DEVICE_RECIPIENT)))
+	{
+	  /* XXX: record the assigned address of this device??? */
+	  printf ("Set Address: %d\n", dev->dev_req.value);
+	  r = USB_EVENT_DEVICE_ADDRESSED;
+	}
+      else
+	r = USB_EVENT_CTRL_WRITE_FINISH;
+      dev->state = WAIT_SETUP;
+      pthread_mutex_lock (&usbc_ep0.mutex);
+      usbc_ep0.buf = usb_setup;
+      usbc_ep0.len = 8;
+      usbc_ep0.state = USB_STATE_READY;
+      write (usbc_ep0.eventfd, &l, sizeof (l));
+      pthread_mutex_unlock (&usbc_ep0.mutex);
+    }
+  else
+    {
+      puts ("handle_in0 error");
+      dev->state = STALLED;
+      pthread_mutex_lock (&usbc_ep0.mutex);
+      usbc_ep0.state = USB_STATE_STALL;
+      write (usbc_ep0.eventfd, &l, sizeof (l));
+      pthread_mutex_unlock (&usbc_ep0.mutex);
+    }
+
+  return r;
+}
+
+static void handle_out0 (struct usb_dev *dev)
+{
+  const uint64_t l = 1;
+
+  if (dev->state == OUT_DATA)
+    /* Usual case.  */
+    handle_datastage_out (dev);
+  else if (dev->state == WAIT_STATUS_OUT)
+    {
+      /*
+       * Control READ transfer finished by ZLP.
+       * Leave ENDP0 status RX_NAK, TX_NAK.
+       */
+      dev->state = WAIT_SETUP;
+      pthread_mutex_lock (&usbc_ep0.mutex);
+      usbc_ep0.buf = usb_setup;
+      usbc_ep0.len = 8;
+      usbc_ep0.state = USB_STATE_READY;
+      write (usbc_ep0.eventfd, &l, sizeof (l));
+      pthread_mutex_unlock (&usbc_ep0.mutex);
+    }
+  else
+    {
+      /*
+       * dev->state == IN_DATA || dev->state == LAST_IN_DATA
+       * (Host aborts the transfer before finish)
+       * Or else, unexpected state.
+       * STALL the endpoint, until we receive the next SETUP token.
+       */
+      puts ("handle_out0 error");
+      dev->state = STALLED;
+      pthread_mutex_lock (&usbc_ep0.mutex);
+      usbc_ep0.state = USB_STATE_STALL;
+      write (usbc_ep0.eventfd, &l, sizeof (l));
+      pthread_mutex_unlock (&usbc_ep0.mutex);
+    }
+}
+
+
+static int
+usb_handle_transfer (struct usb_dev *dev, uint8_t dir, uint8_t ep_num)
+{
+  if (ep_num == 0)
+    {
+      if (dir)
+	return USB_MAKE_EV (handle_in0 (dev));
+      else
+	{
+	  handle_out0 (dev);
+	  return USB_EVENT_OK;
+	}
+    }
+  else
+    {
+      uint16_t len;
+
+      if (dir)
+	{
+	  len = usbc_ep_in[ep_num].len;
+	  return USB_MAKE_TXRX (ep_num, 1, len);
+	}
+      else
+	{
+	  len = usbc_ep_out[ep_num].len;
+	  return  USB_MAKE_TXRX (ep_num, 0, len);
+	}
+    }
+
+  return USB_EVENT_OK;
+}
+
+int
+usb_lld_ctrl_ack (struct usb_dev *dev)
+{
+  const uint64_t l = 1;
+
+  dev->state = WAIT_STATUS_IN;
+  pthread_mutex_lock (&usbc_ep0.mutex);
+  usbc_ep0.buf = usb_setup;
+  usbc_ep0.len = 0;
+  usbc_ep0.state = USB_STATE_READY;
+  write (usbc_ep0.eventfd, &l, sizeof (l));
+  pthread_mutex_unlock (&usbc_ep0.mutex);
+  return USB_EVENT_OK;
+}
+
+int
+usb_lld_ctrl_recv (struct usb_dev *dev, void *p, size_t len)
+{
+  struct ctrl_data *data_p = &dev->ctrl_data;
+  const uint64_t l = 1;
+
+  data_p->addr = p;
+  data_p->len = len;
+  dev->state = OUT_DATA;
+  if (len > USB_MAX_PACKET_SIZE)
+    len = USB_MAX_PACKET_SIZE;
+  pthread_mutex_lock (&usbc_ep0.mutex);
+  usbc_ep0.state = USB_STATE_READY;
+  usbc_ep0.buf = p;
+  usbc_ep0.len = len;
+  write (usbc_ep0.eventfd, &l, sizeof (l));
+  pthread_mutex_unlock (&usbc_ep0.mutex);
+  return USB_EVENT_OK;
+}
+
+int
+usb_lld_ctrl_send (struct usb_dev *dev, const void *buf, size_t buflen)
+{
+  struct ctrl_data *data_p = &dev->ctrl_data;
+  uint32_t len_asked = dev->dev_req.len;
+  uint32_t len;
+  const uint64_t l = 1;
+
+  data_p->addr = (void *)buf;
+  data_p->len = buflen;
+
+  /* Restrict the data length to be the one host asks for */
+  if (data_p->len > len_asked)
+    data_p->len = len_asked;
+
+  if (data_p->len != 0 && (data_p->len % USB_MAX_PACKET_SIZE) == 0)
+    data_p->require_zlp = 1;
+
+  if (data_p->len < USB_MAX_PACKET_SIZE)
+    {
+      len = data_p->len;
+      dev->state = LAST_IN_DATA;
+    }
+  else
+    {
+      len = USB_MAX_PACKET_SIZE;
+      dev->state = IN_DATA;
+    }
+
+  pthread_mutex_lock (&usbc_ep0.mutex);
+  usbc_ep0.buf = data_p->addr;
+  usbc_ep0.len = len;
+  usbc_ep0.state = USB_STATE_READY;
+  data_p->len -= len;
+  data_p->addr += len;
+  write (usbc_ep0.eventfd, &l, sizeof (l));
+  pthread_mutex_unlock (&usbc_ep0.mutex);
+  return USB_EVENT_OK;
+}
+
+uint8_t
+usb_lld_current_configuration (struct usb_dev *dev)
+{
+  return dev->configuration;
+}
+
+
+void
+usb_lld_ctrl_error (struct usb_dev *dev)
+{
+  const uint64_t l = 1;
+
+  puts ("ctrl_error");
+  dev->state = STALLED;
+  pthread_mutex_lock (&usbc_ep0.mutex);
+  usbc_ep0.state = USB_STATE_STALL;
+  write (usbc_ep0.eventfd, &l, sizeof (l));
+  pthread_mutex_unlock (&usbc_ep0.mutex);
+}
+
+/* FIXME: ??? */
+void
+usb_lld_reset (struct usb_dev *dev, uint8_t feature)
+{
+  usb_lld_set_configuration (dev, 0);
+  dev->feature = feature;
+  usbc_ep0.state = USB_STATE_READY;
+}
+
+void
+usb_lld_set_configuration (struct usb_dev *dev, uint8_t config)
+{
+  dev->configuration = config;
+}
+
+
+void
+usb_lld_setup_endp (struct usb_dev *dev, int ep_num, int rx_en, int tx_en)
+{
+  (void)dev;
+
+  if (ep_num == 0)
+    return;
+
+  if (rx_en)
+    {
+      usbc_ep_out[ep_num].buf = NULL;
+      usbc_ep_out[ep_num].len = 0;
+      usbc_ep_out[ep_num].state = USB_STATE_NAK;
+    }
+
+  if (tx_en)
+    {
+      usbc_ep_in[ep_num].buf = NULL;
+      usbc_ep_in[ep_num].len = 0;
+      usbc_ep_in[ep_num].state = USB_STATE_NAK;
+    }
+}
+
+
+void
+usb_lld_stall_tx (int ep_num)
+{
+  printf ("stall tx %d", ep_num);
+  usbc_ep_in[ep_num].state = USB_STATE_STALL;
+}
+
+void
+usb_lld_stall_rx (int ep_num)
+{
+  printf ("stall rx %d", ep_num);
+  usbc_ep_out[ep_num].state = USB_STATE_STALL;
+}
+
+void
+usb_lld_rx_enable_buf (int ep_num, void *buf, size_t len)
+{
+  struct usb_control *usbc_p = &usbc_ep_out[ep_num];
+  const uint64_t l = 1;
+
+  pthread_mutex_lock (&usbc_p->mutex);
+  usbc_p->state = USB_STATE_READY;
+  usbc_p->buf = buf;
+  usbc_p->len = len;
+
+  write (usbc_p->eventfd, &l, sizeof (l));
+  pthread_mutex_unlock (&usbc_p->mutex);
+  printf ("usb_lld_rx_enable_buf: %d\n", ep_num);
+}
+
+
+void
+usb_lld_tx_enable_buf (int ep_num, const void *buf, size_t len)
+{
+  struct usb_control *usbc_p = &usbc_ep_in[ep_num];
+  const uint64_t l = 1;
+
+  pthread_mutex_lock (&usbc_p->mutex);
+  usbc_p->state = USB_STATE_READY;
+  usbc_p->buf = (void *)buf;
+  usbc_p->len = len;
+  write (usbc_p->eventfd, &l, sizeof (l));
+  pthread_mutex_unlock (&usbc_p->mutex);
+  printf ("usb_lld_tx_enable_buf: %d\n", ep_num);
+}
-- 
cgit v1.2.3