2015-10-01 12:50:18 +02:00
|
|
|
/*
|
|
|
|
Copyright (c) 2015, Arduino LLC
|
|
|
|
Original code (pre-library): Copyright (c) 2011, Peter Barrett
|
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software for
|
|
|
|
any purpose with or without fee is hereby granted, provided that the
|
|
|
|
above copyright notice and this permission notice appear in all copies.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS 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, DIRECT, 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.
|
|
|
|
*/
|
2015-06-05 17:54:46 +02:00
|
|
|
|
|
|
|
#include "HID.h"
|
|
|
|
|
|
|
|
#if defined(USBCON)
|
|
|
|
|
2015-10-01 17:35:26 +02:00
|
|
|
HID_& HID()
|
|
|
|
{
|
|
|
|
static HID_ obj;
|
|
|
|
return obj;
|
|
|
|
}
|
2015-06-05 17:54:46 +02:00
|
|
|
|
2015-10-07 13:21:21 +02:00
|
|
|
int HID_::getInterface(uint8_t* interfaceCount)
|
2015-06-05 17:54:46 +02:00
|
|
|
{
|
2015-10-07 13:21:21 +02:00
|
|
|
*interfaceCount += 1; // uses 1
|
2015-10-01 12:53:24 +02:00
|
|
|
HIDDescriptor hidInterface = {
|
2015-10-10 12:49:45 +02:00
|
|
|
D_INTERFACE(pluggedInterface, 1, USB_DEVICE_CLASS_HUMAN_INTERFACE, HID_SUBCLASS_NONE, HID_PROTOCOL_NONE),
|
2015-10-01 12:50:18 +02:00
|
|
|
D_HIDREPORT(descriptorSize),
|
2015-10-07 20:39:50 +02:00
|
|
|
D_ENDPOINT(USB_ENDPOINT_IN(pluggedEndpoint), USB_ENDPOINT_TYPE_INTERRUPT, USB_EP_SIZE, 0x01)
|
2015-07-02 12:12:15 +02:00
|
|
|
};
|
2015-09-30 16:19:21 +02:00
|
|
|
return USB_SendControl(0, &hidInterface, sizeof(hidInterface));
|
2015-06-05 17:54:46 +02:00
|
|
|
}
|
|
|
|
|
2015-10-07 19:02:40 +02:00
|
|
|
int HID_::getDescriptor(USBSetup& setup)
|
2015-06-05 17:54:46 +02:00
|
|
|
{
|
2015-10-08 16:38:57 +02:00
|
|
|
// Check if this is a HID Class Descriptor request
|
|
|
|
if (setup.bmRequestType != REQUEST_DEVICETOHOST_STANDARD_INTERFACE) { return 0; }
|
|
|
|
if (setup.wValueH != HID_REPORT_DESCRIPTOR_TYPE) { return 0; }
|
|
|
|
|
|
|
|
// In a HID Class Descriptor wIndex cointains the interface number
|
|
|
|
if (setup.wIndex != pluggedInterface) { return 0; }
|
2015-10-01 12:35:24 +02:00
|
|
|
|
2015-10-07 19:02:40 +02:00
|
|
|
int total = 0;
|
2015-10-12 12:17:42 +02:00
|
|
|
HIDSubDescriptor* node;
|
2015-10-07 19:02:40 +02:00
|
|
|
for (node = rootNode; node; node = node->next) {
|
|
|
|
int res = USB_SendControl(TRANSFER_PGM, node->data, node->length);
|
|
|
|
if (res == -1)
|
|
|
|
return -1;
|
|
|
|
total += res;
|
|
|
|
}
|
2015-10-12 17:31:35 +02:00
|
|
|
|
|
|
|
// Reset the protocol on reenumeration. Normally the host should not assume the state of the protocol
|
|
|
|
// due to the USB specs, but Windows and Linux just assumes its in report mode.
|
|
|
|
protocol = HID_REPORT_PROTOCOL;
|
|
|
|
|
2015-10-07 19:02:40 +02:00
|
|
|
return total;
|
2015-06-05 17:54:46 +02:00
|
|
|
}
|
|
|
|
|
2015-10-13 15:13:31 +02:00
|
|
|
uint8_t HID_::getShortName(char *name)
|
|
|
|
{
|
|
|
|
name[0] = 'H';
|
|
|
|
name[1] = 'I';
|
|
|
|
name[2] = 'D';
|
|
|
|
name[3] = 'A' + (descriptorSize & 0x0F);
|
|
|
|
name[4] = 'A' + ((descriptorSize >> 4) & 0x0F);
|
|
|
|
return 5;
|
|
|
|
}
|
|
|
|
|
2015-10-12 12:17:42 +02:00
|
|
|
void HID_::AppendDescriptor(HIDSubDescriptor *node)
|
2015-07-02 12:12:15 +02:00
|
|
|
{
|
2015-10-01 12:39:57 +02:00
|
|
|
if (!rootNode) {
|
2015-07-02 12:12:15 +02:00
|
|
|
rootNode = node;
|
|
|
|
} else {
|
2015-10-12 12:17:42 +02:00
|
|
|
HIDSubDescriptor *current = rootNode;
|
2015-10-01 12:50:18 +02:00
|
|
|
while (current->next) {
|
2015-07-02 12:12:15 +02:00
|
|
|
current = current->next;
|
|
|
|
}
|
|
|
|
current->next = node;
|
|
|
|
}
|
2015-10-01 12:50:18 +02:00
|
|
|
descriptorSize += node->length;
|
2015-07-02 12:12:15 +02:00
|
|
|
}
|
|
|
|
|
2015-10-25 08:24:55 +01:00
|
|
|
int HID_::SendReport(uint8_t id, const void* data, int len)
|
2015-06-05 17:54:46 +02:00
|
|
|
{
|
2015-10-27 08:06:21 +01:00
|
|
|
auto ret = USB_Send(pluggedEndpoint, &id, 1);
|
|
|
|
if(ret >= 0){
|
|
|
|
ret += USB_Send(pluggedEndpoint | TRANSFER_RELEASE, data, len);
|
|
|
|
}
|
|
|
|
return ret;
|
2015-06-05 17:54:46 +02:00
|
|
|
}
|
|
|
|
|
2015-10-07 19:02:40 +02:00
|
|
|
bool HID_::setup(USBSetup& setup)
|
2015-06-05 17:54:46 +02:00
|
|
|
{
|
2015-10-07 20:39:50 +02:00
|
|
|
if (pluggedInterface != setup.wIndex) {
|
2015-06-05 17:54:46 +02:00
|
|
|
return false;
|
2015-10-01 12:50:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t request = setup.bRequest;
|
|
|
|
uint8_t requestType = setup.bmRequestType;
|
|
|
|
|
|
|
|
if (requestType == REQUEST_DEVICETOHOST_CLASS_INTERFACE)
|
|
|
|
{
|
|
|
|
if (request == HID_GET_REPORT) {
|
|
|
|
// TODO: HID_GetReport();
|
|
|
|
return true;
|
2015-06-05 17:54:46 +02:00
|
|
|
}
|
2015-10-01 12:50:18 +02:00
|
|
|
if (request == HID_GET_PROTOCOL) {
|
|
|
|
// TODO: Send8(protocol);
|
|
|
|
return true;
|
|
|
|
}
|
2015-10-11 14:39:36 +02:00
|
|
|
if (request == HID_GET_IDLE) {
|
|
|
|
// TODO: Send8(idle);
|
|
|
|
}
|
2015-10-01 12:50:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (requestType == REQUEST_HOSTTODEVICE_CLASS_INTERFACE)
|
|
|
|
{
|
|
|
|
if (request == HID_SET_PROTOCOL) {
|
2015-10-10 13:11:31 +02:00
|
|
|
// The USB Host tells us if we are in boot or report mode.
|
|
|
|
// This only works with a real boot compatible device.
|
2015-10-01 12:50:18 +02:00
|
|
|
protocol = setup.wValueL;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (request == HID_SET_IDLE) {
|
|
|
|
idle = setup.wValueL;
|
|
|
|
return true;
|
2015-06-05 17:54:46 +02:00
|
|
|
}
|
2015-10-07 19:11:23 +02:00
|
|
|
if (request == HID_SET_REPORT)
|
|
|
|
{
|
2015-10-10 13:11:31 +02:00
|
|
|
//uint8_t reportID = setup.wValueL;
|
|
|
|
//uint16_t length = setup.wLength;
|
|
|
|
//uint8_t data[length];
|
|
|
|
// Make sure to not read more data than USB_EP_SIZE.
|
|
|
|
// You can read multiple times through a loop.
|
|
|
|
// The first byte (may!) contain the reportID on a multreport.
|
|
|
|
//USB_RecvControl(data, length);
|
2015-10-07 19:11:23 +02:00
|
|
|
}
|
2015-06-05 17:54:46 +02:00
|
|
|
}
|
2015-10-01 12:50:18 +02:00
|
|
|
|
|
|
|
return false;
|
2015-06-05 17:54:46 +02:00
|
|
|
}
|
|
|
|
|
2015-10-12 12:14:07 +02:00
|
|
|
HID_::HID_(void) : PluggableUSBModule(1, 1, epType),
|
2015-10-01 12:50:18 +02:00
|
|
|
rootNode(NULL), descriptorSize(0),
|
2015-10-12 17:31:35 +02:00
|
|
|
protocol(HID_REPORT_PROTOCOL), idle(1)
|
2015-06-05 17:54:46 +02:00
|
|
|
{
|
2015-09-30 18:45:09 +02:00
|
|
|
epType[0] = EP_TYPE_INTERRUPT_IN;
|
2015-10-01 17:35:26 +02:00
|
|
|
PluggableUSB().plug(this);
|
2015-06-05 17:54:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int HID_::begin(void)
|
|
|
|
{
|
2015-09-28 15:58:16 +02:00
|
|
|
return 0;
|
2015-06-05 17:54:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* if defined(USBCON) */
|