pi1541/uspi/lib/usbstandardhub.c
2018-05-20 14:53:34 +10:00

473 lines
12 KiB
C

//
// usbstandardhub.c
//
// USPi - An USB driver for Raspberry Pi written in C
// Copyright (C) 2014 R. Stange <rsta2@o2online.de>
//
// This program 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.
//
// This program 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/>.
//
#include <uspi/usbstandardhub.h>
#include <uspi/usbdevicefactory.h>
#include <uspios.h>
#include <uspi/macros.h>
#include <uspi/assert.h>
boolean USBStandardHubEnumeratePorts (TUSBStandardHub *pThis);
static const char FromHub[] = "usbhub";
void USBStandardHub (TUSBStandardHub *pThis, TUSBDevice *pDevice)
{
assert (pThis != 0);
USBDeviceCopy (&pThis->m_USBDevice, pDevice);
pThis->m_USBDevice.Configure = USBStandardHubConfigure;
pThis->m_pHubDesc = 0;
pThis->m_nPorts = 0;
for (unsigned nPort = 0; nPort < USB_HUB_MAX_PORTS; nPort++)
{
pThis->m_pDevice[nPort] = 0;
pThis->m_pStatus[nPort] = 0;
}
}
void _USBStandardHub (TUSBStandardHub *pThis)
{
assert (pThis != 0);
for (unsigned nPort = 0; nPort < pThis->m_nPorts; nPort++)
{
if (pThis->m_pStatus[nPort] != 0)
{
free (pThis->m_pStatus[nPort]);
pThis->m_pStatus[nPort] = 0;
}
if (pThis->m_pDevice[nPort] != 0)
{
_USBDevice (pThis->m_pDevice[nPort]);
free (pThis->m_pDevice[nPort]);
pThis->m_pDevice[nPort] = 0;
}
}
pThis->m_nPorts = 0;
if (pThis->m_pHubDesc != 0)
{
free (pThis->m_pHubDesc);
pThis->m_pHubDesc = 0;
}
_USBDevice (&pThis->m_USBDevice);
}
boolean USBStandardHubInitialize (TUSBStandardHub *pThis)
{
assert (pThis != 0);
return USBDeviceInitialize (&pThis->m_USBDevice);
}
boolean USBStandardHubConfigure (TUSBDevice *pUSBDevice)
{
TUSBStandardHub *pThis = (TUSBStandardHub *) pUSBDevice;
assert (pThis != 0);
const TUSBDeviceDescriptor *pDeviceDesc = USBDeviceGetDeviceDescriptor (&pThis->m_USBDevice);
assert (pDeviceDesc != 0);
if ( pDeviceDesc->bDeviceClass != USB_DEVICE_CLASS_HUB
|| pDeviceDesc->bDeviceSubClass != 0
|| pDeviceDesc->bDeviceProtocol != 2 // hub with multiple TTs
|| pDeviceDesc->bNumConfigurations != 1)
{
LogWrite (FromHub, LOG_ERROR, "Unsupported hub (proto %u)",
(unsigned) pDeviceDesc->bDeviceProtocol);
return FALSE;
}
const TUSBConfigurationDescriptor *pConfigDesc =
(TUSBConfigurationDescriptor *) USBDeviceGetDescriptor (&pThis->m_USBDevice, DESCRIPTOR_CONFIGURATION);
if ( pConfigDesc == 0
|| pConfigDesc->bNumInterfaces != 1)
{
USBDeviceConfigurationError (&pThis->m_USBDevice, FromHub);
return FALSE;
}
const TUSBInterfaceDescriptor *pInterfaceDesc;
while ((pInterfaceDesc = (TUSBInterfaceDescriptor *) USBDeviceGetDescriptor (&pThis->m_USBDevice, DESCRIPTOR_INTERFACE)) != 0)
{
if ( pInterfaceDesc->bInterfaceClass != USB_DEVICE_CLASS_HUB
|| pInterfaceDesc->bInterfaceSubClass != 0
|| pInterfaceDesc->bInterfaceProtocol != 2)
{
continue;
}
if (pInterfaceDesc->bNumEndpoints != 1)
{
USBDeviceConfigurationError (&pThis->m_USBDevice, FromHub);
return FALSE;
}
const TUSBEndpointDescriptor *pEndpointDesc =
(TUSBEndpointDescriptor *) USBDeviceGetDescriptor (&pThis->m_USBDevice, DESCRIPTOR_ENDPOINT);
if ( pEndpointDesc == 0
|| (pEndpointDesc->bEndpointAddress & 0x80) != 0x80 // input EP
|| (pEndpointDesc->bmAttributes & 0x3F) != 0x03) // interrupt EP
{
USBDeviceConfigurationError (&pThis->m_USBDevice, FromHub);
return FALSE;
}
break;
}
if (pInterfaceDesc == 0)
{
USBDeviceConfigurationError (&pThis->m_USBDevice, FromHub);
return FALSE;
}
if (!USBDeviceConfigure (&pThis->m_USBDevice))
{
LogWrite (FromHub, LOG_ERROR, "Cannot set configuration");
return FALSE;
}
TUSBHostController *pHost = USBDeviceGetHost (&pThis->m_USBDevice);
assert (pHost != 0);
if (pInterfaceDesc->bAlternateSetting != 0)
{
if (DWHCIDeviceControlMessage (pHost, USBDeviceGetEndpoint0 (&pThis->m_USBDevice),
REQUEST_OUT | REQUEST_TO_INTERFACE, SET_INTERFACE,
pInterfaceDesc->bAlternateSetting,
pInterfaceDesc->bInterfaceNumber, 0, 0) < 0)
{
LogWrite (FromHub, LOG_ERROR, "Cannot set interface");
return FALSE;
}
}
assert (pThis->m_pHubDesc == 0);
pThis->m_pHubDesc = (TUSBHubDescriptor *) malloc (sizeof (TUSBHubDescriptor));
assert (pThis->m_pHubDesc != 0);
if (DWHCIDeviceGetDescriptor (pHost, USBDeviceGetEndpoint0 (&pThis->m_USBDevice),
DESCRIPTOR_HUB, DESCRIPTOR_INDEX_DEFAULT,
pThis->m_pHubDesc, sizeof *pThis->m_pHubDesc,
REQUEST_IN | REQUEST_CLASS)
!= (int) sizeof *pThis->m_pHubDesc)
{
LogWrite (FromHub, LOG_ERROR, "Cannot get hub descriptor");
free (pThis->m_pHubDesc);
pThis->m_pHubDesc = 0;
return FALSE;
}
#ifndef NDEBUG
//DebugHexdump (pThis->m_pHubDesc, sizeof *pThis->m_pHubDesc, FromHub);
#endif
pThis->m_nPorts = pThis->m_pHubDesc->bNbrPorts;
if (pThis->m_nPorts > USB_HUB_MAX_PORTS)
{
LogWrite (FromHub, LOG_ERROR, "Too many ports (%u)", pThis->m_nPorts);
free (pThis->m_pHubDesc);
pThis->m_pHubDesc = 0;
return FALSE;
}
if (!USBStandardHubEnumeratePorts (pThis))
{
LogWrite (FromHub, LOG_ERROR, "Port enumeration failed");
return FALSE;
}
return TRUE;
}
boolean USBStandardHubEnumeratePorts (TUSBStandardHub *pThis)
{
assert (pThis != 0);
TUSBHostController *pHost = USBDeviceGetHost (&pThis->m_USBDevice);
assert (pHost != 0);
TUSBEndpoint *pEndpoint0 = USBDeviceGetEndpoint0 (&pThis->m_USBDevice);
assert (pEndpoint0 != 0);
assert (pThis->m_nPorts > 0);
// first power on all ports
for (unsigned nPort = 0; nPort < pThis->m_nPorts; nPort++)
{
if (DWHCIDeviceControlMessage (pHost, pEndpoint0,
REQUEST_OUT | REQUEST_CLASS | REQUEST_TO_OTHER,
SET_FEATURE, PORT_POWER, nPort+1, 0, 0) < 0)
{
LogWrite (FromHub, LOG_ERROR, "Cannot power port %u", nPort+1);
return FALSE;
}
}
// pThis->m_pHubDesc->bPwrOn2PwrGood delay seems to be not enough
// for some low speed devices, so we use the maximum here
MsDelay (510);
// now detect devices, reset and initialize them
for (unsigned nPort = 0; nPort < pThis->m_nPorts; nPort++)
{
assert (pThis->m_pStatus[nPort] == 0);
pThis->m_pStatus[nPort] = malloc (sizeof (TUSBPortStatus));
assert (pThis->m_pStatus[nPort] != 0);
if (DWHCIDeviceControlMessage (pHost, pEndpoint0,
REQUEST_IN | REQUEST_CLASS | REQUEST_TO_OTHER,
GET_STATUS, 0, nPort+1, pThis->m_pStatus[nPort], 4) != 4)
{
LogWrite (FromHub, LOG_ERROR, "Cannot get status of port %u", nPort+1);
continue;
}
assert (pThis->m_pStatus[nPort]->wPortStatus & PORT_POWER__MASK);
if (!(pThis->m_pStatus[nPort]->wPortStatus & PORT_CONNECTION__MASK))
{
continue;
}
if (DWHCIDeviceControlMessage (pHost, pEndpoint0,
REQUEST_OUT | REQUEST_CLASS | REQUEST_TO_OTHER,
SET_FEATURE, PORT_RESET, nPort+1, 0, 0) < 0)
{
LogWrite (FromHub, LOG_ERROR, "Cannot reset port %u", nPort+1);
continue;
}
MsDelay (100);
if (DWHCIDeviceControlMessage (pHost, pEndpoint0,
REQUEST_IN | REQUEST_CLASS | REQUEST_TO_OTHER,
GET_STATUS, 0, nPort+1, pThis->m_pStatus[nPort], 4) != 4)
{
return FALSE;
}
//LogWrite (FromHub, LOG_DEBUG, "Port %u status is 0x%04X", nPort+1, (unsigned) pThis->m_pStatus[nPort]->wPortStatus);
if (!(pThis->m_pStatus[nPort]->wPortStatus & PORT_ENABLE__MASK))
{
LogWrite (FromHub, LOG_ERROR, "Port %u is not enabled", nPort+1);
continue;
}
// check for over-current
if (pThis->m_pStatus[nPort]->wPortStatus & PORT_OVER_CURRENT__MASK)
{
DWHCIDeviceControlMessage (pHost, pEndpoint0,
REQUEST_OUT | REQUEST_CLASS | REQUEST_TO_OTHER,
CLEAR_FEATURE, PORT_POWER, nPort+1, 0, 0);
LogWrite (FromHub, LOG_ERROR, "Over-current condition on port %u", nPort+1);
return FALSE;
}
TUSBSpeed Speed = USBSpeedUnknown;
if (pThis->m_pStatus[nPort]->wPortStatus & PORT_LOW_SPEED__MASK)
{
Speed = USBSpeedLow;
}
else if (pThis->m_pStatus[nPort]->wPortStatus & PORT_HIGH_SPEED__MASK)
{
Speed = USBSpeedHigh;
}
else
{
Speed = USBSpeedFull;
}
// first create default device
assert (pThis->m_pDevice[nPort] == 0);
pThis->m_pDevice[nPort] = malloc (sizeof (TUSBDevice));
assert (pThis->m_pDevice[nPort] != 0);
USBDevice (pThis->m_pDevice[nPort], pHost, Speed, USBDeviceGetAddress (&pThis->m_USBDevice), nPort+1);
if (!USBDeviceInitialize (pThis->m_pDevice[nPort]))
{
_USBDevice (pThis->m_pDevice[nPort]);
free (pThis->m_pDevice[nPort]);
pThis->m_pDevice[nPort] = 0;
continue;
}
TString *pNames = USBStandardHubGetDeviceNames (pThis->m_pDevice[nPort]);
assert (pNames != 0);
LogWrite (FromHub, LOG_NOTICE, "Port %u: Device %s found", nPort+1, StringGet (pNames));
_String (pNames);
free (pNames);
}
// now configure devices
for (unsigned nPort = 0; nPort < pThis->m_nPorts; nPort++)
{
if (pThis->m_pDevice[nPort] == 0)
{
continue;
}
// now create specific device from default device
TUSBDevice *pChild = USBDeviceFactoryGetDevice (pThis->m_pDevice[nPort]);
if (pChild != 0)
{
_USBDevice (pThis->m_pDevice[nPort]); // delete default device
free (pThis->m_pDevice[nPort]);
pThis->m_pDevice[nPort] = pChild; // assign specific device
if (!(*pThis->m_pDevice[nPort]->Configure) (pThis->m_pDevice[nPort]))
{
LogWrite (FromHub, LOG_ERROR, "Port %u: Cannot configure device", nPort+1);
continue;
}
LogWrite (FromHub, LOG_DEBUG, "Port %u: Device configured", nPort+1);
}
else
{
LogWrite (FromHub, LOG_NOTICE, "Port %u: Device is not supported", nPort+1);
_USBDevice (pThis->m_pDevice[nPort]);
free (pThis->m_pDevice[nPort]);
pThis->m_pDevice[nPort] = 0;
}
}
// again check for over-current
TUSBHubStatus *pHubStatus = malloc (sizeof (TUSBHubStatus));
assert (pHubStatus != 0);
if (DWHCIDeviceControlMessage (pHost, pEndpoint0,
REQUEST_IN | REQUEST_CLASS,
GET_STATUS, 0, 0, pHubStatus, sizeof *pHubStatus) != (int) sizeof *pHubStatus)
{
LogWrite (FromHub, LOG_ERROR, "Cannot get hub status");
free (pHubStatus);
return FALSE;
}
if (pHubStatus->wHubStatus & HUB_OVER_CURRENT__MASK)
{
for (unsigned nPort = 0; nPort < pThis->m_nPorts; nPort++)
{
DWHCIDeviceControlMessage (pHost, pEndpoint0,
REQUEST_OUT | REQUEST_CLASS | REQUEST_TO_OTHER,
CLEAR_FEATURE, PORT_POWER, nPort+1, 0, 0);
}
LogWrite (FromHub, LOG_ERROR, "Hub over-current condition");
free (pHubStatus);
return FALSE;
}
free (pHubStatus);
pHubStatus = 0;
boolean bResult = TRUE;
for (unsigned nPort = 0; nPort < pThis->m_nPorts; nPort++)
{
if (DWHCIDeviceControlMessage (pHost, pEndpoint0,
REQUEST_IN | REQUEST_CLASS | REQUEST_TO_OTHER,
GET_STATUS, 0, nPort+1, pThis->m_pStatus[nPort], 4) != 4)
{
continue;
}
if (pThis->m_pStatus[nPort]->wPortStatus & PORT_OVER_CURRENT__MASK)
{
DWHCIDeviceControlMessage (pHost, pEndpoint0,
REQUEST_OUT | REQUEST_CLASS | REQUEST_TO_OTHER,
CLEAR_FEATURE, PORT_POWER, nPort+1, 0, 0);
LogWrite (FromHub, LOG_ERROR, "Over-current condition on port %u", nPort+1);
bResult = FALSE;
}
}
return bResult;
}
TString *USBStandardHubGetDeviceNames (TUSBDevice *pDevice)
{
assert (pDevice != 0);
TString *pResult = (TString *) malloc (sizeof (TString));
assert (pResult != 0);
String (pResult);
for (unsigned nSelector = DeviceNameVendor; nSelector < DeviceNameUnknown; nSelector++)
{
TString *pName = USBDeviceGetName (pDevice, (TDeviceNameSelector) nSelector);
assert (pName != 0);
if (StringCompare (pName, "unknown") != 0)
{
if (StringGetLength (pResult) > 0)
{
StringAppend (pResult, ", ");
}
StringAppend (pResult, StringGet (pName));
}
_String (pName);
free (pName);
}
if (StringGetLength (pResult) == 0)
{
StringSet (pResult, "unknown");
}
return pResult;
}