SDL_dinputjoystick.c 50 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284
  1. /*
  2. Simple DirectMedia Layer
  3. Copyright (C) 1997-2020 Sam Lantinga <slouken@libsdl.org>
  4. This software is provided 'as-is', without any express or implied
  5. warranty. In no event will the authors be held liable for any damages
  6. arising from the use of this software.
  7. Permission is granted to anyone to use this software for any purpose,
  8. including commercial applications, and to alter it and redistribute it
  9. freely, subject to the following restrictions:
  10. 1. The origin of this software must not be misrepresented; you must not
  11. claim that you wrote the original software. If you use this software
  12. in a product, an acknowledgment in the product documentation would be
  13. appreciated but is not required.
  14. 2. Altered source versions must be plainly marked as such, and must not be
  15. misrepresented as being the original software.
  16. 3. This notice may not be removed or altered from any source distribution.
  17. */
  18. #include "../../SDL_internal.h"
  19. #include "../SDL_sysjoystick.h"
  20. #if SDL_JOYSTICK_DINPUT
  21. #include "SDL_windowsjoystick_c.h"
  22. #include "SDL_dinputjoystick_c.h"
  23. #include "SDL_xinputjoystick_c.h"
  24. #include "../hidapi/SDL_hidapijoystick_c.h"
  25. #ifndef DIDFT_OPTIONAL
  26. #define DIDFT_OPTIONAL 0x80000000
  27. #endif
  28. #define INPUT_QSIZE 32 /* Buffer up to 32 input messages */
  29. #define JOY_AXIS_THRESHOLD (((SDL_JOYSTICK_AXIS_MAX)-(SDL_JOYSTICK_AXIS_MIN))/100) /* 1% motion */
  30. #define CONVERT_MAGNITUDE(x) (((x)*10000) / 0x7FFF)
  31. /* external variables referenced. */
  32. extern HWND SDL_HelperWindow;
  33. /* local variables */
  34. static SDL_bool coinitialized = SDL_FALSE;
  35. static LPDIRECTINPUT8 dinput = NULL;
  36. static PRAWINPUTDEVICELIST SDL_RawDevList = NULL;
  37. static UINT SDL_RawDevListCount = 0;
  38. /* Taken from Wine - Thanks! */
  39. static DIOBJECTDATAFORMAT dfDIJoystick2[] = {
  40. { &GUID_XAxis, DIJOFS_X, DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  41. { &GUID_YAxis, DIJOFS_Y, DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  42. { &GUID_ZAxis, DIJOFS_Z, DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  43. { &GUID_RxAxis, DIJOFS_RX, DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  44. { &GUID_RyAxis, DIJOFS_RY, DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  45. { &GUID_RzAxis, DIJOFS_RZ, DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  46. { &GUID_Slider, DIJOFS_SLIDER(0), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  47. { &GUID_Slider, DIJOFS_SLIDER(1), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  48. { &GUID_POV, DIJOFS_POV(0), DIDFT_OPTIONAL | DIDFT_POV | DIDFT_ANYINSTANCE, 0 },
  49. { &GUID_POV, DIJOFS_POV(1), DIDFT_OPTIONAL | DIDFT_POV | DIDFT_ANYINSTANCE, 0 },
  50. { &GUID_POV, DIJOFS_POV(2), DIDFT_OPTIONAL | DIDFT_POV | DIDFT_ANYINSTANCE, 0 },
  51. { &GUID_POV, DIJOFS_POV(3), DIDFT_OPTIONAL | DIDFT_POV | DIDFT_ANYINSTANCE, 0 },
  52. { NULL, DIJOFS_BUTTON(0), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  53. { NULL, DIJOFS_BUTTON(1), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  54. { NULL, DIJOFS_BUTTON(2), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  55. { NULL, DIJOFS_BUTTON(3), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  56. { NULL, DIJOFS_BUTTON(4), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  57. { NULL, DIJOFS_BUTTON(5), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  58. { NULL, DIJOFS_BUTTON(6), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  59. { NULL, DIJOFS_BUTTON(7), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  60. { NULL, DIJOFS_BUTTON(8), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  61. { NULL, DIJOFS_BUTTON(9), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  62. { NULL, DIJOFS_BUTTON(10), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  63. { NULL, DIJOFS_BUTTON(11), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  64. { NULL, DIJOFS_BUTTON(12), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  65. { NULL, DIJOFS_BUTTON(13), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  66. { NULL, DIJOFS_BUTTON(14), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  67. { NULL, DIJOFS_BUTTON(15), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  68. { NULL, DIJOFS_BUTTON(16), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  69. { NULL, DIJOFS_BUTTON(17), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  70. { NULL, DIJOFS_BUTTON(18), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  71. { NULL, DIJOFS_BUTTON(19), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  72. { NULL, DIJOFS_BUTTON(20), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  73. { NULL, DIJOFS_BUTTON(21), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  74. { NULL, DIJOFS_BUTTON(22), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  75. { NULL, DIJOFS_BUTTON(23), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  76. { NULL, DIJOFS_BUTTON(24), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  77. { NULL, DIJOFS_BUTTON(25), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  78. { NULL, DIJOFS_BUTTON(26), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  79. { NULL, DIJOFS_BUTTON(27), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  80. { NULL, DIJOFS_BUTTON(28), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  81. { NULL, DIJOFS_BUTTON(29), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  82. { NULL, DIJOFS_BUTTON(30), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  83. { NULL, DIJOFS_BUTTON(31), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  84. { NULL, DIJOFS_BUTTON(32), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  85. { NULL, DIJOFS_BUTTON(33), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  86. { NULL, DIJOFS_BUTTON(34), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  87. { NULL, DIJOFS_BUTTON(35), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  88. { NULL, DIJOFS_BUTTON(36), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  89. { NULL, DIJOFS_BUTTON(37), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  90. { NULL, DIJOFS_BUTTON(38), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  91. { NULL, DIJOFS_BUTTON(39), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  92. { NULL, DIJOFS_BUTTON(40), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  93. { NULL, DIJOFS_BUTTON(41), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  94. { NULL, DIJOFS_BUTTON(42), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  95. { NULL, DIJOFS_BUTTON(43), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  96. { NULL, DIJOFS_BUTTON(44), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  97. { NULL, DIJOFS_BUTTON(45), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  98. { NULL, DIJOFS_BUTTON(46), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  99. { NULL, DIJOFS_BUTTON(47), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  100. { NULL, DIJOFS_BUTTON(48), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  101. { NULL, DIJOFS_BUTTON(49), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  102. { NULL, DIJOFS_BUTTON(50), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  103. { NULL, DIJOFS_BUTTON(51), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  104. { NULL, DIJOFS_BUTTON(52), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  105. { NULL, DIJOFS_BUTTON(53), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  106. { NULL, DIJOFS_BUTTON(54), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  107. { NULL, DIJOFS_BUTTON(55), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  108. { NULL, DIJOFS_BUTTON(56), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  109. { NULL, DIJOFS_BUTTON(57), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  110. { NULL, DIJOFS_BUTTON(58), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  111. { NULL, DIJOFS_BUTTON(59), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  112. { NULL, DIJOFS_BUTTON(60), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  113. { NULL, DIJOFS_BUTTON(61), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  114. { NULL, DIJOFS_BUTTON(62), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  115. { NULL, DIJOFS_BUTTON(63), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  116. { NULL, DIJOFS_BUTTON(64), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  117. { NULL, DIJOFS_BUTTON(65), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  118. { NULL, DIJOFS_BUTTON(66), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  119. { NULL, DIJOFS_BUTTON(67), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  120. { NULL, DIJOFS_BUTTON(68), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  121. { NULL, DIJOFS_BUTTON(69), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  122. { NULL, DIJOFS_BUTTON(70), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  123. { NULL, DIJOFS_BUTTON(71), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  124. { NULL, DIJOFS_BUTTON(72), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  125. { NULL, DIJOFS_BUTTON(73), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  126. { NULL, DIJOFS_BUTTON(74), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  127. { NULL, DIJOFS_BUTTON(75), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  128. { NULL, DIJOFS_BUTTON(76), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  129. { NULL, DIJOFS_BUTTON(77), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  130. { NULL, DIJOFS_BUTTON(78), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  131. { NULL, DIJOFS_BUTTON(79), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  132. { NULL, DIJOFS_BUTTON(80), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  133. { NULL, DIJOFS_BUTTON(81), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  134. { NULL, DIJOFS_BUTTON(82), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  135. { NULL, DIJOFS_BUTTON(83), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  136. { NULL, DIJOFS_BUTTON(84), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  137. { NULL, DIJOFS_BUTTON(85), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  138. { NULL, DIJOFS_BUTTON(86), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  139. { NULL, DIJOFS_BUTTON(87), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  140. { NULL, DIJOFS_BUTTON(88), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  141. { NULL, DIJOFS_BUTTON(89), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  142. { NULL, DIJOFS_BUTTON(90), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  143. { NULL, DIJOFS_BUTTON(91), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  144. { NULL, DIJOFS_BUTTON(92), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  145. { NULL, DIJOFS_BUTTON(93), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  146. { NULL, DIJOFS_BUTTON(94), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  147. { NULL, DIJOFS_BUTTON(95), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  148. { NULL, DIJOFS_BUTTON(96), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  149. { NULL, DIJOFS_BUTTON(97), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  150. { NULL, DIJOFS_BUTTON(98), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  151. { NULL, DIJOFS_BUTTON(99), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  152. { NULL, DIJOFS_BUTTON(100), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  153. { NULL, DIJOFS_BUTTON(101), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  154. { NULL, DIJOFS_BUTTON(102), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  155. { NULL, DIJOFS_BUTTON(103), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  156. { NULL, DIJOFS_BUTTON(104), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  157. { NULL, DIJOFS_BUTTON(105), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  158. { NULL, DIJOFS_BUTTON(106), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  159. { NULL, DIJOFS_BUTTON(107), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  160. { NULL, DIJOFS_BUTTON(108), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  161. { NULL, DIJOFS_BUTTON(109), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  162. { NULL, DIJOFS_BUTTON(110), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  163. { NULL, DIJOFS_BUTTON(111), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  164. { NULL, DIJOFS_BUTTON(112), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  165. { NULL, DIJOFS_BUTTON(113), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  166. { NULL, DIJOFS_BUTTON(114), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  167. { NULL, DIJOFS_BUTTON(115), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  168. { NULL, DIJOFS_BUTTON(116), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  169. { NULL, DIJOFS_BUTTON(117), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  170. { NULL, DIJOFS_BUTTON(118), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  171. { NULL, DIJOFS_BUTTON(119), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  172. { NULL, DIJOFS_BUTTON(120), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  173. { NULL, DIJOFS_BUTTON(121), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  174. { NULL, DIJOFS_BUTTON(122), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  175. { NULL, DIJOFS_BUTTON(123), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  176. { NULL, DIJOFS_BUTTON(124), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  177. { NULL, DIJOFS_BUTTON(125), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  178. { NULL, DIJOFS_BUTTON(126), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  179. { NULL, DIJOFS_BUTTON(127), DIDFT_OPTIONAL | DIDFT_BUTTON | DIDFT_ANYINSTANCE, 0 },
  180. { &GUID_XAxis, FIELD_OFFSET(DIJOYSTATE2, lVX), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  181. { &GUID_YAxis, FIELD_OFFSET(DIJOYSTATE2, lVY), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  182. { &GUID_ZAxis, FIELD_OFFSET(DIJOYSTATE2, lVZ), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  183. { &GUID_RxAxis, FIELD_OFFSET(DIJOYSTATE2, lVRx), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  184. { &GUID_RyAxis, FIELD_OFFSET(DIJOYSTATE2, lVRy), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  185. { &GUID_RzAxis, FIELD_OFFSET(DIJOYSTATE2, lVRz), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  186. { &GUID_Slider, FIELD_OFFSET(DIJOYSTATE2, rglVSlider[0]), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  187. { &GUID_Slider, FIELD_OFFSET(DIJOYSTATE2, rglVSlider[1]), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  188. { &GUID_XAxis, FIELD_OFFSET(DIJOYSTATE2, lAX), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  189. { &GUID_YAxis, FIELD_OFFSET(DIJOYSTATE2, lAY), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  190. { &GUID_ZAxis, FIELD_OFFSET(DIJOYSTATE2, lAZ), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  191. { &GUID_RxAxis, FIELD_OFFSET(DIJOYSTATE2, lARx), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  192. { &GUID_RyAxis, FIELD_OFFSET(DIJOYSTATE2, lARy), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  193. { &GUID_RzAxis, FIELD_OFFSET(DIJOYSTATE2, lARz), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  194. { &GUID_Slider, FIELD_OFFSET(DIJOYSTATE2, rglASlider[0]), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  195. { &GUID_Slider, FIELD_OFFSET(DIJOYSTATE2, rglASlider[1]), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  196. { &GUID_XAxis, FIELD_OFFSET(DIJOYSTATE2, lFX), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  197. { &GUID_YAxis, FIELD_OFFSET(DIJOYSTATE2, lFY), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  198. { &GUID_ZAxis, FIELD_OFFSET(DIJOYSTATE2, lFZ), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  199. { &GUID_RxAxis, FIELD_OFFSET(DIJOYSTATE2, lFRx), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  200. { &GUID_RyAxis, FIELD_OFFSET(DIJOYSTATE2, lFRy), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  201. { &GUID_RzAxis, FIELD_OFFSET(DIJOYSTATE2, lFRz), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  202. { &GUID_Slider, FIELD_OFFSET(DIJOYSTATE2, rglFSlider[0]), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  203. { &GUID_Slider, FIELD_OFFSET(DIJOYSTATE2, rglFSlider[1]), DIDFT_OPTIONAL | DIDFT_AXIS | DIDFT_ANYINSTANCE, 0 },
  204. };
  205. const DIDATAFORMAT SDL_c_dfDIJoystick2 = {
  206. sizeof(DIDATAFORMAT),
  207. sizeof(DIOBJECTDATAFORMAT),
  208. DIDF_ABSAXIS,
  209. sizeof(DIJOYSTATE2),
  210. SDL_arraysize(dfDIJoystick2),
  211. dfDIJoystick2
  212. };
  213. /* Convert a DirectInput return code to a text message */
  214. static int
  215. SetDIerror(const char *function, HRESULT code)
  216. {
  217. /*
  218. return SDL_SetError("%s() [%s]: %s", function,
  219. DXGetErrorString9A(code), DXGetErrorDescription9A(code));
  220. */
  221. return SDL_SetError("%s() DirectX error 0x%8.8lx", function, code);
  222. }
  223. #if 0 /* Microsoft recommended implementation, but slower than checking raw devices */
  224. #define COBJMACROS
  225. #include <wbemidl.h>
  226. #include <oleauto.h>
  227. static const IID CLSID_WbemLocator = { 0x4590f811, 0x1d3a, 0x11d0,{ 0x89, 0x1f, 0x00, 0xaa, 0x00, 0x4b, 0x2e, 0x24 } };
  228. static const IID IID_IWbemLocator = { 0xdc12a687, 0x737f, 0x11cf,{ 0x88, 0x4d, 0x00, 0xaa, 0x00, 0x4b, 0x2e, 0x24 } };
  229. static SDL_bool
  230. WIN_IsXInputDevice(const WCHAR *name, const GUID* pGuidProductFromDirectInput)
  231. {
  232. IWbemLocator* pIWbemLocator = NULL;
  233. IEnumWbemClassObject* pEnumDevices = NULL;
  234. IWbemClassObject* pDevices[20];
  235. IWbemServices* pIWbemServices = NULL;
  236. BSTR bstrNamespace = NULL;
  237. BSTR bstrDeviceID = NULL;
  238. BSTR bstrClassName = NULL;
  239. DWORD uReturned = 0;
  240. SDL_bool bIsXinputDevice = SDL_FALSE;
  241. UINT iDevice = 0;
  242. VARIANT var;
  243. HRESULT hr;
  244. if (!SDL_XINPUT_Enabled()) {
  245. return SDL_FALSE;
  246. }
  247. if (SDL_wcsstr(name, L" XINPUT ") != NULL) {
  248. /* This is a duplicate interface for a controller that will show up with XInput,
  249. e.g. Xbox One Elite Series 2 in Bluetooth mode.
  250. */
  251. return SDL_TRUE;
  252. }
  253. SDL_zeroa(pDevices);
  254. // Create WMI
  255. hr = CoCreateInstance(&CLSID_WbemLocator,
  256. NULL,
  257. CLSCTX_INPROC_SERVER,
  258. &IID_IWbemLocator,
  259. (LPVOID*)&pIWbemLocator);
  260. if (FAILED(hr) || pIWbemLocator == NULL)
  261. goto LCleanup;
  262. bstrNamespace = SysAllocString(L"\\\\.\\root\\cimv2"); if (bstrNamespace == NULL) goto LCleanup;
  263. bstrClassName = SysAllocString(L"Win32_PNPEntity"); if (bstrClassName == NULL) goto LCleanup;
  264. bstrDeviceID = SysAllocString(L"DeviceID"); if (bstrDeviceID == NULL) goto LCleanup;
  265. // Connect to WMI
  266. hr = IWbemLocator_ConnectServer(pIWbemLocator, bstrNamespace, NULL, NULL, 0L,
  267. 0L, NULL, NULL, &pIWbemServices);
  268. if (FAILED(hr) || pIWbemServices == NULL) {
  269. goto LCleanup;
  270. }
  271. // Switch security level to IMPERSONATE.
  272. CoSetProxyBlanket((IUnknown *)pIWbemServices, RPC_C_AUTHN_WINNT, RPC_C_AUTHZ_NONE, NULL,
  273. RPC_C_AUTHN_LEVEL_CALL, RPC_C_IMP_LEVEL_IMPERSONATE, NULL, EOAC_NONE);
  274. hr = IWbemServices_CreateInstanceEnum(pIWbemServices, bstrClassName, 0, NULL, &pEnumDevices);
  275. if (FAILED(hr) || pEnumDevices == NULL)
  276. goto LCleanup;
  277. // Loop over all devices
  278. for (;;) {
  279. // Get 20 at a time
  280. hr = IEnumWbemClassObject_Next(pEnumDevices, 10000, SDL_arraysize(pDevices), pDevices, &uReturned);
  281. if (FAILED(hr)) {
  282. goto LCleanup;
  283. }
  284. if (uReturned == 0) {
  285. break;
  286. }
  287. for (iDevice = 0; iDevice < uReturned; iDevice++) {
  288. // For each device, get its device ID
  289. hr = IWbemClassObject_Get(pDevices[iDevice], bstrDeviceID, 0L, &var, NULL, NULL);
  290. if (SUCCEEDED(hr) && var.vt == VT_BSTR && var.bstrVal != NULL) {
  291. // Check if the device ID contains "IG_". If it does, then it's an XInput device
  292. // This information can not be found from DirectInput
  293. if (SDL_wcsstr(var.bstrVal, L"IG_")) {
  294. char *bstrVal = WIN_StringToUTF8(var.bstrVal);
  295. // If it does, then get the VID/PID from var.bstrVal
  296. DWORD dwPid = 0, dwVid = 0, dwVidPid;
  297. const char *strVid, *strPid;
  298. strVid = SDL_strstr(bstrVal, "VID_");
  299. if (strVid && SDL_sscanf(strVid, "VID_%4X", &dwVid) != 1)
  300. dwVid = 0;
  301. strPid = SDL_strstr(bstrVal, "PID_");
  302. if (strPid && SDL_sscanf(strPid, "PID_%4X", &dwPid) != 1)
  303. dwPid = 0;
  304. SDL_free(bstrVal);
  305. // Compare the VID/PID to the DInput device
  306. dwVidPid = MAKELONG(dwVid, dwPid);
  307. if (dwVidPid == pGuidProductFromDirectInput->Data1) {
  308. bIsXinputDevice = SDL_TRUE;
  309. goto LCleanup;
  310. }
  311. }
  312. }
  313. IWbemClassObject_Release(pDevices[iDevice]);
  314. }
  315. }
  316. LCleanup:
  317. if (bstrNamespace) {
  318. SysFreeString(bstrNamespace);
  319. }
  320. if (bstrDeviceID) {
  321. SysFreeString(bstrDeviceID);
  322. }
  323. if (bstrClassName) {
  324. SysFreeString(bstrClassName);
  325. }
  326. for (iDevice = 0; iDevice < SDL_arraysize(pDevices); iDevice++) {
  327. if (pDevices[iDevice]) {
  328. IWbemClassObject_Release(pDevices[iDevice]);
  329. }
  330. }
  331. if (pEnumDevices) {
  332. IEnumWbemClassObject_Release(pEnumDevices);
  333. }
  334. if (pIWbemLocator) {
  335. IWbemLocator_Release(pIWbemLocator);
  336. }
  337. if (pIWbemServices) {
  338. IWbemServices_Release(pIWbemServices);
  339. }
  340. return bIsXinputDevice;
  341. }
  342. #endif /* 0 */
  343. static SDL_bool
  344. SDL_IsXInputDevice(const WCHAR *name, const GUID* pGuidProductFromDirectInput)
  345. {
  346. UINT i;
  347. if (!SDL_XINPUT_Enabled()) {
  348. return SDL_FALSE;
  349. }
  350. if (SDL_wcsstr(name, L" XINPUT ") != NULL) {
  351. /* This is a duplicate interface for a controller that will show up with XInput,
  352. e.g. Xbox One Elite Series 2 in Bluetooth mode.
  353. */
  354. return SDL_TRUE;
  355. }
  356. if (SDL_memcmp(&pGuidProductFromDirectInput->Data4[2], "PIDVID", 6) == 0) {
  357. Uint16 vendor_id = (Uint16)LOWORD(pGuidProductFromDirectInput->Data1);
  358. Uint16 product_id = (Uint16)HIWORD(pGuidProductFromDirectInput->Data1);
  359. SDL_GameControllerType type = SDL_GetJoystickGameControllerType("", vendor_id, product_id, -1, 0, 0, 0);
  360. if (type == SDL_CONTROLLER_TYPE_XBOX360 ||
  361. type == SDL_CONTROLLER_TYPE_XBOXONE ||
  362. (vendor_id == 0x28DE && product_id == 0x11FF)) {
  363. return SDL_TRUE;
  364. }
  365. }
  366. /* Go through RAWINPUT (WinXP and later) to find HID devices. */
  367. /* Cache this if we end up using it. */
  368. if (SDL_RawDevList == NULL) {
  369. if ((GetRawInputDeviceList(NULL, &SDL_RawDevListCount, sizeof(RAWINPUTDEVICELIST)) == -1) || (!SDL_RawDevListCount)) {
  370. return SDL_FALSE; /* oh well. */
  371. }
  372. SDL_RawDevList = (PRAWINPUTDEVICELIST)SDL_malloc(sizeof(RAWINPUTDEVICELIST) * SDL_RawDevListCount);
  373. if (SDL_RawDevList == NULL) {
  374. SDL_OutOfMemory();
  375. return SDL_FALSE;
  376. }
  377. if (GetRawInputDeviceList(SDL_RawDevList, &SDL_RawDevListCount, sizeof(RAWINPUTDEVICELIST)) == -1) {
  378. SDL_free(SDL_RawDevList);
  379. SDL_RawDevList = NULL;
  380. return SDL_FALSE; /* oh well. */
  381. }
  382. }
  383. for (i = 0; i < SDL_RawDevListCount; i++) {
  384. RID_DEVICE_INFO rdi;
  385. char devName[128];
  386. UINT rdiSize = sizeof(rdi);
  387. UINT nameSize = SDL_arraysize(devName);
  388. rdi.cbSize = sizeof(rdi);
  389. if ((SDL_RawDevList[i].dwType == RIM_TYPEHID) &&
  390. (GetRawInputDeviceInfoA(SDL_RawDevList[i].hDevice, RIDI_DEVICEINFO, &rdi, &rdiSize) != ((UINT)-1)) &&
  391. (MAKELONG(rdi.hid.dwVendorId, rdi.hid.dwProductId) == ((LONG)pGuidProductFromDirectInput->Data1)) &&
  392. (GetRawInputDeviceInfoA(SDL_RawDevList[i].hDevice, RIDI_DEVICENAME, devName, &nameSize) != ((UINT)-1)) &&
  393. (SDL_strstr(devName, "IG_") != NULL)) {
  394. return SDL_TRUE;
  395. }
  396. }
  397. return SDL_FALSE;
  398. }
  399. void FreeRumbleEffectData(DIEFFECT *effect)
  400. {
  401. if (!effect) {
  402. return;
  403. }
  404. SDL_free(effect->rgdwAxes);
  405. SDL_free(effect->rglDirection);
  406. SDL_free(effect->lpvTypeSpecificParams);
  407. SDL_free(effect);
  408. }
  409. DIEFFECT *CreateRumbleEffectData(Sint16 magnitude)
  410. {
  411. DIEFFECT *effect;
  412. DIPERIODIC *periodic;
  413. /* Create the effect */
  414. effect = (DIEFFECT *)SDL_calloc(1, sizeof(*effect));
  415. if (!effect) {
  416. return NULL;
  417. }
  418. effect->dwSize = sizeof(*effect);
  419. effect->dwGain = 10000;
  420. effect->dwFlags = DIEFF_OBJECTOFFSETS;
  421. effect->dwDuration = SDL_MAX_RUMBLE_DURATION_MS * 1000; /* In microseconds. */
  422. effect->dwTriggerButton = DIEB_NOTRIGGER;
  423. effect->cAxes = 2;
  424. effect->rgdwAxes = (DWORD *)SDL_calloc(effect->cAxes, sizeof(DWORD));
  425. if (!effect->rgdwAxes) {
  426. FreeRumbleEffectData(effect);
  427. return NULL;
  428. }
  429. effect->rglDirection = (LONG *)SDL_calloc(effect->cAxes, sizeof(LONG));
  430. if (!effect->rglDirection) {
  431. FreeRumbleEffectData(effect);
  432. return NULL;
  433. }
  434. effect->dwFlags |= DIEFF_CARTESIAN;
  435. periodic = (DIPERIODIC *)SDL_calloc(1, sizeof(*periodic));
  436. if (!periodic) {
  437. FreeRumbleEffectData(effect);
  438. return NULL;
  439. }
  440. periodic->dwMagnitude = CONVERT_MAGNITUDE(magnitude);
  441. periodic->dwPeriod = 1000000;
  442. effect->cbTypeSpecificParams = sizeof(*periodic);
  443. effect->lpvTypeSpecificParams = periodic;
  444. return effect;
  445. }
  446. int
  447. SDL_DINPUT_JoystickInit(void)
  448. {
  449. HRESULT result;
  450. HINSTANCE instance;
  451. result = WIN_CoInitialize();
  452. if (FAILED(result)) {
  453. return SetDIerror("CoInitialize", result);
  454. }
  455. coinitialized = SDL_TRUE;
  456. result = CoCreateInstance(&CLSID_DirectInput8, NULL, CLSCTX_INPROC_SERVER,
  457. &IID_IDirectInput8, (LPVOID)&dinput);
  458. if (FAILED(result)) {
  459. return SetDIerror("CoCreateInstance", result);
  460. }
  461. /* Because we used CoCreateInstance, we need to Initialize it, first. */
  462. instance = GetModuleHandle(NULL);
  463. if (instance == NULL) {
  464. return SDL_SetError("GetModuleHandle() failed with error code %lu.", GetLastError());
  465. }
  466. result = IDirectInput8_Initialize(dinput, instance, DIRECTINPUT_VERSION);
  467. if (FAILED(result)) {
  468. return SetDIerror("IDirectInput::Initialize", result);
  469. }
  470. return 0;
  471. }
  472. /* helper function for direct input, gets called for each connected joystick */
  473. static BOOL CALLBACK
  474. EnumJoysticksCallback(const DIDEVICEINSTANCE * pdidInstance, VOID * pContext)
  475. {
  476. JoyStick_DeviceData *pNewJoystick;
  477. JoyStick_DeviceData *pPrevJoystick = NULL;
  478. const DWORD devtype = (pdidInstance->dwDevType & 0xFF);
  479. Uint16 *guid16;
  480. Uint16 vendor = 0;
  481. Uint16 product = 0;
  482. Uint16 version = 0;
  483. WCHAR hidPath[MAX_PATH];
  484. char *name;
  485. if (devtype == DI8DEVTYPE_SUPPLEMENTAL) {
  486. /* Add any supplemental devices that should be ignored here */
  487. #define MAKE_TABLE_ENTRY(VID, PID) ((((DWORD)PID)<<16)|VID)
  488. static DWORD ignored_devices[] = {
  489. MAKE_TABLE_ENTRY(0, 0)
  490. };
  491. #undef MAKE_TABLE_ENTRY
  492. unsigned int i;
  493. for (i = 0; i < SDL_arraysize(ignored_devices); ++i) {
  494. if (pdidInstance->guidProduct.Data1 == ignored_devices[i]) {
  495. return DIENUM_CONTINUE;
  496. }
  497. }
  498. }
  499. if (SDL_IsXInputDevice(pdidInstance->tszProductName, &pdidInstance->guidProduct)) {
  500. return DIENUM_CONTINUE; /* ignore XInput devices here, keep going. */
  501. }
  502. {
  503. HRESULT result;
  504. LPDIRECTINPUTDEVICE8 device;
  505. LPDIRECTINPUTDEVICE8 InputDevice;
  506. DIPROPGUIDANDPATH dipdw2;
  507. result = IDirectInput8_CreateDevice(dinput, &(pdidInstance->guidInstance), &device, NULL);
  508. if (FAILED(result)) {
  509. return DIENUM_CONTINUE; /* better luck next time? */
  510. }
  511. /* Now get the IDirectInputDevice8 interface, instead. */
  512. result = IDirectInputDevice8_QueryInterface(device, &IID_IDirectInputDevice8, (LPVOID *)&InputDevice);
  513. /* We are done with this object. Use the stored one from now on. */
  514. IDirectInputDevice8_Release(device);
  515. if (FAILED(result)) {
  516. return DIENUM_CONTINUE; /* better luck next time? */
  517. }
  518. dipdw2.diph.dwSize = sizeof(dipdw2);
  519. dipdw2.diph.dwHeaderSize = sizeof(dipdw2.diph);
  520. dipdw2.diph.dwObj = 0; // device property
  521. dipdw2.diph.dwHow = DIPH_DEVICE;
  522. result = IDirectInputDevice8_GetProperty(InputDevice, DIPROP_GUIDANDPATH, &dipdw2.diph);
  523. IDirectInputDevice8_Release(InputDevice);
  524. if (FAILED(result)) {
  525. return DIENUM_CONTINUE; /* better luck next time? */
  526. }
  527. /* Get device path, compare that instead of GUID, additionally update GUIDs of joysticks with matching paths, in case they're not open yet. */
  528. SDL_wcslcpy(hidPath, dipdw2.wszPath, SDL_arraysize(hidPath));
  529. }
  530. pNewJoystick = *(JoyStick_DeviceData **)pContext;
  531. while (pNewJoystick) {
  532. if (SDL_wcscmp(pNewJoystick->hidPath, hidPath) == 0) {
  533. /* if we are replacing the front of the list then update it */
  534. if (pNewJoystick == *(JoyStick_DeviceData **)pContext) {
  535. *(JoyStick_DeviceData **)pContext = pNewJoystick->pNext;
  536. } else if (pPrevJoystick) {
  537. pPrevJoystick->pNext = pNewJoystick->pNext;
  538. }
  539. // Update with new guid/etc, if it has changed
  540. pNewJoystick->dxdevice = *pdidInstance;
  541. pNewJoystick->pNext = SYS_Joystick;
  542. SYS_Joystick = pNewJoystick;
  543. return DIENUM_CONTINUE; /* already have this joystick loaded, just keep going */
  544. }
  545. pPrevJoystick = pNewJoystick;
  546. pNewJoystick = pNewJoystick->pNext;
  547. }
  548. pNewJoystick = (JoyStick_DeviceData *)SDL_malloc(sizeof(JoyStick_DeviceData));
  549. if (!pNewJoystick) {
  550. return DIENUM_CONTINUE; /* better luck next time? */
  551. }
  552. SDL_zerop(pNewJoystick);
  553. SDL_wcslcpy(pNewJoystick->hidPath, hidPath, SDL_arraysize(pNewJoystick->hidPath));
  554. SDL_memcpy(&pNewJoystick->dxdevice, pdidInstance, sizeof(DIDEVICEINSTANCE));
  555. SDL_memset(pNewJoystick->guid.data, 0, sizeof(pNewJoystick->guid.data));
  556. if (SDL_memcmp(&pdidInstance->guidProduct.Data4[2], "PIDVID", 6) == 0) {
  557. vendor = (Uint16)LOWORD(pdidInstance->guidProduct.Data1);
  558. product = (Uint16)HIWORD(pdidInstance->guidProduct.Data1);
  559. }
  560. name = WIN_StringToUTF8(pdidInstance->tszProductName);
  561. pNewJoystick->joystickname = SDL_CreateJoystickName(vendor, product, NULL, name);
  562. SDL_free(name);
  563. if (!pNewJoystick->joystickname) {
  564. SDL_free(pNewJoystick);
  565. return DIENUM_CONTINUE; /* better luck next time? */
  566. }
  567. guid16 = (Uint16 *)pNewJoystick->guid.data;
  568. if (SDL_memcmp(&pdidInstance->guidProduct.Data4[2], "PIDVID", 6) == 0) {
  569. *guid16++ = SDL_SwapLE16(SDL_HARDWARE_BUS_USB);
  570. *guid16++ = 0;
  571. *guid16++ = SDL_SwapLE16(vendor);
  572. *guid16++ = 0;
  573. *guid16++ = SDL_SwapLE16(product);
  574. *guid16++ = 0;
  575. *guid16++ = SDL_SwapLE16(version);
  576. *guid16++ = 0;
  577. } else {
  578. *guid16++ = SDL_SwapLE16(SDL_HARDWARE_BUS_BLUETOOTH);
  579. *guid16++ = 0;
  580. SDL_strlcpy((char*)guid16, pNewJoystick->joystickname, sizeof(pNewJoystick->guid.data) - 4);
  581. }
  582. if (SDL_ShouldIgnoreJoystick(pNewJoystick->joystickname, pNewJoystick->guid)) {
  583. SDL_free(pNewJoystick->joystickname);
  584. SDL_free(pNewJoystick);
  585. return DIENUM_CONTINUE;
  586. }
  587. #ifdef SDL_JOYSTICK_HIDAPI
  588. if (HIDAPI_IsDevicePresent(vendor, product, 0, pNewJoystick->joystickname)) {
  589. /* The HIDAPI driver is taking care of this device */
  590. SDL_free(pNewJoystick->joystickname);
  591. SDL_free(pNewJoystick);
  592. return DIENUM_CONTINUE;
  593. }
  594. #endif
  595. WINDOWS_AddJoystickDevice(pNewJoystick);
  596. return DIENUM_CONTINUE; /* get next device, please */
  597. }
  598. void
  599. SDL_DINPUT_JoystickDetect(JoyStick_DeviceData **pContext)
  600. {
  601. IDirectInput8_EnumDevices(dinput, DI8DEVCLASS_GAMECTRL, EnumJoysticksCallback, pContext, DIEDFL_ATTACHEDONLY);
  602. if (SDL_RawDevList) {
  603. SDL_free(SDL_RawDevList); /* in case we used this in DirectInput detection */
  604. SDL_RawDevList = NULL;
  605. }
  606. SDL_RawDevListCount = 0;
  607. }
  608. static BOOL CALLBACK
  609. EnumDevObjectsCallback(LPCDIDEVICEOBJECTINSTANCE dev, LPVOID pvRef)
  610. {
  611. SDL_Joystick *joystick = (SDL_Joystick *)pvRef;
  612. HRESULT result;
  613. input_t *in = &joystick->hwdata->Inputs[joystick->hwdata->NumInputs];
  614. if (dev->dwType & DIDFT_BUTTON) {
  615. in->type = BUTTON;
  616. in->num = joystick->nbuttons;
  617. in->ofs = DIJOFS_BUTTON(in->num);
  618. joystick->nbuttons++;
  619. } else if (dev->dwType & DIDFT_POV) {
  620. in->type = HAT;
  621. in->num = joystick->nhats;
  622. in->ofs = DIJOFS_POV(in->num);
  623. joystick->nhats++;
  624. } else if (dev->dwType & DIDFT_AXIS) {
  625. DIPROPRANGE diprg;
  626. DIPROPDWORD dilong;
  627. in->type = AXIS;
  628. in->num = joystick->naxes;
  629. if (!SDL_memcmp(&dev->guidType, &GUID_XAxis, sizeof(dev->guidType)))
  630. in->ofs = DIJOFS_X;
  631. else if (!SDL_memcmp(&dev->guidType, &GUID_YAxis, sizeof(dev->guidType)))
  632. in->ofs = DIJOFS_Y;
  633. else if (!SDL_memcmp(&dev->guidType, &GUID_ZAxis, sizeof(dev->guidType)))
  634. in->ofs = DIJOFS_Z;
  635. else if (!SDL_memcmp(&dev->guidType, &GUID_RxAxis, sizeof(dev->guidType)))
  636. in->ofs = DIJOFS_RX;
  637. else if (!SDL_memcmp(&dev->guidType, &GUID_RyAxis, sizeof(dev->guidType)))
  638. in->ofs = DIJOFS_RY;
  639. else if (!SDL_memcmp(&dev->guidType, &GUID_RzAxis, sizeof(dev->guidType)))
  640. in->ofs = DIJOFS_RZ;
  641. else if (!SDL_memcmp(&dev->guidType, &GUID_Slider, sizeof(dev->guidType))) {
  642. in->ofs = DIJOFS_SLIDER(joystick->hwdata->NumSliders);
  643. ++joystick->hwdata->NumSliders;
  644. } else {
  645. return DIENUM_CONTINUE; /* not an axis we can grok */
  646. }
  647. diprg.diph.dwSize = sizeof(diprg);
  648. diprg.diph.dwHeaderSize = sizeof(diprg.diph);
  649. diprg.diph.dwObj = dev->dwType;
  650. diprg.diph.dwHow = DIPH_BYID;
  651. diprg.lMin = SDL_JOYSTICK_AXIS_MIN;
  652. diprg.lMax = SDL_JOYSTICK_AXIS_MAX;
  653. result =
  654. IDirectInputDevice8_SetProperty(joystick->hwdata->InputDevice,
  655. DIPROP_RANGE, &diprg.diph);
  656. if (FAILED(result)) {
  657. return DIENUM_CONTINUE; /* don't use this axis */
  658. }
  659. /* Set dead zone to 0. */
  660. dilong.diph.dwSize = sizeof(dilong);
  661. dilong.diph.dwHeaderSize = sizeof(dilong.diph);
  662. dilong.diph.dwObj = dev->dwType;
  663. dilong.diph.dwHow = DIPH_BYID;
  664. dilong.dwData = 0;
  665. result =
  666. IDirectInputDevice8_SetProperty(joystick->hwdata->InputDevice,
  667. DIPROP_DEADZONE, &dilong.diph);
  668. if (FAILED(result)) {
  669. return DIENUM_CONTINUE; /* don't use this axis */
  670. }
  671. joystick->naxes++;
  672. } else {
  673. /* not supported at this time */
  674. return DIENUM_CONTINUE;
  675. }
  676. joystick->hwdata->NumInputs++;
  677. if (joystick->hwdata->NumInputs == MAX_INPUTS) {
  678. return DIENUM_STOP; /* too many */
  679. }
  680. return DIENUM_CONTINUE;
  681. }
  682. /* Sort using the data offset into the DInput struct.
  683. * This gives a reasonable ordering for the inputs.
  684. */
  685. static int
  686. SortDevFunc(const void *a, const void *b)
  687. {
  688. const input_t *inputA = (const input_t*)a;
  689. const input_t *inputB = (const input_t*)b;
  690. if (inputA->ofs < inputB->ofs)
  691. return -1;
  692. if (inputA->ofs > inputB->ofs)
  693. return 1;
  694. return 0;
  695. }
  696. /* Sort the input objects and recalculate the indices for each input. */
  697. static void
  698. SortDevObjects(SDL_Joystick *joystick)
  699. {
  700. input_t *inputs = joystick->hwdata->Inputs;
  701. int nButtons = 0;
  702. int nHats = 0;
  703. int nAxis = 0;
  704. int n;
  705. SDL_qsort(inputs, joystick->hwdata->NumInputs, sizeof(input_t), SortDevFunc);
  706. for (n = 0; n < joystick->hwdata->NumInputs; n++) {
  707. switch (inputs[n].type) {
  708. case BUTTON:
  709. inputs[n].num = nButtons;
  710. nButtons++;
  711. break;
  712. case HAT:
  713. inputs[n].num = nHats;
  714. nHats++;
  715. break;
  716. case AXIS:
  717. inputs[n].num = nAxis;
  718. nAxis++;
  719. break;
  720. }
  721. }
  722. }
  723. int
  724. SDL_DINPUT_JoystickOpen(SDL_Joystick * joystick, JoyStick_DeviceData *joystickdevice)
  725. {
  726. HRESULT result;
  727. LPDIRECTINPUTDEVICE8 device;
  728. DIPROPDWORD dipdw;
  729. joystick->hwdata->buffered = SDL_TRUE;
  730. joystick->hwdata->Capabilities.dwSize = sizeof(DIDEVCAPS);
  731. SDL_zero(dipdw);
  732. dipdw.diph.dwSize = sizeof(DIPROPDWORD);
  733. dipdw.diph.dwHeaderSize = sizeof(DIPROPHEADER);
  734. result =
  735. IDirectInput8_CreateDevice(dinput,
  736. &(joystickdevice->dxdevice.guidInstance), &device, NULL);
  737. if (FAILED(result)) {
  738. return SetDIerror("IDirectInput::CreateDevice", result);
  739. }
  740. /* Now get the IDirectInputDevice8 interface, instead. */
  741. result = IDirectInputDevice8_QueryInterface(device,
  742. &IID_IDirectInputDevice8,
  743. (LPVOID *)& joystick->
  744. hwdata->InputDevice);
  745. /* We are done with this object. Use the stored one from now on. */
  746. IDirectInputDevice8_Release(device);
  747. if (FAILED(result)) {
  748. return SetDIerror("IDirectInputDevice8::QueryInterface", result);
  749. }
  750. /* Acquire shared access. Exclusive access is required for forces,
  751. * though. */
  752. result =
  753. IDirectInputDevice8_SetCooperativeLevel(joystick->hwdata->
  754. InputDevice, SDL_HelperWindow,
  755. DISCL_EXCLUSIVE |
  756. DISCL_BACKGROUND);
  757. if (FAILED(result)) {
  758. return SetDIerror("IDirectInputDevice8::SetCooperativeLevel", result);
  759. }
  760. /* Use the extended data structure: DIJOYSTATE2. */
  761. result =
  762. IDirectInputDevice8_SetDataFormat(joystick->hwdata->InputDevice,
  763. &SDL_c_dfDIJoystick2);
  764. if (FAILED(result)) {
  765. return SetDIerror("IDirectInputDevice8::SetDataFormat", result);
  766. }
  767. /* Get device capabilities */
  768. result =
  769. IDirectInputDevice8_GetCapabilities(joystick->hwdata->InputDevice,
  770. &joystick->hwdata->Capabilities);
  771. if (FAILED(result)) {
  772. return SetDIerror("IDirectInputDevice8::GetCapabilities", result);
  773. }
  774. /* Force capable? */
  775. if (joystick->hwdata->Capabilities.dwFlags & DIDC_FORCEFEEDBACK) {
  776. result = IDirectInputDevice8_Acquire(joystick->hwdata->InputDevice);
  777. if (FAILED(result)) {
  778. return SetDIerror("IDirectInputDevice8::Acquire", result);
  779. }
  780. /* reset all actuators. */
  781. result =
  782. IDirectInputDevice8_SendForceFeedbackCommand(joystick->hwdata->
  783. InputDevice,
  784. DISFFC_RESET);
  785. /* Not necessarily supported, ignore if not supported.
  786. if (FAILED(result)) {
  787. return SetDIerror("IDirectInputDevice8::SendForceFeedbackCommand", result);
  788. }
  789. */
  790. result = IDirectInputDevice8_Unacquire(joystick->hwdata->InputDevice);
  791. if (FAILED(result)) {
  792. return SetDIerror("IDirectInputDevice8::Unacquire", result);
  793. }
  794. /* Turn on auto-centering for a ForceFeedback device (until told
  795. * otherwise). */
  796. dipdw.diph.dwObj = 0;
  797. dipdw.diph.dwHow = DIPH_DEVICE;
  798. dipdw.dwData = DIPROPAUTOCENTER_ON;
  799. result =
  800. IDirectInputDevice8_SetProperty(joystick->hwdata->InputDevice,
  801. DIPROP_AUTOCENTER, &dipdw.diph);
  802. /* Not necessarily supported, ignore if not supported.
  803. if (FAILED(result)) {
  804. return SetDIerror("IDirectInputDevice8::SetProperty", result);
  805. }
  806. */
  807. }
  808. /* What buttons and axes does it have? */
  809. IDirectInputDevice8_EnumObjects(joystick->hwdata->InputDevice,
  810. EnumDevObjectsCallback, joystick,
  811. DIDFT_BUTTON | DIDFT_AXIS | DIDFT_POV);
  812. /* Reorder the input objects. Some devices do not report the X axis as
  813. * the first axis, for example. */
  814. SortDevObjects(joystick);
  815. dipdw.diph.dwObj = 0;
  816. dipdw.diph.dwHow = DIPH_DEVICE;
  817. dipdw.dwData = INPUT_QSIZE;
  818. /* Set the buffer size */
  819. result =
  820. IDirectInputDevice8_SetProperty(joystick->hwdata->InputDevice,
  821. DIPROP_BUFFERSIZE, &dipdw.diph);
  822. if (result == DI_POLLEDDEVICE) {
  823. /* This device doesn't support buffering, so we're forced
  824. * to use less reliable polling. */
  825. joystick->hwdata->buffered = SDL_FALSE;
  826. } else if (FAILED(result)) {
  827. return SetDIerror("IDirectInputDevice8::SetProperty", result);
  828. }
  829. return 0;
  830. }
  831. static int
  832. SDL_DINPUT_JoystickInitRumble(SDL_Joystick * joystick, Sint16 magnitude)
  833. {
  834. HRESULT result;
  835. /* Reset and then enable actuators */
  836. result = IDirectInputDevice8_SendForceFeedbackCommand(joystick->hwdata->InputDevice, DISFFC_RESET);
  837. if (result == DIERR_INPUTLOST || result == DIERR_NOTEXCLUSIVEACQUIRED) {
  838. result = IDirectInputDevice8_Acquire(joystick->hwdata->InputDevice);
  839. if (SUCCEEDED(result)) {
  840. result = IDirectInputDevice8_SendForceFeedbackCommand(joystick->hwdata->InputDevice, DISFFC_RESET);
  841. }
  842. }
  843. if (FAILED(result)) {
  844. return SetDIerror("IDirectInputDevice8::SendForceFeedbackCommand(DISFFC_RESET)", result);
  845. }
  846. result = IDirectInputDevice8_SendForceFeedbackCommand(joystick->hwdata->InputDevice, DISFFC_SETACTUATORSON);
  847. if (FAILED(result)) {
  848. return SetDIerror("IDirectInputDevice8::SendForceFeedbackCommand(DISFFC_SETACTUATORSON)", result);
  849. }
  850. /* Create the effect */
  851. joystick->hwdata->ffeffect = CreateRumbleEffectData(magnitude);
  852. if (!joystick->hwdata->ffeffect) {
  853. return SDL_OutOfMemory();
  854. }
  855. result = IDirectInputDevice8_CreateEffect(joystick->hwdata->InputDevice, &GUID_Sine,
  856. joystick->hwdata->ffeffect, &joystick->hwdata->ffeffect_ref, NULL);
  857. if (FAILED(result)) {
  858. return SetDIerror("IDirectInputDevice8::CreateEffect", result);
  859. }
  860. return 0;
  861. }
  862. int
  863. SDL_DINPUT_JoystickRumble(SDL_Joystick * joystick, Uint16 low_frequency_rumble, Uint16 high_frequency_rumble)
  864. {
  865. HRESULT result;
  866. /* Scale and average the two rumble strengths */
  867. Sint16 magnitude = (Sint16)(((low_frequency_rumble / 2) + (high_frequency_rumble / 2)) / 2);
  868. if (!(joystick->hwdata->Capabilities.dwFlags & DIDC_FORCEFEEDBACK)) {
  869. return SDL_Unsupported();
  870. }
  871. if (joystick->hwdata->ff_initialized) {
  872. DIPERIODIC *periodic = ((DIPERIODIC *)joystick->hwdata->ffeffect->lpvTypeSpecificParams);
  873. periodic->dwMagnitude = CONVERT_MAGNITUDE(magnitude);
  874. result = IDirectInputEffect_SetParameters(joystick->hwdata->ffeffect_ref, joystick->hwdata->ffeffect, (DIEP_DURATION | DIEP_TYPESPECIFICPARAMS));
  875. if (result == DIERR_INPUTLOST) {
  876. result = IDirectInputDevice8_Acquire(joystick->hwdata->InputDevice);
  877. if (SUCCEEDED(result)) {
  878. result = IDirectInputEffect_SetParameters(joystick->hwdata->ffeffect_ref, joystick->hwdata->ffeffect, (DIEP_DURATION | DIEP_TYPESPECIFICPARAMS));
  879. }
  880. }
  881. if (FAILED(result)) {
  882. return SetDIerror("IDirectInputDevice8::SetParameters", result);
  883. }
  884. } else {
  885. if (SDL_DINPUT_JoystickInitRumble(joystick, magnitude) < 0) {
  886. return -1;
  887. }
  888. joystick->hwdata->ff_initialized = SDL_TRUE;
  889. }
  890. result = IDirectInputEffect_Start(joystick->hwdata->ffeffect_ref, 1, 0);
  891. if (result == DIERR_INPUTLOST || result == DIERR_NOTEXCLUSIVEACQUIRED) {
  892. result = IDirectInputDevice8_Acquire(joystick->hwdata->InputDevice);
  893. if (SUCCEEDED(result)) {
  894. result = IDirectInputEffect_Start(joystick->hwdata->ffeffect_ref, 1, 0);
  895. }
  896. }
  897. if (FAILED(result)) {
  898. return SetDIerror("IDirectInputDevice8::Start", result);
  899. }
  900. return 0;
  901. }
  902. static Uint8
  903. TranslatePOV(DWORD value)
  904. {
  905. const int HAT_VALS[] = {
  906. SDL_HAT_UP,
  907. SDL_HAT_UP | SDL_HAT_RIGHT,
  908. SDL_HAT_RIGHT,
  909. SDL_HAT_DOWN | SDL_HAT_RIGHT,
  910. SDL_HAT_DOWN,
  911. SDL_HAT_DOWN | SDL_HAT_LEFT,
  912. SDL_HAT_LEFT,
  913. SDL_HAT_UP | SDL_HAT_LEFT
  914. };
  915. if (LOWORD(value) == 0xFFFF)
  916. return SDL_HAT_CENTERED;
  917. /* Round the value up: */
  918. value += 4500 / 2;
  919. value %= 36000;
  920. value /= 4500;
  921. if (value >= 8)
  922. return SDL_HAT_CENTERED; /* shouldn't happen */
  923. return HAT_VALS[value];
  924. }
  925. static void
  926. UpdateDINPUTJoystickState_Buffered(SDL_Joystick * joystick)
  927. {
  928. int i;
  929. HRESULT result;
  930. DWORD numevents;
  931. DIDEVICEOBJECTDATA evtbuf[INPUT_QSIZE];
  932. numevents = INPUT_QSIZE;
  933. result =
  934. IDirectInputDevice8_GetDeviceData(joystick->hwdata->InputDevice,
  935. sizeof(DIDEVICEOBJECTDATA), evtbuf,
  936. &numevents, 0);
  937. if (result == DIERR_INPUTLOST || result == DIERR_NOTACQUIRED) {
  938. IDirectInputDevice8_Acquire(joystick->hwdata->InputDevice);
  939. result =
  940. IDirectInputDevice8_GetDeviceData(joystick->hwdata->InputDevice,
  941. sizeof(DIDEVICEOBJECTDATA),
  942. evtbuf, &numevents, 0);
  943. }
  944. /* Handle the events or punt */
  945. if (FAILED(result)) {
  946. return;
  947. }
  948. for (i = 0; i < (int)numevents; ++i) {
  949. int j;
  950. for (j = 0; j < joystick->hwdata->NumInputs; ++j) {
  951. const input_t *in = &joystick->hwdata->Inputs[j];
  952. if (evtbuf[i].dwOfs != in->ofs)
  953. continue;
  954. switch (in->type) {
  955. case AXIS:
  956. SDL_PrivateJoystickAxis(joystick, in->num, (Sint16)evtbuf[i].dwData);
  957. break;
  958. case BUTTON:
  959. SDL_PrivateJoystickButton(joystick, in->num,
  960. (Uint8)(evtbuf[i].dwData ? SDL_PRESSED : SDL_RELEASED));
  961. break;
  962. case HAT:
  963. {
  964. Uint8 pos = TranslatePOV(evtbuf[i].dwData);
  965. SDL_PrivateJoystickHat(joystick, in->num, pos);
  966. }
  967. break;
  968. }
  969. }
  970. }
  971. }
  972. /* Function to update the state of a joystick - called as a device poll.
  973. * This function shouldn't update the joystick structure directly,
  974. * but instead should call SDL_PrivateJoystick*() to deliver events
  975. * and update joystick device state.
  976. */
  977. static void
  978. UpdateDINPUTJoystickState_Polled(SDL_Joystick * joystick)
  979. {
  980. DIJOYSTATE2 state;
  981. HRESULT result;
  982. int i;
  983. result =
  984. IDirectInputDevice8_GetDeviceState(joystick->hwdata->InputDevice,
  985. sizeof(DIJOYSTATE2), &state);
  986. if (result == DIERR_INPUTLOST || result == DIERR_NOTACQUIRED) {
  987. IDirectInputDevice8_Acquire(joystick->hwdata->InputDevice);
  988. result =
  989. IDirectInputDevice8_GetDeviceState(joystick->hwdata->InputDevice,
  990. sizeof(DIJOYSTATE2), &state);
  991. }
  992. if (result != DI_OK) {
  993. return;
  994. }
  995. /* Set each known axis, button and POV. */
  996. for (i = 0; i < joystick->hwdata->NumInputs; ++i) {
  997. const input_t *in = &joystick->hwdata->Inputs[i];
  998. switch (in->type) {
  999. case AXIS:
  1000. switch (in->ofs) {
  1001. case DIJOFS_X:
  1002. SDL_PrivateJoystickAxis(joystick, in->num, (Sint16)state.lX);
  1003. break;
  1004. case DIJOFS_Y:
  1005. SDL_PrivateJoystickAxis(joystick, in->num, (Sint16)state.lY);
  1006. break;
  1007. case DIJOFS_Z:
  1008. SDL_PrivateJoystickAxis(joystick, in->num, (Sint16)state.lZ);
  1009. break;
  1010. case DIJOFS_RX:
  1011. SDL_PrivateJoystickAxis(joystick, in->num, (Sint16)state.lRx);
  1012. break;
  1013. case DIJOFS_RY:
  1014. SDL_PrivateJoystickAxis(joystick, in->num, (Sint16)state.lRy);
  1015. break;
  1016. case DIJOFS_RZ:
  1017. SDL_PrivateJoystickAxis(joystick, in->num, (Sint16)state.lRz);
  1018. break;
  1019. case DIJOFS_SLIDER(0):
  1020. SDL_PrivateJoystickAxis(joystick, in->num, (Sint16)state.rglSlider[0]);
  1021. break;
  1022. case DIJOFS_SLIDER(1):
  1023. SDL_PrivateJoystickAxis(joystick, in->num, (Sint16)state.rglSlider[1]);
  1024. break;
  1025. }
  1026. break;
  1027. case BUTTON:
  1028. SDL_PrivateJoystickButton(joystick, in->num,
  1029. (Uint8)(state.rgbButtons[in->ofs - DIJOFS_BUTTON0] ? SDL_PRESSED : SDL_RELEASED));
  1030. break;
  1031. case HAT:
  1032. {
  1033. Uint8 pos = TranslatePOV(state.rgdwPOV[in->ofs - DIJOFS_POV(0)]);
  1034. SDL_PrivateJoystickHat(joystick, in->num, pos);
  1035. break;
  1036. }
  1037. }
  1038. }
  1039. }
  1040. void
  1041. SDL_DINPUT_JoystickUpdate(SDL_Joystick * joystick)
  1042. {
  1043. HRESULT result;
  1044. result = IDirectInputDevice8_Poll(joystick->hwdata->InputDevice);
  1045. if (result == DIERR_INPUTLOST || result == DIERR_NOTACQUIRED) {
  1046. IDirectInputDevice8_Acquire(joystick->hwdata->InputDevice);
  1047. IDirectInputDevice8_Poll(joystick->hwdata->InputDevice);
  1048. }
  1049. if (joystick->hwdata->buffered) {
  1050. UpdateDINPUTJoystickState_Buffered(joystick);
  1051. } else {
  1052. UpdateDINPUTJoystickState_Polled(joystick);
  1053. }
  1054. }
  1055. void
  1056. SDL_DINPUT_JoystickClose(SDL_Joystick * joystick)
  1057. {
  1058. if (joystick->hwdata->ffeffect_ref) {
  1059. IDirectInputEffect_Unload(joystick->hwdata->ffeffect_ref);
  1060. joystick->hwdata->ffeffect_ref = NULL;
  1061. }
  1062. if (joystick->hwdata->ffeffect) {
  1063. FreeRumbleEffectData(joystick->hwdata->ffeffect);
  1064. joystick->hwdata->ffeffect = NULL;
  1065. }
  1066. IDirectInputDevice8_Unacquire(joystick->hwdata->InputDevice);
  1067. IDirectInputDevice8_Release(joystick->hwdata->InputDevice);
  1068. joystick->hwdata->ff_initialized = SDL_FALSE;
  1069. }
  1070. void
  1071. SDL_DINPUT_JoystickQuit(void)
  1072. {
  1073. if (dinput != NULL) {
  1074. IDirectInput8_Release(dinput);
  1075. dinput = NULL;
  1076. }
  1077. if (coinitialized) {
  1078. WIN_CoUninitialize();
  1079. coinitialized = SDL_FALSE;
  1080. }
  1081. }
  1082. #else /* !SDL_JOYSTICK_DINPUT */
  1083. typedef struct JoyStick_DeviceData JoyStick_DeviceData;
  1084. int
  1085. SDL_DINPUT_JoystickInit(void)
  1086. {
  1087. return 0;
  1088. }
  1089. void
  1090. SDL_DINPUT_JoystickDetect(JoyStick_DeviceData **pContext)
  1091. {
  1092. }
  1093. int
  1094. SDL_DINPUT_JoystickOpen(SDL_Joystick * joystick, JoyStick_DeviceData *joystickdevice)
  1095. {
  1096. return SDL_Unsupported();
  1097. }
  1098. int
  1099. SDL_DINPUT_JoystickRumble(SDL_Joystick * joystick, Uint16 low_frequency_rumble, Uint16 high_frequency_rumble)
  1100. {
  1101. return SDL_Unsupported();
  1102. }
  1103. void
  1104. SDL_DINPUT_JoystickUpdate(SDL_Joystick * joystick)
  1105. {
  1106. }
  1107. void
  1108. SDL_DINPUT_JoystickClose(SDL_Joystick * joystick)
  1109. {
  1110. }
  1111. void
  1112. SDL_DINPUT_JoystickQuit(void)
  1113. {
  1114. }
  1115. #endif /* SDL_JOYSTICK_DINPUT */
  1116. /* vi: set ts=4 sw=4 expandtab: */