[PATCH v2 3/3] hid/tests: Move hidp driver-based tests from ntoskrnl.exe.

Rémi Bernon rbernon at codeweavers.com
Mon Aug 30 03:23:28 CDT 2021


Signed-off-by: Rémi Bernon <rbernon at codeweavers.com>
---
 dlls/hid/tests/Makefile.in          |   11 +-
 dlls/hid/tests/hidp.c               | 1554 +++++++++++++++++++++++++++
 dlls/ntoskrnl.exe/tests/Makefile.in |    4 -
 dlls/ntoskrnl.exe/tests/ntoskrnl.c  | 1508 --------------------------
 4 files changed, 1563 insertions(+), 1514 deletions(-)
 create mode 100644 dlls/hid/tests/hidp.c

diff --git a/dlls/hid/tests/Makefile.in b/dlls/hid/tests/Makefile.in
index 6bf377350c8..10ad66377f6 100644
--- a/dlls/hid/tests/Makefile.in
+++ b/dlls/hid/tests/Makefile.in
@@ -1,5 +1,12 @@
 TESTDLL   = hid.dll
-IMPORTS   = hid setupapi
+IMPORTS   = hid setupapi ole32 user32 advapi32 uuid crypt32 newdev wintrust
+PARENTSRC = ../../ntoskrnl.exe/tests
+
+driver_hid_IMPORTS = winecrt0 ntoskrnl hal hidclass
+driver_hid_EXTRADLLFLAGS = -nodefaultlibs -nostartfiles -Wl,--subsystem,native
 
 C_SRCS = \
-	device.c
+	device.c \
+	driver_hid.c \
+	driver_hid.spec \
+	hidp.c
diff --git a/dlls/hid/tests/hidp.c b/dlls/hid/tests/hidp.c
new file mode 100644
index 00000000000..a544bf0f8a0
--- /dev/null
+++ b/dlls/hid/tests/hidp.c
@@ -0,0 +1,1554 @@
+/*
+ * Copyright 2021 Rémi Bernon for CodeWeavers
+ *
+ * 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>
+#include <stddef.h>
+
+#include "ntstatus.h"
+#define WIN32_NO_STATUS
+#include "windef.h"
+#include "winbase.h"
+#include "objbase.h"
+
+#include "ddk/hidclass.h"
+#include "ddk/hidsdi.h"
+#include "ddk/hidpi.h"
+
+#include "driver_tests.h"
+
+#include "wine/test.h"
+
+#define check_member_(file, line, val, exp, fmt, member)               \
+        ok_(file, line)((val).member == (exp).member,                  \
+                        "got " #member " " fmt ", expected " fmt "\n", \
+                        (val).member, (exp).member)
+#define check_member(val, exp, fmt, member) check_member_(__FILE__, __LINE__, val, exp, fmt, member)
+
+#define check_hidp_caps(a, b) check_hidp_caps_(__LINE__, a, b)
+static inline void check_hidp_caps_(int line, HIDP_CAPS *caps, const HIDP_CAPS *exp)
+{
+    check_member_(__FILE__, line, *caps, *exp, "%04x", Usage);
+    check_member_(__FILE__, line, *caps, *exp, "%04x", UsagePage);
+    check_member_(__FILE__, line, *caps, *exp, "%d", InputReportByteLength);
+    check_member_(__FILE__, line, *caps, *exp, "%d", OutputReportByteLength);
+    check_member_(__FILE__, line, *caps, *exp, "%d", FeatureReportByteLength);
+    check_member_(__FILE__, line, *caps, *exp, "%d", NumberLinkCollectionNodes);
+    check_member_(__FILE__, line, *caps, *exp, "%d", NumberInputButtonCaps);
+    check_member_(__FILE__, line, *caps, *exp, "%d", NumberInputValueCaps);
+    check_member_(__FILE__, line, *caps, *exp, "%d", NumberInputDataIndices);
+    check_member_(__FILE__, line, *caps, *exp, "%d", NumberOutputButtonCaps);
+    check_member_(__FILE__, line, *caps, *exp, "%d", NumberOutputValueCaps);
+    check_member_(__FILE__, line, *caps, *exp, "%d", NumberOutputDataIndices);
+    check_member_(__FILE__, line, *caps, *exp, "%d", NumberFeatureButtonCaps);
+    check_member_(__FILE__, line, *caps, *exp, "%d", NumberFeatureValueCaps);
+    check_member_(__FILE__, line, *caps, *exp, "%d", NumberFeatureDataIndices);
+}
+
+#define check_hidp_link_collection_node(a, b) check_hidp_link_collection_node_(__LINE__, a, b)
+static inline void check_hidp_link_collection_node_(int line, HIDP_LINK_COLLECTION_NODE *node,
+                                                    const HIDP_LINK_COLLECTION_NODE *exp)
+{
+    check_member_(__FILE__, line, *node, *exp, "%04x", LinkUsage);
+    check_member_(__FILE__, line, *node, *exp, "%04x", LinkUsagePage);
+    check_member_(__FILE__, line, *node, *exp, "%d", Parent);
+    check_member_(__FILE__, line, *node, *exp, "%d", NumberOfChildren);
+    check_member_(__FILE__, line, *node, *exp, "%d", NextSibling);
+    check_member_(__FILE__, line, *node, *exp, "%d", FirstChild);
+    check_member_(__FILE__, line, *node, *exp, "%d", CollectionType);
+    check_member_(__FILE__, line, *node, *exp, "%d", IsAlias);
+}
+
+#define check_hidp_button_caps(a, b) check_hidp_button_caps_(__LINE__, a, b)
+static inline void check_hidp_button_caps_(int line, HIDP_BUTTON_CAPS *caps, const HIDP_BUTTON_CAPS *exp)
+{
+    check_member_(__FILE__, line, *caps, *exp, "%04x", UsagePage);
+    check_member_(__FILE__, line, *caps, *exp, "%d", ReportID);
+    check_member_(__FILE__, line, *caps, *exp, "%d", IsAlias);
+    check_member_(__FILE__, line, *caps, *exp, "%d", BitField);
+    check_member_(__FILE__, line, *caps, *exp, "%d", LinkCollection);
+    check_member_(__FILE__, line, *caps, *exp, "%04x", LinkUsage);
+    check_member_(__FILE__, line, *caps, *exp, "%04x", LinkUsagePage);
+    check_member_(__FILE__, line, *caps, *exp, "%d", IsRange);
+    check_member_(__FILE__, line, *caps, *exp, "%d", IsStringRange);
+    check_member_(__FILE__, line, *caps, *exp, "%d", IsDesignatorRange);
+    check_member_(__FILE__, line, *caps, *exp, "%d", IsAbsolute);
+
+    if (!caps->IsRange && !exp->IsRange)
+    {
+        check_member_(__FILE__, line, *caps, *exp, "%04x", NotRange.Usage);
+        check_member_(__FILE__, line, *caps, *exp, "%d", NotRange.DataIndex);
+    }
+    else if (caps->IsRange && exp->IsRange)
+    {
+        check_member_(__FILE__, line, *caps, *exp, "%04x", Range.UsageMin);
+        check_member_(__FILE__, line, *caps, *exp, "%04x", Range.UsageMax);
+        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DataIndexMin);
+        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DataIndexMax);
+    }
+
+    if (!caps->IsRange && !exp->IsRange)
+        check_member_(__FILE__, line, *caps, *exp, "%d", NotRange.StringIndex);
+    else if (caps->IsStringRange && exp->IsStringRange)
+    {
+        check_member_(__FILE__, line, *caps, *exp, "%d", Range.StringMin);
+        check_member_(__FILE__, line, *caps, *exp, "%d", Range.StringMax);
+    }
+
+    if (!caps->IsDesignatorRange && !exp->IsDesignatorRange)
+        check_member_(__FILE__, line, *caps, *exp, "%d", NotRange.DesignatorIndex);
+    else if (caps->IsDesignatorRange && exp->IsDesignatorRange)
+    {
+        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DesignatorMin);
+        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DesignatorMax);
+    }
+}
+
+#define check_hidp_value_caps(a, b) check_hidp_value_caps_(__LINE__, a, b)
+static inline void check_hidp_value_caps_(int line, HIDP_VALUE_CAPS *caps, const HIDP_VALUE_CAPS *exp)
+{
+    check_member_(__FILE__, line, *caps, *exp, "%04x", UsagePage);
+    check_member_(__FILE__, line, *caps, *exp, "%d", ReportID);
+    check_member_(__FILE__, line, *caps, *exp, "%d", IsAlias);
+    check_member_(__FILE__, line, *caps, *exp, "%d", BitField);
+    check_member_(__FILE__, line, *caps, *exp, "%d", LinkCollection);
+    check_member_(__FILE__, line, *caps, *exp, "%d", LinkUsage);
+    check_member_(__FILE__, line, *caps, *exp, "%d", LinkUsagePage);
+    check_member_(__FILE__, line, *caps, *exp, "%d", IsRange);
+    check_member_(__FILE__, line, *caps, *exp, "%d", IsStringRange);
+    check_member_(__FILE__, line, *caps, *exp, "%d", IsDesignatorRange);
+    check_member_(__FILE__, line, *caps, *exp, "%d", IsAbsolute);
+
+    check_member_(__FILE__, line, *caps, *exp, "%d", HasNull);
+    check_member_(__FILE__, line, *caps, *exp, "%d", BitSize);
+    check_member_(__FILE__, line, *caps, *exp, "%d", ReportCount);
+    check_member_(__FILE__, line, *caps, *exp, "%d", UnitsExp);
+    check_member_(__FILE__, line, *caps, *exp, "%d", Units);
+    check_member_(__FILE__, line, *caps, *exp, "%d", LogicalMin);
+    check_member_(__FILE__, line, *caps, *exp, "%d", LogicalMax);
+    check_member_(__FILE__, line, *caps, *exp, "%d", PhysicalMin);
+    check_member_(__FILE__, line, *caps, *exp, "%d", PhysicalMax);
+
+    if (!caps->IsRange && !exp->IsRange)
+    {
+        check_member_(__FILE__, line, *caps, *exp, "%04x", NotRange.Usage);
+        check_member_(__FILE__, line, *caps, *exp, "%d", NotRange.DataIndex);
+    }
+    else if (caps->IsRange && exp->IsRange)
+    {
+        check_member_(__FILE__, line, *caps, *exp, "%04x", Range.UsageMin);
+        check_member_(__FILE__, line, *caps, *exp, "%04x", Range.UsageMax);
+        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DataIndexMin);
+        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DataIndexMax);
+    }
+
+    if (!caps->IsRange && !exp->IsRange)
+        check_member_(__FILE__, line, *caps, *exp, "%d", NotRange.StringIndex);
+    else if (caps->IsStringRange && exp->IsStringRange)
+    {
+        check_member_(__FILE__, line, *caps, *exp, "%d", Range.StringMin);
+        check_member_(__FILE__, line, *caps, *exp, "%d", Range.StringMax);
+    }
+
+    if (!caps->IsDesignatorRange && !exp->IsDesignatorRange)
+        check_member_(__FILE__, line, *caps, *exp, "%d", NotRange.DesignatorIndex);
+    else if (caps->IsDesignatorRange && exp->IsDesignatorRange)
+    {
+        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DesignatorMin);
+        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DesignatorMax);
+    }
+}
+
+static BOOL sync_ioctl(HANDLE file, DWORD code, void *in_buf, DWORD in_len, void *out_buf, DWORD *ret_len)
+{
+    OVERLAPPED ovl = {0};
+    DWORD out_len = ret_len ? *ret_len : 0;
+    BOOL ret;
+
+    ovl.hEvent = CreateEventW(NULL, TRUE, FALSE, NULL);
+    ret = DeviceIoControl(file, code, in_buf, in_len, out_buf, out_len, &out_len, &ovl);
+    if (!ret && GetLastError() == ERROR_IO_PENDING) ret = GetOverlappedResult(file, &ovl, &out_len, TRUE);
+    CloseHandle(ovl.hEvent);
+
+    if (ret_len) *ret_len = out_len;
+    return ret;
+}
+
+static void test_hidp(HANDLE file, HANDLE async_file, int report_id, BOOL polled)
+{
+    const HIDP_CAPS expect_hidp_caps[] =
+    {
+        /* without report id */
+        {
+            .Usage = HID_USAGE_GENERIC_JOYSTICK,
+            .UsagePage = HID_USAGE_PAGE_GENERIC,
+            .InputReportByteLength = 26,
+            .OutputReportByteLength = 3,
+            .FeatureReportByteLength = 22,
+            .NumberLinkCollectionNodes = 10,
+            .NumberInputButtonCaps = 17,
+            .NumberInputValueCaps = 7,
+            .NumberInputDataIndices = 47,
+            .NumberFeatureButtonCaps = 1,
+            .NumberFeatureValueCaps = 6,
+            .NumberFeatureDataIndices = 8,
+        },
+        /* with report id */
+        {
+            .Usage = HID_USAGE_GENERIC_JOYSTICK,
+            .UsagePage = HID_USAGE_PAGE_GENERIC,
+            .InputReportByteLength = 25,
+            .OutputReportByteLength = 2,
+            .FeatureReportByteLength = 21,
+            .NumberLinkCollectionNodes = 10,
+            .NumberInputButtonCaps = 17,
+            .NumberInputValueCaps = 7,
+            .NumberInputDataIndices = 47,
+            .NumberFeatureButtonCaps = 1,
+            .NumberFeatureValueCaps = 6,
+            .NumberFeatureDataIndices = 8,
+        },
+    };
+    const HIDP_BUTTON_CAPS expect_button_caps[] =
+    {
+        {
+            .UsagePage = HID_USAGE_PAGE_BUTTON,
+            .ReportID = report_id,
+            .BitField = 2,
+            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
+            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
+            .LinkCollection = 1,
+            .IsRange = TRUE,
+            .IsAbsolute = TRUE,
+            .Range.UsageMin = 1,
+            .Range.UsageMax = 8,
+            .Range.DataIndexMin = 2,
+            .Range.DataIndexMax = 9,
+        },
+        {
+            .UsagePage = HID_USAGE_PAGE_BUTTON,
+            .ReportID = report_id,
+            .BitField = 3,
+            .LinkCollection = 1,
+            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
+            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
+            .IsRange = TRUE,
+            .IsAbsolute = TRUE,
+            .Range.UsageMin = 0x18,
+            .Range.UsageMax = 0x1f,
+            .Range.DataIndexMin = 10,
+            .Range.DataIndexMax = 17,
+        },
+        {
+            .UsagePage = HID_USAGE_PAGE_KEYBOARD,
+            .ReportID = report_id,
+            .BitField = 0x1fc,
+            .LinkCollection = 1,
+            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
+            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
+            .IsRange = TRUE,
+            .IsAbsolute = FALSE,
+            .Range.UsageMin = 0x8,
+            .Range.UsageMax = 0xf,
+            .Range.DataIndexMin = 18,
+            .Range.DataIndexMax = 25,
+        },
+        {
+            .UsagePage = HID_USAGE_PAGE_BUTTON,
+            .ReportID = report_id,
+            .BitField = 2,
+            .LinkCollection = 1,
+            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
+            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
+            .IsRange = FALSE,
+            .IsAbsolute = TRUE,
+            .NotRange.Usage = 0x20,
+            .NotRange.Reserved1 = 0x20,
+            .NotRange.DataIndex = 26,
+            .NotRange.Reserved4 = 26,
+        },
+    };
+    const HIDP_VALUE_CAPS expect_value_caps[] =
+    {
+        {
+            .UsagePage = HID_USAGE_PAGE_GENERIC,
+            .ReportID = report_id,
+            .BitField = 2,
+            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
+            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
+            .LinkCollection = 1,
+            .IsAbsolute = TRUE,
+            .BitSize = 8,
+            .ReportCount = 1,
+            .LogicalMin = -128,
+            .LogicalMax = 127,
+            .NotRange.Usage = HID_USAGE_GENERIC_Y,
+            .NotRange.Reserved1 = HID_USAGE_GENERIC_Y,
+        },
+        {
+            .UsagePage = HID_USAGE_PAGE_GENERIC,
+            .ReportID = report_id,
+            .BitField = 2,
+            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
+            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
+            .LinkCollection = 1,
+            .IsAbsolute = TRUE,
+            .BitSize = 8,
+            .ReportCount = 1,
+            .LogicalMin = -128,
+            .LogicalMax = 127,
+            .NotRange.Usage = HID_USAGE_GENERIC_X,
+            .NotRange.Reserved1 = HID_USAGE_GENERIC_X,
+            .NotRange.DataIndex = 1,
+            .NotRange.Reserved4 = 1,
+        },
+        {
+            .UsagePage = HID_USAGE_PAGE_BUTTON,
+            .ReportID = report_id,
+            .BitField = 2,
+            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
+            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
+            .LinkCollection = 1,
+            .IsAbsolute = TRUE,
+            .ReportCount = 1,
+            .LogicalMax = 1,
+            .IsRange = TRUE,
+            .Range.UsageMin = 0x21,
+            .Range.UsageMax = 0x22,
+            .Range.DataIndexMin = 27,
+            .Range.DataIndexMax = 28,
+        },
+        {
+            .UsagePage = HID_USAGE_PAGE_GENERIC,
+            .ReportID = report_id,
+            .BitField = 2,
+            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
+            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
+            .LinkCollection = 1,
+            .IsAbsolute = TRUE,
+            .BitSize = 4,
+            .ReportCount = 2,
+            .LogicalMin = 1,
+            .LogicalMax = 8,
+            .NotRange.Usage = HID_USAGE_GENERIC_HATSWITCH,
+            .NotRange.Reserved1 = HID_USAGE_GENERIC_HATSWITCH,
+            .NotRange.DataIndex = 29,
+            .NotRange.Reserved4 = 29,
+        },
+    };
+    static const HIDP_LINK_COLLECTION_NODE expect_collections[] =
+    {
+        {
+            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
+            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
+            .CollectionType = 1,
+            .NumberOfChildren = 7,
+            .FirstChild = 9,
+        },
+        {
+            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
+            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
+            .CollectionType = 2,
+        },
+    };
+    static const HIDP_DATA expect_data[] =
+    {
+        { .DataIndex = 0, },
+        { .DataIndex = 1, },
+        { .DataIndex = 5, .RawValue = 1, },
+        { .DataIndex = 7, .RawValue = 1, },
+        { .DataIndex = 19, .RawValue = 1, },
+        { .DataIndex = 21, .RawValue = 1, },
+        { .DataIndex = 30, },
+        { .DataIndex = 31, },
+        { .DataIndex = 32, .RawValue = 0xfeedcafe, },
+        { .DataIndex = 37, .RawValue = 1, },
+        { .DataIndex = 39, .RawValue = 1, },
+    };
+
+    OVERLAPPED overlapped = {0}, overlapped2 = {0};
+    HIDP_LINK_COLLECTION_NODE collections[16];
+    PHIDP_PREPARSED_DATA preparsed_data;
+    USAGE_AND_PAGE usage_and_pages[16];
+    HIDP_BUTTON_CAPS button_caps[32];
+    HIDP_VALUE_CAPS value_caps[16];
+    char buffer[200], report[200];
+    DWORD collection_count;
+    DWORD waveform_list;
+    HIDP_DATA data[64];
+    USAGE usages[16];
+    NTSTATUS status;
+    HIDP_CAPS caps;
+    unsigned int i;
+    USHORT count;
+    ULONG value;
+    BOOL ret;
+
+    ret = HidD_GetPreparsedData(file, &preparsed_data);
+    ok(ret, "HidD_GetPreparsedData failed with error %u\n", GetLastError());
+
+    memset(buffer, 0, sizeof(buffer));
+    status = HidP_GetCaps((PHIDP_PREPARSED_DATA)buffer, &caps);
+    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetCaps returned %#x\n", status);
+    status = HidP_GetCaps(preparsed_data, &caps);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetCaps returned %#x\n", status);
+    check_hidp_caps(&caps, &expect_hidp_caps[report_id]);
+
+    collection_count = 0;
+    status = HidP_GetLinkCollectionNodes(collections, &collection_count, preparsed_data);
+    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetLinkCollectionNodes returned %#x\n", status);
+    ok(collection_count == caps.NumberLinkCollectionNodes, "got %d collection nodes, expected %d\n",
+       collection_count, caps.NumberLinkCollectionNodes);
+    collection_count = ARRAY_SIZE(collections);
+    status = HidP_GetLinkCollectionNodes(collections, &collection_count, (PHIDP_PREPARSED_DATA)buffer);
+    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetLinkCollectionNodes returned %#x\n", status);
+    status = HidP_GetLinkCollectionNodes(collections, &collection_count, preparsed_data);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetLinkCollectionNodes returned %#x\n", status);
+    ok(collection_count == caps.NumberLinkCollectionNodes, "got %d collection nodes, expected %d\n",
+       collection_count, caps.NumberLinkCollectionNodes);
+
+    for (i = 0; i < ARRAY_SIZE(expect_collections); ++i)
+    {
+        winetest_push_context("collections[%d]", i);
+        check_hidp_link_collection_node(&collections[i], &expect_collections[i]);
+        winetest_pop_context();
+    }
+
+    count = ARRAY_SIZE(button_caps);
+    status = HidP_GetButtonCaps(HidP_Output, button_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetButtonCaps returned %#x\n", status);
+    status = HidP_GetButtonCaps(HidP_Feature + 1, button_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_GetButtonCaps returned %#x\n", status);
+    count = 0;
+    status = HidP_GetButtonCaps(HidP_Input, button_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetButtonCaps returned %#x\n", status);
+    ok(count == caps.NumberInputButtonCaps, "HidP_GetButtonCaps returned count %d, expected %d\n",
+       count, caps.NumberInputButtonCaps);
+    count = ARRAY_SIZE(button_caps);
+    status = HidP_GetButtonCaps(HidP_Input, button_caps, &count, (PHIDP_PREPARSED_DATA)buffer);
+    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetButtonCaps returned %#x\n", status);
+    memset(button_caps, 0, sizeof(button_caps));
+    status = HidP_GetButtonCaps(HidP_Input, button_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetButtonCaps returned %#x\n", status);
+    ok(count == caps.NumberInputButtonCaps, "HidP_GetButtonCaps returned count %d, expected %d\n",
+       count, caps.NumberInputButtonCaps);
+
+    for (i = 0; i < ARRAY_SIZE(expect_button_caps); ++i)
+    {
+        winetest_push_context("button_caps[%d]", i);
+        check_hidp_button_caps(&button_caps[i], &expect_button_caps[i]);
+        winetest_pop_context();
+    }
+
+    count = ARRAY_SIZE(button_caps) - 1;
+    status = HidP_GetSpecificButtonCaps(HidP_Output, 0, 0, 0, button_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificButtonCaps returned %#x\n", status);
+    status = HidP_GetSpecificButtonCaps(HidP_Feature + 1, 0, 0, 0, button_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_GetSpecificButtonCaps returned %#x\n", status);
+    count = 0;
+    status = HidP_GetSpecificButtonCaps(HidP_Input, 0, 0, 0, button_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetSpecificButtonCaps returned %#x\n", status);
+    ok(count == caps.NumberInputButtonCaps, "HidP_GetSpecificButtonCaps returned count %d, expected %d\n",
+       count, caps.NumberInputButtonCaps);
+    count = ARRAY_SIZE(button_caps) - 1;
+    status = HidP_GetSpecificButtonCaps(HidP_Input, 0, 0, 0, button_caps, &count, (PHIDP_PREPARSED_DATA)buffer);
+    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetSpecificButtonCaps returned %#x\n", status);
+
+    status = HidP_GetSpecificButtonCaps(HidP_Input, 0, 0, 0, button_caps + 1, &count, preparsed_data);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetSpecificButtonCaps returned %#x\n", status);
+    ok(count == caps.NumberInputButtonCaps, "HidP_GetSpecificButtonCaps returned count %d, expected %d\n",
+       count, caps.NumberInputButtonCaps);
+    check_hidp_button_caps(&button_caps[1], &button_caps[0]);
+
+    status = HidP_GetSpecificButtonCaps(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, 5, button_caps + 1,
+                                        &count, preparsed_data);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetSpecificButtonCaps returned %#x\n", status);
+    ok(count == 1, "HidP_GetSpecificButtonCaps returned count %d, expected %d\n", count, 1);
+    check_hidp_button_caps(&button_caps[1], &button_caps[0]);
+
+    count = 0xbeef;
+    status = HidP_GetSpecificButtonCaps(HidP_Input, 0xfffe, 0, 0, button_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificButtonCaps returned %#x\n", status);
+    ok(count == 0, "HidP_GetSpecificButtonCaps returned count %d, expected %d\n", count, 0);
+    count = 0xbeef;
+    status = HidP_GetSpecificButtonCaps(HidP_Input, 0, 0xfffe, 0, button_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificButtonCaps returned %#x\n", status);
+    ok(count == 0, "HidP_GetSpecificButtonCaps returned count %d, expected %d\n", count, 0);
+    count = 0xbeef;
+    status = HidP_GetSpecificButtonCaps(HidP_Input, 0, 0, 0xfffe, button_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificButtonCaps returned %#x\n", status);
+    ok(count == 0, "HidP_GetSpecificButtonCaps returned count %d, expected %d\n", count, 0);
+
+    count = ARRAY_SIZE(value_caps);
+    status = HidP_GetValueCaps(HidP_Output, value_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetValueCaps returned %#x\n", status);
+    status = HidP_GetValueCaps(HidP_Feature + 1, value_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_GetValueCaps returned %#x\n", status);
+    count = 0;
+    status = HidP_GetValueCaps(HidP_Input, value_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetValueCaps returned %#x\n", status);
+    ok(count == caps.NumberInputValueCaps, "HidP_GetValueCaps returned count %d, expected %d\n",
+       count, caps.NumberInputValueCaps);
+    count = ARRAY_SIZE(value_caps);
+    status = HidP_GetValueCaps(HidP_Input, value_caps, &count, (PHIDP_PREPARSED_DATA)buffer);
+    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetValueCaps returned %#x\n", status);
+    status = HidP_GetValueCaps(HidP_Input, value_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetValueCaps returned %#x\n", status);
+    ok(count == caps.NumberInputValueCaps, "HidP_GetValueCaps returned count %d, expected %d\n",
+       count, caps.NumberInputValueCaps);
+
+    for (i = 0; i < ARRAY_SIZE(expect_value_caps); ++i)
+    {
+        winetest_push_context("value_caps[%d]", i);
+        check_hidp_value_caps(&value_caps[i], &expect_value_caps[i]);
+        winetest_pop_context();
+    }
+
+    count = ARRAY_SIZE(value_caps) - 4;
+    status = HidP_GetSpecificValueCaps(HidP_Output, 0, 0, 0, value_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificValueCaps returned %#x\n", status);
+    status = HidP_GetSpecificValueCaps(HidP_Feature + 1, 0, 0, 0, value_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_GetSpecificValueCaps returned %#x\n", status);
+    count = 0;
+    status = HidP_GetSpecificValueCaps(HidP_Input, 0, 0, 0, value_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetSpecificValueCaps returned %#x\n", status);
+    ok(count == caps.NumberInputValueCaps, "HidP_GetSpecificValueCaps returned count %d, expected %d\n",
+       count, caps.NumberInputValueCaps);
+    count = ARRAY_SIZE(value_caps) - 4;
+    status = HidP_GetSpecificValueCaps(HidP_Input, 0, 0, 0, value_caps + 4, &count, (PHIDP_PREPARSED_DATA)buffer);
+    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetSpecificValueCaps returned %#x\n", status);
+
+    status = HidP_GetSpecificValueCaps(HidP_Input, 0, 0, 0, value_caps + 4, &count, preparsed_data);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetSpecificValueCaps returned %#x\n", status);
+    ok(count == caps.NumberInputValueCaps, "HidP_GetSpecificValueCaps returned count %d, expected %d\n",
+       count, caps.NumberInputValueCaps);
+    check_hidp_value_caps(&value_caps[4], &value_caps[0]);
+    check_hidp_value_caps(&value_caps[5], &value_caps[1]);
+    check_hidp_value_caps(&value_caps[6], &value_caps[2]);
+    check_hidp_value_caps(&value_caps[7], &value_caps[3]);
+
+    count = 1;
+    status = HidP_GetSpecificValueCaps(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_HATSWITCH,
+                                       value_caps + 4, &count, preparsed_data);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetSpecificValueCaps returned %#x\n", status);
+    ok(count == 1, "HidP_GetSpecificValueCaps returned count %d, expected %d\n", count, 1);
+    check_hidp_value_caps(&value_caps[4], &value_caps[3]);
+
+    count = 0xdead;
+    status = HidP_GetSpecificValueCaps(HidP_Input, 0xfffe, 0, 0, value_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificValueCaps returned %#x\n", status);
+    ok(count == 0, "HidP_GetSpecificValueCaps returned count %d, expected %d\n", count, 0);
+    count = 0xdead;
+    status = HidP_GetSpecificValueCaps(HidP_Input, 0, 0xfffe, 0, value_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificValueCaps returned %#x\n", status);
+    ok(count == 0, "HidP_GetSpecificValueCaps returned count %d, expected %d\n", count, 0);
+    count = 0xdead;
+    status = HidP_GetSpecificValueCaps(HidP_Input, 0, 0, 0xfffe, value_caps, &count, preparsed_data);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificValueCaps returned %#x\n", status);
+    ok(count == 0, "HidP_GetSpecificValueCaps returned count %d, expected %d\n", count, 0);
+
+    status = HidP_InitializeReportForID(HidP_Input, 0, (PHIDP_PREPARSED_DATA)buffer, report, sizeof(report));
+    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_InitializeReportForID returned %#x\n", status);
+    status = HidP_InitializeReportForID(HidP_Feature + 1, 0, preparsed_data, report, sizeof(report));
+    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_InitializeReportForID returned %#x\n", status);
+    status = HidP_InitializeReportForID(HidP_Input, 0, preparsed_data, report, sizeof(report));
+    ok(status == HIDP_STATUS_INVALID_REPORT_LENGTH, "HidP_InitializeReportForID returned %#x\n", status);
+    status = HidP_InitializeReportForID(HidP_Input, 0, preparsed_data, report, caps.InputReportByteLength + 1);
+    ok(status == HIDP_STATUS_INVALID_REPORT_LENGTH, "HidP_InitializeReportForID returned %#x\n", status);
+    status = HidP_InitializeReportForID(HidP_Input, 1 - report_id, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_REPORT_DOES_NOT_EXIST, "HidP_InitializeReportForID returned %#x\n", status);
+
+    memset(report, 0xcd, sizeof(report));
+    status = HidP_InitializeReportForID(HidP_Input, report_id, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
+
+    memset(buffer, 0xcd, sizeof(buffer));
+    memset(buffer, 0, caps.InputReportByteLength);
+    buffer[0] = report_id;
+    ok(!memcmp(buffer, report, sizeof(buffer)), "unexpected report data\n");
+
+    status = HidP_SetUsageValueArray(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X, buffer,
+                                     sizeof(buffer), preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_NOT_VALUE_ARRAY, "HidP_SetUsageValueArray returned %#x\n", status);
+    memset(buffer, 0xcd, sizeof(buffer));
+    status = HidP_SetUsageValueArray(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_HATSWITCH, buffer,
+                                     0, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_SetUsageValueArray returned %#x\n", status);
+    status = HidP_SetUsageValueArray(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_HATSWITCH, buffer,
+                                     8, preparsed_data, report, caps.InputReportByteLength);
+    todo_wine
+    ok(status == HIDP_STATUS_NOT_IMPLEMENTED, "HidP_SetUsageValueArray returned %#x\n", status);
+
+    status = HidP_GetUsageValueArray(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X, buffer,
+                                     sizeof(buffer), preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_NOT_VALUE_ARRAY, "HidP_GetUsageValueArray returned %#x\n", status);
+    memset(buffer, 0xcd, sizeof(buffer));
+    status = HidP_GetUsageValueArray(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_HATSWITCH, buffer,
+                                     0, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetUsageValueArray returned %#x\n", status);
+    status = HidP_GetUsageValueArray(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_HATSWITCH, buffer,
+                                     8, preparsed_data, report, caps.InputReportByteLength);
+    todo_wine
+    ok(status == HIDP_STATUS_NOT_IMPLEMENTED, "HidP_GetUsageValueArray returned %#x\n", status);
+
+    value = -128;
+    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
+                                value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
+    value = 0xdeadbeef;
+    status = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
+                                &value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
+    ok(value == 0x80, "got value %x, expected %#x\n", value, 0x80);
+    value = 0xdeadbeef;
+    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
+                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
+    ok(value == -128, "got value %x, expected %#x\n", value, -128);
+
+    value = 127;
+    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
+                                value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
+    value = 0xdeadbeef;
+    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
+                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
+    ok(value == 127, "got value %x, expected %#x\n", value, 127);
+
+    value = 0;
+    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
+                                value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
+    value = 0xdeadbeef;
+    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
+                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
+    ok(value == 0, "got value %x, expected %#x\n", value, 0);
+
+    value = 0x7fffffff;
+    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
+    value = 0xdeadbeef;
+    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_VALUE_OUT_OF_RANGE, "HidP_GetScaledUsageValue returned %#x\n", status);
+    ok(value == 0, "got value %x, expected %#x\n", value, 0);
+    value = 0xdeadbeef;
+    status = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                &value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
+    ok(value == 0x7fffffff, "got value %x, expected %#x\n", value, 0x7fffffff);
+
+    value = 0x3fffffff;
+    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
+    value = 0xdeadbeef;
+    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
+    ok(value == 0x7fffffff, "got value %x, expected %#x\n", value, 0x7fffffff);
+
+    value = 0;
+    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
+    value = 0xdeadbeef;
+    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
+    ok(value == 0x80000000, "got value %x, expected %#x\n", value, 0x80000000);
+
+    value = 0;
+    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RX,
+                                value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
+    value = 0xdeadbeef;
+    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RX,
+                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
+    ok(value == 0, "got value %x, expected %#x\n", value, 0);
+
+    value = 0xfeedcafe;
+    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RY,
+                                value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
+    value = 0xdeadbeef;
+    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RY,
+                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_BAD_LOG_PHY_VALUES, "HidP_GetScaledUsageValue returned %#x\n", status);
+    ok(value == 0, "got value %x, expected %#x\n", value, 0);
+    status = HidP_SetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RY,
+                                      0, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_BAD_LOG_PHY_VALUES, "HidP_GetScaledUsageValue returned %#x\n", status);
+    ok(value == 0, "got value %x, expected %#x\n", value, 0);
+
+    value = HidP_MaxUsageListLength(HidP_Feature + 1, 0, preparsed_data);
+    ok(value == 0, "HidP_MaxUsageListLength(HidP_Feature + 1, 0) returned %d, expected %d\n", value, 0);
+    value = HidP_MaxUsageListLength(HidP_Input, 0, preparsed_data);
+    ok(value == 50, "HidP_MaxUsageListLength(HidP_Input, 0) returned %d, expected %d\n", value, 50);
+    value = HidP_MaxUsageListLength(HidP_Input, HID_USAGE_PAGE_BUTTON, preparsed_data);
+    ok(value == 32, "HidP_MaxUsageListLength(HidP_Input, HID_USAGE_PAGE_BUTTON) returned %d, expected %d\n", value, 32);
+    value = HidP_MaxUsageListLength(HidP_Input, HID_USAGE_PAGE_LED, preparsed_data);
+    ok(value == 8, "HidP_MaxUsageListLength(HidP_Input, HID_USAGE_PAGE_LED) returned %d, expected %d\n", value, 8);
+    value = HidP_MaxUsageListLength(HidP_Feature, HID_USAGE_PAGE_BUTTON, preparsed_data);
+    ok(value == 8, "HidP_MaxUsageListLength(HidP_Feature, HID_USAGE_PAGE_BUTTON) returned %d, expected %d\n", value, 8);
+    value = HidP_MaxUsageListLength(HidP_Feature, HID_USAGE_PAGE_LED, preparsed_data);
+    ok(value == 0, "HidP_MaxUsageListLength(HidP_Feature, HID_USAGE_PAGE_LED) returned %d, expected %d\n", value, 0);
+
+    usages[0] = 0xff;
+    value = 1;
+    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, usages, &value,
+                            preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_SetUsages returned %#x\n", status);
+    usages[1] = 2;
+    usages[2] = 0xff;
+    value = 3;
+    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, usages, &value,
+                            preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_SetUsages returned %#x\n", status);
+    usages[0] = 4;
+    usages[1] = 6;
+    value = 2;
+    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, usages, &value,
+                            preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsages returned %#x\n", status);
+    usages[0] = 4;
+    usages[1] = 6;
+    value = 2;
+    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_LED, 0, usages, &value, preparsed_data,
+                            report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsages returned %#x\n", status);
+
+    value = ARRAY_SIZE(usages);
+    status = HidP_GetUsages(HidP_Input, HID_USAGE_PAGE_KEYBOARD, 0, usages, &value, preparsed_data,
+                            report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsages returned %#x\n", status);
+    ok(value == 0, "got usage count %d, expected %d\n", value, 2);
+
+    usages[0] = 0x9;
+    usages[1] = 0xb;
+    usages[2] = 0xa;
+    value = 3;
+    ok(report[6] == 0, "got report[6] %x expected 0\n", report[6]);
+    ok(report[7] == 0, "got report[7] %x expected 0\n", report[7]);
+    memcpy(buffer, report, caps.InputReportByteLength);
+    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_KEYBOARD, 0, usages, &value, preparsed_data,
+                            report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_SetUsages returned %#x\n", status);
+    buffer[6] = 2;
+    buffer[7] = 4;
+    ok(!memcmp(buffer, report, caps.InputReportByteLength), "unexpected report data\n");
+
+    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_LED, 0, 6, 1,
+                                preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_SetUsageValue returned %#x\n", status);
+
+    value = 0xdeadbeef;
+    status = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_LED, 0, 6, &value,
+                                preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_SetUsageValue returned %#x\n", status);
+    ok(value == 0xdeadbeef, "got value %x, expected %#x\n", value, 0xdeadbeef);
+
+    value = 1;
+    status = HidP_GetUsages(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, usages, &value,
+                            preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetUsages returned %#x\n", status);
+    ok(value == 2, "got usage count %d, expected %d\n", value, 2);
+    value = ARRAY_SIZE(usages);
+    memset(usages, 0xcd, sizeof(usages));
+    status = HidP_GetUsages(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, usages, &value,
+                            preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsages returned %#x\n", status);
+    ok(value == 2, "got usage count %d, expected %d\n", value, 2);
+    ok(usages[0] == 4, "got usages[0] %x, expected %x\n", usages[0], 4);
+    ok(usages[1] == 6, "got usages[1] %x, expected %x\n", usages[1], 6);
+
+    value = ARRAY_SIZE(usages);
+    memset(usages, 0xcd, sizeof(usages));
+    status = HidP_GetUsages(HidP_Input, HID_USAGE_PAGE_LED, 0, usages, &value, preparsed_data,
+                            report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsages returned %#x\n", status);
+    ok(value == 2, "got usage count %d, expected %d\n", value, 2);
+    ok(usages[0] == 6, "got usages[0] %x, expected %x\n", usages[0], 6);
+    ok(usages[1] == 4, "got usages[1] %x, expected %x\n", usages[1], 4);
+
+    value = ARRAY_SIZE(usage_and_pages);
+    memset(usage_and_pages, 0xcd, sizeof(usage_and_pages));
+    status = HidP_GetUsagesEx(HidP_Input, 0, usage_and_pages, &value, preparsed_data, report,
+                              caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsagesEx returned %#x\n", status);
+    ok(value == 6, "got usage count %d, expected %d\n", value, 4);
+    ok(usage_and_pages[0].UsagePage == HID_USAGE_PAGE_BUTTON, "got usage_and_pages[0] UsagePage %x, expected %x\n",
+       usage_and_pages[0].UsagePage, HID_USAGE_PAGE_BUTTON);
+    ok(usage_and_pages[1].UsagePage == HID_USAGE_PAGE_BUTTON, "got usage_and_pages[1] UsagePage %x, expected %x\n",
+       usage_and_pages[1].UsagePage, HID_USAGE_PAGE_BUTTON);
+    ok(usage_and_pages[2].UsagePage == HID_USAGE_PAGE_KEYBOARD, "got usage_and_pages[2] UsagePage %x, expected %x\n",
+       usage_and_pages[2].UsagePage, HID_USAGE_PAGE_KEYBOARD);
+    ok(usage_and_pages[3].UsagePage == HID_USAGE_PAGE_KEYBOARD, "got usage_and_pages[3] UsagePage %x, expected %x\n",
+       usage_and_pages[3].UsagePage, HID_USAGE_PAGE_KEYBOARD);
+    ok(usage_and_pages[4].UsagePage == HID_USAGE_PAGE_LED, "got usage_and_pages[4] UsagePage %x, expected %x\n",
+       usage_and_pages[4].UsagePage, HID_USAGE_PAGE_LED);
+    ok(usage_and_pages[5].UsagePage == HID_USAGE_PAGE_LED, "got usage_and_pages[5] UsagePage %x, expected %x\n",
+       usage_and_pages[5].UsagePage, HID_USAGE_PAGE_LED);
+    ok(usage_and_pages[0].Usage == 4, "got usage_and_pages[0] Usage %x, expected %x\n",
+       usage_and_pages[0].Usage, 4);
+    ok(usage_and_pages[1].Usage == 6, "got usage_and_pages[1] Usage %x, expected %x\n",
+       usage_and_pages[1].Usage, 6);
+    ok(usage_and_pages[2].Usage == 9, "got usage_and_pages[2] Usage %x, expected %x\n",
+       usage_and_pages[2].Usage, 9);
+    ok(usage_and_pages[3].Usage == 11, "got usage_and_pages[3] Usage %x, expected %x\n",
+       usage_and_pages[3].Usage, 11);
+    ok(usage_and_pages[4].Usage == 6, "got usage_and_pages[4] Usage %x, expected %x\n",
+       usage_and_pages[4].Usage, 6);
+    ok(usage_and_pages[5].Usage == 4, "got usage_and_pages[5] Usage %x, expected %x\n",
+       usage_and_pages[5].Usage, 4);
+
+    value = HidP_MaxDataListLength(HidP_Feature + 1, preparsed_data);
+    ok(value == 0, "HidP_MaxDataListLength(HidP_Feature + 1) returned %d, expected %d\n", value, 0);
+    value = HidP_MaxDataListLength(HidP_Input, preparsed_data);
+    ok(value == 58, "HidP_MaxDataListLength(HidP_Input) returned %d, expected %d\n", value, 58);
+    value = HidP_MaxDataListLength(HidP_Output, preparsed_data);
+    ok(value == 0, "HidP_MaxDataListLength(HidP_Output) returned %d, expected %d\n", value, 0);
+    value = HidP_MaxDataListLength(HidP_Feature, preparsed_data);
+    ok(value == 14, "HidP_MaxDataListLength(HidP_Feature) returned %d, expected %d\n", value, 14);
+
+    value = 1;
+    status = HidP_GetData(HidP_Input, data, &value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetData returned %#x\n", status);
+    ok(value == 11, "got data count %d, expected %d\n", value, 11);
+    memset(data, 0, sizeof(data));
+    status = HidP_GetData(HidP_Input, data, &value, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetData returned %#x\n", status);
+    for (i = 0; i < ARRAY_SIZE(expect_data); ++i)
+    {
+        winetest_push_context("data[%d]", i);
+        check_member(data[i], expect_data[i], "%d", DataIndex);
+        check_member(data[i], expect_data[i], "%d", RawValue);
+        winetest_pop_context();
+    }
+
+    /* HID nary usage collections are set with 1-based usage index in their declaration order */
+
+    memset(report, 0, caps.InputReportByteLength);
+    status = HidP_InitializeReportForID(HidP_Input, report_id, preparsed_data, report,
+                                        caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
+    value = 2;
+    usages[0] = 0x8e;
+    usages[1] = 0x8f;
+    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_KEYBOARD, 0, usages, &value, preparsed_data,
+                            report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsages returned %#x\n", status);
+    ok(report[caps.InputReportByteLength - 2] == 3, "unexpected usage index %d, expected 3\n",
+       report[caps.InputReportByteLength - 2]);
+    ok(report[caps.InputReportByteLength - 1] == 4, "unexpected usage index %d, expected 4\n",
+       report[caps.InputReportByteLength - 1]);
+    status = HidP_UnsetUsages(HidP_Input, HID_USAGE_PAGE_KEYBOARD, 0, usages, &value, preparsed_data,
+                              report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_UnsetUsages returned %#x\n", status);
+    ok(report[caps.InputReportByteLength - 2] == 0, "unexpected usage index %d, expected 0\n",
+       report[caps.InputReportByteLength - 2]);
+    ok(report[caps.InputReportByteLength - 1] == 0, "unexpected usage index %d, expected 0\n",
+       report[caps.InputReportByteLength - 1]);
+    status = HidP_UnsetUsages(HidP_Input, HID_USAGE_PAGE_KEYBOARD, 0, usages, &value, preparsed_data,
+                              report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_BUTTON_NOT_PRESSED, "HidP_UnsetUsages returned %#x\n", status);
+    value = 1;
+    usages[0] = 0x8c;
+    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_KEYBOARD, 0, usages, &value, preparsed_data,
+                            report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsages returned %#x\n", status);
+    ok(report[caps.InputReportByteLength - 2] == 1, "unexpected usage index %d, expected 1\n",
+       report[caps.InputReportByteLength - 2]);
+
+    memset(report, 0xcd, sizeof(report));
+    status = HidP_InitializeReportForID(HidP_Feature, 3, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_REPORT_DOES_NOT_EXIST, "HidP_InitializeReportForID returned %#x\n", status);
+
+    memset(report, 0xcd, sizeof(report));
+    status = HidP_InitializeReportForID(HidP_Feature, report_id, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
+
+    memset(buffer, 0xcd, sizeof(buffer));
+    memset(buffer, 0, caps.FeatureReportByteLength);
+    buffer[0] = report_id;
+    ok(!memcmp(buffer, report, sizeof(buffer)), "unexpected report data\n");
+
+    for (i = 0; i < caps.NumberLinkCollectionNodes; ++i)
+    {
+        if (collections[i].LinkUsagePage != HID_USAGE_PAGE_HAPTICS) continue;
+        if (collections[i].LinkUsage == HID_USAGE_HAPTICS_WAVEFORM_LIST) break;
+    }
+    ok(i < caps.NumberLinkCollectionNodes,
+       "HID_USAGE_HAPTICS_WAVEFORM_LIST collection not found\n");
+    waveform_list = i;
+
+    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
+                                HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, (PHIDP_PREPARSED_DATA)buffer,
+                                report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_SetUsageValue returned %#x\n", status);
+    status = HidP_SetUsageValue(HidP_Feature + 1, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
+                                HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, preparsed_data, report,
+                                caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_SetUsageValue returned %#x\n", status);
+    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
+                                HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, preparsed_data, report,
+                                caps.FeatureReportByteLength + 1);
+    ok(status == HIDP_STATUS_INVALID_REPORT_LENGTH, "HidP_SetUsageValue returned %#x\n", status);
+    report[0] = 1 - report_id;
+    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
+                                HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, preparsed_data, report,
+                                caps.FeatureReportByteLength);
+    ok(status == (report_id ? HIDP_STATUS_SUCCESS : HIDP_STATUS_INCOMPATIBLE_REPORT_ID),
+       "HidP_SetUsageValue returned %#x\n", status);
+    report[0] = 2;
+    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
+                                HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, preparsed_data, report,
+                                caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_INCOMPATIBLE_REPORT_ID, "HidP_SetUsageValue returned %#x\n", status);
+    report[0] = report_id;
+    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, 0xdead, 3, HID_USAGE_HAPTICS_WAVEFORM_RUMBLE,
+                                preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_SetUsageValue returned %#x\n", status);
+
+    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
+                                HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, preparsed_data, report,
+                                caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
+
+    memset(buffer, 0xcd, sizeof(buffer));
+    memset(buffer, 0, caps.FeatureReportByteLength);
+    buffer[0] = report_id;
+    value = HID_USAGE_HAPTICS_WAVEFORM_RUMBLE;
+    memcpy(buffer + 1, &value, 2);
+    ok(!memcmp(buffer, report, sizeof(buffer)), "unexpected report data\n");
+
+    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3, &value,
+                                (PHIDP_PREPARSED_DATA)buffer, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetUsageValue returned %#x\n", status);
+    status = HidP_GetUsageValue(HidP_Feature + 1, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
+                                &value, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_GetUsageValue returned %#x\n", status);
+    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3, &value,
+                                preparsed_data, report, caps.FeatureReportByteLength + 1);
+    ok(status == HIDP_STATUS_INVALID_REPORT_LENGTH, "HidP_GetUsageValue returned %#x\n", status);
+    report[0] = 1 - report_id;
+    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3, &value,
+                                preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == (report_id ? HIDP_STATUS_SUCCESS : HIDP_STATUS_INCOMPATIBLE_REPORT_ID),
+       "HidP_GetUsageValue returned %#x\n", status);
+    report[0] = 2;
+    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3, &value,
+                                preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_INCOMPATIBLE_REPORT_ID, "HidP_GetUsageValue returned %#x\n", status);
+    report[0] = report_id;
+    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, 0xdead, 3, &value,
+                                preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetUsageValue returned %#x\n", status);
+
+    value = 0xdeadbeef;
+    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3, &value,
+                                preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
+    ok(value == HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, "got value %x, expected %#x\n", value,
+       HID_USAGE_HAPTICS_WAVEFORM_RUMBLE);
+
+    memset(buffer, 0xff, sizeof(buffer));
+    status = HidP_SetUsageValueArray(HidP_Feature, HID_USAGE_PAGE_HAPTICS, 0, HID_USAGE_HAPTICS_WAVEFORM_CUTOFF_TIME, buffer,
+                                     0, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_SetUsageValueArray returned %#x\n", status);
+    status = HidP_SetUsageValueArray(HidP_Feature, HID_USAGE_PAGE_HAPTICS, 0, HID_USAGE_HAPTICS_WAVEFORM_CUTOFF_TIME, buffer,
+                                     64, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValueArray returned %#x\n", status);
+    ok(!memcmp(report + 9, buffer, 8), "unexpected report data\n");
+
+    memset(buffer, 0, sizeof(buffer));
+    status = HidP_GetUsageValueArray(HidP_Feature, HID_USAGE_PAGE_HAPTICS, 0, HID_USAGE_HAPTICS_WAVEFORM_CUTOFF_TIME, buffer,
+                                     0, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetUsageValueArray returned %#x\n", status);
+    status = HidP_GetUsageValueArray(HidP_Feature, HID_USAGE_PAGE_HAPTICS, 0, HID_USAGE_HAPTICS_WAVEFORM_CUTOFF_TIME, buffer,
+                                     64, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValueArray returned %#x\n", status);
+    memset(buffer + 16, 0xff, 8);
+    ok(!memcmp(buffer, buffer + 16, 16), "unexpected report value\n");
+
+
+    value = 0x7fffffff;
+    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                value, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
+    value = 0xdeadbeef;
+    status = HidP_GetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                      (LONG *)&value, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_VALUE_OUT_OF_RANGE, "HidP_GetScaledUsageValue returned %#x\n", status);
+    ok(value == 0, "got value %x, expected %#x\n", value, 0);
+    value = 0xdeadbeef;
+    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                &value, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
+    ok(value == 0x7fffffff, "got value %x, expected %#x\n", value, 0x7fffffff);
+
+    value = 0x7fff;
+    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                value, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
+    value = 0xdeadbeef;
+    status = HidP_GetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                      (LONG *)&value, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
+    ok(value == 0x0003ffff, "got value %x, expected %#x\n", value, 0x0003ffff);
+
+    value = 0;
+    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                value, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
+    value = 0xdeadbeef;
+    status = HidP_GetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                      (LONG *)&value, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
+    ok(value == 0xfff90000, "got value %x, expected %#x\n", value, 0xfff90000);
+    status = HidP_SetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                      0x1000, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetScaledUsageValue returned %#x\n", status);
+    value = 0;
+    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                &value, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
+    ok(value == 0xfffff518, "got value %x, expected %#x\n", value, 0xfffff518);
+    status = HidP_SetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                      0, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetScaledUsageValue returned %#x\n", status);
+    value = 0;
+    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                &value, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
+    ok(value == 0xfffff45e, "got value %x, expected %#x\n", value, 0xfffff45e);
+    status = HidP_SetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                      0xdead, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetScaledUsageValue returned %#x\n", status);
+    value = 0;
+    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                &value, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
+    ok(value == 0xfffffe7d, "got value %x, expected %#x\n", value, 0xfffffe7d);
+    status = HidP_SetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                      0xbeef, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetScaledUsageValue returned %#x\n", status);
+    value = 0;
+    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
+                                &value, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
+    ok(value == 0xfffffd0b, "got value %x, expected %#x\n", value, 0xfffffd0b);
+
+
+    memset(report, 0xcd, sizeof(report));
+    status = HidP_InitializeReportForID(HidP_Input, report_id, preparsed_data, report, caps.InputReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_GetInputReport(file, report, 0);
+    ok(!ret, "HidD_GetInputReport succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "HidD_GetInputReport returned error %u\n", GetLastError());
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_GetInputReport(file, report, caps.InputReportByteLength - 1);
+    ok(!ret, "HidD_GetInputReport succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
+       "HidD_GetInputReport returned error %u\n", GetLastError());
+
+    SetLastError(0xdeadbeef);
+    memset(buffer, 0x5a, sizeof(buffer));
+    ret = HidD_GetInputReport(file, buffer, caps.InputReportByteLength);
+    if (report_id || broken(!ret) /* w7u */)
+    {
+        ok(!ret, "HidD_GetInputReport succeeded, last error %u\n", GetLastError());
+        ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
+           "HidD_GetInputReport returned error %u\n", GetLastError());
+    }
+    else
+    {
+        ok(ret, "HidD_GetInputReport failed, last error %u\n", GetLastError());
+        ok(buffer[0] == 0x5a, "got buffer[0] %x, expected 0x5a\n", (BYTE)buffer[0]);
+    }
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_GetInputReport(file, report, caps.InputReportByteLength);
+    ok(ret, "HidD_GetInputReport failed, last error %u\n", GetLastError());
+    ok(report[0] == report_id, "got report[0] %02x, expected %02x\n", report[0], report_id);
+
+    SetLastError(0xdeadbeef);
+    value = caps.InputReportByteLength * 2;
+    ret = sync_ioctl(file, IOCTL_HID_GET_INPUT_REPORT, NULL, 0, report, &value);
+    ok(ret, "IOCTL_HID_GET_INPUT_REPORT failed, last error %u\n", GetLastError());
+    ok(value == 3, "got length %u, expected 3\n", value);
+    ok(report[0] == report_id, "got report[0] %02x, expected %02x\n", report[0], report_id);
+
+
+    memset(report, 0xcd, sizeof(report));
+    status = HidP_InitializeReportForID(HidP_Feature, report_id, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_GetFeature(file, report, 0);
+    ok(!ret, "HidD_GetFeature succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "HidD_GetFeature returned error %u\n", GetLastError());
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_GetFeature(file, report, caps.FeatureReportByteLength - 1);
+    ok(!ret, "HidD_GetFeature succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
+       "HidD_GetFeature returned error %u\n", GetLastError());
+
+    SetLastError(0xdeadbeef);
+    memset(buffer, 0x5a, sizeof(buffer));
+    ret = HidD_GetFeature(file, buffer, caps.FeatureReportByteLength);
+    if (report_id || broken(!ret))
+    {
+        ok(!ret, "HidD_GetFeature succeeded, last error %u\n", GetLastError());
+        ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
+           "HidD_GetFeature returned error %u\n", GetLastError());
+    }
+    else
+    {
+        ok(ret, "HidD_GetFeature failed, last error %u\n", GetLastError());
+        ok(buffer[0] == 0x5a, "got buffer[0] %x, expected 0x5a\n", (BYTE)buffer[0]);
+    }
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_GetFeature(file, report, caps.FeatureReportByteLength);
+    ok(ret, "HidD_GetFeature failed, last error %u\n", GetLastError());
+    ok(report[0] == report_id, "got report[0] %02x, expected %02x\n", report[0], report_id);
+
+    value = caps.FeatureReportByteLength * 2;
+    SetLastError(0xdeadbeef);
+    ret = sync_ioctl(file, IOCTL_HID_GET_FEATURE, NULL, 0, report, &value);
+    ok(ret, "IOCTL_HID_GET_FEATURE failed, last error %u\n", GetLastError());
+    ok(value == 3, "got length %u, expected 3\n", value);
+    ok(report[0] == report_id, "got report[0] %02x, expected %02x\n", report[0], report_id);
+
+
+    memset(report, 0xcd, sizeof(report));
+    status = HidP_InitializeReportForID(HidP_Feature, report_id, preparsed_data, report, caps.FeatureReportByteLength);
+    ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_SetFeature(file, report, 0);
+    ok(!ret, "HidD_SetFeature succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "HidD_SetFeature returned error %u\n", GetLastError());
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_SetFeature(file, report, caps.FeatureReportByteLength - 1);
+    ok(!ret, "HidD_SetFeature succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
+       "HidD_SetFeature returned error %u\n", GetLastError());
+
+    SetLastError(0xdeadbeef);
+    memset(buffer, 0x5a, sizeof(buffer));
+    ret = HidD_SetFeature(file, buffer, caps.FeatureReportByteLength);
+    if (report_id || broken(!ret))
+    {
+        ok(!ret, "HidD_SetFeature succeeded, last error %u\n", GetLastError());
+        ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
+           "HidD_SetFeature returned error %u\n", GetLastError());
+    }
+    else
+    {
+        ok(ret, "HidD_SetFeature failed, last error %u\n", GetLastError());
+    }
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_SetFeature(file, report, caps.FeatureReportByteLength);
+    ok(ret, "HidD_SetFeature failed, last error %u\n", GetLastError());
+
+    value = caps.FeatureReportByteLength * 2;
+    SetLastError(0xdeadbeef);
+    ret = sync_ioctl(file, IOCTL_HID_SET_FEATURE, NULL, 0, report, &value);
+    ok(!ret, "IOCTL_HID_SET_FEATURE succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "IOCTL_HID_SET_FEATURE returned error %u\n", GetLastError());
+    value = 0;
+    SetLastError(0xdeadbeef);
+    ret = sync_ioctl(file, IOCTL_HID_SET_FEATURE, report, caps.FeatureReportByteLength * 2, NULL, &value);
+    ok(ret, "IOCTL_HID_SET_FEATURE failed, last error %u\n", GetLastError());
+    ok(value == 3, "got length %u, expected 3\n", value);
+
+
+    memset(report, 0xcd, sizeof(report));
+    status = HidP_InitializeReportForID(HidP_Output, report_id, preparsed_data, report, caps.OutputReportByteLength);
+    ok(status == HIDP_STATUS_REPORT_DOES_NOT_EXIST, "HidP_InitializeReportForID returned %#x\n", status);
+    memset(report, 0, caps.OutputReportByteLength);
+    report[0] = report_id;
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_SetOutputReport(file, report, 0);
+    ok(!ret, "HidD_SetOutputReport succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "HidD_SetOutputReport returned error %u\n", GetLastError());
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_SetOutputReport(file, report, caps.OutputReportByteLength - 1);
+    ok(!ret, "HidD_SetOutputReport succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
+       "HidD_SetOutputReport returned error %u\n", GetLastError());
+
+    SetLastError(0xdeadbeef);
+    memset(buffer, 0x5a, sizeof(buffer));
+    ret = HidD_SetOutputReport(file, buffer, caps.OutputReportByteLength);
+    if (report_id || broken(!ret))
+    {
+        ok(!ret, "HidD_SetOutputReport succeeded, last error %u\n", GetLastError());
+        ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
+           "HidD_SetOutputReport returned error %u\n", GetLastError());
+    }
+    else
+    {
+        ok(ret, "HidD_SetOutputReport failed, last error %u\n", GetLastError());
+    }
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_SetOutputReport(file, report, caps.OutputReportByteLength);
+    ok(ret, "HidD_SetOutputReport failed, last error %u\n", GetLastError());
+
+    value = caps.OutputReportByteLength * 2;
+    SetLastError(0xdeadbeef);
+    ret = sync_ioctl(file, IOCTL_HID_SET_OUTPUT_REPORT, NULL, 0, report, &value);
+    ok(!ret, "IOCTL_HID_SET_OUTPUT_REPORT succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "IOCTL_HID_SET_OUTPUT_REPORT returned error %u\n", GetLastError());
+    value = 0;
+    SetLastError(0xdeadbeef);
+    ret = sync_ioctl(file, IOCTL_HID_SET_OUTPUT_REPORT, report, caps.OutputReportByteLength * 2, NULL, &value);
+    ok(ret, "IOCTL_HID_SET_OUTPUT_REPORT failed, last error %u\n", GetLastError());
+    ok(value == 3, "got length %u, expected 3\n", value);
+
+
+    SetLastError(0xdeadbeef);
+    ret = WriteFile(file, report, 0, &value, NULL);
+    ok(!ret, "WriteFile succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "WriteFile returned error %u\n", GetLastError());
+    ok(value == 0, "WriteFile returned %x\n", value);
+    SetLastError(0xdeadbeef);
+    ret = WriteFile(file, report, caps.OutputReportByteLength - 1, &value, NULL);
+    ok(!ret, "WriteFile succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_PARAMETER || GetLastError() == ERROR_INVALID_USER_BUFFER,
+       "WriteFile returned error %u\n", GetLastError());
+    ok(value == 0, "WriteFile returned %x\n", value);
+
+    memset(report, 0xcd, sizeof(report));
+    report[0] = 0xa5;
+    SetLastError(0xdeadbeef);
+    ret = WriteFile(file, report, caps.OutputReportByteLength * 2, &value, NULL);
+    if (report_id || broken(!ret) /* w7u */)
+    {
+        ok(!ret, "WriteFile succeeded\n");
+        ok(GetLastError() == ERROR_INVALID_PARAMETER, "WriteFile returned error %u\n", GetLastError());
+        ok(value == 0, "WriteFile wrote %u\n", value);
+        SetLastError(0xdeadbeef);
+        report[0] = report_id;
+        ret = WriteFile(file, report, caps.OutputReportByteLength, &value, NULL);
+    }
+
+    if (report_id)
+    {
+        ok(ret, "WriteFile failed, last error %u\n", GetLastError());
+        ok(value == 2, "WriteFile wrote %u\n", value);
+    }
+    else
+    {
+        ok(ret, "WriteFile failed, last error %u\n", GetLastError());
+        ok(value == 3, "WriteFile wrote %u\n", value);
+    }
+
+
+    memset(report, 0xcd, sizeof(report));
+    SetLastError(0xdeadbeef);
+    ret = ReadFile(file, report, 0, &value, NULL);
+    ok(!ret && GetLastError() == ERROR_INVALID_USER_BUFFER, "ReadFile failed, last error %u\n", GetLastError());
+    ok(value == 0, "ReadFile returned %x\n", value);
+    SetLastError(0xdeadbeef);
+    ret = ReadFile(file, report, caps.InputReportByteLength - 1, &value, NULL);
+    ok(!ret && GetLastError() == ERROR_INVALID_USER_BUFFER, "ReadFile failed, last error %u\n", GetLastError());
+    ok(value == 0, "ReadFile returned %x\n", value);
+
+    if (polled)
+    {
+        memset(report, 0xcd, sizeof(report));
+        SetLastError(0xdeadbeef);
+        ret = ReadFile(file, report, caps.InputReportByteLength, &value, NULL);
+        ok(ret, "ReadFile failed, last error %u\n", GetLastError());
+        ok(value == (report_id ? 3 : 4), "ReadFile returned %x\n", value);
+        ok(report[0] == report_id, "unexpected report data\n");
+
+        overlapped.hEvent = CreateEventA(NULL, FALSE, FALSE, NULL);
+        overlapped2.hEvent = CreateEventA(NULL, FALSE, FALSE, NULL);
+
+        /* drain available input reports */
+        SetLastError(0xdeadbeef);
+        while (ReadFile(async_file, report, caps.InputReportByteLength, NULL, &overlapped))
+            ResetEvent(overlapped.hEvent);
+        ok(GetLastError() == ERROR_IO_PENDING, "ReadFile returned error %u\n", GetLastError());
+        ret = GetOverlappedResult(async_file, &overlapped, &value, TRUE);
+        ok(ret, "GetOverlappedResult failed, last error %u\n", GetLastError());
+        ok(value == (report_id ? 3 : 4), "GetOverlappedResult returned length %u, expected 3\n", value);
+        ResetEvent(overlapped.hEvent);
+
+        memcpy(buffer, report, caps.InputReportByteLength);
+        memcpy(buffer + caps.InputReportByteLength, report, caps.InputReportByteLength);
+
+        SetLastError(0xdeadbeef);
+        ret = ReadFile(async_file, report, caps.InputReportByteLength, NULL, &overlapped);
+        ok(!ret, "ReadFile succeeded\n");
+        ok(GetLastError() == ERROR_IO_PENDING, "ReadFile returned error %u\n", GetLastError());
+
+        SetLastError(0xdeadbeef);
+        ret = ReadFile(async_file, buffer, caps.InputReportByteLength, NULL, &overlapped2);
+        ok(!ret, "ReadFile succeeded\n");
+        ok(GetLastError() == ERROR_IO_PENDING, "ReadFile returned error %u\n", GetLastError());
+
+        /* wait for second report to be ready */
+        ret = GetOverlappedResult(async_file, &overlapped2, &value, TRUE);
+        ok(ret, "GetOverlappedResult failed, last error %u\n", GetLastError());
+        ok(value == (report_id ? 3 : 4), "GetOverlappedResult returned length %u, expected 3\n", value);
+        /* first report should be ready and the same */
+        ret = GetOverlappedResult(async_file, &overlapped, &value, FALSE);
+        ok(ret, "GetOverlappedResult failed, last error %u\n", GetLastError());
+        ok(value == (report_id ? 3 : 4), "GetOverlappedResult returned length %u, expected 3\n", value);
+        ok(memcmp(report, buffer + caps.InputReportByteLength, caps.InputReportByteLength),
+           "expected different report\n");
+        ok(!memcmp(report, buffer, caps.InputReportByteLength), "expected identical reports\n");
+
+        CloseHandle(overlapped.hEvent);
+        CloseHandle(overlapped2.hEvent);
+    }
+
+
+    HidD_FreePreparsedData(preparsed_data);
+}
+
+struct test_hid_params
+{
+    BOOL report_id;
+    BOOL polled;
+};
+
+static void test_hid_device(void *args)
+{
+    struct test_hid_params *params = args;
+    BOOL report_id = params->report_id, polled = params->polled;
+    char buffer[200];
+    SP_DEVICE_INTERFACE_DETAIL_DATA_A *iface_detail = (void *)buffer;
+    SP_DEVICE_INTERFACE_DATA iface = {sizeof(iface)};
+    SP_DEVINFO_DATA device = {sizeof(device)};
+    ULONG count, poll_freq, out_len;
+    HANDLE file, async_file;
+    BOOL ret, found = FALSE;
+    OBJECT_ATTRIBUTES attr;
+    UNICODE_STRING string;
+    IO_STATUS_BLOCK io;
+    NTSTATUS status;
+    unsigned int i;
+    HDEVINFO set;
+    GUID guid;
+
+    winetest_push_context("id %d%s", report_id, polled ? " poll" : "");
+
+    HidD_GetHidGuid(&guid);
+
+    set = SetupDiGetClassDevsA(&guid, NULL, NULL, DIGCF_DEVICEINTERFACE | DIGCF_PRESENT);
+    ok(set != INVALID_HANDLE_VALUE, "failed to get device list, error %#x\n", GetLastError());
+
+    for (i = 0; SetupDiEnumDeviceInfo(set, i, &device); ++i)
+    {
+        ret = SetupDiEnumDeviceInterfaces(set, &device, &guid, 0, &iface);
+        ok(ret, "failed to get interface, error %#x\n", GetLastError());
+        ok(IsEqualGUID(&iface.InterfaceClassGuid, &guid),
+                "wrong class %s\n", debugstr_guid(&iface.InterfaceClassGuid));
+        ok(iface.Flags == SPINT_ACTIVE, "got flags %#x\n", iface.Flags);
+
+        iface_detail->cbSize = sizeof(*iface_detail);
+        ret = SetupDiGetDeviceInterfaceDetailA(set, &iface, iface_detail, sizeof(buffer), NULL, NULL);
+        ok(ret, "failed to get interface path, error %#x\n", GetLastError());
+
+        if (strstr(iface_detail->DevicePath, "\\\\?\\hid#winetest#1"))
+        {
+            found = TRUE;
+            break;
+        }
+    }
+
+    SetupDiDestroyDeviceInfoList(set);
+
+    todo_wine ok(found, "didn't find device\n");
+
+    file = CreateFileA(iface_detail->DevicePath, FILE_READ_ACCESS | FILE_WRITE_ACCESS,
+            FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);
+    ok(file != INVALID_HANDLE_VALUE, "got error %u\n", GetLastError());
+
+    count = 0xdeadbeef;
+    SetLastError(0xdeadbeef);
+    ret = HidD_GetNumInputBuffers(file, &count);
+    ok(ret, "HidD_GetNumInputBuffers failed last error %u\n", GetLastError());
+    ok(count == 32, "HidD_GetNumInputBuffers returned %u\n", count);
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_SetNumInputBuffers(file, 1);
+    ok(!ret, "HidD_SetNumInputBuffers succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_PARAMETER, "HidD_SetNumInputBuffers returned error %u\n", GetLastError());
+    SetLastError(0xdeadbeef);
+    ret = HidD_SetNumInputBuffers(file, 513);
+    ok(!ret, "HidD_SetNumInputBuffers succeeded\n");
+    ok(GetLastError() == ERROR_INVALID_PARAMETER, "HidD_SetNumInputBuffers returned error %u\n", GetLastError());
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_SetNumInputBuffers(file, 16);
+    ok(ret, "HidD_SetNumInputBuffers failed last error %u\n", GetLastError());
+
+    count = 0xdeadbeef;
+    SetLastError(0xdeadbeef);
+    ret = HidD_GetNumInputBuffers(file, &count);
+    ok(ret, "HidD_GetNumInputBuffers failed last error %u\n", GetLastError());
+    ok(count == 16, "HidD_GetNumInputBuffers returned %u\n", count);
+
+    async_file = CreateFileA(iface_detail->DevicePath, FILE_READ_ACCESS | FILE_WRITE_ACCESS,
+            FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING,
+            FILE_FLAG_OVERLAPPED | FILE_FLAG_NO_BUFFERING, NULL);
+    ok(async_file != INVALID_HANDLE_VALUE, "got error %u\n", GetLastError());
+
+    count = 0xdeadbeef;
+    SetLastError(0xdeadbeef);
+    ret = HidD_GetNumInputBuffers(async_file, &count);
+    ok(ret, "HidD_GetNumInputBuffers failed last error %u\n", GetLastError());
+    ok(count == 32, "HidD_GetNumInputBuffers returned %u\n", count);
+
+    SetLastError(0xdeadbeef);
+    ret = HidD_SetNumInputBuffers(async_file, 2);
+    ok(ret, "HidD_SetNumInputBuffers failed last error %u\n", GetLastError());
+
+    count = 0xdeadbeef;
+    SetLastError(0xdeadbeef);
+    ret = HidD_GetNumInputBuffers(async_file, &count);
+    ok(ret, "HidD_GetNumInputBuffers failed last error %u\n", GetLastError());
+    ok(count == 2, "HidD_GetNumInputBuffers returned %u\n", count);
+    count = 0xdeadbeef;
+    SetLastError(0xdeadbeef);
+    ret = HidD_GetNumInputBuffers(file, &count);
+    ok(ret, "HidD_GetNumInputBuffers failed last error %u\n", GetLastError());
+    ok(count == 16, "HidD_GetNumInputBuffers returned %u\n", count);
+
+    if (polled)
+    {
+        out_len = sizeof(ULONG);
+        SetLastError(0xdeadbeef);
+        ret = sync_ioctl(file, IOCTL_HID_GET_POLL_FREQUENCY_MSEC, NULL, 0, &poll_freq, &out_len);
+        ok(ret, "IOCTL_HID_GET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
+        ok(out_len == sizeof(ULONG), "got out_len %u, expected sizeof(ULONG)\n", out_len);
+        todo_wine ok(poll_freq == 5, "got poll_freq %u, expected 5\n", poll_freq);
+
+        out_len = 0;
+        poll_freq = 500;
+        SetLastError(0xdeadbeef);
+        ret = sync_ioctl(file, IOCTL_HID_SET_POLL_FREQUENCY_MSEC, &poll_freq, sizeof(ULONG), NULL, &out_len);
+        ok(ret, "IOCTL_HID_SET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
+        ok(out_len == 0, "got out_len %u, expected 0\n", out_len);
+
+        out_len = 0;
+        poll_freq = 10001;
+        SetLastError(0xdeadbeef);
+        ret = sync_ioctl(file, IOCTL_HID_SET_POLL_FREQUENCY_MSEC, &poll_freq, sizeof(ULONG), NULL, &out_len);
+        ok(ret, "IOCTL_HID_SET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
+        ok(out_len == 0, "got out_len %u, expected 0\n", out_len);
+
+        out_len = 0;
+        poll_freq = 0;
+        SetLastError(0xdeadbeef);
+        ret = sync_ioctl(file, IOCTL_HID_SET_POLL_FREQUENCY_MSEC, &poll_freq, sizeof(ULONG), NULL, &out_len);
+        ok(ret, "IOCTL_HID_SET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
+        ok(out_len == 0, "got out_len %u, expected 0\n", out_len);
+
+        out_len = sizeof(ULONG);
+        SetLastError(0xdeadbeef);
+        ret = sync_ioctl(file, IOCTL_HID_GET_POLL_FREQUENCY_MSEC, NULL, 0, &poll_freq, &out_len);
+        ok(ret, "IOCTL_HID_GET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
+        ok(out_len == sizeof(ULONG), "got out_len %u, expected sizeof(ULONG)\n", out_len);
+        ok(poll_freq == 10000, "got poll_freq %u, expected 10000\n", poll_freq);
+
+        out_len = 0;
+        poll_freq = 500;
+        SetLastError(0xdeadbeef);
+        ret = sync_ioctl(file, IOCTL_HID_SET_POLL_FREQUENCY_MSEC, &poll_freq, sizeof(ULONG), NULL, &out_len);
+        ok(ret, "IOCTL_HID_SET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
+        ok(out_len == 0, "got out_len %u, expected 0\n", out_len);
+
+        out_len = sizeof(ULONG);
+        SetLastError(0xdeadbeef);
+        ret = sync_ioctl(async_file, IOCTL_HID_GET_POLL_FREQUENCY_MSEC, NULL, 0, &poll_freq, &out_len);
+        ok(ret, "IOCTL_HID_GET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
+        ok(out_len == sizeof(ULONG), "got out_len %u, expected sizeof(ULONG)\n", out_len);
+        ok(poll_freq == 500, "got poll_freq %u, expected 500\n", poll_freq);
+    }
+
+    test_hidp(file, async_file, report_id, polled);
+
+    CloseHandle(async_file);
+    CloseHandle(file);
+
+    RtlInitUnicodeString(&string, L"\\??\\root#winetest#0#{deadbeef-29ef-4538-a5fd-b69573a362c0}");
+    InitializeObjectAttributes(&attr, &string, OBJ_CASE_INSENSITIVE, NULL, NULL);
+    status = NtOpenFile(&file, SYNCHRONIZE, &attr, &io, 0, FILE_SYNCHRONOUS_IO_NONALERT);
+    todo_wine ok(status == STATUS_UNSUCCESSFUL, "got %#x\n", status);
+
+    winetest_pop_context();
+}
+
+static void test_hid_driver(struct testsign_context *ctx, DWORD report_id, DWORD polled)
+{
+    struct test_hid_params params = {.report_id = report_id, .polled = polled};
+    SP_DEVINFO_DATA device = {sizeof(device)};
+    char cwd[MAX_PATH], tempdir[MAX_PATH];
+    LSTATUS status;
+    HKEY hkey;
+
+    GetCurrentDirectoryA(ARRAY_SIZE(cwd), cwd);
+    GetTempPathA(ARRAY_SIZE(tempdir), tempdir);
+    SetCurrentDirectoryA(tempdir);
+
+    status = RegCreateKeyExW(HKEY_LOCAL_MACHINE, L"System\\CurrentControlSet\\Services\\winetest", 0, NULL, REG_OPTION_VOLATILE, KEY_ALL_ACCESS, NULL, &hkey, NULL);
+    ok(!status, "RegCreateKeyExW returned %#x\n", status);
+
+    status = RegSetValueExW(hkey, L"ReportID", 0, REG_DWORD, (void *)&report_id, sizeof(report_id));
+    ok(!status, "RegSetValueExW returned %#x\n", status);
+
+    status = RegSetValueExW(hkey, L"PolledMode", 0, REG_DWORD, (void *)&polled, sizeof(polled));
+    ok(!status, "RegSetValueExW returned %#x\n", status);
+
+    subtest("driver_hid");
+    run_driver_test(ctx, L"driver_hid.dll", test_hid_device, &params);
+}
+
+START_TEST(hidp)
+{
+    struct testsign_context ctx;
+    if (!driver_test_init(&ctx))
+        return;
+
+    test_hid_driver(&ctx, 0, FALSE);
+    test_hid_driver(&ctx, 1, FALSE);
+    test_hid_driver(&ctx, 0, TRUE);
+    test_hid_driver(&ctx, 1, TRUE);
+
+    driver_test_cleanup(&ctx);
+}
diff --git a/dlls/ntoskrnl.exe/tests/Makefile.in b/dlls/ntoskrnl.exe/tests/Makefile.in
index 85a6925138a..ab1db85adbb 100644
--- a/dlls/ntoskrnl.exe/tests/Makefile.in
+++ b/dlls/ntoskrnl.exe/tests/Makefile.in
@@ -7,8 +7,6 @@ driver2_IMPORTS = winecrt0 ntoskrnl hal
 driver2_EXTRADLLFLAGS = -nodefaultlibs -nostartfiles -Wl,--subsystem,native
 driver3_IMPORTS = winecrt0 ntoskrnl hal
 driver3_EXTRADLLFLAGS = -nodefaultlibs -nostartfiles -Wl,--subsystem,native
-driver_hid_IMPORTS = winecrt0 ntoskrnl hal hidclass
-driver_hid_EXTRADLLFLAGS = -nodefaultlibs -nostartfiles -Wl,--subsystem,native
 driver_netio_IMPORTS = winecrt0 ntoskrnl hal netio
 driver_netio_EXTRADLLFLAGS = -nodefaultlibs -nostartfiles -Wl,--subsystem,native
 driver_pnp_IMPORTS = winecrt0 ntoskrnl hal
@@ -21,8 +19,6 @@ SOURCES = \
 	driver2.spec \
 	driver3.c \
 	driver3.spec \
-	driver_hid.c \
-	driver_hid.spec \
 	driver_netio.c \
 	driver_netio.spec \
 	driver_pnp.c \
diff --git a/dlls/ntoskrnl.exe/tests/ntoskrnl.c b/dlls/ntoskrnl.exe/tests/ntoskrnl.c
index a0636099def..9c8b1f96371 100644
--- a/dlls/ntoskrnl.exe/tests/ntoskrnl.c
+++ b/dlls/ntoskrnl.exe/tests/ntoskrnl.c
@@ -1341,1508 +1341,6 @@ static void test_pnp_devices(void *args)
     UnregisterClassA("ntoskrnl_test_wc", GetModuleHandleA(NULL));
 }
 
-#define check_member_(file, line, val, exp, fmt, member)               \
-        ok_(file, line)((val).member == (exp).member,                  \
-                        "got " #member " " fmt ", expected " fmt "\n", \
-                        (val).member, (exp).member)
-#define check_member(val, exp, fmt, member) check_member_(__FILE__, __LINE__, val, exp, fmt, member)
-
-#define check_hidp_caps(a, b) check_hidp_caps_(__LINE__, a, b)
-static inline void check_hidp_caps_(int line, HIDP_CAPS *caps, const HIDP_CAPS *exp)
-{
-    check_member_(__FILE__, line, *caps, *exp, "%04x", Usage);
-    check_member_(__FILE__, line, *caps, *exp, "%04x", UsagePage);
-    check_member_(__FILE__, line, *caps, *exp, "%d", InputReportByteLength);
-    check_member_(__FILE__, line, *caps, *exp, "%d", OutputReportByteLength);
-    check_member_(__FILE__, line, *caps, *exp, "%d", FeatureReportByteLength);
-    check_member_(__FILE__, line, *caps, *exp, "%d", NumberLinkCollectionNodes);
-    check_member_(__FILE__, line, *caps, *exp, "%d", NumberInputButtonCaps);
-    check_member_(__FILE__, line, *caps, *exp, "%d", NumberInputValueCaps);
-    check_member_(__FILE__, line, *caps, *exp, "%d", NumberInputDataIndices);
-    check_member_(__FILE__, line, *caps, *exp, "%d", NumberOutputButtonCaps);
-    check_member_(__FILE__, line, *caps, *exp, "%d", NumberOutputValueCaps);
-    check_member_(__FILE__, line, *caps, *exp, "%d", NumberOutputDataIndices);
-    check_member_(__FILE__, line, *caps, *exp, "%d", NumberFeatureButtonCaps);
-    check_member_(__FILE__, line, *caps, *exp, "%d", NumberFeatureValueCaps);
-    check_member_(__FILE__, line, *caps, *exp, "%d", NumberFeatureDataIndices);
-}
-
-#define check_hidp_link_collection_node(a, b) check_hidp_link_collection_node_(__LINE__, a, b)
-static inline void check_hidp_link_collection_node_(int line, HIDP_LINK_COLLECTION_NODE *node,
-                                                    const HIDP_LINK_COLLECTION_NODE *exp)
-{
-    check_member_(__FILE__, line, *node, *exp, "%04x", LinkUsage);
-    check_member_(__FILE__, line, *node, *exp, "%04x", LinkUsagePage);
-    check_member_(__FILE__, line, *node, *exp, "%d", Parent);
-    check_member_(__FILE__, line, *node, *exp, "%d", NumberOfChildren);
-    check_member_(__FILE__, line, *node, *exp, "%d", NextSibling);
-    check_member_(__FILE__, line, *node, *exp, "%d", FirstChild);
-    check_member_(__FILE__, line, *node, *exp, "%d", CollectionType);
-    check_member_(__FILE__, line, *node, *exp, "%d", IsAlias);
-}
-
-#define check_hidp_button_caps(a, b) check_hidp_button_caps_(__LINE__, a, b)
-static inline void check_hidp_button_caps_(int line, HIDP_BUTTON_CAPS *caps, const HIDP_BUTTON_CAPS *exp)
-{
-    check_member_(__FILE__, line, *caps, *exp, "%04x", UsagePage);
-    check_member_(__FILE__, line, *caps, *exp, "%d", ReportID);
-    check_member_(__FILE__, line, *caps, *exp, "%d", IsAlias);
-    check_member_(__FILE__, line, *caps, *exp, "%d", BitField);
-    check_member_(__FILE__, line, *caps, *exp, "%d", LinkCollection);
-    check_member_(__FILE__, line, *caps, *exp, "%04x", LinkUsage);
-    check_member_(__FILE__, line, *caps, *exp, "%04x", LinkUsagePage);
-    check_member_(__FILE__, line, *caps, *exp, "%d", IsRange);
-    check_member_(__FILE__, line, *caps, *exp, "%d", IsStringRange);
-    check_member_(__FILE__, line, *caps, *exp, "%d", IsDesignatorRange);
-    check_member_(__FILE__, line, *caps, *exp, "%d", IsAbsolute);
-
-    if (!caps->IsRange && !exp->IsRange)
-    {
-        check_member_(__FILE__, line, *caps, *exp, "%04x", NotRange.Usage);
-        check_member_(__FILE__, line, *caps, *exp, "%d", NotRange.DataIndex);
-    }
-    else if (caps->IsRange && exp->IsRange)
-    {
-        check_member_(__FILE__, line, *caps, *exp, "%04x", Range.UsageMin);
-        check_member_(__FILE__, line, *caps, *exp, "%04x", Range.UsageMax);
-        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DataIndexMin);
-        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DataIndexMax);
-    }
-
-    if (!caps->IsRange && !exp->IsRange)
-        check_member_(__FILE__, line, *caps, *exp, "%d", NotRange.StringIndex);
-    else if (caps->IsStringRange && exp->IsStringRange)
-    {
-        check_member_(__FILE__, line, *caps, *exp, "%d", Range.StringMin);
-        check_member_(__FILE__, line, *caps, *exp, "%d", Range.StringMax);
-    }
-
-    if (!caps->IsDesignatorRange && !exp->IsDesignatorRange)
-        check_member_(__FILE__, line, *caps, *exp, "%d", NotRange.DesignatorIndex);
-    else if (caps->IsDesignatorRange && exp->IsDesignatorRange)
-    {
-        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DesignatorMin);
-        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DesignatorMax);
-    }
-}
-
-#define check_hidp_value_caps(a, b) check_hidp_value_caps_(__LINE__, a, b)
-static inline void check_hidp_value_caps_(int line, HIDP_VALUE_CAPS *caps, const HIDP_VALUE_CAPS *exp)
-{
-    check_member_(__FILE__, line, *caps, *exp, "%04x", UsagePage);
-    check_member_(__FILE__, line, *caps, *exp, "%d", ReportID);
-    check_member_(__FILE__, line, *caps, *exp, "%d", IsAlias);
-    check_member_(__FILE__, line, *caps, *exp, "%d", BitField);
-    check_member_(__FILE__, line, *caps, *exp, "%d", LinkCollection);
-    check_member_(__FILE__, line, *caps, *exp, "%d", LinkUsage);
-    check_member_(__FILE__, line, *caps, *exp, "%d", LinkUsagePage);
-    check_member_(__FILE__, line, *caps, *exp, "%d", IsRange);
-    check_member_(__FILE__, line, *caps, *exp, "%d", IsStringRange);
-    check_member_(__FILE__, line, *caps, *exp, "%d", IsDesignatorRange);
-    check_member_(__FILE__, line, *caps, *exp, "%d", IsAbsolute);
-
-    check_member_(__FILE__, line, *caps, *exp, "%d", HasNull);
-    check_member_(__FILE__, line, *caps, *exp, "%d", BitSize);
-    check_member_(__FILE__, line, *caps, *exp, "%d", ReportCount);
-    check_member_(__FILE__, line, *caps, *exp, "%d", UnitsExp);
-    check_member_(__FILE__, line, *caps, *exp, "%d", Units);
-    check_member_(__FILE__, line, *caps, *exp, "%d", LogicalMin);
-    check_member_(__FILE__, line, *caps, *exp, "%d", LogicalMax);
-    check_member_(__FILE__, line, *caps, *exp, "%d", PhysicalMin);
-    check_member_(__FILE__, line, *caps, *exp, "%d", PhysicalMax);
-
-    if (!caps->IsRange && !exp->IsRange)
-    {
-        check_member_(__FILE__, line, *caps, *exp, "%04x", NotRange.Usage);
-        check_member_(__FILE__, line, *caps, *exp, "%d", NotRange.DataIndex);
-    }
-    else if (caps->IsRange && exp->IsRange)
-    {
-        check_member_(__FILE__, line, *caps, *exp, "%04x", Range.UsageMin);
-        check_member_(__FILE__, line, *caps, *exp, "%04x", Range.UsageMax);
-        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DataIndexMin);
-        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DataIndexMax);
-    }
-
-    if (!caps->IsRange && !exp->IsRange)
-        check_member_(__FILE__, line, *caps, *exp, "%d", NotRange.StringIndex);
-    else if (caps->IsStringRange && exp->IsStringRange)
-    {
-        check_member_(__FILE__, line, *caps, *exp, "%d", Range.StringMin);
-        check_member_(__FILE__, line, *caps, *exp, "%d", Range.StringMax);
-    }
-
-    if (!caps->IsDesignatorRange && !exp->IsDesignatorRange)
-        check_member_(__FILE__, line, *caps, *exp, "%d", NotRange.DesignatorIndex);
-    else if (caps->IsDesignatorRange && exp->IsDesignatorRange)
-    {
-        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DesignatorMin);
-        check_member_(__FILE__, line, *caps, *exp, "%d", Range.DesignatorMax);
-    }
-}
-
-static BOOL sync_ioctl(HANDLE file, DWORD code, void *in_buf, DWORD in_len, void *out_buf, DWORD *ret_len)
-{
-    OVERLAPPED ovl = {0};
-    DWORD out_len = ret_len ? *ret_len : 0;
-    BOOL ret;
-
-    ovl.hEvent = CreateEventW(NULL, TRUE, FALSE, NULL);
-    ret = DeviceIoControl(file, code, in_buf, in_len, out_buf, out_len, &out_len, &ovl);
-    if (!ret && GetLastError() == ERROR_IO_PENDING) ret = GetOverlappedResult(file, &ovl, &out_len, TRUE);
-    CloseHandle(ovl.hEvent);
-
-    if (ret_len) *ret_len = out_len;
-    return ret;
-}
-
-static void test_hidp(HANDLE file, HANDLE async_file, int report_id, BOOL polled)
-{
-    const HIDP_CAPS expect_hidp_caps[] =
-    {
-        /* without report id */
-        {
-            .Usage = HID_USAGE_GENERIC_JOYSTICK,
-            .UsagePage = HID_USAGE_PAGE_GENERIC,
-            .InputReportByteLength = 26,
-            .OutputReportByteLength = 3,
-            .FeatureReportByteLength = 22,
-            .NumberLinkCollectionNodes = 10,
-            .NumberInputButtonCaps = 17,
-            .NumberInputValueCaps = 7,
-            .NumberInputDataIndices = 47,
-            .NumberFeatureButtonCaps = 1,
-            .NumberFeatureValueCaps = 6,
-            .NumberFeatureDataIndices = 8,
-        },
-        /* with report id */
-        {
-            .Usage = HID_USAGE_GENERIC_JOYSTICK,
-            .UsagePage = HID_USAGE_PAGE_GENERIC,
-            .InputReportByteLength = 25,
-            .OutputReportByteLength = 2,
-            .FeatureReportByteLength = 21,
-            .NumberLinkCollectionNodes = 10,
-            .NumberInputButtonCaps = 17,
-            .NumberInputValueCaps = 7,
-            .NumberInputDataIndices = 47,
-            .NumberFeatureButtonCaps = 1,
-            .NumberFeatureValueCaps = 6,
-            .NumberFeatureDataIndices = 8,
-        },
-    };
-    const HIDP_BUTTON_CAPS expect_button_caps[] =
-    {
-        {
-            .UsagePage = HID_USAGE_PAGE_BUTTON,
-            .ReportID = report_id,
-            .BitField = 2,
-            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
-            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
-            .LinkCollection = 1,
-            .IsRange = TRUE,
-            .IsAbsolute = TRUE,
-            .Range.UsageMin = 1,
-            .Range.UsageMax = 8,
-            .Range.DataIndexMin = 2,
-            .Range.DataIndexMax = 9,
-        },
-        {
-            .UsagePage = HID_USAGE_PAGE_BUTTON,
-            .ReportID = report_id,
-            .BitField = 3,
-            .LinkCollection = 1,
-            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
-            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
-            .IsRange = TRUE,
-            .IsAbsolute = TRUE,
-            .Range.UsageMin = 0x18,
-            .Range.UsageMax = 0x1f,
-            .Range.DataIndexMin = 10,
-            .Range.DataIndexMax = 17,
-        },
-        {
-            .UsagePage = HID_USAGE_PAGE_KEYBOARD,
-            .ReportID = report_id,
-            .BitField = 0x1fc,
-            .LinkCollection = 1,
-            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
-            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
-            .IsRange = TRUE,
-            .IsAbsolute = FALSE,
-            .Range.UsageMin = 0x8,
-            .Range.UsageMax = 0xf,
-            .Range.DataIndexMin = 18,
-            .Range.DataIndexMax = 25,
-        },
-        {
-            .UsagePage = HID_USAGE_PAGE_BUTTON,
-            .ReportID = report_id,
-            .BitField = 2,
-            .LinkCollection = 1,
-            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
-            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
-            .IsRange = FALSE,
-            .IsAbsolute = TRUE,
-            .NotRange.Usage = 0x20,
-            .NotRange.Reserved1 = 0x20,
-            .NotRange.DataIndex = 26,
-            .NotRange.Reserved4 = 26,
-        },
-    };
-    const HIDP_VALUE_CAPS expect_value_caps[] =
-    {
-        {
-            .UsagePage = HID_USAGE_PAGE_GENERIC,
-            .ReportID = report_id,
-            .BitField = 2,
-            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
-            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
-            .LinkCollection = 1,
-            .IsAbsolute = TRUE,
-            .BitSize = 8,
-            .ReportCount = 1,
-            .LogicalMin = -128,
-            .LogicalMax = 127,
-            .NotRange.Usage = HID_USAGE_GENERIC_Y,
-            .NotRange.Reserved1 = HID_USAGE_GENERIC_Y,
-        },
-        {
-            .UsagePage = HID_USAGE_PAGE_GENERIC,
-            .ReportID = report_id,
-            .BitField = 2,
-            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
-            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
-            .LinkCollection = 1,
-            .IsAbsolute = TRUE,
-            .BitSize = 8,
-            .ReportCount = 1,
-            .LogicalMin = -128,
-            .LogicalMax = 127,
-            .NotRange.Usage = HID_USAGE_GENERIC_X,
-            .NotRange.Reserved1 = HID_USAGE_GENERIC_X,
-            .NotRange.DataIndex = 1,
-            .NotRange.Reserved4 = 1,
-        },
-        {
-            .UsagePage = HID_USAGE_PAGE_BUTTON,
-            .ReportID = report_id,
-            .BitField = 2,
-            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
-            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
-            .LinkCollection = 1,
-            .IsAbsolute = TRUE,
-            .ReportCount = 1,
-            .LogicalMax = 1,
-            .IsRange = TRUE,
-            .Range.UsageMin = 0x21,
-            .Range.UsageMax = 0x22,
-            .Range.DataIndexMin = 27,
-            .Range.DataIndexMax = 28,
-        },
-        {
-            .UsagePage = HID_USAGE_PAGE_GENERIC,
-            .ReportID = report_id,
-            .BitField = 2,
-            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
-            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
-            .LinkCollection = 1,
-            .IsAbsolute = TRUE,
-            .BitSize = 4,
-            .ReportCount = 2,
-            .LogicalMin = 1,
-            .LogicalMax = 8,
-            .NotRange.Usage = HID_USAGE_GENERIC_HATSWITCH,
-            .NotRange.Reserved1 = HID_USAGE_GENERIC_HATSWITCH,
-            .NotRange.DataIndex = 29,
-            .NotRange.Reserved4 = 29,
-        },
-    };
-    static const HIDP_LINK_COLLECTION_NODE expect_collections[] =
-    {
-        {
-            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
-            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
-            .CollectionType = 1,
-            .NumberOfChildren = 7,
-            .FirstChild = 9,
-        },
-        {
-            .LinkUsage = HID_USAGE_GENERIC_JOYSTICK,
-            .LinkUsagePage = HID_USAGE_PAGE_GENERIC,
-            .CollectionType = 2,
-        },
-    };
-    static const HIDP_DATA expect_data[] =
-    {
-        { .DataIndex = 0, },
-        { .DataIndex = 1, },
-        { .DataIndex = 5, .RawValue = 1, },
-        { .DataIndex = 7, .RawValue = 1, },
-        { .DataIndex = 19, .RawValue = 1, },
-        { .DataIndex = 21, .RawValue = 1, },
-        { .DataIndex = 30, },
-        { .DataIndex = 31, },
-        { .DataIndex = 32, .RawValue = 0xfeedcafe, },
-        { .DataIndex = 37, .RawValue = 1, },
-        { .DataIndex = 39, .RawValue = 1, },
-    };
-
-    OVERLAPPED overlapped = {0}, overlapped2 = {0};
-    HIDP_LINK_COLLECTION_NODE collections[16];
-    PHIDP_PREPARSED_DATA preparsed_data;
-    USAGE_AND_PAGE usage_and_pages[16];
-    HIDP_BUTTON_CAPS button_caps[32];
-    HIDP_VALUE_CAPS value_caps[16];
-    char buffer[200], report[200];
-    DWORD collection_count;
-    DWORD waveform_list;
-    HIDP_DATA data[64];
-    USAGE usages[16];
-    NTSTATUS status;
-    HIDP_CAPS caps;
-    unsigned int i;
-    USHORT count;
-    ULONG value;
-    BOOL ret;
-
-    ret = HidD_GetPreparsedData(file, &preparsed_data);
-    ok(ret, "HidD_GetPreparsedData failed with error %u\n", GetLastError());
-
-    memset(buffer, 0, sizeof(buffer));
-    status = HidP_GetCaps((PHIDP_PREPARSED_DATA)buffer, &caps);
-    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetCaps returned %#x\n", status);
-    status = HidP_GetCaps(preparsed_data, &caps);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetCaps returned %#x\n", status);
-    check_hidp_caps(&caps, &expect_hidp_caps[report_id]);
-
-    collection_count = 0;
-    status = HidP_GetLinkCollectionNodes(collections, &collection_count, preparsed_data);
-    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetLinkCollectionNodes returned %#x\n", status);
-    ok(collection_count == caps.NumberLinkCollectionNodes, "got %d collection nodes, expected %d\n",
-       collection_count, caps.NumberLinkCollectionNodes);
-    collection_count = ARRAY_SIZE(collections);
-    status = HidP_GetLinkCollectionNodes(collections, &collection_count, (PHIDP_PREPARSED_DATA)buffer);
-    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetLinkCollectionNodes returned %#x\n", status);
-    status = HidP_GetLinkCollectionNodes(collections, &collection_count, preparsed_data);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetLinkCollectionNodes returned %#x\n", status);
-    ok(collection_count == caps.NumberLinkCollectionNodes, "got %d collection nodes, expected %d\n",
-       collection_count, caps.NumberLinkCollectionNodes);
-
-    for (i = 0; i < ARRAY_SIZE(expect_collections); ++i)
-    {
-        winetest_push_context("collections[%d]", i);
-        check_hidp_link_collection_node(&collections[i], &expect_collections[i]);
-        winetest_pop_context();
-    }
-
-    count = ARRAY_SIZE(button_caps);
-    status = HidP_GetButtonCaps(HidP_Output, button_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetButtonCaps returned %#x\n", status);
-    status = HidP_GetButtonCaps(HidP_Feature + 1, button_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_GetButtonCaps returned %#x\n", status);
-    count = 0;
-    status = HidP_GetButtonCaps(HidP_Input, button_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetButtonCaps returned %#x\n", status);
-    ok(count == caps.NumberInputButtonCaps, "HidP_GetButtonCaps returned count %d, expected %d\n",
-       count, caps.NumberInputButtonCaps);
-    count = ARRAY_SIZE(button_caps);
-    status = HidP_GetButtonCaps(HidP_Input, button_caps, &count, (PHIDP_PREPARSED_DATA)buffer);
-    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetButtonCaps returned %#x\n", status);
-    memset(button_caps, 0, sizeof(button_caps));
-    status = HidP_GetButtonCaps(HidP_Input, button_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetButtonCaps returned %#x\n", status);
-    ok(count == caps.NumberInputButtonCaps, "HidP_GetButtonCaps returned count %d, expected %d\n",
-       count, caps.NumberInputButtonCaps);
-
-    for (i = 0; i < ARRAY_SIZE(expect_button_caps); ++i)
-    {
-        winetest_push_context("button_caps[%d]", i);
-        check_hidp_button_caps(&button_caps[i], &expect_button_caps[i]);
-        winetest_pop_context();
-    }
-
-    count = ARRAY_SIZE(button_caps) - 1;
-    status = HidP_GetSpecificButtonCaps(HidP_Output, 0, 0, 0, button_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificButtonCaps returned %#x\n", status);
-    status = HidP_GetSpecificButtonCaps(HidP_Feature + 1, 0, 0, 0, button_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_GetSpecificButtonCaps returned %#x\n", status);
-    count = 0;
-    status = HidP_GetSpecificButtonCaps(HidP_Input, 0, 0, 0, button_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetSpecificButtonCaps returned %#x\n", status);
-    ok(count == caps.NumberInputButtonCaps, "HidP_GetSpecificButtonCaps returned count %d, expected %d\n",
-       count, caps.NumberInputButtonCaps);
-    count = ARRAY_SIZE(button_caps) - 1;
-    status = HidP_GetSpecificButtonCaps(HidP_Input, 0, 0, 0, button_caps, &count, (PHIDP_PREPARSED_DATA)buffer);
-    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetSpecificButtonCaps returned %#x\n", status);
-
-    status = HidP_GetSpecificButtonCaps(HidP_Input, 0, 0, 0, button_caps + 1, &count, preparsed_data);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetSpecificButtonCaps returned %#x\n", status);
-    ok(count == caps.NumberInputButtonCaps, "HidP_GetSpecificButtonCaps returned count %d, expected %d\n",
-       count, caps.NumberInputButtonCaps);
-    check_hidp_button_caps(&button_caps[1], &button_caps[0]);
-
-    status = HidP_GetSpecificButtonCaps(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, 5, button_caps + 1,
-                                        &count, preparsed_data);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetSpecificButtonCaps returned %#x\n", status);
-    ok(count == 1, "HidP_GetSpecificButtonCaps returned count %d, expected %d\n", count, 1);
-    check_hidp_button_caps(&button_caps[1], &button_caps[0]);
-
-    count = 0xbeef;
-    status = HidP_GetSpecificButtonCaps(HidP_Input, 0xfffe, 0, 0, button_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificButtonCaps returned %#x\n", status);
-    ok(count == 0, "HidP_GetSpecificButtonCaps returned count %d, expected %d\n", count, 0);
-    count = 0xbeef;
-    status = HidP_GetSpecificButtonCaps(HidP_Input, 0, 0xfffe, 0, button_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificButtonCaps returned %#x\n", status);
-    ok(count == 0, "HidP_GetSpecificButtonCaps returned count %d, expected %d\n", count, 0);
-    count = 0xbeef;
-    status = HidP_GetSpecificButtonCaps(HidP_Input, 0, 0, 0xfffe, button_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificButtonCaps returned %#x\n", status);
-    ok(count == 0, "HidP_GetSpecificButtonCaps returned count %d, expected %d\n", count, 0);
-
-    count = ARRAY_SIZE(value_caps);
-    status = HidP_GetValueCaps(HidP_Output, value_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetValueCaps returned %#x\n", status);
-    status = HidP_GetValueCaps(HidP_Feature + 1, value_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_GetValueCaps returned %#x\n", status);
-    count = 0;
-    status = HidP_GetValueCaps(HidP_Input, value_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetValueCaps returned %#x\n", status);
-    ok(count == caps.NumberInputValueCaps, "HidP_GetValueCaps returned count %d, expected %d\n",
-       count, caps.NumberInputValueCaps);
-    count = ARRAY_SIZE(value_caps);
-    status = HidP_GetValueCaps(HidP_Input, value_caps, &count, (PHIDP_PREPARSED_DATA)buffer);
-    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetValueCaps returned %#x\n", status);
-    status = HidP_GetValueCaps(HidP_Input, value_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetValueCaps returned %#x\n", status);
-    ok(count == caps.NumberInputValueCaps, "HidP_GetValueCaps returned count %d, expected %d\n",
-       count, caps.NumberInputValueCaps);
-
-    for (i = 0; i < ARRAY_SIZE(expect_value_caps); ++i)
-    {
-        winetest_push_context("value_caps[%d]", i);
-        check_hidp_value_caps(&value_caps[i], &expect_value_caps[i]);
-        winetest_pop_context();
-    }
-
-    count = ARRAY_SIZE(value_caps) - 4;
-    status = HidP_GetSpecificValueCaps(HidP_Output, 0, 0, 0, value_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificValueCaps returned %#x\n", status);
-    status = HidP_GetSpecificValueCaps(HidP_Feature + 1, 0, 0, 0, value_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_GetSpecificValueCaps returned %#x\n", status);
-    count = 0;
-    status = HidP_GetSpecificValueCaps(HidP_Input, 0, 0, 0, value_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetSpecificValueCaps returned %#x\n", status);
-    ok(count == caps.NumberInputValueCaps, "HidP_GetSpecificValueCaps returned count %d, expected %d\n",
-       count, caps.NumberInputValueCaps);
-    count = ARRAY_SIZE(value_caps) - 4;
-    status = HidP_GetSpecificValueCaps(HidP_Input, 0, 0, 0, value_caps + 4, &count, (PHIDP_PREPARSED_DATA)buffer);
-    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetSpecificValueCaps returned %#x\n", status);
-
-    status = HidP_GetSpecificValueCaps(HidP_Input, 0, 0, 0, value_caps + 4, &count, preparsed_data);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetSpecificValueCaps returned %#x\n", status);
-    ok(count == caps.NumberInputValueCaps, "HidP_GetSpecificValueCaps returned count %d, expected %d\n",
-       count, caps.NumberInputValueCaps);
-    check_hidp_value_caps(&value_caps[4], &value_caps[0]);
-    check_hidp_value_caps(&value_caps[5], &value_caps[1]);
-    check_hidp_value_caps(&value_caps[6], &value_caps[2]);
-    check_hidp_value_caps(&value_caps[7], &value_caps[3]);
-
-    count = 1;
-    status = HidP_GetSpecificValueCaps(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_HATSWITCH,
-                                       value_caps + 4, &count, preparsed_data);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetSpecificValueCaps returned %#x\n", status);
-    ok(count == 1, "HidP_GetSpecificValueCaps returned count %d, expected %d\n", count, 1);
-    check_hidp_value_caps(&value_caps[4], &value_caps[3]);
-
-    count = 0xdead;
-    status = HidP_GetSpecificValueCaps(HidP_Input, 0xfffe, 0, 0, value_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificValueCaps returned %#x\n", status);
-    ok(count == 0, "HidP_GetSpecificValueCaps returned count %d, expected %d\n", count, 0);
-    count = 0xdead;
-    status = HidP_GetSpecificValueCaps(HidP_Input, 0, 0xfffe, 0, value_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificValueCaps returned %#x\n", status);
-    ok(count == 0, "HidP_GetSpecificValueCaps returned count %d, expected %d\n", count, 0);
-    count = 0xdead;
-    status = HidP_GetSpecificValueCaps(HidP_Input, 0, 0, 0xfffe, value_caps, &count, preparsed_data);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetSpecificValueCaps returned %#x\n", status);
-    ok(count == 0, "HidP_GetSpecificValueCaps returned count %d, expected %d\n", count, 0);
-
-    status = HidP_InitializeReportForID(HidP_Input, 0, (PHIDP_PREPARSED_DATA)buffer, report, sizeof(report));
-    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_InitializeReportForID returned %#x\n", status);
-    status = HidP_InitializeReportForID(HidP_Feature + 1, 0, preparsed_data, report, sizeof(report));
-    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_InitializeReportForID returned %#x\n", status);
-    status = HidP_InitializeReportForID(HidP_Input, 0, preparsed_data, report, sizeof(report));
-    ok(status == HIDP_STATUS_INVALID_REPORT_LENGTH, "HidP_InitializeReportForID returned %#x\n", status);
-    status = HidP_InitializeReportForID(HidP_Input, 0, preparsed_data, report, caps.InputReportByteLength + 1);
-    ok(status == HIDP_STATUS_INVALID_REPORT_LENGTH, "HidP_InitializeReportForID returned %#x\n", status);
-    status = HidP_InitializeReportForID(HidP_Input, 1 - report_id, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_REPORT_DOES_NOT_EXIST, "HidP_InitializeReportForID returned %#x\n", status);
-
-    memset(report, 0xcd, sizeof(report));
-    status = HidP_InitializeReportForID(HidP_Input, report_id, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
-
-    memset(buffer, 0xcd, sizeof(buffer));
-    memset(buffer, 0, caps.InputReportByteLength);
-    buffer[0] = report_id;
-    ok(!memcmp(buffer, report, sizeof(buffer)), "unexpected report data\n");
-
-    status = HidP_SetUsageValueArray(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X, buffer,
-                                     sizeof(buffer), preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_NOT_VALUE_ARRAY, "HidP_SetUsageValueArray returned %#x\n", status);
-    memset(buffer, 0xcd, sizeof(buffer));
-    status = HidP_SetUsageValueArray(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_HATSWITCH, buffer,
-                                     0, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_SetUsageValueArray returned %#x\n", status);
-    status = HidP_SetUsageValueArray(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_HATSWITCH, buffer,
-                                     8, preparsed_data, report, caps.InputReportByteLength);
-    todo_wine
-    ok(status == HIDP_STATUS_NOT_IMPLEMENTED, "HidP_SetUsageValueArray returned %#x\n", status);
-
-    status = HidP_GetUsageValueArray(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X, buffer,
-                                     sizeof(buffer), preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_NOT_VALUE_ARRAY, "HidP_GetUsageValueArray returned %#x\n", status);
-    memset(buffer, 0xcd, sizeof(buffer));
-    status = HidP_GetUsageValueArray(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_HATSWITCH, buffer,
-                                     0, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetUsageValueArray returned %#x\n", status);
-    status = HidP_GetUsageValueArray(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_HATSWITCH, buffer,
-                                     8, preparsed_data, report, caps.InputReportByteLength);
-    todo_wine
-    ok(status == HIDP_STATUS_NOT_IMPLEMENTED, "HidP_GetUsageValueArray returned %#x\n", status);
-
-    value = -128;
-    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
-                                value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
-    value = 0xdeadbeef;
-    status = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
-                                &value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
-    ok(value == 0x80, "got value %x, expected %#x\n", value, 0x80);
-    value = 0xdeadbeef;
-    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
-                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
-    ok(value == -128, "got value %x, expected %#x\n", value, -128);
-
-    value = 127;
-    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
-                                value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
-    value = 0xdeadbeef;
-    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
-                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
-    ok(value == 127, "got value %x, expected %#x\n", value, 127);
-
-    value = 0;
-    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
-                                value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
-    value = 0xdeadbeef;
-    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X,
-                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
-    ok(value == 0, "got value %x, expected %#x\n", value, 0);
-
-    value = 0x7fffffff;
-    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
-    value = 0xdeadbeef;
-    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_VALUE_OUT_OF_RANGE, "HidP_GetScaledUsageValue returned %#x\n", status);
-    ok(value == 0, "got value %x, expected %#x\n", value, 0);
-    value = 0xdeadbeef;
-    status = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                &value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
-    ok(value == 0x7fffffff, "got value %x, expected %#x\n", value, 0x7fffffff);
-
-    value = 0x3fffffff;
-    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
-    value = 0xdeadbeef;
-    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
-    ok(value == 0x7fffffff, "got value %x, expected %#x\n", value, 0x7fffffff);
-
-    value = 0;
-    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
-    value = 0xdeadbeef;
-    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
-    ok(value == 0x80000000, "got value %x, expected %#x\n", value, 0x80000000);
-
-    value = 0;
-    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RX,
-                                value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
-    value = 0xdeadbeef;
-    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RX,
-                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
-    ok(value == 0, "got value %x, expected %#x\n", value, 0);
-
-    value = 0xfeedcafe;
-    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RY,
-                                value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
-    value = 0xdeadbeef;
-    status = HidP_GetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RY,
-                                      (LONG *)&value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_BAD_LOG_PHY_VALUES, "HidP_GetScaledUsageValue returned %#x\n", status);
-    ok(value == 0, "got value %x, expected %#x\n", value, 0);
-    status = HidP_SetScaledUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RY,
-                                      0, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_BAD_LOG_PHY_VALUES, "HidP_GetScaledUsageValue returned %#x\n", status);
-    ok(value == 0, "got value %x, expected %#x\n", value, 0);
-
-    value = HidP_MaxUsageListLength(HidP_Feature + 1, 0, preparsed_data);
-    ok(value == 0, "HidP_MaxUsageListLength(HidP_Feature + 1, 0) returned %d, expected %d\n", value, 0);
-    value = HidP_MaxUsageListLength(HidP_Input, 0, preparsed_data);
-    ok(value == 50, "HidP_MaxUsageListLength(HidP_Input, 0) returned %d, expected %d\n", value, 50);
-    value = HidP_MaxUsageListLength(HidP_Input, HID_USAGE_PAGE_BUTTON, preparsed_data);
-    ok(value == 32, "HidP_MaxUsageListLength(HidP_Input, HID_USAGE_PAGE_BUTTON) returned %d, expected %d\n", value, 32);
-    value = HidP_MaxUsageListLength(HidP_Input, HID_USAGE_PAGE_LED, preparsed_data);
-    ok(value == 8, "HidP_MaxUsageListLength(HidP_Input, HID_USAGE_PAGE_LED) returned %d, expected %d\n", value, 8);
-    value = HidP_MaxUsageListLength(HidP_Feature, HID_USAGE_PAGE_BUTTON, preparsed_data);
-    ok(value == 8, "HidP_MaxUsageListLength(HidP_Feature, HID_USAGE_PAGE_BUTTON) returned %d, expected %d\n", value, 8);
-    value = HidP_MaxUsageListLength(HidP_Feature, HID_USAGE_PAGE_LED, preparsed_data);
-    ok(value == 0, "HidP_MaxUsageListLength(HidP_Feature, HID_USAGE_PAGE_LED) returned %d, expected %d\n", value, 0);
-
-    usages[0] = 0xff;
-    value = 1;
-    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, usages, &value,
-                            preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_SetUsages returned %#x\n", status);
-    usages[1] = 2;
-    usages[2] = 0xff;
-    value = 3;
-    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, usages, &value,
-                            preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_SetUsages returned %#x\n", status);
-    usages[0] = 4;
-    usages[1] = 6;
-    value = 2;
-    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, usages, &value,
-                            preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsages returned %#x\n", status);
-    usages[0] = 4;
-    usages[1] = 6;
-    value = 2;
-    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_LED, 0, usages, &value, preparsed_data,
-                            report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsages returned %#x\n", status);
-
-    value = ARRAY_SIZE(usages);
-    status = HidP_GetUsages(HidP_Input, HID_USAGE_PAGE_KEYBOARD, 0, usages, &value, preparsed_data,
-                            report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsages returned %#x\n", status);
-    ok(value == 0, "got usage count %d, expected %d\n", value, 2);
-
-    usages[0] = 0x9;
-    usages[1] = 0xb;
-    usages[2] = 0xa;
-    value = 3;
-    ok(report[6] == 0, "got report[6] %x expected 0\n", report[6]);
-    ok(report[7] == 0, "got report[7] %x expected 0\n", report[7]);
-    memcpy(buffer, report, caps.InputReportByteLength);
-    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_KEYBOARD, 0, usages, &value, preparsed_data,
-                            report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_SetUsages returned %#x\n", status);
-    buffer[6] = 2;
-    buffer[7] = 4;
-    ok(!memcmp(buffer, report, caps.InputReportByteLength), "unexpected report data\n");
-
-    status = HidP_SetUsageValue(HidP_Input, HID_USAGE_PAGE_LED, 0, 6, 1,
-                                preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_SetUsageValue returned %#x\n", status);
-
-    value = 0xdeadbeef;
-    status = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_LED, 0, 6, &value,
-                                preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_SetUsageValue returned %#x\n", status);
-    ok(value == 0xdeadbeef, "got value %x, expected %#x\n", value, 0xdeadbeef);
-
-    value = 1;
-    status = HidP_GetUsages(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, usages, &value,
-                            preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetUsages returned %#x\n", status);
-    ok(value == 2, "got usage count %d, expected %d\n", value, 2);
-    value = ARRAY_SIZE(usages);
-    memset(usages, 0xcd, sizeof(usages));
-    status = HidP_GetUsages(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, usages, &value,
-                            preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsages returned %#x\n", status);
-    ok(value == 2, "got usage count %d, expected %d\n", value, 2);
-    ok(usages[0] == 4, "got usages[0] %x, expected %x\n", usages[0], 4);
-    ok(usages[1] == 6, "got usages[1] %x, expected %x\n", usages[1], 6);
-
-    value = ARRAY_SIZE(usages);
-    memset(usages, 0xcd, sizeof(usages));
-    status = HidP_GetUsages(HidP_Input, HID_USAGE_PAGE_LED, 0, usages, &value, preparsed_data,
-                            report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsages returned %#x\n", status);
-    ok(value == 2, "got usage count %d, expected %d\n", value, 2);
-    ok(usages[0] == 6, "got usages[0] %x, expected %x\n", usages[0], 6);
-    ok(usages[1] == 4, "got usages[1] %x, expected %x\n", usages[1], 4);
-
-    value = ARRAY_SIZE(usage_and_pages);
-    memset(usage_and_pages, 0xcd, sizeof(usage_and_pages));
-    status = HidP_GetUsagesEx(HidP_Input, 0, usage_and_pages, &value, preparsed_data, report,
-                              caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsagesEx returned %#x\n", status);
-    ok(value == 6, "got usage count %d, expected %d\n", value, 4);
-    ok(usage_and_pages[0].UsagePage == HID_USAGE_PAGE_BUTTON, "got usage_and_pages[0] UsagePage %x, expected %x\n",
-       usage_and_pages[0].UsagePage, HID_USAGE_PAGE_BUTTON);
-    ok(usage_and_pages[1].UsagePage == HID_USAGE_PAGE_BUTTON, "got usage_and_pages[1] UsagePage %x, expected %x\n",
-       usage_and_pages[1].UsagePage, HID_USAGE_PAGE_BUTTON);
-    ok(usage_and_pages[2].UsagePage == HID_USAGE_PAGE_KEYBOARD, "got usage_and_pages[2] UsagePage %x, expected %x\n",
-       usage_and_pages[2].UsagePage, HID_USAGE_PAGE_KEYBOARD);
-    ok(usage_and_pages[3].UsagePage == HID_USAGE_PAGE_KEYBOARD, "got usage_and_pages[3] UsagePage %x, expected %x\n",
-       usage_and_pages[3].UsagePage, HID_USAGE_PAGE_KEYBOARD);
-    ok(usage_and_pages[4].UsagePage == HID_USAGE_PAGE_LED, "got usage_and_pages[4] UsagePage %x, expected %x\n",
-       usage_and_pages[4].UsagePage, HID_USAGE_PAGE_LED);
-    ok(usage_and_pages[5].UsagePage == HID_USAGE_PAGE_LED, "got usage_and_pages[5] UsagePage %x, expected %x\n",
-       usage_and_pages[5].UsagePage, HID_USAGE_PAGE_LED);
-    ok(usage_and_pages[0].Usage == 4, "got usage_and_pages[0] Usage %x, expected %x\n",
-       usage_and_pages[0].Usage, 4);
-    ok(usage_and_pages[1].Usage == 6, "got usage_and_pages[1] Usage %x, expected %x\n",
-       usage_and_pages[1].Usage, 6);
-    ok(usage_and_pages[2].Usage == 9, "got usage_and_pages[2] Usage %x, expected %x\n",
-       usage_and_pages[2].Usage, 9);
-    ok(usage_and_pages[3].Usage == 11, "got usage_and_pages[3] Usage %x, expected %x\n",
-       usage_and_pages[3].Usage, 11);
-    ok(usage_and_pages[4].Usage == 6, "got usage_and_pages[4] Usage %x, expected %x\n",
-       usage_and_pages[4].Usage, 6);
-    ok(usage_and_pages[5].Usage == 4, "got usage_and_pages[5] Usage %x, expected %x\n",
-       usage_and_pages[5].Usage, 4);
-
-    value = HidP_MaxDataListLength(HidP_Feature + 1, preparsed_data);
-    ok(value == 0, "HidP_MaxDataListLength(HidP_Feature + 1) returned %d, expected %d\n", value, 0);
-    value = HidP_MaxDataListLength(HidP_Input, preparsed_data);
-    ok(value == 58, "HidP_MaxDataListLength(HidP_Input) returned %d, expected %d\n", value, 58);
-    value = HidP_MaxDataListLength(HidP_Output, preparsed_data);
-    ok(value == 0, "HidP_MaxDataListLength(HidP_Output) returned %d, expected %d\n", value, 0);
-    value = HidP_MaxDataListLength(HidP_Feature, preparsed_data);
-    ok(value == 14, "HidP_MaxDataListLength(HidP_Feature) returned %d, expected %d\n", value, 14);
-
-    value = 1;
-    status = HidP_GetData(HidP_Input, data, &value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetData returned %#x\n", status);
-    ok(value == 11, "got data count %d, expected %d\n", value, 11);
-    memset(data, 0, sizeof(data));
-    status = HidP_GetData(HidP_Input, data, &value, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetData returned %#x\n", status);
-    for (i = 0; i < ARRAY_SIZE(expect_data); ++i)
-    {
-        winetest_push_context("data[%d]", i);
-        check_member(data[i], expect_data[i], "%d", DataIndex);
-        check_member(data[i], expect_data[i], "%d", RawValue);
-        winetest_pop_context();
-    }
-
-    /* HID nary usage collections are set with 1-based usage index in their declaration order */
-
-    memset(report, 0, caps.InputReportByteLength);
-    status = HidP_InitializeReportForID(HidP_Input, report_id, preparsed_data, report,
-                                        caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
-    value = 2;
-    usages[0] = 0x8e;
-    usages[1] = 0x8f;
-    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_KEYBOARD, 0, usages, &value, preparsed_data,
-                            report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsages returned %#x\n", status);
-    ok(report[caps.InputReportByteLength - 2] == 3, "unexpected usage index %d, expected 3\n",
-       report[caps.InputReportByteLength - 2]);
-    ok(report[caps.InputReportByteLength - 1] == 4, "unexpected usage index %d, expected 4\n",
-       report[caps.InputReportByteLength - 1]);
-    status = HidP_UnsetUsages(HidP_Input, HID_USAGE_PAGE_KEYBOARD, 0, usages, &value, preparsed_data,
-                              report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_UnsetUsages returned %#x\n", status);
-    ok(report[caps.InputReportByteLength - 2] == 0, "unexpected usage index %d, expected 0\n",
-       report[caps.InputReportByteLength - 2]);
-    ok(report[caps.InputReportByteLength - 1] == 0, "unexpected usage index %d, expected 0\n",
-       report[caps.InputReportByteLength - 1]);
-    status = HidP_UnsetUsages(HidP_Input, HID_USAGE_PAGE_KEYBOARD, 0, usages, &value, preparsed_data,
-                              report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_BUTTON_NOT_PRESSED, "HidP_UnsetUsages returned %#x\n", status);
-    value = 1;
-    usages[0] = 0x8c;
-    status = HidP_SetUsages(HidP_Input, HID_USAGE_PAGE_KEYBOARD, 0, usages, &value, preparsed_data,
-                            report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsages returned %#x\n", status);
-    ok(report[caps.InputReportByteLength - 2] == 1, "unexpected usage index %d, expected 1\n",
-       report[caps.InputReportByteLength - 2]);
-
-    memset(report, 0xcd, sizeof(report));
-    status = HidP_InitializeReportForID(HidP_Feature, 3, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_REPORT_DOES_NOT_EXIST, "HidP_InitializeReportForID returned %#x\n", status);
-
-    memset(report, 0xcd, sizeof(report));
-    status = HidP_InitializeReportForID(HidP_Feature, report_id, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
-
-    memset(buffer, 0xcd, sizeof(buffer));
-    memset(buffer, 0, caps.FeatureReportByteLength);
-    buffer[0] = report_id;
-    ok(!memcmp(buffer, report, sizeof(buffer)), "unexpected report data\n");
-
-    for (i = 0; i < caps.NumberLinkCollectionNodes; ++i)
-    {
-        if (collections[i].LinkUsagePage != HID_USAGE_PAGE_HAPTICS) continue;
-        if (collections[i].LinkUsage == HID_USAGE_HAPTICS_WAVEFORM_LIST) break;
-    }
-    ok(i < caps.NumberLinkCollectionNodes,
-       "HID_USAGE_HAPTICS_WAVEFORM_LIST collection not found\n");
-    waveform_list = i;
-
-    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
-                                HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, (PHIDP_PREPARSED_DATA)buffer,
-                                report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_SetUsageValue returned %#x\n", status);
-    status = HidP_SetUsageValue(HidP_Feature + 1, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
-                                HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, preparsed_data, report,
-                                caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_SetUsageValue returned %#x\n", status);
-    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
-                                HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, preparsed_data, report,
-                                caps.FeatureReportByteLength + 1);
-    ok(status == HIDP_STATUS_INVALID_REPORT_LENGTH, "HidP_SetUsageValue returned %#x\n", status);
-    report[0] = 1 - report_id;
-    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
-                                HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, preparsed_data, report,
-                                caps.FeatureReportByteLength);
-    ok(status == (report_id ? HIDP_STATUS_SUCCESS : HIDP_STATUS_INCOMPATIBLE_REPORT_ID),
-       "HidP_SetUsageValue returned %#x\n", status);
-    report[0] = 2;
-    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
-                                HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, preparsed_data, report,
-                                caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_INCOMPATIBLE_REPORT_ID, "HidP_SetUsageValue returned %#x\n", status);
-    report[0] = report_id;
-    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, 0xdead, 3, HID_USAGE_HAPTICS_WAVEFORM_RUMBLE,
-                                preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_SetUsageValue returned %#x\n", status);
-
-    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
-                                HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, preparsed_data, report,
-                                caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
-
-    memset(buffer, 0xcd, sizeof(buffer));
-    memset(buffer, 0, caps.FeatureReportByteLength);
-    buffer[0] = report_id;
-    value = HID_USAGE_HAPTICS_WAVEFORM_RUMBLE;
-    memcpy(buffer + 1, &value, 2);
-    ok(!memcmp(buffer, report, sizeof(buffer)), "unexpected report data\n");
-
-    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3, &value,
-                                (PHIDP_PREPARSED_DATA)buffer, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_INVALID_PREPARSED_DATA, "HidP_GetUsageValue returned %#x\n", status);
-    status = HidP_GetUsageValue(HidP_Feature + 1, HID_USAGE_PAGE_ORDINAL, waveform_list, 3,
-                                &value, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_INVALID_REPORT_TYPE, "HidP_GetUsageValue returned %#x\n", status);
-    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3, &value,
-                                preparsed_data, report, caps.FeatureReportByteLength + 1);
-    ok(status == HIDP_STATUS_INVALID_REPORT_LENGTH, "HidP_GetUsageValue returned %#x\n", status);
-    report[0] = 1 - report_id;
-    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3, &value,
-                                preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == (report_id ? HIDP_STATUS_SUCCESS : HIDP_STATUS_INCOMPATIBLE_REPORT_ID),
-       "HidP_GetUsageValue returned %#x\n", status);
-    report[0] = 2;
-    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3, &value,
-                                preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_INCOMPATIBLE_REPORT_ID, "HidP_GetUsageValue returned %#x\n", status);
-    report[0] = report_id;
-    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, 0xdead, 3, &value,
-                                preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetUsageValue returned %#x\n", status);
-
-    value = 0xdeadbeef;
-    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_ORDINAL, waveform_list, 3, &value,
-                                preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
-    ok(value == HID_USAGE_HAPTICS_WAVEFORM_RUMBLE, "got value %x, expected %#x\n", value,
-       HID_USAGE_HAPTICS_WAVEFORM_RUMBLE);
-
-    memset(buffer, 0xff, sizeof(buffer));
-    status = HidP_SetUsageValueArray(HidP_Feature, HID_USAGE_PAGE_HAPTICS, 0, HID_USAGE_HAPTICS_WAVEFORM_CUTOFF_TIME, buffer,
-                                     0, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_SetUsageValueArray returned %#x\n", status);
-    status = HidP_SetUsageValueArray(HidP_Feature, HID_USAGE_PAGE_HAPTICS, 0, HID_USAGE_HAPTICS_WAVEFORM_CUTOFF_TIME, buffer,
-                                     64, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValueArray returned %#x\n", status);
-    ok(!memcmp(report + 9, buffer, 8), "unexpected report data\n");
-
-    memset(buffer, 0, sizeof(buffer));
-    status = HidP_GetUsageValueArray(HidP_Feature, HID_USAGE_PAGE_HAPTICS, 0, HID_USAGE_HAPTICS_WAVEFORM_CUTOFF_TIME, buffer,
-                                     0, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_BUFFER_TOO_SMALL, "HidP_GetUsageValueArray returned %#x\n", status);
-    status = HidP_GetUsageValueArray(HidP_Feature, HID_USAGE_PAGE_HAPTICS, 0, HID_USAGE_HAPTICS_WAVEFORM_CUTOFF_TIME, buffer,
-                                     64, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValueArray returned %#x\n", status);
-    memset(buffer + 16, 0xff, 8);
-    ok(!memcmp(buffer, buffer + 16, 16), "unexpected report value\n");
-
-
-    value = 0x7fffffff;
-    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                value, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
-    value = 0xdeadbeef;
-    status = HidP_GetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                      (LONG *)&value, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_VALUE_OUT_OF_RANGE, "HidP_GetScaledUsageValue returned %#x\n", status);
-    ok(value == 0, "got value %x, expected %#x\n", value, 0);
-    value = 0xdeadbeef;
-    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                &value, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
-    ok(value == 0x7fffffff, "got value %x, expected %#x\n", value, 0x7fffffff);
-
-    value = 0x7fff;
-    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                value, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
-    value = 0xdeadbeef;
-    status = HidP_GetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                      (LONG *)&value, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
-    ok(value == 0x0003ffff, "got value %x, expected %#x\n", value, 0x0003ffff);
-
-    value = 0;
-    status = HidP_SetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                value, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetUsageValue returned %#x\n", status);
-    value = 0xdeadbeef;
-    status = HidP_GetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                      (LONG *)&value, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetScaledUsageValue returned %#x\n", status);
-    ok(value == 0xfff90000, "got value %x, expected %#x\n", value, 0xfff90000);
-    status = HidP_SetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                      0x1000, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetScaledUsageValue returned %#x\n", status);
-    value = 0;
-    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                &value, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
-    ok(value == 0xfffff518, "got value %x, expected %#x\n", value, 0xfffff518);
-    status = HidP_SetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                      0, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetScaledUsageValue returned %#x\n", status);
-    value = 0;
-    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                &value, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
-    ok(value == 0xfffff45e, "got value %x, expected %#x\n", value, 0xfffff45e);
-    status = HidP_SetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                      0xdead, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetScaledUsageValue returned %#x\n", status);
-    value = 0;
-    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                &value, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
-    ok(value == 0xfffffe7d, "got value %x, expected %#x\n", value, 0xfffffe7d);
-    status = HidP_SetScaledUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                      0xbeef, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_SetScaledUsageValue returned %#x\n", status);
-    value = 0;
-    status = HidP_GetUsageValue(HidP_Feature, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z,
-                                &value, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
-    ok(value == 0xfffffd0b, "got value %x, expected %#x\n", value, 0xfffffd0b);
-
-
-    memset(report, 0xcd, sizeof(report));
-    status = HidP_InitializeReportForID(HidP_Input, report_id, preparsed_data, report, caps.InputReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_GetInputReport(file, report, 0);
-    ok(!ret, "HidD_GetInputReport succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "HidD_GetInputReport returned error %u\n", GetLastError());
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_GetInputReport(file, report, caps.InputReportByteLength - 1);
-    ok(!ret, "HidD_GetInputReport succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
-       "HidD_GetInputReport returned error %u\n", GetLastError());
-
-    SetLastError(0xdeadbeef);
-    memset(buffer, 0x5a, sizeof(buffer));
-    ret = HidD_GetInputReport(file, buffer, caps.InputReportByteLength);
-    if (report_id || broken(!ret) /* w7u */)
-    {
-        ok(!ret, "HidD_GetInputReport succeeded, last error %u\n", GetLastError());
-        ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
-           "HidD_GetInputReport returned error %u\n", GetLastError());
-    }
-    else
-    {
-        ok(ret, "HidD_GetInputReport failed, last error %u\n", GetLastError());
-        ok(buffer[0] == 0x5a, "got buffer[0] %x, expected 0x5a\n", (BYTE)buffer[0]);
-    }
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_GetInputReport(file, report, caps.InputReportByteLength);
-    ok(ret, "HidD_GetInputReport failed, last error %u\n", GetLastError());
-    ok(report[0] == report_id, "got report[0] %02x, expected %02x\n", report[0], report_id);
-
-    SetLastError(0xdeadbeef);
-    value = caps.InputReportByteLength * 2;
-    ret = sync_ioctl(file, IOCTL_HID_GET_INPUT_REPORT, NULL, 0, report, &value);
-    ok(ret, "IOCTL_HID_GET_INPUT_REPORT failed, last error %u\n", GetLastError());
-    ok(value == 3, "got length %u, expected 3\n", value);
-    ok(report[0] == report_id, "got report[0] %02x, expected %02x\n", report[0], report_id);
-
-
-    memset(report, 0xcd, sizeof(report));
-    status = HidP_InitializeReportForID(HidP_Feature, report_id, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_GetFeature(file, report, 0);
-    ok(!ret, "HidD_GetFeature succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "HidD_GetFeature returned error %u\n", GetLastError());
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_GetFeature(file, report, caps.FeatureReportByteLength - 1);
-    ok(!ret, "HidD_GetFeature succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
-       "HidD_GetFeature returned error %u\n", GetLastError());
-
-    SetLastError(0xdeadbeef);
-    memset(buffer, 0x5a, sizeof(buffer));
-    ret = HidD_GetFeature(file, buffer, caps.FeatureReportByteLength);
-    if (report_id || broken(!ret))
-    {
-        ok(!ret, "HidD_GetFeature succeeded, last error %u\n", GetLastError());
-        ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
-           "HidD_GetFeature returned error %u\n", GetLastError());
-    }
-    else
-    {
-        ok(ret, "HidD_GetFeature failed, last error %u\n", GetLastError());
-        ok(buffer[0] == 0x5a, "got buffer[0] %x, expected 0x5a\n", (BYTE)buffer[0]);
-    }
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_GetFeature(file, report, caps.FeatureReportByteLength);
-    ok(ret, "HidD_GetFeature failed, last error %u\n", GetLastError());
-    ok(report[0] == report_id, "got report[0] %02x, expected %02x\n", report[0], report_id);
-
-    value = caps.FeatureReportByteLength * 2;
-    SetLastError(0xdeadbeef);
-    ret = sync_ioctl(file, IOCTL_HID_GET_FEATURE, NULL, 0, report, &value);
-    ok(ret, "IOCTL_HID_GET_FEATURE failed, last error %u\n", GetLastError());
-    ok(value == 3, "got length %u, expected 3\n", value);
-    ok(report[0] == report_id, "got report[0] %02x, expected %02x\n", report[0], report_id);
-
-
-    memset(report, 0xcd, sizeof(report));
-    status = HidP_InitializeReportForID(HidP_Feature, report_id, preparsed_data, report, caps.FeatureReportByteLength);
-    ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_SetFeature(file, report, 0);
-    ok(!ret, "HidD_SetFeature succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "HidD_SetFeature returned error %u\n", GetLastError());
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_SetFeature(file, report, caps.FeatureReportByteLength - 1);
-    ok(!ret, "HidD_SetFeature succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
-       "HidD_SetFeature returned error %u\n", GetLastError());
-
-    SetLastError(0xdeadbeef);
-    memset(buffer, 0x5a, sizeof(buffer));
-    ret = HidD_SetFeature(file, buffer, caps.FeatureReportByteLength);
-    if (report_id || broken(!ret))
-    {
-        ok(!ret, "HidD_SetFeature succeeded, last error %u\n", GetLastError());
-        ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
-           "HidD_SetFeature returned error %u\n", GetLastError());
-    }
-    else
-    {
-        ok(ret, "HidD_SetFeature failed, last error %u\n", GetLastError());
-    }
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_SetFeature(file, report, caps.FeatureReportByteLength);
-    ok(ret, "HidD_SetFeature failed, last error %u\n", GetLastError());
-
-    value = caps.FeatureReportByteLength * 2;
-    SetLastError(0xdeadbeef);
-    ret = sync_ioctl(file, IOCTL_HID_SET_FEATURE, NULL, 0, report, &value);
-    ok(!ret, "IOCTL_HID_SET_FEATURE succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "IOCTL_HID_SET_FEATURE returned error %u\n", GetLastError());
-    value = 0;
-    SetLastError(0xdeadbeef);
-    ret = sync_ioctl(file, IOCTL_HID_SET_FEATURE, report, caps.FeatureReportByteLength * 2, NULL, &value);
-    ok(ret, "IOCTL_HID_SET_FEATURE failed, last error %u\n", GetLastError());
-    ok(value == 3, "got length %u, expected 3\n", value);
-
-
-    memset(report, 0xcd, sizeof(report));
-    status = HidP_InitializeReportForID(HidP_Output, report_id, preparsed_data, report, caps.OutputReportByteLength);
-    ok(status == HIDP_STATUS_REPORT_DOES_NOT_EXIST, "HidP_InitializeReportForID returned %#x\n", status);
-    memset(report, 0, caps.OutputReportByteLength);
-    report[0] = report_id;
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_SetOutputReport(file, report, 0);
-    ok(!ret, "HidD_SetOutputReport succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "HidD_SetOutputReport returned error %u\n", GetLastError());
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_SetOutputReport(file, report, caps.OutputReportByteLength - 1);
-    ok(!ret, "HidD_SetOutputReport succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
-       "HidD_SetOutputReport returned error %u\n", GetLastError());
-
-    SetLastError(0xdeadbeef);
-    memset(buffer, 0x5a, sizeof(buffer));
-    ret = HidD_SetOutputReport(file, buffer, caps.OutputReportByteLength);
-    if (report_id || broken(!ret))
-    {
-        ok(!ret, "HidD_SetOutputReport succeeded, last error %u\n", GetLastError());
-        ok(GetLastError() == ERROR_INVALID_PARAMETER || broken(GetLastError() == ERROR_CRC),
-           "HidD_SetOutputReport returned error %u\n", GetLastError());
-    }
-    else
-    {
-        ok(ret, "HidD_SetOutputReport failed, last error %u\n", GetLastError());
-    }
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_SetOutputReport(file, report, caps.OutputReportByteLength);
-    ok(ret, "HidD_SetOutputReport failed, last error %u\n", GetLastError());
-
-    value = caps.OutputReportByteLength * 2;
-    SetLastError(0xdeadbeef);
-    ret = sync_ioctl(file, IOCTL_HID_SET_OUTPUT_REPORT, NULL, 0, report, &value);
-    ok(!ret, "IOCTL_HID_SET_OUTPUT_REPORT succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "IOCTL_HID_SET_OUTPUT_REPORT returned error %u\n", GetLastError());
-    value = 0;
-    SetLastError(0xdeadbeef);
-    ret = sync_ioctl(file, IOCTL_HID_SET_OUTPUT_REPORT, report, caps.OutputReportByteLength * 2, NULL, &value);
-    ok(ret, "IOCTL_HID_SET_OUTPUT_REPORT failed, last error %u\n", GetLastError());
-    ok(value == 3, "got length %u, expected 3\n", value);
-
-
-    SetLastError(0xdeadbeef);
-    ret = WriteFile(file, report, 0, &value, NULL);
-    ok(!ret, "WriteFile succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_USER_BUFFER, "WriteFile returned error %u\n", GetLastError());
-    ok(value == 0, "WriteFile returned %x\n", value);
-    SetLastError(0xdeadbeef);
-    ret = WriteFile(file, report, caps.OutputReportByteLength - 1, &value, NULL);
-    ok(!ret, "WriteFile succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_PARAMETER || GetLastError() == ERROR_INVALID_USER_BUFFER,
-       "WriteFile returned error %u\n", GetLastError());
-    ok(value == 0, "WriteFile returned %x\n", value);
-
-    memset(report, 0xcd, sizeof(report));
-    report[0] = 0xa5;
-    SetLastError(0xdeadbeef);
-    ret = WriteFile(file, report, caps.OutputReportByteLength * 2, &value, NULL);
-    if (report_id || broken(!ret) /* w7u */)
-    {
-        ok(!ret, "WriteFile succeeded\n");
-        ok(GetLastError() == ERROR_INVALID_PARAMETER, "WriteFile returned error %u\n", GetLastError());
-        ok(value == 0, "WriteFile wrote %u\n", value);
-        SetLastError(0xdeadbeef);
-        report[0] = report_id;
-        ret = WriteFile(file, report, caps.OutputReportByteLength, &value, NULL);
-    }
-
-    if (report_id)
-    {
-        ok(ret, "WriteFile failed, last error %u\n", GetLastError());
-        ok(value == 2, "WriteFile wrote %u\n", value);
-    }
-    else
-    {
-        ok(ret, "WriteFile failed, last error %u\n", GetLastError());
-        ok(value == 3, "WriteFile wrote %u\n", value);
-    }
-
-
-    memset(report, 0xcd, sizeof(report));
-    SetLastError(0xdeadbeef);
-    ret = ReadFile(file, report, 0, &value, NULL);
-    ok(!ret && GetLastError() == ERROR_INVALID_USER_BUFFER, "ReadFile failed, last error %u\n", GetLastError());
-    ok(value == 0, "ReadFile returned %x\n", value);
-    SetLastError(0xdeadbeef);
-    ret = ReadFile(file, report, caps.InputReportByteLength - 1, &value, NULL);
-    ok(!ret && GetLastError() == ERROR_INVALID_USER_BUFFER, "ReadFile failed, last error %u\n", GetLastError());
-    ok(value == 0, "ReadFile returned %x\n", value);
-
-    if (polled)
-    {
-        memset(report, 0xcd, sizeof(report));
-        SetLastError(0xdeadbeef);
-        ret = ReadFile(file, report, caps.InputReportByteLength, &value, NULL);
-        ok(ret, "ReadFile failed, last error %u\n", GetLastError());
-        ok(value == (report_id ? 3 : 4), "ReadFile returned %x\n", value);
-        ok(report[0] == report_id, "unexpected report data\n");
-
-        overlapped.hEvent = CreateEventA(NULL, FALSE, FALSE, NULL);
-        overlapped2.hEvent = CreateEventA(NULL, FALSE, FALSE, NULL);
-
-        /* drain available input reports */
-        SetLastError(0xdeadbeef);
-        while (ReadFile(async_file, report, caps.InputReportByteLength, NULL, &overlapped))
-            ResetEvent(overlapped.hEvent);
-        ok(GetLastError() == ERROR_IO_PENDING, "ReadFile returned error %u\n", GetLastError());
-        ret = GetOverlappedResult(async_file, &overlapped, &value, TRUE);
-        ok(ret, "GetOverlappedResult failed, last error %u\n", GetLastError());
-        ok(value == (report_id ? 3 : 4), "GetOverlappedResult returned length %u, expected 3\n", value);
-        ResetEvent(overlapped.hEvent);
-
-        memcpy(buffer, report, caps.InputReportByteLength);
-        memcpy(buffer + caps.InputReportByteLength, report, caps.InputReportByteLength);
-
-        SetLastError(0xdeadbeef);
-        ret = ReadFile(async_file, report, caps.InputReportByteLength, NULL, &overlapped);
-        ok(!ret, "ReadFile succeeded\n");
-        ok(GetLastError() == ERROR_IO_PENDING, "ReadFile returned error %u\n", GetLastError());
-
-        SetLastError(0xdeadbeef);
-        ret = ReadFile(async_file, buffer, caps.InputReportByteLength, NULL, &overlapped2);
-        ok(!ret, "ReadFile succeeded\n");
-        ok(GetLastError() == ERROR_IO_PENDING, "ReadFile returned error %u\n", GetLastError());
-
-        /* wait for second report to be ready */
-        ret = GetOverlappedResult(async_file, &overlapped2, &value, TRUE);
-        ok(ret, "GetOverlappedResult failed, last error %u\n", GetLastError());
-        ok(value == (report_id ? 3 : 4), "GetOverlappedResult returned length %u, expected 3\n", value);
-        /* first report should be ready and the same */
-        ret = GetOverlappedResult(async_file, &overlapped, &value, FALSE);
-        ok(ret, "GetOverlappedResult failed, last error %u\n", GetLastError());
-        ok(value == (report_id ? 3 : 4), "GetOverlappedResult returned length %u, expected 3\n", value);
-        ok(memcmp(report, buffer + caps.InputReportByteLength, caps.InputReportByteLength),
-           "expected different report\n");
-        ok(!memcmp(report, buffer, caps.InputReportByteLength), "expected identical reports\n");
-
-        CloseHandle(overlapped.hEvent);
-        CloseHandle(overlapped2.hEvent);
-    }
-
-
-    HidD_FreePreparsedData(preparsed_data);
-}
-
-struct test_hid_params
-{
-    BOOL report_id;
-    BOOL polled;
-};
-
-static void test_hid_device(void *args)
-{
-    struct test_hid_params *params = args;
-    BOOL report_id = params->report_id, polled = params->polled;
-    char buffer[200];
-    SP_DEVICE_INTERFACE_DETAIL_DATA_A *iface_detail = (void *)buffer;
-    SP_DEVICE_INTERFACE_DATA iface = {sizeof(iface)};
-    SP_DEVINFO_DATA device = {sizeof(device)};
-    ULONG count, poll_freq, out_len;
-    HANDLE file, async_file;
-    BOOL ret, found = FALSE;
-    OBJECT_ATTRIBUTES attr;
-    UNICODE_STRING string;
-    IO_STATUS_BLOCK io;
-    NTSTATUS status;
-    unsigned int i;
-    HDEVINFO set;
-
-    winetest_push_context("id %d%s", report_id, polled ? " poll" : "");
-
-    set = SetupDiGetClassDevsA(&GUID_DEVINTERFACE_HID, NULL, NULL, DIGCF_DEVICEINTERFACE | DIGCF_PRESENT);
-    ok(set != INVALID_HANDLE_VALUE, "failed to get device list, error %#x\n", GetLastError());
-
-    for (i = 0; SetupDiEnumDeviceInfo(set, i, &device); ++i)
-    {
-        ret = SetupDiEnumDeviceInterfaces(set, &device, &GUID_DEVINTERFACE_HID, 0, &iface);
-        ok(ret, "failed to get interface, error %#x\n", GetLastError());
-        ok(IsEqualGUID(&iface.InterfaceClassGuid, &GUID_DEVINTERFACE_HID),
-                "wrong class %s\n", debugstr_guid(&iface.InterfaceClassGuid));
-        ok(iface.Flags == SPINT_ACTIVE, "got flags %#x\n", iface.Flags);
-
-        iface_detail->cbSize = sizeof(*iface_detail);
-        ret = SetupDiGetDeviceInterfaceDetailA(set, &iface, iface_detail, sizeof(buffer), NULL, NULL);
-        ok(ret, "failed to get interface path, error %#x\n", GetLastError());
-
-        if (strstr(iface_detail->DevicePath, "\\\\?\\hid#winetest#1"))
-        {
-            found = TRUE;
-            break;
-        }
-    }
-
-    SetupDiDestroyDeviceInfoList(set);
-
-    todo_wine ok(found, "didn't find device\n");
-
-    file = CreateFileA(iface_detail->DevicePath, FILE_READ_ACCESS | FILE_WRITE_ACCESS,
-            FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);
-    ok(file != INVALID_HANDLE_VALUE, "got error %u\n", GetLastError());
-
-    count = 0xdeadbeef;
-    SetLastError(0xdeadbeef);
-    ret = HidD_GetNumInputBuffers(file, &count);
-    ok(ret, "HidD_GetNumInputBuffers failed last error %u\n", GetLastError());
-    ok(count == 32, "HidD_GetNumInputBuffers returned %u\n", count);
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_SetNumInputBuffers(file, 1);
-    ok(!ret, "HidD_SetNumInputBuffers succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_PARAMETER, "HidD_SetNumInputBuffers returned error %u\n", GetLastError());
-    SetLastError(0xdeadbeef);
-    ret = HidD_SetNumInputBuffers(file, 513);
-    ok(!ret, "HidD_SetNumInputBuffers succeeded\n");
-    ok(GetLastError() == ERROR_INVALID_PARAMETER, "HidD_SetNumInputBuffers returned error %u\n", GetLastError());
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_SetNumInputBuffers(file, 16);
-    ok(ret, "HidD_SetNumInputBuffers failed last error %u\n", GetLastError());
-
-    count = 0xdeadbeef;
-    SetLastError(0xdeadbeef);
-    ret = HidD_GetNumInputBuffers(file, &count);
-    ok(ret, "HidD_GetNumInputBuffers failed last error %u\n", GetLastError());
-    ok(count == 16, "HidD_GetNumInputBuffers returned %u\n", count);
-
-    async_file = CreateFileA(iface_detail->DevicePath, FILE_READ_ACCESS | FILE_WRITE_ACCESS,
-            FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING,
-            FILE_FLAG_OVERLAPPED | FILE_FLAG_NO_BUFFERING, NULL);
-    ok(async_file != INVALID_HANDLE_VALUE, "got error %u\n", GetLastError());
-
-    count = 0xdeadbeef;
-    SetLastError(0xdeadbeef);
-    ret = HidD_GetNumInputBuffers(async_file, &count);
-    ok(ret, "HidD_GetNumInputBuffers failed last error %u\n", GetLastError());
-    ok(count == 32, "HidD_GetNumInputBuffers returned %u\n", count);
-
-    SetLastError(0xdeadbeef);
-    ret = HidD_SetNumInputBuffers(async_file, 2);
-    ok(ret, "HidD_SetNumInputBuffers failed last error %u\n", GetLastError());
-
-    count = 0xdeadbeef;
-    SetLastError(0xdeadbeef);
-    ret = HidD_GetNumInputBuffers(async_file, &count);
-    ok(ret, "HidD_GetNumInputBuffers failed last error %u\n", GetLastError());
-    ok(count == 2, "HidD_GetNumInputBuffers returned %u\n", count);
-    count = 0xdeadbeef;
-    SetLastError(0xdeadbeef);
-    ret = HidD_GetNumInputBuffers(file, &count);
-    ok(ret, "HidD_GetNumInputBuffers failed last error %u\n", GetLastError());
-    ok(count == 16, "HidD_GetNumInputBuffers returned %u\n", count);
-
-    if (polled)
-    {
-        out_len = sizeof(ULONG);
-        SetLastError(0xdeadbeef);
-        ret = sync_ioctl(file, IOCTL_HID_GET_POLL_FREQUENCY_MSEC, NULL, 0, &poll_freq, &out_len);
-        ok(ret, "IOCTL_HID_GET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
-        ok(out_len == sizeof(ULONG), "got out_len %u, expected sizeof(ULONG)\n", out_len);
-        todo_wine ok(poll_freq == 5, "got poll_freq %u, expected 5\n", poll_freq);
-
-        out_len = 0;
-        poll_freq = 500;
-        SetLastError(0xdeadbeef);
-        ret = sync_ioctl(file, IOCTL_HID_SET_POLL_FREQUENCY_MSEC, &poll_freq, sizeof(ULONG), NULL, &out_len);
-        ok(ret, "IOCTL_HID_SET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
-        ok(out_len == 0, "got out_len %u, expected 0\n", out_len);
-
-        out_len = 0;
-        poll_freq = 10001;
-        SetLastError(0xdeadbeef);
-        ret = sync_ioctl(file, IOCTL_HID_SET_POLL_FREQUENCY_MSEC, &poll_freq, sizeof(ULONG), NULL, &out_len);
-        ok(ret, "IOCTL_HID_SET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
-        ok(out_len == 0, "got out_len %u, expected 0\n", out_len);
-
-        out_len = 0;
-        poll_freq = 0;
-        SetLastError(0xdeadbeef);
-        ret = sync_ioctl(file, IOCTL_HID_SET_POLL_FREQUENCY_MSEC, &poll_freq, sizeof(ULONG), NULL, &out_len);
-        ok(ret, "IOCTL_HID_SET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
-        ok(out_len == 0, "got out_len %u, expected 0\n", out_len);
-
-        out_len = sizeof(ULONG);
-        SetLastError(0xdeadbeef);
-        ret = sync_ioctl(file, IOCTL_HID_GET_POLL_FREQUENCY_MSEC, NULL, 0, &poll_freq, &out_len);
-        ok(ret, "IOCTL_HID_GET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
-        ok(out_len == sizeof(ULONG), "got out_len %u, expected sizeof(ULONG)\n", out_len);
-        ok(poll_freq == 10000, "got poll_freq %u, expected 10000\n", poll_freq);
-
-        out_len = 0;
-        poll_freq = 500;
-        SetLastError(0xdeadbeef);
-        ret = sync_ioctl(file, IOCTL_HID_SET_POLL_FREQUENCY_MSEC, &poll_freq, sizeof(ULONG), NULL, &out_len);
-        ok(ret, "IOCTL_HID_SET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
-        ok(out_len == 0, "got out_len %u, expected 0\n", out_len);
-
-        out_len = sizeof(ULONG);
-        SetLastError(0xdeadbeef);
-        ret = sync_ioctl(async_file, IOCTL_HID_GET_POLL_FREQUENCY_MSEC, NULL, 0, &poll_freq, &out_len);
-        ok(ret, "IOCTL_HID_GET_POLL_FREQUENCY_MSEC failed last error %u\n", GetLastError());
-        ok(out_len == sizeof(ULONG), "got out_len %u, expected sizeof(ULONG)\n", out_len);
-        ok(poll_freq == 500, "got poll_freq %u, expected 500\n", poll_freq);
-    }
-
-    test_hidp(file, async_file, report_id, polled);
-
-    CloseHandle(async_file);
-    CloseHandle(file);
-
-    RtlInitUnicodeString(&string, L"\\??\\root#winetest#0#{deadbeef-29ef-4538-a5fd-b69573a362c0}");
-    InitializeObjectAttributes(&attr, &string, OBJ_CASE_INSENSITIVE, NULL, NULL);
-    status = NtOpenFile(&file, SYNCHRONIZE, &attr, &io, 0, FILE_SYNCHRONOUS_IO_NONALERT);
-    todo_wine ok(status == STATUS_UNSUCCESSFUL, "got %#x\n", status);
-
-    winetest_pop_context();
-}
-
-static void test_hid_driver(struct testsign_context *ctx, DWORD report_id, DWORD polled)
-{
-    struct test_hid_params params = {.report_id = report_id, .polled = polled};
-    SP_DEVINFO_DATA device = {sizeof(device)};
-    char cwd[MAX_PATH], tempdir[MAX_PATH];
-    LSTATUS status;
-    HKEY hkey;
-
-    GetCurrentDirectoryA(ARRAY_SIZE(cwd), cwd);
-    GetTempPathA(ARRAY_SIZE(tempdir), tempdir);
-    SetCurrentDirectoryA(tempdir);
-
-    status = RegCreateKeyExW(HKEY_LOCAL_MACHINE, L"System\\CurrentControlSet\\Services\\winetest", 0, NULL, REG_OPTION_VOLATILE, KEY_ALL_ACCESS, NULL, &hkey, NULL);
-    ok(!status, "RegCreateKeyExW returned %#x\n", status);
-
-    status = RegSetValueExW(hkey, L"ReportID", 0, REG_DWORD, (void *)&report_id, sizeof(report_id));
-    ok(!status, "RegSetValueExW returned %#x\n", status);
-
-    status = RegSetValueExW(hkey, L"PolledMode", 0, REG_DWORD, (void *)&polled, sizeof(polled));
-    ok(!status, "RegSetValueExW returned %#x\n", status);
-
-    run_driver_test(ctx, L"driver_hid.dll", test_hid_device, &params);
-}
-
 START_TEST(ntoskrnl)
 {
     WCHAR filename[MAX_PATH], filename2[MAX_PATH];
@@ -2909,12 +1407,6 @@ START_TEST(ntoskrnl)
     subtest("driver_pnp");
     run_driver_test(&ctx, L"driver_pnp.dll", test_pnp_devices, NULL);
 
-    subtest("driver_hid");
-    test_hid_driver(&ctx, 0, FALSE);
-    test_hid_driver(&ctx, 1, FALSE);
-    test_hid_driver(&ctx, 0, TRUE);
-    test_hid_driver(&ctx, 1, TRUE);
-
 out:
     driver_test_cleanup(&ctx);
 }
-- 
2.33.0




More information about the wine-devel mailing list