250 lines
6.2 KiB
C
250 lines
6.2 KiB
C
|
//
|
||
|
// usbstring.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/usbstring.h>
|
||
|
#include <uspi/usbdevice.h>
|
||
|
#include <uspi/dwhcidevice.h>
|
||
|
#include <uspi/util.h>
|
||
|
#include <uspi/assert.h>
|
||
|
#include <uspios.h>
|
||
|
|
||
|
#define USBSTR_MIN_LENGTH 4
|
||
|
|
||
|
#define USBSTR_DEFAULT_LANGID 0x409
|
||
|
|
||
|
void USBString (TUSBString *pThis, struct TUSBDevice *pDevice)
|
||
|
{
|
||
|
assert (pThis != 0);
|
||
|
|
||
|
pThis->m_pDevice = pDevice;
|
||
|
assert (pThis->m_pDevice != 0);
|
||
|
|
||
|
pThis->m_pUSBString = 0;
|
||
|
|
||
|
pThis->m_pString = malloc (sizeof (TString));
|
||
|
assert (pThis->m_pString != 0);
|
||
|
String (pThis->m_pString);
|
||
|
}
|
||
|
|
||
|
void USBStringCopy (TUSBString *pThis, TUSBString *pParent)
|
||
|
{
|
||
|
assert (pThis != 0);
|
||
|
assert (pParent != 0);
|
||
|
pThis->m_pDevice = pParent->m_pDevice;
|
||
|
|
||
|
pThis->m_pUSBString = 0;
|
||
|
if (pParent->m_pUSBString != 0)
|
||
|
{
|
||
|
pThis->m_pUSBString = (TUSBStringDescriptor *) malloc (pParent->m_pUSBString->bLength);
|
||
|
assert (pThis->m_pUSBString != 0);
|
||
|
memcpy (pThis->m_pUSBString, pParent->m_pUSBString, pParent->m_pUSBString->bLength);
|
||
|
}
|
||
|
|
||
|
assert (pParent->m_pString != 0);
|
||
|
pThis->m_pString = (TString *) malloc (sizeof (TString));
|
||
|
assert (pThis->m_pString != 0);
|
||
|
String2 (pThis->m_pString, StringGet (pParent->m_pString));
|
||
|
}
|
||
|
|
||
|
void _USBString (TUSBString *pThis)
|
||
|
{
|
||
|
assert (pThis != 0);
|
||
|
|
||
|
assert (pThis->m_pString != 0);
|
||
|
_String (pThis->m_pString);
|
||
|
free (pThis->m_pString);
|
||
|
pThis->m_pString = 0;
|
||
|
|
||
|
if (pThis->m_pUSBString != 0)
|
||
|
{
|
||
|
free (pThis->m_pUSBString);
|
||
|
pThis->m_pUSBString = 0;
|
||
|
}
|
||
|
|
||
|
pThis->m_pDevice = 0;
|
||
|
}
|
||
|
|
||
|
boolean USBStringGetFromDescriptor (TUSBString *pThis, u8 ucID, u16 usLanguageID)
|
||
|
{
|
||
|
assert (pThis != 0);
|
||
|
assert (ucID > 0);
|
||
|
|
||
|
if (pThis->m_pUSBString != 0)
|
||
|
{
|
||
|
free (pThis->m_pUSBString);
|
||
|
}
|
||
|
pThis->m_pUSBString = (TUSBStringDescriptor *) malloc (USBSTR_MIN_LENGTH);
|
||
|
assert (pThis->m_pUSBString != 0);
|
||
|
|
||
|
assert (pThis->m_pDevice != 0);
|
||
|
if (DWHCIDeviceControlMessage (USBDeviceGetHost (pThis->m_pDevice),
|
||
|
USBDeviceGetEndpoint0 (pThis->m_pDevice),
|
||
|
REQUEST_IN, GET_DESCRIPTOR,
|
||
|
(DESCRIPTOR_STRING << 8) | ucID, usLanguageID,
|
||
|
pThis->m_pUSBString, USBSTR_MIN_LENGTH) < 0)
|
||
|
{
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
u8 ucLength = pThis->m_pUSBString->bLength;
|
||
|
if ( ucLength < 2
|
||
|
|| (ucLength & 1) != 0
|
||
|
|| pThis->m_pUSBString->bDescriptorType != DESCRIPTOR_STRING)
|
||
|
{
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (ucLength > USBSTR_MIN_LENGTH)
|
||
|
{
|
||
|
free (pThis->m_pUSBString);
|
||
|
pThis->m_pUSBString = (TUSBStringDescriptor *) malloc (ucLength);
|
||
|
assert (pThis->m_pUSBString != 0);
|
||
|
|
||
|
if (DWHCIDeviceControlMessage (USBDeviceGetHost (pThis->m_pDevice),
|
||
|
USBDeviceGetEndpoint0 (pThis->m_pDevice),
|
||
|
REQUEST_IN, GET_DESCRIPTOR,
|
||
|
(DESCRIPTOR_STRING << 8) | ucID, usLanguageID,
|
||
|
pThis->m_pUSBString, ucLength) != (int) ucLength)
|
||
|
{
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if ( pThis->m_pUSBString->bLength != ucLength
|
||
|
|| (pThis->m_pUSBString->bLength & 1) != 0
|
||
|
|| pThis->m_pUSBString->bDescriptorType != DESCRIPTOR_STRING)
|
||
|
{
|
||
|
return FALSE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// convert to ASCII string
|
||
|
assert (pThis->m_pUSBString->bLength >= 2);
|
||
|
assert ((pThis->m_pUSBString->bLength & 1) == 0);
|
||
|
size_t nLength = (pThis->m_pUSBString->bLength-2) / 2;
|
||
|
|
||
|
assert (nLength <= (255-2) / 2);
|
||
|
char Buffer[nLength+1];
|
||
|
|
||
|
for (unsigned i = 0; i < nLength; i++)
|
||
|
{
|
||
|
u16 usChar = pThis->m_pUSBString->bString[i];
|
||
|
if ( usChar < ' '
|
||
|
|| usChar > '~')
|
||
|
{
|
||
|
usChar = '_';
|
||
|
}
|
||
|
|
||
|
Buffer[i] = (char) usChar;
|
||
|
}
|
||
|
Buffer[nLength] = '\0';
|
||
|
|
||
|
assert (pThis->m_pString != 0);
|
||
|
_String (pThis->m_pString);
|
||
|
free (pThis->m_pString);
|
||
|
|
||
|
pThis->m_pString = malloc (sizeof (TString));
|
||
|
assert (pThis->m_pString != 0);
|
||
|
String2 (pThis->m_pString, Buffer);
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
const char *USBStringGet (TUSBString *pThis)
|
||
|
{
|
||
|
assert (pThis != 0);
|
||
|
return StringGet (pThis->m_pString);
|
||
|
}
|
||
|
|
||
|
u16 USBStringGetLanguageID (TUSBString *pThis)
|
||
|
{
|
||
|
assert (pThis != 0);
|
||
|
|
||
|
TUSBStringDescriptor *pLanguageIDs = (TUSBStringDescriptor *) malloc (USBSTR_MIN_LENGTH);
|
||
|
assert (pLanguageIDs != 0);
|
||
|
|
||
|
assert (pThis->m_pDevice != 0);
|
||
|
if (DWHCIDeviceGetDescriptor (USBDeviceGetHost (pThis->m_pDevice),
|
||
|
USBDeviceGetEndpoint0 (pThis->m_pDevice),
|
||
|
DESCRIPTOR_STRING, 0,
|
||
|
pLanguageIDs, USBSTR_MIN_LENGTH, REQUEST_IN) < 0)
|
||
|
{
|
||
|
free (pLanguageIDs);
|
||
|
|
||
|
return USBSTR_DEFAULT_LANGID;
|
||
|
}
|
||
|
|
||
|
u8 ucLength = pLanguageIDs->bLength;
|
||
|
if ( ucLength < 4
|
||
|
|| (ucLength & 1) != 0
|
||
|
|| pLanguageIDs->bDescriptorType != DESCRIPTOR_STRING)
|
||
|
{
|
||
|
free (pLanguageIDs);
|
||
|
|
||
|
return USBSTR_DEFAULT_LANGID;
|
||
|
}
|
||
|
|
||
|
if (ucLength > USBSTR_MIN_LENGTH)
|
||
|
{
|
||
|
free (pLanguageIDs);
|
||
|
pLanguageIDs = (TUSBStringDescriptor *) malloc (ucLength);
|
||
|
assert (pLanguageIDs != 0);
|
||
|
|
||
|
if (DWHCIDeviceGetDescriptor (USBDeviceGetHost (pThis->m_pDevice),
|
||
|
USBDeviceGetEndpoint0 (pThis->m_pDevice),
|
||
|
DESCRIPTOR_STRING, 0,
|
||
|
pLanguageIDs, ucLength, REQUEST_IN) != (int) ucLength)
|
||
|
{
|
||
|
free (pLanguageIDs);
|
||
|
|
||
|
return USBSTR_DEFAULT_LANGID;
|
||
|
}
|
||
|
|
||
|
if ( pLanguageIDs->bLength != ucLength
|
||
|
|| (pLanguageIDs->bLength & 1) != 0
|
||
|
|| pLanguageIDs->bDescriptorType != DESCRIPTOR_STRING)
|
||
|
{
|
||
|
free (pLanguageIDs);
|
||
|
|
||
|
return USBSTR_DEFAULT_LANGID;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
assert (pLanguageIDs->bLength >= 4);
|
||
|
assert ((pLanguageIDs->bLength & 1) == 0);
|
||
|
size_t nLength = (pLanguageIDs->bLength-2) / 2;
|
||
|
|
||
|
// search for default language ID
|
||
|
for (unsigned i = 0; i < nLength; i++)
|
||
|
{
|
||
|
if (pLanguageIDs->bString[i] == USBSTR_DEFAULT_LANGID)
|
||
|
{
|
||
|
free (pLanguageIDs);
|
||
|
|
||
|
return USBSTR_DEFAULT_LANGID;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// default language ID not found, use first ID
|
||
|
u16 usResult = pLanguageIDs->bString[0];
|
||
|
|
||
|
free (pLanguageIDs);
|
||
|
|
||
|
return usResult;
|
||
|
}
|