mirror of
https://github.com/felis/USB_Host_Shield_2.0.git
synced 2024-03-22 11:31:26 +01:00
Fix hub code, fix enumeration.
This commit is contained in:
parent
e01df49bf3
commit
53b7f82d94
7 changed files with 184 additions and 101 deletions
113
Usb.cpp
113
Usb.cpp
|
@ -563,6 +563,25 @@ uint8_t USB::DefaultAddressing(uint8_t parent, uint8_t port, bool lowspeed) {
|
||||||
return 0;
|
return 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
uint8_t USB::AttemptConfig(uint8_t driver, uint8_t parent, uint8_t port, bool lowspeed) {
|
||||||
|
uint8_t rcode = 0;
|
||||||
|
//printf("AttemptConfig: parent = %i, port = %i\r\n", parent, port);
|
||||||
|
|
||||||
|
rcode = devConfig[driver]->ConfigureDevice(parent, port, lowspeed);
|
||||||
|
if (rcode == USB_ERROR_CONFIG_REQUIRES_ADDITIONAL_RESET) {
|
||||||
|
if (parent == 0) {
|
||||||
|
// Send a bus reset on the root interface.
|
||||||
|
regWr(rHCTL, bmBUSRST); //issue bus reset
|
||||||
|
delay(102); // delay 102ms, compensate for clock inaccuracy.
|
||||||
|
} else {
|
||||||
|
// reset parent port
|
||||||
|
devConfig[parent]->ResetHubPort(port);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
rcode = devConfig[driver]->Init(parent, port, lowspeed);
|
||||||
|
return rcode;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This is broken. We need to enumerate differently.
|
* This is broken. We need to enumerate differently.
|
||||||
* It causes major problems with several devices if detected in an unexpected order.
|
* It causes major problems with several devices if detected in an unexpected order.
|
||||||
|
@ -604,40 +623,96 @@ uint8_t USB::DefaultAddressing(uint8_t parent, uint8_t port, bool lowspeed) {
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
uint8_t USB::Configuring(uint8_t parent, uint8_t port, bool lowspeed) {
|
uint8_t USB::Configuring(uint8_t parent, uint8_t port, bool lowspeed) {
|
||||||
|
//uint8_t bAddress = 0;
|
||||||
|
//printf("Configuring: parent = %i, port = %i\r\n", parent, port);
|
||||||
uint8_t rcode = 0;
|
uint8_t rcode = 0;
|
||||||
|
uint8_t buf[sizeof (USB_DEVICE_DESCRIPTOR)];
|
||||||
|
UsbDevice *p = NULL;
|
||||||
|
EpInfo *oldep_ptr = NULL;
|
||||||
|
EpInfo epInfo;
|
||||||
|
|
||||||
for (; devConfigIndex < USB_NUMDEVICES; devConfigIndex++) {
|
epInfo.epAddr = 0;
|
||||||
if (!devConfig[devConfigIndex])
|
epInfo.maxPktSize = 8;
|
||||||
continue;
|
epInfo.epAttribs = 0;
|
||||||
|
epInfo.bmNakPower = USB_NAK_MAX_POWER;
|
||||||
|
|
||||||
rcode = devConfig[devConfigIndex]->ConfigureDevice(parent, port, lowspeed);
|
delay(2000);
|
||||||
if (rcode == USB_ERROR_CONFIG_REQUIRES_ADDITIONAL_RESET) {
|
AddressPool &addrPool = GetAddressPool();
|
||||||
if (parent == 0) {
|
// Get pointer to pseudo device with address 0 assigned
|
||||||
// Send a bus reset on the root interface.
|
p = addrPool.GetUsbDevicePtr(0);
|
||||||
regWr(rHCTL, bmBUSRST); //issue bus reset
|
if (!p) {
|
||||||
delay(102); // delay 102ms, compensate for clock inaccuracy.
|
//printf("Configuring error: USB_ERROR_ADDRESS_NOT_FOUND_IN_POOL\r\n");
|
||||||
} else {
|
return USB_ERROR_ADDRESS_NOT_FOUND_IN_POOL;
|
||||||
// reset parent port
|
}
|
||||||
devConfig[parent]->ResetHubPort(port);
|
|
||||||
}
|
// Save old pointer to EP_RECORD of address 0
|
||||||
|
oldep_ptr = p->epinfo;
|
||||||
|
|
||||||
|
// Temporary assign new pointer to epInfo to p->epinfo in order to
|
||||||
|
// avoid toggle inconsistence
|
||||||
|
|
||||||
|
p->epinfo = &epInfo;
|
||||||
|
|
||||||
|
p->lowspeed = lowspeed;
|
||||||
|
// Get device descriptor
|
||||||
|
rcode = getDevDescr(0, 0, sizeof (USB_DEVICE_DESCRIPTOR), (uint8_t*)buf);
|
||||||
|
|
||||||
|
// Restore p->epinfo
|
||||||
|
p->epinfo = oldep_ptr;
|
||||||
|
|
||||||
|
if (rcode) {
|
||||||
|
//printf("Configuring error: Can't get USB_DEVICE_DESCRIPTOR\r\n");
|
||||||
|
return rcode;
|
||||||
|
}
|
||||||
|
|
||||||
|
// to-do?
|
||||||
|
// Allocate new address according to device class
|
||||||
|
//bAddress = addrPool.AllocAddress(parent, false, port);
|
||||||
|
|
||||||
|
//if (!bAddress)
|
||||||
|
// return USB_ERROR_OUT_OF_ADDRESS_SPACE_IN_POOL;
|
||||||
|
|
||||||
|
uint16_t vid = (uint16_t)((USB_DEVICE_DESCRIPTOR*)buf)->idVendor;
|
||||||
|
uint16_t pid = (uint16_t)((USB_DEVICE_DESCRIPTOR*)buf)->idProduct;
|
||||||
|
uint8_t klass = ((USB_DEVICE_DESCRIPTOR*)buf)->bDeviceClass;
|
||||||
|
|
||||||
|
|
||||||
|
// Attempt to configure if VID/PID or device class matches with a driver
|
||||||
|
for (devConfigIndex = 0; devConfigIndex < USB_NUMDEVICES; devConfigIndex++) {
|
||||||
|
if (!devConfig[devConfigIndex]) continue; // no driver
|
||||||
|
if (devConfig[devConfigIndex]->GetAddress()) continue; // consumed
|
||||||
|
if (devConfig[devConfigIndex]->VIDPIDOK(vid, pid) || devConfig[devConfigIndex]->DEVCLASSOK(klass)) {
|
||||||
|
rcode = AttemptConfig(devConfigIndex, parent, port, lowspeed);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
rcode = devConfig[devConfigIndex]->Init(parent, port, lowspeed);
|
}
|
||||||
if (!rcode) {
|
|
||||||
|
if (devConfigIndex < USB_NUMDEVICES) {
|
||||||
|
if (rcode) {
|
||||||
|
//printf("Configuring error: %i\r\n", rcode);
|
||||||
devConfigIndex = 0;
|
devConfigIndex = 0;
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
return rcode;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// blindly attempt to configure
|
||||||
|
for (devConfigIndex = 0; devConfigIndex < USB_NUMDEVICES; devConfigIndex++) {
|
||||||
|
if (!devConfig[devConfigIndex]) continue;
|
||||||
|
if (devConfig[devConfigIndex]->GetAddress()) continue; // consumed
|
||||||
|
rcode = AttemptConfig(devConfigIndex, parent, port, lowspeed);
|
||||||
|
|
||||||
//printf("ERROR ENUMERATING %2.2x\r\n", rcode);
|
//printf("ERROR ENUMERATING %2.2x\r\n", rcode);
|
||||||
if (!(rcode == USB_DEV_CONFIG_ERROR_DEVICE_NOT_SUPPORTED || rcode == USB_ERROR_CLASS_INSTANCE_ALREADY_IN_USE)) {
|
if (!(rcode == USB_DEV_CONFIG_ERROR_DEVICE_NOT_SUPPORTED || rcode == USB_ERROR_CLASS_INSTANCE_ALREADY_IN_USE)) {
|
||||||
// in case of an error dev_index should be reset to 0
|
// in case of an error dev_index should be reset to 0
|
||||||
// in order to start from the very beginning the
|
// in order to start from the very beginning the
|
||||||
// next time the program gets here
|
// next time the program gets here
|
||||||
if (rcode != USB_DEV_CONFIG_ERROR_DEVICE_INIT_INCOMPLETE)
|
//if (rcode != USB_DEV_CONFIG_ERROR_DEVICE_INIT_INCOMPLETE)
|
||||||
devConfigIndex = 0;
|
// devConfigIndex = 0;
|
||||||
return rcode;
|
return rcode;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// if we get here that means that the device class is not supported by any of registered classes
|
// if we get here that means that the device class is not supported by any of registered classes
|
||||||
devConfigIndex = 0;
|
|
||||||
|
|
||||||
rcode = DefaultAddressing(parent, port, lowspeed);
|
rcode = DefaultAddressing(parent, port, lowspeed);
|
||||||
|
|
||||||
|
|
11
Usb.h
11
Usb.h
|
@ -141,12 +141,14 @@ typedef MAX3421e<P10, P9> MAX3421E; // Official Arduinos (UNO, Duemilanove, Mega
|
||||||
|
|
||||||
class USBDeviceConfig {
|
class USBDeviceConfig {
|
||||||
public:
|
public:
|
||||||
virtual uint8_t Init(uint8_t parent, uint8_t port, bool lowspeed) = 0;
|
virtual uint8_t Init(uint8_t parent, uint8_t port, bool lowspeed) { return 0; }
|
||||||
virtual uint8_t ConfigureDevice(uint8_t parent, uint8_t port, bool lowspeed) {return 0; }
|
virtual uint8_t ConfigureDevice(uint8_t parent, uint8_t port, bool lowspeed) {return 0; }
|
||||||
virtual uint8_t Release() = 0;
|
virtual uint8_t Release() { return 0; }
|
||||||
virtual uint8_t Poll() = 0;
|
virtual uint8_t Poll() { return 0; }
|
||||||
virtual uint8_t GetAddress() = 0;
|
virtual uint8_t GetAddress() { return 0; }
|
||||||
virtual void ResetHubPort(uint8_t port) { return; } // Note used for hubs only!
|
virtual void ResetHubPort(uint8_t port) { return; } // Note used for hubs only!
|
||||||
|
virtual boolean VIDPIDOK(uint16_t vid, uint16_t pid) { return false; }
|
||||||
|
virtual boolean DEVCLASSOK(uint8_t klass) { return false; }
|
||||||
};
|
};
|
||||||
|
|
||||||
/* USB Setup Packet Structure */
|
/* USB Setup Packet Structure */
|
||||||
|
@ -256,6 +258,7 @@ private:
|
||||||
uint8_t SetAddress(uint8_t addr, uint8_t ep, EpInfo **ppep, uint16_t &nak_limit);
|
uint8_t SetAddress(uint8_t addr, uint8_t ep, EpInfo **ppep, uint16_t &nak_limit);
|
||||||
uint8_t OutTransfer(EpInfo *pep, uint16_t nak_limit, uint16_t nbytes, uint8_t *data);
|
uint8_t OutTransfer(EpInfo *pep, uint16_t nak_limit, uint16_t nbytes, uint8_t *data);
|
||||||
uint8_t InTransfer(EpInfo *pep, uint16_t nak_limit, uint16_t *nbytesptr, uint8_t *data);
|
uint8_t InTransfer(EpInfo *pep, uint16_t nak_limit, uint16_t *nbytesptr, uint8_t *data);
|
||||||
|
uint8_t AttemptConfig(uint8_t driver, uint8_t parent, uint8_t port, bool lowspeed);
|
||||||
};
|
};
|
||||||
|
|
||||||
#if 0 //defined(USB_METHODS_INLINE)
|
#if 0 //defined(USB_METHODS_INLINE)
|
||||||
|
|
|
@ -35,7 +35,7 @@ public:
|
||||||
#define CP_MASK_COMPARE_CLASS 1
|
#define CP_MASK_COMPARE_CLASS 1
|
||||||
#define CP_MASK_COMPARE_SUBCLASS 2
|
#define CP_MASK_COMPARE_SUBCLASS 2
|
||||||
#define CP_MASK_COMPARE_PROTOCOL 4
|
#define CP_MASK_COMPARE_PROTOCOL 4
|
||||||
#define CP_MASK_COMPARE_ALL 7
|
#define CP_MASK_COMPARE_ALL 7
|
||||||
|
|
||||||
// Configuration Descriptor Parser Class Template
|
// Configuration Descriptor Parser Class Template
|
||||||
|
|
||||||
|
@ -104,10 +104,10 @@ bool ConfigDescParser<CLASS_ID, SUBCLASS_ID, PROTOCOL_ID, MASK>::ParseDescriptor
|
||||||
dscrType = *((uint8_t*) theBuffer.pValue + 1);
|
dscrType = *((uint8_t*) theBuffer.pValue + 1);
|
||||||
stateParseDescr = 2;
|
stateParseDescr = 2;
|
||||||
case 2:
|
case 2:
|
||||||
// This is a sort of hack. Assuming that two bytes are allready in the buffer
|
// This is a sort of hack. Assuming that two bytes are all ready in the buffer
|
||||||
// the pointer is positioned two bytes ahead in order for the rest of descriptor
|
// the pointer is positioned two bytes ahead in order for the rest of descriptor
|
||||||
// to be read right after the size and the type fields.
|
// to be read right after the size and the type fields.
|
||||||
// This should be used carefuly. varBuffer should be used directly to handle data
|
// This should be used carefully. varBuffer should be used directly to handle data
|
||||||
// in the buffer.
|
// in the buffer.
|
||||||
theBuffer.pValue = varBuffer + 2;
|
theBuffer.pValue = varBuffer + 2;
|
||||||
stateParseDescr = 3;
|
stateParseDescr = 3;
|
||||||
|
|
|
@ -277,7 +277,7 @@ uint8_t BulkOnly::ConfigureDevice(uint8_t parent, uint8_t port, bool lowspeed) {
|
||||||
EpInfo *oldep_ptr = NULL;
|
EpInfo *oldep_ptr = NULL;
|
||||||
USBTRACE("MS ConfigureDevice\r\n");
|
USBTRACE("MS ConfigureDevice\r\n");
|
||||||
ClearAllEP();
|
ClearAllEP();
|
||||||
delay(2000);
|
//delay(2000);
|
||||||
AddressPool &addrPool = pUsb->GetAddressPool();
|
AddressPool &addrPool = pUsb->GetAddressPool();
|
||||||
|
|
||||||
|
|
||||||
|
@ -409,7 +409,6 @@ uint8_t BulkOnly::Init(uint8_t parent, uint8_t port, bool lowspeed) {
|
||||||
// Assign epInfo to epinfo pointer
|
// Assign epInfo to epinfo pointer
|
||||||
pUsb->setEpInfoEntry(bAddress, bNumEP, epInfo);
|
pUsb->setEpInfoEntry(bAddress, bNumEP, epInfo);
|
||||||
|
|
||||||
USBTRACE("MS ConfigureDevice\r\n");
|
|
||||||
USBTRACE2("Conf:", bConfNum);
|
USBTRACE2("Conf:", bConfNum);
|
||||||
|
|
||||||
// Set Configuration Value
|
// Set Configuration Value
|
||||||
|
@ -528,6 +527,42 @@ Fail:
|
||||||
return rcode;
|
return rcode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* For driver use only.
|
||||||
|
*
|
||||||
|
* @param conf
|
||||||
|
* @param iface
|
||||||
|
* @param alt
|
||||||
|
* @param proto
|
||||||
|
* @param pep
|
||||||
|
*/
|
||||||
|
void BulkOnly::EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t proto, const USB_ENDPOINT_DESCRIPTOR * pep) {
|
||||||
|
ErrorMessage<uint8_t > (PSTR("Conf.Val"), conf);
|
||||||
|
ErrorMessage<uint8_t > (PSTR("Iface Num"), iface);
|
||||||
|
ErrorMessage<uint8_t > (PSTR("Alt.Set"), alt);
|
||||||
|
|
||||||
|
bConfNum = conf;
|
||||||
|
|
||||||
|
uint8_t index;
|
||||||
|
|
||||||
|
if ((pep->bmAttributes & 0x03) == 3 && (pep->bEndpointAddress & 0x80) == 0x80)
|
||||||
|
index = epInterruptInIndex;
|
||||||
|
else
|
||||||
|
if ((pep->bmAttributes & 0x02) == 2)
|
||||||
|
index = ((pep->bEndpointAddress & 0x80) == 0x80) ? epDataInIndex : epDataOutIndex;
|
||||||
|
else
|
||||||
|
return;
|
||||||
|
|
||||||
|
// Fill in the endpoint info structure
|
||||||
|
epInfo[index].epAddr = (pep->bEndpointAddress & 0x0F);
|
||||||
|
epInfo[index].maxPktSize = (uint8_t)pep->wMaxPacketSize;
|
||||||
|
epInfo[index].epAttribs = 0;
|
||||||
|
|
||||||
|
bNumEP++;
|
||||||
|
|
||||||
|
PrintEndpointDescriptor(pep);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* For driver use only.
|
* For driver use only.
|
||||||
*
|
*
|
||||||
|
@ -1256,42 +1291,6 @@ uint8_t BulkOnly::HandleSCSIError(uint8_t status) {
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* For driver use only.
|
|
||||||
*
|
|
||||||
* @param conf
|
|
||||||
* @param iface
|
|
||||||
* @param alt
|
|
||||||
* @param proto
|
|
||||||
* @param pep
|
|
||||||
*/
|
|
||||||
void BulkOnly::EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t proto, const USB_ENDPOINT_DESCRIPTOR * pep) {
|
|
||||||
ErrorMessage<uint8_t > (PSTR("Conf.Val"), conf);
|
|
||||||
ErrorMessage<uint8_t > (PSTR("Iface Num"), iface);
|
|
||||||
ErrorMessage<uint8_t > (PSTR("Alt.Set"), alt);
|
|
||||||
|
|
||||||
bConfNum = conf;
|
|
||||||
|
|
||||||
uint8_t index;
|
|
||||||
|
|
||||||
if ((pep->bmAttributes & 0x03) == 3 && (pep->bEndpointAddress & 0x80) == 0x80)
|
|
||||||
index = epInterruptInIndex;
|
|
||||||
else
|
|
||||||
if ((pep->bmAttributes & 0x02) == 2)
|
|
||||||
index = ((pep->bEndpointAddress & 0x80) == 0x80) ? epDataInIndex : epDataOutIndex;
|
|
||||||
else
|
|
||||||
return;
|
|
||||||
|
|
||||||
// Fill in the endpoint info structure
|
|
||||||
epInfo[index].epAddr = (pep->bEndpointAddress & 0x0F);
|
|
||||||
epInfo[index].maxPktSize = (uint8_t)pep->wMaxPacketSize;
|
|
||||||
epInfo[index].epAttribs = 0;
|
|
||||||
|
|
||||||
bNumEP++;
|
|
||||||
|
|
||||||
PrintEndpointDescriptor(pep);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* @param ep_ptr
|
* @param ep_ptr
|
||||||
|
|
|
@ -275,6 +275,8 @@ public:
|
||||||
|
|
||||||
// UsbConfigXtracter implementation
|
// UsbConfigXtracter implementation
|
||||||
virtual void EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t proto, const USB_ENDPOINT_DESCRIPTOR *ep);
|
virtual void EndpointXtract(uint8_t conf, uint8_t iface, uint8_t alt, uint8_t proto, const USB_ENDPOINT_DESCRIPTOR *ep);
|
||||||
|
virtual boolean DEVCLASSOK(uint8_t klass) { return (klass == USB_CLASS_MASS_STORAGE); }
|
||||||
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
uint8_t Inquiry(uint8_t lun, uint16_t size, uint8_t *buf);
|
uint8_t Inquiry(uint8_t lun, uint16_t size, uint8_t *buf);
|
||||||
|
|
72
usbhub.cpp
72
usbhub.cpp
|
@ -22,7 +22,7 @@ USBHub::USBHub(USB *p) :
|
||||||
pUsb(p),
|
pUsb(p),
|
||||||
bAddress(0),
|
bAddress(0),
|
||||||
bNbrPorts(0),
|
bNbrPorts(0),
|
||||||
bInitState(0),
|
//bInitState(0),
|
||||||
qNextPollTime(0),
|
qNextPollTime(0),
|
||||||
bPollEnable(false) {
|
bPollEnable(false) {
|
||||||
epInfo[0].epAddr = 0;
|
epInfo[0].epAddr = 0;
|
||||||
|
@ -47,12 +47,13 @@ uint8_t USBHub::Init(uint8_t parent, uint8_t port, bool lowspeed) {
|
||||||
uint8_t len = 0;
|
uint8_t len = 0;
|
||||||
uint16_t cd_len = 0;
|
uint16_t cd_len = 0;
|
||||||
|
|
||||||
//USBTRACE("\r\nHub Init Start");
|
//USBTRACE("\r\nHub Init Start ");
|
||||||
|
//D_PrintHex<uint8_t > (bInitState, 0x80);
|
||||||
|
|
||||||
AddressPool &addrPool = pUsb->GetAddressPool();
|
AddressPool &addrPool = pUsb->GetAddressPool();
|
||||||
|
|
||||||
switch (bInitState) {
|
//switch (bInitState) {
|
||||||
case 0:
|
// case 0:
|
||||||
if (bAddress)
|
if (bAddress)
|
||||||
return USB_ERROR_CLASS_INSTANCE_ALREADY_IN_USE;
|
return USB_ERROR_CLASS_INSTANCE_ALREADY_IN_USE;
|
||||||
|
|
||||||
|
@ -129,9 +130,9 @@ uint8_t USBHub::Init(uint8_t parent, uint8_t port, bool lowspeed) {
|
||||||
if (rcode)
|
if (rcode)
|
||||||
goto FailSetDevTblEntry;
|
goto FailSetDevTblEntry;
|
||||||
|
|
||||||
bInitState = 1;
|
// bInitState = 1;
|
||||||
|
|
||||||
case 1:
|
// case 1:
|
||||||
// Get hub descriptor
|
// Get hub descriptor
|
||||||
rcode = GetHubDescriptor(0, 8, buf);
|
rcode = GetHubDescriptor(0, 8, buf);
|
||||||
|
|
||||||
|
@ -141,9 +142,9 @@ uint8_t USBHub::Init(uint8_t parent, uint8_t port, bool lowspeed) {
|
||||||
// Save number of ports for future use
|
// Save number of ports for future use
|
||||||
bNbrPorts = ((HubDescriptor*)buf)->bNbrPorts;
|
bNbrPorts = ((HubDescriptor*)buf)->bNbrPorts;
|
||||||
|
|
||||||
bInitState = 2;
|
// bInitState = 2;
|
||||||
|
|
||||||
case 2:
|
// case 2:
|
||||||
// Read configuration Descriptor in Order To Obtain Proper Configuration Value
|
// Read configuration Descriptor in Order To Obtain Proper Configuration Value
|
||||||
rcode = pUsb->getConfDescr(bAddress, 0, 8, 0, buf);
|
rcode = pUsb->getConfDescr(bAddress, 0, 8, 0, buf);
|
||||||
|
|
||||||
|
@ -171,18 +172,19 @@ uint8_t USBHub::Init(uint8_t parent, uint8_t port, bool lowspeed) {
|
||||||
if (rcode)
|
if (rcode)
|
||||||
goto FailSetConfDescr;
|
goto FailSetConfDescr;
|
||||||
|
|
||||||
bInitState = 3;
|
// bInitState = 3;
|
||||||
|
|
||||||
case 3:
|
// case 3:
|
||||||
// Power on all ports
|
// Power on all ports
|
||||||
for (uint8_t j = 1; j <= bNbrPorts; j++)
|
for (uint8_t j = 1; j <= bNbrPorts; j++)
|
||||||
SetPortFeature(HUB_FEATURE_PORT_POWER, j, 0); //HubPortPowerOn(j);
|
SetPortFeature(HUB_FEATURE_PORT_POWER, j, 0); //HubPortPowerOn(j);
|
||||||
|
|
||||||
pUsb->SetHubPreMask();
|
pUsb->SetHubPreMask();
|
||||||
bPollEnable = true;
|
bPollEnable = true;
|
||||||
bInitState = 0;
|
// bInitState = 0;
|
||||||
}
|
//}
|
||||||
bInitState = 0;
|
//bInitState = 0;
|
||||||
|
//USBTRACE("...OK\r\n");
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
// Oleg, No debugging?? -- xxxajk
|
// Oleg, No debugging?? -- xxxajk
|
||||||
|
@ -202,6 +204,7 @@ FailSetConfDescr:
|
||||||
goto Fail;
|
goto Fail;
|
||||||
|
|
||||||
Fail:
|
Fail:
|
||||||
|
USBTRACE("...FAIL\r\n");
|
||||||
return rcode;
|
return rcode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -241,17 +244,17 @@ uint8_t USBHub::CheckHubStatus() {
|
||||||
if (rcode)
|
if (rcode)
|
||||||
return rcode;
|
return rcode;
|
||||||
|
|
||||||
if (buf[0] & 0x01) // Hub Status Change
|
//if (buf[0] & 0x01) // Hub Status Change
|
||||||
{
|
//{
|
||||||
//pUsb->PrintHubStatus(addr);
|
// pUsb->PrintHubStatus(addr);
|
||||||
//rcode = GetHubStatus(1, 0, 1, 4, buf);
|
// rcode = GetHubStatus(1, 0, 1, 4, buf);
|
||||||
//if (rcode)
|
// if (rcode)
|
||||||
//{
|
// {
|
||||||
// USB_HOST_SERIAL.print("GetHubStatus Error");
|
// USB_HOST_SERIAL.print("GetHubStatus Error");
|
||||||
// USB_HOST_SERIAL.println(rcode, HEX);
|
// USB_HOST_SERIAL.println(rcode, HEX);
|
||||||
// return rcode;
|
// return rcode;
|
||||||
//}
|
// }
|
||||||
}
|
//}
|
||||||
for (uint8_t port = 1, mask = 0x02; port < 8; mask <<= 1, port++) {
|
for (uint8_t port = 1, mask = 0x02; port < 8; mask <<= 1, port++) {
|
||||||
if (buf[0] & mask) {
|
if (buf[0] & mask) {
|
||||||
HubEvent evt;
|
HubEvent evt;
|
||||||
|
@ -308,18 +311,17 @@ void USBHub::ResetHubPort(uint8_t port) {
|
||||||
SetPortFeature(HUB_FEATURE_PORT_RESET, port, 0);
|
SetPortFeature(HUB_FEATURE_PORT_RESET, port, 0);
|
||||||
|
|
||||||
|
|
||||||
for(;;) {
|
for (int i = 0; i < 3; i++) {
|
||||||
rcode = GetPortStatus(port, 4, evt.evtBuff);
|
rcode = GetPortStatus(port, 4, evt.evtBuff);
|
||||||
if(rcode) return; // Some kind of error, bail.
|
if (rcode) break; // Some kind of error, bail.
|
||||||
rcode = evt.bmEvent;
|
if (evt.bmEvent == bmHUB_PORT_EVENT_RESET_COMPLETE || evt.bmEvent == bmHUB_PORT_EVENT_LS_RESET_COMPLETE) {
|
||||||
if (rcode == bmHUB_PORT_EVENT_RESET_COMPLETE || rcode == bmHUB_PORT_EVENT_LS_RESET_COMPLETE) {
|
break;
|
||||||
ClearPortFeature(HUB_FEATURE_C_PORT_RESET, port, 0);
|
}
|
||||||
ClearPortFeature(HUB_FEATURE_C_PORT_CONNECTION, port, 0);
|
delay(100); // simulate polling.
|
||||||
delay(20);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
delay(100); // simulate polling.
|
|
||||||
}
|
}
|
||||||
|
ClearPortFeature(HUB_FEATURE_C_PORT_RESET, port, 0);
|
||||||
|
ClearPortFeature(HUB_FEATURE_C_PORT_CONNECTION, port, 0);
|
||||||
|
delay(20);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t USBHub::PortStatusChange(uint8_t port, HubEvent &evt) {
|
uint8_t USBHub::PortStatusChange(uint8_t port, HubEvent &evt) {
|
||||||
|
|
4
usbhub.h
4
usbhub.h
|
@ -183,7 +183,7 @@ class USBHub : USBDeviceConfig {
|
||||||
|
|
||||||
uint8_t bAddress; // address
|
uint8_t bAddress; // address
|
||||||
uint8_t bNbrPorts; // number of ports
|
uint8_t bNbrPorts; // number of ports
|
||||||
uint8_t bInitState; // initialization state variable
|
// uint8_t bInitState; // initialization state variable
|
||||||
uint32_t qNextPollTime; // next poll time
|
uint32_t qNextPollTime; // next poll time
|
||||||
bool bPollEnable; // poll enable flag
|
bool bPollEnable; // poll enable flag
|
||||||
|
|
||||||
|
@ -211,6 +211,8 @@ public:
|
||||||
virtual uint8_t GetAddress() {
|
virtual uint8_t GetAddress() {
|
||||||
return bAddress;
|
return bAddress;
|
||||||
};
|
};
|
||||||
|
virtual boolean DEVCLASSOK(uint8_t klass) { return (klass == 0x09); }
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// Clear Hub Feature
|
// Clear Hub Feature
|
||||||
|
|
Loading…
Reference in a new issue