USB Host stack

The USB Host stack server - usb provides the generic core functionality, including abstraction of Host Controller Device, managing devices enumeration and communication with device drivers. The USB Host stack is accessible to other processes through a port /dev/usb.

Host Controller Devices

The USB Host stack allows using different types and multiple instances of HCDs. It provides generic types hcd_t and hcd_ops_t. Specific drivers for different types of HCDs such as ehci, ohci etc. are a part of the phoenix-rtos-devices repository in the form of a static library named, e.g. libusbehci. When building the USB Host stack, one should set the environmental variable USB_HCD_LIBS to an appropriate value denoting host controllers available on the platform. Each Host Controller driver library should register its hcd_ops_t instance using a GCC constructor. It allows the USB Host stack to communicate with an HCD driver using callbacks within a hcd_ops_t structure.

The USB Host stack during initialization first fetches the platform-dependent information on the available HCD instances using hcd_info_t structure. (At this moment it is done using hcd_getInfo() function. In the future it should be done using a device tree.) Then it matches instances with previously registered HCD ops and creates hcd_t instances, with which it would then communicate in terms of scheduling transfers and detecting devices connection/disconnection.

Hubs

Hubs are the basis of the USB devices tree. Each HCD has its own Root Hub with at least one port. Both Root Hubs and additional physical hub devices are managed using the hub driver, which is the only USB class driver implemented as a part of the USB stack, while other USB classes are implemented as separate processes. The hub driver is responsible for managing port status changes, e.g. device connection or disconnection. When a new device is connected the hub module performs the enumeration process and binds a device with appropriate drivers. On the device disconnection, it shall unbind a device from drivers, and destroy a device and all its resources.

Drivers

Drivers are separate processes, which communicate with the Host stack using messages and are represented in the USB Host stack as usb_drv_t instances. A driver first registers itself using a connect USB message (implemented in libusb as usb_connect() function). It then waits for events such as insertion or deletion sent by the USB Host stack. insertion message is sent, once the USB Host stack binds the newly connected device to a particular driver. A device can be a composite device containing multiple interfaces. Each interface can be bound to a different driver. A insertion message is sent for every interface separately. It works the same with deletion messages.

Drivers are bound to interfaces after matching device or interface descriptors with driver filters described by one or many usb_device_id_t structures. An array of those structures is provided to the USB Host stack, when registering a driver using usb_connect() function. It consists of the following fields (from most specific to most general):

  • pid (product ID)
  • vid (vendor ID)
  • protocol
  • subclass
  • dclass (device class)

Each of those fields can have a fixed value or a wildcard value USBDRV_ANY. An interface can match multiple drivers, but the host stack shall choose the most specific one to bind the interface to. For example, it shall prefer a driver, which matches pid and vid fields over a driver that matches dclass and subclass fields.

It's the driver's responsibility to create ports in order to give other processes access to resources of a particular device, e.g. /dev/umass0, /dev/umass1, /dev/usbacm0, etc.

Pipes

Pipes are a software abstraction of a USB endpoint. Drivers communicate with specific endpoints using pipes. A pipe is characterized by a direction (in or out) and type (control, bulk, interrupt isochronous). A device driver first opens a pipe by sending a USB open message (implemented in libusb as usb_open() function). A driver gives details on a pipe it requests to open. If the USB Host stack finds an endpoint on a given device interface with given direction and type, it creates a pipe, allocates a id unique in the context of the driver and returns the ID to the driver. The driver can then send transfers using this pipe ID. A pipe ID can be thought as a UN*X-like file descriptors - it is a key to communicate with a specific endpoint.

A driver writes or reads data using a pipe by sending urb messages. The USB Host stack schedules a transfer on a corresponding Host Controller Device, once it processes a urb. It then informs a driver, that a transfer has been finished either in a synchronous way (calling msgRespond() to the urb message - bulk and control transfers) or asynchronous (a new event message - interrupt and isochronous message).