|
-
-
- #include <Arduino.h>
- #include "USBHost.h"
-
-
- static USBDriver *available_drivers = NULL;
- static uint8_t enumbuf[256] __attribute__ ((aligned(16)));
- static setup_t enumsetup __attribute__ ((aligned(16)));
- static uint16_t enumlen;
- static Device_t *devlist=NULL;
-
-
-
-
- volatile bool USBHost::enumeration_busy = false;
-
- static void pipe_set_maxlen(Pipe_t *pipe, uint32_t maxlen);
- static void pipe_set_addr(Pipe_t *pipe, uint32_t addr);
-
-
-
- void USBHost::driver_ready_for_device(USBDriver *driver)
- {
- driver->device = NULL;
- driver->next = NULL;
- if (available_drivers == NULL) {
- available_drivers = driver;
- } else {
-
- USBDriver *last = available_drivers;
- while (last->next) last = last->next;
- last->next = driver;
- }
- }
-
-
-
- Device_t * USBHost::new_Device(uint32_t speed, uint32_t hub_addr, uint32_t hub_port)
- {
- Device_t *dev;
-
- print("new_Device: ");
- switch (speed) {
- case 0: print("12"); break;
- case 1: print("1.5"); break;
- case 2: print("480"); break;
- default: print("??");
- }
- println(" Mbit/sec");
- dev = allocate_Device();
- if (!dev) return NULL;
- memset(dev, 0, sizeof(Device_t));
- dev->speed = speed;
- dev->address = 0;
- dev->hub_address = hub_addr;
- dev->hub_port = hub_port;
- dev->control_pipe = new_Pipe(dev, 0, 0, 0, 8);
- if (!dev->control_pipe) {
- free_Device(dev);
- return NULL;
- }
- dev->control_pipe->callback_function = &enumeration;
- dev->control_pipe->direction = 1;
-
-
- USBHost::enumeration_busy = true;
- mk_setup(enumsetup, 0x80, 6, 0x0100, 0, 8);
- queue_Control_Transfer(dev, &enumsetup, enumbuf, NULL);
- if (devlist == NULL) {
- devlist = dev;
- } else {
- Device_t *p;
- for (p = devlist; p->next; p = p->next) ;
- p->next = dev;
- }
- return dev;
- }
-
-
-
- void USBHost::enumeration(const Transfer_t *transfer)
- {
- Device_t *dev;
- uint32_t len;
-
-
- if (transfer->driver) {
- transfer->driver->control(transfer);
- return;
- }
-
- println("enumeration:");
-
-
- dev = transfer->pipe->device;
-
- while (1) {
-
-
-
-
-
-
-
- switch (dev->enum_state) {
- case 0:
- pipe_set_maxlen(dev->control_pipe, enumbuf[7]);
- mk_setup(enumsetup, 0, 5, assign_address(), 0, 0);
- queue_Control_Transfer(dev, &enumsetup, NULL, NULL);
- dev->enum_state = 1;
- return;
- case 1:
- dev->address = enumsetup.wValue;
- pipe_set_addr(dev->control_pipe, enumsetup.wValue);
- mk_setup(enumsetup, 0x80, 6, 0x0100, 0, 18);
- queue_Control_Transfer(dev, &enumsetup, enumbuf, NULL);
- dev->enum_state = 2;
- return;
- case 2:
- dev->bDeviceClass = enumbuf[4];
- dev->bDeviceSubClass = enumbuf[5];
- dev->bDeviceProtocol = enumbuf[6];
- dev->idVendor = enumbuf[8] | (enumbuf[9] << 8);
- dev->idProduct = enumbuf[10] | (enumbuf[11] << 8);
- enumbuf[0] = enumbuf[14];
- enumbuf[1] = enumbuf[15];
- enumbuf[2] = enumbuf[16];
- if ((enumbuf[0] | enumbuf[1] | enumbuf[2]) > 0) {
- dev->enum_state = 3;
- } else {
- dev->enum_state = 11;
- }
- break;
- case 3:
- len = sizeof(enumbuf) - 4;
- mk_setup(enumsetup, 0x80, 6, 0x0300, 0, len);
- queue_Control_Transfer(dev, &enumsetup, enumbuf + 4, NULL);
- dev->enum_state = 4;
- return;
- case 4:
- if (enumbuf[4] < 4 || enumbuf[5] != 3) {
- dev->enum_state = 11;
- } else {
- dev->LanguageID = enumbuf[6] | (enumbuf[7] << 8);
- if (enumbuf[0]) dev->enum_state = 5;
- else if (enumbuf[1]) dev->enum_state = 7;
- else if (enumbuf[2]) dev->enum_state = 9;
- else dev->enum_state = 11;
- }
- break;
- case 5:
- len = sizeof(enumbuf) - 4;
- mk_setup(enumsetup, 0x80, 6, 0x0300 | enumbuf[0], dev->LanguageID, len);
- queue_Control_Transfer(dev, &enumsetup, enumbuf + 4, NULL);
- dev->enum_state = 6;
- return;
- case 6:
-
- if (enumbuf[1]) dev->enum_state = 7;
- else if (enumbuf[2]) dev->enum_state = 9;
- else dev->enum_state = 11;
- break;
- case 7:
- len = sizeof(enumbuf) - 4;
- mk_setup(enumsetup, 0x80, 6, 0x0300 | enumbuf[1], dev->LanguageID, len);
- queue_Control_Transfer(dev, &enumsetup, enumbuf + 4, NULL);
- dev->enum_state = 8;
- return;
- case 8:
-
- if (enumbuf[2]) dev->enum_state = 9;
- else dev->enum_state = 11;
- break;
- case 9:
- len = sizeof(enumbuf) - 4;
- mk_setup(enumsetup, 0x80, 6, 0x0300 | enumbuf[2], dev->LanguageID, len);
- queue_Control_Transfer(dev, &enumsetup, enumbuf + 4, NULL);
- dev->enum_state = 10;
- return;
- case 10:
-
- dev->enum_state = 11;
- break;
- case 11:
- mk_setup(enumsetup, 0x80, 6, 0x0200, 0, 9);
- queue_Control_Transfer(dev, &enumsetup, enumbuf, NULL);
- dev->enum_state = 12;
- return;
- case 12:
- enumlen = enumbuf[2] | (enumbuf[3] << 8);
- println("Config data length = ", enumlen);
- if (enumlen > sizeof(enumbuf)) {
-
- }
- mk_setup(enumsetup, 0x80, 6, 0x0200, 0, enumlen);
- queue_Control_Transfer(dev, &enumsetup, enumbuf, NULL);
- dev->enum_state = 13;
- return;
- case 13:
- println("bNumInterfaces = ", enumbuf[4]);
- println("bConfigurationValue = ", enumbuf[5]);
- dev->bmAttributes = enumbuf[7];
- dev->bMaxPower = enumbuf[8];
-
- mk_setup(enumsetup, 0, 9, enumbuf[5], 0, 0);
- queue_Control_Transfer(dev, &enumsetup, NULL, NULL);
- dev->enum_state = 14;
- return;
- case 14:
- claim_drivers(dev);
- dev->enum_state = 15;
-
-
-
-
- USBHost::enumeration_busy = false;
- return;
- case 15:
-
- default:
- return;
- }
- }
- }
-
- void USBHost::claim_drivers(Device_t *dev)
- {
- USBDriver *driver, *prev=NULL;
-
-
- for (driver=available_drivers; driver != NULL; driver = driver->next) {
- if (driver->claim(dev, 0, enumbuf + 9, enumlen - 9)) {
- if (prev) {
- prev->next = driver->next;
- } else {
- available_drivers = driver->next;
- }
- driver->device = dev;
- driver->next = NULL;
- dev->drivers = driver;
- return;
- }
- prev = driver;
- }
-
- const uint8_t *p = enumbuf + 9;
- const uint8_t *end = enumbuf + enumlen;
- while (p < end) {
- uint8_t desclen = *p;
- uint8_t desctype = *(p+1);
- print("Descriptor ");
- print(desctype);
- print(" = ");
- if (desctype == 4) println("INTERFACE");
- else if (desctype == 5) println("ENDPOINT");
- else if (desctype == 6) println("DEV_QUALIFIER");
- else if (desctype == 7) println("OTHER_SPEED");
- else if (desctype == 11) println("IAD");
- else if (desctype == 33) println("HID");
- else println(" ???");
- if (desctype == 11 && desclen == 8) {
-
-
- }
- if (desctype == 4 && desclen == 9) {
-
- prev = NULL;
- for (driver=available_drivers; driver != NULL; driver = driver->next) {
-
-
-
-
- if (driver->claim(dev, 1, p, end - p)) {
-
-
- if (prev) {
- prev->next = driver->next;
- } else {
- available_drivers = driver->next;
- }
-
- if (dev->drivers) {
- dev->drivers->next = driver;
- }
- dev->drivers = driver;
- driver->next = NULL;
- driver->device = dev;
-
- }
- prev = driver;
- }
- }
- p += desclen;
- }
- }
-
- static bool address_in_use(uint32_t addr)
- {
- for (Device_t *p = devlist; p; p = p->next) {
- if (p->address == addr) return true;
- }
- return false;
- }
-
- uint32_t USBHost::assign_address(void)
- {
- static uint8_t last_assigned_address=0;
- uint32_t addr = last_assigned_address;
- while (1) {
- if (++addr > 127) addr = 1;
- if (!address_in_use(addr)) {
- last_assigned_address = addr;
- return addr;
- }
- }
- }
-
- static void pipe_set_maxlen(Pipe_t *pipe, uint32_t maxlen)
- {
- pipe->qh.capabilities[0] = (pipe->qh.capabilities[0] & 0x8000FFFF) | (maxlen << 16);
- }
-
- static void pipe_set_addr(Pipe_t *pipe, uint32_t addr)
- {
- pipe->qh.capabilities[0] = (pipe->qh.capabilities[0] & 0xFFFFFF80) | addr;
- }
-
-
- void USBHost::disconnect_Device(Device_t *dev)
- {
- if (!dev) return;
- println("disconnect_Device:");
-
-
-
-
- print_driverlist("available_drivers", available_drivers);
- print_driverlist("dev->drivers", dev->drivers);
- for (USBDriver *p = dev->drivers; p; ) {
- println("disconnect driver ", (uint32_t)p, HEX);
- p->disconnect();
- USBDriver *next = p->next;
- p->next = available_drivers;
- available_drivers = p;
- p = next;
- }
- print_driverlist("available_drivers", available_drivers);
-
-
- for (Pipe_t *p = dev->data_pipes; p; ) {
- Pipe_t *next = p->next;
- delete_Pipe(p);
- p = next;
- }
- delete_Pipe(dev->control_pipe);
-
-
- Device_t *prev_dev = NULL;
- for (Device_t *p = devlist; p; p = p->next) {
- if (p == dev) {
- if (prev_dev == NULL) {
- devlist = p->next;
- } else {
- prev_dev->next = p->next;
- }
- println("removed Device_t from devlist");
- free_Device(p);
- break;
- }
- prev_dev = p;
- }
- }
-
|