Leaked source code of windows server 2003
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.

2161 lines
73 KiB

  1. /*++
  2. Copyright (c) 1995 Microsoft Corporation
  3. All rights reserved.
  4. Module Name:
  5. printupg.cxx
  6. Abstract:
  7. Code to implement printupg. Please refer to printupg.hxx for an overview
  8. of printupg feature set.
  9. Author:
  10. Larry Zhu (LZhu) 20-Feb-2001
  11. Revision History:
  12. --*/
  13. #include "precomp.h"
  14. #pragma hdrstop
  15. #include "printupg.hxx"
  16. /*++
  17. Routine Name
  18. PSetupIsCompatibleDriver
  19. Routine Description:
  20. Check whether the driver is blocked or warned and if the driver is blocked
  21. or warned, return the blocking status and the replacement driver, if any.
  22. Arguments:
  23. pszServer - The server that needs to check for bad driver. If
  24. pszServer is NULL, that means the local machine
  25. pszDriverModel - The name of the driver to check
  26. pszDriverPath - The path of the driver, this can be a full path
  27. or the filename
  28. pszEnvironment - The environment of the server, such as
  29. "Windows NT x86"
  30. dwVersion - The major version of the driver
  31. pFileTimeDriver - The FileTime of the driver
  32. puBlockingStatus - Points to status of blocking
  33. ppszReplacementDriver - Points to the NULL terminating name of the replacement
  34. driver. A *NULL* string (instead of an empty string)
  35. means there is no replacement driver
  36. Return Value:
  37. A BOOL - TRUE if success; FALSE otherwise, Call GetLastError()
  38. to get the Error code. Pass a NULL to ppszReplacementDriver
  39. will not receive the name of replacement driver otherwise
  40. Call PSetupFreeMem to free the received string pointed by
  41. ppszReplacementDriver
  42. --*/
  43. BOOL
  44. PSetupIsCompatibleDriver(
  45. IN LPCTSTR pszServer, OPTIONAL
  46. IN LPCTSTR pszDriverModel,
  47. IN LPCTSTR pszDriverPath, // main rendering driver dll
  48. IN LPCTSTR pszEnvironment,
  49. IN DWORD dwVersion,
  50. IN FILETIME *pFileTimeDriver,
  51. OUT DWORD *pdwBlockingStatus,
  52. OUT LPTSTR *ppszReplacementDriver OPTIONAL // caller must free it.
  53. )
  54. {
  55. HRESULT hRetval = E_FAIL;
  56. UINT uBlockingStatus = 0;
  57. TString strReplacementDriver;
  58. hRetval = pszDriverModel && pszDriverPath && pszEnvironment && pFileTimeDriver && pdwBlockingStatus ? S_OK : E_INVALIDARG;
  59. if (SUCCEEDED(hRetval))
  60. {
  61. *pdwBlockingStatus = 0;
  62. if (ppszReplacementDriver)
  63. {
  64. *ppszReplacementDriver = NULL;
  65. }
  66. hRetval = IsLocalMachineServer();
  67. }
  68. if (SUCCEEDED(hRetval))
  69. {
  70. hRetval = InternalCompatibleDriverCheck(pszDriverModel,
  71. pszDriverPath,
  72. pszEnvironment,
  73. pFileTimeDriver,
  74. cszUpgradeInf,
  75. dwVersion,
  76. S_OK == hRetval ? TRUE : FALSE, // bIsServer
  77. &uBlockingStatus,
  78. &strReplacementDriver);
  79. }
  80. if (SUCCEEDED(hRetval))
  81. {
  82. *pdwBlockingStatus = uBlockingStatus;
  83. if ((BSP_PRINTER_DRIVER_OK != (*pdwBlockingStatus & BSP_BLOCKING_LEVEL_MASK)) && ppszReplacementDriver && !strReplacementDriver.bEmpty())
  84. {
  85. *ppszReplacementDriver = AllocStr(strReplacementDriver);
  86. hRetval = ppszReplacementDriver ? S_OK : E_OUTOFMEMORY;
  87. if (SUCCEEDED(hRetval))
  88. {
  89. *pdwBlockingStatus |= BSP_INBOX_DRIVER_AVAILABLE;
  90. }
  91. }
  92. }
  93. SetLastError(SUCCEEDED(hRetval) ? ERROR_SUCCESS : HRESULT_CODE(hRetval));
  94. return SUCCEEDED(hRetval);
  95. }
  96. /*++
  97. Routine Name:
  98. AddPrinterDriverUsingCorrectLevelWithPrintUpgRetry
  99. Routine Description:
  100. This routine tries to add the printer driver and it is blocked or
  101. warned, it popups a message box either indicates the driver is blocked
  102. and installation will abort or at the case of warned driver, whether
  103. to preceed the driver installation.
  104. Arguments:
  105. pszServer - The server that needs to check for bad driver. If
  106. pszServer is NULL, that means the local machine
  107. pDriverInfo6 - Points to DRINVER_INFO_6 structor
  108. dwAddDrvFlags - Flags used in AddPrinterDriver
  109. bIsDriverPathFullPath - Whether the driverpath is a full path
  110. bOfferReplacement - Whether to offer replacement
  111. bPopupUI - Whether to popup UI
  112. ppszReplacementDriver - Points to the replacement driver
  113. pdwBlockingStatus - Points to blocking status of the driver
  114. Return Value:
  115. An BOOL - GetLastError() on failure
  116. --*/
  117. BOOL
  118. AddPrinterDriverUsingCorrectLevelWithPrintUpgRetry(
  119. IN LPCTSTR pszServer, OPTIONAL
  120. IN DRIVER_INFO_6 *pDriverInfo6,
  121. IN DWORD dwAddDrvFlags,
  122. IN BOOL bIsDriverPathFullPath,
  123. IN BOOL bOfferReplacement,
  124. IN BOOL bPopupUI,
  125. OUT LPTSTR *ppszReplacementDriver,
  126. OUT DWORD *pdwBlockingStatus
  127. )
  128. {
  129. HRESULT hRetval = E_FAIL;
  130. DWORD dwLevel = 6;
  131. FILETIME DriverFileTime;
  132. hRetval = pDriverInfo6 && ppszReplacementDriver && pdwBlockingStatus ? S_OK : E_INVALIDARG;
  133. //
  134. // Set APD_NO_UI flag and call AddPrinterDriver.
  135. //
  136. if (SUCCEEDED(hRetval))
  137. {
  138. *pdwBlockingStatus = BSP_PRINTER_DRIVER_OK;
  139. *ppszReplacementDriver = NULL;
  140. dwAddDrvFlags |= APD_NO_UI;
  141. hRetval = AddPrinterDriverEx(const_cast<LPTSTR>(pszServer),
  142. dwLevel,
  143. reinterpret_cast<BYTE*>(pDriverInfo6),
  144. dwAddDrvFlags) ? ERROR_SUCCESS : GetLastErrorAsHResult();
  145. }
  146. for ( dwLevel = 4; (FAILED(hRetval) && (ERROR_INVALID_LEVEL == HRESULT_CODE(hRetval))) && (dwLevel > 1) ; --dwLevel ) {
  147. //
  148. // Since DRIVER_INFO_2, 3, 4 are subsets of DRIVER_INFO_6 and all fields
  149. // are at the beginning these calls can be made with same buffer
  150. //
  151. hRetval = AddPrinterDriverEx(const_cast<LPTSTR>(pszServer),
  152. dwLevel,
  153. reinterpret_cast<BYTE*>(pDriverInfo6),
  154. dwAddDrvFlags) ? S_OK : GetLastErrorAsHResult();
  155. }
  156. //
  157. // Set the blocking status information (either blocked or warned) from the
  158. // server.
  159. //
  160. // Show printupg ui and get the replacement driver from printupg.inf on
  161. // client and get user's response and retry AddPrinterDriver with level
  162. // dwLevel.
  163. //
  164. if (FAILED(hRetval) && bPopupUI && ((ERROR_PRINTER_DRIVER_BLOCKED == HRESULT_CODE(hRetval)) || (ERROR_PRINTER_DRIVER_WARNED == HRESULT_CODE(hRetval))))
  165. {
  166. *pdwBlockingStatus = (ERROR_PRINTER_DRIVER_BLOCKED == HRESULT_CODE(hRetval)) ? BSP_PRINTER_DRIVER_BLOCKED : BSP_PRINTER_DRIVER_WARNED;
  167. hRetval = PrintUpgRetry(pszServer,
  168. dwLevel,
  169. pDriverInfo6,
  170. dwAddDrvFlags,
  171. bIsDriverPathFullPath,
  172. bOfferReplacement,
  173. pdwBlockingStatus,
  174. ppszReplacementDriver);
  175. }
  176. SetLastError(SUCCEEDED(hRetval) ? ERROR_SUCCESS : HRESULT_CODE(hRetval));
  177. return SUCCEEDED(hRetval);
  178. }
  179. /*++
  180. Routine Name:
  181. BlockedDriverPrintUpgUI
  182. Routine Description:
  183. This routine checks the printupg.inf and see if the driver is there. If the
  184. driver is there, consider it blocked even if it is warned. Then it popups a
  185. message box either indicates the driver is blocked and ask whether the user
  186. wants to proceed to install a replacement driver it has one.
  187. Arguments:
  188. pszServer - The server that needs to check for bad driver. If
  189. pszServer is NULL, that means the local machine
  190. pDriverInfo6 - Points to DRINVER_INFO_6 structor
  191. bIsDriverPathFullPath - Whether the driverpath is a full path
  192. bOfferReplacement - Whether to offer replacement
  193. bPopupUI - Whether to popup UI
  194. ppszReplacementDriver - Points to the replacement driver
  195. pdwBlockingStatus - Points to blocking status of the driver
  196. Return Value:
  197. An BOOL - GetLastError() on failure
  198. --*/
  199. BOOL
  200. BlockedDriverPrintUpgUI(
  201. IN LPCTSTR pszServer, OPTIONAL
  202. IN DRIVER_INFO_6 *pDriverInfo6,
  203. IN BOOL bIsDriverPathFullPath,
  204. IN BOOL bOfferReplacement,
  205. IN BOOL bPopupUI,
  206. OUT LPTSTR *ppszReplacementDriver,
  207. OUT DWORD *pdwBlockingStatus
  208. )
  209. {
  210. HRESULT hRetval = E_FAIL;
  211. DWORD dwBlockingStatusOnClient = BSP_PRINTER_DRIVER_OK;
  212. hRetval = pDriverInfo6 && ppszReplacementDriver && pdwBlockingStatus ? S_OK : E_INVALIDARG;
  213. if (SUCCEEDED(hRetval))
  214. {
  215. *pdwBlockingStatus = BSP_PRINTER_DRIVER_OK;
  216. *ppszReplacementDriver = NULL;
  217. hRetval = IsDriverBadLocally(pszServer,
  218. pDriverInfo6,
  219. bIsDriverPathFullPath,
  220. &dwBlockingStatusOnClient,
  221. ppszReplacementDriver);
  222. //
  223. // Get the replacement driver information
  224. //
  225. if (SUCCEEDED(hRetval) && (BSP_PRINTER_DRIVER_OK != (dwBlockingStatusOnClient & BSP_BLOCKING_LEVEL_MASK)))
  226. {
  227. *pdwBlockingStatus |= (dwBlockingStatusOnClient & ~BSP_BLOCKING_LEVEL_MASK) | BSP_PRINTER_DRIVER_BLOCKED;
  228. }
  229. }
  230. if (SUCCEEDED(hRetval) && bPopupUI && (BSP_PRINTER_DRIVER_BLOCKED & *pdwBlockingStatus))
  231. {
  232. hRetval = PrintUpgRetry(pszServer,
  233. 6, // dwLevel, do not care in the case of blocked driver
  234. pDriverInfo6,
  235. 0, // dwAddDrvFlags, do not care in the case of blocked driver
  236. bIsDriverPathFullPath,
  237. bOfferReplacement,
  238. pdwBlockingStatus,
  239. ppszReplacementDriver);
  240. }
  241. SetLastError(SUCCEEDED(hRetval) ? ERROR_SUCCESS : HRESULT_CODE(hRetval));
  242. return SUCCEEDED(hRetval);
  243. }
  244. /*++
  245. Routine Name:
  246. PrintUpgRetry
  247. Routine Description:
  248. This routine popups up a message box either indicates the driver is blocked
  249. or warned and asks the user how to preceed. In the case of warned driver,
  250. whether to preceed the driver installation and retry AddPrinterDriver if
  251. the user wants to proceed with a warned driver.
  252. Arguments:
  253. pszServer - Remote machine that has the driver files
  254. dwLevel - Driver info level, since DRIVER_INFO_6
  255. is a super set of level 4, 3, and 2,
  256. the driver info structure is shared
  257. pDriverInfo6 - Points to DRIVER_INFO_6 structure
  258. dwAddDrvFlags - Flags used to AddPrinterDriver
  259. bIsDriverPathFullPath - Whether the driverpath is a full path
  260. bOffereReplacement - Whether to offer a replacement driver
  261. pdwBlockingStatus - Points to blocking status on server
  262. ppszReplacementDriver - Points to the replacement driver
  263. Return Value:
  264. An HRESULT
  265. --*/
  266. HRESULT
  267. PrintUpgRetry(
  268. IN LPCTSTR pszServer, OPTIONAL
  269. IN DWORD dwLevel,
  270. IN DRIVER_INFO_6 *pDriverInfo6,
  271. IN DWORD dwAddDrvFlags,
  272. IN BOOL bIsDriverPathFullPath,
  273. IN BOOL bOfferReplacement,
  274. IN OUT DWORD *pdwBlockingStatus,
  275. OUT LPTSTR *ppszReplacementDriver
  276. )
  277. {
  278. HRESULT hRetval = E_FAIL;
  279. hRetval = pDriverInfo6 && pdwBlockingStatus && ppszReplacementDriver && ((6 == dwLevel) || (dwLevel >= 2) && (dwLevel <= 4)) ? S_OK : E_INVALIDARG;
  280. if (SUCCEEDED(hRetval))
  281. {
  282. *ppszReplacementDriver = NULL;
  283. hRetval = PrintUpgUI(pszServer,
  284. pDriverInfo6,
  285. bIsDriverPathFullPath,
  286. bOfferReplacement,
  287. pdwBlockingStatus,
  288. ppszReplacementDriver);
  289. //
  290. // There are 4 cases here:
  291. // 1. For warned driver and the user instructs to install it, try
  292. // AddPrinterDriverEx again with APD_INSTALL_WARNED_DRIVER.
  293. // 2. If the user wants to cancel, set the last error correctly and
  294. // abort.
  295. // 3. If the user wants to install the replacement driver, we will
  296. // set the error code correctly and do not install the replacement
  297. // driver at this moment since we shall clean ourself up first.
  298. // 4. If other errors occur, we will just return the correct error
  299. // code.
  300. //
  301. // For case 1, 2, 4, we do not return the replacement driver because
  302. // I can not see any reason to do so.
  303. //
  304. if (FAILED(hRetval) || !(*pdwBlockingStatus & BSP_PRINTER_DRIVER_REPLACED)) // except case 3
  305. {
  306. LocalFreeMem(*ppszReplacementDriver);
  307. *ppszReplacementDriver = NULL;
  308. }
  309. if (SUCCEEDED(hRetval) && (*pdwBlockingStatus & BSP_PRINTER_DRIVER_PROCEEDED)) // case 1
  310. {
  311. dwAddDrvFlags |= APD_INSTALL_WARNED_DRIVER;
  312. hRetval = AddPrinterDriverEx(const_cast<LPTSTR>(pszServer),
  313. dwLevel,
  314. reinterpret_cast<BYTE*>(pDriverInfo6),
  315. dwAddDrvFlags) ? ERROR_SUCCESS : GetLastErrorAsHResult();
  316. }
  317. else if (SUCCEEDED(hRetval) && (*pdwBlockingStatus & BSP_PRINTER_DRIVER_CANCELLED)) // case 2
  318. {
  319. hRetval = HRESULT_FROM_WIN32(ERROR_CANCELLED);
  320. }
  321. else if (SUCCEEDED(hRetval) && (*pdwBlockingStatus & BSP_PRINTER_DRIVER_REPLACED)) // case 3
  322. {
  323. hRetval = HResultFromWin32((*pdwBlockingStatus & BSP_PRINTER_DRIVER_BLOCKED) ? ERROR_PRINTER_DRIVER_BLOCKED : ERROR_PRINTER_DRIVER_WARNED);
  324. }
  325. }
  326. return hRetval;
  327. }
  328. /*++
  329. Routine Name:
  330. PrintUpgUI
  331. Routine Description:
  332. This routine asks ntprint.dll to popup a message box either indicates
  333. the driver is blocked and installation will abort or at the case of
  334. warned driver, whether to preceed the driver installation.
  335. Arguments:
  336. pszServer - Remote machine that has the driver files
  337. pDriverInfo6 - Points to DRINVER_INFO_6 structor
  338. bIsDriverPathFullPath - Whether the driverpath is a full path
  339. bOffereReplacement - Whether to offer a replacement driver
  340. pdwBlockingStatus - Points to blocking status on server
  341. ppszReplacementDriver - Points to the replacement driver
  342. Return Value:
  343. An HRESULT
  344. --*/
  345. HRESULT
  346. PrintUpgUI(
  347. IN LPCTSTR pszServer, OPTIONAL
  348. IN DRIVER_INFO_6 *pDriverInfo6,
  349. IN BOOL bIsDriverPathFullPath,
  350. IN BOOL bOfferReplacement,
  351. IN OUT DWORD *pdwBlockingStatus,
  352. OUT LPTSTR *ppszReplacementDriver
  353. )
  354. {
  355. HRESULT hRetval = E_FAIL;
  356. DWORD dwBlockingStatusOnClient = BSP_PRINTER_DRIVER_OK;
  357. hRetval = pDriverInfo6 && pdwBlockingStatus && ppszReplacementDriver ? S_OK : E_INVALIDARG;
  358. //
  359. // Take the replacement driver information from local client.
  360. //
  361. if (SUCCEEDED(hRetval) && bOfferReplacement)
  362. {
  363. *ppszReplacementDriver = NULL;
  364. hRetval = IsDriverBadLocally(pszServer,
  365. pDriverInfo6,
  366. bIsDriverPathFullPath,
  367. &dwBlockingStatusOnClient,
  368. ppszReplacementDriver);
  369. //
  370. // Get the replacement driver information
  371. //
  372. if (SUCCEEDED(hRetval))
  373. {
  374. *pdwBlockingStatus |= (dwBlockingStatusOnClient & ~BSP_BLOCKING_LEVEL_MASK);
  375. }
  376. }
  377. //
  378. // Get the user's response
  379. //
  380. if (SUCCEEDED(hRetval))
  381. {
  382. *pdwBlockingStatus &= ~BSP_USER_RESPONSE_MASK;
  383. hRetval = InternalPrintUpgUI(pDriverInfo6->pName,
  384. pDriverInfo6->pDriverPath, // main rendering driver dll
  385. pDriverInfo6->pEnvironment,
  386. pDriverInfo6->cVersion,
  387. pdwBlockingStatus);
  388. }
  389. return hRetval;
  390. }
  391. /*++
  392. Routine Name
  393. InfIsCompatibleDriver
  394. Routine Description:
  395. Check whether the driver is blocked or warned and if the driver is blocked
  396. or warned, return the blocking status and the replacement driver, if any.
  397. Arguments:
  398. pszDriverModel - The name of the driver to check
  399. pszDriverPath - The path of the driver, this can be a full path
  400. or the filename
  401. pszEnvironment - The environment of the server, such as
  402. "Windows NT x86"
  403. hInf - Handle to printupg inf
  404. puBlockingStatus - Points to status of blocking
  405. ppszReplacementDriver - Points to the NULL terminating name of the replacement
  406. driver. A *NULL* string (instead of an empty string)
  407. means there is no replacement driver
  408. Return Value:
  409. A BOOL - TRUE if success; FALSE otherwise, Call GetLastError()
  410. to get the Error code. Pass a NULL to ppszReplacementDriver
  411. will not receive the name of replacement driver otherwise
  412. Call PSetupFreeMem to free the received string pointed by
  413. ppszReplacementDriver
  414. --*/
  415. BOOL
  416. InfIsCompatibleDriver(
  417. IN LPCTSTR pszDriverModel,
  418. IN LPCTSTR pszDriverPath, // main rendering driver dll
  419. IN LPCTSTR pszEnvironment,
  420. IN HINF hInf,
  421. OUT DWORD *pdwBlockingStatus,
  422. OUT LPTSTR *ppszReplacementDriver OPTIONAL // caller must free it.
  423. )
  424. {
  425. HRESULT hRetval = E_FAIL;
  426. UINT uBlockingStatus = 0;
  427. DWORD dwMajorVersion = 0;
  428. BOOL bIsServer = FALSE;
  429. TString strReplacementDriver;
  430. FILETIME DriverFileTime;
  431. hRetval = pszDriverModel && pszDriverPath && pszEnvironment && (hInf != INVALID_HANDLE_VALUE) && pdwBlockingStatus ? S_OK : E_INVALIDARG;
  432. if (SUCCEEDED(hRetval))
  433. {
  434. *pdwBlockingStatus = 0;
  435. if (ppszReplacementDriver)
  436. {
  437. *ppszReplacementDriver = NULL;
  438. }
  439. hRetval = GetPrinterDriverVersion(pszDriverPath, &dwMajorVersion, NULL);
  440. }
  441. if (SUCCEEDED(hRetval))
  442. {
  443. hRetval = IsLocalMachineServer();
  444. }
  445. if (SUCCEEDED(hRetval))
  446. {
  447. bIsServer = S_OK == hRetval ? TRUE : FALSE;
  448. hRetval = GetFileTimeByName(pszDriverPath, &DriverFileTime);
  449. }
  450. if (S_OK == hRetval)
  451. {
  452. hRetval = InternalCompatibleInfDriverCheck(pszDriverModel,
  453. pszDriverPath,
  454. pszEnvironment,
  455. &DriverFileTime,
  456. hInf,
  457. dwMajorVersion,
  458. bIsServer,
  459. &uBlockingStatus,
  460. &strReplacementDriver);
  461. }
  462. if (SUCCEEDED(hRetval))
  463. {
  464. *pdwBlockingStatus = uBlockingStatus;
  465. if ((BSP_PRINTER_DRIVER_OK != (*pdwBlockingStatus & BSP_BLOCKING_LEVEL_MASK)) && ppszReplacementDriver && !strReplacementDriver.bEmpty())
  466. {
  467. *ppszReplacementDriver = AllocStr(strReplacementDriver);
  468. hRetval = ppszReplacementDriver ? S_OK : E_OUTOFMEMORY;
  469. if (SUCCEEDED(hRetval))
  470. {
  471. *pdwBlockingStatus |= BSP_INBOX_DRIVER_AVAILABLE;
  472. }
  473. }
  474. }
  475. SetLastError(SUCCEEDED(hRetval) ? ERROR_SUCCESS : HRESULT_CODE(hRetval));
  476. return SUCCEEDED(hRetval);
  477. }
  478. /*++
  479. Routine Name:
  480. GetPrinterDriverVersion
  481. Routine Description:
  482. Gets version information about an executable file. If the file is not an
  483. executable, it will return 0 for both major and minor version.
  484. Arguments:
  485. pszFileName - file name, this is either a full path for
  486. the file is under the path in the search
  487. sequence specified used the LoadLibrary
  488. function.
  489. pdwFileMajorVersion - pointer to major version
  490. pdwFileMinorVersion - pointer to minor version
  491. Return Value:
  492. An HRESULT
  493. --*/
  494. HRESULT
  495. GetPrinterDriverVersion(
  496. IN LPCTSTR pszFileName,
  497. OUT DWORD *pdwFileMajorVersion, OPTIONAL
  498. OUT DWORD *pdwFileMinorVersion OPTIONAL
  499. )
  500. {
  501. HRESULT hRetval = E_FAIL;
  502. DWORD dwSize = 0;
  503. UINT uLen = 0;
  504. BYTE *pbBlock = NULL;
  505. VS_FIXEDFILEINFO *pFileVersion = NULL;
  506. DWORD dwHandle = 0;
  507. hRetval = pszFileName && *pszFileName ? S_OK : E_INVALIDARG;
  508. if (SUCCEEDED(hRetval))
  509. {
  510. dwSize = GetFileVersionInfoSize(const_cast<LPTSTR>(pszFileName), &dwHandle );
  511. if (dwSize == 0)
  512. {
  513. hRetval = GetLastErrorAsHResult();
  514. }
  515. }
  516. if (SUCCEEDED(hRetval))
  517. {
  518. pbBlock = new BYTE[dwSize];
  519. hRetval = pbBlock ? S_OK : E_OUTOFMEMORY;
  520. }
  521. if (SUCCEEDED(hRetval))
  522. {
  523. hRetval = GetFileVersionInfo(const_cast<LPTSTR>(pszFileName), 0, dwSize, pbBlock) ? S_OK : GetLastErrorAsHResult();
  524. }
  525. //
  526. // VerQueryValue does not set last error.
  527. //
  528. if (SUCCEEDED(hRetval))
  529. {
  530. hRetval = VerQueryValue(pbBlock, _T("\\"), reinterpret_cast<VOID **> (&pFileVersion), &uLen) && pFileVersion && uLen ? S_OK : E_INVALIDARG;
  531. }
  532. if (SUCCEEDED(hRetval))
  533. {
  534. hRetval = GetDriverVersionFromFileVersion(pFileVersion, pdwFileMajorVersion, pdwFileMinorVersion);
  535. }
  536. delete [] pbBlock;
  537. return hRetval;
  538. }
  539. /*++
  540. Routine Name:
  541. GetDriverVersionFromFileVersion
  542. Routine Description:
  543. Gets driver info from a FileVersion structure.
  544. Arguments:
  545. pFileVersion - Points to a file info structure
  546. pdwFileMajorVersion - pointer to major version
  547. pdwFileMinorVersion - pointer to minor version
  548. Return Value:
  549. An HRESULT
  550. --*/
  551. HRESULT
  552. GetDriverVersionFromFileVersion(
  553. IN VS_FIXEDFILEINFO *pFileVersion,
  554. OUT DWORD *pdwFileMajorVersion, OPTIONAL
  555. OUT DWORD *pdwFileMinorVersion OPTIONAL
  556. )
  557. {
  558. HRESULT hRetval = E_FAIL;
  559. hRetval = pFileVersion ? S_OK : E_INVALIDARG;
  560. //
  561. // Return versions for drivers designed for Windows NT/Windows 2000,
  562. // and marked as printer drivers.
  563. // Hold for all dlls Pre-Daytona.
  564. // After Daytona, printer driver writers must support
  565. // version control or we'll dump them as Version 0 drivers.
  566. //
  567. if (SUCCEEDED(hRetval))
  568. {
  569. if (pdwFileMajorVersion)
  570. {
  571. *pdwFileMajorVersion = 0;
  572. }
  573. if (pdwFileMinorVersion)
  574. {
  575. *pdwFileMinorVersion = 0;
  576. }
  577. if (VOS_NT_WINDOWS32 == pFileVersion->dwFileOS)
  578. {
  579. if ((VFT_DRV == pFileVersion->dwFileType) && (VFT2_DRV_VERSIONED_PRINTER == pFileVersion->dwFileSubtype))
  580. {
  581. if (pdwFileMajorVersion)
  582. {
  583. *pdwFileMajorVersion = pFileVersion->dwFileVersionMS;
  584. }
  585. if (pdwFileMinorVersion)
  586. {
  587. *pdwFileMinorVersion = pFileVersion->dwFileVersionLS;
  588. }
  589. }
  590. else if (pdwFileMajorVersion)
  591. {
  592. if (pFileVersion->dwProductVersionMS == pFileVersion->dwFileVersionMS)
  593. {
  594. *pdwFileMajorVersion = 0;
  595. }
  596. else
  597. {
  598. *pdwFileMajorVersion = pFileVersion->dwFileVersionMS;
  599. }
  600. }
  601. }
  602. }
  603. return hRetval;
  604. }
  605. /*++
  606. Routine Name:
  607. IsDriverBadLocally
  608. Routine Description:
  609. This routine loads the file and gets the FILETIME of the driver and check
  610. whether the driver is blokced or warned according the printupg.inf on the
  611. local machine.
  612. Arguments:
  613. pszServer - The server that has the driver files
  614. pDriverInfo6 - Points to DRINVER_INFO_6 structor
  615. dwAddDrvFlags - Flags used in AddPrinterDriver
  616. bIsDriverPathFullPath - Whether the driverpath is a full path
  617. pdwBlockingStatus - Points to blocking status of the driver
  618. ppszReplacementDriver - Points to the replacement driver
  619. Return Value:
  620. An HRESULT
  621. --*/
  622. HRESULT
  623. IsDriverBadLocally(
  624. IN LPCTSTR pszServer, OPTIONAL
  625. IN DRIVER_INFO_6 *pDriverInfo6,
  626. IN BOOL bIsDriverPathFullPath,
  627. OUT DWORD *pdwBlockingStatus,
  628. OUT LPTSTR *ppszReplacementDriver
  629. )
  630. {
  631. HRESULT hRetval = E_FAIL;
  632. DWORD dwMajorVersion = 0;
  633. FILETIME DriverFileTime;
  634. TString strDriverFullPath;
  635. hRetval = pDriverInfo6 && pdwBlockingStatus && ppszReplacementDriver ? S_OK : E_INVALIDARG;
  636. if (SUCCEEDED(hRetval))
  637. {
  638. hRetval = GetPrinterDriverPath(pszServer,
  639. pDriverInfo6->pDriverPath,
  640. pDriverInfo6->pEnvironment,
  641. bIsDriverPathFullPath,
  642. &strDriverFullPath);
  643. }
  644. if (SUCCEEDED(hRetval))
  645. {
  646. hRetval = GetFileTimeByName(strDriverFullPath, &DriverFileTime);
  647. }
  648. if (SUCCEEDED(hRetval))
  649. {
  650. hRetval = GetPrinterDriverVersion(strDriverFullPath, &dwMajorVersion, NULL);
  651. }
  652. if (SUCCEEDED(hRetval))
  653. {
  654. hRetval = PSetupIsCompatibleDriver(NULL,
  655. pDriverInfo6->pName,
  656. pDriverInfo6->pDriverPath,
  657. pDriverInfo6->pEnvironment,
  658. dwMajorVersion,
  659. &DriverFileTime,
  660. pdwBlockingStatus,
  661. ppszReplacementDriver) ? S_OK : GetLastErrorAsHResult();
  662. }
  663. if (FAILED(hRetval) && ppszReplacementDriver && *ppszReplacementDriver)
  664. {
  665. LocalFreeMem(*ppszReplacementDriver);
  666. *ppszReplacementDriver = NULL;
  667. }
  668. return hRetval;
  669. }
  670. /*++
  671. Routine Name:
  672. InternalPrintUpgUI
  673. Routine Description:
  674. This routine asks ntprint.dll to popup a message box either indicates
  675. the driver is blocked and installation will abort or at the case of
  676. warned driver, whether to preceed the driver installation.
  677. Arguments:
  678. pszDriverModel - The name of the driver to check
  679. pszDriverPath - The path of the driver, this can be a full path
  680. or the filename
  681. pszEnvironment - The environment of the server, such as
  682. "Windows NT x86"
  683. dwVersion - The major version of the driver
  684. pdwBlockingStatus - Points to blocking status on the client
  685. ppszReplacementDriver - Points to the name of replacement driver
  686. Return Value:
  687. An HRESULT - When this function is successful, user choose
  688. either proceed to install warned driver or install
  689. replacement driver.
  690. --*/
  691. HRESULT
  692. InternalPrintUpgUI(
  693. IN LPCTSTR pszDriverModel,
  694. IN LPCTSTR pszDriverPath, // main rendering driver dll
  695. IN LPCTSTR pszEnvironment,
  696. IN DWORD dwVersion,
  697. IN OUT DWORD *pdwBlockingStatus
  698. )
  699. {
  700. DWORD hRetval = E_FAIL;
  701. HWND hWndParent = NULL;
  702. DWORD dwStatusResponsed = BSP_PRINTER_DRIVER_OK;
  703. hRetval = pszDriverModel && pszDriverPath && pszEnvironment && pdwBlockingStatus ? S_OK : E_INVALIDARG;
  704. if (SUCCEEDED(hRetval) && (BSP_PRINTER_DRIVER_OK != (*pdwBlockingStatus & BSP_BLOCKING_LEVEL_MASK)))
  705. {
  706. *pdwBlockingStatus &= ~BSP_USER_RESPONSE_MASK;
  707. hWndParent = SUCCEEDED(GetCurrentThreadLastPopup(&hWndParent)) ? hWndParent : NULL;
  708. //
  709. // Ask the user what they want to do. If they don't want to proceed,
  710. // then the error is what the would get from the localspl call.
  711. //
  712. // PSetupShowBlockedDriverUI can not fail!
  713. //
  714. *pdwBlockingStatus |= (PSetupShowBlockedDriverUI(hWndParent, *pdwBlockingStatus) & BSP_USER_RESPONSE_MASK);
  715. }
  716. return hRetval;
  717. }
  718. //
  719. // This function is trying to get the last active popup of the top
  720. // level owner of the current thread active window.
  721. //
  722. HRESULT
  723. GetCurrentThreadLastPopup(
  724. OUT HWND *phwnd
  725. )
  726. {
  727. HWND hwndOwner, hwndParent;
  728. HRESULT hr = E_INVALIDARG;
  729. GUITHREADINFO ti = {0};
  730. if( phwnd )
  731. {
  732. hr = E_FAIL;
  733. *phwnd = NULL;
  734. ti.cbSize = sizeof(ti);
  735. if( GetGUIThreadInfo(0, &ti) && ti.hwndActive )
  736. {
  737. *phwnd = ti.hwndActive;
  738. // climb up to the top parent in case it's a child window...
  739. while( hwndParent = GetParent(*phwnd) )
  740. {
  741. *phwnd = hwndParent;
  742. }
  743. // get the owner in case the top parent is owned
  744. hwndOwner = GetWindow(*phwnd, GW_OWNER);
  745. if( hwndOwner )
  746. {
  747. *phwnd = hwndOwner;
  748. }
  749. // get the last popup of the owner window
  750. if( *phwnd)
  751. {
  752. *phwnd = GetLastActivePopup(*phwnd);
  753. }
  754. hr = (*phwnd) ? S_OK : E_FAIL;
  755. }
  756. }
  757. return hr;
  758. }
  759. /*++
  760. Routine Name
  761. GetPrinterDriverPath
  762. Routine Description:
  763. Get the file full path of the driver.
  764. Arguments:
  765. pszServer - The server that needs to check for bad driver. If
  766. pszServer is NULL, that means the local machine
  767. pszDriverPath - The path of the driver, this can be a full path or
  768. the filename
  769. pszEnvironment - The environment of the server, such as
  770. "Windows NT x86"
  771. bIsDriverpathFullPath - Whether pszDriverPath is the full path or just the
  772. the file name of the driver
  773. pstrFull - The full path of the driver
  774. Return Value:
  775. An HRESULT
  776. --*/
  777. HRESULT
  778. GetPrinterDriverPath(
  779. IN LPCTSTR pszServer, OPTIONAL
  780. IN LPCTSTR pszDriverPath,
  781. IN LPCTSTR pszEnvironment,
  782. IN BOOL bIsDriverPathFullPath,
  783. OUT TString *pstrFullPath
  784. )
  785. {
  786. HRESULT hRetval = E_FAIL;
  787. TCHAR szDir[MAX_PATH] = {0};
  788. DWORD dwNeeded = 0;
  789. hRetval = pszEnvironment && *pszEnvironment && pszDriverPath && *pszDriverPath && pstrFullPath ? S_OK : E_INVALIDARG;
  790. if (SUCCEEDED(hRetval) && !bIsDriverPathFullPath)
  791. {
  792. hRetval = GetPrinterDriverDirectory(const_cast<LPTSTR>(pszServer), // This API is designed wrongly.
  793. const_cast<LPTSTR>(pszEnvironment),
  794. 1, // This value must be 1
  795. (LPBYTE)szDir,
  796. sizeof(szDir),
  797. &dwNeeded) ? S_OK : GetLastErrorAsHResult();
  798. }
  799. if (FAILED(hRetval) && (ERROR_INSUFFICIENT_BUFFER == HRESULT_CODE(hRetval)))
  800. {
  801. hRetval = HRESULT_FROM_WIN32(ERROR_FILENAME_EXCED_RANGE);
  802. }
  803. //
  804. // Add 1 '\\' between driverpath and filename. Note if the path is too
  805. // long StrNCatBuff will truncate it and StrNCatBuff always NULL terminates
  806. // the buffer.
  807. //
  808. if (SUCCEEDED(hRetval))
  809. {
  810. hRetval = HResultFromWin32(StrNCatBuff(szDir, COUNTOF(szDir), szDir, *szDir ? _T("\\") : _T(""), pszDriverPath, NULL));
  811. }
  812. if (SUCCEEDED(hRetval))
  813. {
  814. hRetval = pstrFullPath->bUpdate(szDir) ? S_OK : E_OUTOFMEMORY;
  815. }
  816. return hRetval;
  817. }
  818. /*++
  819. Routine Name
  820. GetFileTimeByName
  821. Routine Description:
  822. Get the file time of the file given a full path.
  823. Arguments:
  824. pszPath - Full path of the driver
  825. pFileTime - Points to the file time
  826. Return Value:
  827. An HRESULT
  828. --*/
  829. HRESULT
  830. GetFileTimeByName(
  831. IN LPCTSTR pszPath,
  832. OUT FILETIME *pFileTime
  833. )
  834. {
  835. HRESULT hRetval = E_FAIL;
  836. HANDLE hFile = INVALID_HANDLE_VALUE;
  837. hRetval = pszPath && *pszPath && pFileTime ? S_OK : E_INVALIDARG;
  838. if (SUCCEEDED(hRetval))
  839. {
  840. hFile = CreateFile(pszPath,
  841. GENERIC_READ,
  842. FILE_SHARE_READ,
  843. NULL,
  844. OPEN_EXISTING,
  845. FILE_ATTRIBUTE_NORMAL,
  846. NULL);
  847. hRetval = (INVALID_HANDLE_VALUE != hFile) ? S_OK : GetLastErrorAsHResult();
  848. }
  849. if (SUCCEEDED(hRetval))
  850. {
  851. hRetval = GetFileTime(hFile, NULL, NULL, pFileTime) ? S_OK : GetLastErrorAsHResult();
  852. }
  853. if (hFile != INVALID_HANDLE_VALUE)
  854. {
  855. CloseHandle(hFile);
  856. }
  857. return hRetval;
  858. }
  859. /*++
  860. Routine Name
  861. InternalCompatibleDriverCheck
  862. Routine Description:
  863. Check whether the driver is blocked or warned and if the driver
  864. is blocked or warned, return the replacement driver.
  865. Arguments:
  866. pszDriverModel - The name of the driver to check
  867. pszDriverPath - The path of the driver, this can be a full path or
  868. the filename
  869. pszEnvironment - The environment of the server, such as
  870. "Windows NT x86"
  871. pFileTimeDriver - The FileTime of the driver
  872. pszPrintUpgInf - The PrintUpg Inf filename
  873. uVersion - The major version of the driver
  874. bIsServer - Where the printing services runs on an NT Server SKU
  875. puBlockingStatus - Points to status of blocking
  876. pstrReplacementDriver - The replacement driver.
  877. Return Value:
  878. An HRESULT
  879. --*/
  880. HRESULT
  881. InternalCompatibleDriverCheck(
  882. IN LPCTSTR pszDriverModel,
  883. IN LPCTSTR pszDriverPath,
  884. IN LPCTSTR pszEnvironment,
  885. IN FILETIME *pFileTimeDriver,
  886. IN LPCTSTR pszPrintUpgInf,
  887. IN UINT uVersion,
  888. IN BOOL bIsServer,
  889. OUT UINT *puBlockingStatus,
  890. OUT TString *pstrReplacementDriver
  891. )
  892. {
  893. DBG_MSG(DBG_TRACE, (_T("In the InternalCompatibleDriverCheck routine\n")));
  894. HRESULT hRetval = E_FAIL;
  895. HINF hInf = INVALID_HANDLE_VALUE;
  896. hRetval = pszDriverModel && pszDriverPath && pszEnvironment && pFileTimeDriver && pszPrintUpgInf && puBlockingStatus && pstrReplacementDriver ? S_OK : E_INVALIDARG;
  897. if (SUCCEEDED(hRetval))
  898. {
  899. *puBlockingStatus = 0;
  900. //
  901. // NOTICE-rorleth-2002/01/10: fixed bug 761840 by using flag INF_STYLE_CACHE_IGNORE
  902. // in case we're called from the spooler's context and printupg.pnf needs to be
  903. // created, the user's hive would not be unloaded on logoff as the cryptoapi
  904. // cache lives partially in the user's hive but the spooler is still running
  905. //
  906. hInf = SetupOpenInfFile(pszPrintUpgInf, NULL, INF_STYLE_WIN4 | INF_STYLE_CACHE_IGNORE, NULL);
  907. hRetval = (INVALID_HANDLE_VALUE != hInf) ? S_OK : GetLastErrorAsHResult();
  908. }
  909. if (SUCCEEDED(hRetval))
  910. {
  911. hRetval = InternalCompatibleInfDriverCheck(pszDriverModel,
  912. pszDriverPath,
  913. pszEnvironment,
  914. pFileTimeDriver,
  915. hInf,
  916. uVersion,
  917. bIsServer,
  918. puBlockingStatus,
  919. pstrReplacementDriver);
  920. }
  921. if (INVALID_HANDLE_VALUE != hInf)
  922. {
  923. SetupCloseInfFile(hInf);
  924. }
  925. return hRetval;
  926. }
  927. /*++
  928. Routine Name
  929. InternalCompatibleInfDriverCheck
  930. Routine Description:
  931. Check whether the driver is blocked or warned and if the driver
  932. is blocked or warned, return the replacement driver.
  933. Arguments:
  934. pszDriverModel - The name of the driver to check
  935. pszDriverPath - The path of the driver, this can be a full path or
  936. the filename
  937. pszEnvironment - The environment of the server, such as
  938. "Windows NT x86"
  939. pFileTimeDriver - The FileTime of the driver
  940. hPrintUpgInf - The handle to the PrintUpg Inf file
  941. uVersion - The major version of the driver
  942. bIsServer - Where the printing services runs on an NT Server SKU
  943. puBlockingStatus - Points to status of blocking
  944. pstrReplacementDriver - The replacement driver.
  945. Return Value:
  946. An HRESULT
  947. --*/
  948. HRESULT
  949. InternalCompatibleInfDriverCheck(
  950. IN LPCTSTR pszModelName,
  951. IN LPCTSTR pszDriverPath,
  952. IN LPCTSTR pszEnvironment,
  953. IN FILETIME *pFileTimeDriver,
  954. IN HINF hPrintUpgInf,
  955. IN UINT uVersion,
  956. IN BOOL bIsServer,
  957. OUT UINT *puBlockingStatus,
  958. OUT TString *pstrReplacementDriver OPTIONAL
  959. )
  960. {
  961. HRESULT hRetval = E_FAIL;
  962. HINF hInf = INVALID_HANDLE_VALUE;
  963. UINT uWarnLevelSrv = 0;
  964. UINT uWarnLevelWks = 0;
  965. hRetval = pszModelName && pszDriverPath && pszEnvironment && pFileTimeDriver && (INVALID_HANDLE_VALUE != hPrintUpgInf) && puBlockingStatus && pstrReplacementDriver? S_OK : E_INVALIDARG;
  966. if (SUCCEEDED(hRetval))
  967. {
  968. hRetval = IsDriverDllInExcludedSection(pszDriverPath, hPrintUpgInf);
  969. }
  970. if (S_FALSE == hRetval)
  971. {
  972. hRetval = IsDriverInMappingSection(pszModelName,
  973. pszEnvironment,
  974. uVersion,
  975. hPrintUpgInf,
  976. pFileTimeDriver,
  977. &uWarnLevelSrv,
  978. &uWarnLevelWks,
  979. pstrReplacementDriver);
  980. if (S_OK == hRetval)
  981. {
  982. hRetval = GetBlockingStatusByWksType(uWarnLevelSrv, uWarnLevelWks, bIsServer, puBlockingStatus);
  983. }
  984. }
  985. if (SUCCEEDED(hRetval))
  986. {
  987. DBG_MSG(DBG_TRACE, (_T("Driver \"%s\" Driverpath \"%s\" Environment \"%s\" Version %d WarnLevel Server %d WarnLevel Wks %d bIsServer %d *Blocking Status* 0X%X *Replacement Driver* \"%s\"\n"), pszModelName, pszDriverPath, pszEnvironment,uVersion, uWarnLevelSrv, uWarnLevelSrv, bIsServer, *puBlockingStatus, static_cast<LPCTSTR>(*pstrReplacementDriver)));
  988. hRetval = (BSP_PRINTER_DRIVER_OK == (*puBlockingStatus & BSP_BLOCKING_LEVEL_MASK)) ? S_OK : S_FALSE;
  989. }
  990. return hRetval;
  991. }
  992. /*++
  993. Routine Name
  994. IsDriverInMappingSection
  995. Routine Description:
  996. Check whether the driver is mapped, aka a bad driver.
  997. Arguments:
  998. pszDriverModel - The name of the driver to check
  999. pszEnvironment - The environment of the server, such as
  1000. uVersion - The major version of the driver
  1001. hPrintUpgInf - The handle to the PrintUpg Inf file
  1002. pFileTimeDriver - Points to the file time of the driver
  1003. pdwWarnLevelSrv - Points to the warning level for server SKU
  1004. pdwWarnLevelWks - Points to the warning level for wks SKU
  1005. pstrReplacementDriver - The replacement driver
  1006. Return Value:
  1007. An HRESULT - S_OK means the driver is a bad driver and is mapped to
  1008. some inbox driver, S_FALSE means the driver is not.
  1009. --*/
  1010. HRESULT
  1011. IsDriverInMappingSection(
  1012. IN LPCTSTR pszModelName,
  1013. IN LPCTSTR pszEnvironment,
  1014. IN UINT uVersion,
  1015. IN HINF hPrintUpgInf,
  1016. IN FILETIME *pFileTimeDriver,
  1017. OUT UINT *puWarnLevelSrv,
  1018. OUT UINT *puWarnLevelWks,
  1019. OUT TString *pstrReplacementDriver
  1020. )
  1021. {
  1022. DBG_MSG(DBG_TRACE, (_T("In the IsDriverInMappingSection routine\n")));
  1023. HRESULT hRetval = E_FAIL;
  1024. UINT uWarnLevelSrv = 0;
  1025. UINT uWarnLevelWks = 0;
  1026. INFCONTEXT InfContext;
  1027. TString strMappingSection;
  1028. TString strReplacementDriver;
  1029. hRetval = pszModelName && pszEnvironment && (INVALID_HANDLE_VALUE != hPrintUpgInf) && pFileTimeDriver && puWarnLevelSrv && puWarnLevelWks && pstrReplacementDriver ? S_OK : E_INVALIDARG;
  1030. if (SUCCEEDED(hRetval))
  1031. {
  1032. *puWarnLevelSrv = 0;
  1033. *puWarnLevelWks = 0;
  1034. hRetval = GetSectionName(pszEnvironment, uVersion, &strMappingSection);
  1035. }
  1036. if (SUCCEEDED(hRetval))
  1037. {
  1038. hRetval = SetupFindFirstLine(hPrintUpgInf, strMappingSection, pszModelName, &InfContext) ? S_FALSE : GetLastErrorAsHResult();
  1039. }
  1040. //
  1041. // This code assumes that:
  1042. //
  1043. // There can be multiple lines for the same printer driver, but they
  1044. // are sorted in non-descreasing order by date, the last field of the
  1045. // line. The fist line that has the date no older than the driver's
  1046. // date is used.
  1047. //
  1048. // An interesting case would be like (since date is optional)
  1049. //
  1050. // "HP LaserJet 4" = "HP LaserJet 4", 1, 2, "11/28/1999"
  1051. // "HP LaserJet 4" = "HP LaserJet 4", 2, 1
  1052. //
  1053. // If a date is empty then the driver of all dates are blocked, hence
  1054. // an empty date means close to a very late date in the future.
  1055. //
  1056. for (;S_FALSE == hRetval;)
  1057. {
  1058. hRetval = IsDateInLineNoOlderThanDriverDate(&InfContext, pFileTimeDriver, &uWarnLevelSrv, &uWarnLevelWks, &strReplacementDriver);
  1059. if (S_FALSE == hRetval)
  1060. {
  1061. hRetval = SetupFindNextMatchLine(&InfContext, pszModelName, &InfContext) ? S_FALSE : GetLastErrorAsHResult();
  1062. }
  1063. }
  1064. //
  1065. // ERROR_LINE_NOT_FOUND is an HRESULT!
  1066. //
  1067. if (FAILED(hRetval) && (HRESULT_CODE(ERROR_LINE_NOT_FOUND) == HRESULT_CODE(hRetval)))
  1068. {
  1069. DBG_MSG(DBG_TRACE, (_T("Driver \"%s\" is not mapped\n"), pszModelName));
  1070. hRetval = S_FALSE;
  1071. }
  1072. if (S_OK == hRetval)
  1073. {
  1074. DBG_MSG(DBG_TRACE, (_T("Driver \"%s\" is mapped\n"), pszModelName));
  1075. *puWarnLevelSrv = uWarnLevelSrv;
  1076. *puWarnLevelWks = uWarnLevelWks;
  1077. hRetval = pstrReplacementDriver->bUpdate(strReplacementDriver) ? S_OK : E_OUTOFMEMORY;
  1078. }
  1079. return hRetval;
  1080. }
  1081. /*++
  1082. Routine Name
  1083. IsDateInLineNoOlderThanDriverDate
  1084. Routine Description:
  1085. This routines process the current line of inf and determinate whether the
  1086. date in the line is not older than that of driver.
  1087. Arguments:
  1088. pInfContext - Points to the current context of an INF
  1089. pDriverFileTime - File time of the actual driver
  1090. pdwWarnLevelSrv - Points to the warning level for server SKU
  1091. pdwWarnLevelWks - Points to the warning level for wks SKU
  1092. pstrReplacementDriver - The replacement driver.
  1093. Return Value:
  1094. An HRESULT - S_OK means the date in the current line is no older
  1095. than that of the driver
  1096. --*/
  1097. HRESULT
  1098. IsDateInLineNoOlderThanDriverDate(
  1099. IN INFCONTEXT *pInfContext,
  1100. IN FILETIME *pDriverFileTime,
  1101. OUT UINT *puWarnLevelSrv,
  1102. OUT UINT *puWarnLevelWks,
  1103. OUT TString *pstrReplacementDriver
  1104. )
  1105. {
  1106. HRESULT hRetval = E_FAIL;
  1107. INT iWarnLevel = 0;
  1108. FILETIME FileTimeInInf;
  1109. hRetval = pInfContext && pDriverFileTime && puWarnLevelSrv && puWarnLevelWks && pstrReplacementDriver ? S_OK : E_INVALIDARG;
  1110. if (SUCCEEDED(hRetval))
  1111. {
  1112. hRetval = SetupGetIntField(pInfContext, kWarnLevelSrv, &iWarnLevel) ? S_OK: GetLastErrorAsHResult();
  1113. if (SUCCEEDED(hRetval))
  1114. {
  1115. *puWarnLevelSrv = iWarnLevel;
  1116. hRetval = SetupGetIntField(pInfContext, kWarnLevelWks, &iWarnLevel) ? S_OK: GetLastErrorAsHResult();
  1117. }
  1118. if (SUCCEEDED(hRetval))
  1119. {
  1120. *puWarnLevelWks = iWarnLevel;
  1121. hRetval = InfGetString(pInfContext, kReplacementDriver, pstrReplacementDriver);
  1122. }
  1123. if (SUCCEEDED(hRetval))
  1124. {
  1125. hRetval = InfGetStringAsFileTime(pInfContext, kFileTime, &FileTimeInInf);
  1126. //
  1127. // Date field is optional.
  1128. //
  1129. if (FAILED(hRetval) && (ERROR_INVALID_PARAMETER == HRESULT_CODE(hRetval)))
  1130. {
  1131. DBG_MSG(DBG_TRACE, (_T("Date in inf is empty, drivers of all dates are blocked.\n")));
  1132. hRetval = S_OK;
  1133. }
  1134. else if (SUCCEEDED(hRetval))
  1135. {
  1136. hRetval = CompareFileTime(pDriverFileTime, &FileTimeInInf) <= 0 ? S_OK : S_FALSE ;
  1137. }
  1138. }
  1139. }
  1140. return hRetval;
  1141. }
  1142. /*++
  1143. Routine Name
  1144. GetSectionName
  1145. Routine Description:
  1146. Get the Section name in terms of environment and driver version.
  1147. Arguments:
  1148. pszEnvironment - The environment of the server, such as
  1149. uVersion - The major version of the driver
  1150. pstrSection - Points the name of section of driver mapping
  1151. Return Value:
  1152. An HRESULT
  1153. --*/
  1154. HRESULT
  1155. GetSectionName(
  1156. IN LPCTSTR pszEnvironment,
  1157. IN UINT uVersion,
  1158. OUT TString *pstrSection
  1159. )
  1160. {
  1161. HRESULT hRetval = E_FAIL;
  1162. hRetval = pszEnvironment && pstrSection ? S_OK : E_INVALIDARG;
  1163. if (SUCCEEDED(hRetval))
  1164. {
  1165. hRetval = pstrSection->bFormat(_T("%s_%s_%s %d"), cszPrintDriverMapping, pszEnvironment, cszVersion, uVersion);
  1166. }
  1167. return hRetval;
  1168. }
  1169. /*++
  1170. Routine Name
  1171. InfGetString
  1172. Routine Description:
  1173. This routine is a wrapper to SetupGetStringField using TString.
  1174. Arguments:
  1175. pInfContext - The context of the inf
  1176. uFieldIndex - The field index of the string to retrieve
  1177. pstrField - Points to the string field as TString
  1178. Return Value:
  1179. An HRESULT
  1180. --*/
  1181. HRESULT
  1182. InfGetString(
  1183. IN INFCONTEXT *pInfContext,
  1184. IN UINT uFieldIndex,
  1185. OUT TString *pstrField
  1186. )
  1187. {
  1188. HRESULT hRetval = E_FAIL;
  1189. TCHAR szField[MAX_PATH] = {0};
  1190. DWORD dwNeeded = 0;
  1191. TCHAR *pszField = NULL;
  1192. hRetval = pInfContext && pstrField ? S_OK : E_INVALIDARG;
  1193. if (SUCCEEDED(hRetval))
  1194. {
  1195. hRetval = SetupGetStringField(pInfContext,
  1196. uFieldIndex,
  1197. szField,
  1198. COUNTOF(szField),
  1199. &dwNeeded) ? S_OK : GetLastErrorAsHResult();
  1200. if (SUCCEEDED(hRetval))
  1201. {
  1202. hRetval = pstrField->bUpdate(szField) ? S_OK : E_OUTOFMEMORY;
  1203. }
  1204. else if (FAILED(hRetval) && (ERROR_INSUFFICIENT_BUFFER == HRESULT_CODE(hRetval)))
  1205. {
  1206. pszField = new TCHAR[dwNeeded];
  1207. hRetval = pszField ? S_OK : E_OUTOFMEMORY;
  1208. DBG_MSG(DBG_TRACE, (_T("Long string encountered\n")));
  1209. if (SUCCEEDED(hRetval))
  1210. {
  1211. hRetval = SetupGetStringField(pInfContext,
  1212. uFieldIndex,
  1213. pszField,
  1214. dwNeeded,
  1215. &dwNeeded) ? S_OK : GetLastErrorAsHResult();
  1216. }
  1217. if (SUCCEEDED(hRetval))
  1218. {
  1219. hRetval = pstrField->bUpdate(pszField) ? S_OK : E_OUTOFMEMORY;
  1220. }
  1221. }
  1222. }
  1223. delete [] pszField;
  1224. return hRetval;
  1225. }
  1226. /*++
  1227. Routine Name
  1228. InfGetStringAsFileTime
  1229. Routine Description:
  1230. This routine get the time of driver in printupg and converts it to FILETIME.
  1231. Arguments:
  1232. pInfContext - The context of the inf
  1233. uFieldIndex - The field index of the string to retrieve
  1234. pFielTime - Points to the FILETIME structure
  1235. Return Value:
  1236. An HRESULT
  1237. --*/
  1238. HRESULT
  1239. InfGetStringAsFileTime(
  1240. IN INFCONTEXT *pInfContext,
  1241. IN UINT uFieldIndex,
  1242. OUT FILETIME *pFileTime
  1243. )
  1244. {
  1245. HRESULT hRetval = E_FAIL;
  1246. TString strDate;
  1247. hRetval = pInfContext && pFileTime ? S_OK : E_INVALIDARG;
  1248. if (SUCCEEDED(hRetval))
  1249. {
  1250. hRetval = InfGetString(pInfContext, uFieldIndex, &strDate);
  1251. }
  1252. if (SUCCEEDED(hRetval))
  1253. {
  1254. DBG_MSG(DBG_TRACE, (_T("FileTime in INF as string \"%s\"\n"), static_cast<LPCTSTR>(strDate)));
  1255. hRetval = StringTimeToFileTime(strDate, pFileTime);
  1256. }
  1257. return hRetval;
  1258. }
  1259. /*++
  1260. Routine Name
  1261. StringTimeToFileTime
  1262. Routine Description:
  1263. Converts a string of time in the form of "11/27/1999" to FILETIME.
  1264. Arguments:
  1265. pszFileTime - The file time as string such as "11/27/1999"
  1266. pFileTime - Points to the converted FILETIME
  1267. Return Value:
  1268. An HRESULT
  1269. --*/
  1270. HRESULT
  1271. StringTimeToFileTime(
  1272. IN LPCTSTR pszFileTime,
  1273. OUT FILETIME *pFileTime
  1274. )
  1275. {
  1276. HRESULT hRetval = E_FAIL;
  1277. SYSTEMTIME SystemTime;
  1278. hRetval = pszFileTime && pFileTime ? S_OK : E_INVALIDARG;
  1279. if (SUCCEEDED(hRetval))
  1280. {
  1281. //
  1282. // StringToDate should take pszFileTime as const.
  1283. //
  1284. hRetval = StringToDate(const_cast<LPTSTR>(pszFileTime), &SystemTime) ? S_OK : GetLastErrorAsHResult();
  1285. }
  1286. if (SUCCEEDED(hRetval))
  1287. {
  1288. hRetval = SystemTimeToFileTime(&SystemTime, pFileTime) ? S_OK : GetLastErrorAsHResult();
  1289. }
  1290. return hRetval;
  1291. }
  1292. /*++
  1293. Routine Name
  1294. GetBlockingStatusByWksType
  1295. Routine Description:
  1296. Fill out the status of blocking according to the type of SKU that runs the
  1297. service.
  1298. Arguments:
  1299. uWarnLevelSrv - The warn level for server SKU
  1300. uWarnLevelSrv - The warn level for wks SKU
  1301. bIsServer - Whether the SKU running printing service is server
  1302. puBlockingStatus - Points to the result as status of blocking
  1303. Return Value:
  1304. An HRESULT
  1305. --*/
  1306. HRESULT
  1307. GetBlockingStatusByWksType(
  1308. IN UINT uWarnLevelSrv,
  1309. IN UINT uWarnLevelWks,
  1310. IN BOOL bIsServer,
  1311. OUT UINT *puBlockingStatus
  1312. )
  1313. {
  1314. HRESULT hRetval = E_FAIL;
  1315. UINT uWarnLevel = 0;
  1316. hRetval = puBlockingStatus ? S_OK : E_INVALIDARG;
  1317. if (SUCCEEDED(hRetval))
  1318. {
  1319. *puBlockingStatus &= ~BSP_BLOCKING_LEVEL_MASK;
  1320. *puBlockingStatus |= BSP_PRINTER_DRIVER_OK;
  1321. uWarnLevel = bIsServer ? uWarnLevelSrv : uWarnLevelWks;
  1322. switch (uWarnLevel)
  1323. {
  1324. case kBlocked:
  1325. *puBlockingStatus |= BSP_PRINTER_DRIVER_BLOCKED;
  1326. break;
  1327. case kWarned:
  1328. *puBlockingStatus |= BSP_PRINTER_DRIVER_WARNED;
  1329. break;
  1330. default:
  1331. hRetval = E_FAIL;
  1332. break;
  1333. }
  1334. }
  1335. return hRetval;
  1336. }
  1337. /*++
  1338. Routine Name
  1339. IsDriverDllInExcludedSection
  1340. Routine Description:
  1341. Determine Whether the driver dll name is in the excluded section of printupg.
  1342. Arguments:
  1343. pszDriverPath - The path of the driver and this can be a full path or
  1344. the file name
  1345. hPrintUpgInf - The handle to printupg INF file
  1346. Return Value:
  1347. An HRESULT - S_OK means the driver dll is in the excluded section,
  1348. S_FALSE means it is not.
  1349. --*/
  1350. HRESULT
  1351. IsDriverDllInExcludedSection(
  1352. IN LPCTSTR pszDriverPath,
  1353. IN HINF hPrintUpgInf
  1354. )
  1355. {
  1356. DBG_MSG(DBG_TRACE, (_T("In the IsDriverDllInExcludedSection routine\n")));
  1357. HRESULT hRetval = E_FAIL;
  1358. TString strDriverFileName;
  1359. INFCONTEXT InfContext;
  1360. hRetval = pszDriverPath && (INVALID_HANDLE_VALUE != hPrintUpgInf) ? S_OK : E_INVALIDARG;
  1361. if (SUCCEEDED(hRetval))
  1362. {
  1363. hRetval = strDriverFileName.bUpdate(FileNamePart(pszDriverPath)) ? S_OK : E_OUTOFMEMORY;
  1364. }
  1365. if (SUCCEEDED(hRetval) && !strDriverFileName.bEmpty())
  1366. {
  1367. hRetval = SetupFindFirstLine(hPrintUpgInf,
  1368. cszExcludeSection,
  1369. strDriverFileName,
  1370. &InfContext) ? S_OK : GetLastErrorAsHResult();
  1371. //
  1372. // ERROR_LINE_NOT_FOUND is an HRESULT!
  1373. //
  1374. if (FAILED(hRetval) && (HRESULT_CODE(ERROR_LINE_NOT_FOUND) == HRESULT_CODE(hRetval)))
  1375. {
  1376. hRetval = S_FALSE;
  1377. }
  1378. DBG_MSG(DBG_TRACE, (_T("Driver Path %s driver dll %s excluded section \"%s\" Is not excluded? %d\n"), pszDriverPath, static_cast<LPCTSTR>(strDriverFileName), cszExcludeSection, hRetval));
  1379. }
  1380. return hRetval;
  1381. }
  1382. /*++
  1383. Routine Name
  1384. IsLocalMachineServer
  1385. Routine Description:
  1386. This routine determines whether the local machine is a server SKU.
  1387. Arguments:
  1388. None
  1389. Return Value:
  1390. An HRESULT - S_OK if it is a server, S_FALSE otherwise.
  1391. --*/
  1392. HRESULT
  1393. IsLocalMachineServer(
  1394. VOID
  1395. )
  1396. {
  1397. HRESULT hRetval = E_FAIL;
  1398. DWORDLONG dwlConditionMask = 0;
  1399. OSVERSIONINFOEX OsVerEx;
  1400. (VOID)ZeroMemory(&OsVerEx, sizeof(OSVERSIONINFOEX));
  1401. OsVerEx.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
  1402. OsVerEx.wProductType = VER_NT_SERVER;
  1403. VER_SET_CONDITION( dwlConditionMask, VER_PRODUCT_TYPE, VER_EQUAL );
  1404. hRetval = VerifyVersionInfo(&OsVerEx, VER_PRODUCT_TYPE, dwlConditionMask) ? S_OK : GetLastErrorAsHResult();
  1405. if (FAILED(hRetval) && (ERROR_OLD_WIN_VERSION == HRESULT_CODE(hRetval)))
  1406. {
  1407. hRetval = S_FALSE;
  1408. }
  1409. return hRetval;
  1410. }
  1411. #if DBG_PRINTUPG
  1412. LPTSTR
  1413. ReadDigit(
  1414. LPTSTR ptr,
  1415. LPWORD pW
  1416. )
  1417. {
  1418. TCHAR c;
  1419. //
  1420. // Skip spaces
  1421. //
  1422. while ( !iswdigit(c = *ptr) && c != TEXT('\0') )
  1423. ++ptr;
  1424. if ( c == TEXT('\0') )
  1425. return NULL;
  1426. //
  1427. // Read field
  1428. //
  1429. for ( *pW = 0 ; iswdigit(c = *ptr) ; ++ptr )
  1430. *pW = *pW * 10 + c - TEXT('0');
  1431. return ptr;
  1432. }
  1433. BOOL
  1434. StringToDate(
  1435. LPTSTR pszDate,
  1436. SYSTEMTIME *pInfTime
  1437. )
  1438. {
  1439. BOOL bRet = FALSE;
  1440. ZeroMemory(pInfTime, sizeof(*pInfTime));
  1441. bRet = (pszDate = ReadDigit(pszDate, &(pInfTime->wMonth))) &&
  1442. (pszDate = ReadDigit(pszDate, &(pInfTime->wDay))) &&
  1443. (pszDate = ReadDigit(pszDate, &(pInfTime->wYear)));
  1444. //
  1445. // Y2K compatible check
  1446. //
  1447. if ( bRet && pInfTime->wYear < 100 ) {
  1448. if ( pInfTime->wYear < 10 )
  1449. pInfTime->wYear += 2000;
  1450. else
  1451. pInfTime->wYear += 1900;
  1452. }
  1453. return bRet;
  1454. }
  1455. LPTSTR
  1456. FileNamePart(
  1457. IN LPCTSTR pszFullName
  1458. )
  1459. {
  1460. LPTSTR pszSlash, pszTemp;
  1461. if ( !pszFullName )
  1462. return NULL;
  1463. //
  1464. // First find the : for the drive
  1465. //
  1466. if ( pszTemp = lstrchr(pszFullName, TEXT(':')) )
  1467. pszFullName = pszFullName + 1;
  1468. for ( pszTemp = (LPTSTR)pszFullName ;
  1469. pszSlash = lstrchr(pszTemp, TEXT('\\')) ;
  1470. pszTemp = pszSlash + 1 )
  1471. ;
  1472. return *pszTemp ? pszTemp : NULL;
  1473. }
  1474. PVOID
  1475. LocalAllocMem(
  1476. IN UINT cbSize
  1477. )
  1478. {
  1479. return LocalAlloc( LPTR, cbSize );
  1480. }
  1481. VOID
  1482. LocalFreeMem(
  1483. IN PVOID p
  1484. )
  1485. {
  1486. LocalFree(p);
  1487. }
  1488. //
  1489. // For some reason these are needed by spllib when you use StrNCatBuf.
  1490. // This doesn't make any sense, but just implement them.
  1491. //
  1492. extern "C"
  1493. LPVOID
  1494. DllAllocSplMem(
  1495. DWORD cbSize
  1496. )
  1497. {
  1498. return LocalAllocMem(cbSize);
  1499. }
  1500. HRESULT
  1501. TestPrintUpgOne(
  1502. IN LPCTSTR pszDriverModel,
  1503. IN LPCTSTR pszDriverPath,
  1504. IN LPCTSTR pszEnvironment,
  1505. IN LPCTSTR pszDriverTime,
  1506. IN LPCTSTR pszPrintUpgInf,
  1507. IN UINT uVersion,
  1508. IN BOOL bIsServer,
  1509. IN UINT uBlockingStatusInput,
  1510. IN LPCTSTR pszReplacementDriver,
  1511. IN BOOL bSuccess
  1512. )
  1513. {
  1514. HRESULT hRetval = E_FAIL;
  1515. UINT uBlockingStatus = 0;
  1516. TString strReplacementDriver;
  1517. FILETIME FileTimeDriver;
  1518. hRetval = pszDriverModel && pszDriverPath && pszEnvironment && pszDriverTime && pszPrintUpgInf && pszReplacementDriver ? S_OK : E_INVALIDARG;
  1519. if (SUCCEEDED(hRetval))
  1520. {
  1521. hRetval = StringTimeToFileTime(pszDriverTime, &FileTimeDriver);
  1522. }
  1523. //if (SUCCEEDED(hRetval))
  1524. //{
  1525. // hRetval = IsEnvironmentAndVersionNeededToCheck(pszEnvironment, uVersion);
  1526. //}
  1527. //if (S_OK == hRetval)
  1528. if (SUCCEEDED(hRetval))
  1529. {
  1530. DBG_MSG(DBG_TRACE, (_T("TEST case: Driver of time %s Expected status %d Expected Replacement \"%s\"\n"), pszDriverTime, uBlockingStatusInput, pszReplacementDriver));
  1531. hRetval = InternalCompatibleDriverCheck(pszDriverModel,
  1532. pszDriverPath,
  1533. pszEnvironment,
  1534. &FileTimeDriver,
  1535. pszPrintUpgInf,
  1536. uVersion,
  1537. bIsServer,
  1538. &uBlockingStatus,
  1539. &strReplacementDriver);
  1540. }
  1541. if (SUCCEEDED(hRetval))
  1542. {
  1543. hRetval = ((uBlockingStatusInput & BSP_BLOCKING_LEVEL_MASK) == (uBlockingStatus & BSP_BLOCKING_LEVEL_MASK)) ? S_OK : S_FALSE;
  1544. }
  1545. if ((S_OK == hRetval) && (BSP_PRINTER_DRIVER_OK != (uBlockingStatus & BSP_BLOCKING_LEVEL_MASK)))
  1546. {
  1547. hRetval = !lstrcmp(pszReplacementDriver, strReplacementDriver) ? S_OK : S_FALSE;
  1548. //
  1549. // we can not test whether BSP_INBOX_DRIVER_AVAILABLE is set
  1550. //
  1551. //if((S_OK == hRetval) && pszReplacemtDriver && *ppszReplacementDriver)
  1552. //{
  1553. // hRetval = (BSP_INBOX_DRIVER_AVAILABLE & uBlockingStatus) ? S_OK : S_FALSE;
  1554. //}
  1555. }
  1556. if (SUCCEEDED(hRetval))
  1557. {
  1558. hRetval = (((S_FALSE == hRetval) && !bSuccess) || (S_OK == hRetval) && bSuccess) ? S_OK : E_FAIL;
  1559. }
  1560. return hRetval;
  1561. }
  1562. TCHAR szModel1[] = _T("HP LaserJet 4");
  1563. TCHAR szModel2[] = _T("HP LaserJet 4P");
  1564. TCHAR szModel3[] = _T("");
  1565. TCHAR szModel4[] = _T("HP LaserJet 4M Plus");
  1566. TCHAR szModel5[] = _T("HP LaserJet 4Si");
  1567. TCHAR szModel6[] = _T("HP LaserJet 4PPP");
  1568. TCHAR szModel7[] = _T("Lexmark Optra T612");
  1569. TCHAR szModel8[] = _T("HP LaserJet 4V");
  1570. TCHAR szModel9[] = _T("Apple LaserJet 400PS");
  1571. TCHAR szModel0[] = _T("HP Laserjet 4Si mx");
  1572. TCHAR szDriverPath1[] = _T("H:\\WINDOWS\\system32\\spool\\drivers\\w32x86\\3\\ps5ui.dll");
  1573. TCHAR szDriverPath2[] = _T("pscript5.dll");
  1574. TCHAR szDriverPath3[] = _T("H:\\WINDOWS\\system32\\spool\\drivers\\w32x86\\3\\ps5ui.dll");
  1575. TCHAR szDriverPath4[] = _T("psCrIpt5.dll");
  1576. TCHAR szDriverPath5[] = _T("ps5ui.dll");
  1577. TCHAR szEnvironment1[] = _T("Windows NT x86");
  1578. TCHAR szEnvironment2[] = _T("Windows 4.0");
  1579. TCHAR szEnvironment3[] = _T("Windows IA64");
  1580. TCHAR szEnvironment4[] = _T("Windows XP");
  1581. TCHAR szDriverTime1[] = _T("11/27/1999");
  1582. TCHAR szDriverTime2[] = _T("11/27/2999");
  1583. TCHAR szDriverTime3[] = _T("11/27/2001");
  1584. TCHAR szDriverTime4[] = _T("10/27/1999");
  1585. TCHAR szDriverTime5[] = _T("11/28/1998");
  1586. TCHAR szDriverTime6[] = _T("11/28/1999");
  1587. TCHAR szDriverTime7[] = _T("10/28/1999");
  1588. DWORD dwStatus1 = BSP_PRINTER_DRIVER_OK;
  1589. DWORD dwStatus2 = BSP_PRINTER_DRIVER_WARNED;
  1590. DWORD dwStatus3 = BSP_PRINTER_DRIVER_BLOCKED;
  1591. TCHAR cszUpgradeInf[] = _T("printupg.inf");
  1592. TCHAR cszPrintDriverMapping[] = _T("Printer Driver Mapping");
  1593. TCHAR cszVersion[] = _T("Version");
  1594. TCHAR cszExcludeSection[] = _T("Excluded Driver Files");
  1595. TCHAR cszPrintUpgInf[] = _T("\\\\lzhu0\\zdrive\\sdroot\\printscan\\print\\spooler\\test\\printupg\\printupg.inf");
  1596. HRESULT
  1597. TestPrintUpgAll(
  1598. VOID
  1599. )
  1600. {
  1601. HRESULT hRetval = S_OK;
  1602. const struct PrintUpgTest aPrintUpgTests [] =
  1603. {
  1604. //
  1605. // model driverPath environment drivertime ver isServer status replacement isSuccess
  1606. //
  1607. {szModel1, szDriverPath1, szEnvironment1, szDriverTime4, 2, TRUE, dwStatus2, szModel1, TRUE},
  1608. {szModel1, szDriverPath1, szEnvironment1, szDriverTime1, 2, FALSE, dwStatus3, szModel1, TRUE},
  1609. {szModel1, szDriverPath1, szEnvironment1, szDriverTime2, 2, FALSE, dwStatus1, szModel1, TRUE},
  1610. {szModel1, szDriverPath1, szEnvironment1, szDriverTime2, 2, TRUE, dwStatus1, szModel1, TRUE},
  1611. {szModel2, szDriverPath1, szEnvironment1, szDriverTime1, 2, TRUE, dwStatus3, szModel6, TRUE},
  1612. {szModel2, szDriverPath1, szEnvironment1, szDriverTime1, 2, FALSE, dwStatus2, szModel6, TRUE},
  1613. {szModel2, szDriverPath1, szEnvironment1, szDriverTime2, 2, FALSE, dwStatus1, szModel6, TRUE},
  1614. {szModel2, szDriverPath1, szEnvironment1, szDriverTime2, 2, FALSE, dwStatus1, szModel1, TRUE},
  1615. {szModel1, szDriverPath1, szEnvironment1, szDriverTime1, 2, FALSE, dwStatus2, szModel1, FALSE},
  1616. {szModel1, szDriverPath1, szEnvironment1, szDriverTime1, 2, TRUE, dwStatus3, szModel1, FALSE},
  1617. {szModel1, szDriverPath1, szEnvironment1, szDriverTime2, 2, FALSE, dwStatus2, szModel1, FALSE},
  1618. {szModel1, szDriverPath1, szEnvironment1, szDriverTime2, 2, TRUE, dwStatus3, szModel1, FALSE},
  1619. {szModel2, szDriverPath1, szEnvironment1, szDriverTime1, 2, TRUE, dwStatus3, szModel1, FALSE},
  1620. {szModel2, szDriverPath1, szEnvironment1, szDriverTime2, 2, FALSE, dwStatus2, szModel6, FALSE},
  1621. {szModel2, szDriverPath1, szEnvironment1, szDriverTime1, 2, FALSE, dwStatus1, szModel6, FALSE},
  1622. {szModel2, szDriverPath1, szEnvironment1, szDriverTime1, 2, FALSE, dwStatus2, szModel1, FALSE},
  1623. //
  1624. // case 16 is next
  1625. //
  1626. {szModel7, szDriverPath1, szEnvironment1, szDriverTime1, 2, TRUE, dwStatus3, szModel3, TRUE},
  1627. {szModel7, szDriverPath1, szEnvironment1, szDriverTime1, 2, FALSE, dwStatus2, szModel3, TRUE},
  1628. {szModel7, szDriverPath1, szEnvironment1, szDriverTime2, 2, FALSE, dwStatus1, szModel3, TRUE},
  1629. {szModel7, szDriverPath1, szEnvironment1, szDriverTime2, 2, TRUE, dwStatus1, szModel3, TRUE},
  1630. {szModel8, szDriverPath1, szEnvironment1, szDriverTime1, 2, TRUE, dwStatus2, szModel3, TRUE},
  1631. {szModel8, szDriverPath1, szEnvironment1, szDriverTime1, 2, FALSE, dwStatus3, szModel3, TRUE},
  1632. {szModel8, szDriverPath1, szEnvironment1, szDriverTime2, 2, FALSE, dwStatus1, szModel3, TRUE},
  1633. {szModel8, szDriverPath1, szEnvironment1, szDriverTime2, 2, FALSE, dwStatus1, szModel3, TRUE},
  1634. {szModel7, szDriverPath1, szEnvironment1, szDriverTime1, 2, FALSE, dwStatus3, szModel7, FALSE},
  1635. {szModel7, szDriverPath1, szEnvironment1, szDriverTime1, 2, TRUE, dwStatus2, szModel7, FALSE},
  1636. {szModel7, szDriverPath1, szEnvironment1, szDriverTime2, 2, FALSE, dwStatus2, szModel7, FALSE},
  1637. {szModel7, szDriverPath1, szEnvironment1, szDriverTime2, 2, TRUE, dwStatus3, szModel7, FALSE},
  1638. {szModel8, szDriverPath1, szEnvironment1, szDriverTime1, 2, TRUE, dwStatus2, szModel7, FALSE},
  1639. {szModel8, szDriverPath1, szEnvironment1, szDriverTime2, 2, FALSE, dwStatus3, szModel8, FALSE},
  1640. {szModel8, szDriverPath1, szEnvironment1, szDriverTime1, 2, FALSE, dwStatus1, szModel8, FALSE},
  1641. {szModel8, szDriverPath1, szEnvironment1, szDriverTime1, 2, FALSE, dwStatus2, szModel7, FALSE},
  1642. //
  1643. // case 32 is next
  1644. //
  1645. {szModel4, szDriverPath2, szEnvironment1, szDriverTime1, 2, TRUE, dwStatus3, szModel4, FALSE},
  1646. {szModel4, szDriverPath5, szEnvironment1, szDriverTime1, 2, TRUE, dwStatus3, szModel4, FALSE},
  1647. {szModel4, szDriverPath5, szEnvironment1, szDriverTime1, 2, TRUE, dwStatus2, szModel4, FALSE},
  1648. {szModel4, szDriverPath4, szEnvironment1, szDriverTime1, 2, TRUE, dwStatus1, szModel7, TRUE},
  1649. {szModel4, szDriverPath1, szEnvironment1, szDriverTime1, 3, TRUE, dwStatus3, szModel4, TRUE},
  1650. {szModel4, szDriverPath1, szEnvironment1, szDriverTime1, 3, FALSE, dwStatus2, szModel4, TRUE},
  1651. {szModel4, szDriverPath1, szEnvironment1, szDriverTime2, 3, FALSE, dwStatus1, szModel4, TRUE},
  1652. {szModel4, szDriverPath1, szEnvironment1, szDriverTime2, 3, TRUE, dwStatus1, szModel4, TRUE},
  1653. {szModel4, szDriverPath1, szEnvironment1, szDriverTime1, 3, FALSE, dwStatus3, szModel4, FALSE},
  1654. {szModel4, szDriverPath1, szEnvironment1, szDriverTime1, 3, TRUE, dwStatus2, szModel4, FALSE},
  1655. {szModel4, szDriverPath1, szEnvironment1, szDriverTime2, 3, FALSE, dwStatus2, szModel4, FALSE},
  1656. {szModel4, szDriverPath1, szEnvironment1, szDriverTime2, 3, TRUE, dwStatus3, szModel4, FALSE},
  1657. {szModel0, szDriverPath1, szEnvironment1, szDriverTime1, 3, TRUE, dwStatus3, szModel3, TRUE},
  1658. {szModel0, szDriverPath1, szEnvironment1, szDriverTime1, 3, FALSE, dwStatus2, szModel3, TRUE},
  1659. {szModel0, szDriverPath1, szEnvironment1, szDriverTime2, 3, FALSE, dwStatus1, szModel3, TRUE},
  1660. {szModel0, szDriverPath1, szEnvironment1, szDriverTime2, 3, TRUE, dwStatus1, szModel3, TRUE},
  1661. //
  1662. // case 48 is next
  1663. //
  1664. {szModel4, szDriverPath1, szEnvironment3, szDriverTime1, 3, TRUE, dwStatus2, szModel8, FALSE},
  1665. {szModel4, szDriverPath1, szEnvironment3, szDriverTime1, 3, FALSE, dwStatus2, szModel2, TRUE},
  1666. {szModel4, szDriverPath1, szEnvironment3, szDriverTime2, 3, FALSE, dwStatus1, szModel8, TRUE},
  1667. {szModel4, szDriverPath1, szEnvironment3, szDriverTime2, 3, TRUE, dwStatus1, szModel8, TRUE},
  1668. {szModel4, szDriverPath1, szEnvironment3, szDriverTime1, 3, FALSE, dwStatus2, szModel8, FALSE},
  1669. {szModel4, szDriverPath1, szEnvironment3, szDriverTime1, 3, TRUE, dwStatus3, szModel8, FALSE},
  1670. {szModel4, szDriverPath1, szEnvironment3, szDriverTime2, 3, FALSE, dwStatus2, szModel8, FALSE},
  1671. {szModel4, szDriverPath1, szEnvironment3, szDriverTime2, 3, TRUE, dwStatus3, szModel8, FALSE},
  1672. {szModel5, szDriverPath1, szEnvironment3, szDriverTime1, 3, TRUE, dwStatus3, szModel3, TRUE},
  1673. {szModel5, szDriverPath1, szEnvironment3, szDriverTime1, 3, FALSE, dwStatus2, szModel3, TRUE},
  1674. {szModel5, szDriverPath1, szEnvironment3, szDriverTime2, 3, FALSE, dwStatus1, szModel3, TRUE},
  1675. {szModel5, szDriverPath1, szEnvironment3, szDriverTime2, 3, TRUE, dwStatus1, szModel3, TRUE},
  1676. {szModel5, szDriverPath1, szEnvironment2, szDriverTime1, 3, TRUE, dwStatus1, szModel3, TRUE},
  1677. {szModel5, szDriverPath1, szEnvironment3, szDriverTime1, 1, FALSE, dwStatus2, szModel3, FALSE},
  1678. {szModel9, szDriverPath1, szEnvironment1, szDriverTime2, 3, FALSE, dwStatus1, szModel3, TRUE},
  1679. {szModel9, szDriverPath1, szEnvironment1, szDriverTime2, 3, TRUE, dwStatus2, szModel3, FALSE},
  1680. //
  1681. // case 64 is next
  1682. //
  1683. {szModel4, szDriverPath1, szEnvironment3, szDriverTime1, 2, TRUE, dwStatus3, szModel8, TRUE},
  1684. {szModel4, szDriverPath1, szEnvironment3, szDriverTime1, 3, FALSE, dwStatus2, szModel2, TRUE},
  1685. {szModel1, szDriverPath1, szEnvironment1, szDriverTime5, 2, TRUE, dwStatus3, szModel1, TRUE},
  1686. {szModel1, szDriverPath1, szEnvironment1, szDriverTime6, 2, TRUE, dwStatus2, szModel1, TRUE},
  1687. {szModel1, szDriverPath1, szEnvironment1, szDriverTime7, 2, TRUE, dwStatus2, szModel1, TRUE},
  1688. {szModel2, szDriverPath1, szEnvironment3, szDriverTime2, 3, TRUE, dwStatus2, szModel3, TRUE},
  1689. {szModel2, szDriverPath1, szEnvironment3, szDriverTime2, 3, FALSE, dwStatus3, szModel3, TRUE},
  1690. {szModel2, szDriverPath1, szEnvironment4, szDriverTime2, 3, FALSE, dwStatus1, szModel3, TRUE},
  1691. {szModel2, szDriverPath1, szEnvironment3, szDriverTime2, 4, FALSE, dwStatus1, szModel3, TRUE},
  1692. };
  1693. DBG_MSG(DBG_TRACE, (L"tests started\n"));
  1694. for (int i = 0; SUCCEEDED(hRetval) && i < COUNTOF(aPrintUpgTests); i++)
  1695. {
  1696. DBG_MSG(DBG_TRACE, (_T("************************ %d **********************\n"), i));
  1697. hRetval = TestPrintUpgOne(aPrintUpgTests[i].pszDriverModel,
  1698. aPrintUpgTests[i].pszDriverPath,
  1699. aPrintUpgTests[i].pszEnvironment,
  1700. aPrintUpgTests[i].pszDriverTime,
  1701. cszPrintUpgInf,
  1702. aPrintUpgTests[i].uVersion,
  1703. aPrintUpgTests[i].bIsServer,
  1704. aPrintUpgTests[i].uBlockingStatus,
  1705. aPrintUpgTests[i].pszReplacementDriver,
  1706. aPrintUpgTests[i].bSuccess);
  1707. }
  1708. DBG_MSG(DBG_TRACE, (L"tests ended\n"));
  1709. return hRetval;
  1710. }
  1711. #endif // DBG_PRINTUPG