2015-06-30 12:39:40 +00:00
|
|
|
/*
|
|
|
|
* Human Input Devices
|
|
|
|
*
|
|
|
|
* Copyright (C) 2015 Aric Stewart
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library 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
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
2016-09-19 13:45:34 +00:00
|
|
|
#include "ntstatus.h"
|
2015-06-30 12:39:40 +00:00
|
|
|
#define WIN32_NO_STATUS
|
|
|
|
#include "windef.h"
|
|
|
|
#include "winbase.h"
|
|
|
|
#include "winternl.h"
|
|
|
|
#include "winioctl.h"
|
|
|
|
#include "ddk/wdm.h"
|
|
|
|
|
|
|
|
#include "hidusage.h"
|
|
|
|
#include "ddk/hidpi.h"
|
2019-10-30 19:25:58 +00:00
|
|
|
#include "wine/hid.h"
|
2015-06-30 12:39:40 +00:00
|
|
|
#include "wine/debug.h"
|
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(hidp);
|
|
|
|
|
2021-06-21 07:05:22 +00:00
|
|
|
static NTSTATUS get_value_caps_range( WINE_HIDP_PREPARSED_DATA *preparsed, HIDP_REPORT_TYPE report_type, ULONG report_len,
|
|
|
|
const struct hid_value_caps **caps, const struct hid_value_caps **caps_end )
|
|
|
|
{
|
|
|
|
if (preparsed->magic != HID_MAGIC) return HIDP_STATUS_INVALID_PREPARSED_DATA;
|
|
|
|
|
|
|
|
switch (report_type)
|
|
|
|
{
|
|
|
|
case HidP_Input:
|
|
|
|
if (report_len && report_len != preparsed->caps.InputReportByteLength)
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_LENGTH;
|
|
|
|
*caps = HID_INPUT_VALUE_CAPS( preparsed );
|
|
|
|
break;
|
|
|
|
case HidP_Output:
|
|
|
|
if (report_len && report_len != preparsed->caps.OutputReportByteLength)
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_LENGTH;
|
|
|
|
*caps = HID_OUTPUT_VALUE_CAPS( preparsed );
|
|
|
|
break;
|
|
|
|
case HidP_Feature:
|
|
|
|
if (report_len && report_len != preparsed->caps.FeatureReportByteLength)
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_LENGTH;
|
|
|
|
*caps = HID_FEATURE_VALUE_CAPS( preparsed );
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_TYPE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*caps_end = *caps + preparsed->value_caps_count[report_type];
|
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2021-06-21 07:05:23 +00:00
|
|
|
struct caps_filter
|
|
|
|
{
|
|
|
|
BOOLEAN buttons;
|
|
|
|
USAGE usage_page;
|
|
|
|
USHORT collection;
|
|
|
|
USAGE usage;
|
|
|
|
};
|
|
|
|
|
|
|
|
static BOOL match_value_caps( const struct hid_value_caps *caps, const struct caps_filter *filter )
|
|
|
|
{
|
|
|
|
if (!caps->usage_min && !caps->usage_max) return FALSE;
|
|
|
|
if (filter->buttons && !HID_VALUE_CAPS_IS_BUTTON( caps )) return FALSE;
|
|
|
|
if (filter->usage_page && filter->usage_page != caps->usage_page) return FALSE;
|
|
|
|
if (filter->collection && filter->collection != caps->link_collection) return FALSE;
|
|
|
|
if (!filter->usage) return TRUE;
|
|
|
|
return caps->usage_min <= filter->usage && caps->usage_max >= filter->usage;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef NTSTATUS (*enum_value_caps_callback)( const struct hid_value_caps *caps, void *user );
|
|
|
|
|
|
|
|
static NTSTATUS enum_value_caps( WINE_HIDP_PREPARSED_DATA *preparsed, HIDP_REPORT_TYPE report_type,
|
|
|
|
const struct caps_filter *filter, enum_value_caps_callback callback,
|
|
|
|
void *user, USHORT *count )
|
|
|
|
{
|
|
|
|
const struct hid_value_caps *caps, *caps_end;
|
|
|
|
NTSTATUS status;
|
|
|
|
LONG remaining = *count;
|
|
|
|
|
|
|
|
for (status = get_value_caps_range( preparsed, report_type, 0, &caps, &caps_end );
|
|
|
|
status == HIDP_STATUS_SUCCESS && caps != caps_end; caps++)
|
|
|
|
{
|
|
|
|
if (!match_value_caps( caps, filter )) continue;
|
|
|
|
if (remaining-- > 0) status = callback( caps, user );
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status != HIDP_STATUS_SUCCESS) return status;
|
|
|
|
|
|
|
|
*count -= remaining;
|
|
|
|
if (*count == 0) return HIDP_STATUS_USAGE_NOT_FOUND;
|
|
|
|
if (remaining < 0) return HIDP_STATUS_BUFFER_TOO_SMALL;
|
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2015-06-30 12:40:00 +00:00
|
|
|
static NTSTATUS get_report_data(BYTE *report, INT reportLength, INT startBit, INT valueSize, PULONG value)
|
|
|
|
{
|
|
|
|
|
|
|
|
if ((startBit + valueSize) / 8 > reportLength)
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_LENGTH;
|
|
|
|
|
|
|
|
if (valueSize == 1)
|
|
|
|
{
|
|
|
|
ULONG byte_index = startBit / 8;
|
|
|
|
ULONG bit_index = startBit - (byte_index * 8);
|
|
|
|
INT mask = (1 << bit_index);
|
2019-05-06 13:46:42 +00:00
|
|
|
*value = !!(report[byte_index] & mask);
|
2015-06-30 12:40:00 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-05-06 13:46:42 +00:00
|
|
|
ULONG remaining_bits = valueSize;
|
2016-11-07 19:44:33 +00:00
|
|
|
ULONG byte_index = startBit / 8;
|
2019-05-06 13:46:42 +00:00
|
|
|
ULONG bit_index = startBit % 8;
|
2015-06-30 12:40:00 +00:00
|
|
|
ULONG data = 0;
|
2016-11-07 19:44:33 +00:00
|
|
|
ULONG shift = 0;
|
2019-05-06 13:46:42 +00:00
|
|
|
while (remaining_bits)
|
2015-06-30 12:40:00 +00:00
|
|
|
{
|
2019-05-06 13:46:42 +00:00
|
|
|
ULONG copy_bits = 8 - bit_index;
|
|
|
|
if (remaining_bits < copy_bits)
|
|
|
|
copy_bits = remaining_bits;
|
|
|
|
|
2019-05-20 15:31:37 +00:00
|
|
|
data |= ((report[byte_index] >> bit_index) & ((1 << copy_bits) - 1)) << shift;
|
2019-05-06 13:46:42 +00:00
|
|
|
|
|
|
|
shift += copy_bits;
|
|
|
|
bit_index = 0;
|
|
|
|
byte_index++;
|
|
|
|
remaining_bits -= copy_bits;
|
2015-06-30 12:40:00 +00:00
|
|
|
}
|
|
|
|
*value = data;
|
|
|
|
}
|
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-11-23 21:09:34 +00:00
|
|
|
static NTSTATUS set_report_data(BYTE *report, INT reportLength, INT startBit, INT valueSize, ULONG value)
|
|
|
|
{
|
|
|
|
if ((startBit + valueSize) / 8 > reportLength)
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_LENGTH;
|
|
|
|
|
|
|
|
if (valueSize == 1)
|
|
|
|
{
|
|
|
|
ULONG byte_index = startBit / 8;
|
|
|
|
ULONG bit_index = startBit - (byte_index * 8);
|
|
|
|
if (value)
|
|
|
|
report[byte_index] |= (1 << bit_index);
|
|
|
|
else
|
|
|
|
report[byte_index] &= ~(1 << bit_index);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ULONG byte_index = (startBit + valueSize - 1) / 8;
|
|
|
|
ULONG data = value;
|
|
|
|
ULONG remainingBits = valueSize;
|
|
|
|
while (remainingBits)
|
|
|
|
{
|
|
|
|
BYTE subvalue = data & 0xff;
|
|
|
|
|
|
|
|
data >>= 8;
|
|
|
|
|
|
|
|
if (remainingBits >= 8)
|
|
|
|
{
|
|
|
|
report[byte_index] = subvalue;
|
|
|
|
byte_index --;
|
|
|
|
remainingBits -= 8;
|
|
|
|
}
|
|
|
|
else if (remainingBits > 0)
|
|
|
|
{
|
|
|
|
BYTE mask = (0xff << (8-remainingBits)) & subvalue;
|
|
|
|
report[byte_index] |= mask;
|
|
|
|
remainingBits = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2020-02-26 13:03:46 +00:00
|
|
|
static NTSTATUS get_report_data_array(BYTE *report, UINT reportLength, UINT startBit, UINT elemSize,
|
|
|
|
UINT numElements, PCHAR values, UINT valuesSize)
|
|
|
|
{
|
|
|
|
BYTE byte, *end, *p = report + startBit / 8;
|
|
|
|
ULONG size = elemSize * numElements;
|
|
|
|
ULONG m, bit_index = startBit % 8;
|
|
|
|
BYTE *data = (BYTE*)values;
|
|
|
|
|
|
|
|
if ((startBit + size) / 8 > reportLength)
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_LENGTH;
|
|
|
|
|
|
|
|
if (valuesSize < (size + 7) / 8)
|
|
|
|
return HIDP_STATUS_BUFFER_TOO_SMALL;
|
|
|
|
|
|
|
|
end = report + (startBit + size + 7) / 8;
|
|
|
|
|
|
|
|
data--;
|
|
|
|
byte = *p++;
|
|
|
|
while (p != end)
|
|
|
|
{
|
|
|
|
*(++data) = byte >> bit_index;
|
|
|
|
byte = *p++;
|
|
|
|
*data |= byte << (8 - bit_index);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Handle the end and mask out bits beyond */
|
|
|
|
m = (startBit + size) % 8;
|
|
|
|
m = m ? m : 8;
|
|
|
|
|
|
|
|
if (m > bit_index)
|
|
|
|
*(++data) = (byte >> bit_index) & ((1 << (m - bit_index)) - 1);
|
|
|
|
else
|
|
|
|
*data &= (1 << (m + 8 - bit_index)) - 1;
|
|
|
|
|
|
|
|
if (++data < (BYTE*)values + valuesSize)
|
|
|
|
memset(data, 0, (BYTE*)values + valuesSize - data);
|
|
|
|
|
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2021-06-21 07:05:21 +00:00
|
|
|
NTSTATUS WINAPI HidP_GetButtonCaps( HIDP_REPORT_TYPE report_type, HIDP_BUTTON_CAPS *caps, USHORT *caps_count,
|
|
|
|
PHIDP_PREPARSED_DATA preparsed_data )
|
2015-06-30 12:39:50 +00:00
|
|
|
{
|
2021-06-21 07:05:21 +00:00
|
|
|
return HidP_GetSpecificButtonCaps( report_type, 0, 0, 0, caps, caps_count, preparsed_data );
|
2015-06-30 12:39:50 +00:00
|
|
|
}
|
|
|
|
|
2021-06-21 07:05:23 +00:00
|
|
|
NTSTATUS WINAPI HidP_GetCaps( PHIDP_PREPARSED_DATA preparsed_data, HIDP_CAPS *caps )
|
2015-06-30 12:39:40 +00:00
|
|
|
{
|
2021-06-21 07:05:23 +00:00
|
|
|
WINE_HIDP_PREPARSED_DATA *preparsed = (WINE_HIDP_PREPARSED_DATA *)preparsed_data;
|
2015-06-30 12:39:40 +00:00
|
|
|
|
2021-06-21 07:05:23 +00:00
|
|
|
TRACE( "preparsed_data %p, caps %p.\n", preparsed_data, caps );
|
2015-06-30 12:39:40 +00:00
|
|
|
|
2021-06-21 07:05:23 +00:00
|
|
|
if (preparsed->magic != HID_MAGIC) return HIDP_STATUS_INVALID_PREPARSED_DATA;
|
2015-06-30 12:39:40 +00:00
|
|
|
|
2021-06-21 07:05:23 +00:00
|
|
|
*caps = preparsed->caps;
|
|
|
|
caps->NumberInputButtonCaps = preparsed->new_caps.NumberInputButtonCaps;
|
|
|
|
caps->NumberOutputButtonCaps = preparsed->new_caps.NumberOutputButtonCaps;
|
|
|
|
caps->NumberFeatureButtonCaps = preparsed->new_caps.NumberFeatureButtonCaps;
|
2015-06-30 12:39:40 +00:00
|
|
|
|
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
2015-06-30 12:39:55 +00:00
|
|
|
|
2019-05-13 10:34:19 +00:00
|
|
|
static NTSTATUS find_usage(HIDP_REPORT_TYPE ReportType, USAGE UsagePage, USHORT LinkCollection,
|
2015-06-30 12:40:41 +00:00
|
|
|
USAGE Usage, PHIDP_PREPARSED_DATA PreparsedData, PCHAR Report,
|
2021-06-07 09:07:01 +00:00
|
|
|
USHORT bit_size, WINE_HID_ELEMENT *element)
|
2015-06-30 12:40:04 +00:00
|
|
|
{
|
|
|
|
PWINE_HIDP_PREPARSED_DATA data = (PWINE_HIDP_PREPARSED_DATA)PreparsedData;
|
2019-05-13 10:33:56 +00:00
|
|
|
WINE_HID_ELEMENT *elems = HID_ELEMS(data);
|
2015-06-30 12:40:04 +00:00
|
|
|
WINE_HID_REPORT *report = NULL;
|
|
|
|
USHORT v_count = 0, r_count = 0;
|
|
|
|
int i;
|
|
|
|
|
2015-06-30 12:40:41 +00:00
|
|
|
TRACE("(%i, %x, %i, %i, %p, %p)\n", ReportType, UsagePage, LinkCollection, Usage,
|
|
|
|
PreparsedData, Report);
|
2015-06-30 12:40:04 +00:00
|
|
|
|
|
|
|
if (data->magic != HID_MAGIC)
|
|
|
|
return HIDP_STATUS_INVALID_PREPARSED_DATA;
|
|
|
|
switch(ReportType)
|
|
|
|
{
|
|
|
|
case HidP_Input:
|
|
|
|
v_count = data->caps.NumberInputValueCaps;
|
|
|
|
break;
|
|
|
|
case HidP_Output:
|
|
|
|
v_count = data->caps.NumberOutputValueCaps;
|
|
|
|
break;
|
|
|
|
case HidP_Feature:
|
|
|
|
v_count = data->caps.NumberFeatureValueCaps;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_TYPE;
|
|
|
|
}
|
2019-05-13 10:33:56 +00:00
|
|
|
r_count = data->reportCount[ReportType];
|
|
|
|
report = &data->reports[data->reportIdx[ReportType][(BYTE)Report[0]]];
|
2015-06-30 12:40:04 +00:00
|
|
|
|
2019-05-13 10:33:56 +00:00
|
|
|
if (!r_count || !v_count)
|
2015-06-30 12:40:04 +00:00
|
|
|
return HIDP_STATUS_USAGE_NOT_FOUND;
|
|
|
|
|
2019-05-13 10:33:56 +00:00
|
|
|
if (report->reportID && report->reportID != Report[0])
|
2015-06-30 12:40:04 +00:00
|
|
|
return HIDP_STATUS_REPORT_DOES_NOT_EXIST;
|
|
|
|
|
|
|
|
for (i = 0; i < report->elementCount; i++)
|
|
|
|
{
|
2021-06-04 09:30:35 +00:00
|
|
|
HIDP_VALUE_CAPS *value = &elems[report->elementIdx + i].caps;
|
2019-05-03 09:59:06 +00:00
|
|
|
|
2021-06-07 09:07:01 +00:00
|
|
|
if ((elems[report->elementIdx + i].caps.BitSize == 1) != (bit_size == 1) ||
|
2019-05-03 09:59:06 +00:00
|
|
|
value->UsagePage != UsagePage)
|
|
|
|
continue;
|
|
|
|
|
2021-06-04 09:30:33 +00:00
|
|
|
if (value->IsRange && value->Range.UsageMin <= Usage && Usage <= value->Range.UsageMax)
|
2019-05-03 09:59:06 +00:00
|
|
|
{
|
2019-05-13 10:33:56 +00:00
|
|
|
*element = elems[report->elementIdx + i];
|
2021-06-04 09:30:33 +00:00
|
|
|
element->valueStartBit += value->BitSize * (Usage - value->Range.UsageMin);
|
2021-06-07 09:07:01 +00:00
|
|
|
element->bitCount = elems[report->elementIdx + i].caps.BitSize;
|
2019-05-03 09:59:06 +00:00
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
2021-06-04 09:30:33 +00:00
|
|
|
else if (value->NotRange.Usage == Usage)
|
2015-06-30 12:40:04 +00:00
|
|
|
{
|
2019-05-13 10:33:56 +00:00
|
|
|
*element = elems[report->elementIdx + i];
|
2021-06-07 09:07:01 +00:00
|
|
|
element->bitCount = elems[report->elementIdx + i].caps.BitSize;
|
2015-06-30 12:40:41 +00:00
|
|
|
return HIDP_STATUS_SUCCESS;
|
2015-06-30 12:40:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return HIDP_STATUS_USAGE_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
2019-02-12 06:06:03 +00:00
|
|
|
static LONG sign_extend(ULONG value, const WINE_HID_ELEMENT *element)
|
|
|
|
{
|
|
|
|
UINT bit_count = element->bitCount;
|
|
|
|
|
|
|
|
if ((value & (1 << (bit_count - 1)))
|
2021-06-07 09:07:01 +00:00
|
|
|
&& element->caps.BitSize != 1
|
2021-06-04 09:30:35 +00:00
|
|
|
&& element->caps.LogicalMin < 0)
|
2019-02-12 06:06:03 +00:00
|
|
|
{
|
|
|
|
value -= (1 << bit_count);
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2019-02-12 06:06:04 +00:00
|
|
|
static LONG logical_to_physical(LONG value, const WINE_HID_ELEMENT *element)
|
|
|
|
{
|
2021-06-04 09:30:35 +00:00
|
|
|
if (element->caps.PhysicalMin || element->caps.PhysicalMax)
|
2019-02-12 06:06:04 +00:00
|
|
|
{
|
2021-06-04 09:30:35 +00:00
|
|
|
value = (((ULONGLONG)(value - element->caps.LogicalMin)
|
|
|
|
* (element->caps.PhysicalMax - element->caps.PhysicalMin))
|
|
|
|
/ (element->caps.LogicalMax - element->caps.LogicalMin))
|
|
|
|
+ element->caps.PhysicalMin;
|
2019-02-12 06:06:04 +00:00
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2015-06-30 12:40:41 +00:00
|
|
|
NTSTATUS WINAPI HidP_GetScaledUsageValue(HIDP_REPORT_TYPE ReportType, USAGE UsagePage,
|
|
|
|
USHORT LinkCollection, USAGE Usage, PLONG UsageValue,
|
|
|
|
PHIDP_PREPARSED_DATA PreparsedData, PCHAR Report, ULONG ReportLength)
|
|
|
|
{
|
|
|
|
NTSTATUS rc;
|
2019-05-03 09:59:06 +00:00
|
|
|
WINE_HID_ELEMENT element;
|
2015-06-30 12:40:41 +00:00
|
|
|
TRACE("(%i, %x, %i, %i, %p, %p, %p, %i)\n", ReportType, UsagePage, LinkCollection, Usage, UsageValue,
|
|
|
|
PreparsedData, Report, ReportLength);
|
|
|
|
|
2021-06-07 09:07:01 +00:00
|
|
|
rc = find_usage(ReportType, UsagePage, LinkCollection, Usage, PreparsedData, Report, 0, &element);
|
2015-06-30 12:40:41 +00:00
|
|
|
|
|
|
|
if (rc == HIDP_STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
ULONG rawValue;
|
|
|
|
rc = get_report_data((BYTE*)Report, ReportLength,
|
2019-05-03 09:59:06 +00:00
|
|
|
element.valueStartBit, element.bitCount, &rawValue);
|
2015-06-30 12:40:41 +00:00
|
|
|
if (rc != HIDP_STATUS_SUCCESS)
|
|
|
|
return rc;
|
2019-05-03 09:59:06 +00:00
|
|
|
*UsageValue = logical_to_physical(sign_extend(rawValue, &element), &element);
|
2015-06-30 12:40:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
NTSTATUS WINAPI HidP_GetUsageValue(HIDP_REPORT_TYPE ReportType, USAGE UsagePage, USHORT LinkCollection,
|
|
|
|
USAGE Usage, PULONG UsageValue, PHIDP_PREPARSED_DATA PreparsedData,
|
|
|
|
PCHAR Report, ULONG ReportLength)
|
|
|
|
{
|
2019-05-03 09:59:06 +00:00
|
|
|
WINE_HID_ELEMENT element;
|
2015-06-30 12:40:41 +00:00
|
|
|
NTSTATUS rc;
|
|
|
|
|
|
|
|
TRACE("(%i, %x, %i, %i, %p, %p, %p, %i)\n", ReportType, UsagePage, LinkCollection, Usage, UsageValue,
|
|
|
|
PreparsedData, Report, ReportLength);
|
|
|
|
|
2021-06-07 09:07:01 +00:00
|
|
|
rc = find_usage(ReportType, UsagePage, LinkCollection, Usage, PreparsedData, Report, 0, &element);
|
2015-06-30 12:40:41 +00:00
|
|
|
|
|
|
|
if (rc == HIDP_STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
return get_report_data((BYTE*)Report, ReportLength,
|
2019-05-03 09:59:06 +00:00
|
|
|
element.valueStartBit, element.bitCount, UsageValue);
|
2015-06-30 12:40:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2015-06-30 12:40:04 +00:00
|
|
|
|
2020-02-26 13:03:46 +00:00
|
|
|
NTSTATUS WINAPI HidP_GetUsageValueArray(HIDP_REPORT_TYPE ReportType, USAGE UsagePage, USHORT LinkCollection,
|
|
|
|
USAGE Usage, PCHAR UsageValue, USHORT UsageValueByteLength,
|
|
|
|
PHIDP_PREPARSED_DATA PreparsedData, PCHAR Report, ULONG ReportLength)
|
|
|
|
{
|
|
|
|
WINE_HID_ELEMENT element;
|
|
|
|
NTSTATUS rc;
|
|
|
|
|
|
|
|
TRACE("(%i, %x, %i, %i, %p, %u, %p, %p, %i)\n", ReportType, UsagePage, LinkCollection, Usage, UsageValue,
|
|
|
|
UsageValueByteLength, PreparsedData, Report, ReportLength);
|
|
|
|
|
2021-06-07 09:07:01 +00:00
|
|
|
rc = find_usage(ReportType, UsagePage, LinkCollection, Usage, PreparsedData, Report, 0, &element);
|
2020-02-26 13:03:46 +00:00
|
|
|
|
|
|
|
if (rc == HIDP_STATUS_SUCCESS)
|
|
|
|
{
|
2021-06-04 09:30:35 +00:00
|
|
|
if (element.caps.IsRange || element.caps.ReportCount <= 1 || !element.bitCount)
|
2020-02-26 13:03:46 +00:00
|
|
|
return HIDP_STATUS_NOT_VALUE_ARRAY;
|
|
|
|
|
|
|
|
return get_report_data_array((BYTE*)Report, ReportLength, element.valueStartBit, element.bitCount,
|
2021-06-04 09:30:35 +00:00
|
|
|
element.caps.ReportCount, UsageValue, UsageValueByteLength);
|
2020-02-26 13:03:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-06-30 12:40:00 +00:00
|
|
|
NTSTATUS WINAPI HidP_GetUsages(HIDP_REPORT_TYPE ReportType, USAGE UsagePage, USHORT LinkCollection,
|
|
|
|
PUSAGE UsageList, PULONG UsageLength, PHIDP_PREPARSED_DATA PreparsedData,
|
|
|
|
PCHAR Report, ULONG ReportLength)
|
|
|
|
{
|
|
|
|
PWINE_HIDP_PREPARSED_DATA data = (PWINE_HIDP_PREPARSED_DATA)PreparsedData;
|
2019-05-13 10:33:56 +00:00
|
|
|
WINE_HID_ELEMENT *elems = HID_ELEMS(data);
|
2015-06-30 12:40:00 +00:00
|
|
|
WINE_HID_REPORT *report = NULL;
|
|
|
|
BOOL found = FALSE;
|
|
|
|
USHORT b_count = 0, r_count = 0;
|
|
|
|
int i,uCount;
|
|
|
|
|
|
|
|
TRACE("(%i, %x, %i, %p, %p, %p, %p, %i)\n", ReportType, UsagePage, LinkCollection, UsageList,
|
|
|
|
UsageLength, PreparsedData, Report, ReportLength);
|
|
|
|
|
|
|
|
if (data->magic != HID_MAGIC)
|
2016-12-08 14:34:02 +00:00
|
|
|
{
|
|
|
|
*UsageLength = 0;
|
2015-06-30 12:40:00 +00:00
|
|
|
return HIDP_STATUS_INVALID_PREPARSED_DATA;
|
2016-12-08 14:34:02 +00:00
|
|
|
}
|
2015-06-30 12:40:00 +00:00
|
|
|
|
|
|
|
switch(ReportType)
|
|
|
|
{
|
|
|
|
case HidP_Input:
|
|
|
|
b_count = data->caps.NumberInputButtonCaps;
|
|
|
|
break;
|
|
|
|
case HidP_Output:
|
|
|
|
b_count = data->caps.NumberOutputButtonCaps;
|
|
|
|
break;
|
|
|
|
case HidP_Feature:
|
|
|
|
b_count = data->caps.NumberFeatureButtonCaps;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_TYPE;
|
|
|
|
}
|
2019-05-13 10:33:56 +00:00
|
|
|
r_count = data->reportCount[ReportType];
|
|
|
|
report = &data->reports[data->reportIdx[ReportType][(BYTE)Report[0]]];
|
2015-06-30 12:40:00 +00:00
|
|
|
|
2019-05-13 10:33:56 +00:00
|
|
|
if (!r_count || !b_count)
|
2015-06-30 12:40:00 +00:00
|
|
|
return HIDP_STATUS_USAGE_NOT_FOUND;
|
|
|
|
|
2019-05-13 10:33:56 +00:00
|
|
|
if (report->reportID && report->reportID != Report[0])
|
2015-06-30 12:40:00 +00:00
|
|
|
return HIDP_STATUS_REPORT_DOES_NOT_EXIST;
|
|
|
|
|
|
|
|
uCount = 0;
|
|
|
|
for (i = 0; i < report->elementCount && uCount < *UsageLength; i++)
|
|
|
|
{
|
2021-06-07 09:07:01 +00:00
|
|
|
if (elems[report->elementIdx + i].caps.BitSize == 1 &&
|
2021-06-04 09:30:35 +00:00
|
|
|
elems[report->elementIdx + i].caps.UsagePage == UsagePage)
|
2015-06-30 12:40:00 +00:00
|
|
|
{
|
|
|
|
int k;
|
2019-05-13 10:33:56 +00:00
|
|
|
WINE_HID_ELEMENT *element = &elems[report->elementIdx + i];
|
2015-06-30 12:40:00 +00:00
|
|
|
for (k=0; k < element->bitCount; k++)
|
|
|
|
{
|
|
|
|
UINT v = 0;
|
|
|
|
NTSTATUS rc = get_report_data((BYTE*)Report, ReportLength,
|
|
|
|
element->valueStartBit + k, 1, &v);
|
|
|
|
if (rc != HIDP_STATUS_SUCCESS)
|
|
|
|
return rc;
|
|
|
|
found = TRUE;
|
|
|
|
if (v)
|
|
|
|
{
|
|
|
|
if (uCount == *UsageLength)
|
|
|
|
return HIDP_STATUS_BUFFER_TOO_SMALL;
|
2021-06-04 09:30:35 +00:00
|
|
|
UsageList[uCount] = element->caps.Range.UsageMin + k;
|
2015-06-30 12:40:00 +00:00
|
|
|
uCount++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-08 14:34:02 +00:00
|
|
|
*UsageLength = uCount;
|
|
|
|
|
2015-06-30 12:40:00 +00:00
|
|
|
if (!found)
|
|
|
|
return HIDP_STATUS_USAGE_NOT_FOUND;
|
|
|
|
|
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2021-06-21 07:05:21 +00:00
|
|
|
NTSTATUS WINAPI HidP_GetValueCaps( HIDP_REPORT_TYPE report_type, HIDP_VALUE_CAPS *caps, USHORT *caps_count,
|
|
|
|
PHIDP_PREPARSED_DATA preparsed_data )
|
2015-06-30 12:39:55 +00:00
|
|
|
{
|
2021-06-21 07:05:21 +00:00
|
|
|
return HidP_GetSpecificValueCaps( report_type, 0, 0, 0, caps, caps_count, preparsed_data );
|
2015-06-30 12:39:55 +00:00
|
|
|
}
|
2015-06-30 12:40:10 +00:00
|
|
|
|
2021-06-21 07:05:22 +00:00
|
|
|
NTSTATUS WINAPI HidP_InitializeReportForID( HIDP_REPORT_TYPE report_type, UCHAR report_id,
|
|
|
|
PHIDP_PREPARSED_DATA preparsed_data, char *report_buf, ULONG report_len )
|
2015-06-30 12:40:10 +00:00
|
|
|
{
|
2021-06-21 07:05:22 +00:00
|
|
|
WINE_HIDP_PREPARSED_DATA *preparsed = (WINE_HIDP_PREPARSED_DATA *)preparsed_data;
|
|
|
|
const struct hid_value_caps *caps, *end;
|
|
|
|
NTSTATUS status;
|
2015-06-30 12:40:10 +00:00
|
|
|
|
2021-06-21 07:05:22 +00:00
|
|
|
TRACE( "report_type %d, report_id %x, preparsed_data %p, report_buf %p, report_len %u.\n", report_type,
|
|
|
|
report_id, preparsed_data, report_buf, report_len );
|
2015-06-30 12:40:10 +00:00
|
|
|
|
2021-06-21 07:05:22 +00:00
|
|
|
if (!report_len) return HIDP_STATUS_INVALID_REPORT_LENGTH;
|
2015-06-30 12:40:10 +00:00
|
|
|
|
2021-06-21 07:05:22 +00:00
|
|
|
status = get_value_caps_range( preparsed, report_type, report_len, &caps, &end );
|
|
|
|
if (status != HIDP_STATUS_SUCCESS) return status;
|
2015-06-30 12:40:10 +00:00
|
|
|
|
2021-06-21 07:05:22 +00:00
|
|
|
while (caps != end && (caps->report_id != report_id || (!caps->usage_min && !caps->usage_max))) caps++;
|
|
|
|
if (caps == end) return HIDP_STATUS_REPORT_DOES_NOT_EXIST;
|
2015-06-30 12:40:10 +00:00
|
|
|
|
2021-06-21 07:05:22 +00:00
|
|
|
memset( report_buf, 0, report_len );
|
|
|
|
report_buf[0] = report_id;
|
2015-06-30 12:40:10 +00:00
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
2015-06-30 12:40:32 +00:00
|
|
|
|
|
|
|
ULONG WINAPI HidP_MaxUsageListLength(HIDP_REPORT_TYPE ReportType, USAGE UsagePage, PHIDP_PREPARSED_DATA PreparsedData)
|
|
|
|
{
|
|
|
|
PWINE_HIDP_PREPARSED_DATA data = (PWINE_HIDP_PREPARSED_DATA)PreparsedData;
|
2019-05-13 10:33:56 +00:00
|
|
|
WINE_HID_ELEMENT *elems = HID_ELEMS(data);
|
2015-06-30 12:40:32 +00:00
|
|
|
WINE_HID_REPORT *report = NULL;
|
|
|
|
int r_count;
|
|
|
|
int i;
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
TRACE("(%i, %x, %p)\n", ReportType, UsagePage, PreparsedData);
|
|
|
|
|
|
|
|
if (data->magic != HID_MAGIC)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
switch(ReportType)
|
|
|
|
{
|
|
|
|
case HidP_Input:
|
|
|
|
report = HID_INPUT_REPORTS(data);
|
|
|
|
break;
|
|
|
|
case HidP_Output:
|
|
|
|
report = HID_OUTPUT_REPORTS(data);
|
|
|
|
break;
|
|
|
|
case HidP_Feature:
|
|
|
|
report = HID_FEATURE_REPORTS(data);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_TYPE;
|
|
|
|
}
|
2019-05-13 10:33:56 +00:00
|
|
|
r_count = data->reportCount[ReportType];
|
|
|
|
|
2015-06-30 12:40:32 +00:00
|
|
|
|
2019-05-13 10:33:56 +00:00
|
|
|
if (!r_count)
|
2015-06-30 12:40:32 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (i = 0; i < r_count; i++)
|
|
|
|
{
|
|
|
|
int j;
|
2019-05-13 10:33:56 +00:00
|
|
|
for (j = 0; j < report[i].elementCount; j++)
|
2015-06-30 12:40:32 +00:00
|
|
|
{
|
2021-06-07 09:07:01 +00:00
|
|
|
if (elems[report[i].elementIdx + j].caps.BitSize == 1 &&
|
2021-06-04 09:30:35 +00:00
|
|
|
(UsagePage == 0 || elems[report[i].elementIdx + j].caps.UsagePage == UsagePage))
|
2015-06-30 12:40:32 +00:00
|
|
|
{
|
2021-06-04 09:30:35 +00:00
|
|
|
if (elems[report[i].elementIdx + j].caps.IsRange)
|
|
|
|
count += (elems[report[i].elementIdx + j].caps.Range.UsageMax -
|
|
|
|
elems[report[i].elementIdx + j].caps.Range.UsageMin) + 1;
|
2015-06-30 12:40:32 +00:00
|
|
|
else
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
2016-09-19 13:45:34 +00:00
|
|
|
|
2018-11-23 21:09:34 +00:00
|
|
|
NTSTATUS WINAPI HidP_SetUsageValue(HIDP_REPORT_TYPE ReportType, USAGE UsagePage, USHORT LinkCollection,
|
|
|
|
USAGE Usage, ULONG UsageValue, PHIDP_PREPARSED_DATA PreparsedData,
|
|
|
|
CHAR *Report, ULONG ReportLength)
|
|
|
|
{
|
2019-05-03 09:59:06 +00:00
|
|
|
WINE_HID_ELEMENT element;
|
2018-11-23 21:09:34 +00:00
|
|
|
NTSTATUS rc;
|
|
|
|
|
|
|
|
TRACE("(%i, %x, %i, %i, %i, %p, %p, %i)\n", ReportType, UsagePage, LinkCollection, Usage, UsageValue,
|
|
|
|
PreparsedData, Report, ReportLength);
|
|
|
|
|
2021-06-07 09:07:01 +00:00
|
|
|
rc = find_usage(ReportType, UsagePage, LinkCollection, Usage, PreparsedData, Report, 0, &element);
|
2018-11-23 21:09:34 +00:00
|
|
|
|
|
|
|
if (rc == HIDP_STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
return set_report_data((BYTE*)Report, ReportLength,
|
2019-05-03 09:59:06 +00:00
|
|
|
element.valueStartBit, element.bitCount, UsageValue);
|
2018-11-23 21:09:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2021-06-14 07:15:50 +00:00
|
|
|
NTSTATUS WINAPI HidP_SetUsageValueArray( HIDP_REPORT_TYPE report_type, USAGE usage_page, USHORT collection,
|
|
|
|
USAGE usage, char *value_buf, USHORT value_len,
|
|
|
|
PHIDP_PREPARSED_DATA preparsed_data, char *report_buf, ULONG report_len )
|
|
|
|
{
|
|
|
|
FIXME( "report_type %d, usage_page %x, collection %d, usage %x, value_buf %p, value_len %u, "
|
|
|
|
"preparsed_data %p, report_buf %p, report_len %u stub!\n",
|
|
|
|
report_type, usage_page, collection, usage, value_buf, value_len, preparsed_data, report_buf, report_len );
|
|
|
|
|
|
|
|
return HIDP_STATUS_NOT_IMPLEMENTED;
|
|
|
|
}
|
2018-11-23 21:09:34 +00:00
|
|
|
|
2019-05-03 09:59:02 +00:00
|
|
|
NTSTATUS WINAPI HidP_SetUsages(HIDP_REPORT_TYPE ReportType, USAGE UsagePage, USHORT LinkCollection,
|
|
|
|
PUSAGE UsageList, PULONG UsageLength, PHIDP_PREPARSED_DATA PreparsedData,
|
|
|
|
PCHAR Report, ULONG ReportLength)
|
|
|
|
{
|
2019-05-03 09:59:06 +00:00
|
|
|
WINE_HID_ELEMENT element;
|
2019-05-03 09:59:02 +00:00
|
|
|
NTSTATUS rc;
|
|
|
|
ULONG i;
|
|
|
|
|
|
|
|
TRACE("(%i, %x, %i, %p, %p, %p, %p, %i)\n", ReportType, UsagePage, LinkCollection, UsageList,
|
|
|
|
UsageLength, PreparsedData, Report, ReportLength);
|
|
|
|
|
|
|
|
for (i = 0; i < *UsageLength; i++)
|
|
|
|
{
|
2019-05-13 10:34:19 +00:00
|
|
|
rc = find_usage(ReportType, UsagePage, LinkCollection,
|
2021-06-07 09:07:01 +00:00
|
|
|
UsageList[i], PreparsedData, Report, 1, &element);
|
2019-05-03 09:59:02 +00:00
|
|
|
if (rc == HIDP_STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
rc = set_report_data((BYTE*)Report, ReportLength,
|
2019-05-03 09:59:06 +00:00
|
|
|
element.valueStartBit, element.bitCount, -1);
|
2019-05-14 10:43:45 +00:00
|
|
|
}
|
2019-05-03 09:59:02 +00:00
|
|
|
|
2019-05-14 10:43:45 +00:00
|
|
|
if (rc != HIDP_STATUS_SUCCESS)
|
|
|
|
{
|
|
|
|
*UsageLength = i;
|
|
|
|
return rc;
|
2019-05-03 09:59:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-19 13:45:34 +00:00
|
|
|
NTSTATUS WINAPI HidP_TranslateUsagesToI8042ScanCodes(USAGE *ChangedUsageList,
|
|
|
|
ULONG UsageListLength, HIDP_KEYBOARD_DIRECTION KeyAction,
|
|
|
|
HIDP_KEYBOARD_MODIFIER_STATE *ModifierState,
|
|
|
|
PHIDP_INSERT_SCANCODES InsertCodesProcedure, VOID *InsertCodesContext)
|
|
|
|
{
|
|
|
|
FIXME("stub: %p, %i, %i, %p, %p, %p\n", ChangedUsageList, UsageListLength,
|
|
|
|
KeyAction, ModifierState, InsertCodesProcedure, InsertCodesContext);
|
|
|
|
|
|
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
|
|
}
|
2016-11-03 16:39:49 +00:00
|
|
|
|
2021-06-21 07:05:23 +00:00
|
|
|
static NTSTATUS get_button_caps( const struct hid_value_caps *caps, void *user )
|
2016-11-15 14:05:38 +00:00
|
|
|
{
|
2021-06-21 07:05:23 +00:00
|
|
|
HIDP_BUTTON_CAPS **iter = user, *dst = *iter;
|
|
|
|
dst->UsagePage = caps->usage_page;
|
|
|
|
dst->ReportID = caps->report_id;
|
|
|
|
dst->LinkCollection = caps->link_collection;
|
|
|
|
dst->LinkUsagePage = caps->link_usage_page;
|
|
|
|
dst->LinkUsage = caps->link_usage;
|
|
|
|
dst->BitField = caps->bit_field;
|
|
|
|
dst->IsAlias = FALSE;
|
|
|
|
dst->IsAbsolute = HID_VALUE_CAPS_IS_ABSOLUTE( caps );
|
|
|
|
if (!(dst->IsRange = caps->is_range))
|
2016-11-15 14:05:38 +00:00
|
|
|
{
|
2021-06-21 07:05:23 +00:00
|
|
|
dst->NotRange.Usage = caps->usage_min;
|
|
|
|
dst->NotRange.DataIndex = caps->data_index_min;
|
2016-11-15 14:05:38 +00:00
|
|
|
}
|
2021-06-21 07:05:23 +00:00
|
|
|
else
|
2016-11-15 14:05:38 +00:00
|
|
|
{
|
2021-06-21 07:05:23 +00:00
|
|
|
dst->Range.UsageMin = caps->usage_min;
|
|
|
|
dst->Range.UsageMax = caps->usage_max;
|
|
|
|
dst->Range.DataIndexMin = caps->data_index_min;
|
|
|
|
dst->Range.DataIndexMax = caps->data_index_max;
|
2016-11-15 14:05:38 +00:00
|
|
|
}
|
2021-06-21 07:05:23 +00:00
|
|
|
if (!(dst->IsStringRange = caps->is_string_range))
|
|
|
|
dst->NotRange.StringIndex = caps->string_min;
|
|
|
|
else
|
2016-11-15 14:05:38 +00:00
|
|
|
{
|
2021-06-21 07:05:23 +00:00
|
|
|
dst->Range.StringMin = caps->string_min;
|
|
|
|
dst->Range.StringMax = caps->string_max;
|
2016-11-15 14:05:38 +00:00
|
|
|
}
|
2021-06-21 07:05:23 +00:00
|
|
|
if ((dst->IsDesignatorRange = caps->is_designator_range))
|
|
|
|
dst->NotRange.DesignatorIndex = caps->designator_min;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dst->Range.DesignatorMin = caps->designator_min;
|
|
|
|
dst->Range.DesignatorMax = caps->designator_max;
|
|
|
|
}
|
|
|
|
*iter += 1;
|
2016-11-15 14:05:38 +00:00
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2021-06-21 07:05:23 +00:00
|
|
|
NTSTATUS WINAPI HidP_GetSpecificButtonCaps( HIDP_REPORT_TYPE report_type, USAGE usage_page, USHORT collection,
|
|
|
|
USAGE usage, HIDP_BUTTON_CAPS *caps, USHORT *caps_count,
|
|
|
|
PHIDP_PREPARSED_DATA preparsed_data )
|
|
|
|
{
|
|
|
|
WINE_HIDP_PREPARSED_DATA *preparsed = (WINE_HIDP_PREPARSED_DATA *)preparsed_data;
|
|
|
|
const struct caps_filter filter = {.buttons = TRUE, .usage_page = usage_page, .collection = collection, .usage = usage};
|
|
|
|
|
|
|
|
TRACE( "report_type %d, usage_page %x, collection %d, usage %x, caps %p, caps_count %p, preparsed_data %p.\n",
|
|
|
|
report_type, usage_page, collection, usage, caps, caps_count, preparsed_data );
|
|
|
|
|
|
|
|
return enum_value_caps( preparsed, report_type, &filter, get_button_caps, &caps, caps_count );
|
|
|
|
}
|
2016-11-15 14:05:38 +00:00
|
|
|
|
2016-11-03 16:39:49 +00:00
|
|
|
NTSTATUS WINAPI HidP_GetSpecificValueCaps(HIDP_REPORT_TYPE ReportType,
|
|
|
|
USAGE UsagePage, USHORT LinkCollection, USAGE Usage,
|
|
|
|
HIDP_VALUE_CAPS *ValueCaps, USHORT *ValueCapsLength, PHIDP_PREPARSED_DATA PreparsedData)
|
|
|
|
{
|
|
|
|
WINE_HIDP_PREPARSED_DATA *data = (PWINE_HIDP_PREPARSED_DATA)PreparsedData;
|
2019-05-13 10:33:56 +00:00
|
|
|
WINE_HID_ELEMENT *elems = HID_ELEMS(data);
|
2016-11-03 16:39:49 +00:00
|
|
|
WINE_HID_REPORT *report = NULL;
|
|
|
|
USHORT v_count = 0, r_count = 0;
|
|
|
|
int i,j,u;
|
|
|
|
|
|
|
|
TRACE("(%i, 0x%x, %i, 0x%x, %p %p %p)\n", ReportType, UsagePage, LinkCollection,
|
|
|
|
Usage, ValueCaps, ValueCapsLength, PreparsedData);
|
|
|
|
|
|
|
|
if (data->magic != HID_MAGIC)
|
|
|
|
return HIDP_STATUS_INVALID_PREPARSED_DATA;
|
|
|
|
|
|
|
|
switch(ReportType)
|
|
|
|
{
|
|
|
|
case HidP_Input:
|
|
|
|
v_count = data->caps.NumberInputValueCaps;
|
|
|
|
report = HID_INPUT_REPORTS(data);
|
|
|
|
break;
|
|
|
|
case HidP_Output:
|
|
|
|
v_count = data->caps.NumberOutputValueCaps;
|
|
|
|
report = HID_OUTPUT_REPORTS(data);
|
|
|
|
break;
|
|
|
|
case HidP_Feature:
|
|
|
|
v_count = data->caps.NumberFeatureValueCaps;
|
|
|
|
report = HID_FEATURE_REPORTS(data);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_TYPE;
|
|
|
|
}
|
2019-05-13 10:33:56 +00:00
|
|
|
r_count = data->reportCount[ReportType];
|
2016-11-03 16:39:49 +00:00
|
|
|
|
2019-05-13 10:33:56 +00:00
|
|
|
if (!r_count || !v_count)
|
2016-11-03 16:39:49 +00:00
|
|
|
{
|
|
|
|
*ValueCapsLength = 0;
|
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
v_count = min(v_count, *ValueCapsLength);
|
|
|
|
|
|
|
|
u = 0;
|
|
|
|
for (j = 0; j < r_count && u < v_count; j++)
|
|
|
|
{
|
2019-05-13 10:33:56 +00:00
|
|
|
for (i = 0; i < report[j].elementCount && u < v_count; i++)
|
2016-11-03 16:39:49 +00:00
|
|
|
{
|
2021-06-07 09:07:01 +00:00
|
|
|
if (elems[report[j].elementIdx + i].caps.BitSize != 1 &&
|
2021-06-04 09:30:35 +00:00
|
|
|
(UsagePage == 0 || UsagePage == elems[report[j].elementIdx + i].caps.UsagePage) &&
|
|
|
|
(LinkCollection == 0 || LinkCollection == elems[report[j].elementIdx + i].caps.LinkCollection) &&
|
|
|
|
(Usage == 0 || Usage == elems[report[j].elementIdx + i].caps.NotRange.Usage))
|
2016-11-03 16:39:49 +00:00
|
|
|
{
|
2021-06-04 09:30:35 +00:00
|
|
|
ValueCaps[u++] = elems[report[j].elementIdx + i].caps;
|
2016-11-03 16:39:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
TRACE("Matched %i usages\n", u);
|
|
|
|
|
|
|
|
*ValueCapsLength = u;
|
|
|
|
|
|
|
|
return HIDP_STATUS_SUCCESS;
|
|
|
|
}
|
2017-01-25 13:07:04 +00:00
|
|
|
|
|
|
|
NTSTATUS WINAPI HidP_GetUsagesEx(HIDP_REPORT_TYPE ReportType, USHORT LinkCollection, USAGE_AND_PAGE *ButtonList,
|
|
|
|
ULONG *UsageLength, PHIDP_PREPARSED_DATA PreparsedData, CHAR *Report, ULONG ReportLength)
|
|
|
|
{
|
|
|
|
WINE_HIDP_PREPARSED_DATA *data = (WINE_HIDP_PREPARSED_DATA*)PreparsedData;
|
2019-05-13 10:33:56 +00:00
|
|
|
WINE_HID_ELEMENT *elems = HID_ELEMS(data);
|
2017-01-25 13:07:04 +00:00
|
|
|
WINE_HID_REPORT *report = NULL;
|
|
|
|
USHORT b_count = 0, r_count = 0;
|
|
|
|
int i,uCount = 0;
|
|
|
|
NTSTATUS rc;
|
|
|
|
|
|
|
|
TRACE("(%i, %i, %p, %p(%i), %p, %p, %i)\n", ReportType, LinkCollection, ButtonList,
|
|
|
|
UsageLength, *UsageLength, PreparsedData, Report, ReportLength);
|
|
|
|
|
|
|
|
if (data->magic != HID_MAGIC)
|
|
|
|
return HIDP_STATUS_INVALID_PREPARSED_DATA;
|
|
|
|
|
|
|
|
switch(ReportType)
|
|
|
|
{
|
|
|
|
case HidP_Input:
|
|
|
|
b_count = data->caps.NumberInputButtonCaps;
|
|
|
|
break;
|
|
|
|
case HidP_Output:
|
|
|
|
b_count = data->caps.NumberOutputButtonCaps;
|
|
|
|
break;
|
|
|
|
case HidP_Feature:
|
|
|
|
b_count = data->caps.NumberFeatureButtonCaps;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_TYPE;
|
|
|
|
}
|
2019-05-13 10:33:56 +00:00
|
|
|
r_count = data->reportCount[ReportType];
|
|
|
|
report = &data->reports[data->reportIdx[ReportType][(BYTE)Report[0]]];
|
2017-01-25 13:07:04 +00:00
|
|
|
|
2019-05-13 10:33:56 +00:00
|
|
|
if (!r_count || !b_count)
|
2017-01-25 13:07:04 +00:00
|
|
|
return HIDP_STATUS_USAGE_NOT_FOUND;
|
|
|
|
|
2019-05-13 10:33:56 +00:00
|
|
|
if (report->reportID && report->reportID != Report[0])
|
2017-01-25 13:07:04 +00:00
|
|
|
return HIDP_STATUS_REPORT_DOES_NOT_EXIST;
|
|
|
|
|
|
|
|
for (i = 0; i < report->elementCount; i++)
|
|
|
|
{
|
2021-06-07 09:07:01 +00:00
|
|
|
if (elems[report->elementIdx + i].caps.BitSize == 1)
|
2017-01-25 13:07:04 +00:00
|
|
|
{
|
|
|
|
int k;
|
2019-05-13 10:33:56 +00:00
|
|
|
WINE_HID_ELEMENT *element = &elems[report->elementIdx + i];
|
2017-01-25 13:07:04 +00:00
|
|
|
for (k=0; k < element->bitCount; k++)
|
|
|
|
{
|
|
|
|
UINT v = 0;
|
|
|
|
NTSTATUS rc = get_report_data((BYTE*)Report, ReportLength,
|
|
|
|
element->valueStartBit + k, 1, &v);
|
|
|
|
if (rc != HIDP_STATUS_SUCCESS)
|
|
|
|
return rc;
|
|
|
|
if (v)
|
|
|
|
{
|
|
|
|
if (uCount < *UsageLength)
|
|
|
|
{
|
2021-06-04 09:30:35 +00:00
|
|
|
ButtonList[uCount].Usage = element->caps.Range.UsageMin + k;
|
|
|
|
ButtonList[uCount].UsagePage = element->caps.UsagePage;
|
2017-01-25 13:07:04 +00:00
|
|
|
}
|
|
|
|
uCount++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TRACE("Returning %i usages\n", uCount);
|
|
|
|
|
|
|
|
if (*UsageLength < uCount)
|
|
|
|
rc = HIDP_STATUS_BUFFER_TOO_SMALL;
|
|
|
|
else
|
|
|
|
rc = HIDP_STATUS_SUCCESS;
|
|
|
|
|
|
|
|
*UsageLength = uCount;
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
2017-01-27 13:17:53 +00:00
|
|
|
|
|
|
|
ULONG WINAPI HidP_MaxDataListLength(HIDP_REPORT_TYPE ReportType, PHIDP_PREPARSED_DATA PreparsedData)
|
|
|
|
{
|
|
|
|
WINE_HIDP_PREPARSED_DATA *data = (WINE_HIDP_PREPARSED_DATA *)PreparsedData;
|
|
|
|
TRACE("(%i, %p)\n", ReportType, PreparsedData);
|
|
|
|
if (data->magic != HID_MAGIC)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
switch(ReportType)
|
|
|
|
{
|
|
|
|
case HidP_Input:
|
|
|
|
return data->caps.NumberInputDataIndices;
|
|
|
|
case HidP_Output:
|
|
|
|
return data->caps.NumberOutputDataIndices;
|
|
|
|
case HidP_Feature:
|
|
|
|
return data->caps.NumberFeatureDataIndices;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2017-01-27 13:18:01 +00:00
|
|
|
|
|
|
|
NTSTATUS WINAPI HidP_GetData(HIDP_REPORT_TYPE ReportType, HIDP_DATA *DataList, ULONG *DataLength,
|
|
|
|
PHIDP_PREPARSED_DATA PreparsedData,CHAR *Report, ULONG ReportLength)
|
|
|
|
{
|
|
|
|
WINE_HIDP_PREPARSED_DATA *data = (WINE_HIDP_PREPARSED_DATA*)PreparsedData;
|
2019-05-13 10:33:56 +00:00
|
|
|
WINE_HID_ELEMENT *elems = HID_ELEMS(data);
|
2017-01-27 13:18:01 +00:00
|
|
|
WINE_HID_REPORT *report = NULL;
|
|
|
|
USHORT r_count = 0;
|
|
|
|
int i,uCount = 0;
|
|
|
|
NTSTATUS rc;
|
|
|
|
|
|
|
|
TRACE("(%i, %p, %p(%i), %p, %p, %i)\n", ReportType, DataList, DataLength,
|
|
|
|
DataLength?*DataLength:0, PreparsedData, Report, ReportLength);
|
|
|
|
|
|
|
|
if (data->magic != HID_MAGIC)
|
|
|
|
return 0;
|
|
|
|
|
2019-05-13 10:33:56 +00:00
|
|
|
if (ReportType != HidP_Input && ReportType != HidP_Output && ReportType != HidP_Feature)
|
|
|
|
return HIDP_STATUS_INVALID_REPORT_TYPE;
|
2017-01-27 13:18:01 +00:00
|
|
|
|
2019-05-13 10:33:56 +00:00
|
|
|
r_count = data->reportCount[ReportType];
|
|
|
|
report = &data->reports[data->reportIdx[ReportType][(BYTE)Report[0]]];
|
2017-01-27 13:18:01 +00:00
|
|
|
|
2019-05-13 10:33:56 +00:00
|
|
|
if (!r_count || (report->reportID && report->reportID != Report[0]))
|
2017-01-27 13:18:01 +00:00
|
|
|
return HIDP_STATUS_REPORT_DOES_NOT_EXIST;
|
|
|
|
|
|
|
|
for (i = 0; i < report->elementCount; i++)
|
|
|
|
{
|
2019-05-13 10:33:56 +00:00
|
|
|
WINE_HID_ELEMENT *element = &elems[report->elementIdx + i];
|
2021-06-07 09:07:01 +00:00
|
|
|
if (element->caps.BitSize == 1)
|
2017-01-27 13:18:01 +00:00
|
|
|
{
|
|
|
|
int k;
|
|
|
|
for (k=0; k < element->bitCount; k++)
|
|
|
|
{
|
|
|
|
UINT v = 0;
|
|
|
|
NTSTATUS rc = get_report_data((BYTE*)Report, ReportLength,
|
|
|
|
element->valueStartBit + k, 1, &v);
|
|
|
|
if (rc != HIDP_STATUS_SUCCESS)
|
|
|
|
return rc;
|
|
|
|
if (v)
|
|
|
|
{
|
|
|
|
if (uCount < *DataLength)
|
|
|
|
{
|
2021-06-04 09:30:35 +00:00
|
|
|
DataList[uCount].DataIndex = element->caps.Range.DataIndexMin + k;
|
2021-06-04 09:30:33 +00:00
|
|
|
DataList[uCount].On = v;
|
2017-01-27 13:18:01 +00:00
|
|
|
}
|
|
|
|
uCount++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (uCount < *DataLength)
|
|
|
|
{
|
|
|
|
UINT v;
|
|
|
|
NTSTATUS rc = get_report_data((BYTE*)Report, ReportLength,
|
|
|
|
element->valueStartBit, element->bitCount, &v);
|
|
|
|
if (rc != HIDP_STATUS_SUCCESS)
|
|
|
|
return rc;
|
2021-06-04 09:30:35 +00:00
|
|
|
DataList[uCount].DataIndex = element->caps.NotRange.DataIndex;
|
2021-06-04 09:30:33 +00:00
|
|
|
DataList[uCount].RawValue = v;
|
2017-01-27 13:18:01 +00:00
|
|
|
}
|
|
|
|
uCount++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*DataLength < uCount)
|
|
|
|
rc = HIDP_STATUS_BUFFER_TOO_SMALL;
|
|
|
|
else
|
|
|
|
rc = HIDP_STATUS_SUCCESS;
|
|
|
|
|
|
|
|
*DataLength = uCount;
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
2020-02-11 18:13:44 +00:00
|
|
|
|
|
|
|
NTSTATUS WINAPI HidP_GetLinkCollectionNodes(HIDP_LINK_COLLECTION_NODE *LinkCollectionNode,
|
|
|
|
ULONG *LinkCollectionNodeLength, PHIDP_PREPARSED_DATA PreparsedData)
|
|
|
|
{
|
2020-02-11 18:13:46 +00:00
|
|
|
WINE_HIDP_PREPARSED_DATA *data = (WINE_HIDP_PREPARSED_DATA*)PreparsedData;
|
|
|
|
WINE_HID_LINK_COLLECTION_NODE *nodes = HID_NODES(data);
|
|
|
|
ULONG i;
|
2020-02-11 18:13:44 +00:00
|
|
|
|
2020-02-11 18:13:46 +00:00
|
|
|
TRACE("(%p, %p, %p)\n", LinkCollectionNode, LinkCollectionNodeLength, PreparsedData);
|
2020-02-11 18:13:44 +00:00
|
|
|
|
2021-06-10 08:06:24 +00:00
|
|
|
if (data->magic != HID_MAGIC)
|
|
|
|
return HIDP_STATUS_INVALID_PREPARSED_DATA;
|
|
|
|
|
2020-02-11 18:13:46 +00:00
|
|
|
if (*LinkCollectionNodeLength < data->caps.NumberLinkCollectionNodes)
|
|
|
|
return HIDP_STATUS_BUFFER_TOO_SMALL;
|
|
|
|
|
|
|
|
for (i = 0; i < data->caps.NumberLinkCollectionNodes; ++i)
|
|
|
|
{
|
|
|
|
LinkCollectionNode[i].LinkUsage = nodes[i].LinkUsage;
|
|
|
|
LinkCollectionNode[i].LinkUsagePage = nodes[i].LinkUsagePage;
|
|
|
|
LinkCollectionNode[i].Parent = nodes[i].Parent;
|
|
|
|
LinkCollectionNode[i].NumberOfChildren = nodes[i].NumberOfChildren;
|
|
|
|
LinkCollectionNode[i].NextSibling = nodes[i].NextSibling;
|
|
|
|
LinkCollectionNode[i].FirstChild = nodes[i].FirstChild;
|
|
|
|
LinkCollectionNode[i].CollectionType = nodes[i].CollectionType;
|
|
|
|
LinkCollectionNode[i].IsAlias = nodes[i].IsAlias;
|
|
|
|
}
|
|
|
|
*LinkCollectionNodeLength = data->caps.NumberLinkCollectionNodes;
|
|
|
|
|
|
|
|
return HIDP_STATUS_SUCCESS;
|
2020-02-11 18:13:44 +00:00
|
|
|
}
|