2009-02-23 08:08:22 +01:00
|
|
|
/*
|
|
|
|
LUFA Library
|
|
|
|
Copyright (C) Dean Camera, 2009.
|
|
|
|
|
|
|
|
dean [at] fourwalledcubicle [dot] com
|
|
|
|
www.fourwalledcubicle.com
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
Copyright 2009 Dean Camera (dean [at] fourwalledcubicle [dot] com)
|
|
|
|
|
|
|
|
Permission to use, copy, modify, and distribute this software
|
|
|
|
and its documentation for any purpose and without fee is hereby
|
|
|
|
granted, provided that the above copyright notice appear in all
|
|
|
|
copies and that both that the copyright notice and this
|
|
|
|
permission notice and warranty disclaimer appear in supporting
|
|
|
|
documentation, and that the name of the author not be used in
|
|
|
|
advertising or publicity pertaining to distribution of the
|
|
|
|
software without specific, written prior permission.
|
|
|
|
|
|
|
|
The author disclaim all warranties with regard to this
|
|
|
|
software, including all implied warranties of merchantability
|
|
|
|
and fitness. In no event shall the author be liable for any
|
|
|
|
special, indirect or consequential damages or any damages
|
|
|
|
whatsoever resulting from loss of use, data or profits, whether
|
|
|
|
in an action of contract, negligence or other tortious action,
|
|
|
|
arising out of or in connection with the use or performance of
|
|
|
|
this software.
|
|
|
|
*/
|
|
|
|
|
2009-06-15 10:58:55 +02:00
|
|
|
#include "../../HighLevel/USBMode.h"
|
|
|
|
#if defined(USB_CAN_BE_HOST)
|
|
|
|
|
2009-02-23 08:08:22 +01:00
|
|
|
#include "HIDParser.h"
|
|
|
|
|
2009-04-19 13:43:21 +02:00
|
|
|
uint8_t USB_ProcessHIDReport(const uint8_t* ReportData, uint16_t ReportSize, HID_ReportInfo_t* const ParserData)
|
2009-02-23 08:08:22 +01:00
|
|
|
{
|
2009-09-02 09:16:52 +02:00
|
|
|
HID_StateTable_t StateTable[HID_STATETABLE_STACK_DEPTH];
|
|
|
|
HID_StateTable_t* CurrStateTable = &StateTable[0];
|
2009-09-09 06:18:37 +02:00
|
|
|
HID_CollectionPath_t* CurrCollectionPath = NULL;
|
2009-09-09 10:34:24 +02:00
|
|
|
HID_ReportSizeInfo_t* CurrReportIDInfo = &ParserData->ReportIDSizes[0];
|
2009-11-04 14:16:53 +01:00
|
|
|
uint16_t UsageList[HID_USAGE_STACK_DEPTH];
|
2009-12-04 02:06:26 +01:00
|
|
|
uint8_t UsageListSize = 0;
|
|
|
|
HID_MinMax_t UsageMinMax = {0, 0};
|
2009-02-23 08:08:22 +01:00
|
|
|
|
2009-11-04 08:14:38 +01:00
|
|
|
memset(ParserData, 0x00, sizeof(HID_ReportInfo_t));
|
2009-09-09 10:34:24 +02:00
|
|
|
memset(CurrStateTable, 0x00, sizeof(HID_StateTable_t));
|
|
|
|
memset(CurrReportIDInfo, 0x00, sizeof(HID_ReportSizeInfo_t));
|
2009-02-23 08:08:22 +01:00
|
|
|
|
2009-11-04 08:14:38 +01:00
|
|
|
ParserData->TotalDeviceReports = 1;
|
|
|
|
|
2009-02-23 08:08:22 +01:00
|
|
|
while (ReportSize)
|
|
|
|
{
|
2009-08-31 15:58:03 +02:00
|
|
|
uint8_t HIDReportItem = *ReportData;
|
2009-02-23 08:08:22 +01:00
|
|
|
uint32_t ReportItemData = 0;
|
|
|
|
|
2009-08-31 15:58:03 +02:00
|
|
|
ReportData++;
|
2009-08-11 10:36:25 +02:00
|
|
|
ReportSize--;
|
|
|
|
|
|
|
|
switch (HIDReportItem & DATA_SIZE_MASK)
|
2009-02-23 08:08:22 +01:00
|
|
|
{
|
|
|
|
case DATA_SIZE_4:
|
2009-08-11 10:36:25 +02:00
|
|
|
ReportItemData = *((uint32_t*)ReportData);
|
|
|
|
ReportSize -= 4;
|
|
|
|
ReportData += 4;
|
2009-02-23 08:08:22 +01:00
|
|
|
break;
|
|
|
|
case DATA_SIZE_2:
|
2009-08-11 10:36:25 +02:00
|
|
|
ReportItemData = *((uint16_t*)ReportData);
|
|
|
|
ReportSize -= 2;
|
|
|
|
ReportData += 2;
|
2009-02-23 08:08:22 +01:00
|
|
|
break;
|
|
|
|
case DATA_SIZE_1:
|
2009-08-11 10:36:25 +02:00
|
|
|
ReportItemData = *((uint8_t*)ReportData);
|
|
|
|
ReportSize -= 1;
|
|
|
|
ReportData += 1;
|
2009-02-23 08:08:22 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2009-08-11 10:36:25 +02:00
|
|
|
switch (HIDReportItem & (TYPE_MASK | TAG_MASK))
|
2009-02-23 08:08:22 +01:00
|
|
|
{
|
|
|
|
case (TYPE_GLOBAL | TAG_GLOBAL_PUSH):
|
2009-08-11 10:36:25 +02:00
|
|
|
if (CurrStateTable == &StateTable[HID_STATETABLE_STACK_DEPTH - 1])
|
2009-02-23 08:08:22 +01:00
|
|
|
return HID_PARSE_HIDStackOverflow;
|
|
|
|
|
2009-08-11 13:19:22 +02:00
|
|
|
memcpy((CurrStateTable + 1),
|
|
|
|
CurrStateTable,
|
2009-02-23 08:08:22 +01:00
|
|
|
sizeof(HID_ReportItem_t));
|
|
|
|
|
|
|
|
CurrStateTable++;
|
|
|
|
break;
|
|
|
|
case (TYPE_GLOBAL | TAG_GLOBAL_POP):
|
|
|
|
if (CurrStateTable == &StateTable[0])
|
|
|
|
return HID_PARSE_HIDStackUnderflow;
|
|
|
|
|
|
|
|
CurrStateTable--;
|
|
|
|
break;
|
|
|
|
case (TYPE_GLOBAL | TAG_GLOBAL_USAGEPAGE):
|
|
|
|
CurrStateTable->Attributes.Usage.Page = ReportItemData;
|
|
|
|
break;
|
|
|
|
case (TYPE_GLOBAL | TAG_GLOBAL_LOGICALMIN):
|
|
|
|
CurrStateTable->Attributes.Logical.Minimum = ReportItemData;
|
|
|
|
break;
|
|
|
|
case (TYPE_GLOBAL | TAG_GLOBAL_LOGICALMAX):
|
|
|
|
CurrStateTable->Attributes.Logical.Maximum = ReportItemData;
|
|
|
|
break;
|
|
|
|
case (TYPE_GLOBAL | TAG_GLOBAL_PHYSMIN):
|
|
|
|
CurrStateTable->Attributes.Physical.Minimum = ReportItemData;
|
|
|
|
break;
|
|
|
|
case (TYPE_GLOBAL | TAG_GLOBAL_PHYSMAX):
|
|
|
|
CurrStateTable->Attributes.Physical.Maximum = ReportItemData;
|
|
|
|
break;
|
|
|
|
case (TYPE_GLOBAL | TAG_GLOBAL_UNITEXP):
|
|
|
|
CurrStateTable->Attributes.Unit.Exponent = ReportItemData;
|
|
|
|
break;
|
|
|
|
case (TYPE_GLOBAL | TAG_GLOBAL_UNIT):
|
|
|
|
CurrStateTable->Attributes.Unit.Type = ReportItemData;
|
|
|
|
break;
|
|
|
|
case (TYPE_GLOBAL | TAG_GLOBAL_REPORTSIZE):
|
|
|
|
CurrStateTable->Attributes.BitSize = ReportItemData;
|
|
|
|
break;
|
|
|
|
case (TYPE_GLOBAL | TAG_GLOBAL_REPORTCOUNT):
|
|
|
|
CurrStateTable->ReportCount = ReportItemData;
|
|
|
|
break;
|
|
|
|
case (TYPE_GLOBAL | TAG_GLOBAL_REPORTID):
|
|
|
|
CurrStateTable->ReportID = ReportItemData;
|
2009-09-09 10:34:24 +02:00
|
|
|
|
|
|
|
if (ParserData->UsingReportIDs)
|
|
|
|
{
|
|
|
|
CurrReportIDInfo = NULL;
|
|
|
|
|
|
|
|
for (uint8_t i = 0; i < ParserData->TotalDeviceReports; i++)
|
|
|
|
{
|
|
|
|
if (ParserData->ReportIDSizes[i].ReportID == CurrStateTable->ReportID)
|
|
|
|
{
|
|
|
|
CurrReportIDInfo = &ParserData->ReportIDSizes[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (CurrReportIDInfo == NULL)
|
|
|
|
{
|
2009-11-04 14:16:53 +01:00
|
|
|
if (ParserData->TotalDeviceReports == HID_MAX_REPORT_IDS)
|
2009-09-09 10:34:24 +02:00
|
|
|
return HID_PARSE_InsufficientReportIDItems;
|
|
|
|
|
2009-11-04 14:16:53 +01:00
|
|
|
CurrReportIDInfo = &ParserData->ReportIDSizes[ParserData->TotalDeviceReports++];
|
2009-09-09 10:34:24 +02:00
|
|
|
memset(CurrReportIDInfo, 0x00, sizeof(HID_ReportSizeInfo_t));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ParserData->UsingReportIDs = true;
|
|
|
|
|
2009-11-04 14:16:53 +01:00
|
|
|
CurrReportIDInfo->ReportID = CurrStateTable->ReportID;
|
2009-02-23 08:08:22 +01:00
|
|
|
break;
|
|
|
|
case (TYPE_LOCAL | TAG_LOCAL_USAGE):
|
2009-11-04 14:16:53 +01:00
|
|
|
if (UsageListSize == HID_USAGE_STACK_DEPTH)
|
|
|
|
return HID_PARSE_UsageListOverflow;
|
2009-02-23 08:08:22 +01:00
|
|
|
|
2009-11-04 14:16:53 +01:00
|
|
|
UsageList[UsageListSize++] = ReportItemData;
|
2009-02-23 08:08:22 +01:00
|
|
|
break;
|
|
|
|
case (TYPE_LOCAL | TAG_LOCAL_USAGEMIN):
|
2009-12-04 02:06:26 +01:00
|
|
|
UsageMinMax.Minimum = ReportItemData;
|
2009-02-23 08:08:22 +01:00
|
|
|
break;
|
|
|
|
case (TYPE_LOCAL | TAG_LOCAL_USAGEMAX):
|
2009-12-04 02:06:26 +01:00
|
|
|
UsageMinMax.Maximum = ReportItemData;
|
2009-02-23 08:08:22 +01:00
|
|
|
break;
|
|
|
|
case (TYPE_MAIN | TAG_MAIN_COLLECTION):
|
|
|
|
if (CurrCollectionPath == NULL)
|
|
|
|
{
|
|
|
|
CurrCollectionPath = &ParserData->CollectionPaths[0];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
HID_CollectionPath_t* ParentCollectionPath = CurrCollectionPath;
|
|
|
|
|
|
|
|
CurrCollectionPath = &ParserData->CollectionPaths[1];
|
|
|
|
|
2009-11-04 08:14:38 +01:00
|
|
|
while (CurrCollectionPath->Parent != NULL)
|
2009-02-23 08:08:22 +01:00
|
|
|
{
|
2009-08-11 10:36:25 +02:00
|
|
|
if (CurrCollectionPath == &ParserData->CollectionPaths[HID_MAX_COLLECTIONS - 1])
|
2009-02-23 08:08:22 +01:00
|
|
|
return HID_PARSE_InsufficientCollectionPaths;
|
|
|
|
|
|
|
|
CurrCollectionPath++;
|
|
|
|
}
|
|
|
|
|
|
|
|
CurrCollectionPath->Parent = ParentCollectionPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
CurrCollectionPath->Type = ReportItemData;
|
|
|
|
CurrCollectionPath->Usage.Page = CurrStateTable->Attributes.Usage.Page;
|
|
|
|
|
2009-11-04 14:16:53 +01:00
|
|
|
if (UsageListSize)
|
2009-02-23 08:08:22 +01:00
|
|
|
{
|
2009-11-04 14:16:53 +01:00
|
|
|
CurrCollectionPath->Usage.Usage = UsageList[0];
|
2009-02-23 08:08:22 +01:00
|
|
|
|
2009-11-04 14:16:53 +01:00
|
|
|
for (uint8_t i = 0; i < UsageListSize; i++)
|
|
|
|
UsageList[i] = UsageList[i + 1];
|
2009-12-04 03:06:38 +01:00
|
|
|
|
2009-11-04 14:16:53 +01:00
|
|
|
UsageListSize--;
|
2009-02-23 08:08:22 +01:00
|
|
|
}
|
2009-12-04 02:06:26 +01:00
|
|
|
else if (UsageMinMax.Minimum <= UsageMinMax.Maximum)
|
|
|
|
{
|
|
|
|
CurrCollectionPath->Usage.Usage = UsageMinMax.Minimum++;
|
|
|
|
}
|
2009-02-23 08:08:22 +01:00
|
|
|
|
|
|
|
break;
|
|
|
|
case (TYPE_MAIN | TAG_MAIN_ENDCOLLECTION):
|
|
|
|
if (CurrCollectionPath == NULL)
|
|
|
|
return HID_PARSE_UnexpectedEndCollection;
|
|
|
|
|
|
|
|
CurrCollectionPath = CurrCollectionPath->Parent;
|
|
|
|
break;
|
|
|
|
case (TYPE_MAIN | TAG_MAIN_INPUT):
|
|
|
|
case (TYPE_MAIN | TAG_MAIN_OUTPUT):
|
|
|
|
case (TYPE_MAIN | TAG_MAIN_FEATURE):
|
2009-10-05 07:20:30 +02:00
|
|
|
for (uint8_t ReportItemNum = 0; ReportItemNum < CurrStateTable->ReportCount; ReportItemNum++)
|
2009-02-23 08:08:22 +01:00
|
|
|
{
|
2009-09-02 09:16:52 +02:00
|
|
|
HID_ReportItem_t NewReportItem;
|
2009-02-23 08:08:22 +01:00
|
|
|
|
2009-09-02 09:16:52 +02:00
|
|
|
memcpy(&NewReportItem.Attributes,
|
2009-02-23 08:08:22 +01:00
|
|
|
&CurrStateTable->Attributes,
|
|
|
|
sizeof(HID_ReportItem_Attributes_t));
|
|
|
|
|
2009-09-02 09:16:52 +02:00
|
|
|
NewReportItem.ItemFlags = ReportItemData;
|
|
|
|
NewReportItem.CollectionPath = CurrCollectionPath;
|
|
|
|
NewReportItem.ReportID = CurrStateTable->ReportID;
|
2009-02-23 08:08:22 +01:00
|
|
|
|
2009-11-04 14:16:53 +01:00
|
|
|
if (UsageListSize)
|
2009-02-23 08:08:22 +01:00
|
|
|
{
|
2009-11-04 14:16:53 +01:00
|
|
|
NewReportItem.Attributes.Usage.Usage = UsageList[0];
|
2009-02-23 08:08:22 +01:00
|
|
|
|
2009-11-04 14:16:53 +01:00
|
|
|
for (uint8_t i = 0; i < UsageListSize; i++)
|
|
|
|
UsageList[i] = UsageList[i + 1];
|
2009-02-23 08:08:22 +01:00
|
|
|
|
2009-11-04 14:16:53 +01:00
|
|
|
UsageListSize--;
|
2009-02-23 08:08:22 +01:00
|
|
|
}
|
2009-12-04 02:06:26 +01:00
|
|
|
else if (UsageMinMax.Minimum <= UsageMinMax.Maximum)
|
|
|
|
{
|
2009-12-04 03:06:38 +01:00
|
|
|
NewReportItem.Attributes.Usage.Usage = UsageMinMax.Minimum++;
|
2009-12-04 02:06:26 +01:00
|
|
|
}
|
2009-12-04 03:06:38 +01:00
|
|
|
|
2009-11-04 14:16:53 +01:00
|
|
|
uint8_t ItemTag = (HIDReportItem & TAG_MASK);
|
|
|
|
|
|
|
|
if (ItemTag == TAG_MAIN_INPUT)
|
|
|
|
NewReportItem.ItemType = REPORT_ITEM_TYPE_In;
|
|
|
|
else if (ItemTag == TAG_MAIN_OUTPUT)
|
|
|
|
NewReportItem.ItemType = REPORT_ITEM_TYPE_Out;
|
|
|
|
else
|
|
|
|
NewReportItem.ItemType = REPORT_ITEM_TYPE_Feature;
|
2009-09-21 14:23:09 +02:00
|
|
|
|
2009-11-04 08:14:38 +01:00
|
|
|
NewReportItem.BitOffset = CurrReportIDInfo->ReportSizeBits[NewReportItem.ItemType];
|
|
|
|
|
|
|
|
CurrReportIDInfo->ReportSizeBits[NewReportItem.ItemType] += CurrStateTable->Attributes.BitSize;
|
2009-09-02 09:16:52 +02:00
|
|
|
|
2009-11-04 08:14:38 +01:00
|
|
|
if (ParserData->LargestReportSizeBits < NewReportItem.BitOffset)
|
|
|
|
ParserData->LargestReportSizeBits = NewReportItem.BitOffset;
|
2009-09-21 14:23:09 +02:00
|
|
|
|
2009-11-03 12:11:51 +01:00
|
|
|
if (!(ReportItemData & IOF_CONSTANT) && CALLBACK_HIDParser_FilterHIDReportItem(&NewReportItem))
|
2009-09-02 09:16:52 +02:00
|
|
|
{
|
|
|
|
if (ParserData->TotalReportItems == HID_MAX_REPORTITEMS)
|
|
|
|
return HID_PARSE_InsufficientReportItems;
|
|
|
|
|
|
|
|
memcpy(&ParserData->ReportItems[ParserData->TotalReportItems],
|
|
|
|
&NewReportItem, sizeof(HID_ReportItem_t));
|
2009-02-23 08:08:22 +01:00
|
|
|
|
2009-09-02 09:16:52 +02:00
|
|
|
ParserData->TotalReportItems++;
|
|
|
|
}
|
2009-02-23 08:08:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2009-08-11 10:36:25 +02:00
|
|
|
if ((HIDReportItem & TYPE_MASK) == TYPE_MAIN)
|
2009-02-23 08:08:22 +01:00
|
|
|
{
|
2009-12-04 02:06:26 +01:00
|
|
|
UsageMinMax.Minimum = 0;
|
|
|
|
UsageMinMax.Maximum = 0;
|
|
|
|
UsageListSize = 0;
|
2009-02-23 08:08:22 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-22 10:07:48 +02:00
|
|
|
if (!(ParserData->TotalReportItems))
|
|
|
|
return HID_PARSE_NoUnfilteredReportItems;
|
|
|
|
|
2009-02-23 08:08:22 +01:00
|
|
|
return HID_PARSE_Successful;
|
|
|
|
}
|
|
|
|
|
2009-04-19 13:43:21 +02:00
|
|
|
bool USB_GetHIDReportItemInfo(const uint8_t* ReportData, HID_ReportItem_t* const ReportItem)
|
2009-02-23 08:08:22 +01:00
|
|
|
{
|
|
|
|
uint16_t DataBitsRem = ReportItem->Attributes.BitSize;
|
|
|
|
uint16_t CurrentBit = ReportItem->BitOffset;
|
|
|
|
uint32_t BitMask = (1 << 0);
|
|
|
|
|
2009-12-04 02:06:26 +01:00
|
|
|
ReportItem->PreviousValue = ReportItem->Value;
|
2009-02-23 08:08:22 +01:00
|
|
|
ReportItem->Value = 0;
|
|
|
|
|
|
|
|
if (ReportItem->ReportID)
|
|
|
|
{
|
|
|
|
if (ReportItem->ReportID != ReportData[0])
|
|
|
|
return false;
|
|
|
|
|
|
|
|
ReportData++;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (DataBitsRem--)
|
|
|
|
{
|
|
|
|
if (ReportData[CurrentBit / 8] & (1 << (CurrentBit % 8)))
|
|
|
|
ReportItem->Value |= BitMask;
|
|
|
|
|
|
|
|
CurrentBit++;
|
|
|
|
BitMask <<= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-12-04 02:09:39 +01:00
|
|
|
void USB_SetHIDReportItemInfo(uint8_t* ReportData, HID_ReportItem_t* const ReportItem)
|
2009-02-23 08:08:22 +01:00
|
|
|
{
|
|
|
|
uint16_t DataBitsRem = ReportItem->Attributes.BitSize;
|
|
|
|
uint16_t CurrentBit = ReportItem->BitOffset;
|
|
|
|
uint32_t BitMask = (1 << 0);
|
|
|
|
|
|
|
|
if (ReportItem->ReportID)
|
|
|
|
{
|
|
|
|
ReportData[0] = ReportItem->ReportID;
|
|
|
|
ReportData++;
|
|
|
|
}
|
|
|
|
|
2009-12-04 02:06:26 +01:00
|
|
|
ReportItem->PreviousValue = ReportItem->Value;
|
|
|
|
|
2009-02-23 08:08:22 +01:00
|
|
|
while (DataBitsRem--)
|
|
|
|
{
|
|
|
|
if (ReportItem->Value & (1 << (CurrentBit % 8)))
|
|
|
|
ReportData[CurrentBit / 8] |= BitMask;
|
|
|
|
|
|
|
|
CurrentBit++;
|
|
|
|
BitMask <<= 1;
|
|
|
|
}
|
|
|
|
}
|
2009-06-14 17:55:13 +02:00
|
|
|
|
2009-09-21 08:08:39 +02:00
|
|
|
uint16_t USB_GetHIDReportSize(HID_ReportInfo_t* const ParserData, const uint8_t ReportID, const uint8_t ReportType)
|
2009-09-20 14:01:25 +02:00
|
|
|
{
|
|
|
|
for (uint8_t i = 0; i < HID_MAX_REPORT_IDS; i++)
|
|
|
|
{
|
|
|
|
if (ParserData->ReportIDSizes[i].ReportID == ReportID)
|
2009-09-20 14:34:07 +02:00
|
|
|
return ParserData->ReportIDSizes[i].ReportSizeBits[ReportType];
|
2009-09-20 14:01:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-06-14 17:55:13 +02:00
|
|
|
#endif
|