2 * Copyright (c) 2011 Andrew Nguyen
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 #define DIRECTINPUT_VERSION 0x0800
26 #include "wine/test.h"
30 static BOOL CALLBACK dummy_callback(const DIDEVICEINSTANCEA *instance, void *context)
32 ok(0, "Callback was invoked with parameters (%p, %p)\n", instance, context);
36 static void test_preinitialization(void)
43 } create_device_tests[] =
45 {NULL, FALSE, E_POINTER},
46 {NULL, TRUE, E_POINTER},
47 {&GUID_Unknown, FALSE, E_POINTER},
48 {&GUID_Unknown, TRUE, DIERR_NOTINITIALIZED},
49 {&GUID_SysMouse, FALSE, E_POINTER},
50 {&GUID_SysMouse, TRUE, DIERR_NOTINITIALIZED},
56 LPDIENUMDEVICESCALLBACKA lpCallback;
60 } enum_devices_tests[] =
62 {0, NULL, 0, DIERR_INVALIDPARAM},
63 {0, NULL, ~0u, DIERR_INVALIDPARAM},
64 {0, dummy_callback, 0, DIERR_NOTINITIALIZED},
65 {0, dummy_callback, ~0u, DIERR_INVALIDPARAM, 1},
66 {0xdeadbeef, NULL, 0, DIERR_INVALIDPARAM},
67 {0xdeadbeef, NULL, ~0u, DIERR_INVALIDPARAM},
68 {0xdeadbeef, dummy_callback, 0, DIERR_INVALIDPARAM, 1},
69 {0xdeadbeef, dummy_callback, ~0u, DIERR_INVALIDPARAM, 1},
75 IDirectInputDevice8A *pDID;
77 hr = CoCreateInstance(&CLSID_DirectInput8, NULL, CLSCTX_INPROC_SERVER, &IID_IDirectInput8A, (void **)&pDI);
80 skip("Failed to instantiate a IDirectInputA instance: 0x%08x\n", hr);
84 for (i = 0; i < sizeof(create_device_tests)/sizeof(create_device_tests[0]); i++)
86 if (create_device_tests[i].pdev) pDID = (void *)0xdeadbeef;
87 hr = IDirectInput8_CreateDevice(pDI, create_device_tests[i].rguid,
88 create_device_tests[i].pdev ? &pDID : NULL,
90 ok(hr == create_device_tests[i].expected_hr, "[%d] IDirectInput8_CreateDevice returned 0x%08x\n", i, hr);
91 if (create_device_tests[i].pdev)
92 ok(pDID == NULL, "[%d] Output interface pointer is %p\n", i, pDID);
95 for (i = 0; i < sizeof(enum_devices_tests)/sizeof(enum_devices_tests[0]); i++)
97 hr = IDirectInput8_EnumDevices(pDI, enum_devices_tests[i].dwDevType,
98 enum_devices_tests[i].lpCallback,
100 enum_devices_tests[i].dwFlags);
101 if (enum_devices_tests[i].todo)
104 ok(hr == enum_devices_tests[i].expected_hr, "[%d] IDirectInput8_EnumDevice returned 0x%08x\n", i, hr);
107 ok(hr == enum_devices_tests[i].expected_hr, "[%d] IDirectInput8_EnumDevice returned 0x%08x\n", i, hr);
110 hr = IDirectInput8_GetDeviceStatus(pDI, NULL);
112 ok(hr == E_POINTER, "IDirectInput8_GetDeviceStatus returned 0x%08x\n", hr);
114 hr = IDirectInput8_GetDeviceStatus(pDI, &GUID_Unknown);
115 ok(hr == DIERR_NOTINITIALIZED, "IDirectInput8_GetDeviceStatus returned 0x%08x\n", hr);
117 hr = IDirectInput8_GetDeviceStatus(pDI, &GUID_SysMouse);
118 ok(hr == DIERR_NOTINITIALIZED, "IDirectInput8_GetDeviceStatus returned 0x%08x\n", hr);
120 hr = IDirectInput8_RunControlPanel(pDI, NULL, 0);
121 ok(hr == DIERR_NOTINITIALIZED, "IDirectInput8_RunControlPanel returned 0x%08x\n", hr);
123 hr = IDirectInput8_RunControlPanel(pDI, NULL, ~0u);
124 ok(hr == DIERR_INVALIDPARAM, "IDirectInput8_RunControlPanel returned 0x%08x\n", hr);
126 hr = IDirectInput8_RunControlPanel(pDI, (HWND)0xdeadbeef, 0);
127 ok(hr == E_HANDLE, "IDirectInput8_RunControlPanel returned 0x%08x\n", hr);
129 hr = IDirectInput8_RunControlPanel(pDI, (HWND)0xdeadbeef, ~0u);
130 ok(hr == E_HANDLE, "IDirectInput8_RunControlPanel returned 0x%08x\n", hr);
132 IDirectInput8_Release(pDI);
135 static void test_DirectInput8Create(void)
144 } invalid_param_list[] =
146 {FALSE, 0, &IID_IDirectInputA, FALSE, E_POINTER},
147 {FALSE, 0, &IID_IDirectInputA, TRUE, DIERR_NOINTERFACE},
148 {FALSE, 0, &IID_IDirectInput8A, FALSE, E_POINTER},
149 {FALSE, 0, &IID_IDirectInput8A, TRUE, DIERR_INVALIDPARAM},
150 {FALSE, DIRECTINPUT_VERSION, &IID_IDirectInputA, FALSE, E_POINTER},
151 {FALSE, DIRECTINPUT_VERSION, &IID_IDirectInputA, TRUE, DIERR_NOINTERFACE},
152 {FALSE, DIRECTINPUT_VERSION, &IID_IDirectInput8A, FALSE, E_POINTER},
153 {FALSE, DIRECTINPUT_VERSION, &IID_IDirectInput8A, TRUE, DIERR_INVALIDPARAM},
154 {FALSE, DIRECTINPUT_VERSION - 1, &IID_IDirectInputA, FALSE, E_POINTER},
155 {FALSE, DIRECTINPUT_VERSION - 1, &IID_IDirectInputA, TRUE, DIERR_NOINTERFACE},
156 {FALSE, DIRECTINPUT_VERSION - 1, &IID_IDirectInput8A, FALSE, E_POINTER},
157 {FALSE, DIRECTINPUT_VERSION - 1, &IID_IDirectInput8A, TRUE, DIERR_INVALIDPARAM},
158 {FALSE, DIRECTINPUT_VERSION + 1, &IID_IDirectInputA, FALSE, E_POINTER},
159 {FALSE, DIRECTINPUT_VERSION + 1, &IID_IDirectInputA, TRUE, DIERR_NOINTERFACE},
160 {FALSE, DIRECTINPUT_VERSION + 1, &IID_IDirectInput8A, FALSE, E_POINTER},
161 {FALSE, DIRECTINPUT_VERSION + 1, &IID_IDirectInput8A, TRUE, DIERR_INVALIDPARAM},
162 {TRUE, 0, &IID_IDirectInputA, FALSE, E_POINTER},
163 {TRUE, 0, &IID_IDirectInputA, TRUE, DIERR_NOINTERFACE},
164 {TRUE, 0, &IID_IDirectInput8A, FALSE, E_POINTER},
165 {TRUE, 0, &IID_IDirectInput8A, TRUE, DIERR_NOTINITIALIZED},
166 {TRUE, DIRECTINPUT_VERSION, &IID_IDirectInputA, FALSE, E_POINTER},
167 {TRUE, DIRECTINPUT_VERSION, &IID_IDirectInputA, TRUE, DIERR_NOINTERFACE},
168 {TRUE, DIRECTINPUT_VERSION, &IID_IDirectInput8A, FALSE, E_POINTER},
169 {TRUE, DIRECTINPUT_VERSION - 1, &IID_IDirectInputA, FALSE, E_POINTER},
170 {TRUE, DIRECTINPUT_VERSION - 1, &IID_IDirectInputA, TRUE, DIERR_NOINTERFACE},
171 {TRUE, DIRECTINPUT_VERSION - 1, &IID_IDirectInput8A, FALSE, E_POINTER},
172 {TRUE, DIRECTINPUT_VERSION - 1, &IID_IDirectInput8A, TRUE, DIERR_BETADIRECTINPUTVERSION},
173 {TRUE, DIRECTINPUT_VERSION + 1, &IID_IDirectInputA, FALSE, E_POINTER},
174 {TRUE, DIRECTINPUT_VERSION + 1, &IID_IDirectInputA, TRUE, DIERR_NOINTERFACE},
175 {TRUE, DIRECTINPUT_VERSION + 1, &IID_IDirectInput8A, FALSE, E_POINTER},
176 {TRUE, DIRECTINPUT_VERSION + 1, &IID_IDirectInput8A, TRUE, DIERR_OLDDIRECTINPUTVERSION},
179 static REFIID no_interface_list[] = {&IID_IDirectInputA, &IID_IDirectInputW,
180 &IID_IDirectInput2A, &IID_IDirectInput2W,
181 &IID_IDirectInput7A, &IID_IDirectInput7W,
182 &IID_IDirectInputDeviceA, &IID_IDirectInputDeviceW,
183 &IID_IDirectInputDevice2A, &IID_IDirectInputDevice2W,
184 &IID_IDirectInputDevice7A, &IID_IDirectInputDevice7W,
185 &IID_IDirectInputDevice8A, &IID_IDirectInputDevice8W,
186 &IID_IDirectInputEffect};
188 static REFIID iid_list[] = {&IID_IUnknown, &IID_IDirectInput8A, &IID_IDirectInput8W};
194 for (i = 0; i < sizeof(invalid_param_list)/sizeof(invalid_param_list[0]); i++)
196 if (invalid_param_list[i].ppdi) pUnk = (void *)0xdeadbeef;
197 hr = DirectInput8Create(invalid_param_list[i].hinst ? hInstance : NULL,
198 invalid_param_list[i].dwVersion,
199 invalid_param_list[i].riid,
200 invalid_param_list[i].ppdi ? (void **)&pUnk : NULL,
202 ok(hr == invalid_param_list[i].expected_hr, "[%d] DirectInput8Create returned 0x%08x\n", i, hr);
203 if (invalid_param_list[i].ppdi)
204 ok(pUnk == NULL, "[%d] Output interface pointer is %p\n", i, pUnk);
207 for (i = 0; i < sizeof(no_interface_list)/sizeof(no_interface_list[0]); i++)
209 pUnk = (void *)0xdeadbeef;
210 hr = DirectInput8Create(hInstance, DIRECTINPUT_VERSION, no_interface_list[i], (void **)&pUnk, NULL);
211 ok(hr == DIERR_NOINTERFACE, "[%d] DirectInput8Create returned 0x%08x\n", i, hr);
212 ok(pUnk == NULL, "[%d] Output interface pointer is %p\n", i, pUnk);
215 for (i = 0; i < sizeof(iid_list)/sizeof(iid_list[0]); i++)
218 hr = DirectInput8Create(hInstance, DIRECTINPUT_VERSION, iid_list[i], (void **)&pUnk, NULL);
219 ok(hr == DI_OK, "[%d] DirectInput8Create returned 0x%08x\n", i, hr);
220 ok(pUnk != NULL, "[%d] Output interface pointer is NULL\n", i);
222 IUnknown_Release(pUnk);
226 static void test_QueryInterface(void)
228 static REFIID iid_list[] = {&IID_IUnknown, &IID_IDirectInput8A, &IID_IDirectInput8W};
234 } no_interface_list[] =
236 {&IID_IDirectInputA, 1},
237 {&IID_IDirectInputW, 1},
238 {&IID_IDirectInput2A, 1},
239 {&IID_IDirectInput2W, 1},
240 {&IID_IDirectInput7A, 1},
241 {&IID_IDirectInput7W, 1},
242 {&IID_IDirectInputDeviceA},
243 {&IID_IDirectInputDeviceW},
244 {&IID_IDirectInputDevice2A},
245 {&IID_IDirectInputDevice2W},
246 {&IID_IDirectInputDevice7A},
247 {&IID_IDirectInputDevice7W},
248 {&IID_IDirectInputDevice8A},
249 {&IID_IDirectInputDevice8W},
250 {&IID_IDirectInputEffect},
258 hr = DirectInput8Create(hInstance, DIRECTINPUT_VERSION, &IID_IDirectInput8A, (void **)&pDI, NULL);
261 win_skip("Failed to instantiate a IDirectInputA instance: 0x%08x\n", hr);
265 hr = IDirectInput8_QueryInterface(pDI, NULL, NULL);
266 ok(hr == E_POINTER, "IDirectInput8_QueryInterface returned 0x%08x\n", hr);
268 pUnk = (void *)0xdeadbeef;
269 hr = IDirectInput8_QueryInterface(pDI, NULL, (void **)&pUnk);
270 ok(hr == E_POINTER, "IDirectInput8_QueryInterface returned 0x%08x\n", hr);
271 ok(pUnk == (void *)0xdeadbeef, "Output interface pointer is %p\n", pUnk);
273 hr = IDirectInput8_QueryInterface(pDI, &IID_IUnknown, NULL);
274 ok(hr == E_POINTER, "IDirectInput8_QueryInterface returned 0x%08x\n", hr);
276 for (i = 0; i < sizeof(iid_list)/sizeof(iid_list[0]); i++)
279 hr = IDirectInput8_QueryInterface(pDI, iid_list[i], (void **)&pUnk);
280 ok(hr == S_OK, "[%d] IDirectInput8_QueryInterface returned 0x%08x\n", i, hr);
281 ok(pUnk != NULL, "[%d] Output interface pointer is NULL\n", i);
282 if (pUnk) IUnknown_Release(pUnk);
285 for (i = 0; i < sizeof(no_interface_list)/sizeof(no_interface_list[0]); i++)
287 pUnk = (void *)0xdeadbeef;
288 hr = IDirectInput8_QueryInterface(pDI, no_interface_list[i].riid, (void **)&pUnk);
289 if (no_interface_list[i].test_todo)
292 ok(hr == E_NOINTERFACE, "[%d] IDirectInput8_QueryInterface returned 0x%08x\n", i, hr);
294 ok(pUnk == NULL, "[%d] Output interface pointer is %p\n", i, pUnk);
296 if (pUnk) IUnknown_Release(pUnk);
300 ok(hr == E_NOINTERFACE, "[%d] IDirectInput8_QueryInterface returned 0x%08x\n", i, hr);
301 ok(pUnk == NULL, "[%d] Output interface pointer is %p\n", i, pUnk);
305 IDirectInput8_Release(pDI);
308 static void test_CreateDevice(void)
312 IDirectInputDevice8A *pDID;
314 hr = DirectInput8Create(hInstance, DIRECTINPUT_VERSION, &IID_IDirectInput8A, (void **)&pDI, NULL);
317 win_skip("Failed to instantiate a IDirectInputA instance: 0x%08x\n", hr);
321 hr = IDirectInput8_CreateDevice(pDI, NULL, NULL, NULL);
322 ok(hr == E_POINTER, "IDirectInput8_CreateDevice returned 0x%08x\n", hr);
324 pDID = (void *)0xdeadbeef;
325 hr = IDirectInput8_CreateDevice(pDI, NULL, &pDID, NULL);
326 ok(hr == E_POINTER, "IDirectInput8_CreateDevice returned 0x%08x\n", hr);
327 ok(pDID == NULL, "Output interface pointer is %p\n", pDID);
329 hr = IDirectInput8_CreateDevice(pDI, &GUID_Unknown, NULL, NULL);
330 ok(hr == E_POINTER, "IDirectInput8_CreateDevice returned 0x%08x\n", hr);
332 pDID = (void *)0xdeadbeef;
333 hr = IDirectInput8_CreateDevice(pDI, &GUID_Unknown, &pDID, NULL);
334 ok(hr == DIERR_DEVICENOTREG, "IDirectInput8_CreateDevice returned 0x%08x\n", hr);
335 ok(pDID == NULL, "Output interface pointer is %p\n", pDID);
337 hr = IDirectInput8_CreateDevice(pDI, &GUID_SysMouse, NULL, NULL);
338 ok(hr == E_POINTER, "IDirectInput8_CreateDevice returned 0x%08x\n", hr);
340 hr = IDirectInput8_CreateDevice(pDI, &GUID_SysMouse, &pDID, NULL);
341 ok(hr == DI_OK, "IDirectInput8_CreateDevice returned 0x%08x\n", hr);
343 IDirectInputDevice_Release(pDID);
344 IDirectInput8_Release(pDI);
347 struct enum_devices_test
349 unsigned int device_count;
353 static BOOL CALLBACK enum_devices_callback(const DIDEVICEINSTANCEA *instance, void *context)
355 struct enum_devices_test *enum_test = context;
357 enum_test->device_count++;
358 return enum_test->return_value;
361 static void test_EnumDevices(void)
365 struct enum_devices_test enum_test, enum_test_return;
367 hr = DirectInput8Create(hInstance, DIRECTINPUT_VERSION, &IID_IDirectInput8A, (void **)&pDI, NULL);
370 win_skip("Failed to instantiate a IDirectInputA instance: 0x%08x\n", hr);
374 hr = IDirectInput8_EnumDevices(pDI, 0, NULL, NULL, 0);
375 ok(hr == DIERR_INVALIDPARAM, "IDirectInput8_EnumDevices returned 0x%08x\n", hr);
377 hr = IDirectInput8_EnumDevices(pDI, 0, NULL, NULL, ~0u);
378 ok(hr == DIERR_INVALIDPARAM, "IDirectInput8_EnumDevices returned 0x%08x\n", hr);
380 /* Test crashes on Wine. */
383 hr = IDirectInput8_EnumDevices(pDI, 0, enum_devices_callback, NULL, ~0u);
384 ok(hr == DIERR_INVALIDPARAM, "IDirectInput8_EnumDevices returned 0x%08x\n", hr);
387 hr = IDirectInput8_EnumDevices(pDI, 0xdeadbeef, NULL, NULL, 0);
388 ok(hr == DIERR_INVALIDPARAM, "IDirectInput8_EnumDevices returned 0x%08x\n", hr);
390 hr = IDirectInput8_EnumDevices(pDI, 0xdeadbeef, NULL, NULL, ~0u);
391 ok(hr == DIERR_INVALIDPARAM, "IDirectInput8_EnumDevices returned 0x%08x\n", hr);
393 hr = IDirectInput8_EnumDevices(pDI, 0xdeadbeef, enum_devices_callback, NULL, 0);
395 ok(hr == DIERR_INVALIDPARAM, "IDirectInput8_EnumDevices returned 0x%08x\n", hr);
397 hr = IDirectInput8_EnumDevices(pDI, 0xdeadbeef, enum_devices_callback, NULL, ~0u);
399 ok(hr == DIERR_INVALIDPARAM, "IDirectInput8_EnumDevices returned 0x%08x\n", hr);
401 enum_test.device_count = 0;
402 enum_test.return_value = DIENUM_CONTINUE;
403 hr = IDirectInput8_EnumDevices(pDI, 0, enum_devices_callback, &enum_test, 0);
404 ok(hr == DI_OK, "IDirectInput8_EnumDevices returned 0x%08x\n", hr);
405 ok(enum_test.device_count != 0, "Device count is %u\n", enum_test.device_count);
407 /* Enumeration only stops with an explicit DIENUM_STOP. */
408 enum_test_return.device_count = 0;
409 enum_test_return.return_value = 42;
410 hr = IDirectInput8_EnumDevices(pDI, 0, enum_devices_callback, &enum_test_return, 0);
411 ok(hr == DI_OK, "IDirectInput8_EnumDevices returned 0x%08x\n", hr);
412 ok(enum_test_return.device_count == enum_test.device_count,
413 "Device count is %u vs. %u\n", enum_test_return.device_count, enum_test.device_count);
415 enum_test.device_count = 0;
416 enum_test.return_value = DIENUM_STOP;
417 hr = IDirectInput8_EnumDevices(pDI, 0, enum_devices_callback, &enum_test, 0);
418 ok(hr == DI_OK, "IDirectInput8_EnumDevices returned 0x%08x\n", hr);
419 ok(enum_test.device_count == 1, "Device count is %u\n", enum_test.device_count);
421 IDirectInput8_Release(pDI);
424 struct enum_semantics_test
426 unsigned int device_count;
429 LPDIACTIONFORMAT lpdiaf;
430 const char* username;
433 static DIACTION actionMapping[]=
436 { 0, 0x01008A01 /* DIAXIS_DRIVINGR_STEER */, 0, { "Steer" } },
438 { 1, 0x01000C01 /* DIBUTTON_DRIVINGR_SHIFTUP */, 0, { "Upshift" } },
440 { 2, DIKEYBOARD_SPACE, 0, { "Missile" } },
442 { 3, DIMOUSE_BUTTON0, 0, { "Select" } },
444 { 4, DIMOUSE_YAXIS, 0, { "Y Axis" } }
447 static BOOL CALLBACK enum_semantics_callback(LPCDIDEVICEINSTANCE lpddi, IDirectInputDevice8A *lpdid, DWORD dwFlags, DWORD dwRemaining, void *context)
449 struct enum_semantics_test *data = context;
451 if (context == NULL) return DIENUM_STOP;
453 data->device_count++;
455 if (IsEqualGUID(&lpddi->guidInstance, &GUID_SysKeyboard)) data->keyboard = TRUE;
457 if (IsEqualGUID(&lpddi->guidInstance, &GUID_SysMouse)) data->mouse = TRUE;
459 return DIENUM_CONTINUE;
462 static BOOL CALLBACK set_action_map_callback(LPCDIDEVICEINSTANCE lpddi, IDirectInputDevice8A *lpdid, DWORD dwFlags, DWORD dwRemaining, void *context)
465 struct enum_semantics_test *data = context;
467 /* Building and setting an action map */
468 /* It should not use any pre-stored mappings so we use DIDBAM_INITIALIZE */
469 hr = IDirectInputDevice8_BuildActionMap(lpdid, data->lpdiaf, NULL, DIDBAM_INITIALIZE);
470 ok (SUCCEEDED(hr), "BuildActionMap failed hr=%08x\n", hr);
472 hr = IDirectInputDevice8_SetActionMap(lpdid, data->lpdiaf, data->username, 0);
473 ok (SUCCEEDED(hr), "SetActionMap failed hr=%08x\n", hr);
475 return DIENUM_CONTINUE;
478 static void test_EnumDevicesBySemantics(void)
482 DIACTIONFORMATA diaf;
483 const GUID ACTION_MAPPING_GUID = { 0x1, 0x2, 0x3, { 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0xa, 0xb } };
484 struct enum_semantics_test data = { 0, FALSE, FALSE, &diaf, NULL };
485 int device_total = 0;
487 hr = DirectInput8Create(hInstance, DIRECTINPUT_VERSION, &IID_IDirectInput8A, (void **)&pDI, NULL);
490 win_skip("Failed to instantiate a IDirectInputA instance: 0x%08x\n", hr);
494 memset (&diaf, 0, sizeof(diaf));
495 diaf.dwSize = sizeof(diaf);
496 diaf.dwActionSize = sizeof(DIACTION);
497 diaf.dwNumActions = sizeof(actionMapping) / sizeof(actionMapping[0]);
498 diaf.dwDataSize = 4 * diaf.dwNumActions;
499 diaf.rgoAction = actionMapping;
500 diaf.guidActionMap = ACTION_MAPPING_GUID;
501 diaf.dwGenre = 0x01000000; /* DIVIRTUAL_DRIVING_RACE */
502 diaf.dwBufferSize = 32;
504 /* Test enumerating all attached and installed devices */
505 data.keyboard = FALSE;
507 data.device_count = 0;
508 hr = IDirectInput8_EnumDevicesBySemantics(pDI, NULL, &diaf, enum_semantics_callback, &data, DIEDBSFL_ATTACHEDONLY);
509 ok (data.device_count > 0, "EnumDevicesBySemantics did not call the callback hr=%08x\n", hr);
510 ok (data.keyboard, "EnumDevicesBySemantics should enumerate the keyboard\n");
511 ok (data.mouse, "EnumDevicesBySemantics should enumerate the mouse\n");
513 /* Enumerate Force feedback devices. We should get no mouse nor keyboard */
514 data.keyboard = FALSE;
516 hr = IDirectInput8_EnumDevicesBySemantics(pDI, NULL, &diaf, enum_semantics_callback, &data, DIEDBSFL_FORCEFEEDBACK);
517 ok (SUCCEEDED(hr), "EnumDevicesBySemantics failed hr=%08x\n", hr);
518 ok (!data.keyboard, "Keyboard should not be enumerated when asking for forcefeedback\n");
519 ok (!data.mouse, "Mouse should not be enumerated when asking for forcefeedback\n");
521 /* Enumerate available devices. That is devices not owned by any user.
522 Before setting the action map for all devices we still have them available. */
523 data.device_count = 0;
524 hr = IDirectInput8_EnumDevicesBySemantics(pDI, NULL, &diaf, enum_semantics_callback, &data, DIEDBSFL_AVAILABLEDEVICES);
525 ok (SUCCEEDED(hr), "EnumDevicesBySemantics failed hr=%08x\n", hr);
526 ok (data.device_count > 0, "There should be devices available before action mapping available=%d\n", data.device_count);
528 /* Keep the device total */
529 device_total = data.device_count;
531 /* This enumeration builds and sets the action map for all devices with a NULL username */
532 hr = IDirectInput8_EnumDevicesBySemantics(pDI, NULL, &diaf, set_action_map_callback, &data, DIEDBSFL_ATTACHEDONLY);
533 ok (SUCCEEDED(hr), "EnumDevicesBySemantics failed: hr=%08x\n", hr);
535 /* After a successful action mapping we should have no devices available */
536 data.device_count = 0;
537 hr = IDirectInput8_EnumDevicesBySemantics(pDI, NULL, &diaf, enum_semantics_callback, &data, DIEDBSFL_AVAILABLEDEVICES);
538 ok (SUCCEEDED(hr), "EnumDevicesBySemantics failed hr=%08x\n", hr);
539 todo_wine ok (data.device_count == 0, "No device should be available after action mapping available=%d\n", data.device_count);
541 /* Now we'll give all the devices to a specific user */
542 data.username = "Sh4d0w M4g3";
543 hr = IDirectInput8_EnumDevicesBySemantics(pDI, NULL, &diaf, set_action_map_callback, &data, DIEDBSFL_ATTACHEDONLY);
544 ok (SUCCEEDED(hr), "EnumDevicesBySemantics failed: hr=%08x\n", hr);
546 /* Testing with the default user, DIEDBSFL_THISUSER has no effect */
547 data.device_count = 0;
548 hr = IDirectInput8_EnumDevicesBySemantics(pDI, NULL, &diaf, enum_semantics_callback, &data, DIEDBSFL_THISUSER);
549 ok (SUCCEEDED(hr), "EnumDevicesBySemantics failed hr=%08x\n", hr);
550 ok (data.device_count == device_total, "THISUSER has no effect with NULL username owned=%d, expected=%d\n", data.device_count, device_total);
552 /* Using an empty user string is the same as passing NULL, DIEDBSFL_THISUSER has no effect */
553 data.device_count = 0;
554 hr = IDirectInput8_EnumDevicesBySemantics(pDI, "", &diaf, enum_semantics_callback, &data, DIEDBSFL_THISUSER);
555 ok (SUCCEEDED(hr), "EnumDevicesBySemantics failed hr=%08x\n", hr);
556 ok (data.device_count == device_total, "THISUSER has no effect with \"\" as username owned=%d, expected=%d\n", data.device_count, device_total);
558 /* Testing with a user with no ownership of the devices */
559 data.device_count = 0;
560 hr = IDirectInput8_EnumDevicesBySemantics(pDI, "Ninja Brian", &diaf, enum_semantics_callback, &data, DIEDBSFL_THISUSER);
561 ok (SUCCEEDED(hr), "EnumDevicesBySemantics failed hr=%08x\n", hr);
562 todo_wine ok (data.device_count == 0, "This user should own no devices owned=%d\n", data.device_count);
564 /* Sh4d0w M4g3 has ownership of all devices */
565 data.device_count = 0;
566 hr = IDirectInput8_EnumDevicesBySemantics(pDI, "Sh4d0w M4g3", &diaf, enum_semantics_callback, &data, DIEDBSFL_THISUSER);
567 ok (SUCCEEDED(hr), "EnumDevicesBySemantics failed hr=%08x\n", hr);
568 ok (data.device_count == device_total, "This user should own %d devices owned=%d\n", device_total, data.device_count);
570 /* The call fails with a zeroed GUID */
571 memset(&diaf.guidActionMap, 0, sizeof(GUID));
572 hr = IDirectInput8_EnumDevicesBySemantics(pDI, NULL, &diaf, enum_semantics_callback, NULL, 0);
573 todo_wine ok(FAILED(hr), "EnumDevicesBySemantics succeeded with invalid GUID hr=%08x\n", hr);
575 IDirectInput8_Release(pDI);
578 static void test_GetDeviceStatus(void)
583 hr = DirectInput8Create(hInstance, DIRECTINPUT_VERSION, &IID_IDirectInput8A, (void **)&pDI, NULL);
586 win_skip("Failed to instantiate a IDirectInputA instance: 0x%08x\n", hr);
590 hr = IDirectInput8_GetDeviceStatus(pDI, NULL);
592 ok(hr == E_POINTER, "IDirectInput8_GetDeviceStatus returned 0x%08x\n", hr);
594 hr = IDirectInput8_GetDeviceStatus(pDI, &GUID_Unknown);
596 ok(hr == DIERR_DEVICENOTREG, "IDirectInput8_GetDeviceStatus returned 0x%08x\n", hr);
598 hr = IDirectInput8_GetDeviceStatus(pDI, &GUID_SysMouse);
599 ok(hr == DI_OK, "IDirectInput8_GetDeviceStatus returned 0x%08x\n", hr);
601 IDirectInput8_Release(pDI);
604 static void test_RunControlPanel(void)
609 hr = DirectInput8Create(hInstance, DIRECTINPUT_VERSION, &IID_IDirectInput8A, (void **)&pDI, NULL);
612 win_skip("Failed to instantiate a IDirectInputA instance: 0x%08x\n", hr);
616 if (winetest_interactive)
618 hr = IDirectInput8_RunControlPanel(pDI, NULL, 0);
619 ok(hr == S_OK, "IDirectInput8_RunControlPanel returned 0x%08x\n", hr);
621 hr = IDirectInput8_RunControlPanel(pDI, GetDesktopWindow(), 0);
622 ok(hr == S_OK, "IDirectInput8_RunControlPanel returned 0x%08x\n", hr);
625 hr = IDirectInput8_RunControlPanel(pDI, NULL, ~0u);
626 ok(hr == DIERR_INVALIDPARAM, "IDirectInput8_RunControlPanel returned 0x%08x\n", hr);
628 hr = IDirectInput8_RunControlPanel(pDI, (HWND)0xdeadbeef, 0);
629 ok(hr == E_HANDLE, "IDirectInput8_RunControlPanel returned 0x%08x\n", hr);
631 hr = IDirectInput8_RunControlPanel(pDI, (HWND)0xdeadbeef, ~0u);
632 ok(hr == E_HANDLE, "IDirectInput8_RunControlPanel returned 0x%08x\n", hr);
634 IDirectInput8_Release(pDI);
637 static void test_Initialize(void)
642 hr = DirectInput8Create(hInstance, DIRECTINPUT_VERSION, &IID_IDirectInput8A, (void **)&pDI, NULL);
645 win_skip("Failed to instantiate a IDirectInputA instance: 0x%08x\n", hr);
649 hr = IDirectInput8_Initialize(pDI, NULL, 0);
650 ok(hr == DIERR_INVALIDPARAM, "IDirectInput8_Initialize returned 0x%08x\n", hr);
652 hr = IDirectInput8_Initialize(pDI, NULL, DIRECTINPUT_VERSION);
653 ok(hr == DIERR_INVALIDPARAM, "IDirectInput8_Initialize returned 0x%08x\n", hr);
655 hr = IDirectInput8_Initialize(pDI, hInstance, 0);
656 ok(hr == DIERR_NOTINITIALIZED, "IDirectInput8_Initialize returned 0x%08x\n", hr);
658 /* Invalid DirectInput versions less than DIRECTINPUT_VERSION yield DIERR_BETADIRECTINPUTVERSION. */
659 hr = IDirectInput8_Initialize(pDI, hInstance, DIRECTINPUT_VERSION - 1);
660 ok(hr == DIERR_BETADIRECTINPUTVERSION, "IDirectInput8_Initialize returned 0x%08x\n", hr);
662 /* Invalid DirectInput versions greater than DIRECTINPUT_VERSION yield DIERR_BETADIRECTINPUTVERSION. */
663 hr = IDirectInput8_Initialize(pDI, hInstance, DIRECTINPUT_VERSION + 1);
664 ok(hr == DIERR_OLDDIRECTINPUTVERSION, "IDirectInput8_Initialize returned 0x%08x\n", hr);
666 hr = IDirectInput8_Initialize(pDI, hInstance, DIRECTINPUT_VERSION);
667 ok(hr == DI_OK, "IDirectInput8_Initialize returned 0x%08x\n", hr);
669 /* Parameters are still validated after successful initialization. */
670 hr = IDirectInput8_Initialize(pDI, hInstance, 0);
671 ok(hr == DIERR_NOTINITIALIZED, "IDirectInput8_Initialize returned 0x%08x\n", hr);
673 IDirectInput8_Release(pDI);
678 hInstance = GetModuleHandleA(NULL);
681 test_preinitialization();
682 test_DirectInput8Create();
683 test_QueryInterface();
686 test_EnumDevicesBySemantics();
687 test_GetDeviceStatus();
688 test_RunControlPanel();