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.

645 lines
18 KiB

  1. //Copyright (c) 1998 - 1999 Microsoft Corporation
  2. /*++
  3. Module Name:
  4. rdpdrstp
  5. Abstract:
  6. This module implements Terminal Server RDPDR device redirector
  7. setup functions in C for user-mode NT.
  8. Environment:
  9. User mode
  10. Author:
  11. Tadb
  12. --*/
  13. // Toggle stand-alone testing.
  14. //#define UNITTEST 1
  15. #include "stdafx.h"
  16. #ifdef UNITTEST
  17. #include <windows.h>
  18. #include <tchar.h>
  19. #include <time.h>
  20. #include <stdio.h>
  21. #include <setupapi.h>
  22. #include <prsht.h>
  23. #include "objbase.h" // for CoInitialize()
  24. #endif
  25. #include <devguid.h>
  26. #include <cfgmgr32.h>
  27. #include <winspool.h>
  28. #include <rdpdrstp.h>
  29. #include "newdev.h"
  30. #define SIZECHARS(x) (sizeof((x))/sizeof(*x))
  31. //#define USBMON_DLL TEXT("USBMON.DLL")
  32. //#define USB_MON_NAME TEXT("USB Monitor")
  33. #ifndef UNITTEST
  34. #include "logmsg.h"
  35. #endif
  36. #ifdef UNITTEST
  37. #define LOGMESSAGE1(arg1, arg2) ;
  38. #define LOGMESSAGE0(arg1) ;
  39. #endif
  40. ////////////////////////////////////////////////////////////
  41. //
  42. // Internal Types
  43. //
  44. typedef BOOL (InstallDevInstFuncType)(
  45. HWND hwndParent, LPCWSTR DeviceInstanceId,
  46. BOOL UpdateDriver,
  47. PDWORD pReboot,
  48. BOOL silentInstall
  49. );
  50. BOOL RDPDRINST_GUIModeSetupInstall(
  51. IN HWND hwndParent,
  52. IN WCHAR *pPNPID,
  53. IN TCHAR *pDeviceID
  54. )
  55. /*++
  56. Routine Description:
  57. This is the single entry point for RDPDR (Terminal Server Device Redirector)
  58. GUI-mode setup install routine.
  59. It currently simply creates and installs a dev node for RDPDR to interact with
  60. PnP.
  61. Arguments:
  62. hwndParent Handle to parent window for GUI required by this function.
  63. Return Value:
  64. TRUE on success. FALSE, otherwise.
  65. --*/
  66. {
  67. HDEVINFO devInfoSet;
  68. SP_DEVINFO_DATA deviceInfoData;
  69. TCHAR className[MAX_CLASS_NAME_LEN];
  70. WCHAR pnpID[256];
  71. DWORD len;
  72. WCHAR devInstanceID[MAX_PATH];
  73. InstallDevInstFuncType *pInstallDevInst;
  74. HINSTANCE hndl = NULL;
  75. //MONITOR_INFO_2 mi;
  76. //mi.pDLLName = USBMON_DLL;
  77. //mi.pEnvironment = NULL;
  78. //mi.pName = USB_MON_NAME;
  79. // Add the USB port monitor
  80. //if (!AddMonitor(NULL, 2, (PBYTE)&mi)) {
  81. // if (GetLastError() != ERROR_PRINT_MONITOR_ALREADY_INSTALLED) {
  82. // LOGMESSAGE1(_T("AddMonitor failed. Error code: %ld."), GetLastError());
  83. // return FALSE;
  84. // }
  85. //}
  86. //
  87. // Create the device info list.
  88. //
  89. devInfoSet = SetupDiCreateDeviceInfoList(&GUID_DEVCLASS_SYSTEM, hwndParent);
  90. if (devInfoSet == INVALID_HANDLE_VALUE) {
  91. LOGMESSAGE1(_T("Error creating device info list. Error code: %ld."),
  92. GetLastError());
  93. return FALSE;
  94. }
  95. //
  96. // Get the "official" system class name.
  97. //
  98. ZeroMemory(&deviceInfoData, sizeof(SP_DEVINFO_DATA));
  99. deviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA);
  100. if (!SetupDiClassNameFromGuid(&GUID_DEVCLASS_SYSTEM,
  101. className,
  102. sizeof(className)/sizeof(TCHAR),
  103. NULL
  104. ))
  105. {
  106. LOGMESSAGE1(_T("Error fetching system class name. Error code: %ld."),
  107. GetLastError());
  108. return FALSE;
  109. }
  110. //
  111. // Create the dev node.
  112. //
  113. if (!SetupDiCreateDeviceInfo(devInfoSet,
  114. pDeviceID,
  115. &GUID_DEVCLASS_SYSTEM,
  116. NULL,
  117. hwndParent,
  118. 0L, // No flags.
  119. &deviceInfoData
  120. ))
  121. {
  122. // If it already exists, then we are done ... because this was an
  123. // upgrade.
  124. if (GetLastError() == ERROR_DEVINST_ALREADY_EXISTS)
  125. {
  126. SetupDiDestroyDeviceInfoList(devInfoSet);
  127. return TRUE;
  128. }
  129. else {
  130. LOGMESSAGE1(_T("Error creating device node. Error code: %ld."),
  131. GetLastError());
  132. SetupDiDestroyDeviceInfoList(devInfoSet);
  133. return FALSE;
  134. }
  135. }
  136. else if (!SetupDiSetSelectedDevice(devInfoSet, &deviceInfoData)) {
  137. LOGMESSAGE1(_T("Error selecting device node. Error code: %ld."),
  138. GetLastError());
  139. goto WhackTheDevNodeAndReturnError;
  140. }
  141. //
  142. // Add the RDPDR PnP ID.
  143. //
  144. // Create the PnP ID string.
  145. wcscpy(pnpID, pPNPID);
  146. len = wcslen(pnpID);
  147. // This is a multi_sz string, so we need to terminate with an extra null.
  148. pnpID[len+1] = 0;
  149. // Add it to the registry entry for the dev node.
  150. if (!SetupDiSetDeviceRegistryProperty(
  151. devInfoSet, &deviceInfoData,
  152. SPDRP_HARDWAREID, (CONST BYTE *)pnpID,
  153. (len + 2) * sizeof(WCHAR))) {
  154. LOGMESSAGE1(_T("Error setting device registry property. Error code: %ld."),
  155. GetLastError());
  156. goto WhackTheDevNodeAndReturnError;
  157. }
  158. //
  159. // Register the, as of yet, phantom dev node with PnP to turn it into a real
  160. // dev node.
  161. //
  162. if (!SetupDiRegisterDeviceInfo(devInfoSet, &deviceInfoData, 0, NULL,
  163. NULL, NULL)) {
  164. LOGMESSAGE1(_T("Error registering device node with PnP. Error code: %ld."),
  165. GetLastError());
  166. goto WhackTheDevNodeAndReturnError;
  167. }
  168. //
  169. // Get the device instance ID.
  170. //
  171. if (!SetupDiGetDeviceInstanceIdW(devInfoSet, &deviceInfoData, devInstanceID,
  172. SIZECHARS(devInstanceID), NULL)) {
  173. LOGMESSAGE1(_T("Error getting the device instance id. Error code: %ld."),
  174. GetLastError());
  175. goto WhackTheDevNodeAndReturnError;
  176. }
  177. //
  178. // Use newdev.dll to install RDPDR as the driver for this new dev node.
  179. //
  180. hndl = LoadLibrary(TEXT("newdev.dll"));
  181. if (hndl == NULL) {
  182. LOGMESSAGE1(_T("Error loading newdev.dll. Error code: %ld."),
  183. GetLastError());
  184. goto WhackTheDevNodeAndReturnError;
  185. }
  186. pInstallDevInst = (InstallDevInstFuncType *)GetProcAddress(hndl, "InstallDevInstEx");
  187. if (pInstallDevInst == NULL) {
  188. LOGMESSAGE1(_T("Error fetching InstallDevInst func. Error code: %ld."),
  189. GetLastError());
  190. goto WhackTheDevNodeAndReturnError;
  191. }
  192. if ((*pInstallDevInst)(hwndParent, devInstanceID, FALSE, NULL, TRUE)) {
  193. // Clean up and return success!
  194. SetupDiDestroyDeviceInfoList(devInfoSet);
  195. FreeLibrary(hndl);
  196. return TRUE;
  197. }
  198. else {
  199. LOGMESSAGE1(_T("Error in newdev install. Error code: %ld."),
  200. GetLastError());
  201. }
  202. //
  203. // Whack the dev node and return failure.
  204. //
  205. WhackTheDevNodeAndReturnError:
  206. SetupDiCallClassInstaller(DIF_REMOVE, devInfoSet, &deviceInfoData);
  207. SetupDiDestroyDeviceInfoList(devInfoSet);
  208. if (hndl != NULL) {
  209. FreeLibrary(hndl);
  210. }
  211. return FALSE;
  212. }
  213. BOOL RDPDRINST_GUIModeSetupUninstall(HWND hwndParent, WCHAR *pPNPID, GUID *pGuid)
  214. /*++
  215. Routine Description:
  216. This is the single entry point for RDPDR (Terminal Server Device Redirector)
  217. GUI-mode setup uninstall routine.
  218. It currently simply remove the dev node created so that RDPDR can interact
  219. with PnP.
  220. Arguments:
  221. hwndParent Handle to parent window for GUI required by this function.
  222. Return Value:
  223. TRUE on success. FALSE, otherwise.
  224. --*/
  225. {
  226. HDEVINFO devInfoSet;
  227. SP_DEVINFO_DATA deviceInfoData;
  228. DWORD iLoop;
  229. BOOL bMoreDevices;
  230. WCHAR pnpID[256];
  231. BOOL result;
  232. //
  233. // Get the set of all devices with the RDPDR PnP ID.
  234. //
  235. devInfoSet = SetupDiGetClassDevs(pGuid, NULL, hwndParent,
  236. DIGCF_PRESENT);
  237. if (devInfoSet == INVALID_HANDLE_VALUE) {
  238. LOGMESSAGE1(_T("Error getting RDPDR devices from PnP. Error code: %ld."),
  239. GetLastError());
  240. return FALSE;
  241. }
  242. // Assume that we will be successful.
  243. result = TRUE;
  244. // Get the first device.
  245. iLoop=0;
  246. deviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA);
  247. bMoreDevices=SetupDiEnumDeviceInfo(devInfoSet, iLoop, &deviceInfoData);
  248. // Get the details for all matching device interfaces.
  249. while (bMoreDevices)
  250. {
  251. // Get the PnP ID for the device.
  252. if (!SetupDiGetDeviceRegistryProperty(devInfoSet, &deviceInfoData,
  253. SPDRP_HARDWAREID, NULL, (BYTE *)pnpID,
  254. sizeof(pnpID), NULL)) {
  255. LOGMESSAGE1(_T("Error fetching PnP ID in RDPDR device node remove. Error code: %ld."),
  256. GetLastError());
  257. }
  258. // If the current device matches RDPDR, then remove it.
  259. else if (!wcscmp(pnpID, pPNPID))
  260. {
  261. if (!SetupDiCallClassInstaller(DIF_REMOVE, devInfoSet, &deviceInfoData)) {
  262. // If we failed here, set the return status to indicate failure, but
  263. // don't give up on any other RDPDR dev nodes.
  264. LOGMESSAGE1(_T("Error removing RDPDR device node. Error code: %ld."),
  265. GetLastError());
  266. result = FALSE;
  267. }
  268. }
  269. // Get the next one device interface.
  270. bMoreDevices=SetupDiEnumDeviceInfo(devInfoSet, ++iLoop, &deviceInfoData);
  271. }
  272. // Release the device info list.
  273. SetupDiDestroyDeviceInfoList(devInfoSet);
  274. return result;
  275. }
  276. ULONG RDPDRINST_DetectInstall()
  277. /*++
  278. Routine Description:
  279. Return the number of RDPDR.SYS devices found.
  280. Arguments:
  281. NA
  282. Return Value:
  283. TRUE on success. FALSE, otherwise.
  284. --*/
  285. {
  286. HDEVINFO devInfoSet;
  287. SP_DEVINFO_DATA deviceInfoData;
  288. DWORD iLoop;
  289. BOOL bMoreDevices;
  290. ULONG count;
  291. TCHAR pnpID[256];
  292. GUID *pGuid=(GUID *)&GUID_DEVCLASS_SYSTEM;
  293. //
  294. // Get the set of all devices with the RDPDR PnP ID.
  295. //
  296. devInfoSet = SetupDiGetClassDevs(pGuid, NULL, NULL,
  297. DIGCF_PRESENT);
  298. if (devInfoSet == INVALID_HANDLE_VALUE) {
  299. LOGMESSAGE1(_T("ERRORgetting RDPDRINST_DetectInstall:RDPDR devices from PnP. Error code: %ld."),
  300. GetLastError());
  301. return 0;
  302. }
  303. // Get the first device.
  304. iLoop=0;
  305. deviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA);
  306. bMoreDevices=SetupDiEnumDeviceInfo(devInfoSet, iLoop, &deviceInfoData);
  307. // Get the details for all matching device interfaces.
  308. count = 0;
  309. while (bMoreDevices)
  310. {
  311. // Get the PnP ID for the device.
  312. if (!SetupDiGetDeviceRegistryProperty(devInfoSet, &deviceInfoData,
  313. SPDRP_HARDWAREID, NULL, (BYTE *)pnpID,
  314. sizeof(pnpID), NULL)) {
  315. LOGMESSAGE1(_T("ERROR:fetching PnP ID in RDPDR device node remove. Error code: %ld."),
  316. GetLastError());
  317. }
  318. // If the current device matches the RDPDR PNP ID
  319. if (!_tcscmp(pnpID, TRDPDRPNPID)) {
  320. count++;
  321. }
  322. // Get the next one device interface.
  323. bMoreDevices=SetupDiEnumDeviceInfo(devInfoSet, ++iLoop, &deviceInfoData);
  324. }
  325. // Release the device info list.
  326. SetupDiDestroyDeviceInfoList(devInfoSet);
  327. return count;
  328. }
  329. BOOL IsRDPDrInstalled ()
  330. {
  331. ULONG ulReturn;
  332. LOGMESSAGE0(_T("Entered IsRDPDrInstalled"));
  333. ulReturn = RDPDRINST_DetectInstall();
  334. LOGMESSAGE1(_T("Returning IsRDPDrInstalled (ulReturn = %d)"), ulReturn);
  335. return 0 != ulReturn;
  336. }
  337. //
  338. // Unit-Test
  339. //
  340. #ifdef UNITTEST
  341. void __cdecl main()
  342. {
  343. RDPDRINST_GUIModeSetupInstall(NULL);
  344. RDPDRINST_GUIModeSetupUninstall(NULL);
  345. }
  346. #endif
  347. #ifdef TSOC_CONSOLE_SHADOWING
  348. //
  349. // Need to instantiate the device class GUIDs so we can use the "Net" class
  350. // GUID below...
  351. //
  352. /*
  353. DWORD
  354. InstallRootEnumeratedDevice(
  355. IN HWND hwndParent,
  356. IN PCTSTR DeviceName,
  357. IN PCTSTR HardwareIdList,
  358. IN PCTSTR FullInfPath,
  359. OUT PBOOL RebootRequired OPTIONAL
  360. )
  361. */
  362. /*++
  363. Routine Description:
  364. This routine creates and installs a new, root-enumerated devnode
  365. representing a network adapter.
  366. Arguments:
  367. hwndParent - Supplies the window handle to be used as the parent of any
  368. UI that is generated as a result of this device's installation.
  369. DeviceName - Supplies the full name of the devnode to be created (e.g.,
  370. "Root\VMWARE\0000"). Note that if this devnode already exists, the API
  371. will fail.
  372. HardwareIdList - Supplies a multi-sz list containing one or more hardware
  373. IDs to be associated with the device. These are necessary in order to
  374. match up with an INF driver node when we go to do the device
  375. installation.
  376. FullInfPath - Supplies the full path to the INF to be used when installing
  377. this device.
  378. RebootRequired - Optionally, supplies the address of a boolean that is set,
  379. upon successful return, to indicate whether or not a reboot is required
  380. to bring the newly-installed device on-line.
  381. Return Value:
  382. If the function succeeds, the return value is NO_ERROR.
  383. If the function fails, the return value is a Win32 error code indicating
  384. the cause of the failure.
  385. --*/
  386. /*
  387. {
  388. HDEVINFO DeviceInfoSet;
  389. SP_DEVINFO_DATA DeviceInfoData;
  390. DWORD HardwareIdListSize, CurIdSize;
  391. PCTSTR p;
  392. DWORD Err;
  393. //
  394. // Create the container for the to-be-created device information element.
  395. //
  396. DeviceInfoSet = SetupDiCreateDeviceInfoList(&GUID_DEVCLASS_DISPLAY, hwndParent);
  397. if (DeviceInfoSet == INVALID_HANDLE_VALUE)
  398. {
  399. LOGMESSAGE1(_T("SetupDiCreateDeviceInfoList failed. LastError = %ld"), GetLastError());
  400. return GetLastError();
  401. }
  402. //
  403. // Now create the element.
  404. //
  405. // ** Note that if the desire is to always have a unique devnode be created
  406. // (i.e., have an auto-generated name), then the caller would need to pass
  407. // in just the device part of the name (i.e., just the middle part of
  408. // "Root\<DeviceId>\<UniqueInstanceId>"). In that case, we'd need to pass
  409. // the DICD_GENERATE_ID flag in the next-to-last argument of the call below.
  410. //
  411. DeviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA);
  412. if (!SetupDiCreateDeviceInfo(DeviceInfoSet,
  413. DeviceName,
  414. &GUID_DEVCLASS_DISPLAY,
  415. NULL,
  416. hwndParent,
  417. 0,
  418. &DeviceInfoData))
  419. {
  420. LOGMESSAGE1(_T("SetupDiCreateDeviceInfo failed. LastError = %ld"), GetLastError());
  421. Err = GetLastError();
  422. SetupDiDestroyDeviceInfoList(DeviceInfoSet);
  423. return Err;
  424. }
  425. //
  426. // Now compute the size of the hardware ID list we're going to associate
  427. // with the device.
  428. //
  429. HardwareIdListSize = 1; // initialize to 1 for extra null terminating char
  430. for(p = HardwareIdList; *p; p += CurIdSize)
  431. {
  432. CurIdSize = lstrlen(p) + 1;
  433. HardwareIdListSize += CurIdSize;
  434. }
  435. //
  436. // (Need size in bytes, not characters, for call below.)
  437. //
  438. HardwareIdListSize *= sizeof(TCHAR);
  439. //
  440. // Store the hardware ID list to the device's HardwareID property.
  441. //
  442. if (!SetupDiSetDeviceRegistryProperty(DeviceInfoSet,
  443. &DeviceInfoData,
  444. SPDRP_HARDWAREID,
  445. (LPBYTE)HardwareIdList,
  446. HardwareIdListSize))
  447. {
  448. LOGMESSAGE1(_T("SetupDiSetDeviceRegistryProperty failed. LastError = %ld"), GetLastError());
  449. Err = GetLastError();
  450. SetupDiDestroyDeviceInfoList(DeviceInfoSet);
  451. return Err;
  452. }
  453. //
  454. // OK, now we can register our device information element. This transforms
  455. // the element from a mere registry presence into an actual devnode in the
  456. // PnP hardware tree.
  457. //
  458. if (!SetupDiCallClassInstaller(DIF_REGISTERDEVICE,
  459. DeviceInfoSet,
  460. &DeviceInfoData))
  461. {
  462. LOGMESSAGE1(_T("SetupDiCallClassInstaller failed. LastError = %ld"), GetLastError());
  463. Err = GetLastError();
  464. SetupDiDestroyDeviceInfoList(DeviceInfoSet);
  465. return Err;
  466. }
  467. //
  468. // OK, the device information element has now been registered. From here
  469. // on, if we encounter any failure we'll also need to explicitly remove
  470. // this device from the system before bailing.
  471. //
  472. //
  473. // Now we're ready to install the device. (We need to initialize the
  474. // caller-supplied "RebootRequired" buffer to zero, because the call below
  475. // simply ORs in reboot-needed flags, as it performs device installations
  476. // that require reboot.)
  477. //
  478. if(RebootRequired)
  479. {
  480. *RebootRequired = FALSE;
  481. }
  482. if (!UpdateDriverForPlugAndPlayDevices(hwndParent,
  483. HardwareIdList, // use the first ID
  484. FullInfPath,
  485. INSTALLFLAG_FORCE,
  486. RebootRequired))
  487. {
  488. Err = GetLastError();
  489. LOGMESSAGE1(_T("UpdateDriverForPlugAndPlayDevices failed. LastError = %ld"), GetLastError());
  490. if(Err == NO_ERROR)
  491. {
  492. //
  493. // The only time we should get NO_ERROR here is when
  494. // UpdateDriverForPlugAndPlayDevices didn't find anything to do.
  495. // That should never be the case here. However, since something
  496. // obviously went awry, go ahead and force some error, so the
  497. // caller knows things didn't work out.
  498. //
  499. Err = ERROR_NO_SUCH_DEVINST;
  500. }
  501. SetupDiCallClassInstaller(DIF_REMOVE,
  502. DeviceInfoSet,
  503. &DeviceInfoData
  504. );
  505. SetupDiDestroyDeviceInfoList(DeviceInfoSet);
  506. return Err;
  507. }
  508. //
  509. // We're done! We successfully installed the device.
  510. //
  511. SetupDiDestroyDeviceInfoList(DeviceInfoSet);
  512. return NO_ERROR;
  513. }
  514. */
  515. #endif// TSOC_CONSOLE_SHADOWING