USB Host functions to support Custom Class USB Devices.
More...
|
| User API |
| User API reference of the Custom Class.
|
|
| Configuration |
| Configuration of the USB Host Custom Class in µVision.
|
|
USB Host functions to support Custom Class USB Devices.
The Custom Class in the USB Host Component is used for attaching USB Devices with a specific USB Class to your system. This can either be one of the standard classes that are not directly supported by the USB Middleware or a vendor specific class. Using these functions, you can add support for any USB Device class to the system.
Refer to:
Software Structure
The application starts the USB Host by calling USBH_Initialize. The USB Host Core will wait until an USB Custom Class device is attached to the system. As soon as this happens the Core will configure and initialize the device and it will be ready to be used by the application.
The transmit functions USBH_PipeSend and USBH_PipeReceive will be called by the user thread directly to communicate with the Custom Class device.
As soon as the Custom Class Device is detached from the system, the callback functions USBH_CustomClass_Unconfigure and USBH_CustomClass_Uninitialize signal the removal to the user application.
Implementation
To create an USB Host with support for the Custom class:
User Code Templates
There are two user code templates available that show how to create support for a Custom Class device:
- USBH_User_CustomClass.c shows in general how to use the Custom Class functions.
- USBH_PL2303.c is an actual implementation of the Custom Class functions to add support for Prolific's PL2303 UART to serial RS232 adapter.
User Code Template USBH_User_CustomClass.c
The following source code can be used to implement support for a Custom USB Device Class in the user application.
#include <stdint.h>
#include "rl_usb.h"
#define CUSTOM_CLASS_IF_CLASS USB_DEVICE_CLASS_VENDOR_SPECIFIC
#define CUSTOM_CLASS_IF_SUBCLASS 0
#define CUSTOM_CLASS_IF_PROTOCOL 0
uint8_t USBH_CustomClass_Ctrl = 0;
USBH_PIPE *USBH_CustomClass_Pipes[8] = { 0 };
USB_INTERFACE_DESCRIPTOR *ptr_if_desc;
USB_ENDPOINT_DESCRIPTOR *ptr_ep_desc;
uint8_t num, i;
USBH_CustomClass_Ctrl = ptr_dev->
ctrl;
ptr_if_desc = (USB_INTERFACE_DESCRIPTOR *)((uint32_t)ptr_cfg_desc + ptr_cfg_desc->bLength);
num = ptr_if_desc->bNumEndpoints;
switch (ptr_if_desc->bInterfaceClass) {
case CUSTOM_CLASS_IF_CLASS:
switch (ptr_if_desc->bInterfaceSubClass) {
case CUSTOM_CLASS_IF_SUBCLASS:
switch (ptr_if_desc->bInterfaceProtocol) {
case CUSTOM_CLASS_IF_PROTOCOL:
ptr_ep_desc = (USB_ENDPOINT_DESCRIPTOR *)((uint32_t)ptr_if_desc + ptr_if_desc->bLength);
i = 0;
while (num--) {
ptr_pipe =
USBH_PipeCreate (ptr_dev->
ctrl, ptr_dev->
dev_addr, ptr_dev->
dev_speed, 0, 0, ptr_ep_desc->bEndpointAddress, ptr_ep_desc->bmAttributes & USB_ENDPOINT_TYPE_MASK, ptr_ep_desc->wMaxPacketSize & 0x7FF, ptr_ep_desc->bInterval);
if (!ptr_pipe) {
for (i = 0; i < 8; i++) {
if (USBH_CustomClass_Pipes[i]) {
USBH_CustomClass_Pipes[i] = NULL;
}
}
return 0xFF;
}
USBH_CustomClass_Pipes[i++] = ptr_pipe;
ptr_ep_desc++;
}
return 0;
}
break;
}
break;
}
return 0xFF;
}
uint8_t i;
USBH_CustomClass_Ctrl = 0;
for (i = 0; i < 8; i++) {
if (USBH_CustomClass_Pipes[i]) {
USBH_CustomClass_Pipes[i] = NULL;
}
}
}
}
}
User Code Template USBH_PL2303.c
The following source code implements support for Prolific's PL2303 UART to serial RS232 adapter using the Custom Class functions.
#include <stdint.h>
#include "rl_usb.h"
#define CUSTOM_CLASS_IF_CLASS USB_DEVICE_CLASS_VENDOR_SPECIFIC
#define CUSTOM_CLASS_IF_SUBCLASS 0
#define CUSTOM_CLASS_IF_PROTOCOL 0
uint8_t USBH_CustomClass_Ctrl = 0;
USBH_PIPE *USBH_CustomClass_Pipes[8] = { 0 };
USB_INTERFACE_DESCRIPTOR *ptr_if_desc;
USB_ENDPOINT_DESCRIPTOR *ptr_ep_desc;
uint8_t num, i;
USBH_CustomClass_Ctrl = ptr_dev->
ctrl;
ptr_if_desc = (USB_INTERFACE_DESCRIPTOR *)((uint32_t)ptr_cfg_desc + ptr_cfg_desc->bLength);
num = ptr_if_desc->bNumEndpoints;
if ((ptr_dev_desc->idVendor != 0x067B) || (ptr_dev_desc->idProduct != 0x2303))
return 0xFF;
switch (ptr_if_desc->bInterfaceClass) {
case CUSTOM_CLASS_IF_CLASS:
switch (ptr_if_desc->bInterfaceSubClass) {
case CUSTOM_CLASS_IF_SUBCLASS:
switch (ptr_if_desc->bInterfaceProtocol) {
case CUSTOM_CLASS_IF_PROTOCOL:
ptr_ep_desc = (USB_ENDPOINT_DESCRIPTOR *)((uint32_t)ptr_if_desc + ptr_if_desc->bLength);
i = 0;
while (num--) {
ptr_pipe =
USBH_PipeCreate (ptr_dev->
ctrl, ptr_dev->
dev_addr, ptr_dev->
dev_speed, 0, 0, ptr_ep_desc->bEndpointAddress, ptr_ep_desc->bmAttributes & USB_ENDPOINT_TYPE_MASK, ptr_ep_desc->wMaxPacketSize & 0x7FF, ptr_ep_desc->bInterval);
if (!ptr_pipe) {
for (i = 0; i < 8; i++) {
if (USBH_CustomClass_Pipes[i]) {
USBH_CustomClass_Pipes[i] = NULL;
}
}
return 0xFF;
}
USBH_CustomClass_Pipes[i++] = ptr_pipe;
ptr_ep_desc++;
}
return 0;
}
break;
}
break;
}
return 0xFF;
}
uint8_t i;
USBH_CustomClass_Ctrl = 0;
for (i = 0; i < 8; i++) {
if (USBH_CustomClass_Pipes[i]) {
USBH_CustomClass_Pipes[i] = NULL;
}
}
}
uint8_t buf[8];
setup_packet.
wValue = U16_LE(0x8484);
setup_packet.
wIndex = U16_LE(0);
setup_packet.
wValue = U16_LE(0x0404);
setup_packet.
wValue = U16_LE(0x8484);
setup_packet.
wValue = U16_LE(0x8383);
setup_packet.
wValue = U16_LE(0x8484);
setup_packet.
wValue = U16_LE(0x0404);
setup_packet.
wIndex = U16_LE(1);
setup_packet.
wValue = U16_LE(0x8484);
setup_packet.
wIndex = U16_LE(0);
setup_packet.
wValue = U16_LE(0x8383);
setup_packet.
wValue = U16_LE(0);
setup_packet.
wIndex = U16_LE(1);
setup_packet.
wValue = U16_LE(1);
setup_packet.
wIndex = U16_LE(0);
setup_packet.
wValue = U16_LE(2);
setup_packet.
wIndex = U16_LE(0x44);
setup_packet.
wValue = U16_LE(0);
setup_packet.
wIndex = U16_LE(0);
*((uint32_t *)&buf[0]) = U32_LE(9600);
buf[4] = 0;
buf[5] = 0;
buf[6] = 8;
}
}