Sweden-Number/dlls/xinput1_3/tests/xinput.c

863 lines
37 KiB
C

/*
* The Wine project - Xinput Joystick Library
* Copyright 2008 Andrew Fenn
*
* 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 <windows.h>
#include <stdio.h>
#include "winioctl.h"
#include "xinput.h"
#include "shlwapi.h"
#include "setupapi.h"
#include "ddk/hidsdi.h"
#include "ddk/hidclass.h"
#include "wine/test.h"
static DWORD (WINAPI *pXInputGetState)(DWORD, XINPUT_STATE*);
static DWORD (WINAPI *pXInputGetStateEx)(DWORD, XINPUT_STATE*);
static DWORD (WINAPI *pXInputGetCapabilities)(DWORD,DWORD,XINPUT_CAPABILITIES*);
static DWORD (WINAPI *pXInputSetState)(DWORD, XINPUT_VIBRATION*);
static void (WINAPI *pXInputEnable)(BOOL);
static DWORD (WINAPI *pXInputGetKeystroke)(DWORD, DWORD, PXINPUT_KEYSTROKE);
static DWORD (WINAPI *pXInputGetDSoundAudioDeviceGuids)(DWORD, GUID*, GUID*);
static DWORD (WINAPI *pXInputGetBatteryInformation)(DWORD, BYTE, XINPUT_BATTERY_INFORMATION*);
static void dump_gamepad(XINPUT_GAMEPAD *data)
{
trace("-- Gamepad Variables --\n");
trace("Gamepad.wButtons: %#x\n", data->wButtons);
trace("Gamepad.bLeftTrigger: %d\n", data->bLeftTrigger);
trace("Gamepad.bRightTrigger: %d\n", data->bRightTrigger);
trace("Gamepad.sThumbLX: %d\n", data->sThumbLX);
trace("Gamepad.sThumbLY: %d\n", data->sThumbLY);
trace("Gamepad.sThumbRX: %d\n", data->sThumbRX);
trace("Gamepad.sThumbRY: %d\n\n", data->sThumbRY);
}
static void test_set_state(void)
{
XINPUT_VIBRATION vibrator;
DWORD controllerNum;
DWORD result;
for(controllerNum = 0; controllerNum < XUSER_MAX_COUNT; controllerNum++)
{
ZeroMemory(&vibrator, sizeof(XINPUT_VIBRATION));
vibrator.wLeftMotorSpeed = 32767;
vibrator.wRightMotorSpeed = 32767;
result = pXInputSetState(controllerNum, &vibrator);
if (result == ERROR_DEVICE_NOT_CONNECTED) continue;
Sleep(250);
vibrator.wLeftMotorSpeed = 0;
vibrator.wRightMotorSpeed = 0;
result = pXInputSetState(controllerNum, &vibrator);
ok(result == ERROR_SUCCESS, "XInputSetState failed with (%d)\n", result);
/* Disabling XInput here, queueing a vibration and then re-enabling XInput
* is used to prove that vibrations are auto enabled when resuming XInput.
* If XInputEnable(1) is removed below the vibration will never play. */
if (pXInputEnable) pXInputEnable(0);
Sleep(250);
vibrator.wLeftMotorSpeed = 65535;
vibrator.wRightMotorSpeed = 65535;
result = pXInputSetState(controllerNum, &vibrator);
ok(result == ERROR_SUCCESS, "XInputSetState failed with (%d)\n", result);
if (pXInputEnable) pXInputEnable(1);
Sleep(250);
vibrator.wLeftMotorSpeed = 0;
vibrator.wRightMotorSpeed = 0;
result = pXInputSetState(controllerNum, &vibrator);
ok(result == ERROR_SUCCESS, "XInputSetState failed with (%d)\n", result);
}
result = pXInputSetState(XUSER_MAX_COUNT+1, &vibrator);
ok(result == ERROR_BAD_ARGUMENTS, "XInputSetState returned (%d)\n", result);
}
static void test_get_state(void)
{
XINPUT_STATE state;
DWORD controllerNum, i, result, good = XUSER_MAX_COUNT;
for (i = 0; i < (pXInputGetStateEx ? 2 : 1); i++)
{
for (controllerNum = 0; controllerNum < XUSER_MAX_COUNT; controllerNum++)
{
ZeroMemory(&state, sizeof(state));
if (i == 0)
result = pXInputGetState(controllerNum, &state);
else
result = pXInputGetStateEx(controllerNum, &state);
ok(result == ERROR_SUCCESS || result == ERROR_DEVICE_NOT_CONNECTED,
"%s failed with (%d)\n", i == 0 ? "XInputGetState" : "XInputGetStateEx", result);
if (ERROR_DEVICE_NOT_CONNECTED == result)
{
skip("Controller %d is not connected\n", controllerNum);
continue;
}
trace("-- Results for controller %d --\n", controllerNum);
if (i == 0)
{
good = controllerNum;
trace("XInputGetState: %d\n", result);
}
else
trace("XInputGetStateEx: %d\n", result);
trace("State->dwPacketNumber: %d\n", state.dwPacketNumber);
dump_gamepad(&state.Gamepad);
}
}
result = pXInputGetState(0, NULL);
ok(result == ERROR_BAD_ARGUMENTS, "XInputGetState returned (%d)\n", result);
result = pXInputGetState(XUSER_MAX_COUNT, &state);
ok(result == ERROR_BAD_ARGUMENTS, "XInputGetState returned (%d)\n", result);
result = pXInputGetState(XUSER_MAX_COUNT+1, &state);
ok(result == ERROR_BAD_ARGUMENTS, "XInputGetState returned (%d)\n", result);
if (pXInputGetStateEx)
{
result = pXInputGetStateEx(XUSER_MAX_COUNT, &state);
ok(result == ERROR_BAD_ARGUMENTS, "XInputGetState returned (%d)\n", result);
result = pXInputGetStateEx(XUSER_MAX_COUNT+1, &state);
ok(result == ERROR_BAD_ARGUMENTS, "XInputGetState returned (%d)\n", result);
}
if (winetest_interactive && good < XUSER_MAX_COUNT)
{
DWORD now = GetTickCount(), packet = 0;
XINPUT_GAMEPAD *game = &state.Gamepad;
trace("You have 20 seconds to test the joystick freely\n");
do
{
Sleep(100);
pXInputGetState(good, &state);
if (state.dwPacketNumber == packet)
continue;
packet = state.dwPacketNumber;
trace("Buttons 0x%04X Triggers %3d/%3d LT %6d/%6d RT %6d/%6d\n",
game->wButtons, game->bLeftTrigger, game->bRightTrigger,
game->sThumbLX, game->sThumbLY, game->sThumbRX, game->sThumbRY);
}
while(GetTickCount() - now < 20000);
trace("Test over...\n");
}
}
static void test_get_keystroke(void)
{
XINPUT_KEYSTROKE keystroke;
DWORD controllerNum;
DWORD result;
for(controllerNum = 0; controllerNum < XUSER_MAX_COUNT; controllerNum++)
{
ZeroMemory(&keystroke, sizeof(XINPUT_KEYSTROKE));
result = pXInputGetKeystroke(controllerNum, XINPUT_FLAG_GAMEPAD, &keystroke);
ok(result == ERROR_EMPTY || result == ERROR_SUCCESS || result == ERROR_DEVICE_NOT_CONNECTED,
"XInputGetKeystroke failed with (%d)\n", result);
if (ERROR_DEVICE_NOT_CONNECTED == result)
{
skip("Controller %d is not connected\n", controllerNum);
}
}
ZeroMemory(&keystroke, sizeof(XINPUT_KEYSTROKE));
result = pXInputGetKeystroke(XUSER_MAX_COUNT+1, XINPUT_FLAG_GAMEPAD, &keystroke);
ok(result == ERROR_BAD_ARGUMENTS, "XInputGetKeystroke returned (%d)\n", result);
}
static void test_get_capabilities(void)
{
XINPUT_CAPABILITIES capabilities;
DWORD controllerNum;
DWORD result;
for(controllerNum = 0; controllerNum < XUSER_MAX_COUNT; controllerNum++)
{
ZeroMemory(&capabilities, sizeof(XINPUT_CAPABILITIES));
result = pXInputGetCapabilities(controllerNum, XINPUT_FLAG_GAMEPAD, &capabilities);
ok(result == ERROR_SUCCESS || result == ERROR_DEVICE_NOT_CONNECTED, "XInputGetCapabilities failed with (%d)\n", result);
if (ERROR_DEVICE_NOT_CONNECTED == result)
{
skip("Controller %d is not connected\n", controllerNum);
continue;
}
/* Important to show that the results changed between 1.3 and 1.4 XInput version */
dump_gamepad(&capabilities.Gamepad);
}
ZeroMemory(&capabilities, sizeof(XINPUT_CAPABILITIES));
result = pXInputGetCapabilities(XUSER_MAX_COUNT+1, XINPUT_FLAG_GAMEPAD, &capabilities);
ok(result == ERROR_BAD_ARGUMENTS, "XInputGetCapabilities returned (%d)\n", result);
}
static void test_get_dsoundaudiodevice(void)
{
DWORD controllerNum;
DWORD result;
GUID soundRender, soundCapture;
GUID testGuid = {0xFFFFFFFF, 0xFFFF, 0xFFFF, {0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF}};
GUID emptyGuid = {0x0, 0x0, 0x0, {0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0}};
for(controllerNum = 0; controllerNum < XUSER_MAX_COUNT; controllerNum++)
{
soundRender = soundCapture = testGuid;
result = pXInputGetDSoundAudioDeviceGuids(controllerNum, &soundRender, &soundCapture);
ok(result == ERROR_SUCCESS || result == ERROR_DEVICE_NOT_CONNECTED, "XInputGetDSoundAudioDeviceGuids failed with (%d)\n", result);
if (ERROR_DEVICE_NOT_CONNECTED == result)
{
skip("Controller %d is not connected\n", controllerNum);
continue;
}
if (!IsEqualGUID(&soundRender, &emptyGuid))
ok(!IsEqualGUID(&soundRender, &testGuid), "Broken GUID returned for sound render device\n");
else
trace("Headset phone not attached\n");
if (!IsEqualGUID(&soundCapture, &emptyGuid))
ok(!IsEqualGUID(&soundCapture, &testGuid), "Broken GUID returned for sound capture device\n");
else
trace("Headset microphone not attached\n");
}
result = pXInputGetDSoundAudioDeviceGuids(XUSER_MAX_COUNT+1, &soundRender, &soundCapture);
ok(result == ERROR_BAD_ARGUMENTS, "XInputGetDSoundAudioDeviceGuids returned (%d)\n", result);
}
static void test_get_batteryinformation(void)
{
DWORD controllerNum;
DWORD result;
XINPUT_BATTERY_INFORMATION batteryInfo;
for(controllerNum = 0; controllerNum < XUSER_MAX_COUNT; controllerNum++)
{
ZeroMemory(&batteryInfo, sizeof(XINPUT_BATTERY_INFORMATION));
result = pXInputGetBatteryInformation(controllerNum, BATTERY_DEVTYPE_GAMEPAD, &batteryInfo);
ok(result == ERROR_SUCCESS || result == ERROR_DEVICE_NOT_CONNECTED, "XInputGetBatteryInformation failed with (%d)\n", result);
if (ERROR_DEVICE_NOT_CONNECTED == result)
{
ok(batteryInfo.BatteryLevel == BATTERY_TYPE_DISCONNECTED, "Failed to report device as being disconnected.\n");
skip("Controller %d is not connected\n", controllerNum);
}
}
result = pXInputGetBatteryInformation(XUSER_MAX_COUNT+1, BATTERY_DEVTYPE_GAMEPAD, &batteryInfo);
ok(result == ERROR_BAD_ARGUMENTS, "XInputGetBatteryInformation returned (%d)\n", result);
}
#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)
static void check_hid_caps(DWORD index, HANDLE device, PHIDP_PREPARSED_DATA preparsed,
HIDD_ATTRIBUTES *attrs, HIDP_CAPS *hid_caps)
{
const HIDP_CAPS expect_hid_caps =
{
.Usage = HID_USAGE_GENERIC_GAMEPAD,
.UsagePage = HID_USAGE_PAGE_GENERIC,
.InputReportByteLength =
attrs->VendorID == 0x045e && attrs->ProductID == 0x02ff ? 16 :
15,
.OutputReportByteLength = 0,
.FeatureReportByteLength = 0,
.NumberLinkCollectionNodes = 4,
.NumberInputButtonCaps = 1,
.NumberInputValueCaps = 6,
.NumberInputDataIndices =
attrs->VendorID == 0x045e && attrs->ProductID == 0x02ff ? 22 :
16,
.NumberFeatureButtonCaps = 0,
.NumberFeatureValueCaps = 0,
.NumberFeatureDataIndices = 0,
};
const HIDP_BUTTON_CAPS expect_button_caps[] =
{
{
.UsagePage = HID_USAGE_PAGE_BUTTON,
.BitField = 2,
.LinkUsage = HID_USAGE_GENERIC_GAMEPAD,
.LinkUsagePage = HID_USAGE_PAGE_GENERIC,
.IsRange = TRUE,
.IsAbsolute = TRUE,
.Range.UsageMin = 0x01,
.Range.UsageMax =
attrs->VendorID == 0x045e && attrs->ProductID == 0x02ff ? 0x10 :
0x0a,
.Range.DataIndexMin = 5,
.Range.DataIndexMax =
attrs->VendorID == 0x045e && attrs->ProductID == 0x02ff ? 20 :
14,
},
};
const HIDP_VALUE_CAPS expect_value_caps[] =
{
{
.UsagePage = HID_USAGE_PAGE_GENERIC,
.BitField = 2,
.LinkUsagePage = HID_USAGE_PAGE_GENERIC,
.LinkCollection = 1,
.IsAbsolute = TRUE,
.BitSize = 16,
.ReportCount = 1,
.LogicalMax = -1,
.PhysicalMax = -1,
.NotRange.Usage = HID_USAGE_GENERIC_Y,
.NotRange.DataIndex = 0,
},
{
.UsagePage = HID_USAGE_PAGE_GENERIC,
.BitField = 2,
.LinkUsagePage = HID_USAGE_PAGE_GENERIC,
.LinkCollection = 1,
.IsAbsolute = TRUE,
.BitSize = 16,
.ReportCount = 1,
.LogicalMax = -1,
.PhysicalMax = -1,
.NotRange.Usage = HID_USAGE_GENERIC_X,
.NotRange.DataIndex = 1,
},
{
.UsagePage = HID_USAGE_PAGE_GENERIC,
.BitField = 2,
.LinkUsagePage = HID_USAGE_PAGE_GENERIC,
.LinkCollection = 2,
.IsAbsolute = TRUE,
.BitSize = 16,
.ReportCount = 1,
.LogicalMax = -1,
.PhysicalMax = -1,
.NotRange.Usage = HID_USAGE_GENERIC_RY,
.NotRange.DataIndex = 2,
},
{
.UsagePage = HID_USAGE_PAGE_GENERIC,
.BitField = 2,
.LinkUsagePage = HID_USAGE_PAGE_GENERIC,
.LinkCollection = 2,
.IsAbsolute = TRUE,
.BitSize = 16,
.ReportCount = 1,
.LogicalMax = -1,
.PhysicalMax = -1,
.NotRange.Usage = HID_USAGE_GENERIC_RX,
.NotRange.DataIndex = 3,
},
{
.UsagePage = HID_USAGE_PAGE_GENERIC,
.BitField = 2,
.LinkUsagePage = HID_USAGE_PAGE_GENERIC,
.LinkCollection = 3,
.IsAbsolute = TRUE,
.BitSize = 16,
.ReportCount = 1,
.LogicalMax = -1,
.PhysicalMax = -1,
.NotRange.Usage = HID_USAGE_GENERIC_Z,
.NotRange.DataIndex = 4,
},
{
.UsagePage = HID_USAGE_PAGE_GENERIC,
.BitField = 66,
.LinkUsagePage = HID_USAGE_PAGE_GENERIC,
.LinkUsage = HID_USAGE_GENERIC_GAMEPAD,
.IsAbsolute = TRUE,
.HasNull = TRUE,
.BitSize = 4,
.Units = 14,
.ReportCount = 1,
.LogicalMin = 1,
.LogicalMax = 8,
.PhysicalMin = 0x0000,
.PhysicalMax = 0x103b,
.NotRange.Usage = HID_USAGE_GENERIC_HATSWITCH,
.NotRange.DataIndex =
attrs->VendorID == 0x045e && attrs->ProductID == 0x02ff ? 21 :
15,
},
};
static const HIDP_LINK_COLLECTION_NODE expect_collections[] =
{
{
.LinkUsage = HID_USAGE_GENERIC_GAMEPAD,
.LinkUsagePage = HID_USAGE_PAGE_GENERIC,
.CollectionType = 1,
.NumberOfChildren = 3,
.FirstChild = 3,
},
{ .LinkUsagePage = HID_USAGE_PAGE_GENERIC, .NextSibling = 0, },
{ .LinkUsagePage = HID_USAGE_PAGE_GENERIC, .NextSibling = 1, },
{ .LinkUsagePage = HID_USAGE_PAGE_GENERIC, .NextSibling = 2, },
};
HIDP_LINK_COLLECTION_NODE collections[16];
HIDP_BUTTON_CAPS button_caps[16];
HIDP_VALUE_CAPS value_caps[16];
XINPUT_STATE last_state, state;
XINPUT_CAPABILITIES xi_caps;
char buffer[200] = {0};
ULONG length, value;
USAGE usages[15];
NTSTATUS status;
USHORT count;
DWORD i, res;
BOOL ret;
res = pXInputGetCapabilities(index, 0, &xi_caps);
ok(res == ERROR_SUCCESS, "XInputGetCapabilities %d returned %u\n", index, res);
res = pXInputGetState(index, &state);
ok(res == ERROR_SUCCESS, "XInputGetState %d returned %u\n", index, res);
ok(hid_caps->UsagePage == HID_USAGE_PAGE_GENERIC, "unexpected usage page %04x\n", hid_caps->UsagePage);
ok(hid_caps->Usage == HID_USAGE_GENERIC_GAMEPAD, "unexpected usage %04x\n", hid_caps->Usage);
check_member(*hid_caps, expect_hid_caps, "%04x", Usage);
check_member(*hid_caps, expect_hid_caps, "%04x", UsagePage);
check_member(*hid_caps, expect_hid_caps, "%d", InputReportByteLength);
check_member(*hid_caps, expect_hid_caps, "%d", OutputReportByteLength);
check_member(*hid_caps, expect_hid_caps, "%d", FeatureReportByteLength);
check_member(*hid_caps, expect_hid_caps, "%d", NumberLinkCollectionNodes);
check_member(*hid_caps, expect_hid_caps, "%d", NumberInputButtonCaps);
check_member(*hid_caps, expect_hid_caps, "%d", NumberInputValueCaps);
check_member(*hid_caps, expect_hid_caps, "%d", NumberInputDataIndices);
check_member(*hid_caps, expect_hid_caps, "%d", NumberOutputButtonCaps);
check_member(*hid_caps, expect_hid_caps, "%d", NumberOutputValueCaps);
check_member(*hid_caps, expect_hid_caps, "%d", NumberOutputDataIndices);
check_member(*hid_caps, expect_hid_caps, "%d", NumberFeatureButtonCaps);
check_member(*hid_caps, expect_hid_caps, "%d", NumberFeatureValueCaps);
check_member(*hid_caps, expect_hid_caps, "%d", NumberFeatureDataIndices);
length = hid_caps->NumberLinkCollectionNodes;
status = HidP_GetLinkCollectionNodes(collections, &length, preparsed);
ok(status == HIDP_STATUS_SUCCESS, "HidP_GetLinkCollectionNodes returned %#x\n", status);
ok(length == ARRAY_SIZE(expect_collections), "got %d collections\n", length);
for (i = 0; i < min(length, ARRAY_SIZE(expect_collections)); ++i)
{
winetest_push_context("collections[%d]", i);
check_member(collections[i], expect_collections[i], "%04x", LinkUsage);
check_member(collections[i], expect_collections[i], "%04x", LinkUsagePage);
check_member(collections[i], expect_collections[i], "%d", Parent);
check_member(collections[i], expect_collections[i], "%d", NumberOfChildren);
check_member(collections[i], expect_collections[i], "%d", NextSibling);
check_member(collections[i], expect_collections[i], "%d", FirstChild);
check_member(collections[i], expect_collections[i], "%d", CollectionType);
check_member(collections[i], expect_collections[i], "%d", IsAlias);
winetest_pop_context();
}
count = hid_caps->NumberInputButtonCaps;
status = HidP_GetButtonCaps(HidP_Input, button_caps, &count, preparsed);
ok(status == HIDP_STATUS_SUCCESS, "HidP_GetButtonCaps returned %#x\n", status);
ok(count == ARRAY_SIZE(expect_button_caps), "got %d button caps\n", count);
for (i = 0; i < ARRAY_SIZE(expect_button_caps); ++i)
{
winetest_push_context("button_caps[%d]", i);
check_member(button_caps[i], expect_button_caps[i], "%04x", UsagePage);
check_member(button_caps[i], expect_button_caps[i], "%d", ReportID);
check_member(button_caps[i], expect_button_caps[i], "%d", IsAlias);
check_member(button_caps[i], expect_button_caps[i], "%d", BitField);
check_member(button_caps[i], expect_button_caps[i], "%d", LinkCollection);
check_member(button_caps[i], expect_button_caps[i], "%04x", LinkUsage);
check_member(button_caps[i], expect_button_caps[i], "%04x", LinkUsagePage);
check_member(button_caps[i], expect_button_caps[i], "%d", IsRange);
check_member(button_caps[i], expect_button_caps[i], "%d", IsStringRange);
check_member(button_caps[i], expect_button_caps[i], "%d", IsDesignatorRange);
check_member(button_caps[i], expect_button_caps[i], "%d", IsAbsolute);
if (!button_caps[i].IsRange && !expect_button_caps[i].IsRange)
{
check_member(button_caps[i], expect_button_caps[i], "%04x", NotRange.Usage);
check_member(button_caps[i], expect_button_caps[i], "%d", NotRange.DataIndex);
}
else if (button_caps[i].IsRange && expect_button_caps[i].IsRange)
{
check_member(button_caps[i], expect_button_caps[i], "%04x", Range.UsageMin);
check_member(button_caps[i], expect_button_caps[i], "%04x", Range.UsageMax);
check_member(button_caps[i], expect_button_caps[i], "%d", Range.DataIndexMin);
check_member(button_caps[i], expect_button_caps[i], "%d", Range.DataIndexMax);
}
if (!button_caps[i].IsRange && !expect_button_caps[i].IsRange)
check_member(button_caps[i], expect_button_caps[i], "%d", NotRange.StringIndex);
else if (button_caps[i].IsStringRange && expect_button_caps[i].IsStringRange)
{
check_member(button_caps[i], expect_button_caps[i], "%d", Range.StringMin);
check_member(button_caps[i], expect_button_caps[i], "%d", Range.StringMax);
}
if (!button_caps[i].IsDesignatorRange && !expect_button_caps[i].IsDesignatorRange)
check_member(button_caps[i], expect_button_caps[i], "%d", NotRange.DesignatorIndex);
else if (button_caps[i].IsDesignatorRange && expect_button_caps[i].IsDesignatorRange)
{
check_member(button_caps[i], expect_button_caps[i], "%d", Range.DesignatorMin);
check_member(button_caps[i], expect_button_caps[i], "%d", Range.DesignatorMax);
}
winetest_pop_context();
}
count = hid_caps->NumberInputValueCaps;
status = HidP_GetValueCaps(HidP_Input, value_caps, &count, preparsed);
ok(status == HIDP_STATUS_SUCCESS, "HidP_GetValueCaps returned %#x\n", status);
ok(count == ARRAY_SIZE(expect_value_caps), "got %d value caps\n", count);
for (i = 0; i < min(count, ARRAY_SIZE(expect_value_caps)); ++i)
{
winetest_push_context("value_caps[%d]", i);
check_member(value_caps[i], expect_value_caps[i], "%04x", UsagePage);
check_member(value_caps[i], expect_value_caps[i], "%d", ReportID);
check_member(value_caps[i], expect_value_caps[i], "%d", IsAlias);
check_member(value_caps[i], expect_value_caps[i], "%d", BitField);
check_member(value_caps[i], expect_value_caps[i], "%d", LinkCollection);
check_member(value_caps[i], expect_value_caps[i], "%d", LinkUsage);
check_member(value_caps[i], expect_value_caps[i], "%d", LinkUsagePage);
check_member(value_caps[i], expect_value_caps[i], "%d", IsRange);
check_member(value_caps[i], expect_value_caps[i], "%d", IsStringRange);
check_member(value_caps[i], expect_value_caps[i], "%d", IsDesignatorRange);
check_member(value_caps[i], expect_value_caps[i], "%d", IsAbsolute);
check_member(value_caps[i], expect_value_caps[i], "%d", HasNull);
check_member(value_caps[i], expect_value_caps[i], "%d", BitSize);
check_member(value_caps[i], expect_value_caps[i], "%d", ReportCount);
check_member(value_caps[i], expect_value_caps[i], "%d", UnitsExp);
check_member(value_caps[i], expect_value_caps[i], "%d", Units);
check_member(value_caps[i], expect_value_caps[i], "%d", LogicalMin);
check_member(value_caps[i], expect_value_caps[i], "%d", LogicalMax);
check_member(value_caps[i], expect_value_caps[i], "%d", PhysicalMin);
check_member(value_caps[i], expect_value_caps[i], "%d", PhysicalMax);
if (!value_caps[i].IsRange && !expect_value_caps[i].IsRange)
{
check_member(value_caps[i], expect_value_caps[i], "%04x", NotRange.Usage);
check_member(value_caps[i], expect_value_caps[i], "%d", NotRange.DataIndex);
}
else if (value_caps[i].IsRange && expect_value_caps[i].IsRange)
{
check_member(value_caps[i], expect_value_caps[i], "%04x", Range.UsageMin);
check_member(value_caps[i], expect_value_caps[i], "%04x", Range.UsageMax);
check_member(value_caps[i], expect_value_caps[i], "%d", Range.DataIndexMin);
check_member(value_caps[i], expect_value_caps[i], "%d", Range.DataIndexMax);
}
if (!value_caps[i].IsRange && !expect_value_caps[i].IsRange)
check_member(value_caps[i], expect_value_caps[i], "%d", NotRange.StringIndex);
else if (value_caps[i].IsStringRange && expect_value_caps[i].IsStringRange)
{
check_member(value_caps[i], expect_value_caps[i], "%d", Range.StringMin);
check_member(value_caps[i], expect_value_caps[i], "%d", Range.StringMax);
}
if (!value_caps[i].IsDesignatorRange && !expect_value_caps[i].IsDesignatorRange)
check_member(value_caps[i], expect_value_caps[i], "%d", NotRange.DesignatorIndex);
else if (value_caps[i].IsDesignatorRange && expect_value_caps[i].IsDesignatorRange)
{
check_member(value_caps[i], expect_value_caps[i], "%d", Range.DesignatorMin);
check_member(value_caps[i], expect_value_caps[i], "%d", Range.DesignatorMax);
}
winetest_pop_context();
}
status = HidP_InitializeReportForID(HidP_Input, 0, preparsed, buffer, hid_caps->InputReportByteLength);
ok(status == HIDP_STATUS_SUCCESS, "HidP_InitializeReportForID returned %#x\n", status);
SetLastError(0xdeadbeef);
memset(buffer, 0, sizeof(buffer));
ret = HidD_GetInputReport(device, buffer, hid_caps->InputReportByteLength);
ok(!ret, "HidD_GetInputReport succeeded\n");
ok(GetLastError() == ERROR_INVALID_PARAMETER, "HidD_GetInputReport returned error %u\n", GetLastError());
if (!winetest_interactive) skip("skipping interactive tests\n");
/* ReadFile on Xbox One For Windows controller seems to never succeed */
else if (attrs->VendorID == 0x045e && attrs->ProductID == 0x02ff) skip("skipping interactive tests (Xbox One For Windows)\n");
else
{
res = pXInputGetState(index, &last_state);
ok(res == ERROR_SUCCESS, "XInputGetState returned %#x\n", res);
trace("press A button on gamepad %d\n", index);
do
{
Sleep(5);
res = pXInputGetState(index, &state);
ok(res == ERROR_SUCCESS, "XInputGetState returned %#x\n", res);
} while (res == ERROR_SUCCESS && state.dwPacketNumber == last_state.dwPacketNumber);
ok(state.Gamepad.wButtons & XINPUT_GAMEPAD_A, "unexpected button state %#x\n", state.Gamepad.wButtons);
/* now read as many reports from the device to get a consistent final state */
for (i = 0; i < (state.dwPacketNumber - last_state.dwPacketNumber); ++i)
{
SetLastError(0xdeadbeef);
memset(buffer, 0, sizeof(buffer));
length = hid_caps->InputReportByteLength;
ret = ReadFile(device, buffer, hid_caps->InputReportByteLength, &length, NULL);
ok(ret, "ReadFile failed, last error %u\n", GetLastError());
ok(length == hid_caps->InputReportByteLength, "ReadFile returned length %u\n", length);
}
last_state = state;
length = ARRAY_SIZE(usages);
status = HidP_GetUsages(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, usages, &length, preparsed, buffer, hid_caps->InputReportByteLength);
ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsages returned %#x\n", status);
ok(length == 1, "got length %u\n", length);
ok(usages[0] == 1, "got usages[0] %u\n", usages[0]);
trace("release A on gamepad %d\n", index);
do
{
Sleep(5);
res = pXInputGetState(index, &state);
ok(res == ERROR_SUCCESS, "XInputGetState returned %#x\n", res);
} while (res == ERROR_SUCCESS && state.dwPacketNumber == last_state.dwPacketNumber);
ok(!state.Gamepad.wButtons, "unexpected button state %#x\n", state.Gamepad.wButtons);
/* now read as many reports from the device to get a consistent final state */
for (i = 0; i < (state.dwPacketNumber - last_state.dwPacketNumber); ++i)
{
SetLastError(0xdeadbeef);
memset(buffer, 0, sizeof(buffer));
length = hid_caps->InputReportByteLength;
ret = ReadFile(device, buffer, hid_caps->InputReportByteLength, &length, NULL);
ok(ret, "ReadFile failed, last error %u\n", GetLastError());
ok(length == hid_caps->InputReportByteLength, "ReadFile returned length %u\n", length);
}
last_state = state;
length = ARRAY_SIZE(usages);
status = HidP_GetUsages(HidP_Input, HID_USAGE_PAGE_BUTTON, 0, usages, &length, preparsed, buffer, hid_caps->InputReportByteLength);
ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsages returned %#x\n", status);
ok(length == 0, "got length %u\n", length);
trace("press both LT and RT on gamepad %d\n", index);
do
{
do
{
Sleep(5);
res = pXInputGetState(index, &state);
ok(res == ERROR_SUCCESS, "XInputGetState returned %#x\n", res);
} while (res == ERROR_SUCCESS && state.dwPacketNumber == last_state.dwPacketNumber);
/* now read as many reports from the device to get a consistent final state */
for (i = 0; i < (state.dwPacketNumber - last_state.dwPacketNumber); ++i)
{
SetLastError(0xdeadbeef);
memset(buffer, 0, sizeof(buffer));
length = hid_caps->InputReportByteLength;
ret = ReadFile(device, buffer, hid_caps->InputReportByteLength, &length, NULL);
ok(ret, "ReadFile failed, last error %u\n", GetLastError());
ok(length == hid_caps->InputReportByteLength, "ReadFile returned length %u\n", length);
}
last_state = state;
value = 0;
status = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_X, &value, preparsed, buffer, hid_caps->InputReportByteLength);
ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
ok(value == state.Gamepad.sThumbLX + 32768, "got LX value %d\n", value);
value = 0;
status = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Y, &value, preparsed, buffer, hid_caps->InputReportByteLength);
ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
ok(value == 32767 - state.Gamepad.sThumbLY, "got LY value %d\n", value);
value = 0;
status = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RX, &value, preparsed, buffer, hid_caps->InputReportByteLength);
ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
ok(value == state.Gamepad.sThumbRX + 32768, "got LX value %d\n", value);
value = 0;
status = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RY, &value, preparsed, buffer, hid_caps->InputReportByteLength);
ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
ok(value == 32767 - state.Gamepad.sThumbRY, "got LY value %d\n", value);
value = 0;
status = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_Z, &value, preparsed, buffer, hid_caps->InputReportByteLength);
ok(status == HIDP_STATUS_SUCCESS, "HidP_GetUsageValue returned %#x\n", status);
ok(value == 32768 + (state.Gamepad.bLeftTrigger - state.Gamepad.bRightTrigger) * 128, "got Z value %d (RT %d, LT %d)\n",
value, state.Gamepad.bRightTrigger, state.Gamepad.bLeftTrigger);
value = 0;
status = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0, HID_USAGE_GENERIC_RZ, &value, preparsed, buffer, hid_caps->InputReportByteLength);
ok(status == HIDP_STATUS_USAGE_NOT_FOUND, "HidP_GetUsageValue returned %#x\n", status);
} while (ret && (state.Gamepad.bRightTrigger != 255 || state.Gamepad.bLeftTrigger != 255));
}
}
static BOOL try_open_hid_device(const WCHAR *path, HANDLE *device, PHIDP_PREPARSED_DATA *preparsed,
HIDD_ATTRIBUTES *attrs, HIDP_CAPS *caps)
{
PHIDP_PREPARSED_DATA preparsed_data = NULL;
HANDLE device_file;
device_file = CreateFileW(path, FILE_READ_ACCESS | FILE_WRITE_ACCESS, FILE_SHARE_READ | FILE_SHARE_WRITE,
NULL, OPEN_EXISTING, 0, NULL);
if (device_file == INVALID_HANDLE_VALUE) return FALSE;
if (!HidD_GetPreparsedData(device_file, &preparsed_data)) goto failed;
if (!HidD_GetAttributes(device_file, attrs)) goto failed;
if (HidP_GetCaps(preparsed_data, caps) != HIDP_STATUS_SUCCESS) goto failed;
*device = device_file;
*preparsed = preparsed_data;
return TRUE;
failed:
CloseHandle(device_file);
HidD_FreePreparsedData(preparsed_data);
return FALSE;
}
static void test_hid_reports(void)
{
static const WCHAR prefix[] = L"\\\\?\\HID#VID_0000&PID_0000";
char buffer[sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA_W) + MAX_PATH * sizeof(WCHAR)];
SP_DEVICE_INTERFACE_DETAIL_DATA_W *detail = (void *)buffer;
SP_DEVICE_INTERFACE_DATA iface = {sizeof(iface)};
SP_DEVINFO_DATA devinfo = {sizeof(devinfo)};
PHIDP_PREPARSED_DATA preparsed;
HIDD_ATTRIBUTES attrs;
HIDP_CAPS caps;
HDEVINFO set;
HANDLE device;
UINT32 i = 0, cnt = 0;
GUID hid;
HidD_GetHidGuid(&hid);
set = SetupDiGetClassDevsW(&hid, NULL, NULL, DIGCF_DEVICEINTERFACE);
ok(set != INVALID_HANDLE_VALUE, "SetupDiGetClassDevsW failed, error %u\n", GetLastError());
while (SetupDiEnumDeviceInterfaces(set, NULL, &hid, i++, &iface))
{
detail->cbSize = sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA_W);
if (!SetupDiGetDeviceInterfaceDetailW(set, &iface, detail, sizeof(buffer), NULL, &devinfo))
continue;
if (!try_open_hid_device(detail->DevicePath, &device, &preparsed, &attrs, &caps))
continue;
if (wcslen(detail->DevicePath) <= wcslen(prefix) ||
wcsnicmp(detail->DevicePath + wcslen(prefix), L"&IG_", 4 ))
continue;
trace("found xinput HID device %s\n", wine_dbgstr_w(detail->DevicePath));
check_hid_caps(cnt++, device, preparsed, &attrs, &caps);
CloseHandle(device);
HidD_FreePreparsedData(preparsed);
}
SetupDiDestroyDeviceInfoList(set);
}
START_TEST(xinput)
{
struct
{
const char *name;
int version;
} libs[] = {
{ "xinput1_1.dll", 1 },
{ "xinput1_2.dll", 2 },
{ "xinput1_3.dll", 3 },
{ "xinput1_4.dll", 4 },
{ "xinput9_1_0.dll", 0 } /* legacy for XP/Vista */
};
HMODULE hXinput;
void *pXInputGetStateEx_Ordinal;
int i;
for (i = 0; i < ARRAY_SIZE(libs); i++)
{
hXinput = LoadLibraryA( libs[i].name );
if (!hXinput)
{
win_skip("Could not load %s\n", libs[i].name);
continue;
}
trace("Testing %s\n", libs[i].name);
pXInputEnable = (void*)GetProcAddress(hXinput, "XInputEnable");
pXInputSetState = (void*)GetProcAddress(hXinput, "XInputSetState");
pXInputGetState = (void*)GetProcAddress(hXinput, "XInputGetState");
pXInputGetStateEx = (void*)GetProcAddress(hXinput, "XInputGetStateEx"); /* Win >= 8 */
pXInputGetStateEx_Ordinal = (void*)GetProcAddress(hXinput, (LPCSTR) 100);
pXInputGetKeystroke = (void*)GetProcAddress(hXinput, "XInputGetKeystroke");
pXInputGetCapabilities = (void*)GetProcAddress(hXinput, "XInputGetCapabilities");
pXInputGetDSoundAudioDeviceGuids = (void*)GetProcAddress(hXinput, "XInputGetDSoundAudioDeviceGuids");
pXInputGetBatteryInformation = (void*)GetProcAddress(hXinput, "XInputGetBatteryInformation");
/* XInputGetStateEx may not be present by name, use ordinal in this case */
if (!pXInputGetStateEx)
pXInputGetStateEx = pXInputGetStateEx_Ordinal;
test_hid_reports();
test_set_state();
test_get_state();
test_get_capabilities();
if (libs[i].version != 4)
test_get_dsoundaudiodevice();
else
ok(!pXInputGetDSoundAudioDeviceGuids, "XInputGetDSoundAudioDeviceGuids exists in %s\n", libs[i].name);
if (libs[i].version > 2)
{
test_get_keystroke();
test_get_batteryinformation();
ok(pXInputGetStateEx != NULL, "XInputGetStateEx not found in %s\n", libs[i].name);
}
else
{
ok(!pXInputGetKeystroke, "XInputGetKeystroke exists in %s\n", libs[i].name);
ok(!pXInputGetStateEx, "XInputGetStateEx exists in %s\n", libs[i].name);
ok(!pXInputGetBatteryInformation, "XInputGetBatteryInformation exists in %s\n", libs[i].name);
if (libs[i].version == 0)
ok(!pXInputEnable, "XInputEnable exists in %s\n", libs[i].name);
}
FreeLibrary(hXinput);
}
}