Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1904 lines
59 KiB

  1. /*****************************************************************************
  2. *
  3. * DIHidEnm.c
  4. *
  5. * Copyright (c) 1996 Microsoft Corporation. All Rights Reserved.
  6. *
  7. * Abstract:
  8. *
  9. * Support functions for HID enumeration.
  10. *
  11. * Contents:
  12. *
  13. * DIHid_BuildHidList
  14. *
  15. *****************************************************************************/
  16. #include "dinputpr.h"
  17. /*****************************************************************************
  18. *
  19. * The sqiffle for this file.
  20. *
  21. *****************************************************************************/
  22. #define sqfl sqflHid
  23. #ifdef HID_SUPPORT
  24. /*****************************************************************************
  25. *
  26. * @doc INTERNAL
  27. *
  28. * @global PHIDDEVICELIST | g_hdl |
  29. *
  30. * List of known HID devices.
  31. *
  32. * @global DWORD | g_tmLastHIDRebuild |
  33. *
  34. * The time we last rebuilt the HID list. Zero means that the
  35. * HID list has never been rebuilt. Watch out for wraparound;
  36. * a 32-bit value rolls over after about 30 days.
  37. *
  38. *****************************************************************************/
  39. #define MSREBUILDRATE 20000 /* Twenty seconds */
  40. #define MSREBUILDRATE_FIFTH 5000 /* Two seconds */
  41. PHIDDEVICELIST g_phdl;
  42. DWORD g_tmLastHIDRebuild;
  43. TCHAR g_tszIdLastRemoved[MAX_PATH];
  44. DWORD g_tmLastRemoved = 0;
  45. TCHAR g_tszIdLastUnknown[MAX_PATH];
  46. DWORD g_tmLastUnknown = 0;
  47. #pragma BEGIN_CONST_DATA
  48. /*****************************************************************************
  49. *
  50. * @doc INTERNAL
  51. *
  52. * @func PHIDDEVICEINFO | phdiFindHIDInstanceGUID |
  53. *
  54. * Locates information given an instance GUID for a HID device.
  55. *
  56. * The parameters have already been validated.
  57. *
  58. * The DLL critical must be held across the call; once the
  59. * critical section is released, the returned pointer becomes
  60. * invalid.
  61. *
  62. * @parm IN PCGUID | pguid |
  63. *
  64. * The instance GUID to be located.
  65. *
  66. * @returns
  67. *
  68. * Pointer to the <t HIDDEVICEINFO> that describes
  69. * the device.
  70. *
  71. *****************************************************************************/
  72. PHIDDEVICEINFO EXTERNAL
  73. phdiFindHIDInstanceGUID(PCGUID pguid)
  74. {
  75. PHIDDEVICEINFO phdi;
  76. AssertF(InCrit());
  77. if(g_phdl)
  78. {
  79. int ihdi;
  80. for(ihdi = 0, phdi = g_phdl->rghdi;
  81. ihdi < g_phdl->chdi;
  82. ihdi++, phdi++)
  83. {
  84. if(IsEqualGUID(pguid, &phdi->guid) )
  85. {
  86. goto done;
  87. }
  88. }
  89. /*
  90. * Memphis Bug#68994. App does not detect USB device.
  91. * App was using product guid.
  92. * Fix: We allow match to HID guid, if product guid is specfied
  93. */
  94. for(ihdi = 0, phdi = g_phdl->rghdi;
  95. ihdi < g_phdl->chdi;
  96. ihdi++, phdi++)
  97. {
  98. if(IsEqualGUID(pguid, &phdi->guidProduct) )
  99. {
  100. RPF("Warning: Use instance GUID (NOT product GUID) to refer to a device.");
  101. goto done;
  102. }
  103. }
  104. #ifdef WINNT
  105. /*
  106. * NT Bug#351951.
  107. * If they are directly asking for one of the predefined joystick
  108. * IDs then see if we have a device mapped to that ID. If so,
  109. * pretend they asked for that GUID instead.
  110. */
  111. /*
  112. * Weakly Assert the range of predefined static joystick instance GUIDs
  113. */
  114. AssertF( ( rgGUID_Joystick[0].Data1 & 0x0f ) == 0 );
  115. AssertF( ( rgGUID_Joystick[0x0f].Data1 & 0x0f ) == 0x0f );
  116. /*
  117. * Check the GUID is the same as the first static one ignoring LS 4 bits
  118. */
  119. if( ( (pguid->Data1 & 0xf0) == (rgGUID_Joystick[0].Data1 & 0xf0) )
  120. && !memcmp( ((PBYTE)&rgGUID_Joystick)+1, ((PBYTE)pguid)+1, sizeof(*pguid) - 1 ) )
  121. {
  122. RPF("Using predefined instance GUIDs is bad and should not work!");
  123. phdi = phdiFindJoyId( pguid->Data1 & 0x0f );
  124. goto done;
  125. }
  126. #endif
  127. }
  128. phdi = 0;
  129. done:;
  130. return phdi;
  131. }
  132. /*****************************************************************************
  133. *
  134. * @doc INTERNAL
  135. *
  136. * @func HRESULT | hresFindHIDInstanceGUID |
  137. *
  138. * Locates information given an instance GUID for a HID device.
  139. *
  140. * The parameters have already been validated.
  141. *
  142. * @parm IN PCGUID | pguid |
  143. *
  144. * The instance GUID to be located.
  145. *
  146. * @parm OUT CREATEDCB * | pcdcb |
  147. *
  148. * Receives pointer to the <f CreateDcb> function for the object.
  149. *
  150. *****************************************************************************/
  151. STDMETHODIMP
  152. hresFindHIDInstanceGUID(PCGUID pguid, CREATEDCB *pcdcb)
  153. {
  154. HRESULT hres;
  155. PHIDDEVICEINFO phdi;
  156. EnterProc(hresFindHIDInstanceGUID, (_ "G", pguid));
  157. AssertF(SUCCEEDED(hresFullValidGuid(pguid, 0)));
  158. DllEnterCrit();
  159. phdi = phdiFindHIDInstanceGUID(pguid);
  160. if(phdi)
  161. {
  162. *pcdcb = CHid_New;
  163. hres = S_OK;
  164. } else
  165. {
  166. hres = DIERR_DEVICENOTREG;
  167. }
  168. DllLeaveCrit();
  169. /*
  170. * Don't use ExitOleProcPpv because that will validate that
  171. * *pcdcb == 0 if FAILED(hres), but that's not our job.
  172. */
  173. ExitOleProc();
  174. return hres;
  175. }
  176. /*****************************************************************************
  177. *
  178. * @doc INTERNAL
  179. *
  180. * @func PHIDDEVICEINFO | phdiFindHIDDeviceInterface |
  181. *
  182. * Locates information given a device interface
  183. * (in other words, a \\.\... thing) for a HID device.
  184. *
  185. * The parameters have already been validated.
  186. *
  187. * The DLL critical must be held across the call; once the
  188. * critical section is released, the returned pointer becomes
  189. * invalid.
  190. *
  191. * @parm IN LPCTSTR | ptszPath |
  192. *
  193. * The interface device to be located.
  194. *
  195. * @returns
  196. *
  197. * Pointer to the <t HIDDEVICEINFO> that describes
  198. * the device.
  199. *
  200. *****************************************************************************/
  201. PHIDDEVICEINFO EXTERNAL
  202. phdiFindHIDDeviceInterface(LPCTSTR ptszPath)
  203. {
  204. PHIDDEVICEINFO phdi;
  205. AssertF(InCrit());
  206. if(g_phdl)
  207. {
  208. int ihdi;
  209. for(ihdi = 0, phdi = g_phdl->rghdi; ihdi < g_phdl->chdi;
  210. ihdi++, phdi++)
  211. {
  212. if(phdi->pdidd &&
  213. lstrcmpi(phdi->pdidd->DevicePath, ptszPath) == 0)
  214. {
  215. goto done;
  216. }
  217. }
  218. }
  219. phdi = 0;
  220. done:;
  221. return phdi;
  222. }
  223. /*****************************************************************************
  224. *
  225. * @doc INTERNAL
  226. *
  227. * @func HRESULT | hresFindHIDDeviceInterface |
  228. *
  229. * Locates information given a device interface
  230. * (in other words, a \\.\... thing) for a HID device.
  231. *
  232. * The parameters have already been validated.
  233. *
  234. * @parm IN LPCTSTR | ptszPath |
  235. *
  236. * The interface device to be located.
  237. *
  238. * @parm OUT LPGUID | pguidOut |
  239. *
  240. * Receives the instance GUID of the device found.
  241. *
  242. *****************************************************************************/
  243. STDMETHODIMP
  244. hresFindHIDDeviceInterface(LPCTSTR ptszPath, LPGUID pguidOut)
  245. {
  246. HRESULT hres;
  247. PHIDDEVICEINFO phdi;
  248. EnterProc(hresFindHIDDeviceInterface, (_ "s", ptszPath));
  249. DllEnterCrit();
  250. phdi = phdiFindHIDDeviceInterface(ptszPath);
  251. if(phdi)
  252. {
  253. *pguidOut = phdi->guid;
  254. hres = S_OK;
  255. } else
  256. {
  257. hres = DIERR_DEVICENOTREG;
  258. }
  259. DllLeaveCrit();
  260. ExitOleProc();
  261. return hres;
  262. }
  263. /*****************************************************************************
  264. *
  265. * @doc INTERNAL
  266. *
  267. * @func void | DIHid_ProbeMouse |
  268. *
  269. * That this function exists at all is a total hack to work
  270. * around bugs in Memphis and NT5.
  271. *
  272. * If you call GetSystemMetrics(SM_WHEELPRESENT) or
  273. * GetSystemMetrics(SM_MOUSEBUTTONS), USER32 does not
  274. * return the correct values if your HID mouse is
  275. * not the same as your PS/2 mouse (if any).
  276. *
  277. * For example, if your PS/2 mouse is a regular two-button
  278. * mouse but your HID mouse is a wheel mouse, GetSystemMetrics
  279. * will still say "No wheel, 2 buttons" even though it's wrong.
  280. *
  281. * So what we have to do is wander through all the HID mice in
  282. * the system and record the number of buttons they have,
  283. * and whether they have a wheel.
  284. *
  285. * That way, when we create a system mouse, we can take the
  286. * maximum of every supported device.
  287. *
  288. *****************************************************************************/
  289. void INTERNAL
  290. DIHid_ProbeMouse(PHIDDEVICEINFO phdi, PHIDP_CAPS pcaps,
  291. PHIDP_PREPARSED_DATA ppd)
  292. {
  293. LPVOID pvReport;
  294. HRESULT hres;
  295. /*
  296. * Get the number of buttons in the generic button page.
  297. * This is the only page the MOUHID uses.
  298. */
  299. phdi->osd.uiButtons =
  300. HidP_MaxUsageListLength(HidP_Input, HID_USAGE_PAGE_BUTTON, ppd);
  301. /*
  302. * See if there is a HID_USAGE_GENERIC_WHEEL.
  303. * This is the way that MOUHID detects a wheel.
  304. */
  305. hres = AllocCbPpv(pcaps->InputReportByteLength, &pvReport);
  306. if(SUCCEEDED(hres))
  307. {
  308. ULONG ul;
  309. NTSTATUS stat;
  310. stat = HidP_GetUsageValue(HidP_Input, HID_USAGE_PAGE_GENERIC, 0,
  311. HID_USAGE_GENERIC_WHEEL, &ul, ppd,
  312. pvReport,
  313. pcaps->InputReportByteLength);
  314. if(SUCCEEDED(stat))
  315. {
  316. phdi->osd.uiAxes = 3;
  317. }
  318. FreePv(pvReport);
  319. }
  320. }
  321. /*****************************************************************************
  322. *
  323. * @doc INTERNAL
  324. *
  325. * @func void | DIHid_ParseUsagePage |
  326. *
  327. * Parse the usage page information and create fake type
  328. * information in the old DirectX3-compatible way.
  329. *
  330. *****************************************************************************/
  331. void INTERNAL
  332. DIHid_ParseUsagePage(PHIDDEVICEINFO phdi, PHIDP_CAPS pcaps,
  333. PHIDP_PREPARSED_DATA ppd)
  334. {
  335. switch(pcaps->UsagePage)
  336. {
  337. case HID_USAGE_PAGE_GENERIC:
  338. switch(pcaps->Usage)
  339. {
  340. /*
  341. * MouHID accepts either HID_USAGE_GENERIC_MOUSE or
  342. * HID_USAGE_GENERIC_POINTER, so we will do the same.
  343. */
  344. case HID_USAGE_GENERIC_MOUSE:
  345. case HID_USAGE_GENERIC_POINTER:
  346. DIHid_ProbeMouse(phdi, pcaps, ppd);
  347. phdi->osd.dwDevType =
  348. MAKE_DIDEVICE_TYPE(DIDEVTYPE_MOUSE,
  349. DIDEVTYPEMOUSE_UNKNOWN) |
  350. DIDEVTYPE_HID;
  351. break;
  352. case HID_USAGE_GENERIC_JOYSTICK:
  353. phdi->osd.dwDevType =
  354. MAKE_DIDEVICE_TYPE(DIDEVTYPE_JOYSTICK,
  355. DIDEVTYPEJOYSTICK_UNKNOWN) |
  356. DIDEVTYPE_HID;
  357. break;
  358. case HID_USAGE_GENERIC_GAMEPAD:
  359. phdi->osd.dwDevType =
  360. MAKE_DIDEVICE_TYPE(DIDEVTYPE_JOYSTICK,
  361. DIDEVTYPEJOYSTICK_GAMEPAD) |
  362. DIDEVTYPE_HID;
  363. break;
  364. case HID_USAGE_GENERIC_KEYBOARD:
  365. phdi->osd.dwDevType =
  366. MAKE_DIDEVICE_TYPE(DIDEVTYPE_KEYBOARD,
  367. DIDEVTYPEKEYBOARD_UNKNOWN) |
  368. DIDEVTYPE_HID;
  369. break;
  370. default:
  371. phdi->osd.dwDevType = DIDEVTYPE_DEVICE | DIDEVTYPE_HID;
  372. break;
  373. }
  374. break;
  375. default:
  376. phdi->osd.dwDevType = DIDEVTYPE_DEVICE | DIDEVTYPE_HID;
  377. break;
  378. }
  379. }
  380. /*****************************************************************************
  381. *
  382. * @doc INTERNAL
  383. *
  384. * @func BOOL | DIHid_GetDevicePath |
  385. *
  386. * Obtain the path for the device. This is a simple wrapper
  387. * function to keep DIHid_BuildHidListEntry from getting too
  388. * annoying.
  389. *
  390. * This also gets the devinfo so we can get the
  391. * instance ID string for subsequent use to get the
  392. * friendly name, etc.
  393. *
  394. *****************************************************************************/
  395. BOOL EXTERNAL
  396. DIHid_GetDevicePath(HDEVINFO hdev,
  397. PSP_DEVICE_INTERFACE_DATA pdid,
  398. PSP_DEVICE_INTERFACE_DETAIL_DATA *ppdidd,
  399. OPTIONAL PSP_DEVINFO_DATA pdinf)
  400. {
  401. HRESULT hres;
  402. BOOL fRc;
  403. DWORD cbRequired;
  404. EnterProcI(DIHid_GetDevicePath, (_ "xp", hdev, pdid));
  405. AssertF(*ppdidd == 0);
  406. /*
  407. * Ask for the required size then allocate it then fill it.
  408. *
  409. * Note that we don't need to free the memory on the failure
  410. * path; our caller will do the necessary memory freeing.
  411. *
  412. * Sigh. Windows NT and Windows 98 implement
  413. * SetupDiGetDeviceInterfaceDetail differently if you are
  414. * querying for the buffer size.
  415. *
  416. * Windows 98 returns FALSE, and GetLastError() returns
  417. * ERROR_INSUFFICIENT_BUFFER.
  418. *
  419. * Windows NT returns TRUE.
  420. *
  421. * So we allow the cases either where the call succeeds or
  422. * the call fails with ERROR_INSUFFICIENT_BUFFER.
  423. */
  424. if(SetupDiGetDeviceInterfaceDetail(hdev, pdid, 0, 0,
  425. &cbRequired, 0) ||
  426. GetLastError() == ERROR_INSUFFICIENT_BUFFER)
  427. {
  428. hres = AllocCbPpv(cbRequired, ppdidd);
  429. // Keep prefix happy, manbug 29341
  430. if(SUCCEEDED(hres) && ( *ppdidd != NULL) )
  431. {
  432. PSP_DEVICE_INTERFACE_DETAIL_DATA pdidd = *ppdidd;
  433. /*
  434. * Note, The cbSize field always contains the size of the fixed
  435. * part of the data structure, not a size reflecting the
  436. * variable-length string at the end.
  437. */
  438. pdidd->cbSize = cbX(SP_DEVICE_INTERFACE_DETAIL_DATA);
  439. fRc = SetupDiGetDeviceInterfaceDetail(hdev, pdid, pdidd,
  440. cbRequired, &cbRequired, pdinf);
  441. if(!fRc)
  442. {
  443. FreePpv(ppdidd);
  444. /*
  445. * Set fRc = FALSE again, so the compiler doesn't need
  446. * to blow a register to cache the value zero.
  447. */
  448. fRc = FALSE;
  449. }
  450. } else
  451. {
  452. fRc = FALSE;
  453. }
  454. } else
  455. {
  456. SquirtSqflPtszV(sqfl | sqflError,
  457. TEXT("%hs: SetupDiGetDeviceInterfaceDetail failed 1, ")
  458. TEXT("Error = %d"),
  459. s_szProc, GetLastError());
  460. fRc = FALSE;
  461. }
  462. ExitProcF(fRc);
  463. return fRc;
  464. }
  465. /*****************************************************************************
  466. *
  467. * @doc INTERNAL
  468. *
  469. * @func BOOL | DIHid_GetDeviceInstanceId |
  470. *
  471. * Obtain the instance ID for the device.
  472. *
  473. * The instance ID allows us to get access to device
  474. * properties later.
  475. *
  476. *****************************************************************************/
  477. BOOL EXTERNAL
  478. DIHid_GetDeviceInstanceId(HDEVINFO hdev,
  479. PSP_DEVINFO_DATA pdinf, LPTSTR *pptszId)
  480. {
  481. HRESULT hres;
  482. BOOL fRc;
  483. DWORD ctchRequired;
  484. AssertF(*pptszId == 0);
  485. /*
  486. * Ask for the required size then allocate it then fill it.
  487. *
  488. * Note that we don't need to free the memory on the failure
  489. * path; our caller will do the necessary memory freeing.
  490. */
  491. if(SetupDiGetDeviceInstanceId(hdev, pdinf, NULL, 0,
  492. &ctchRequired) == 0 &&
  493. GetLastError() == ERROR_INSUFFICIENT_BUFFER)
  494. {
  495. hres = AllocCbPpv(cbCtch(ctchRequired), pptszId);
  496. if(SUCCEEDED(hres))
  497. {
  498. fRc = SetupDiGetDeviceInstanceId(hdev, pdinf, *pptszId,
  499. ctchRequired, NULL);
  500. } else
  501. {
  502. fRc = FALSE;
  503. }
  504. } else
  505. {
  506. fRc = FALSE;
  507. }
  508. return fRc;
  509. }
  510. /*****************************************************************************
  511. *
  512. * @doc INTERNAL
  513. *
  514. * @func BOOL | DIHid_GetInstanceGUID |
  515. *
  516. * Read the instance GUID from the registry, or create one if
  517. * necessary.
  518. *
  519. *****************************************************************************/
  520. BOOL INTERNAL
  521. DIHid_GetInstanceGUID(HKEY hk, LPGUID pguid)
  522. {
  523. LONG lRc;
  524. DWORD cb;
  525. cb = cbX(GUID);
  526. lRc = RegQueryValueEx(hk, TEXT("GUID"), 0, 0, (PV)pguid, &cb);
  527. if(lRc != ERROR_SUCCESS)
  528. {
  529. DICreateGuid(pguid);
  530. lRc = RegSetValueEx(hk, TEXT("GUID"), 0, REG_BINARY,
  531. (PV)pguid, cbX(GUID));
  532. }
  533. return lRc == ERROR_SUCCESS;
  534. }
  535. /*****************************************************************************
  536. *
  537. * @doc INTERNAL
  538. *
  539. * @func void | DIHid_EmptyHidList |
  540. *
  541. * Empty the list of HID devices.
  542. *
  543. * This function must be called under the DLL critical section.
  544. *
  545. *****************************************************************************/
  546. void EXTERNAL
  547. DIHid_EmptyHidList(void)
  548. {
  549. AssertF(InCrit());
  550. /*
  551. * Free all the old strings and things in the HIDDEVICEINFO's.
  552. */
  553. if(g_phdl)
  554. {
  555. int ihdi;
  556. for(ihdi = 0; ihdi < g_phdl->chdi; ihdi++)
  557. {
  558. FreePpv(&g_phdl->rghdi[ihdi].pdidd);
  559. FreePpv(&g_phdl->rghdi[ihdi].ptszId);
  560. if(g_phdl->rghdi[ihdi].hk)
  561. {
  562. RegCloseKey(g_phdl->rghdi[ihdi].hk);
  563. }
  564. }
  565. /*
  566. * We invalidated all the pointers, so make sure
  567. * nobody looks at them.
  568. */
  569. g_phdl->chdi = 0;
  570. }
  571. }
  572. /*****************************************************************************
  573. *
  574. * @doc INTERNAL
  575. *
  576. * @func void | DIHid_CheckHidList |
  577. *
  578. * Check the list of HID devices, and free whose what can not be opened
  579. *
  580. * This function must be called under the DLL critical section.
  581. *
  582. *****************************************************************************/
  583. void INTERNAL
  584. DIHid_CheckHidList(void)
  585. {
  586. HANDLE hf;
  587. EnterProcI(DIHid_CheckList, (_ "x", 0x0) );
  588. AssertF(InCrit());
  589. /*
  590. * Free all the information of the device cannot be opened
  591. */
  592. if(g_phdl)
  593. {
  594. int ihdi;
  595. for(ihdi = 0, g_phdl->chdi = 0; ihdi < g_phdl->chdiAlloc; ihdi++)
  596. {
  597. if( g_phdl->rghdi[ihdi].pdidd )
  598. {
  599. /*
  600. * Open the device
  601. */
  602. hf = CreateFile(g_phdl->rghdi[ihdi].pdidd->DevicePath,
  603. GENERIC_READ | GENERIC_WRITE,
  604. FILE_SHARE_READ | FILE_SHARE_WRITE,
  605. 0, /* no SECURITY_ATTRIBUTES */
  606. OPEN_EXISTING,
  607. 0, /* attributes */
  608. 0); /* template */
  609. if(hf == INVALID_HANDLE_VALUE)
  610. {
  611. #if 0
  612. PHIDDEVICEINFO phdi;
  613. PBUSDEVICEINFO pbdi;
  614. CloseHandle(hf);
  615. phdi = &g_phdl->rghdi[ihdi];
  616. DllEnterCrit();
  617. phdi = phdiFindHIDDeviceInterface(g_phdl->rghdi[ihdi].pdidd->DevicePath);
  618. AssertF(phdi != NULL);
  619. pbdi = pbdiFromphdi(phdi);
  620. DllLeaveCrit();
  621. if( pbdi != NULL )
  622. {
  623. if( pbdi->fDeleteIfNotConnected == TRUE )
  624. {
  625. lstrcpy( g_tszIdLastRemoved, pbdi->ptszId );
  626. g_tmLastRemoved = GetTickCount();
  627. DIBusDevice_Remove(pbdi);
  628. pbdi->fDeleteIfNotConnected = FALSE;
  629. }
  630. }
  631. #endif
  632. FreePpv(&g_phdl->rghdi[ihdi].pdidd);
  633. FreePpv(&g_phdl->rghdi[ihdi].ptszId);
  634. if(g_phdl->rghdi[ihdi].hk)
  635. {
  636. RegCloseKey(g_phdl->rghdi[ihdi].hk);
  637. }
  638. ZeroX( g_phdl->rghdi[ihdi] );
  639. SquirtSqflPtszV(sqfl | sqflError,
  640. TEXT("%hs: CreateFile(%s) failed? le=%d"),
  641. s_szProc, g_phdl->rghdi[ihdi].pdidd->DevicePath, GetLastError());
  642. /* Skip erroneous items */
  643. } else
  644. {
  645. CloseHandle(hf);
  646. g_phdl->chdi++;
  647. }
  648. }
  649. }
  650. //re-order the existing devices, put them at the front of the hid list.
  651. for(ihdi = 0; ihdi < g_phdl->chdi; ihdi++)
  652. {
  653. if( !g_phdl->rghdi[ihdi].pdidd )
  654. {
  655. int ihdi2;
  656. //find the existing device from the biggest index in the hid list.
  657. for( ihdi2 = g_phdl->chdiAlloc; ihdi2 >= ihdi+1; ihdi2-- )
  658. {
  659. if( g_phdl->rghdi[ihdi2].pdidd )
  660. {
  661. memcpy( &g_phdl->rghdi[ihdi], &g_phdl->rghdi[ihdi2], sizeof(HIDDEVICEINFO) );
  662. ZeroX( g_phdl->rghdi[ihdi2] );
  663. }
  664. }
  665. }
  666. }
  667. }
  668. ExitProc();
  669. return;
  670. }
  671. /*****************************************************************************
  672. *
  673. * @doc INTERNAL
  674. *
  675. * @func HRESULT | DIHid_CreateDeviceInstanceKeys |
  676. *
  677. * Create the keys associaciated with a particular device.
  678. *
  679. * @parm IN OUT PHIDDEVICEINFO | phdi |
  680. *
  681. * HIDDEVICEINFO we to use/update.
  682. *
  683. * @returns
  684. *
  685. * S_OK for a success
  686. * A COM error for a failure
  687. *
  688. *****************************************************************************/
  689. #define DIHES_UNDEFINED 0
  690. #define DIHES_UNKNOWN 1
  691. #define DIHES_KNOWN 2
  692. HRESULT INTERNAL
  693. DIHid_CreateDeviceInstanceKeys( PHIDDEVICEINFO phdi, PBYTE pState )
  694. {
  695. HRESULT hres;
  696. HKEY hkDin;
  697. USHORT uVid, uPid;
  698. TCHAR tszName[MAX_PATH];
  699. PTCHAR ptszInstance = NULL;
  700. EnterProcI(DIHid_CreateDeviceInstanceKeys, (_ "p", phdi));
  701. /*
  702. * Unfortunately, we need to open our registry keys before we can open
  703. * the device so rather than ask the device for its VID and PID, we
  704. * parse it from the
  705. */
  706. /*
  707. * ISSUE-2001/01/27-MarcAnd Should avoid parsing device ID ourselves
  708. * Need to find some documented way to parse out the three parts of the
  709. * device ID: class, device and instance in case the form changes.
  710. */
  711. /*
  712. * The format of the device ID is a set of strings in the form
  713. * "<class>\<device>\<instance>" with variable case.
  714. * For HID devices, the class should be "hid" and the instance is a set
  715. * of hex values separated by ampersands. The device string should be
  716. * of the form vid_9999&pid_9999 but devices exposed via a gameport can
  717. * use any string that PnP accepts.
  718. */
  719. AssertF( ( phdi->ptszId[0] == TEXT('H') ) || ( phdi->ptszId[0] == TEXT('h') ) );
  720. AssertF( ( phdi->ptszId[1] == TEXT('I') ) || ( phdi->ptszId[1] == TEXT('i') ) );
  721. AssertF( ( phdi->ptszId[2] == TEXT('D') ) || ( phdi->ptszId[2] == TEXT('d') ) );
  722. /*
  723. * Assert that a defined state and a valid VID/PID on entry must both be
  724. * true or both false (so we can use the state after getting VID/PID).
  725. */
  726. if( ( phdi->ProductID != 0 ) && ( phdi->VendorID !=0 ) )
  727. {
  728. AssertF( *pState != DIHES_UNDEFINED );
  729. }
  730. else
  731. {
  732. AssertF( *pState == DIHES_UNDEFINED );
  733. }
  734. if( phdi->ptszId[3] == TEXT('\\') )
  735. {
  736. PTCHAR ptcSrc;
  737. PTCHAR ptDevId;
  738. ptcSrc = ptDevId = &phdi->ptszId[4];
  739. if( ( phdi->ProductID != 0 ) && ( phdi->VendorID !=0 ) )
  740. {
  741. int iLen;
  742. /*
  743. * Create the key name from VID / PID (because it may be
  744. * different from the value derived from the device ID).
  745. */
  746. iLen = wsprintf(tszName, VID_PID_TEMPLATE, phdi->VendorID, phdi->ProductID);
  747. while( *ptcSrc != TEXT('\\') )
  748. {
  749. if( *ptcSrc == TEXT('\0') )
  750. {
  751. break;
  752. }
  753. ptcSrc++;
  754. }
  755. if( ( *ptcSrc != TEXT('\0') )
  756. && ( ptcSrc != ptDevId ) )
  757. {
  758. ptszInstance = &tszName[iLen+2];
  759. lstrcpy( ptszInstance, ptcSrc+1 );
  760. }
  761. }
  762. else
  763. {
  764. PTCHAR ptcDest;
  765. BOOL fFirstAmpersand = FALSE;
  766. /*
  767. * Copy the device ID (minus "HID\") so we can corrupt the copy.
  768. * Convert to upper case and find the other slash on the way.
  769. * Terminate the string at either the separator slash or a second
  770. * ampersand if one is found (VID_1234&PID_1234&COL_12 or REV_12).
  771. * These are device IDs so we're only really interested in keeping
  772. * real VID\PID style IDs in upper case so we don't care if this
  773. * is imperfect for other cases as long as it is reproducable.
  774. */
  775. for( ptcDest = tszName; *ptcSrc != TEXT('\0'); ptcSrc++, ptcDest++ )
  776. {
  777. if( ( *ptcSrc >= TEXT('a') ) && ( *ptcSrc <= TEXT('z') ) )
  778. {
  779. *ptcDest = *ptcSrc - ( TEXT('a') - TEXT('A') );
  780. }
  781. else if( *ptcSrc == TEXT('&') )
  782. {
  783. if( ( ptszInstance != NULL )
  784. || !fFirstAmpersand )
  785. {
  786. fFirstAmpersand = TRUE;
  787. *ptcDest = TEXT('&');
  788. }
  789. else
  790. {
  791. *ptcDest = TEXT('\0');
  792. }
  793. }
  794. else if( *ptcSrc != TEXT('\\' ) )
  795. {
  796. *ptcDest = *ptcSrc;
  797. }
  798. else
  799. {
  800. /*
  801. * Only one slash and not the first char
  802. */
  803. if( ( ptszInstance != NULL )
  804. || ( ptcDest == tszName ) )
  805. {
  806. ptszInstance = NULL;
  807. break;
  808. }
  809. /*
  810. * Separate the device ID and the instance
  811. */
  812. *ptcDest = TEXT('\0');
  813. ptszInstance = ptcDest + 1;
  814. }
  815. }
  816. if( ptszInstance != NULL )
  817. {
  818. #ifndef UNICODE
  819. /*
  820. * ISSUE-2001/02/06-MarcAnd Should have a ParseVIDPIDA
  821. * ParseVIDPID is going to convert this string back to ANSI ifndef UNICODE
  822. */
  823. WCHAR wszName[cbszVIDPID];
  824. AToU( wszName, cA(wszName), ptDevId );
  825. if( ( ptszInstance - tszName == cbszVIDPID )
  826. && ( ParseVIDPID( &uVid, &uPid, wszName ) ) )
  827. #else
  828. if( ( ptszInstance - tszName == cbszVIDPID )
  829. && ( ParseVIDPID( &uVid, &uPid, ptDevId ) ) )
  830. #endif
  831. {
  832. phdi->VendorID = uVid;
  833. phdi->ProductID = uPid;
  834. }
  835. else
  836. {
  837. SquirtSqflPtszV(sqfl | sqflBenign,
  838. TEXT("%hs: ID %s, len %d not VID PID"),
  839. s_szProc, ptDevId, ptszInstance - tszName );
  840. }
  841. /*
  842. * Terminate the instance string
  843. */
  844. *ptcDest = TEXT('\0');
  845. }
  846. }
  847. }
  848. if( ptszInstance == NULL )
  849. {
  850. hres = E_INVALIDARG;
  851. RPF( "Ignoring invalid device ID handle \"%s\" enumerated by system" );
  852. }
  853. else
  854. {
  855. //Open our own reg key under MediaProperties\DirectInput,
  856. //creating it if necessary.
  857. hres = hresMumbleKeyEx(HKEY_LOCAL_MACHINE,
  858. REGSTR_PATH_DITYPEPROP,
  859. DI_KEY_ALL_ACCESS,
  860. REG_OPTION_NON_VOLATILE,
  861. &hkDin);
  862. if( SUCCEEDED(hres) )
  863. {
  864. HKEY hkVidPid;
  865. hres = hresMumbleKeyEx(hkDin,
  866. (LPTSTR)tszName,
  867. DI_KEY_ALL_ACCESS,
  868. REG_OPTION_NON_VOLATILE,
  869. &hkVidPid);
  870. if( SUCCEEDED(hres) )
  871. {
  872. HKEY hkDevInsts;
  873. /*
  874. * Need to create user subkeys even if the device ID is not
  875. * VID/PID as an auto-detect device could use the auto-detect
  876. * HW ID for whatever device it detects.
  877. */
  878. //Create the key for Calibration
  879. HKEY hkCal;
  880. hres = hresMumbleKeyEx(hkVidPid,
  881. TEXT("Calibration"),
  882. DI_KEY_ALL_ACCESS,
  883. REG_OPTION_NON_VOLATILE,
  884. &hkCal);
  885. if (SUCCEEDED(hres))
  886. {
  887. //Create the key for the instance (as the user sees it).
  888. //For this, need to find out how many instances of the particular device
  889. //we have already enumerated.
  890. int ihdi;
  891. int iNum = 0;
  892. TCHAR tszInst[3];
  893. for( ihdi = 0; ihdi < g_phdl->chdi; ihdi++)
  894. {
  895. if ((g_phdl->rghdi[ihdi].VendorID == phdi->VendorID) && (g_phdl->rghdi[ihdi].ProductID == phdi->ProductID))
  896. {
  897. iNum++;
  898. }
  899. }
  900. wsprintf(tszInst, TEXT("%u"), iNum);
  901. hres = hresMumbleKeyEx(hkCal,
  902. tszInst,
  903. DI_KEY_ALL_ACCESS,
  904. REG_OPTION_NON_VOLATILE,
  905. &phdi->hk);
  906. if (SUCCEEDED(hres))
  907. {
  908. DIHid_GetInstanceGUID(phdi->hk, &phdi->guid);
  909. }
  910. else
  911. {
  912. SquirtSqflPtszV(sqfl | sqflError,
  913. TEXT("%hs: RegCreateKeyEx failed on Instance reg key"),
  914. s_szProc);
  915. }
  916. RegCloseKey(hkCal);
  917. }
  918. else
  919. {
  920. SquirtSqflPtszV(sqfl | sqflError,
  921. TEXT("%hs: RegCreateKeyEx failed on Calibration reg key"),
  922. s_szProc);
  923. }
  924. /*
  925. * Try to open the device (plug) instances to find out or
  926. * update the staus of processing on this device but don't
  927. * return any failures.
  928. */
  929. if( SUCCEEDED( hresMumbleKeyEx(hkVidPid,
  930. TEXT("DeviceInstances"),
  931. DI_KEY_ALL_ACCESS,
  932. REG_OPTION_NON_VOLATILE,
  933. &hkDevInsts) ) )
  934. {
  935. LONG lRc;
  936. if( *pState == DIHES_UNDEFINED )
  937. {
  938. DWORD cb = cbX( *pState );
  939. lRc = RegQueryValueEx( hkDevInsts,
  940. ptszInstance, 0, 0, pState, &cb );
  941. if( lRc != ERROR_SUCCESS )
  942. {
  943. SquirtSqflPtszV(sqfl | sqflBenign,
  944. TEXT("%hs: RegQueryValueEx failed (%d) to get state for instance %s"),
  945. s_szProc, lRc, ptszInstance );
  946. /*
  947. * Make sure it was not trashed in the failure
  948. */
  949. *pState = DIHES_UNDEFINED;
  950. }
  951. }
  952. else
  953. {
  954. lRc = RegSetValueEx( hkDevInsts,
  955. ptszInstance, 0, REG_BINARY, pState, cbX( *pState ) );
  956. if( lRc != ERROR_SUCCESS)
  957. {
  958. SquirtSqflPtszV(sqfl | sqflBenign,
  959. TEXT("%hs: RegSetValueEx failed (%d) to update state for instance %s"),
  960. s_szProc, lRc, ptszInstance );
  961. }
  962. }
  963. RegCloseKey(hkDevInsts);
  964. }
  965. else
  966. {
  967. SquirtSqflPtszV(sqfl | sqflBenign,
  968. TEXT("%hs: Failed to open DeviceInstances key"),
  969. s_szProc );
  970. }
  971. RegCloseKey(hkVidPid);
  972. }
  973. else
  974. {
  975. SquirtSqflPtszV(sqfl | sqflError,
  976. TEXT("%hs: RegCreateKeyEx failed on Device reg key"),
  977. s_szProc);
  978. }
  979. RegCloseKey(hkDin);
  980. }
  981. else
  982. {
  983. SquirtSqflPtszV(sqfl | sqflError,
  984. TEXT("%hs: RegOpenKeyEx failed on DirectInput reg key"),
  985. s_szProc);
  986. }
  987. }
  988. ExitOleProc();
  989. return hres;
  990. }
  991. /*****************************************************************************
  992. *
  993. * @doc INTERNAL
  994. *
  995. * @func void | DIHid_GetDevInfo |
  996. *
  997. * Get the HIDDEVICEINFO info from the device
  998. *
  999. * @parm HDEVINFO | hdev |
  1000. *
  1001. * Device to get information
  1002. *
  1003. * @parm PSP_DEVICE_INTERFACE_DATA | pdid |
  1004. *
  1005. * Describes the device
  1006. *
  1007. * @parm OUT PHIDDEVICEINFO | phdi |
  1008. *
  1009. * HIDDEVICEINFO we need to collect
  1010. *
  1011. * @returns
  1012. *
  1013. * Nonzero on success.
  1014. *
  1015. *****************************************************************************/
  1016. BOOL INTERNAL
  1017. DIHid_GetDevInfo( HDEVINFO hdev, PSP_DEVICE_INTERFACE_DATA pdid, PHIDDEVICEINFO phdi )
  1018. {
  1019. BOOL fRc = FALSE;
  1020. SP_DEVINFO_DATA dinf;
  1021. EnterProcI(DIHid_GetDevInfo, (_ "xpp", hdev, pdid, phdi));
  1022. /*
  1023. * Open the registry key for the device so we can obtain
  1024. * auxiliary information.
  1025. */
  1026. dinf.cbSize = cbX(SP_DEVINFO_DATA);
  1027. /*
  1028. * Get the instance GUID and the path to
  1029. * the HID device so we can talk to it.
  1030. */
  1031. if (DIHid_GetDevicePath(hdev, pdid, &phdi->pdidd, &dinf) &&
  1032. DIHid_GetDeviceInstanceId(hdev, &dinf, &phdi->ptszId))
  1033. {
  1034. HANDLE hf;
  1035. TCHAR tszName[MAX_PATH];
  1036. ULONG len = sizeof(tszName);
  1037. BOOL fCreateOK = FALSE;
  1038. BYTE bNewState = DIHES_UNDEFINED;
  1039. BYTE bPreviousState = DIHES_UNDEFINED;
  1040. DIHid_CreateDeviceInstanceKeys( phdi, &bPreviousState );
  1041. //////////////////// BEGIN OF PATCH ////////////////////////
  1042. /*
  1043. * This part is to keep the compatibility with Win2k Gold.
  1044. * Some driver, like Thrustmaster driver, tries to get Joystick ID
  1045. * from old registry key:
  1046. * HKLM\SYSTEM\CurrentControlSet\Control\DeviceClasses\{4d1e55b2-f16f-11cf-88cb-001111000030}\<?????????>\#\Device Parameters\DirectX
  1047. * See Windows bug 395416 for detail.
  1048. */
  1049. {
  1050. HKEY hkDev;
  1051. HRESULT hres;
  1052. /*
  1053. * Open the registry key for the device so we can obtain
  1054. * auxiliary information, creating it if necessary.
  1055. */
  1056. hkDev = SetupDiCreateDeviceInterfaceRegKey(hdev, pdid, 0,
  1057. MAXIMUM_ALLOWED, NULL, NULL);
  1058. if(hkDev && hkDev != INVALID_HANDLE_VALUE)
  1059. {
  1060. hres = hresMumbleKeyEx(hkDev,
  1061. TEXT("DirectX"),
  1062. KEY_ALL_ACCESS,
  1063. REG_OPTION_NON_VOLATILE,
  1064. &phdi->hkOld);
  1065. if(SUCCEEDED(hres) )
  1066. {
  1067. LONG lRc;
  1068. lRc = RegSetValueEx(phdi->hkOld, TEXT("GUID"), 0, REG_BINARY,
  1069. (PV)&phdi->guid, cbX(GUID));
  1070. }
  1071. }
  1072. }
  1073. /////////////////// END OF PATCH /////////////////////
  1074. /*
  1075. * Before we CreateFile on the device, we need to make sure the
  1076. * device is not in the middle of the setup process. If a device has
  1077. * no device description it is probably still being setup (plugged
  1078. * in) so delay opening it to avoid having an open handle on a device
  1079. * that setup is trying to update. If that happens, setup will prompt
  1080. * the user to reboot to use the device.
  1081. * Since HIDs are "RAW" devices (don't need drivers) don't ignore
  1082. * such a device forever.
  1083. */
  1084. if( CM_Get_DevNode_Registry_Property(dinf.DevInst,
  1085. CM_DRP_DEVICEDESC,
  1086. NULL,
  1087. tszName,
  1088. &len,
  1089. 0) == CR_SUCCESS)
  1090. {
  1091. /*
  1092. * Known device.
  1093. */
  1094. fCreateOK = TRUE;
  1095. bNewState = DIHES_KNOWN;
  1096. }
  1097. else
  1098. {
  1099. /*
  1100. * Unknown device. This either means that setup has not yet
  1101. * completed processing the HID or that no device description
  1102. * was set for the device where it matched.
  1103. *
  1104. * For now, see if we've seen this device instance before.
  1105. * If we have, then if it was previously unknown assume it's
  1106. * going to stay that way and start using the device. If it
  1107. * was previously known or we've never seen it before wait for
  1108. * a while in case setup is just taking it's time.
  1109. *
  1110. * If the device was the last one we tried to remove, then it's
  1111. * OK to open it while it's still showing up.
  1112. * ISSUE-2001/02/06-MarcAnd Opening removed devices
  1113. * I assume this is so we get a read failure when it really goes
  1114. * away but I don't know how that's better than ignoring it.
  1115. */
  1116. /*
  1117. * ISSUE-2001/01/27-MarcAnd Should keep real list with status
  1118. * We should keep a complete list of all the devices we know
  1119. * about with status indicating things like "pending on
  1120. * setup since X" or "removed" not these globals.
  1121. */
  1122. if( lstrcmpi(phdi->ptszId, g_tszIdLastRemoved) == 0 )
  1123. {
  1124. fCreateOK = TRUE;
  1125. bNewState = bPreviousState;
  1126. }
  1127. else
  1128. {
  1129. if( bPreviousState == DIHES_UNKNOWN )
  1130. {
  1131. /*
  1132. * We've seen this device before and it was Unknown so
  1133. * don't expect that to change (as Setup does not retry
  1134. * the ID search through the INFs) so just use it.
  1135. */
  1136. fCreateOK = TRUE;
  1137. bNewState = DIHES_UNKNOWN;
  1138. }
  1139. else
  1140. {
  1141. if( lstrcmpi(phdi->ptszId, g_tszIdLastUnknown) == 0 )
  1142. {
  1143. if( GetTickCount() - g_tmLastUnknown < MSREBUILDRATE )
  1144. {
  1145. SquirtSqflPtszV(sqfl | sqflBenign,
  1146. TEXT("%hs: DIHid_BuildHidListEntry: %s pending on setup."),
  1147. s_szProc, phdi->ptszId );
  1148. fRc = FALSE;
  1149. }
  1150. else
  1151. {
  1152. /*
  1153. * Don't wait any longer but we'll need to update
  1154. * the state to "Unknown".
  1155. */
  1156. fCreateOK = TRUE;
  1157. bNewState = DIHES_UNKNOWN;
  1158. g_tszIdLastUnknown[0] = TEXT('0');
  1159. #ifdef XDEBUG
  1160. if( bPreviousState == DIHES_KNOWN )
  1161. {
  1162. SquirtSqflPtszV(sqfl | sqflBenign,
  1163. TEXT("%hs: %s was known but is now unknown!"),
  1164. s_szProc, phdi->ptszId );
  1165. }
  1166. #endif
  1167. }
  1168. }
  1169. else
  1170. {
  1171. lstrcpy( g_tszIdLastUnknown, phdi->ptszId );
  1172. g_tmLastUnknown = GetTickCount();
  1173. fRc = FALSE;
  1174. }
  1175. }
  1176. }
  1177. }
  1178. if( fCreateOK )
  1179. {
  1180. /*
  1181. * bNewState should always have been set if OK to create
  1182. */
  1183. AssertF( bNewState != DIHES_UNDEFINED );
  1184. /*
  1185. * Open the device so we can get its (real) usage page / usage.
  1186. */
  1187. hf = CreateFile(phdi->pdidd->DevicePath,
  1188. GENERIC_READ | GENERIC_WRITE,
  1189. FILE_SHARE_READ | FILE_SHARE_WRITE,
  1190. 0, /* no SECURITY_ATTRIBUTES */
  1191. OPEN_EXISTING,
  1192. 0, /* attributes */
  1193. 0); /* template */
  1194. if(hf != INVALID_HANDLE_VALUE)
  1195. {
  1196. PHIDP_PREPARSED_DATA ppd;
  1197. HIDD_ATTRIBUTES attr;
  1198. if(HidD_GetPreparsedData(hf, &ppd))
  1199. {
  1200. HIDP_CAPS caps;
  1201. if( SUCCEEDED(HidP_GetCaps(ppd, &caps)) )
  1202. {
  1203. DIHid_ParseUsagePage(phdi, &caps, ppd);
  1204. SquirtSqflPtszV(sqfl,
  1205. TEXT("%hs: Have %s"),
  1206. s_szProc, phdi->pdidd->DevicePath);
  1207. /*
  1208. * ISSUE-2001/02/06-MarcAnd Incorrect return possible
  1209. * There's nothing to reset this to false if any of
  1210. * the following code fails.
  1211. */
  1212. fRc = TRUE;
  1213. }
  1214. HidD_FreePreparsedData(ppd);
  1215. }
  1216. else
  1217. {
  1218. SquirtSqflPtszV(sqfl | sqflError,
  1219. TEXT("%hs: GetPreparsedData(%s) failed"),
  1220. s_szProc, phdi->pdidd->DevicePath);
  1221. }
  1222. attr.Size = cbX(attr);
  1223. if( HidD_GetAttributes(hf, &attr) )
  1224. {
  1225. if( ( phdi->ProductID != attr.ProductID )
  1226. || ( phdi->VendorID != attr.VendorID ) )
  1227. {
  1228. SquirtSqflPtszV(sqfl | sqflBenign,
  1229. TEXT("%hs: Device changed from VID_%04X&PID_%04X to VID_%04X&PID_%04X"),
  1230. s_szProc, phdi->ProductID, phdi->VendorID, attr.ProductID, attr.VendorID);
  1231. phdi->ProductID = attr.ProductID;
  1232. phdi->VendorID = attr.VendorID;
  1233. /*
  1234. * Make sure we update the registry.
  1235. */
  1236. bPreviousState = DIHES_UNDEFINED;
  1237. }
  1238. if( bPreviousState != bNewState )
  1239. {
  1240. DIHid_CreateDeviceInstanceKeys( phdi, &bNewState );
  1241. }
  1242. }
  1243. else
  1244. {
  1245. SquirtSqflPtszV(sqfl | sqflError,
  1246. TEXT("%hs: HidD_GetAttributes(%s) failed"),
  1247. s_szProc, phdi->pdidd->DevicePath);
  1248. }
  1249. DICreateStaticGuid(&phdi->guidProduct, attr.ProductID, attr.VendorID);
  1250. CloseHandle(hf);
  1251. }
  1252. else
  1253. {
  1254. SquirtSqflPtszV(sqfl | sqflError,
  1255. TEXT("%hs: CreateFile(%s) failed? le=%d"),
  1256. s_szProc, phdi->pdidd->DevicePath, GetLastError());
  1257. }
  1258. }
  1259. #ifndef WINNT
  1260. /*
  1261. * The following part is to resolve the swap id probleom in OSR.
  1262. * If a USB joystick is not in the lowest available id, after unplug and replug,
  1263. * VJOYD will sign the same joystick at the lowest available id, but DINPUT still
  1264. * remember its last id. To resolve this confliction, we need this code.
  1265. */
  1266. {
  1267. VXDINITPARMS vip;
  1268. CHAR sz[MAX_PATH];
  1269. LPSTR pszPath;
  1270. BOOL fFindIt = FALSE;
  1271. for( phdi->idJoy = 0; phdi->idJoy < cJoyMax; phdi->idJoy++ )
  1272. {
  1273. pszPath = JoyReg_JoyIdToDeviceInterface_95(phdi->idJoy, &vip, sz);
  1274. if(pszPath)
  1275. {
  1276. if( lstrcmpiA(pszPath, (PCHAR)phdi->pdidd->DevicePath) == 0x0 ) {
  1277. fFindIt = TRUE;
  1278. break;
  1279. }
  1280. }
  1281. }
  1282. if( fFindIt ) {
  1283. if( phdi->hk != 0x0 )
  1284. {
  1285. DWORD cb;
  1286. BOOL fRtn;
  1287. cb = cbX(phdi->idJoy);
  1288. fRtn = RegSetValueEx(phdi->hk, TEXT("Joystick Id"), 0, 0, (PV)&phdi->idJoy, cb );
  1289. if( !fRtn )
  1290. {
  1291. // don't worry
  1292. }
  1293. }
  1294. } else
  1295. {
  1296. // Post Dx7 Patch. Win9x only
  1297. // Gravis has gone and created a vjoyd mini driver for their
  1298. // USB gamepad. Their driver replaces joyhid, but does not inform
  1299. // vjoyd about the HID path. So 2 instances of the same device show
  1300. // up in the CPL.
  1301. // This fix is to make a HID device that does not have a matching
  1302. // joyhid entry as being non Joystick.
  1303. /*
  1304. * Post DX7a! Only do this for game controllers
  1305. * Assert the device types in case they get changed
  1306. * Note they are changed in DX8 but the new types are
  1307. * only to be used by DX8 so these assertions should
  1308. * be sufficient.
  1309. */
  1310. CAssertF( DIDEVTYPE_DEVICE == 1 );
  1311. CAssertF( DIDEVTYPE_JOYSTICK == 4 );
  1312. if( GET_DIDEVICE_TYPE( phdi->osd.dwDevType ) == DIDEVTYPE_JOYSTICK )
  1313. {
  1314. /*
  1315. * Change type from joystick to device
  1316. */
  1317. phdi->osd.dwDevType ^= DIDEVTYPE_JOYSTICK | DIDEVTYPE_DEVICE;
  1318. }
  1319. /*
  1320. * Set to the invalid id anyway
  1321. */
  1322. phdi->idJoy = -1;
  1323. }
  1324. }
  1325. #else
  1326. // Executed only on WINNT
  1327. if( phdi->hk != 0x0 )
  1328. {
  1329. DWORD cb;
  1330. cb = cbX(phdi->idJoy);
  1331. if( RegQueryValueEx(phdi->hk, TEXT("Joystick Id"),
  1332. 0, 0, (PV)&phdi->idJoy, &cb ) != ERROR_SUCCESS )
  1333. {
  1334. phdi->idJoy = JOY_BOGUSID;
  1335. }
  1336. }
  1337. #endif // WINNT
  1338. }
  1339. else
  1340. {
  1341. SquirtSqflPtszV(sqfl | sqflError,
  1342. TEXT("%hs: Unable to get GUID or device path"),
  1343. s_szProc);
  1344. }
  1345. /*
  1346. * If we failed, then free the goo we already got.
  1347. */
  1348. if(!fRc)
  1349. {
  1350. if( phdi->hk )
  1351. {
  1352. RegCloseKey(phdi->hk);
  1353. }
  1354. phdi->hk = 0;
  1355. FreePpv(&phdi->pdidd);
  1356. FreePpv(&phdi->ptszId);
  1357. }
  1358. ExitProcF(fRc);
  1359. return fRc;
  1360. }
  1361. #undef DIHES_UNDEFINED
  1362. #undef DIHES_UNKNOWN
  1363. #undef DIHES_KNOWN
  1364. /*****************************************************************************
  1365. *
  1366. * @doc INTERNAL
  1367. *
  1368. * @func void | DIHid_BuildHidListEntry |
  1369. *
  1370. * Builds a single entry in the list of HID devices.
  1371. *
  1372. * @parm HDEVINFO | hdev |
  1373. *
  1374. * Device list being enumerated.
  1375. *
  1376. * @parm PSP_DEVICE_INTERFACE_DATA | pdid |
  1377. *
  1378. * Describes the device that was enumerated.
  1379. *
  1380. * @returns
  1381. *
  1382. * Nonzero on success.
  1383. *
  1384. *****************************************************************************/
  1385. BOOL INTERNAL
  1386. DIHid_BuildHidListEntry(HDEVINFO hdev, PSP_DEVICE_INTERFACE_DATA pdid)
  1387. {
  1388. BOOL fRc = TRUE;
  1389. BOOL fAlreadyExist = FALSE;
  1390. PHIDDEVICEINFO phdi;
  1391. HIDDEVICEINFO hdi;
  1392. EnterProcI(DIHid_BuildHidListEntry, (_ "xp", hdev, pdid));
  1393. if(g_phdl)
  1394. {
  1395. PSP_DEVICE_INTERFACE_DETAIL_DATA pdidd;
  1396. /* GetDevicePath is expecting a NULL */
  1397. pdidd = NULL;
  1398. if( DIHid_GetDevicePath(hdev, pdid, &pdidd, NULL) )
  1399. {
  1400. PHIDDEVICEINFO phdi;
  1401. int ihdi;
  1402. //Check whether the device has been in the list
  1403. for(ihdi = 0, phdi = g_phdl->rghdi;
  1404. ihdi < g_phdl->chdi;
  1405. ihdi++, phdi++)
  1406. {
  1407. if( phdi->pdidd )
  1408. {
  1409. if( lstrcmp( pdidd->DevicePath, phdi->pdidd->DevicePath ) == 0 )
  1410. {
  1411. //already in the list
  1412. fAlreadyExist = TRUE;
  1413. #ifdef WINNT
  1414. // id may be changed, so refresh here.
  1415. // See Windows bug 321711. -qzheng
  1416. if( phdi->hk != 0x0 )
  1417. {
  1418. DWORD cb;
  1419. cb = cbX(phdi->idJoy);
  1420. if( RegQueryValueEx(phdi->hk, TEXT("Joystick Id"),
  1421. 0, 0, (PV)&phdi->idJoy, &cb ) != ERROR_SUCCESS )
  1422. {
  1423. phdi->idJoy = JOY_BOGUSID;
  1424. }
  1425. }
  1426. #endif
  1427. SquirtSqflPtszV(sqfl | sqflTrace,
  1428. TEXT("%hs: Device %s Already Exists in HID List "), s_szProc, pdidd->DevicePath);
  1429. break;
  1430. }
  1431. }
  1432. }
  1433. FreePpv(&pdidd);
  1434. }
  1435. if( fAlreadyExist )
  1436. {
  1437. //device is already there, needn't do anything, just leave
  1438. } else
  1439. {
  1440. PBUSDEVICEINFO pbdi;
  1441. // prefix 29343
  1442. if( g_phdl!= NULL && g_phdl->chdi >= g_phdl->chdiAlloc )
  1443. {
  1444. /*
  1445. * Make sure there is room for this device in the list.
  1446. * Grow by doubling.
  1447. */
  1448. HRESULT hres;
  1449. hres = ReallocCbPpv(cbHdlChdi(g_phdl->chdiAlloc * 2), &g_phdl);
  1450. if(FAILED(hres))
  1451. {
  1452. SquirtSqflPtszV(sqfl | sqflError,
  1453. TEXT("%hs: Realloc failed"), s_szProc);
  1454. fRc = FALSE;
  1455. goto done;
  1456. }
  1457. g_phdl->chdiAlloc *= 2;
  1458. }
  1459. phdi = &g_phdl->rghdi[g_phdl->chdi];
  1460. memset( &hdi, 0, sizeof(hdi) );
  1461. if( DIHid_GetDevInfo(hdev, pdid, &hdi) )
  1462. {
  1463. hdi.fAttached = TRUE;
  1464. pbdi = pbdiFromphdi(&hdi);
  1465. if( pbdi != NULL )
  1466. {
  1467. /*
  1468. * If the devices is just being removed, and PnP is working on the
  1469. * removing, then we won't include it in our list.
  1470. */
  1471. if( lstrcmpi(pbdi->ptszId, g_tszIdLastRemoved) == 0 &&
  1472. GetTickCount() - g_tmLastRemoved < MSREBUILDRATE_FIFTH )
  1473. {
  1474. SquirtSqflPtszV(sqfl | sqflBenign,
  1475. TEXT("%hs: DIHid_BuildHidListEntry: %s pending on removal."),
  1476. s_szProc, pbdi->ptszId );
  1477. fRc = FALSE;
  1478. } else {
  1479. BUS_REGDATA RegData;
  1480. memcpy( phdi, &hdi, sizeof(hdi) );
  1481. g_phdl->chdi++;
  1482. if( SUCCEEDED(DIBusDevice_GetRegData(pbdi->hk, &RegData)) )
  1483. {
  1484. RegData.fAttachOnReboot = TRUE;
  1485. DIBusDevice_SetRegData(pbdi->hk, &RegData);
  1486. }
  1487. }
  1488. } else {
  1489. memcpy( phdi, &hdi, sizeof(hdi) );
  1490. g_phdl->chdi++;
  1491. }
  1492. }
  1493. else
  1494. {
  1495. SquirtSqflPtszV(sqfl | sqflBenign,
  1496. TEXT("%hs: DIHid_GetDevInfo failed error %d, ignoring device"),
  1497. s_szProc, GetLastError() );
  1498. fRc = FALSE;
  1499. }
  1500. }
  1501. } // end of if(g_phdl)
  1502. done:
  1503. ExitProcF(fRc);
  1504. return fRc;
  1505. }
  1506. /*****************************************************************************
  1507. *
  1508. * @doc INTERNAL
  1509. *
  1510. * @func void | DIHid_BuildHidList |
  1511. *
  1512. * Builds the list of HID devices.
  1513. *
  1514. * @parm BOOL | fForce |
  1515. *
  1516. * If nonzero, we force a rebuild of the HID list.
  1517. * Otherwise, we rebuild only if the list hasn't
  1518. * been rebuilt recently.
  1519. *
  1520. *****************************************************************************/
  1521. void EXTERNAL
  1522. DIHid_BuildHidList(BOOL fForce)
  1523. {
  1524. HRESULT hres;
  1525. DWORD dwTick;
  1526. EnterProcI(DIHid_BuildHidList, (_ "u", fForce));
  1527. DllEnterCrit();
  1528. // Force implies a complete rebuild of the list
  1529. // No hanging onto old entries.
  1530. if( fForce )
  1531. {
  1532. DIHid_EmptyHidList();
  1533. }
  1534. //ISSUE-2001/03/29-timgill Meltdown code change may be unnecessary
  1535. dwTick = GetTickCount();
  1536. if(!(g_flEmulation & 0x80000000) && /* Not disabled by emulation */
  1537. HidD_GetHidGuid && /* HID actually exists */
  1538. (fForce || /* Forcing rebuild, or */
  1539. g_tmLastHIDRebuild == 0 || /* Never built before, or */
  1540. dwTick - g_tmLastHIDRebuild > MSREBUILDRATE /* Haven't rebuilt in a while */
  1541. #ifdef WINNT
  1542. || dwTick - Excl_GetConfigChangedTime() < MSREBUILDRATE /* joyConfig is just changed. */
  1543. #endif
  1544. ))
  1545. {
  1546. GUID guidHid;
  1547. HDEVINFO hdev;
  1548. HidD_GetHidGuid(&guidHid);
  1549. hdev = SetupDiGetClassDevs(&guidHid, 0, 0,
  1550. DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);
  1551. if(hdev != INVALID_HANDLE_VALUE)
  1552. {
  1553. DIHid_CheckHidList();
  1554. /*
  1555. * There is no way to query the number of devices.
  1556. * You just have to keep incrementing until you run out.
  1557. *
  1558. * If we already have a g_phdl, then re-use it. Else, create
  1559. * a new one. But always realloc up to the minimum starting
  1560. * point. (This upward realloc is important in case there
  1561. * were no HID devices the last time we checked.)
  1562. */
  1563. if( !g_phdl )
  1564. {
  1565. hres = AllocCbPpv(cbHdlChdi(chdiInit), &g_phdl);
  1566. if(SUCCEEDED(hres))
  1567. {
  1568. g_phdl->chdi = 0;
  1569. g_phdl->chdiAlloc = chdiInit;
  1570. } else
  1571. {
  1572. /* Skip erroneous items */
  1573. SquirtSqflPtszV(sqfl | sqflError,
  1574. TEXT("DIHid_BuildHidListEntry ")
  1575. TEXT("Realloc g_phdl fails."));
  1576. }
  1577. } else
  1578. hres = S_OK;
  1579. if(SUCCEEDED(hres))
  1580. {
  1581. int idev;
  1582. /*
  1583. * To avoid infinite looping on
  1584. * internal error, break on any
  1585. * error once we have tried more than
  1586. * chdiMax devices, since that's the most
  1587. * HID will ever give us.
  1588. */
  1589. for(idev = 0; idev < chdiMax; idev++)
  1590. {
  1591. SP_DEVICE_INTERFACE_DATA did;
  1592. AssertF(g_phdl->chdi <= g_phdl->chdiAlloc);
  1593. /*
  1594. * SetupDI requires that the caller initialize cbSize.
  1595. */
  1596. did.cbSize = cbX(did);
  1597. if(SetupDiEnumDeviceInterfaces(hdev, 0, &guidHid,
  1598. idev, &did))
  1599. {
  1600. if(DIHid_BuildHidListEntry(hdev, &did))
  1601. {
  1602. } else
  1603. {
  1604. /* Skip erroneous items */
  1605. SquirtSqflPtszV(sqfl | sqflError,
  1606. TEXT("DIHid_BuildHidListEntry ")
  1607. TEXT("failed?"));
  1608. }
  1609. } else
  1610. if(GetLastError() == ERROR_NO_MORE_ITEMS)
  1611. {
  1612. break;
  1613. } else
  1614. {
  1615. /* Skip erroneous items */
  1616. SquirtSqflPtszV(sqfl | sqflError,
  1617. TEXT("SetupDiEnumDeviceInterface ")
  1618. TEXT("failed? le=%d"), GetLastError());
  1619. }
  1620. }
  1621. }
  1622. SetupDiDestroyDeviceInfoList(hdev);
  1623. }
  1624. #ifdef WINNT
  1625. if( g_phdl && g_phdl->chdi )
  1626. {
  1627. DIWdm_InitJoyId();
  1628. }
  1629. #endif
  1630. g_tmLastHIDRebuild = GetTickCount();
  1631. }
  1632. DllLeaveCrit();
  1633. ExitProc();
  1634. }
  1635. #endif