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.

601 lines
14 KiB

  1. /*++
  2. Copyright (c) 1997 Microsoft Corporation
  3. All rights reserved
  4. Module Name:
  5. devmode.c
  6. Abstract:
  7. Handles per-user devmode implementation.
  8. Author:
  9. Environment:
  10. User Mode -Win32
  11. Revision History:
  12. --*/
  13. #include "precomp.h"
  14. #pragma hdrstop
  15. #include "local.h"
  16. #include <offsets.h>
  17. /********************************************************************
  18. Forward prototypes
  19. ********************************************************************/
  20. BOOL
  21. bGetDevModeLocation(
  22. IN HKEY hKeyUser, OPTIONAL
  23. IN LPCWSTR pszPrinter,
  24. OUT PHKEY phKey,
  25. OUT LPCWSTR *ppszValue
  26. );
  27. const WCHAR gszPrinterConnections[] = L"Printers\\Connections\\";
  28. const WCHAR gszDevMode[] = L"DevMode";
  29. const WCHAR gszDevModePerUserLocal[] = L"Printers\\DevModePerUser";
  30. /********************************************************************
  31. Private Functions
  32. ********************************************************************/
  33. DWORD
  34. RegOpenConnectionKey(
  35. HKEY hKeyUser,
  36. LPWSTR pszPrinter,
  37. PHKEY phKey
  38. )
  39. {
  40. PWCHAR pszPrinterScratch = NULL;
  41. DWORD dwRetValue = ERROR_SUCCESS;
  42. DWORD cchSize = MAX_UNC_PRINTER_NAME + PRINTER_NAME_SUFFIX_MAX + COUNTOF( gszPrinterConnections );
  43. if (pszPrinter &&
  44. wcslen(pszPrinter) < MAX_UNC_PRINTER_NAME + PRINTER_NAME_SUFFIX_MAX) {
  45. if (pszPrinterScratch = AllocSplMem(cchSize * sizeof(WCHAR))) {
  46. wcscpy(pszPrinterScratch, gszPrinterConnections);
  47. FormatPrinterForRegistryKey(pszPrinter,
  48. &pszPrinterScratch[ COUNTOF( gszPrinterConnections )-1] );
  49. dwRetValue = RegOpenKeyEx(hKeyUser,
  50. pszPrinterScratch,
  51. 0,
  52. KEY_READ | KEY_WRITE,
  53. phKey );
  54. FreeSplMem(pszPrinterScratch);
  55. } else {
  56. dwRetValue = GetLastError();
  57. }
  58. } else {
  59. dwRetValue = ERROR_INVALID_PARAMETER;
  60. }
  61. return dwRetValue;
  62. }
  63. /********************************************************************
  64. Public functions
  65. ********************************************************************/
  66. BOOL
  67. bSetDevModePerUser(
  68. HANDLE hKeyUser,
  69. LPCWSTR pszPrinter,
  70. PDEVMODE pDevMode
  71. )
  72. /*++
  73. Routine Description:
  74. Sets the per-user DevMode in HKCU.
  75. Arguments:
  76. hKeyUser - HKEY_CURRENT_USER handle. OPTIONAL
  77. pszPrinter - Printer to set.
  78. pDevMode - DevMode to save. If NULL, deletes value.
  79. Return Value:
  80. TRUE - Success
  81. FALSE - Failure
  82. --*/
  83. {
  84. HKEY hKey = NULL;
  85. LPWSTR pszValue = NULL;
  86. DWORD Status;
  87. if( !pszPrinter ){
  88. SetLastError( ERROR_INVALID_HANDLE );
  89. return FALSE;
  90. }
  91. //
  92. // Retrieve the location of the DevMode.
  93. //
  94. if( !bGetDevModeLocation( hKeyUser,
  95. pszPrinter,
  96. &hKey,
  97. &pszValue )){
  98. return FALSE;
  99. }
  100. if( !pDevMode ){
  101. //
  102. // NULL, so delete the value.
  103. //
  104. Status = RegDeleteValue( hKey, pszValue );
  105. //
  106. // If value not found, don't fail.
  107. //
  108. if( Status == ERROR_FILE_NOT_FOUND ){
  109. Status = ERROR_SUCCESS;
  110. }
  111. } else {
  112. Status = RegSetValueEx( hKey,
  113. pszValue,
  114. 0,
  115. REG_BINARY,
  116. (PBYTE)pDevMode,
  117. pDevMode->dmSize +
  118. pDevMode->dmDriverExtra );
  119. if( Status == ERROR_SUCCESS ){
  120. //
  121. // Notify everyone that the DevMode has changed.
  122. //
  123. SendNotifyMessage( HWND_BROADCAST,
  124. WM_DEVMODECHANGE,
  125. 0,
  126. (LPARAM)pszPrinter );
  127. }
  128. }
  129. RegCloseKey( hKey );
  130. if( Status != ERROR_SUCCESS ){
  131. SetLastError( Status );
  132. return FALSE;
  133. }
  134. return TRUE;
  135. }
  136. BOOL
  137. bGetDevModePerUser(
  138. HKEY hKeyUser,
  139. LPCWSTR pszPrinter,
  140. PDEVMODE *ppDevMode
  141. )
  142. /*++
  143. Routine Description:
  144. Retrieves the per-user DevMode based on the current user.
  145. Arguments:
  146. hKeyUser - HKEY_CURRENT_USER handle. OPTIONAL
  147. pszPrinter - Printer to get.
  148. ppDevMode - Receives pointer to devmode. Must be freed by callee.
  149. Return Value:
  150. TRUE - Success: able to check if per-user DevMode exists. *ppDevMode
  151. is NULL if no per-user DevMode is there. (TRUE does not indicate
  152. that a per-user DevMode was found, only that we successfully checked.)
  153. FALSE - Failure.
  154. --*/
  155. {
  156. HKEY hKey = NULL;
  157. LPWSTR pszValue = NULL;
  158. LONG Status;
  159. *ppDevMode = NULL;
  160. if( !pszPrinter ){
  161. SetLastError( ERROR_INVALID_HANDLE );
  162. return FALSE;
  163. }
  164. //
  165. // Retrieve the location of the DevMode.
  166. //
  167. if( !bGetDevModeLocation( hKeyUser,
  168. pszPrinter,
  169. &hKey,
  170. &pszValue )){
  171. Status = GetLastError();
  172. } else {
  173. DWORD cbDevModePerUser;
  174. //
  175. // Key exists. See if we can read it and get the per-user DevMode.
  176. //
  177. Status = RegQueryInfoKey( hKey,
  178. NULL,
  179. NULL,
  180. NULL,
  181. NULL,
  182. NULL,
  183. NULL,
  184. NULL,
  185. NULL,
  186. &cbDevModePerUser,
  187. NULL,
  188. NULL );
  189. if( Status == ERROR_SUCCESS ){
  190. if( cbDevModePerUser >= MIN_DEVMODE_SIZEW ){
  191. *ppDevMode = AllocSplMem( cbDevModePerUser );
  192. if( !*ppDevMode ){
  193. Status = GetLastError();
  194. } else {
  195. Status = RegQueryValueEx( hKey,
  196. pszValue,
  197. NULL,
  198. NULL,
  199. (PBYTE)*ppDevMode,
  200. &cbDevModePerUser );
  201. if (ERROR_SUCCESS == Status) {
  202. Status = IsValidDevmode(*ppDevMode, cbDevModePerUser);
  203. //
  204. // If the devmode isn't valid, delete it and treat it
  205. // as if the devmode cannot be found.
  206. //
  207. if (ERROR_SUCCESS != Status) {
  208. //
  209. // If we can delete the key, just consider it as if
  210. // it does not exist.
  211. //
  212. if (ERROR_SUCCESS == RegDeleteValue(hKey, pszValue)) {
  213. Status = ERROR_FILE_NOT_FOUND;
  214. }
  215. }
  216. }
  217. if( Status != ERROR_SUCCESS ){
  218. FreeSplMem( *ppDevMode );
  219. *ppDevMode = NULL;
  220. }
  221. //
  222. // Allow ERROR_FILE_NOT_FOUND to return success. *ppDevMode
  223. // is still NULL, but we return TRUE to indicate that we
  224. // successfully checked the registry--we just didn't find one.
  225. //
  226. if( Status == ERROR_FILE_NOT_FOUND ){
  227. Status = ERROR_SUCCESS;
  228. }
  229. }
  230. }
  231. }
  232. RegCloseKey( hKey );
  233. }
  234. if( Status != ERROR_SUCCESS ){
  235. SetLastError( Status );
  236. return FALSE;
  237. }
  238. return TRUE;
  239. }
  240. BOOL
  241. bCompatibleDevMode(
  242. PPRINTHANDLE pPrintHandle,
  243. PDEVMODE pDevModeBase,
  244. PDEVMODE pDevModeNew
  245. )
  246. /*++
  247. Routine Description:
  248. Check if two DevModes are compatible (e.g., they can be used
  249. interchangably).
  250. This is done by checking size and version information. Not
  251. foolproof, but the best we can do since we can't look at private
  252. information.
  253. Arguments:
  254. pPrintHandle - Printer to check.
  255. pDevModeBase - Known good DevMode.
  256. pDevModeNew - DevMode to check.
  257. Return Value:
  258. TRUE - Appears compatible.
  259. FALSE - Not compatible.
  260. --*/
  261. {
  262. if( !pDevModeBase || ! pDevModeNew ){
  263. return FALSE;
  264. }
  265. return pDevModeBase->dmSize == pDevModeNew->dmSize &&
  266. pDevModeBase->dmDriverExtra == pDevModeNew->dmDriverExtra &&
  267. pDevModeBase->dmSpecVersion == pDevModeNew->dmSpecVersion &&
  268. pDevModeBase->dmDriverVersion == pDevModeNew->dmDriverVersion;
  269. }
  270. /********************************************************************
  271. Support Functions
  272. ********************************************************************/
  273. BOOL
  274. bGetDevModeLocation(
  275. IN HKEY hKeyUser, OPTIONAL
  276. IN LPCWSTR pszPrinter,
  277. OUT PHKEY phKey,
  278. OUT LPCWSTR *ppszValue
  279. )
  280. /*++
  281. Routine Description:
  282. Retrieves the location of the per-user DevMode.
  283. On success, caller is responsible for closing phKey. ppszValue's
  284. life is dependent on pszPrinter.
  285. Arguments:
  286. hKeyUser - HKEY_CURRENT_USER key--optional. If not specified, current
  287. impersonation used.
  288. pszPrinter - Printer to use.
  289. phKey - Receives R/W key of per-user DevMode. On success, this
  290. must be closed by caller.
  291. ppszValue - Receives value of per-user DevMode (where to read/write).
  292. Return Value:
  293. TRUE - Success
  294. FALSE - Failure, LastError set.
  295. --*/
  296. {
  297. HANDLE hKeyClose = NULL;
  298. DWORD Status;
  299. *phKey = NULL;
  300. *ppszValue = NULL;
  301. if( !hKeyUser ){
  302. hKeyUser = GetClientUserHandle( KEY_READ|KEY_WRITE );
  303. hKeyClose = hKeyUser;
  304. }
  305. if( !hKeyUser ){
  306. //
  307. // Failed to get impersonation information. Probably because
  308. // we're not impersonating, so there's no per-user information.
  309. //
  310. Status = GetLastError();
  311. } else {
  312. //
  313. // If it starts with two backslashes, it may be either a connection
  314. // or a masq printer.
  315. //
  316. if( pszPrinter[0] == L'\\' && pszPrinter[1] == L'\\' )
  317. {
  318. //
  319. // Query the registry for pszPrinter and look for DevMode.
  320. // First look at the HKCU:Printer\Connections.
  321. //
  322. if((Status = RegOpenConnectionKey(hKeyUser,(LPWSTR)pszPrinter,phKey)) == ERROR_SUCCESS)
  323. {
  324. *ppszValue = gszDevMode;
  325. }
  326. }
  327. //
  328. // If we didn't find it in Printer\Connection, then it
  329. // must be a local or masq printer.
  330. //
  331. if( !*ppszValue ){
  332. DWORD dwIgnore;
  333. //
  334. // Not a connection or didn't exist in the connections key.
  335. // Look in the Printers\DevModePerUser key.
  336. //
  337. Status = RegCreateKeyEx( hKeyUser,
  338. gszDevModePerUserLocal,
  339. 0,
  340. NULL,
  341. 0,
  342. KEY_READ | KEY_WRITE,
  343. NULL,
  344. phKey,
  345. &dwIgnore );
  346. if( Status == ERROR_SUCCESS ){
  347. *ppszValue = pszPrinter;
  348. }
  349. }
  350. }
  351. if( hKeyClose ){
  352. RegCloseKey( hKeyClose );
  353. }
  354. if( Status != ERROR_SUCCESS ){
  355. SetLastError( Status );
  356. return FALSE;
  357. }
  358. return TRUE;
  359. }
  360. BOOL bGetDevModePerUserEvenForShares(
  361. HKEY hKeyUser,
  362. LPCWSTR pszPrinter,
  363. PDEVMODE *ppDevMode
  364. )
  365. {
  366. BOOL RetVal = FALSE;
  367. HANDLE hPrinter;
  368. if(OpenPrinter((LPWSTR)pszPrinter,&hPrinter,NULL))
  369. {
  370. DWORD PrntrInfoSize=0,PrntrInfoSizeReq=0;
  371. PPRINTER_INFO_2 pPrinterInfo2 = NULL;
  372. if(!GetPrinter(hPrinter,
  373. 2,
  374. (LPBYTE)pPrinterInfo2,
  375. PrntrInfoSize,
  376. &PrntrInfoSizeReq) &&
  377. (GetLastError() == ERROR_INSUFFICIENT_BUFFER) &&
  378. (pPrinterInfo2 = (PPRINTER_INFO_2)AllocSplMem((PrntrInfoSize = PrntrInfoSizeReq))) &&
  379. GetPrinter(hPrinter,
  380. 2,
  381. (LPBYTE)pPrinterInfo2,
  382. PrntrInfoSize,
  383. &PrntrInfoSizeReq))
  384. {
  385. RetVal = bGetDevModePerUser( hKeyUser,
  386. pPrinterInfo2->pPrinterName,
  387. ppDevMode );
  388. }
  389. if(hPrinter)
  390. ClosePrinter(hPrinter);
  391. if(pPrinterInfo2)
  392. FreeSplMem(pPrinterInfo2);
  393. }
  394. else
  395. {
  396. RetVal = bGetDevModePerUser( hKeyUser,
  397. pszPrinter,
  398. ppDevMode );
  399. }
  400. return(RetVal);
  401. }
  402. /*++
  403. Routine Name:
  404. IsValidDevmode
  405. Description:
  406. Check to see whether the devmode passed in is at least as advertised in its
  407. buffer.
  408. Arguments:
  409. pDevmode - The devmode
  410. DevmodeSize - The size of the devmode as advertised in the registry.
  411. Return Value:
  412. An HRESULT
  413. --*/
  414. DWORD
  415. IsValidDevmode(
  416. IN PDEVMODE pDevmode,
  417. IN size_t DevmodeSize
  418. )
  419. {
  420. DWORD Status = ERROR_SUCCESS;
  421. //
  422. // This guarantees that the devmode is at least big enough to get the dmSize
  423. // and dmDriverExtra.
  424. //
  425. Status = DevmodeSize >= MIN_DEVMODE_SIZEW ? ERROR_SUCCESS : ERROR_INVALID_DATA;
  426. //
  427. // The advertised devmode size must be contained inside
  428. //
  429. if (ERROR_SUCCESS == Status)
  430. {
  431. Status = (size_t)(pDevmode->dmSize + pDevmode->dmDriverExtra) <= DevmodeSize ? ERROR_SUCCESS : ERROR_INVALID_DATA;
  432. }
  433. return Status;
  434. }