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

136 lines
3.3 KiB
C

//
// dwhcirootport.cpp
//
// 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/dwhcirootport.h>
#include <uspi/dwhcidevice.h>
#include <uspi/usbdevicefactory.h>
#include <uspi/usbstandardhub.h>
#include <uspi/assert.h>
#include <uspios.h>
static const char FromDWHCIRoot[] = "dwroot";
void DWHCIRootPort (TDWHCIRootPort *pThis, struct TDWHCIDevice *pHost)
{
assert (pThis != 0);
pThis->m_pHost = pHost;
pThis->m_pDevice = 0;
assert (pThis->m_pHost != 0);
}
void _DWHCIRootPort (TDWHCIRootPort *pThis)
{
assert (pThis != 0);
if (pThis->m_pDevice != 0)
{
_USBDevice (pThis->m_pDevice);
free (pThis->m_pDevice);
pThis->m_pDevice = 0;
}
pThis->m_pHost = 0;
}
boolean DWHCIRootPortInitialize (TDWHCIRootPort *pThis)
{
assert (pThis != 0);
assert (pThis->m_pHost != 0);
TUSBSpeed Speed = DWHCIDeviceGetPortSpeed (pThis->m_pHost);
if (Speed == USBSpeedUnknown)
{
LogWrite (FromDWHCIRoot, LOG_ERROR, "Cannot detect port speed");
return FALSE;
}
// first create default device
assert (pThis->m_pDevice == 0);
pThis->m_pDevice = (TUSBDevice *) malloc (sizeof (TUSBDevice));
assert (pThis->m_pDevice != 0);
USBDevice (pThis->m_pDevice, pThis->m_pHost, Speed, 0, 1);
if (!USBDeviceInitialize (pThis->m_pDevice))
{
_USBDevice (pThis->m_pDevice);
free (pThis->m_pDevice);
pThis->m_pDevice = 0;
return FALSE;
}
TString *pNames = USBStandardHubGetDeviceNames (pThis->m_pDevice);
assert (pNames != 0);
LogWrite (FromDWHCIRoot, LOG_NOTICE, "Device %s found", StringGet (pNames));
_String (pNames);
free (pNames);
// now create specific device from default device
TUSBDevice *pChild = USBDeviceFactoryGetDevice (pThis->m_pDevice);
if (pChild != 0)
{
_USBDevice (pThis->m_pDevice); // delete default device
free (pThis->m_pDevice);
pThis->m_pDevice = pChild; // assign specific device
if (!(*pThis->m_pDevice->Configure) (pThis->m_pDevice))
{
LogWrite (FromDWHCIRoot, LOG_ERROR, "Cannot configure device");
_USBDevice (pThis->m_pDevice);
free (pThis->m_pDevice);
pThis->m_pDevice = 0;
return FALSE;
}
LogWrite (FromDWHCIRoot, LOG_DEBUG, "Device configured");
}
else
{
LogWrite (FromDWHCIRoot, LOG_NOTICE, "Device is not supported");
_USBDevice (pThis->m_pDevice);
free (pThis->m_pDevice);
pThis->m_pDevice = 0;
return FALSE;
}
// check for over-current
if (DWHCIDeviceOvercurrentDetected (pThis->m_pHost))
{
LogWrite (FromDWHCIRoot, LOG_ERROR, "Over-current condition");
DWHCIDeviceDisableRootPort (pThis->m_pHost);
_USBDevice (pThis->m_pDevice);
free (pThis->m_pDevice);
pThis->m_pDevice = 0;
return FALSE;
}
return TRUE;
}