/* * 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 "config.h" #include #define NONAMELESSUNION #include "ntstatus.h" #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" #include "parse.h" #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(hidp); 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); *value = (report[byte_index] & mask); } else { ULONG byte_index = startBit / 8; ULONG data = 0; ULONG remainingBits = valueSize; ULONG shift = 0; ULONG begin_offset = startBit % 8; while (remainingBits) { if (remainingBits >= 8) { BYTE mask = 0xff << begin_offset; data |= (report[byte_index] & mask) << shift; byte_index ++; remainingBits -= (8-begin_offset); shift += (8-begin_offset); begin_offset = 0; } else if (remainingBits > 0) { BYTE mask = (0xff >> (8-remainingBits)) << begin_offset; data |= (report[byte_index] & mask) << shift; remainingBits = 0; } } *value = data; } return HIDP_STATUS_SUCCESS; } 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; } NTSTATUS WINAPI HidP_GetButtonCaps(HIDP_REPORT_TYPE ReportType, PHIDP_BUTTON_CAPS ButtonCaps, PUSHORT ButtonCapsLength, PHIDP_PREPARSED_DATA PreparsedData) { PWINE_HIDP_PREPARSED_DATA data = (PWINE_HIDP_PREPARSED_DATA)PreparsedData; WINE_HID_REPORT *report = NULL; USHORT b_count = 0, r_count = 0; int i,j,u; TRACE("(%i, %p, %p, %p)\n",ReportType, ButtonCaps, ButtonCapsLength, PreparsedData); if (data->magic != HID_MAGIC) return HIDP_STATUS_INVALID_PREPARSED_DATA; switch(ReportType) { case HidP_Input: b_count = data->caps.NumberInputButtonCaps; r_count = data->dwInputReportCount; report = HID_INPUT_REPORTS(data); break; case HidP_Output: b_count = data->caps.NumberOutputButtonCaps; r_count = data->dwOutputReportCount; report = HID_OUTPUT_REPORTS(data); break; case HidP_Feature: b_count = data->caps.NumberFeatureButtonCaps; r_count = data->dwFeatureReportCount; report = HID_FEATURE_REPORTS(data); break; default: return HIDP_STATUS_INVALID_REPORT_TYPE; } if (!r_count || !b_count || !report) { *ButtonCapsLength = 0; return HIDP_STATUS_SUCCESS; } b_count = min(b_count, *ButtonCapsLength); u = 0; for (j = 0; j < r_count && u < b_count; j++) { for (i = 0; i < report->elementCount && u < b_count; i++) { if (report->Elements[i].ElementType == ButtonElement) ButtonCaps[u++] = report->Elements[i].caps.button; } report = HID_NEXT_REPORT(data, report); } *ButtonCapsLength = b_count; return HIDP_STATUS_SUCCESS; } NTSTATUS WINAPI HidP_GetCaps(PHIDP_PREPARSED_DATA PreparsedData, PHIDP_CAPS Capabilities) { PWINE_HIDP_PREPARSED_DATA data = (PWINE_HIDP_PREPARSED_DATA)PreparsedData; TRACE("(%p, %p)\n",PreparsedData, Capabilities); if (data->magic != HID_MAGIC) return HIDP_STATUS_INVALID_PREPARSED_DATA; *Capabilities = data->caps; return HIDP_STATUS_SUCCESS; } static NTSTATUS find_value(HIDP_REPORT_TYPE ReportType, USAGE UsagePage, USHORT LinkCollection, USAGE Usage, PHIDP_PREPARSED_DATA PreparsedData, PCHAR Report, WINE_HID_ELEMENT **element) { PWINE_HIDP_PREPARSED_DATA data = (PWINE_HIDP_PREPARSED_DATA)PreparsedData; WINE_HID_REPORT *report = NULL; USHORT v_count = 0, r_count = 0; int i; TRACE("(%i, %x, %i, %i, %p, %p)\n", ReportType, UsagePage, LinkCollection, Usage, PreparsedData, Report); if (data->magic != HID_MAGIC) return HIDP_STATUS_INVALID_PREPARSED_DATA; switch(ReportType) { case HidP_Input: v_count = data->caps.NumberInputValueCaps; r_count = data->dwInputReportCount; report = HID_INPUT_REPORTS(data); break; case HidP_Output: v_count = data->caps.NumberOutputValueCaps; r_count = data->dwOutputReportCount; report = HID_OUTPUT_REPORTS(data); break; case HidP_Feature: v_count = data->caps.NumberFeatureValueCaps; r_count = data->dwFeatureReportCount; report = HID_FEATURE_REPORTS(data); break; default: return HIDP_STATUS_INVALID_REPORT_TYPE; } if (!r_count || !v_count || !report) return HIDP_STATUS_USAGE_NOT_FOUND; for (i = 0; i < r_count; i++) { if (!report->reportID || report->reportID == Report[0]) break; report = HID_NEXT_REPORT(data, report); } if (i == r_count) return HIDP_STATUS_REPORT_DOES_NOT_EXIST; for (i = 0; i < report->elementCount; i++) { if (report->Elements[i].ElementType == ValueElement && report->Elements[i].caps.value.UsagePage == UsagePage && report->Elements[i].caps.value.u.NotRange.Usage == Usage) { *element = &report->Elements[i]; return HIDP_STATUS_SUCCESS; } } return HIDP_STATUS_USAGE_NOT_FOUND; } static LONG sign_extend(ULONG value, const WINE_HID_ELEMENT *element) { UINT bit_count = element->bitCount; if ((value & (1 << (bit_count - 1))) && element->ElementType == ValueElement && element->caps.value.LogicalMin < 0) { value -= (1 << bit_count); } return value; } static LONG logical_to_physical(LONG value, const WINE_HID_ELEMENT *element) { if (element->caps.value.PhysicalMin || element->caps.value.PhysicalMax) { value = (((ULONGLONG)(value - element->caps.value.LogicalMin) * (element->caps.value.PhysicalMax - element->caps.value.PhysicalMin)) / (element->caps.value.LogicalMax - element->caps.value.LogicalMin)) + element->caps.value.PhysicalMin; } return value; } 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; WINE_HID_ELEMENT *element; TRACE("(%i, %x, %i, %i, %p, %p, %p, %i)\n", ReportType, UsagePage, LinkCollection, Usage, UsageValue, PreparsedData, Report, ReportLength); rc = find_value(ReportType, UsagePage, LinkCollection, Usage, PreparsedData, Report, &element); if (rc == HIDP_STATUS_SUCCESS) { ULONG rawValue; rc = get_report_data((BYTE*)Report, ReportLength, element->valueStartBit, element->bitCount, &rawValue); if (rc != HIDP_STATUS_SUCCESS) return rc; *UsageValue = logical_to_physical(sign_extend(rawValue, element), element); } 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) { WINE_HID_ELEMENT *element; NTSTATUS rc; TRACE("(%i, %x, %i, %i, %p, %p, %p, %i)\n", ReportType, UsagePage, LinkCollection, Usage, UsageValue, PreparsedData, Report, ReportLength); rc = find_value(ReportType, UsagePage, LinkCollection, Usage, PreparsedData, Report, &element); if (rc == HIDP_STATUS_SUCCESS) { return get_report_data((BYTE*)Report, ReportLength, element->valueStartBit, element->bitCount, UsageValue); } return rc; } 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; 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) { *UsageLength = 0; return HIDP_STATUS_INVALID_PREPARSED_DATA; } switch(ReportType) { case HidP_Input: b_count = data->caps.NumberInputButtonCaps; r_count = data->dwInputReportCount; report = HID_INPUT_REPORTS(data); break; case HidP_Output: b_count = data->caps.NumberOutputButtonCaps; r_count = data->dwOutputReportCount; report = HID_OUTPUT_REPORTS(data); break; case HidP_Feature: b_count = data->caps.NumberFeatureButtonCaps; r_count = data->dwFeatureReportCount; report = HID_FEATURE_REPORTS(data); break; default: return HIDP_STATUS_INVALID_REPORT_TYPE; } if (!r_count || !b_count || !report) return HIDP_STATUS_USAGE_NOT_FOUND; for (i = 0; i < r_count; i++) { if (!report->reportID || report->reportID == Report[0]) break; report = HID_NEXT_REPORT(data, report); } if (i == r_count) return HIDP_STATUS_REPORT_DOES_NOT_EXIST; uCount = 0; for (i = 0; i < report->elementCount && uCount < *UsageLength; i++) { if (report->Elements[i].ElementType == ButtonElement && report->Elements[i].caps.button.UsagePage == UsagePage) { int k; WINE_HID_ELEMENT *element = &report->Elements[i]; 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; UsageList[uCount] = element->caps.button.u.Range.UsageMin + k; uCount++; } } } } *UsageLength = uCount; if (!found) return HIDP_STATUS_USAGE_NOT_FOUND; return HIDP_STATUS_SUCCESS; } NTSTATUS WINAPI HidP_GetValueCaps(HIDP_REPORT_TYPE ReportType, PHIDP_VALUE_CAPS ValueCaps, PUSHORT ValueCapsLength, PHIDP_PREPARSED_DATA PreparsedData) { PWINE_HIDP_PREPARSED_DATA data = (PWINE_HIDP_PREPARSED_DATA)PreparsedData; WINE_HID_REPORT *report = NULL; USHORT v_count = 0, r_count = 0; int i,j,u; TRACE("(%i, %p, %p, %p)\n", ReportType, ValueCaps, ValueCapsLength, PreparsedData); if (data->magic != HID_MAGIC) return HIDP_STATUS_INVALID_PREPARSED_DATA; switch(ReportType) { case HidP_Input: v_count = data->caps.NumberInputValueCaps; r_count = data->dwInputReportCount; report = HID_INPUT_REPORTS(data); break; case HidP_Output: v_count = data->caps.NumberOutputValueCaps; r_count = data->dwOutputReportCount; report = HID_OUTPUT_REPORTS(data); break; case HidP_Feature: v_count = data->caps.NumberFeatureValueCaps; r_count = data->dwFeatureReportCount; report = HID_FEATURE_REPORTS(data); break; default: return HIDP_STATUS_INVALID_REPORT_TYPE; } if (!r_count || !v_count || !report) { *ValueCapsLength = 0; return HIDP_STATUS_SUCCESS; } v_count = min(v_count, *ValueCapsLength); u = 0; for (j = 0; j < r_count && u < v_count; j++) { for (i = 0; i < report->elementCount && u < v_count; i++) { if (report->Elements[i].ElementType == ValueElement) ValueCaps[u++] = report->Elements[i].caps.value; } report = HID_NEXT_REPORT(data, report); } *ValueCapsLength = v_count; return HIDP_STATUS_SUCCESS; } NTSTATUS WINAPI HidP_InitializeReportForID(HIDP_REPORT_TYPE ReportType, UCHAR ReportID, PHIDP_PREPARSED_DATA PreparsedData, PCHAR Report, ULONG ReportLength) { int size; PWINE_HIDP_PREPARSED_DATA data = (PWINE_HIDP_PREPARSED_DATA)PreparsedData; WINE_HID_REPORT *report = NULL; BOOL found=FALSE; int r_count; int i; TRACE("(%i, %i, %p, %p, %i)\n",ReportType, ReportID, PreparsedData, Report, ReportLength); if (data->magic != HID_MAGIC) return HIDP_STATUS_INVALID_PREPARSED_DATA; switch(ReportType) { case HidP_Input: size = data->caps.InputReportByteLength; report = HID_INPUT_REPORTS(data); r_count = data->dwInputReportCount; break; case HidP_Output: size = data->caps.OutputReportByteLength; report = HID_OUTPUT_REPORTS(data); r_count = data->dwOutputReportCount; break; case HidP_Feature: size = data->caps.FeatureReportByteLength; report = HID_FEATURE_REPORTS(data); r_count = data->dwFeatureReportCount; break; default: return HIDP_STATUS_INVALID_REPORT_TYPE; } if (!r_count || !size || !report) return HIDP_STATUS_REPORT_DOES_NOT_EXIST; if (size != ReportLength) return HIDP_STATUS_INVALID_REPORT_LENGTH; ZeroMemory(Report, size); for (i = 0; i < r_count; i++) { if (report->reportID == ReportID) { found = TRUE; if (report->reportID) Report[0] = ReportID; /* TODO: Handle null and default values */ } report = HID_NEXT_REPORT(data, report); } if (!found) return HIDP_STATUS_REPORT_DOES_NOT_EXIST; return HIDP_STATUS_SUCCESS; } ULONG WINAPI HidP_MaxUsageListLength(HIDP_REPORT_TYPE ReportType, USAGE UsagePage, PHIDP_PREPARSED_DATA PreparsedData) { PWINE_HIDP_PREPARSED_DATA data = (PWINE_HIDP_PREPARSED_DATA)PreparsedData; 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); r_count = data->dwInputReportCount; break; case HidP_Output: report = HID_OUTPUT_REPORTS(data); r_count = data->dwOutputReportCount; break; case HidP_Feature: report = HID_FEATURE_REPORTS(data); r_count = data->dwFeatureReportCount; break; default: return HIDP_STATUS_INVALID_REPORT_TYPE; } if (!r_count || !report) return 0; for (i = 0; i < r_count; i++) { int j; for (j = 0; j < report->elementCount; j++) { if (report->Elements[j].ElementType == ButtonElement && (UsagePage == 0 || report->Elements[j].caps.button.UsagePage == UsagePage)) { if (report->Elements[j].caps.button.IsRange) count += (report->Elements[j].caps.button.u.Range.UsageMax - report->Elements[j].caps.button.u.Range.UsageMin) + 1; else count++; } } report = HID_NEXT_REPORT(data, report); } return count; } NTSTATUS WINAPI HidP_SetUsageValue(HIDP_REPORT_TYPE ReportType, USAGE UsagePage, USHORT LinkCollection, USAGE Usage, ULONG UsageValue, PHIDP_PREPARSED_DATA PreparsedData, CHAR *Report, ULONG ReportLength) { WINE_HID_ELEMENT *element; NTSTATUS rc; TRACE("(%i, %x, %i, %i, %i, %p, %p, %i)\n", ReportType, UsagePage, LinkCollection, Usage, UsageValue, PreparsedData, Report, ReportLength); rc = find_value(ReportType, UsagePage, LinkCollection, Usage, PreparsedData, Report, &element); if (rc == HIDP_STATUS_SUCCESS) { return set_report_data((BYTE*)Report, ReportLength, element->valueStartBit, element->bitCount, UsageValue); } return rc; } 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; } NTSTATUS WINAPI HidP_GetSpecificButtonCaps(HIDP_REPORT_TYPE ReportType, USAGE UsagePage, USHORT LinkCollection, USAGE Usage, HIDP_BUTTON_CAPS *ButtonCaps, USHORT *ButtonCapsLength, PHIDP_PREPARSED_DATA PreparsedData) { WINE_HIDP_PREPARSED_DATA *data = (WINE_HIDP_PREPARSED_DATA*)PreparsedData; WINE_HID_REPORT *report = NULL; USHORT b_count = 0, r_count = 0; int i,j,u; TRACE("(%i, 0x%x, %i, 0x%x, %p %p %p)\n", ReportType, UsagePage, LinkCollection, Usage, ButtonCaps, ButtonCapsLength, PreparsedData); if (data->magic != HID_MAGIC) return HIDP_STATUS_INVALID_PREPARSED_DATA; switch(ReportType) { case HidP_Input: b_count = data->caps.NumberInputButtonCaps; r_count = data->dwInputReportCount; report = HID_INPUT_REPORTS(data); break; case HidP_Output: b_count = data->caps.NumberOutputButtonCaps; r_count = data->dwOutputReportCount; report = HID_OUTPUT_REPORTS(data); break; case HidP_Feature: b_count = data->caps.NumberFeatureButtonCaps; r_count = data->dwFeatureReportCount; report = HID_FEATURE_REPORTS(data); break; default: return HIDP_STATUS_INVALID_REPORT_TYPE; } if (!r_count || !b_count || !report) { *ButtonCapsLength = 0; return HIDP_STATUS_SUCCESS; } b_count = min(b_count, *ButtonCapsLength); u = 0; for (j = 0; j < r_count && u < b_count; j++) { for (i = 0; i < report->elementCount && u < b_count; i++) { if (report->Elements[i].ElementType == ButtonElement && (UsagePage == 0 || UsagePage == report->Elements[i].caps.button.UsagePage) && (LinkCollection == 0 || LinkCollection == report->Elements[i].caps.button.LinkCollection) && (Usage == 0 || ( (!report->Elements[i].caps.button.IsRange && Usage == report->Elements[i].caps.button.u.NotRange.Usage)) || (report->Elements[i].caps.button.IsRange && Usage >=report->Elements[i].caps.button.u.Range.UsageMin && Usage <= report->Elements[i].caps.button.u.Range.UsageMax))) { ButtonCaps[u++] = report->Elements[i].caps.button; } } report = HID_NEXT_REPORT(data, report); } TRACE("Matched %i usages\n", u); *ButtonCapsLength = u; return HIDP_STATUS_SUCCESS; } 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; 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; r_count = data->dwInputReportCount; report = HID_INPUT_REPORTS(data); break; case HidP_Output: v_count = data->caps.NumberOutputValueCaps; r_count = data->dwOutputReportCount; report = HID_OUTPUT_REPORTS(data); break; case HidP_Feature: v_count = data->caps.NumberFeatureValueCaps; r_count = data->dwFeatureReportCount; report = HID_FEATURE_REPORTS(data); break; default: return HIDP_STATUS_INVALID_REPORT_TYPE; } if (!r_count || !v_count || !report) { *ValueCapsLength = 0; return HIDP_STATUS_SUCCESS; } v_count = min(v_count, *ValueCapsLength); u = 0; for (j = 0; j < r_count && u < v_count; j++) { for (i = 0; i < report->elementCount && u < v_count; i++) { if (report->Elements[i].ElementType == ValueElement && (UsagePage == 0 || UsagePage == report->Elements[i].caps.value.UsagePage) && (LinkCollection == 0 || LinkCollection == report->Elements[i].caps.value.LinkCollection) && (Usage == 0 || Usage == report->Elements[i].caps.value.u.NotRange.Usage)) { ValueCaps[u++] = report->Elements[i].caps.value; } } report = HID_NEXT_REPORT(data, report); } TRACE("Matched %i usages\n", u); *ValueCapsLength = u; return HIDP_STATUS_SUCCESS; } 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; 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; r_count = data->dwInputReportCount; report = HID_INPUT_REPORTS(data); break; case HidP_Output: b_count = data->caps.NumberOutputButtonCaps; r_count = data->dwOutputReportCount; report = HID_OUTPUT_REPORTS(data); break; case HidP_Feature: b_count = data->caps.NumberFeatureButtonCaps; r_count = data->dwFeatureReportCount; report = HID_FEATURE_REPORTS(data); break; default: return HIDP_STATUS_INVALID_REPORT_TYPE; } if (!r_count || !b_count || !report) return HIDP_STATUS_USAGE_NOT_FOUND; for (i = 0; i < r_count; i++) { if (!report->reportID || report->reportID == Report[0]) break; report = HID_NEXT_REPORT(data, report); } if (i == r_count) return HIDP_STATUS_REPORT_DOES_NOT_EXIST; for (i = 0; i < report->elementCount; i++) { if (report->Elements[i].ElementType == ButtonElement) { int k; WINE_HID_ELEMENT *element = &report->Elements[i]; 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) { ButtonList[uCount].Usage = element->caps.button.u.Range.UsageMin + k; ButtonList[uCount].UsagePage = element->caps.button.UsagePage; } 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; } 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; } } 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; 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; switch(ReportType) { case HidP_Input: r_count = data->dwInputReportCount; report = HID_INPUT_REPORTS(data); break; case HidP_Output: r_count = data->dwOutputReportCount; report = HID_OUTPUT_REPORTS(data); break; case HidP_Feature: r_count = data->dwFeatureReportCount; report = HID_FEATURE_REPORTS(data); break; default: return HIDP_STATUS_INVALID_REPORT_TYPE; } for (i = 0; i < r_count; i++) { if (!report->reportID || report->reportID == Report[0]) break; report = HID_NEXT_REPORT(data, report); } if (i == r_count) return HIDP_STATUS_REPORT_DOES_NOT_EXIST; for (i = 0; i < report->elementCount; i++) { WINE_HID_ELEMENT *element = &report->Elements[i]; if (element->ElementType == ButtonElement) { 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) { DataList[uCount].DataIndex = element->caps.button.u.Range.DataIndexMin + k; DataList[uCount].u.On = v; } 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; DataList[uCount].DataIndex = element->caps.value.u.NotRange.DataIndex; DataList[uCount].u.RawValue = v; } uCount++; } } if (*DataLength < uCount) rc = HIDP_STATUS_BUFFER_TOO_SMALL; else rc = HIDP_STATUS_SUCCESS; *DataLength = uCount; return rc; }