1 /*
2  * PROJECT:     ReactOS Spooler API
3  * LICENSE:     GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+)
4  * PURPOSE:     Functions related to Printers and printing
5  * COPYRIGHT:   Copyright 2015-2018 Colin Finck (colin@reactos.org)
6  */
7 
8 #include "precomp.h"
9 #include <marshalling/printers.h>
10 //#include <marshalling/printerdrivers.h>
11 #include <strsafe.h>
12 
13 extern HINSTANCE hinstWinSpool;
14 //
15 // See winddiui.h, ReactOS version is limited.
16 // Loading from XyzUI.dll part of XyzDRV.dll set. example TTYUI.DLL or UniDrvUI.DLL.
17 //
18 typedef DWORD (WINAPI *DEVICECAPABILITIES) (HANDLE,PWSTR,WORD,PVOID,PDEVMODEW);
19 static DEVICECAPABILITIES fpDeviceCapabilities;
20 
21 typedef LONG (WINAPI *DEVICEPROPERTYSHEETS) (PPROPSHEETUI_INFO,LPARAM);
22 static DEVICEPROPERTYSHEETS fpDevicePropertySheets;
23 typedef LONG (WINAPI *DOCUMENTPROPERTYSHEETS) (PPROPSHEETUI_INFO,LPARAM);
24 static DOCUMENTPROPERTYSHEETS fpDocumentPropertySheets;
25 
26 typedef LONG (WINAPI *COMMONPROPERTYSHEETUIW) (HWND,PFNPROPSHEETUI,LPARAM,LPDWORD);
27 static COMMONPROPERTYSHEETUIW fpCommonPropertySheetUIW;
28 
29 typedef LONG (WINAPI *QUERYCOLORPROFILE) (HANDLE,PDEVMODEW,ULONG,PVOID,ULONG*,FLONG*);
30 static  QUERYCOLORPROFILE fpQueryColorProfile;
31 
32 typedef BOOL (WINAPI *SPOOLERPRINTEREVENT) (LPWSTR,int,DWORD,LPARAM);
33 static SPOOLERPRINTEREVENT fpPrinterEvent;
34 
35 typedef BOOL (WINAPI *DEVQUERYPRINT) (HANDLE,LPDEVMODEW,DWORD*);
36 static DEVQUERYPRINT fpDevQueryPrint;
37 
38 typedef BOOL (WINAPI *DEVQUERYPRINTEX) (PDEVQUERYPRINT_INFO);
39 static DEVQUERYPRINTEX fpDevQueryPrintEx;
40 
41 //
42 //  PrintUI.dll
43 //
44 LONG WINAPI ConstructPrinterFriendlyName( PWSTR, PVOID, LPDWORD Size );
45 typedef LONG (WINAPI *CONSTRUCTPRINTERFRIENDLYNAME) (PWSTR,PVOID,LPDWORD);
46 static CONSTRUCTPRINTERFRIENDLYNAME fpConstructPrinterFriendlyName;
47 
48 //
49 //  CompstUI User Data
50 //
51 typedef struct _COMPUI_USERDATA
52 {
53   HMODULE hModule;
54   LPWSTR pszPrinterName;
55 } COMPUI_USERDATA, *PCOMPUI_USERDATA;
56 
57 // Local Constants
58 
59 /** And the award for the most confusingly named setting goes to "Device", for storing the default printer of the current user.
60     Ok, I admit that this has historical reasons. It's still not straightforward in any way though! */
61 static const WCHAR wszWindowsKey[] = L"Software\\Microsoft\\Windows NT\\CurrentVersion\\Windows";
62 static const WCHAR wszDeviceValue[] = L"Device";
63 
64 static DWORD
65 _StartDocPrinterSpooled(PSPOOLER_HANDLE pHandle, PDOC_INFO_1W pDocInfo1, PADDJOB_INFO_1W pAddJobInfo1)
66 {
67     DWORD cbNeeded;
68     DWORD dwErrorCode;
69     PJOB_INFO_1W pJobInfo1 = NULL;
70 
71     // Create the spool file.
72     pHandle->hSPLFile = CreateFileW(pAddJobInfo1->Path, GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, CREATE_ALWAYS, 0, NULL);
73     if (pHandle->hSPLFile == INVALID_HANDLE_VALUE)
74     {
75         dwErrorCode = GetLastError();
76         ERR("CreateFileW failed for \"%S\" with error %lu!\n", pAddJobInfo1->Path, dwErrorCode);
77         goto Cleanup;
78     }
79 
80     // Get the size of the job information.
81     GetJobW((HANDLE)pHandle, pAddJobInfo1->JobId, 1, NULL, 0, &cbNeeded);
82     if (GetLastError() != ERROR_INSUFFICIENT_BUFFER)
83     {
84         dwErrorCode = GetLastError();
85         ERR("GetJobW failed with error %lu!\n", dwErrorCode);
86         goto Cleanup;
87     }
88 
89     // Allocate enough memory for the returned job information.
90     pJobInfo1 = HeapAlloc(hProcessHeap, 0, cbNeeded);
91     if (!pJobInfo1)
92     {
93         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
94         ERR("HeapAlloc failed!\n");
95         goto Cleanup;
96     }
97 
98     // Get the job information.
99     if (!GetJobW((HANDLE)pHandle, pAddJobInfo1->JobId, 1, (PBYTE)pJobInfo1, cbNeeded, &cbNeeded))
100     {
101         dwErrorCode = GetLastError();
102         ERR("GetJobW failed with error %lu!\n", dwErrorCode);
103         goto Cleanup;
104     }
105 
106     // Add our document information.
107     if (pDocInfo1->pDatatype)
108         pJobInfo1->pDatatype = pDocInfo1->pDatatype;
109 
110     pJobInfo1->pDocument = pDocInfo1->pDocName;
111 
112     // Set the new job information.
113     if (!SetJobW((HANDLE)pHandle, pAddJobInfo1->JobId, 1, (PBYTE)pJobInfo1, 0))
114     {
115         dwErrorCode = GetLastError();
116         ERR("SetJobW failed with error %lu!\n", dwErrorCode);
117         goto Cleanup;
118     }
119 
120     // We were successful!
121     pHandle->dwJobID = pAddJobInfo1->JobId;
122     dwErrorCode = ERROR_SUCCESS;
123 
124 Cleanup:
125     if (pJobInfo1)
126         HeapFree(hProcessHeap, 0, pJobInfo1);
127 
128     return dwErrorCode;
129 }
130 
131 static DWORD
132 _StartDocPrinterWithRPC(PSPOOLER_HANDLE pHandle, PDOC_INFO_1W pDocInfo1)
133 {
134     DWORD dwErrorCode;
135     WINSPOOL_DOC_INFO_CONTAINER DocInfoContainer;
136 
137     DocInfoContainer.Level = 1;
138     DocInfoContainer.DocInfo.pDocInfo1 = (WINSPOOL_DOC_INFO_1*)pDocInfo1;
139 
140     RpcTryExcept
141     {
142         dwErrorCode = _RpcStartDocPrinter(pHandle->hPrinter, &DocInfoContainer, &pHandle->dwJobID);
143     }
144     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
145     {
146         dwErrorCode = RpcExceptionCode();
147         ERR("_RpcStartDocPrinter failed with exception code %lu!\n", dwErrorCode);
148     }
149     RpcEndExcept;
150 
151     return dwErrorCode;
152 }
153 
154 BOOL WINAPI
155 AbortPrinter(HANDLE hPrinter)
156 {
157     DWORD dwErrorCode;
158     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
159 
160     TRACE("AbortPrinter(%p)\n", hPrinter);
161 
162     // Sanity checks.
163     if (!pHandle)
164     {
165         dwErrorCode = ERROR_INVALID_HANDLE;
166         goto Cleanup;
167     }
168 
169     pHandle->bTrayIcon = pHandle->bStartedDoc = FALSE;
170 
171     if ( pHandle->hSPLFile != INVALID_HANDLE_VALUE && pHandle->bJob )
172     {
173         // Close any open file handle.
174         CloseHandle( pHandle->hSPLFile );
175         pHandle->hSPLFile = INVALID_HANDLE_VALUE;
176 
177         SetJobW( hPrinter, pHandle->dwJobID, 0, NULL, JOB_CONTROL_DELETE );
178 
179         return ScheduleJob( hPrinter, pHandle->dwJobID );
180     }
181 
182     // Do the RPC call.
183     RpcTryExcept
184     {
185         dwErrorCode = _RpcAbortPrinter(&pHandle->hPrinter);
186     }
187     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
188     {
189         dwErrorCode = RpcExceptionCode();
190         ERR("_RpcAbortPrinter failed with exception code %lu!\n", dwErrorCode);
191     }
192     RpcEndExcept;
193 
194 Cleanup:
195     SetLastError(dwErrorCode);
196     return (dwErrorCode == ERROR_SUCCESS);
197 }
198 
199 HANDLE WINAPI
200 AddPrinterA(PSTR pName, DWORD Level, PBYTE pPrinter)
201 {
202     UNICODE_STRING pNameW, usBuffer;
203     PWSTR pwstrNameW;
204     PRINTER_INFO_2W *ppi2w = (PRINTER_INFO_2W*)pPrinter;
205     PRINTER_INFO_2A *ppi2a = (PRINTER_INFO_2A*)pPrinter;
206     HANDLE ret = NULL;
207     PWSTR pwszPrinterName = NULL;
208     PWSTR pwszServerName = NULL;
209     PWSTR pwszShareName = NULL;
210     PWSTR pwszPortName = NULL;
211     PWSTR pwszDriverName = NULL;
212     PWSTR pwszComment = NULL;
213     PWSTR pwszLocation = NULL;
214     PWSTR pwszSepFile = NULL;
215     PWSTR pwszPrintProcessor = NULL;
216     PWSTR pwszDatatype = NULL;
217     PWSTR pwszParameters = NULL;
218     PDEVMODEW pdmw = NULL;
219 
220     TRACE("AddPrinterA(%s, %d, %p)\n", debugstr_a(pName), Level, pPrinter);
221 
222     if(Level != 2)
223     {
224         ERR("Level = %d, unsupported!\n", Level);
225         SetLastError(ERROR_INVALID_LEVEL);
226         return NULL;
227     }
228 
229     pwstrNameW = AsciiToUnicode(&pNameW,pName);
230 
231     if (ppi2a->pShareName)
232     {
233         pwszShareName = AsciiToUnicode(&usBuffer, ppi2a->pShareName);
234         if (!(ppi2w->pShareName = pwszShareName)) goto Cleanup;
235     }
236     if (ppi2a->pPortName)
237     {
238         pwszPortName = AsciiToUnicode(&usBuffer, ppi2a->pPortName);
239         if (!(ppi2w->pPortName = pwszPortName)) goto Cleanup;
240     }
241     if (ppi2a->pDriverName)
242     {
243         pwszDriverName = AsciiToUnicode(&usBuffer, ppi2a->pDriverName);
244         if (!(ppi2w->pDriverName = pwszDriverName)) goto Cleanup;
245     }
246     if (ppi2a->pComment)
247     {
248         pwszComment = AsciiToUnicode(&usBuffer, ppi2a->pComment);
249         if (!(ppi2w->pComment = pwszComment)) goto Cleanup;
250     }
251     if (ppi2a->pLocation)
252     {
253         pwszLocation = AsciiToUnicode(&usBuffer, ppi2a->pLocation);
254         if (!(ppi2w->pLocation = pwszLocation)) goto Cleanup;
255     }
256     if (ppi2a->pSepFile)
257     {
258         pwszSepFile = AsciiToUnicode(&usBuffer, ppi2a->pSepFile);
259         if (!(ppi2w->pSepFile = pwszSepFile)) goto Cleanup;
260     }
261     if (ppi2a->pServerName)
262     {
263         pwszPrintProcessor = AsciiToUnicode(&usBuffer, ppi2a->pPrintProcessor);
264         if (!(ppi2w->pPrintProcessor = pwszPrintProcessor)) goto Cleanup;
265     }
266     if (ppi2a->pDatatype)
267     {
268         pwszDatatype = AsciiToUnicode(&usBuffer, ppi2a->pDatatype);
269         if (!(ppi2w->pDatatype = pwszDatatype)) goto Cleanup;
270     }
271     if (ppi2a->pParameters)
272     {
273         pwszParameters = AsciiToUnicode(&usBuffer, ppi2a->pParameters);
274         if (!(ppi2w->pParameters = pwszParameters)) goto Cleanup;
275     }
276     if ( ppi2a->pDevMode )
277     {
278         RosConvertAnsiDevModeToUnicodeDevmode( ppi2a->pDevMode, &pdmw );
279         ppi2w->pDevMode = pdmw;
280     }
281     if (ppi2a->pServerName)
282     {
283         pwszServerName = AsciiToUnicode(&usBuffer, ppi2a->pServerName);
284         if (!(ppi2w->pPrinterName = pwszServerName)) goto Cleanup;
285     }
286     if (ppi2a->pPrinterName)
287     {
288         pwszPrinterName = AsciiToUnicode(&usBuffer, ppi2a->pPrinterName);
289         if (!(ppi2w->pPrinterName = pwszPrinterName)) goto Cleanup;
290     }
291 
292     ret = AddPrinterW(pwstrNameW, Level, (LPBYTE)ppi2w);
293 
294 Cleanup:
295     if (pdmw) HeapFree(hProcessHeap, 0, pdmw);
296     if (pwszPrinterName) HeapFree(hProcessHeap, 0, pwszPrinterName);
297     if (pwszServerName) HeapFree(hProcessHeap, 0, pwszServerName);
298     if (pwszShareName) HeapFree(hProcessHeap, 0, pwszShareName);
299     if (pwszPortName) HeapFree(hProcessHeap, 0, pwszPortName);
300     if (pwszDriverName) HeapFree(hProcessHeap, 0, pwszDriverName);
301     if (pwszComment) HeapFree(hProcessHeap, 0, pwszComment);
302     if (pwszLocation) HeapFree(hProcessHeap, 0, pwszLocation);
303     if (pwszSepFile) HeapFree(hProcessHeap, 0, pwszSepFile);
304     if (pwszPrintProcessor) HeapFree(hProcessHeap, 0, pwszPrintProcessor);
305     if (pwszDatatype) HeapFree(hProcessHeap, 0, pwszDatatype);
306     if (pwszParameters) HeapFree(hProcessHeap, 0, pwszParameters);
307 
308     RtlFreeUnicodeString(&pNameW);
309     return ret;
310 }
311 
312 HANDLE WINAPI
313 AddPrinterW(PWSTR pName, DWORD Level, PBYTE pPrinter)
314 {
315     DWORD dwErrorCode;
316     WINSPOOL_PRINTER_CONTAINER PrinterContainer;
317     WINSPOOL_DEVMODE_CONTAINER DevModeContainer;
318     WINSPOOL_SECURITY_CONTAINER SecurityContainer;
319     SECURITY_DESCRIPTOR *sd = NULL;
320     DWORD size;
321     HANDLE hPrinter = NULL, hHandle = NULL;
322     PSPOOLER_HANDLE pHandle = NULL;
323 
324     TRACE("AddPrinterW(%S, %lu, %p)\n", pName, Level, pPrinter);
325 
326     DevModeContainer.cbBuf = 0;
327     DevModeContainer.pDevMode = NULL;
328 
329     SecurityContainer.cbBuf = 0;
330     SecurityContainer.pSecurity = NULL;
331 
332     if ( Level != 2 )
333     {
334         FIXME( "Unsupported level %d\n", Level );
335         SetLastError( ERROR_INVALID_LEVEL );
336         return hHandle;
337     }
338     else
339     {
340         PPRINTER_INFO_2W pi2w = (PPRINTER_INFO_2W)pPrinter;
341         if ( pi2w )
342         {
343             if ( pi2w->pDevMode )
344             {
345                 if ( IsValidDevmodeNoSizeW( pi2w->pDevMode ) )
346                 {
347                     DevModeContainer.cbBuf = pi2w->pDevMode->dmSize + pi2w->pDevMode->dmDriverExtra;
348                     DevModeContainer.pDevMode = (PBYTE)pi2w->pDevMode;
349                 }
350             }
351 
352             if ( pi2w->pSecurityDescriptor )
353             {
354                 sd = get_sd( pi2w->pSecurityDescriptor, &size );
355                 if ( sd )
356                 {
357                     SecurityContainer.cbBuf = size;
358                     SecurityContainer.pSecurity = (PBYTE)sd;
359                 }
360             }
361         }
362         else
363         {
364             SetLastError(ERROR_INVALID_PARAMETER);
365             return hHandle;
366         }
367     }
368 
369     PrinterContainer.PrinterInfo.pPrinterInfo1 = (WINSPOOL_PRINTER_INFO_1*)pPrinter;
370     PrinterContainer.Level = Level;
371 
372     // Do the RPC call
373     RpcTryExcept
374     {
375         dwErrorCode = _RpcAddPrinter( pName, &PrinterContainer, &DevModeContainer, &SecurityContainer, &hPrinter );
376     }
377     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
378     {
379         dwErrorCode = RpcExceptionCode();
380     }
381     RpcEndExcept;
382 
383     if (hPrinter)
384     {
385         // Create a new SPOOLER_HANDLE structure.
386         pHandle = HeapAlloc(hProcessHeap, HEAP_ZERO_MEMORY, sizeof(SPOOLER_HANDLE));
387         if (!pHandle)
388         {
389             dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
390             ERR("HeapAlloc failed!\n");
391             _RpcDeletePrinter(hPrinter);
392             _RpcClosePrinter(hPrinter);
393             goto Cleanup;
394         }
395 
396         pHandle->Sig = SPOOLER_HANDLE_SIG;
397         pHandle->hPrinter = hPrinter;
398         pHandle->hSPLFile = INVALID_HANDLE_VALUE;
399         pHandle->hSpoolFileHandle = INVALID_HANDLE_VALUE;
400         hHandle = (HANDLE)pHandle;
401     }
402 
403 Cleanup:
404     if ( sd ) HeapFree( GetProcessHeap(), 0, sd );
405 
406     SetLastError(dwErrorCode);
407     return hHandle;
408 }
409 
410 BOOL WINAPI
411 ClosePrinter(HANDLE hPrinter)
412 {
413     DWORD dwErrorCode;
414     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
415 
416     TRACE("ClosePrinter(%p)\n", hPrinter);
417 
418     // Sanity checks.
419     if ( IntProtectHandle( hPrinter, TRUE ) )
420     {
421         dwErrorCode = ERROR_INVALID_HANDLE;
422         goto Cleanup;
423     }
424 
425     // Do the RPC call.
426     RpcTryExcept
427     {
428         dwErrorCode = _RpcClosePrinter(&pHandle->hPrinter);
429     }
430     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
431     {
432         dwErrorCode = RpcExceptionCode();
433         ERR("_RpcClosePrinter failed with exception code %lu!\n", dwErrorCode);
434     }
435     RpcEndExcept;
436 
437     // Close any open file handle.
438     if (pHandle->hSPLFile != INVALID_HANDLE_VALUE)
439         CloseHandle(pHandle->hSPLFile);
440 
441     pHandle->Sig = -1;
442 
443     // Free the memory for the handle.
444     HeapFree(hProcessHeap, 0, pHandle);
445 
446 Cleanup:
447     SetLastError(dwErrorCode);
448     return (dwErrorCode == ERROR_SUCCESS);
449 }
450 
451 BOOL WINAPI
452 DeletePrinter(HANDLE hPrinter)
453 {
454     DWORD dwErrorCode;
455     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
456 
457     TRACE("DeletePrinter(%p)\n", hPrinter);
458 
459     // Sanity checks.
460     if (!pHandle)
461     {
462         dwErrorCode = ERROR_INVALID_HANDLE;
463         goto Cleanup;
464     }
465 
466     // Do the RPC call.
467     RpcTryExcept
468     {
469         dwErrorCode = _RpcDeletePrinter(&pHandle->hPrinter);
470     }
471     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
472     {
473         dwErrorCode = RpcExceptionCode();
474         ERR("_RpcDeletePrinter failed with exception code %lu!\n", dwErrorCode);
475     }
476     RpcEndExcept;
477 
478 Cleanup:
479     SetLastError(dwErrorCode);
480     return (dwErrorCode == ERROR_SUCCESS);
481 }
482 
483 //
484 // Based on GDI32:printdrv.c:IntGetPrinterDriver.
485 //
486 HMODULE
487 WINAPI
488 LoadPrinterDriver( HANDLE hspool )
489 {
490     INT iTries = 0;
491     DWORD Size = (sizeof(WCHAR) * MAX_PATH) * 2; // DRIVER_INFO_5W + plus strings.
492     PDRIVER_INFO_5W pdi = NULL;
493     HMODULE hLibrary = NULL;
494 
495     do
496     {
497         ++iTries;
498 
499         pdi = RtlAllocateHeap( GetProcessHeap(), 0, Size);
500 
501         if ( !pdi )
502             break;
503 
504         if ( GetPrinterDriverW(hspool, NULL, 5, (LPBYTE)pdi, Size, &Size) )
505         {
506             TRACE("Level 5 Size %d\n",Size);
507 
508             // Name and load configure library (for example, C:\DRIVERS\Pscrptui.dll). Not printui.dll!
509 
510             hLibrary = LoadLibrary(pdi->pConfigFile);
511 
512             FIXME("IGPD : Get Printer Driver Config File : %S\n",pdi->pConfigFile);
513 
514             RtlFreeHeap( GetProcessHeap(), 0, pdi);
515             return hLibrary;
516         }
517 
518         if ( GetLastError() != ERROR_INSUFFICIENT_BUFFER )
519             ++iTries;
520 
521         RtlFreeHeap( GetProcessHeap(), 0, pdi);
522      }
523      while ( iTries < 2 );
524      ERR("No Printer Driver Error %d\n",GetLastError());
525      return NULL;
526 }
527 
528 DWORD WINAPI
529 DeviceCapabilitiesA(LPCSTR pDevice, LPCSTR pPort, WORD fwCapability, LPSTR pOutput, const DEVMODEA* pDevMode)
530 {
531     PWSTR pwszDeviceName = NULL;
532     PDEVMODEW pdmwInput = NULL;
533     BOOL bReturnValue = GDI_ERROR;
534     DWORD cch;
535 
536     FIXME("DeviceCapabilitiesA(%s, %s, %hu, %p, %p)\n", pDevice, pPort, fwCapability, pOutput, pDevMode);
537 
538     if (pDevice)
539     {
540         // Convert pName to a Unicode string pwszDeviceName.
541         cch = strlen(pDevice);
542 
543         pwszDeviceName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(WCHAR));
544         if (!pwszDeviceName)
545         {
546             SetLastError(ERROR_NOT_ENOUGH_MEMORY);
547             ERR("HeapAlloc failed!\n");
548             goto Cleanup;
549         }
550 
551         MultiByteToWideChar(CP_ACP, 0, pDevice, -1, pwszDeviceName, cch + 1);
552     }
553 
554     if (pDevMode)
555     {
556         RosConvertAnsiDevModeToUnicodeDevmode((PDEVMODEA)pDevMode, &pdmwInput);
557     }
558 
559     // pPort is ignored so no need to pass it.
560     bReturnValue = DeviceCapabilitiesW( pwszDeviceName, NULL, fwCapability, (LPWSTR)pOutput, (const DEVMODEW*) pdmwInput );
561 
562 Cleanup:
563     if(pwszDeviceName)
564         HeapFree(hProcessHeap, 0, pwszDeviceName);
565 
566     if (pdmwInput)
567         HeapFree(hProcessHeap, 0, pdmwInput);
568 
569     return bReturnValue;
570 }
571 
572 DWORD WINAPI
573 DeviceCapabilitiesW(LPCWSTR pDevice, LPCWSTR pPort, WORD fwCapability, LPWSTR pOutput, const DEVMODEW* pDevMode)
574 {
575     HANDLE hPrinter;
576     HMODULE hLibrary;
577     DWORD iDevCap = GDI_ERROR;
578 
579     FIXME("DeviceCapabilitiesW(%S, %S, %hu, %p, %p)\n", pDevice, pPort, fwCapability, pOutput, pDevMode);
580 
581     if ( pDevMode )
582     {
583         if (!IsValidDevmodeNoSizeW( (PDEVMODEW)pDevMode ) )
584         {
585             ERR("DeviceCapabilitiesW : Devode Invalid\n");
586             return -1;
587         }
588     }
589 
590     if ( OpenPrinterW( (LPWSTR)pDevice, &hPrinter, NULL ) )
591     {
592         hLibrary = LoadPrinterDriver( hPrinter );
593 
594         if ( hLibrary )
595         {
596             fpDeviceCapabilities = (PVOID)GetProcAddress( hLibrary, "DrvDeviceCapabilities" );
597 
598             if ( fpDeviceCapabilities )
599             {
600                 iDevCap = fpDeviceCapabilities( hPrinter, (PWSTR)pDevice, fwCapability, pOutput, (PDEVMODE)pDevMode );
601             }
602 
603             FreeLibrary(hLibrary);
604         }
605 
606         ClosePrinter( hPrinter );
607     }
608 
609     return iDevCap;
610 }
611 
612 BOOL
613 WINAPI
614 DevQueryPrint( HANDLE hPrinter, LPDEVMODEW pDevMode, DWORD *pResID)
615 {
616     HMODULE hLibrary;
617     BOOL Ret = FALSE;
618 
619     hLibrary = LoadPrinterDriver( hPrinter );
620 
621     if ( hLibrary )
622     {
623         fpDevQueryPrint = (PVOID)GetProcAddress( hLibrary, "DevQueryPrint" );
624 
625         if ( fpDevQueryPrint )
626         {
627             Ret = fpDevQueryPrint( hPrinter, pDevMode, pResID );
628         }
629 
630         FreeLibrary(hLibrary);
631     }
632     return Ret;
633 }
634 
635 BOOL WINAPI
636 DevQueryPrintEx( PDEVQUERYPRINT_INFO pDQPInfo )
637 {
638     HMODULE hLibrary;
639     BOOL Ret = FALSE;
640 
641     hLibrary = LoadPrinterDriver( pDQPInfo->hPrinter );
642 
643     if ( hLibrary )
644     {
645         fpDevQueryPrintEx = (PVOID)GetProcAddress( hLibrary, "DevQueryPrintEx" );
646 
647         if ( fpDevQueryPrintEx )
648         {
649             Ret = fpDevQueryPrintEx( pDQPInfo );
650         }
651 
652         FreeLibrary(hLibrary);
653     }
654     return Ret;
655 }
656 
657 INT WINAPI
658 DocumentEvent( HANDLE hPrinter, HDC hdc, int iEsc, ULONG cbIn, PVOID pvIn, ULONG cbOut, PVOID pvOut)
659 {
660     FIXME("DocumentEvent(%p, %p, %lu, %lu, %p, %lu, %p)\n", hPrinter, hdc, iEsc, cbIn, pvIn, cbOut, pvOut);
661     UNIMPLEMENTED;
662     return DOCUMENTEVENT_UNSUPPORTED;
663 }
664 
665 LONG WINAPI
666 DocumentPropertiesA(HWND hWnd, HANDLE hPrinter, LPSTR pDeviceName, PDEVMODEA pDevModeOutput, PDEVMODEA pDevModeInput, DWORD fMode)
667 {
668     PWSTR pwszDeviceName = NULL;
669     PDEVMODEW pdmwInput = NULL;
670     PDEVMODEW pdmwOutput = NULL;
671     LONG lReturnValue = -1;
672     DWORD cch;
673 
674     FIXME("DocumentPropertiesA(%p, %p, %s, %p, %p, %lu)\n", hWnd, hPrinter, pDeviceName, pDevModeOutput, pDevModeInput, fMode);
675 
676     if (pDeviceName)
677     {
678         // Convert pName to a Unicode string pwszDeviceName.
679         cch = strlen(pDeviceName);
680 
681         pwszDeviceName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(WCHAR));
682         if (!pwszDeviceName)
683         {
684             SetLastError(ERROR_NOT_ENOUGH_MEMORY);
685             ERR("HeapAlloc failed!\n");
686             goto Cleanup;
687         }
688 
689         MultiByteToWideChar(CP_ACP, 0, pDeviceName, -1, pwszDeviceName, cch + 1);
690     }
691 
692     if (pDevModeInput)
693     {
694         // Create working buffer for input to DocumentPropertiesW.
695         RosConvertAnsiDevModeToUnicodeDevmode(pDevModeInput, &pdmwInput);
696     }
697 
698     if (pDevModeOutput)
699     {
700         // Create working buffer for output from DocumentPropertiesW.
701 
702         // Do it RIGHT! Get the F...ing Size!
703         LONG Size = DocumentPropertiesW( hWnd, hPrinter, pwszDeviceName, NULL, NULL, 0 );
704 
705         if ( Size < 0 )
706         {
707             goto Cleanup;
708         }
709 
710         pdmwOutput = HeapAlloc(hProcessHeap, 0, Size);
711         if (!pdmwOutput)
712         {
713             SetLastError(ERROR_NOT_ENOUGH_MEMORY);
714             ERR("HeapAlloc failed!\n");
715             goto Cleanup;
716         }
717     }
718 
719     lReturnValue = DocumentPropertiesW(hWnd, hPrinter, pwszDeviceName, pdmwOutput, pdmwInput, fMode);
720     FIXME("lReturnValue from DocumentPropertiesW is '%ld'.\n", lReturnValue);
721 
722     if (lReturnValue < 0)
723     {
724         FIXME("DocumentPropertiesW failed!\n");
725         goto Cleanup;
726     }
727 
728     if (pdmwOutput)
729     {
730         RosConvertUnicodeDevModeToAnsiDevmode(pdmwOutput, pDevModeOutput);
731     }
732 
733 Cleanup:
734     if(pwszDeviceName)
735         HeapFree(hProcessHeap, 0, pwszDeviceName);
736 
737     if (pdmwInput)
738         HeapFree(hProcessHeap, 0, pdmwInput);
739 
740     if (pdmwOutput)
741         HeapFree(hProcessHeap, 0, pdmwOutput);
742 
743     return lReturnValue;
744 }
745 
746 PRINTER_INFO_9W * get_devmodeW(HANDLE hprn)
747 {
748     PRINTER_INFO_9W *pi9 = NULL;
749     DWORD needed = 0;
750     BOOL res;
751 
752     res = GetPrinterW(hprn, 9, NULL, 0, &needed);
753     if (!res && (GetLastError() == ERROR_INSUFFICIENT_BUFFER))
754     {
755         pi9 = HeapAlloc(hProcessHeap, 0, needed);
756         if (!pi9)
757         {
758             ERR("Failed to allocate PRINTER_INFO_9W of %u bytes\n", needed);
759             return NULL;
760         }
761         res = GetPrinterW(hprn, 9, (LPBYTE)pi9, needed, &needed);
762     }
763 
764     if (res)
765         return pi9;
766 
767     ERR("GetPrinterW failed with %u\n", GetLastError());
768     HeapFree(hProcessHeap, 0, pi9);
769     return NULL;
770 }
771 
772 BOOL
773 FASTCALL
774 CreateUIUserData( ULONG_PTR *puserdata, HANDLE hPrinter )
775 {
776     PCOMPUI_USERDATA pcui_ud = DllAllocSplMem( sizeof(COMPUI_USERDATA) );
777 
778     *puserdata = (ULONG_PTR)pcui_ud;
779     FIXME("CreateUIUserData\n");
780     if ( pcui_ud )
781     {
782         pcui_ud->hModule = LoadPrinterDriver( hPrinter );
783 
784         if ( !pcui_ud->hModule )
785         {
786             DllFreeSplMem( pcui_ud );
787             *puserdata = 0;
788         }
789     }
790     return *puserdata != 0;
791 }
792 
793 VOID
794 FASTCALL
795 DestroyUIUserData( ULONG_PTR *puserdata )
796 {
797     PCOMPUI_USERDATA pcui_ud = (PCOMPUI_USERDATA)*puserdata;
798     FIXME("DestroyUIUserData\n");
799     if ( pcui_ud )
800     {
801         if ( pcui_ud->hModule )
802         {
803             FreeLibrary( pcui_ud->hModule );
804             pcui_ud->hModule = NULL;
805         }
806 
807         if ( pcui_ud->pszPrinterName )
808         {
809             DllFreeSplMem( pcui_ud->pszPrinterName );
810             pcui_ud->pszPrinterName = NULL;
811         }
812 
813         DllFreeSplMem( pcui_ud );
814         *puserdata = 0;
815     }
816 }
817 
818 BOOL
819 FASTCALL
820 IntFixUpDevModeNames( PDOCUMENTPROPERTYHEADER pdphdr )
821 {
822     PRINTER_INFO_2W *pi2 = NULL;
823     DWORD needed = 0;
824     BOOL res;
825 
826     if (!(pdphdr->fMode & DM_OUT_BUFFER) ||
827          pdphdr->fMode & DM_NOPERMISSION || // Do not allow the user to modify properties on the displayed property sheet pages.
828         !pdphdr->pdmOut )
829     {
830         return FALSE;
831     }
832 
833     res = GetPrinterW( pdphdr->hPrinter, 2, NULL, 0, &needed);
834     if (!res && (GetLastError() == ERROR_INSUFFICIENT_BUFFER))
835     {
836         pi2 = HeapAlloc(hProcessHeap, 0, needed);
837         if (!pi2)
838         {
839             ERR("Failed to allocate PRINTER_INFO_2W of %u bytes\n", needed);
840             return FALSE;
841         }
842         res = GetPrinterW( pdphdr->hPrinter, 2, (LPBYTE)pi2, needed, &needed);
843     }
844 
845     if (res)
846     {
847         /* Check if the provided buffer is large enough */
848         DWORD cbDevMode = pi2->pDevMode->dmSize + pi2->pDevMode->dmDriverExtra;
849         if (pdphdr->cbOut < cbDevMode)
850         {
851             ERR("cbOut (%lu) < cbDevMode(%u)\n", pdphdr->cbOut, cbDevMode);
852             res = FALSE;
853             goto Exit;
854         }
855 
856         /* Copy the devmode */
857         RtlCopyMemory(pdphdr->pdmOut, pi2->pDevMode, cbDevMode);
858 
859         TRACE("IFUDMN : Get Printer Name %S\n", pi2->pPrinterName);
860         StringCchCopyW( pdphdr->pdmOut->dmDeviceName, CCHDEVICENAME-1, pi2->pPrinterName );
861         pdphdr->pdmOut->dmDeviceName[CCHDEVICENAME-1] = 0;
862     }
863     else
864     {
865         ERR("IFUDMN : GetPrinterW failed with %u\n", GetLastError());
866     }
867 
868 Exit:
869     HeapFree(hProcessHeap, 0, pi2);
870     return res;
871 }
872 
873 LONG
874 WINAPI
875 CreatePrinterFriendlyName( PCOMPUI_USERDATA pcui_ud, LPWSTR pszPrinterName )
876 {
877     LONG Result = 0;
878     DWORD Size = 0;
879     HMODULE hLibrary = NULL;
880 
881     hLibrary = LoadLibraryA( "printui.dll" );
882 
883     if ( hLibrary )
884     {
885         fpConstructPrinterFriendlyName = (PVOID)GetProcAddress( hLibrary, "ConstructPrinterFriendlyName" );
886 
887         if ( fpConstructPrinterFriendlyName )
888         {
889              if ( !fpConstructPrinterFriendlyName( pszPrinterName, NULL, &Size ) )
890              {
891                  if ( GetLastError() == ERROR_INSUFFICIENT_BUFFER )
892                  {
893                      PWSTR pwstr = DllAllocSplMem( (Size + 1) * sizeof(WCHAR) );
894 
895                      pcui_ud->pszPrinterName = pwstr;
896 
897                      if ( pwstr )
898                          Result = fpConstructPrinterFriendlyName( pszPrinterName, pwstr, &Size );
899                  }
900              }
901         }
902         FreeLibrary( hLibrary );
903     }
904 
905     if ( !Result )
906     {
907         DllFreeSplMem( pcui_ud->pszPrinterName );
908         pcui_ud->pszPrinterName = AllocSplStr( pszPrinterName );
909     }
910 
911     return Result;
912 }
913 
914 //
915 // Tested with XP CompstUI as a callback and works. Fails perfectly.
916 //
917 LONG
918 WINAPI
919 DocumentPropertySheets( PPROPSHEETUI_INFO pCPSUIInfo, LPARAM lparam )
920 {
921     LONG Result = -1;
922     PDOCUMENTPROPERTYHEADER pdphdr;
923 
924     FIXME("DocumentPropertySheets(%p, 0x%lx)\n", pCPSUIInfo, lparam);
925 
926     // If pPSUIInfo is NULL, and if either lParam -> fMode is zero or lParam -> pdmOut is NULL,
927     // this function should return the size, in bytes, of the printer's DEVMODEW structure.
928     if ( !pCPSUIInfo && lparam )
929     {
930         pdphdr = (PDOCUMENTPROPERTYHEADER)lparam;
931 
932         if ( pdphdr->cbSize >= sizeof(PDOCUMENTPROPERTYHEADER) &&
933             !(pdphdr->fMode & DM_PROMPT) )
934         {
935             HMODULE hLibrary = LoadPrinterDriver( pdphdr->hPrinter );
936 
937             if ( hLibrary )
938             {
939                 fpDocumentPropertySheets = (PVOID)GetProcAddress( hLibrary, "DrvDocumentPropertySheets" );
940 
941                 if ( fpDocumentPropertySheets )
942                 {
943                     FIXME("DPS : fpDocumentPropertySheets(%p, 0x%lx) pdmOut %p\n", pCPSUIInfo, lparam, pdphdr->pdmOut);
944                     Result = fpDocumentPropertySheets( pCPSUIInfo, lparam );
945                     FIXME("DPS : fpDocumentPropertySheets result %d cbOut %d\n",Result, pdphdr->cbOut);
946                 }
947                 else
948                 {
949                     //
950                     // ReactOS backup!!! Currently no supporting UI driver.
951                     //
952                     PRINTER_INFO_9W * pi9 = get_devmodeW( pdphdr->hPrinter );
953                     if ( pi9 )
954                     {
955                         Result = pi9->pDevMode->dmSize + pi9->pDevMode->dmDriverExtra;
956                         FIXME("IDPS : Using ReactOS backup!!! DevMode Size %d\n",Result);
957                         HeapFree(hProcessHeap, 0, pi9);
958                     }
959                 }
960 
961                 FreeLibrary(hLibrary);
962 
963                 if ( Result > 0 )
964                 {
965                     IntFixUpDevModeNames( pdphdr );
966                 }
967 
968                 return Result;
969             }
970             else
971             {
972                 SetLastError(ERROR_INVALID_HANDLE);
973             }
974         }
975         else
976         {
977             SetLastError(ERROR_INVALID_PARAMETER);
978         }
979         return Result;
980     }
981 
982     Result = 0;
983 
984     if ( pCPSUIInfo )
985     {
986         PSETRESULT_INFO psri;
987         PPROPSHEETUI_INFO_HEADER ppsuiihdr;
988         PCOMPUI_USERDATA pcui_ud;
989         pdphdr = (PDOCUMENTPROPERTYHEADER)pCPSUIInfo->lParamInit;
990 
991         if ( pdphdr->cbSize < sizeof(PDOCUMENTPROPERTYHEADER) )
992         {
993             SetLastError(ERROR_INVALID_PARAMETER);
994             return Result;
995         }
996 
997         switch ( pCPSUIInfo->Reason )
998         {
999             case PROPSHEETUI_REASON_INIT:
1000             {
1001                 FIXME("DocPS : PROPSHEETUI_REASON_INIT\n");
1002                 if ( CreateUIUserData( &pCPSUIInfo->UserData, pdphdr->hPrinter ) )
1003                 {
1004                     pcui_ud = (PCOMPUI_USERDATA)pCPSUIInfo->UserData;
1005 
1006                     fpDocumentPropertySheets = (PVOID)GetProcAddress( pcui_ud->hModule, "DrvDocumentPropertySheets" );
1007 
1008                     if ( fpDocumentPropertySheets )
1009                     {
1010                         pCPSUIInfo->pfnComPropSheet( pCPSUIInfo->hComPropSheet,
1011                                                      CPSFUNC_SET_FUSION_CONTEXT,
1012                                                      -3,  // What type of handle is this?
1013                                                      0 ); // Not used, must be zero.
1014 
1015                         Result = pCPSUIInfo->pfnComPropSheet( pCPSUIInfo->hComPropSheet,
1016                                                               CPSFUNC_ADD_PFNPROPSHEETUIW,
1017                                                              (LPARAM)fpDocumentPropertySheets,
1018                                                               pCPSUIInfo->lParamInit );
1019                         break;
1020                     }
1021                     FIXME("DocPS : PROPSHEETUI_REASON_INIT Fail\n");
1022                     DestroyUIUserData( &pCPSUIInfo->UserData );
1023                 }
1024             }
1025                 break;
1026 
1027             case PROPSHEETUI_REASON_GET_INFO_HEADER:
1028                 FIXME("DocPS : PROPSHEETUI_REASON_GET_INFO_HEADER\n");
1029 
1030                 ppsuiihdr = (PPROPSHEETUI_INFO_HEADER)lparam;
1031 
1032                 pcui_ud = (PCOMPUI_USERDATA)pCPSUIInfo->UserData;
1033 
1034                 CreatePrinterFriendlyName( pcui_ud, pdphdr->pszPrinterName );
1035 
1036                 ppsuiihdr->Flags  = PSUIHDRF_NOAPPLYNOW|PSUIHDRF_PROPTITLE;
1037                 ppsuiihdr->pTitle = pcui_ud->pszPrinterName;
1038                 ppsuiihdr->hInst  = hinstWinSpool;
1039                 ppsuiihdr->IconID = IDI_CPSUI_DOCUMENT;
1040 
1041                 Result = CPSUI_OK;
1042                 break;
1043 
1044             case PROPSHEETUI_REASON_DESTROY:
1045                 FIXME("DocPS : PROPSHEETUI_REASON_DESTROY\n");
1046                 DestroyUIUserData( &pCPSUIInfo->UserData );
1047                 Result = CPSUI_OK;
1048                 break;
1049 
1050             case PROPSHEETUI_REASON_SET_RESULT:
1051                 FIXME("DocPS : PROPSHEETUI_REASON_SET_RESULT\n");
1052 
1053                 psri = (PSETRESULT_INFO)lparam;
1054 
1055                 pCPSUIInfo->Result = psri->Result;
1056                 if ( pCPSUIInfo->Result > 0 )
1057                 {
1058                     IntFixUpDevModeNames( pdphdr );
1059                 }
1060                 Result = CPSUI_OK;
1061                 break;
1062         }
1063     }
1064     return Result;
1065 }
1066 
1067 LONG
1068 WINAPI
1069 DevicePropertySheets( PPROPSHEETUI_INFO pCPSUIInfo, LPARAM lparam )
1070 {
1071     LONG Result = 0;
1072     PDEVICEPROPERTYHEADER pdphdr;
1073 
1074     FIXME("DevicePropertySheets(%p, 0x%lx)\n", pCPSUIInfo, lparam);
1075 
1076     if ( pCPSUIInfo )
1077     {
1078         PSETRESULT_INFO psri;
1079         PPROPSHEETUI_INFO_HEADER ppsuiihdr;
1080         PCOMPUI_USERDATA pcui_ud;
1081         pdphdr = (PDEVICEPROPERTYHEADER)pCPSUIInfo->lParamInit;
1082 
1083         if ( pdphdr->cbSize < sizeof(DEVICEPROPERTYHEADER) )
1084         {
1085             SetLastError(ERROR_INVALID_PARAMETER);
1086             return Result;
1087         }
1088 
1089         switch ( pCPSUIInfo->Reason )
1090         {
1091             case PROPSHEETUI_REASON_INIT:
1092             {
1093                 FIXME("DevPS : PROPSHEETUI_REASON_INIT\n");
1094                 if ( CreateUIUserData( &pCPSUIInfo->UserData, pdphdr->hPrinter ) )
1095                 {
1096                     pcui_ud = (PCOMPUI_USERDATA)pCPSUIInfo->UserData;
1097 
1098                     fpDevicePropertySheets = (PVOID)GetProcAddress( pcui_ud->hModule, "DrvDevicePropertySheets" );
1099 
1100                     if ( fpDevicePropertySheets )
1101                     {
1102                         pCPSUIInfo->pfnComPropSheet( pCPSUIInfo->hComPropSheet,
1103                                                      CPSFUNC_SET_FUSION_CONTEXT,
1104                                                      -3,  // What type of handle is this?
1105                                                      0 ); // Not used, must be zero.
1106 
1107                         Result = pCPSUIInfo->pfnComPropSheet( pCPSUIInfo->hComPropSheet,
1108                                                               CPSFUNC_ADD_PFNPROPSHEETUIW,
1109                                                              (LPARAM)fpDevicePropertySheets,
1110                                                               pCPSUIInfo->lParamInit );
1111                         break;
1112                     }
1113                     FIXME("DevPS : PROPSHEETUI_REASON_INIT Fail\n");
1114                     DestroyUIUserData( &pCPSUIInfo->UserData );
1115                 }
1116             }
1117                 break;
1118 
1119             case PROPSHEETUI_REASON_GET_INFO_HEADER:
1120                 FIXME("DevPS : PROPSHEETUI_REASON_GET_INFO_HEADER\n");
1121 
1122                 ppsuiihdr = (PPROPSHEETUI_INFO_HEADER)lparam;
1123 
1124                 pcui_ud = (PCOMPUI_USERDATA)pCPSUIInfo->UserData;
1125 
1126                 CreatePrinterFriendlyName( pcui_ud, pdphdr->pszPrinterName );
1127 
1128                 ppsuiihdr->Flags  = PSUIHDRF_NOAPPLYNOW|PSUIHDRF_PROPTITLE;
1129                 ppsuiihdr->pTitle = pcui_ud->pszPrinterName;
1130                 ppsuiihdr->hInst  = hinstWinSpool;
1131                 ppsuiihdr->IconID = IDI_CPSUI_DOCUMENT;
1132 
1133                 Result = CPSUI_OK;
1134                 break;
1135 
1136             case PROPSHEETUI_REASON_DESTROY:
1137                 FIXME("DevPS : PROPSHEETUI_REASON_DESTROY\n");
1138                 DestroyUIUserData( &pCPSUIInfo->UserData );
1139                 Result = CPSUI_OK;
1140                 break;
1141 
1142             case PROPSHEETUI_REASON_SET_RESULT:
1143                 FIXME("DevPS : PROPSHEETUI_REASON_SET_RESULT\n");
1144                 psri = (PSETRESULT_INFO)lparam;
1145                 pCPSUIInfo->Result = psri->Result;
1146                 Result = CPSUI_OK;
1147                 break;
1148         }
1149     }
1150     return Result;
1151 }
1152 
1153 LONG
1154 WINAPI
1155 CallCommonPropertySheetUI(HWND hWnd, PFNPROPSHEETUI pfnPropSheetUI, LPARAM lparam, LPDWORD pResult)
1156 {
1157     HMODULE hLibrary = NULL;
1158     LONG Ret = ERR_CPSUI_GETLASTERROR;
1159 
1160     FIXME("CallCommonPropertySheetUI(%p, %p, 0x%lx, %p)\n", hWnd, pfnPropSheetUI, lparam, pResult);
1161 
1162     if ( ( hLibrary = LoadLibraryA( "compstui.dll" ) ) )
1163     {
1164         fpCommonPropertySheetUIW = (PVOID) GetProcAddress(hLibrary, "CommonPropertySheetUIW");
1165 
1166         if ( fpCommonPropertySheetUIW )
1167         {
1168             Ret = fpCommonPropertySheetUIW( hWnd, pfnPropSheetUI, lparam, pResult );
1169         }
1170 
1171         FreeLibrary(hLibrary);
1172     }
1173     return Ret;
1174 }
1175 
1176 LONG WINAPI
1177 DocumentPropertiesW(HWND hWnd, HANDLE hPrinter, LPWSTR pDeviceName, PDEVMODEW pDevModeOutput, PDEVMODEW pDevModeInput, DWORD fMode)
1178 {
1179     HANDLE hUseHandle = NULL;
1180     DOCUMENTPROPERTYHEADER docprophdr;
1181     LONG Result = IDOK;
1182 
1183     FIXME("DocumentPropertiesW(%p, %p, %S, %p, %p, %lu)\n", hWnd, hPrinter, pDeviceName, pDevModeOutput, pDevModeInput, fMode);
1184 
1185     if (hPrinter)
1186     {
1187         hUseHandle = hPrinter;
1188     }
1189     else if (!OpenPrinterW(pDeviceName, &hUseHandle, NULL))
1190     {
1191         ERR("No handle, and no usable printer name passed in\n");
1192         return -1;
1193     }
1194 
1195     if ( !(fMode & DM_IN_BUFFER ) ||
1196          ( ( pDevModeInput && !IsValidDevmodeNoSizeW( (PDEVMODEW)pDevModeInput ) ) ) )
1197     {
1198         pDevModeInput = NULL;
1199         fMode &= ~DM_IN_BUFFER;
1200     }
1201 
1202     docprophdr.cbSize         = sizeof(DOCUMENTPROPERTYHEADER);
1203     docprophdr.Reserved       = 0;
1204     docprophdr.hPrinter       = hUseHandle;
1205     docprophdr.pszPrinterName = pDeviceName;
1206     docprophdr.cbOut          = 0;
1207 
1208     if ( pDevModeOutput )
1209     {
1210         docprophdr.pdmIn  = NULL;
1211         docprophdr.pdmOut = NULL;
1212         docprophdr.fMode  = 0;
1213         FIXME("DPW : Call DocumentPropertySheets with pDevModeOutput %p\n",pDevModeOutput);
1214         docprophdr.cbOut  = DocumentPropertySheets( NULL, (LPARAM)&docprophdr );
1215     }
1216 
1217     docprophdr.pdmIn  = pDevModeInput;
1218     docprophdr.pdmOut = pDevModeOutput;
1219     docprophdr.fMode  = fMode;
1220 
1221     if ( fMode & DM_IN_PROMPT )
1222     {
1223         Result = CPSUI_CANCEL;
1224 
1225         //
1226         // Now call the Property Sheet for Print > Properties.
1227         //
1228         if ( CallCommonPropertySheetUI( hWnd, (PFNPROPSHEETUI)DocumentPropertySheets, (LPARAM)&docprophdr, (LPDWORD)&Result ) < 0 )
1229         {
1230             FIXME("CallCommonPropertySheetUI return error\n");
1231             Result = ERR_CPSUI_GETLASTERROR;
1232         }
1233         else
1234             Result = (Result == CPSUI_OK) ? IDOK : IDCANCEL;
1235         FIXME("CallCommonPropertySheetUI returned\n");
1236     }
1237     else
1238     {
1239         FIXME("DPW : CallDocumentPropertySheets\n");
1240         Result = DocumentPropertySheets( NULL, (LPARAM)&docprophdr );
1241     }
1242 
1243     if ( Result != ERR_CPSUI_GETLASTERROR || Result != ERR_CPSUI_ALLOCMEM_FAILED )
1244     {
1245         if ( pDevModeOutput )
1246         {
1247             if ( !IsValidDevmodeNoSizeW( pDevModeOutput ) )
1248             {
1249                 ERR("DPW : Improper pDevModeOutput size.\n");
1250                 Result = -1;
1251             }
1252         }
1253         else
1254         {
1255             ERR("No pDevModeOutput\n");
1256         }
1257     }
1258 
1259     if (hUseHandle && !hPrinter)
1260         ClosePrinter(hUseHandle);
1261     return Result;
1262 }
1263 
1264 BOOL
1265 WINAPI
1266 PrinterProperties( HWND hWnd, HANDLE hPrinter )
1267 {
1268     PRINTER_INFO_2W *pi2 = NULL;
1269     DWORD needed = 0;
1270     LONG Ret, Result = 0;
1271     BOOL res;
1272     DEVICEPROPERTYHEADER devprophdr;
1273 
1274     FIXME("PrinterProperties(%p, %p)\n", hWnd, hPrinter);
1275 
1276     devprophdr.cbSize         = sizeof(DEVICEPROPERTYHEADER);
1277     devprophdr.Flags          = DPS_NOPERMISSION;
1278     devprophdr.hPrinter       = hPrinter;
1279     devprophdr.pszPrinterName = NULL;
1280 
1281     res = GetPrinterW( hPrinter, 2, NULL, 0, &needed);
1282     if (!res && (GetLastError() == ERROR_INSUFFICIENT_BUFFER))
1283     {
1284         pi2 = HeapAlloc(hProcessHeap, 0, needed);
1285         if (!pi2)
1286         {
1287             ERR("Failed to allocate PRINTER_INFO_2W of %u bytes\n", needed);
1288             return FALSE;
1289         }
1290         res = GetPrinterW(hPrinter, 2, (LPBYTE)pi2, needed, &needed);
1291     }
1292 
1293     //
1294     // Above can fail, still process w/o printer name.
1295     //
1296     if ( res ) devprophdr.pszPrinterName = pi2->pPrinterName;
1297 
1298     needed = 1;
1299 
1300     if ( ( SetPrinterDataW( hPrinter, L"PrinterPropertiesPermission", REG_DWORD, (LPBYTE)&needed, sizeof(DWORD) ) == ERROR_SUCCESS ) )
1301     {
1302         devprophdr.Flags &= ~DPS_NOPERMISSION;
1303     }
1304 
1305     Ret = CallCommonPropertySheetUI( hWnd, (PFNPROPSHEETUI)DevicePropertySheets, (LPARAM)&devprophdr, (LPDWORD)&Result );
1306 
1307     res = (Ret >= 0);
1308 
1309     if (!res)
1310     {
1311         FIXME("PrinterProperties fail ICPSUI\n");
1312     }
1313 
1314     if (pi2) HeapFree(hProcessHeap, 0, pi2);
1315 
1316     return res;
1317 }
1318 
1319 BOOL WINAPI
1320 EndDocPrinter(HANDLE hPrinter)
1321 {
1322     DWORD dwErrorCode;
1323     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
1324 
1325     TRACE("EndDocPrinter(%p)\n", hPrinter);
1326 
1327     // Sanity checks.
1328     if (!pHandle)
1329     {
1330         dwErrorCode = ERROR_INVALID_HANDLE;
1331         goto Cleanup;
1332     }
1333 
1334     if (pHandle->hSPLFile != INVALID_HANDLE_VALUE)
1335     {
1336         // For spooled jobs, the document is finished by calling _RpcScheduleJob.
1337         RpcTryExcept
1338         {
1339             dwErrorCode = _RpcScheduleJob(pHandle->hPrinter, pHandle->dwJobID);
1340         }
1341         RpcExcept(EXCEPTION_EXECUTE_HANDLER)
1342         {
1343             dwErrorCode = RpcExceptionCode();
1344             ERR("_RpcScheduleJob failed with exception code %lu!\n", dwErrorCode);
1345         }
1346         RpcEndExcept;
1347 
1348         // Close the spool file handle.
1349         CloseHandle(pHandle->hSPLFile);
1350     }
1351     else
1352     {
1353         // In all other cases, just call _RpcEndDocPrinter.
1354         RpcTryExcept
1355         {
1356             dwErrorCode = _RpcEndDocPrinter(pHandle->hPrinter);
1357         }
1358         RpcExcept(EXCEPTION_EXECUTE_HANDLER)
1359         {
1360             dwErrorCode = RpcExceptionCode();
1361             ERR("_RpcEndDocPrinter failed with exception code %lu!\n", dwErrorCode);
1362         }
1363         RpcEndExcept;
1364     }
1365 
1366     // A new document can now be started again.
1367     pHandle->bTrayIcon = pHandle->bJob = pHandle->bStartedDoc = FALSE;
1368 
1369 Cleanup:
1370     SetLastError(dwErrorCode);
1371     return (dwErrorCode == ERROR_SUCCESS);
1372 }
1373 
1374 BOOL WINAPI
1375 EndPagePrinter(HANDLE hPrinter)
1376 {
1377     DWORD dwErrorCode;
1378     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
1379 
1380     TRACE("EndPagePrinter(%p)\n", hPrinter);
1381 
1382     // Sanity checks.
1383     if (!pHandle)
1384     {
1385         dwErrorCode = ERROR_INVALID_HANDLE;
1386         goto Cleanup;
1387     }
1388 
1389     if (pHandle->hSPLFile != INVALID_HANDLE_VALUE)
1390     {
1391         // For spooled jobs, we don't need to do anything.
1392         dwErrorCode = ERROR_SUCCESS;
1393     }
1394     else
1395     {
1396         // In all other cases, just call _RpcEndPagePrinter.
1397         RpcTryExcept
1398         {
1399             dwErrorCode = _RpcEndPagePrinter(pHandle->hPrinter);
1400         }
1401         RpcExcept(EXCEPTION_EXECUTE_HANDLER)
1402         {
1403             dwErrorCode = RpcExceptionCode();
1404             ERR("_RpcEndPagePrinter failed with exception code %lu!\n", dwErrorCode);
1405         }
1406         RpcEndExcept;
1407     }
1408 
1409 Cleanup:
1410     SetLastError(dwErrorCode);
1411     return (dwErrorCode == ERROR_SUCCESS);
1412 }
1413 
1414 BOOL WINAPI
1415 EnumPrintersA(DWORD Flags, PSTR Name, DWORD Level, PBYTE pPrinterEnum, DWORD cbBuf, PDWORD pcbNeeded, PDWORD pcReturned)
1416 {
1417     DWORD dwErrorCode;
1418     DWORD cch;
1419     PWSTR pwszName = NULL;
1420     PSTR pszPrinterName = NULL;
1421     PSTR pszServerName = NULL;
1422     PSTR pszDescription = NULL;
1423     PSTR pszName = NULL;
1424     PSTR pszComment = NULL;
1425     PSTR pszShareName = NULL;
1426     PSTR pszPortName = NULL;
1427     PSTR pszDriverName = NULL;
1428     PSTR pszLocation = NULL;
1429     PSTR pszSepFile = NULL;
1430     PSTR pszPrintProcessor = NULL;
1431     PSTR pszDatatype = NULL;
1432     PSTR pszParameters = NULL;
1433     DWORD i;
1434     PPRINTER_INFO_1W ppi1w = NULL;
1435     PPRINTER_INFO_1A ppi1a = NULL;
1436     PPRINTER_INFO_2W ppi2w = NULL;
1437     PPRINTER_INFO_2A ppi2a = NULL;
1438     PPRINTER_INFO_4W ppi4w = NULL;
1439     PPRINTER_INFO_4A ppi4a = NULL;
1440     PPRINTER_INFO_5W ppi5w = NULL;
1441     PPRINTER_INFO_5A ppi5a = NULL;
1442 
1443     TRACE("EnumPrintersA(%lu, %s, %lu, %p, %lu, %p, %p)\n", Flags, Name, Level, pPrinterEnum, cbBuf, pcbNeeded, pcReturned);
1444 
1445     // Check for invalid levels here for early error return. MSDN says that only 1, 2, 4, and 5 are allowable.
1446     if (Level !=  1 && Level != 2 && Level != 4 && Level != 5)
1447     {
1448         dwErrorCode = ERROR_INVALID_LEVEL;
1449         ERR("Invalid Level!\n");
1450         goto Cleanup;
1451     }
1452 
1453     if (Name)
1454     {
1455         // Convert pName to a Unicode string pwszName.
1456         cch = strlen(Name);
1457 
1458         pwszName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(WCHAR));
1459         if (!pwszName)
1460         {
1461             dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1462             ERR("HeapAlloc failed!\n");
1463             goto Cleanup;
1464         }
1465 
1466         MultiByteToWideChar(CP_ACP, 0, Name, -1, pwszName, cch + 1);
1467     }
1468 
1469     /* Ref: https://stackoverflow.com/questions/41147180/why-enumprintersa-and-enumprintersw-request-the-same-amount-of-memory */
1470     if (!EnumPrintersW(Flags, pwszName, Level, pPrinterEnum, cbBuf, pcbNeeded, pcReturned))
1471     {
1472         dwErrorCode = GetLastError();
1473         goto Cleanup;
1474     }
1475 
1476     /* We are mapping multiple different pointers to the same pPrinterEnum pointer here so that */
1477     /* we can do in-place conversion. We read the Unicode response from the EnumPrintersW and */
1478     /* then we write back the ANSI conversion into the same buffer for our EnumPrintersA output */
1479 
1480     /* mapping to pPrinterEnum for Unicode (w) characters for Levels 1, 2, 4, and 5 */
1481     ppi1w = (PPRINTER_INFO_1W)pPrinterEnum;
1482     ppi2w = (PPRINTER_INFO_2W)pPrinterEnum;
1483     ppi4w = (PPRINTER_INFO_4W)pPrinterEnum;
1484     ppi5w = (PPRINTER_INFO_5W)pPrinterEnum;
1485     /* mapping to pPrinterEnum for ANSI (a) characters for Levels 1, 2, 4, and 5 */
1486     ppi1a = (PPRINTER_INFO_1A)pPrinterEnum;
1487     ppi2a = (PPRINTER_INFO_2A)pPrinterEnum;
1488     ppi4a = (PPRINTER_INFO_4A)pPrinterEnum;
1489     ppi5a = (PPRINTER_INFO_5A)pPrinterEnum;
1490 
1491     for (i = 0; i < *pcReturned; i++)
1492     {
1493         switch (Level)
1494         {
1495             case 1:
1496             {
1497                 if (ppi1w[i].pDescription)
1498                 {
1499                     // Convert Unicode pDescription to a ANSI string pszDescription.
1500                     cch = wcslen(ppi1w[i].pDescription);
1501 
1502                     pszDescription = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1503                     if (!pszDescription)
1504                     {
1505                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1506                         ERR("HeapAlloc failed!\n");
1507                         goto Cleanup;
1508                     }
1509 
1510                     WideCharToMultiByte(CP_ACP, 0, ppi1w[i].pDescription, -1, pszDescription, cch + 1, NULL, NULL);
1511                     StringCchCopyA(ppi1a[i].pDescription, cch + 1, pszDescription);
1512 
1513                     HeapFree(hProcessHeap, 0, pszDescription);
1514                 }
1515 
1516                 if (ppi1w[i].pName)
1517                 {
1518                     // Convert Unicode pName to a ANSI string pszName.
1519                     cch = wcslen(ppi1w[i].pName);
1520 
1521                     pszName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1522                     if (!pszName)
1523                     {
1524                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1525                         ERR("HeapAlloc failed!\n");
1526                         goto Cleanup;
1527                     }
1528 
1529                     WideCharToMultiByte(CP_ACP, 0, ppi1w[i].pName, -1, pszName, cch + 1, NULL, NULL);
1530                     StringCchCopyA(ppi1a[i].pName, cch + 1, pszName);
1531 
1532                     HeapFree(hProcessHeap, 0, pszName);
1533                 }
1534 
1535                 if (ppi1w[i].pComment)
1536                 {
1537                     // Convert Unicode pComment to a ANSI string pszComment.
1538                     cch = wcslen(ppi1w[i].pComment);
1539 
1540                     pszComment = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1541                     if (!pszComment)
1542                     {
1543                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1544                         ERR("HeapAlloc failed!\n");
1545                         goto Cleanup;
1546                     }
1547 
1548                     WideCharToMultiByte(CP_ACP, 0, ppi1w[i].pComment, -1, pszComment, cch + 1, NULL, NULL);
1549                     StringCchCopyA(ppi1a[i].pComment, cch + 1, pszComment);
1550 
1551                     HeapFree(hProcessHeap, 0, pszComment);
1552                 }
1553                 break;
1554             }
1555 
1556 
1557             case 2:
1558             {
1559                 if (ppi2w[i].pServerName)
1560                 {
1561                     // Convert Unicode pServerName to a ANSI string pszServerName.
1562                     cch = wcslen(ppi2w[i].pServerName);
1563 
1564                     pszServerName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1565                     if (!pszServerName)
1566                     {
1567                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1568                         ERR("HeapAlloc failed!\n");
1569                         goto Cleanup;
1570                     }
1571 
1572                     WideCharToMultiByte(CP_ACP, 0, ppi2w[i].pServerName, -1, pszServerName, cch + 1, NULL, NULL);
1573                     StringCchCopyA(ppi2a[i].pServerName, cch + 1, pszServerName);
1574 
1575                     HeapFree(hProcessHeap, 0, pszServerName);
1576                 }
1577 
1578                 if (ppi2w[i].pPrinterName)
1579                 {
1580                     // Convert Unicode pPrinterName to a ANSI string pszPrinterName.
1581                     cch = wcslen(ppi2w[i].pPrinterName);
1582 
1583                     pszPrinterName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1584                     if (!pszPrinterName)
1585                     {
1586                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1587                         ERR("HeapAlloc failed!\n");
1588                         goto Cleanup;
1589                     }
1590 
1591                     WideCharToMultiByte(CP_ACP, 0, ppi2w[i].pPrinterName, -1, pszPrinterName, cch + 1, NULL, NULL);
1592                     StringCchCopyA(ppi2a[i].pPrinterName, cch + 1, pszPrinterName);
1593 
1594                     HeapFree(hProcessHeap, 0, pszPrinterName);
1595                 }
1596 
1597                 if (ppi2w[i].pShareName)
1598                 {
1599                     // Convert Unicode pShareName to a ANSI string pszShareName.
1600                     cch = wcslen(ppi2w[i].pShareName);
1601 
1602                     pszShareName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1603                     if (!pszShareName)
1604                     {
1605                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1606                         ERR("HeapAlloc failed!\n");
1607                         goto Cleanup;
1608                     }
1609 
1610                     WideCharToMultiByte(CP_ACP, 0, ppi2w[i].pShareName, -1, pszShareName, cch + 1, NULL, NULL);
1611                     StringCchCopyA(ppi2a[i].pShareName, cch + 1, pszShareName);
1612 
1613                     HeapFree(hProcessHeap, 0, pszShareName);
1614                 }
1615 
1616                 if (ppi2w[i].pPortName)
1617                 {
1618                     // Convert Unicode pPortName to a ANSI string pszPortName.
1619                     cch = wcslen(ppi2w[i].pPortName);
1620 
1621                     pszPortName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1622                     if (!pszPortName)
1623                     {
1624                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1625                         ERR("HeapAlloc failed!\n");
1626                         goto Cleanup;
1627                     }
1628 
1629                     WideCharToMultiByte(CP_ACP, 0, ppi2w[i].pPortName, -1, pszPortName, cch + 1, NULL, NULL);
1630                     StringCchCopyA(ppi2a[i].pPortName, cch + 1, pszPortName);
1631 
1632                     HeapFree(hProcessHeap, 0, pszPortName);
1633                 }
1634 
1635                 if (ppi2w[i].pDriverName)
1636                 {
1637                     // Convert Unicode pDriverName to a ANSI string pszDriverName.
1638                     cch = wcslen(ppi2w[i].pDriverName);
1639 
1640                     pszDriverName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1641                     if (!pszDriverName)
1642                     {
1643                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1644                         ERR("HeapAlloc failed!\n");
1645                         goto Cleanup;
1646                     }
1647 
1648                     WideCharToMultiByte(CP_ACP, 0, ppi2w[i].pDriverName, -1, pszDriverName, cch + 1, NULL, NULL);
1649                     StringCchCopyA(ppi2a[i].pDriverName, cch + 1, pszDriverName);
1650 
1651                     HeapFree(hProcessHeap, 0, pszDriverName);
1652                 }
1653 
1654                 if (ppi2w[i].pComment)
1655                 {
1656                     // Convert Unicode pComment to a ANSI string pszComment.
1657                     cch = wcslen(ppi2w[i].pComment);
1658 
1659                     pszComment = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1660                     if (!pszComment)
1661                     {
1662                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1663                         ERR("HeapAlloc failed!\n");
1664                         goto Cleanup;
1665                     }
1666 
1667                     WideCharToMultiByte(CP_ACP, 0, ppi2w[i].pComment, -1, pszComment, cch + 1, NULL, NULL);
1668                     StringCchCopyA(ppi2a[i].pComment, cch + 1, pszComment);
1669 
1670                     HeapFree(hProcessHeap, 0, pszComment);
1671                 }
1672 
1673                 if (ppi2w[i].pLocation)
1674                 {
1675                     // Convert Unicode pLocation to a ANSI string pszLocation.
1676                     cch = wcslen(ppi2w[i].pLocation);
1677 
1678                     pszLocation = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1679                     if (!pszLocation)
1680                     {
1681                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1682                         ERR("HeapAlloc failed!\n");
1683                         goto Cleanup;
1684                     }
1685 
1686                     WideCharToMultiByte(CP_ACP, 0, ppi2w[i].pLocation, -1, pszLocation, cch + 1, NULL, NULL);
1687                     StringCchCopyA(ppi2a[i].pLocation, cch + 1, pszLocation);
1688 
1689                     HeapFree(hProcessHeap, 0, pszLocation);
1690                 }
1691 
1692 
1693                 if (ppi2w[i].pSepFile)
1694                 {
1695                     // Convert Unicode pSepFile to a ANSI string pszSepFile.
1696                     cch = wcslen(ppi2w[i].pSepFile);
1697 
1698                     pszSepFile = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1699                     if (!pszSepFile)
1700                     {
1701                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1702                         ERR("HeapAlloc failed!\n");
1703                         goto Cleanup;
1704                     }
1705 
1706                     WideCharToMultiByte(CP_ACP, 0, ppi2w[i].pSepFile, -1, pszSepFile, cch + 1, NULL, NULL);
1707                     StringCchCopyA(ppi2a[i].pSepFile, cch + 1, pszSepFile);
1708 
1709                     HeapFree(hProcessHeap, 0, pszSepFile);
1710                 }
1711 
1712                 if (ppi2w[i].pPrintProcessor)
1713                 {
1714                     // Convert Unicode pPrintProcessor to a ANSI string pszPrintProcessor.
1715                     cch = wcslen(ppi2w[i].pPrintProcessor);
1716 
1717                     pszPrintProcessor = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1718                     if (!pszPrintProcessor)
1719                     {
1720                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1721                         ERR("HeapAlloc failed!\n");
1722                         goto Cleanup;
1723                     }
1724 
1725                     WideCharToMultiByte(CP_ACP, 0, ppi2w[i].pPrintProcessor, -1, pszPrintProcessor, cch + 1, NULL, NULL);
1726                     StringCchCopyA(ppi2a[i].pPrintProcessor, cch + 1, pszPrintProcessor);
1727 
1728                     HeapFree(hProcessHeap, 0, pszPrintProcessor);
1729                 }
1730 
1731 
1732                 if (ppi2w[i].pDatatype)
1733                 {
1734                     // Convert Unicode pDatatype to a ANSI string pszDatatype.
1735                     cch = wcslen(ppi2w[i].pDatatype);
1736 
1737                     pszDatatype = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1738                     if (!pszDatatype)
1739                     {
1740                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1741                         ERR("HeapAlloc failed!\n");
1742                         goto Cleanup;
1743                     }
1744 
1745                     WideCharToMultiByte(CP_ACP, 0, ppi2w[i].pDatatype, -1, pszDatatype, cch + 1, NULL, NULL);
1746                     StringCchCopyA(ppi2a[i].pDatatype, cch + 1, pszDatatype);
1747 
1748                     HeapFree(hProcessHeap, 0, pszDatatype);
1749                 }
1750 
1751                 if (ppi2w[i].pParameters)
1752                 {
1753                     // Convert Unicode pParameters to a ANSI string pszParameters.
1754                     cch = wcslen(ppi2w[i].pParameters);
1755 
1756                     pszParameters = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1757                     if (!pszParameters)
1758                     {
1759                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1760                         ERR("HeapAlloc failed!\n");
1761                         goto Cleanup;
1762                     }
1763 
1764                     WideCharToMultiByte(CP_ACP, 0, ppi2w[i].pParameters, -1, pszParameters, cch + 1, NULL, NULL);
1765                     StringCchCopyA(ppi2a[i].pParameters, cch + 1, pszParameters);
1766 
1767                     HeapFree(hProcessHeap, 0, pszParameters);
1768                 }
1769                 if ( ppi2w[i].pDevMode )
1770                 {
1771                     RosConvertUnicodeDevModeToAnsiDevmode( ppi2w[i].pDevMode, ppi2a[i].pDevMode );
1772                 }
1773                 break;
1774             }
1775 
1776             case 4:
1777             {
1778                 if (ppi4w[i].pPrinterName)
1779                 {
1780                     // Convert Unicode pPrinterName to a ANSI string pszPrinterName.
1781                     cch = wcslen(ppi4w[i].pPrinterName);
1782 
1783                     pszPrinterName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1784                     if (!pszPrinterName)
1785                     {
1786                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1787                         ERR("HeapAlloc failed!\n");
1788                         goto Cleanup;
1789                     }
1790 
1791                     WideCharToMultiByte(CP_ACP, 0, ppi4w[i].pPrinterName, -1, pszPrinterName, cch + 1, NULL, NULL);
1792                     StringCchCopyA(ppi4a[i].pPrinterName, cch + 1, pszPrinterName);
1793 
1794                     HeapFree(hProcessHeap, 0, pszPrinterName);
1795                 }
1796 
1797                 if (ppi4w[i].pServerName)
1798                 {
1799                     // Convert Unicode pServerName to a ANSI string pszServerName.
1800                     cch = wcslen(ppi4w[i].pServerName);
1801 
1802                     pszServerName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1803                     if (!pszServerName)
1804                     {
1805                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1806                         ERR("HeapAlloc failed!\n");
1807                         goto Cleanup;
1808                     }
1809 
1810                     WideCharToMultiByte(CP_ACP, 0, ppi4w[i].pServerName, -1, pszServerName, cch + 1, NULL, NULL);
1811                     StringCchCopyA(ppi4a[i].pServerName, cch + 1, pszServerName);
1812 
1813                     HeapFree(hProcessHeap, 0, pszServerName);
1814                 }
1815                 break;
1816             }
1817 
1818             case 5:
1819             {
1820                 if (ppi5w[i].pPrinterName)
1821                 {
1822                     // Convert Unicode pPrinterName to a ANSI string pszPrinterName.
1823                     cch = wcslen(ppi5w[i].pPrinterName);
1824 
1825                     pszPrinterName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1826                     if (!pszPrinterName)
1827                     {
1828                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1829                         ERR("HeapAlloc failed!\n");
1830                         goto Cleanup;
1831                     }
1832 
1833                     WideCharToMultiByte(CP_ACP, 0, ppi5w[i].pPrinterName, -1, pszPrinterName, cch + 1, NULL, NULL);
1834                     StringCchCopyA(ppi5a[i].pPrinterName, cch + 1, pszPrinterName);
1835 
1836                     HeapFree(hProcessHeap, 0, pszPrinterName);
1837                 }
1838 
1839                 if (ppi5w[i].pPortName)
1840                 {
1841                     // Convert Unicode pPortName to a ANSI string pszPortName.
1842                     cch = wcslen(ppi5w[i].pPortName);
1843 
1844                     pszPortName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
1845                     if (!pszPortName)
1846                     {
1847                         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1848                         ERR("HeapAlloc failed!\n");
1849                         goto Cleanup;
1850                     }
1851 
1852                     WideCharToMultiByte(CP_ACP, 0, ppi5w[i].pPortName, -1, pszPortName, cch + 1, NULL, NULL);
1853                     StringCchCopyA(ppi5a[i].pPortName, cch + 1, pszPortName);
1854 
1855                     HeapFree(hProcessHeap, 0, pszPortName);
1856                 }
1857                 break;
1858             }
1859 
1860         }   // switch
1861     }       // for
1862 
1863     dwErrorCode = ERROR_SUCCESS;
1864 
1865 Cleanup:
1866     if (pwszName)
1867     {
1868         HeapFree(hProcessHeap, 0, pwszName);
1869     }
1870 
1871     SetLastError(dwErrorCode);
1872     return (dwErrorCode == ERROR_SUCCESS);
1873 }
1874 
1875 BOOL WINAPI
1876 EnumPrintersW(DWORD Flags, PWSTR Name, DWORD Level, PBYTE pPrinterEnum, DWORD cbBuf, PDWORD pcbNeeded, PDWORD pcReturned)
1877 {
1878     DWORD dwErrorCode;
1879 
1880     TRACE("EnumPrintersW(%lu, %S, %lu, %p, %lu, %p, %p)\n", Flags, Name, Level, pPrinterEnum, cbBuf, pcbNeeded, pcReturned);
1881 
1882     // Dismiss invalid levels already at this point.
1883     if (Level == 3 || Level > 5)
1884     {
1885         dwErrorCode = ERROR_INVALID_LEVEL;
1886         goto Cleanup;
1887     }
1888 
1889     if (cbBuf && pPrinterEnum)
1890         ZeroMemory(pPrinterEnum, cbBuf);
1891 
1892     // Do the RPC call
1893     RpcTryExcept
1894     {
1895         dwErrorCode = _RpcEnumPrinters(Flags, Name, Level, pPrinterEnum, cbBuf, pcbNeeded, pcReturned);
1896     }
1897     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
1898     {
1899         dwErrorCode = RpcExceptionCode();
1900         ERR("_RpcEnumPrinters failed with exception code %lu!\n", dwErrorCode);
1901     }
1902     RpcEndExcept;
1903 
1904     if (dwErrorCode == ERROR_SUCCESS)
1905     {
1906         // Replace relative offset addresses in the output by absolute pointers.
1907         ASSERT(Level <= 9);
1908         MarshallUpStructuresArray(cbBuf, pPrinterEnum, *pcReturned, pPrinterInfoMarshalling[Level]->pInfo, pPrinterInfoMarshalling[Level]->cbStructureSize, TRUE);
1909     }
1910 
1911 Cleanup:
1912     SetLastError(dwErrorCode);
1913     return (dwErrorCode == ERROR_SUCCESS);
1914 }
1915 
1916 BOOL WINAPI
1917 FlushPrinter(HANDLE hPrinter, PVOID pBuf, DWORD cbBuf, PDWORD pcWritten, DWORD cSleep)
1918 {
1919     TRACE("FlushPrinter(%p, %p, %lu, %p, %lu)\n", hPrinter, pBuf, cbBuf, pcWritten, cSleep);
1920     UNIMPLEMENTED;
1921     return FALSE;
1922 }
1923 
1924 BOOL WINAPI
1925 GetDefaultPrinterA(LPSTR pszBuffer, LPDWORD pcchBuffer)
1926 {
1927     DWORD dwErrorCode;
1928     PWSTR pwszBuffer = NULL;
1929 
1930     TRACE("GetDefaultPrinterA(%p, %p)\n", pszBuffer, pcchBuffer);
1931 
1932     // Sanity check.
1933     if (!pcchBuffer)
1934     {
1935         dwErrorCode = ERROR_INVALID_PARAMETER;
1936         goto Cleanup;
1937     }
1938 
1939     // Check if an ANSI buffer was given and if so, allocate a Unicode buffer of the same size.
1940     if (pszBuffer && *pcchBuffer)
1941     {
1942         pwszBuffer = HeapAlloc(hProcessHeap, 0, *pcchBuffer * sizeof(WCHAR));
1943         if (!pwszBuffer)
1944         {
1945             dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
1946             ERR("HeapAlloc failed!\n");
1947             goto Cleanup;
1948         }
1949     }
1950 
1951     if (!GetDefaultPrinterW(pwszBuffer, pcchBuffer))
1952     {
1953         dwErrorCode = GetLastError();
1954         goto Cleanup;
1955     }
1956 
1957     // We successfully got a string in pwszBuffer, so convert the Unicode string to ANSI.
1958     WideCharToMultiByte(CP_ACP, 0, pwszBuffer, -1, pszBuffer, *pcchBuffer, NULL, NULL);
1959 
1960     dwErrorCode = ERROR_SUCCESS;
1961 
1962 Cleanup:
1963     if (pwszBuffer)
1964         HeapFree(hProcessHeap, 0, pwszBuffer);
1965 
1966     SetLastError(dwErrorCode);
1967     return (dwErrorCode == ERROR_SUCCESS);
1968 }
1969 
1970 BOOL WINAPI
1971 GetDefaultPrinterW(LPWSTR pszBuffer, LPDWORD pcchBuffer)
1972 {
1973     DWORD cbNeeded;
1974     DWORD cchInputBuffer;
1975     DWORD dwErrorCode;
1976     HKEY hWindowsKey = NULL;
1977     PWSTR pwszDevice = NULL;
1978     PWSTR pwszComma;
1979 
1980     TRACE("GetDefaultPrinterW(%p, %p)\n", pszBuffer, pcchBuffer);
1981 
1982     // Sanity check.
1983     if (!pcchBuffer)
1984     {
1985         dwErrorCode = ERROR_INVALID_PARAMETER;
1986         goto Cleanup;
1987     }
1988 
1989     cchInputBuffer = *pcchBuffer;
1990 
1991     // Open the registry key where the default printer for the current user is stored.
1992     dwErrorCode = (DWORD)RegOpenKeyExW(HKEY_CURRENT_USER, wszWindowsKey, 0, KEY_READ, &hWindowsKey);
1993     if (dwErrorCode != ERROR_SUCCESS)
1994     {
1995         ERR("RegOpenKeyExW failed with status %lu!\n", dwErrorCode);
1996         goto Cleanup;
1997     }
1998 
1999     // Determine the size of the required buffer.
2000     dwErrorCode = (DWORD)RegQueryValueExW(hWindowsKey, wszDeviceValue, NULL, NULL, NULL, &cbNeeded);
2001     if (dwErrorCode != ERROR_SUCCESS)
2002     {
2003         ERR("RegQueryValueExW failed with status %lu!\n", dwErrorCode);
2004         goto Cleanup;
2005     }
2006 
2007     // Allocate it.
2008     pwszDevice = HeapAlloc(hProcessHeap, 0, cbNeeded);
2009     if (!pwszDevice)
2010     {
2011         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2012         ERR("HeapAlloc failed!\n");
2013         goto Cleanup;
2014     }
2015 
2016     // Now get the actual value.
2017     dwErrorCode = RegQueryValueExW(hWindowsKey, wszDeviceValue, NULL, NULL, (PBYTE)pwszDevice, &cbNeeded);
2018     if (dwErrorCode != ERROR_SUCCESS)
2019     {
2020         ERR("RegQueryValueExW failed with status %lu!\n", dwErrorCode);
2021         goto Cleanup;
2022     }
2023 
2024     // We get a string "<Printer Name>,winspool,<Port>:".
2025     // Extract the printer name from it.
2026     pwszComma = wcschr(pwszDevice, L',');
2027     if (!pwszComma)
2028     {
2029         ERR("Found no or invalid default printer: %S!\n", pwszDevice);
2030         dwErrorCode = ERROR_INVALID_NAME;
2031         goto Cleanup;
2032     }
2033 
2034     // Store the length of the Printer Name (including the terminating NUL character!) in *pcchBuffer.
2035     *pcchBuffer = pwszComma - pwszDevice + 1;
2036 
2037     // Check if the supplied buffer is large enough.
2038     if ( !pszBuffer || cchInputBuffer < *pcchBuffer)
2039     {
2040         dwErrorCode = ERROR_INSUFFICIENT_BUFFER;
2041         goto Cleanup;
2042     }
2043 
2044     // Copy the default printer.
2045     *pwszComma = 0;
2046     CopyMemory(pszBuffer, pwszDevice, *pcchBuffer * sizeof(WCHAR));
2047 
2048     dwErrorCode = ERROR_SUCCESS;
2049 
2050 Cleanup:
2051     if (hWindowsKey)
2052         RegCloseKey(hWindowsKey);
2053 
2054     if (pwszDevice)
2055         HeapFree(hProcessHeap, 0, pwszDevice);
2056 
2057     SetLastError(dwErrorCode);
2058     return (dwErrorCode == ERROR_SUCCESS);
2059 }
2060 
2061 BOOL WINAPI
2062 GetPrinterA(HANDLE hPrinter, DWORD Level, LPBYTE pPrinter, DWORD cbBuf, LPDWORD pcbNeeded)
2063 {
2064     DWORD dwErrorCode;
2065     PPRINTER_INFO_1A ppi1a = (PPRINTER_INFO_1A)pPrinter;
2066     PPRINTER_INFO_1W ppi1w = (PPRINTER_INFO_1W)pPrinter;
2067     PPRINTER_INFO_2A ppi2a = (PPRINTER_INFO_2A)pPrinter;
2068     PPRINTER_INFO_2W ppi2w = (PPRINTER_INFO_2W)pPrinter;
2069     PPRINTER_INFO_4A ppi4a = (PPRINTER_INFO_4A)pPrinter;
2070     PPRINTER_INFO_4W ppi4w = (PPRINTER_INFO_4W)pPrinter;
2071     PPRINTER_INFO_5A ppi5a = (PPRINTER_INFO_5A)pPrinter;
2072     PPRINTER_INFO_5W ppi5w = (PPRINTER_INFO_5W)pPrinter;
2073     PPRINTER_INFO_7A ppi7a = (PPRINTER_INFO_7A)pPrinter;
2074     PPRINTER_INFO_7W ppi7w = (PPRINTER_INFO_7W)pPrinter;
2075     PPRINTER_INFO_9A ppi9a = (PPRINTER_INFO_9A)pPrinter;
2076     PPRINTER_INFO_9W ppi9w = (PPRINTER_INFO_9W)pPrinter;
2077     DWORD cch;
2078 
2079     TRACE("GetPrinterA(%p, %lu, %p, %lu, %p)\n", hPrinter, Level, pPrinter, cbBuf, pcbNeeded);
2080 
2081     // Check for invalid levels here for early error return. Should be 1-9.
2082     if (Level <  1 || Level > 9)
2083     {
2084         dwErrorCode = ERROR_INVALID_LEVEL;
2085         ERR("Invalid Level!\n");
2086         goto Cleanup;
2087     }
2088 
2089     if (!GetPrinterW(hPrinter, Level, pPrinter, cbBuf, pcbNeeded))
2090     {
2091         dwErrorCode = GetLastError();
2092         goto Cleanup;
2093     }
2094 
2095     switch (Level)
2096     {
2097         case 1:
2098         {
2099             if (ppi1w->pDescription)
2100             {
2101                 PSTR pszDescription;
2102 
2103                 // Convert Unicode pDescription to a ANSI string pszDescription.
2104                 cch = wcslen(ppi1w->pDescription);
2105 
2106                 pszDescription = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2107                 if (!pszDescription)
2108                 {
2109                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2110                     ERR("HeapAlloc failed!\n");
2111                     goto Cleanup;
2112                 }
2113 
2114                 WideCharToMultiByte(CP_ACP, 0, ppi1w->pDescription, -1, pszDescription, cch + 1, NULL, NULL);
2115                 StringCchCopyA(ppi1a->pDescription, cch + 1, pszDescription);
2116 
2117                 HeapFree(hProcessHeap, 0, pszDescription);
2118             }
2119 
2120             if (ppi1w->pName)
2121             {
2122                 PSTR pszName;
2123 
2124                 // Convert Unicode pName to a ANSI string pszName.
2125                 cch = wcslen(ppi1w->pName);
2126 
2127                 pszName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2128                 if (!pszName)
2129                 {
2130                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2131                     ERR("HeapAlloc failed!\n");
2132                     goto Cleanup;
2133                 }
2134 
2135                 WideCharToMultiByte(CP_ACP, 0, ppi1w->pName, -1, pszName, cch + 1, NULL, NULL);
2136                 StringCchCopyA(ppi1a->pName, cch + 1, pszName);
2137 
2138                 HeapFree(hProcessHeap, 0, pszName);
2139             }
2140 
2141             if (ppi1w->pComment)
2142             {
2143                 PSTR pszComment;
2144 
2145                 // Convert Unicode pComment to a ANSI string pszComment.
2146                 cch = wcslen(ppi1w->pComment);
2147 
2148                 pszComment = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2149                 if (!pszComment)
2150                 {
2151                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2152                     ERR("HeapAlloc failed!\n");
2153                     goto Cleanup;
2154                 }
2155 
2156                 WideCharToMultiByte(CP_ACP, 0, ppi1w->pComment, -1, pszComment, cch + 1, NULL, NULL);
2157                 StringCchCopyA(ppi1a->pComment, cch + 1, pszComment);
2158 
2159                 HeapFree(hProcessHeap, 0, pszComment);
2160             }
2161             break;
2162         }
2163 
2164         case 2:
2165         {
2166             if (ppi2w->pServerName)
2167             {
2168                 PSTR pszServerName;
2169 
2170                 // Convert Unicode pServerName to a ANSI string pszServerName.
2171                 cch = wcslen(ppi2w->pServerName);
2172 
2173                 pszServerName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2174                 if (!pszServerName)
2175                 {
2176                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2177                     ERR("HeapAlloc failed!\n");
2178                     goto Cleanup;
2179                 }
2180 
2181                 WideCharToMultiByte(CP_ACP, 0, ppi2w->pServerName, -1, pszServerName, cch + 1, NULL, NULL);
2182                 StringCchCopyA(ppi2a->pServerName, cch + 1, pszServerName);
2183 
2184                 HeapFree(hProcessHeap, 0, pszServerName);
2185             }
2186 
2187             if (ppi2w->pPrinterName)
2188             {
2189                 PSTR pszPrinterName;
2190 
2191                 // Convert Unicode pPrinterName to a ANSI string pszPrinterName.
2192                 cch = wcslen(ppi2w->pPrinterName);
2193 
2194                 pszPrinterName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2195                 if (!pszPrinterName)
2196                 {
2197                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2198                     ERR("HeapAlloc failed!\n");
2199                     goto Cleanup;
2200                 }
2201 
2202                 WideCharToMultiByte(CP_ACP, 0, ppi2w->pPrinterName, -1, pszPrinterName, cch + 1, NULL, NULL);
2203                 StringCchCopyA(ppi2a->pPrinterName, cch + 1, pszPrinterName);
2204 
2205                 HeapFree(hProcessHeap, 0, pszPrinterName);
2206             }
2207 
2208             if (ppi2w->pShareName)
2209             {
2210                 PSTR pszShareName;
2211 
2212                 // Convert Unicode pShareName to a ANSI string pszShareName.
2213                 cch = wcslen(ppi2w->pShareName);
2214 
2215                 pszShareName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2216                 if (!pszShareName)
2217                 {
2218                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2219                     ERR("HeapAlloc failed!\n");
2220                     goto Cleanup;
2221                 }
2222 
2223                 WideCharToMultiByte(CP_ACP, 0, ppi2w->pShareName, -1, pszShareName, cch + 1, NULL, NULL);
2224                 StringCchCopyA(ppi2a->pShareName, cch + 1, pszShareName);
2225 
2226                 HeapFree(hProcessHeap, 0, pszShareName);
2227             }
2228 
2229             if (ppi2w->pPortName)
2230             {
2231                 PSTR pszPortName;
2232 
2233                 // Convert Unicode pPortName to a ANSI string pszPortName.
2234                 cch = wcslen(ppi2w->pPortName);
2235 
2236                 pszPortName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2237                 if (!pszPortName)
2238                 {
2239                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2240                     ERR("HeapAlloc failed!\n");
2241                     goto Cleanup;
2242                 }
2243 
2244                 WideCharToMultiByte(CP_ACP, 0, ppi2w->pPortName, -1, pszPortName, cch + 1, NULL, NULL);
2245                 StringCchCopyA(ppi2a->pPortName, cch + 1, pszPortName);
2246 
2247                 HeapFree(hProcessHeap, 0, pszPortName);
2248             }
2249 
2250             if (ppi2w->pDriverName)
2251             {
2252                 PSTR pszDriverName;
2253 
2254                 // Convert Unicode pDriverName to a ANSI string pszDriverName.
2255                 cch = wcslen(ppi2w->pDriverName);
2256 
2257                 pszDriverName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2258                 if (!pszDriverName)
2259                 {
2260                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2261                     ERR("HeapAlloc failed!\n");
2262                     goto Cleanup;
2263                 }
2264 
2265                 WideCharToMultiByte(CP_ACP, 0, ppi2w->pDriverName, -1, pszDriverName, cch + 1, NULL, NULL);
2266                 StringCchCopyA(ppi2a->pDriverName, cch + 1, pszDriverName);
2267 
2268                 HeapFree(hProcessHeap, 0, pszDriverName);
2269             }
2270 
2271             if (ppi2w->pComment)
2272             {
2273                 PSTR pszComment;
2274 
2275                 // Convert Unicode pComment to a ANSI string pszComment.
2276                 cch = wcslen(ppi2w->pComment);
2277 
2278                 pszComment = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2279                 if (!pszComment)
2280                 {
2281                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2282                     ERR("HeapAlloc failed!\n");
2283                     goto Cleanup;
2284                 }
2285 
2286                 WideCharToMultiByte(CP_ACP, 0, ppi2w->pComment, -1, pszComment, cch + 1, NULL, NULL);
2287                 StringCchCopyA(ppi2a->pComment, cch + 1, pszComment);
2288 
2289                 HeapFree(hProcessHeap, 0, pszComment);
2290             }
2291 
2292             if (ppi2w->pLocation)
2293             {
2294                 PSTR pszLocation;
2295 
2296                 // Convert Unicode pLocation to a ANSI string pszLocation.
2297                 cch = wcslen(ppi2w->pLocation);
2298 
2299                 pszLocation = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2300                 if (!pszLocation)
2301                 {
2302                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2303                     ERR("HeapAlloc failed!\n");
2304                     goto Cleanup;
2305                 }
2306 
2307                 WideCharToMultiByte(CP_ACP, 0, ppi2w->pLocation, -1, pszLocation, cch + 1, NULL, NULL);
2308                 StringCchCopyA(ppi2a->pLocation, cch + 1, pszLocation);
2309 
2310                 HeapFree(hProcessHeap, 0, pszLocation);
2311             }
2312 
2313             if (ppi2w->pSepFile)
2314             {
2315                 PSTR pszSepFile;
2316 
2317                 // Convert Unicode pSepFile to a ANSI string pszSepFile.
2318                 cch = wcslen(ppi2w->pSepFile);
2319 
2320                 pszSepFile = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2321                 if (!pszSepFile)
2322                 {
2323                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2324                     ERR("HeapAlloc failed!\n");
2325                     goto Cleanup;
2326                 }
2327 
2328                 WideCharToMultiByte(CP_ACP, 0, ppi2w->pSepFile, -1, pszSepFile, cch + 1, NULL, NULL);
2329                 StringCchCopyA(ppi2a->pSepFile, cch + 1, pszSepFile);
2330 
2331                 HeapFree(hProcessHeap, 0, pszSepFile);
2332             }
2333 
2334             if (ppi2w->pPrintProcessor)
2335             {
2336                 PSTR pszPrintProcessor;
2337 
2338                 // Convert Unicode pPrintProcessor to a ANSI string pszPrintProcessor.
2339                 cch = wcslen(ppi2w->pPrintProcessor);
2340 
2341                 pszPrintProcessor = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2342                 if (!pszPrintProcessor)
2343                 {
2344                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2345                     ERR("HeapAlloc failed!\n");
2346                     goto Cleanup;
2347                 }
2348 
2349                 WideCharToMultiByte(CP_ACP, 0, ppi2w->pPrintProcessor, -1, pszPrintProcessor, cch + 1, NULL, NULL);
2350                 StringCchCopyA(ppi2a->pPrintProcessor, cch + 1, pszPrintProcessor);
2351 
2352                 HeapFree(hProcessHeap, 0, pszPrintProcessor);
2353             }
2354 
2355             if (ppi2w->pDatatype)
2356             {
2357                 PSTR pszDatatype;
2358 
2359                 // Convert Unicode pDatatype to a ANSI string pszDatatype.
2360                 cch = wcslen(ppi2w->pDatatype);
2361 
2362                 pszDatatype = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2363                 if (!pszDatatype)
2364                 {
2365                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2366                     ERR("HeapAlloc failed!\n");
2367                     goto Cleanup;
2368                 }
2369 
2370                 WideCharToMultiByte(CP_ACP, 0, ppi2w->pDatatype, -1, pszDatatype, cch + 1, NULL, NULL);
2371                 StringCchCopyA(ppi2a->pDatatype, cch + 1, pszDatatype);
2372 
2373                 HeapFree(hProcessHeap, 0, pszDatatype);
2374             }
2375 
2376             if (ppi2w->pParameters)
2377             {
2378                 PSTR pszParameters;
2379 
2380                 // Convert Unicode pParameters to a ANSI string pszParameters.
2381                 cch = wcslen(ppi2w->pParameters);
2382 
2383                 pszParameters = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2384                 if (!pszParameters)
2385                 {
2386                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2387                     ERR("HeapAlloc failed!\n");
2388                     goto Cleanup;
2389                 }
2390 
2391                 WideCharToMultiByte(CP_ACP, 0, ppi2w->pParameters, -1, pszParameters, cch + 1, NULL, NULL);
2392                 StringCchCopyA(ppi2a->pParameters, cch + 1, pszParameters);
2393 
2394                 HeapFree(hProcessHeap, 0, pszParameters);
2395             }
2396             if ( ppi2w->pDevMode )
2397             {
2398                 RosConvertUnicodeDevModeToAnsiDevmode( ppi2w->pDevMode, ppi2a->pDevMode );
2399             }
2400             break;
2401         }
2402 
2403         case 4:
2404         {
2405             if (ppi4w->pPrinterName)
2406             {
2407                 PSTR pszPrinterName;
2408 
2409                 // Convert Unicode pPrinterName to a ANSI string pszPrinterName.
2410                 cch = wcslen(ppi4w->pPrinterName);
2411 
2412                 pszPrinterName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2413                 if (!pszPrinterName)
2414                 {
2415                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2416                     ERR("HeapAlloc failed!\n");
2417                     goto Cleanup;
2418                 }
2419 
2420                 WideCharToMultiByte(CP_ACP, 0, ppi4w->pPrinterName, -1, pszPrinterName, cch + 1, NULL, NULL);
2421                 StringCchCopyA(ppi4a->pPrinterName, cch + 1, pszPrinterName);
2422 
2423                 HeapFree(hProcessHeap, 0, pszPrinterName);
2424             }
2425 
2426             if (ppi4w->pServerName)
2427             {
2428                 PSTR pszServerName;
2429 
2430                 // Convert Unicode pServerName to a ANSI string pszServerName.
2431                 cch = wcslen(ppi4w->pServerName);
2432 
2433                 pszServerName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2434                 if (!pszServerName)
2435                 {
2436                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2437                     ERR("HeapAlloc failed!\n");
2438                     goto Cleanup;
2439                 }
2440 
2441                 WideCharToMultiByte(CP_ACP, 0, ppi4w->pServerName, -1, pszServerName, cch + 1, NULL, NULL);
2442                 StringCchCopyA(ppi4a->pServerName, cch + 1, pszServerName);
2443 
2444                 HeapFree(hProcessHeap, 0, pszServerName);
2445             }
2446             break;
2447         }
2448 
2449         case 5:
2450         {
2451             if (ppi5w->pPrinterName)
2452             {
2453                 PSTR pszPrinterName;
2454 
2455                 // Convert Unicode pPrinterName to a ANSI string pszPrinterName.
2456                 cch = wcslen(ppi5w->pPrinterName);
2457 
2458                 pszPrinterName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2459                 if (!pszPrinterName)
2460                 {
2461                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2462                     ERR("HeapAlloc failed!\n");
2463                     goto Cleanup;
2464                 }
2465 
2466                 WideCharToMultiByte(CP_ACP, 0, ppi5w->pPrinterName, -1, pszPrinterName, cch + 1, NULL, NULL);
2467                 StringCchCopyA(ppi5a->pPrinterName, cch + 1, pszPrinterName);
2468 
2469                 HeapFree(hProcessHeap, 0, pszPrinterName);
2470             }
2471 
2472             if (ppi5w->pPortName)
2473             {
2474                 PSTR pszPortName;
2475 
2476                 // Convert Unicode pPortName to a ANSI string pszPortName.
2477                 cch = wcslen(ppi5w->pPortName);
2478 
2479                 pszPortName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2480                 if (!pszPortName)
2481                 {
2482                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2483                     ERR("HeapAlloc failed!\n");
2484                     goto Cleanup;
2485                 }
2486 
2487                 WideCharToMultiByte(CP_ACP, 0, ppi5w->pPortName, -1, pszPortName, cch + 1, NULL, NULL);
2488                 StringCchCopyA(ppi5a->pPortName, cch + 1, pszPortName);
2489 
2490                 HeapFree(hProcessHeap, 0, pszPortName);
2491             }
2492             break;
2493         }
2494 
2495         case 7:
2496         {
2497             if (ppi7w->pszObjectGUID)
2498             {
2499                 PSTR pszaObjectGUID;
2500 
2501                 // Convert Unicode pszObjectGUID to a ANSI string pszaObjectGUID.
2502                 cch = wcslen(ppi7w->pszObjectGUID);
2503 
2504                 pszaObjectGUID = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(CHAR));
2505                 if (!pszaObjectGUID)
2506                 {
2507                     dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2508                     ERR("HeapAlloc failed!\n");
2509                     goto Cleanup;
2510                 }
2511 
2512                 WideCharToMultiByte(CP_ACP, 0, ppi7w->pszObjectGUID, -1, pszaObjectGUID, cch + 1, NULL, NULL);
2513                 StringCchCopyA(ppi7a->pszObjectGUID, cch + 1, pszaObjectGUID);
2514 
2515                 HeapFree(hProcessHeap, 0, pszaObjectGUID);
2516             }
2517         }
2518             break;
2519         case 8:
2520         case 9:
2521             RosConvertUnicodeDevModeToAnsiDevmode(ppi9w->pDevMode, ppi9a->pDevMode);
2522             break;
2523     }       // switch
2524 
2525     dwErrorCode = ERROR_SUCCESS;
2526 
2527 Cleanup:
2528     SetLastError(dwErrorCode);
2529     return (dwErrorCode == ERROR_SUCCESS);
2530 }
2531 
2532 BOOL WINAPI
2533 GetPrinterW(HANDLE hPrinter, DWORD Level, LPBYTE pPrinter, DWORD cbBuf, LPDWORD pcbNeeded)
2534 {
2535     DWORD dwErrorCode;
2536     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
2537 
2538     TRACE("GetPrinterW(%p, %lu, %p, %lu, %p)\n", hPrinter, Level, pPrinter, cbBuf, pcbNeeded);
2539 
2540     // Sanity checks.
2541     if (!pHandle)
2542     {
2543         dwErrorCode = ERROR_INVALID_HANDLE;
2544         goto Cleanup;
2545     }
2546 
2547     // Dismiss invalid levels already at this point.
2548     if (Level > 9)
2549     {
2550         dwErrorCode = ERROR_INVALID_LEVEL;
2551         goto Cleanup;
2552     }
2553 
2554     if (cbBuf && pPrinter)
2555         ZeroMemory(pPrinter, cbBuf);
2556 
2557     // Do the RPC call
2558     RpcTryExcept
2559     {
2560         dwErrorCode = _RpcGetPrinter(pHandle->hPrinter, Level, pPrinter, cbBuf, pcbNeeded);
2561     }
2562     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
2563     {
2564         dwErrorCode = RpcExceptionCode();
2565         ERR("_RpcGetPrinter failed with exception code %lu!\n", dwErrorCode);
2566     }
2567     RpcEndExcept;
2568 
2569     if (dwErrorCode == ERROR_SUCCESS)
2570     {
2571         // Replace relative offset addresses in the output by absolute pointers.
2572         ASSERT(Level <= 9);
2573         MarshallUpStructure(cbBuf, pPrinter, pPrinterInfoMarshalling[Level]->pInfo, pPrinterInfoMarshalling[Level]->cbStructureSize, TRUE);
2574     }
2575 
2576 Cleanup:
2577     SetLastError(dwErrorCode);
2578     return (dwErrorCode == ERROR_SUCCESS);
2579 }
2580 
2581 BOOL WINAPI
2582 OpenPrinterA(LPSTR pPrinterName, LPHANDLE phPrinter, LPPRINTER_DEFAULTSA pDefault)
2583 {
2584     BOOL bReturnValue = FALSE;
2585     DWORD cch;
2586     PWSTR pwszPrinterName = NULL;
2587     PRINTER_DEFAULTSW wDefault = { 0 };
2588 
2589     TRACE("OpenPrinterA(%s, %p, %p)\n", pPrinterName, phPrinter, pDefault);
2590 
2591     if (pPrinterName)
2592     {
2593         // Convert pPrinterName to a Unicode string pwszPrinterName
2594         cch = strlen(pPrinterName);
2595 
2596         pwszPrinterName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(WCHAR));
2597         if (!pwszPrinterName)
2598         {
2599             SetLastError(ERROR_NOT_ENOUGH_MEMORY);
2600             ERR("HeapAlloc failed!\n");
2601             goto Cleanup;
2602         }
2603 
2604         MultiByteToWideChar(CP_ACP, 0, pPrinterName, -1, pwszPrinterName, cch + 1);
2605     }
2606 
2607     if (pDefault)
2608     {
2609         wDefault.DesiredAccess = pDefault->DesiredAccess;
2610 
2611         if (pDefault->pDatatype)
2612         {
2613             // Convert pDefault->pDatatype to a Unicode string wDefault.pDatatype
2614             cch = strlen(pDefault->pDatatype);
2615 
2616             wDefault.pDatatype = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(WCHAR));
2617             if (!wDefault.pDatatype)
2618             {
2619                 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
2620                 ERR("HeapAlloc failed!\n");
2621                 goto Cleanup;
2622             }
2623 
2624             MultiByteToWideChar(CP_ACP, 0, pDefault->pDatatype, -1, wDefault.pDatatype, cch + 1);
2625         }
2626 
2627         if (pDefault->pDevMode)
2628             wDefault.pDevMode = GdiConvertToDevmodeW(pDefault->pDevMode);
2629     }
2630 
2631     bReturnValue = OpenPrinterW(pwszPrinterName, phPrinter, &wDefault);
2632 
2633     if ( bReturnValue )
2634     {
2635         PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)*phPrinter;
2636         pHandle->bAnsi = TRUE;
2637     }
2638 
2639 Cleanup:
2640     if (wDefault.pDatatype)
2641         HeapFree(hProcessHeap, 0, wDefault.pDatatype);
2642 
2643     if (wDefault.pDevMode)
2644         HeapFree(hProcessHeap, 0, wDefault.pDevMode);
2645 
2646     if (pwszPrinterName)
2647         HeapFree(hProcessHeap, 0, pwszPrinterName);
2648 
2649     return bReturnValue;
2650 }
2651 
2652 BOOL WINAPI
2653 OpenPrinterW(LPWSTR pPrinterName, LPHANDLE phPrinter, LPPRINTER_DEFAULTSW pDefault)
2654 {
2655     DWORD dwErrorCode;
2656     HANDLE hPrinter;
2657     PSPOOLER_HANDLE pHandle;
2658     PWSTR pDatatype = NULL;
2659     WINSPOOL_DEVMODE_CONTAINER DevModeContainer = { 0 };
2660     ACCESS_MASK AccessRequired = 0;
2661 
2662     TRACE("OpenPrinterW(%S, %p, %p)\n", pPrinterName, phPrinter, pDefault);
2663 
2664     // Sanity check
2665     if (!phPrinter)
2666     {
2667         dwErrorCode = ERROR_INVALID_PARAMETER;
2668         goto Cleanup;
2669     }
2670 
2671     // Prepare the additional parameters in the format required by _RpcOpenPrinter
2672     if (pDefault)
2673     {
2674         pDatatype = pDefault->pDatatype;
2675         DevModeContainer.cbBuf = sizeof(DEVMODEW);
2676         DevModeContainer.pDevMode = (BYTE*)pDefault->pDevMode;
2677         AccessRequired = pDefault->DesiredAccess;
2678     }
2679 
2680     // Do the RPC call
2681     RpcTryExcept
2682     {
2683         dwErrorCode = _RpcOpenPrinter(pPrinterName, &hPrinter, pDatatype, &DevModeContainer, AccessRequired);
2684     }
2685     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
2686     {
2687         dwErrorCode = RpcExceptionCode();
2688         ERR("_RpcOpenPrinter failed with exception code %lu!\n", dwErrorCode);
2689     }
2690     RpcEndExcept;
2691 
2692     if (dwErrorCode == ERROR_SUCCESS)
2693     {
2694         // Create a new SPOOLER_HANDLE structure.
2695         pHandle = HeapAlloc(hProcessHeap, HEAP_ZERO_MEMORY, sizeof(SPOOLER_HANDLE));
2696         if (!pHandle)
2697         {
2698             dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
2699             ERR("HeapAlloc failed!\n");
2700             goto Cleanup;
2701         }
2702 
2703         pHandle->Sig = SPOOLER_HANDLE_SIG;
2704         pHandle->hPrinter = hPrinter;
2705         pHandle->hSPLFile = INVALID_HANDLE_VALUE;
2706         pHandle->hSpoolFileHandle = INVALID_HANDLE_VALUE;
2707 
2708         // Return it as phPrinter.
2709         *phPrinter = (HANDLE)pHandle;
2710     }
2711 
2712 Cleanup:
2713     SetLastError(dwErrorCode);
2714     return (dwErrorCode == ERROR_SUCCESS);
2715 }
2716 
2717 //
2718 // Dead API.
2719 //
2720 DWORD WINAPI
2721 PrinterMessageBoxA(HANDLE hPrinter, DWORD Error, HWND hWnd, LPSTR pText, LPSTR pCaption, DWORD dwType)
2722 {
2723     return 50;
2724 }
2725 
2726 DWORD WINAPI
2727 PrinterMessageBoxW(HANDLE hPrinter, DWORD Error, HWND hWnd, LPWSTR pText, LPWSTR pCaption, DWORD dwType)
2728 {
2729     return 50;
2730 }
2731 
2732 BOOL WINAPI
2733 QueryColorProfile(
2734   HANDLE    hPrinter,
2735   PDEVMODEW pdevmode,
2736   ULONG     ulQueryMode,
2737   VOID      *pvProfileData,
2738   ULONG     *pcbProfileData,
2739   FLONG     *pflProfileData )
2740 {
2741     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
2742     BOOL Ret = FALSE;
2743     HMODULE hLibrary;
2744 
2745     FIXME("QueryColorProfile(%p, %p, %l, %p, %p, %p)\n", hPrinter, pdevmode, ulQueryMode, pvProfileData, pcbProfileData, pflProfileData);
2746 
2747     if ( pHandle->bNoColorProfile )
2748     {
2749         Ret = (BOOL)SP_ERROR;
2750     }
2751     else
2752     {
2753 
2754         if ( pdevmode )
2755         {
2756             if (!IsValidDevmodeNoSizeW( pdevmode ) )
2757             {
2758                 ERR("QueryColorProfile : Devode Invalid\n");
2759                 return FALSE;
2760             }
2761         }
2762 
2763         hLibrary = LoadPrinterDriver( hPrinter );
2764 
2765         if ( hLibrary )
2766         {
2767             fpQueryColorProfile = (PVOID)GetProcAddress( hLibrary, "DrvQueryColorProfile" );
2768 
2769             if ( fpQueryColorProfile )
2770             {
2771                 Ret = fpQueryColorProfile( hPrinter, pdevmode, ulQueryMode, pvProfileData, pcbProfileData, pflProfileData );
2772             }
2773             else
2774             {
2775                 pHandle->bNoColorProfile = TRUE;
2776                 Ret = (BOOL)SP_ERROR;
2777             }
2778 
2779             FreeLibrary(hLibrary);
2780         }
2781     }
2782     return Ret;
2783 }
2784 
2785 // Note from GDI32:printdrv.c
2786 //
2787 //  QuerySpoolMode :
2788 //    BOOL return TRUE if successful.
2789 //    dlFont 0x0001 for Downloading fonts. 0x0002 unknown XPS_PASS?.
2790 //    dwVersion is version of EMFSPOOL. Must be 0x00010000. See [MS-EMFSPOOL] page 18.
2791 //
2792 
2793 #define QSM_DOWNLOADINGFONTS 0x0001
2794 
2795 /*
2796    Note from MSDN : "V4 print drivers using RAW mode to send PCL/Postscript have 0 byte spool file"
2797 
2798    Use XPS_PASS instead of RAW to pass information directly to the print filter pipeline in
2799    v4 and v3 XPSDrv drivers. Here's how to proceed with Windows 8:
2800 
2801     Call GetPrinterDriver to retrieve the DRIVER_INFO_8 structure.
2802     Check DRIVER_INFO_8::dwPrinterDriverAttributes for the PRINTER_DRIVER_XPS flag.
2803     Choose your datatype based on the presence or absence of the flag:
2804         If the flag is set, use XPS_PASS.
2805         If the flag isn't set, use RAW.
2806  */
2807 
2808 #define QSM_XPS_PASS         0x0002 // Guessing. PRINTER_DRIVER_XPS?
2809 
2810 BOOL WINAPI
2811 QuerySpoolMode( HANDLE hPrinter, PDWORD downloadFontsFlags, PDWORD dwVersion )
2812 {
2813     PRINTER_INFO_2W *pi2 = NULL;
2814     DWORD needed = 0;
2815     BOOL res;
2816 
2817     FIXME("QuerySpoolMode(%p, %p, %p)\n", hPrinter, downloadFontsFlags, dwVersion);
2818 
2819     res = GetPrinterW( hPrinter, 2, NULL, 0, &needed);
2820     if (!res && (GetLastError() == ERROR_INSUFFICIENT_BUFFER))
2821     {
2822         pi2 = HeapAlloc(hProcessHeap, 0, needed);
2823         if (!pi2)
2824         {
2825             ERR("Failed to allocate PRINTER_INFO_2W of %u bytes\n", needed);
2826             return FALSE;
2827         }
2828         res = GetPrinterW(hPrinter, 2, (LPBYTE)pi2, needed, &needed);
2829     }
2830 
2831     if ( res )
2832     {
2833         *dwVersion = 0x10000;
2834         *downloadFontsFlags = 0;
2835 
2836         if ( pi2->pServerName )
2837         {
2838             *downloadFontsFlags |= QSM_DOWNLOADINGFONTS;
2839         }
2840     }
2841 //
2842 //  Guessing,,,
2843 //  To do : Add GetPrinterDriver for DRIVER_INFO_8, test PRINTER_DRIVER_XPS flag,
2844 //          to set *downloadFontsFlags |= QSM_XPS_PASS;
2845 //
2846 //  Vista+ looks for QSM_XPS_PASS to be set in GDI32.
2847 //
2848     HeapFree(hProcessHeap, 0, pi2);
2849     return res;
2850 }
2851 
2852 //
2853 // This requires IC support.
2854 //
2855 DWORD WINAPI
2856 QueryRemoteFonts( HANDLE hPrinter, PUNIVERSAL_FONT_ID pufi, ULONG NumberOfUFIs )
2857 {
2858     HANDLE hIC;
2859     DWORD Result = -1, cOut, cIn = 0;
2860     PBYTE pOut;
2861 
2862     FIXME("QueryRemoteFonts(%p, %p, %lu)\n", hPrinter, pufi, NumberOfUFIs);
2863 
2864     hIC = CreatePrinterIC( hPrinter, NULL );
2865     if ( hIC )
2866     {
2867         cOut = (NumberOfUFIs * sizeof(UNIVERSAL_FONT_ID)) + sizeof(DWORD); // Include "DWORD" first part to return size.
2868 
2869         pOut = HeapAlloc( hProcessHeap, 0, cOut );
2870         if ( pOut )
2871         {
2872             if ( PlayGdiScriptOnPrinterIC( hIC, (LPBYTE)&cIn, sizeof(DWORD), pOut, cOut, 0 ) )
2873             {
2874                 cIn = *((PDWORD)pOut); // Fisrt part is the size of the UFID object.
2875 
2876                 Result = cIn; // Return the required size.
2877 
2878                 if( NumberOfUFIs < cIn )
2879                 {
2880                     cIn = NumberOfUFIs;
2881                 }
2882                 //     Copy whole object back to GDI32, exclude first DWORD part.
2883                 memcpy( pufi, pOut + sizeof(DWORD), cIn * sizeof(UNIVERSAL_FONT_ID) );
2884             }
2885             HeapFree( hProcessHeap, 0, pOut );
2886         }
2887         DeletePrinterIC( hIC );
2888     }
2889     return Result;
2890 }
2891 
2892 BOOL WINAPI
2893 ReadPrinter(HANDLE hPrinter, PVOID pBuf, DWORD cbBuf, PDWORD pNoBytesRead)
2894 {
2895     DWORD dwErrorCode;
2896     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
2897 
2898     TRACE("ReadPrinter(%p, %p, %lu, %p)\n", hPrinter, pBuf, cbBuf, pNoBytesRead);
2899 
2900     // Sanity checks.
2901     if (!pHandle)
2902     {
2903         dwErrorCode = ERROR_INVALID_HANDLE;
2904         goto Cleanup;
2905     }
2906 
2907     // Do the RPC call
2908     RpcTryExcept
2909     {
2910         dwErrorCode = _RpcReadPrinter(pHandle->hPrinter, pBuf, cbBuf, pNoBytesRead);
2911     }
2912     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
2913     {
2914         dwErrorCode = RpcExceptionCode();
2915         ERR("_RpcReadPrinter failed with exception code %lu!\n", dwErrorCode);
2916     }
2917     RpcEndExcept;
2918 
2919 Cleanup:
2920     SetLastError(dwErrorCode);
2921     return (dwErrorCode == ERROR_SUCCESS);
2922 }
2923 
2924 BOOL WINAPI
2925 ResetPrinterA(HANDLE hPrinter, PPRINTER_DEFAULTSA pDefault)
2926 {
2927     BOOL ret;
2928     UNICODE_STRING pNameW;
2929     PDEVMODEW pdmw = NULL;
2930     PPRINTER_DEFAULTSW pdw = (PPRINTER_DEFAULTSW)pDefault;
2931 
2932     TRACE("ResetPrinterA(%p, %p)\n", hPrinter, pDefault);
2933 
2934     if ( pDefault->pDatatype == (LPSTR)-1 )
2935     {
2936         pdw->pDatatype = (LPWSTR)-1;
2937     }
2938     else
2939     {
2940         pdw->pDatatype = AsciiToUnicode( &pNameW, pDefault->pDatatype );
2941     }
2942     if ( pDefault->pDevMode == (LPDEVMODEA)-1)
2943     {
2944         pdw->pDevMode = (LPDEVMODEW)-1;
2945     }
2946     else
2947     {
2948         if ( pDefault->pDevMode )//&& IsValidDevmodeNoSizeW( pDefault->pDevMode ) )
2949         {
2950             RosConvertAnsiDevModeToUnicodeDevmode( pDefault->pDevMode, &pdmw );
2951             pdw->pDevMode = pdmw;
2952         }
2953     }
2954 
2955     ret = ResetPrinterW( hPrinter, pdw );
2956 
2957     if (pdmw) HeapFree(hProcessHeap, 0, pdmw);
2958 
2959     RtlFreeUnicodeString( &pNameW );
2960 
2961     return ret;
2962 }
2963 
2964 BOOL WINAPI
2965 ResetPrinterW(HANDLE hPrinter, PPRINTER_DEFAULTSW pDefault)
2966 {
2967     TRACE("ResetPrinterW(%p, %p)\n", hPrinter, pDefault);
2968     UNIMPLEMENTED;
2969     return FALSE;
2970 }
2971 
2972 BOOL WINAPI
2973 SeekPrinter( HANDLE hPrinter, LARGE_INTEGER liDistanceToMove, PLARGE_INTEGER pliNewPointer, DWORD dwMoveMethod, BOOL bWrite )
2974 {
2975     DWORD dwErrorCode;
2976     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
2977 
2978     FIXME("SeekPrinter(%p, %I64u, %p, %lu, %d)\n", hPrinter, liDistanceToMove.QuadPart, pliNewPointer, dwMoveMethod, bWrite);
2979 
2980     // Sanity checks.
2981     if (!pHandle)
2982     {
2983         dwErrorCode = ERROR_INVALID_HANDLE;
2984         goto Cleanup;
2985     }
2986 
2987     // Do the RPC call
2988     RpcTryExcept
2989     {
2990         dwErrorCode = _RpcSeekPrinter(pHandle->hPrinter, liDistanceToMove, pliNewPointer, dwMoveMethod, bWrite);
2991     }
2992     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
2993     {
2994         dwErrorCode = RpcExceptionCode();
2995         ERR("_RpcSeekPrinter failed with exception code %lu!\n", dwErrorCode);
2996     }
2997     RpcEndExcept;
2998 
2999 Cleanup:
3000     SetLastError(dwErrorCode);
3001     return (dwErrorCode == ERROR_SUCCESS);
3002 }
3003 
3004 BOOL WINAPI
3005 SetDefaultPrinterA(LPCSTR pszPrinter)
3006 {
3007     BOOL bReturnValue = FALSE;
3008     DWORD cch;
3009     PWSTR pwszPrinter = NULL;
3010 
3011     TRACE("SetDefaultPrinterA(%s)\n", pszPrinter);
3012 
3013     if (pszPrinter)
3014     {
3015         // Convert pszPrinter to a Unicode string pwszPrinter
3016         cch = strlen(pszPrinter);
3017 
3018         pwszPrinter = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(WCHAR));
3019         if (!pwszPrinter)
3020         {
3021             SetLastError(ERROR_NOT_ENOUGH_MEMORY);
3022             ERR("HeapAlloc failed!\n");
3023             goto Cleanup;
3024         }
3025 
3026         MultiByteToWideChar(CP_ACP, 0, pszPrinter, -1, pwszPrinter, cch + 1);
3027     }
3028 
3029     bReturnValue = SetDefaultPrinterW(pwszPrinter);
3030 
3031 Cleanup:
3032     if (pwszPrinter)
3033         HeapFree(hProcessHeap, 0, pwszPrinter);
3034 
3035     return bReturnValue;
3036 }
3037 
3038 BOOL WINAPI
3039 SetDefaultPrinterW(LPCWSTR pszPrinter)
3040 {
3041     const WCHAR wszDevicesKey[] = L"Software\\Microsoft\\Windows NT\\CurrentVersion\\Devices";
3042 
3043     DWORD cbDeviceValueData;
3044     DWORD cbPrinterValueData = 0;
3045     DWORD cchPrinter;
3046     DWORD dwErrorCode;
3047     HKEY hDevicesKey = NULL;
3048     HKEY hWindowsKey = NULL;
3049     PWSTR pwszDeviceValueData = NULL;
3050     WCHAR wszPrinter[MAX_PRINTER_NAME + 1];
3051 
3052     TRACE("SetDefaultPrinterW(%S)\n", pszPrinter);
3053 
3054     // Open the Devices registry key.
3055     dwErrorCode = (DWORD)RegOpenKeyExW(HKEY_CURRENT_USER, wszDevicesKey, 0, KEY_READ, &hDevicesKey);
3056     if (dwErrorCode != ERROR_SUCCESS)
3057     {
3058         ERR("RegOpenKeyExW failed with status %lu!\n", dwErrorCode);
3059         goto Cleanup;
3060     }
3061 
3062     // Did the caller give us a printer to set as default?
3063     if (pszPrinter && *pszPrinter)
3064     {
3065         // Check if the given printer exists and query the value data size.
3066         dwErrorCode = (DWORD)RegQueryValueExW(hDevicesKey, pszPrinter, NULL, NULL, NULL, &cbPrinterValueData);
3067         if (dwErrorCode == ERROR_FILE_NOT_FOUND)
3068         {
3069             dwErrorCode = ERROR_INVALID_PRINTER_NAME;
3070             goto Cleanup;
3071         }
3072         else if (dwErrorCode != ERROR_SUCCESS)
3073         {
3074             ERR("RegQueryValueExW failed with status %lu!\n", dwErrorCode);
3075             goto Cleanup;
3076         }
3077 
3078         cchPrinter = wcslen(pszPrinter);
3079     }
3080     else
3081     {
3082         // If there is already a default printer, we're done!
3083         cchPrinter = _countof(wszPrinter);
3084         if (GetDefaultPrinterW(wszPrinter, &cchPrinter))
3085         {
3086             dwErrorCode = ERROR_SUCCESS;
3087             goto Cleanup;
3088         }
3089 
3090         // Otherwise, get us the first printer from the "Devices" key to later set it as default and query the value data size.
3091         cchPrinter = _countof(wszPrinter);
3092         dwErrorCode = (DWORD)RegEnumValueW(hDevicesKey, 0, wszPrinter, &cchPrinter, NULL, NULL, NULL, &cbPrinterValueData);
3093         if (dwErrorCode != ERROR_MORE_DATA)
3094             goto Cleanup;
3095 
3096         pszPrinter = wszPrinter;
3097     }
3098 
3099     // We now need to query the value data, which has the format "winspool,<Port>:"
3100     // and make "<Printer Name>,winspool,<Port>:" out of it.
3101     // Allocate a buffer large enough for the final data.
3102     cbDeviceValueData = (cchPrinter + 1) * sizeof(WCHAR) + cbPrinterValueData;
3103     pwszDeviceValueData = HeapAlloc(hProcessHeap, 0, cbDeviceValueData);
3104     if (!pwszDeviceValueData)
3105     {
3106         dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
3107         ERR("HeapAlloc failed!\n");
3108         goto Cleanup;
3109     }
3110 
3111     // Copy the Printer Name and a comma into it.
3112     CopyMemory(pwszDeviceValueData, pszPrinter, cchPrinter * sizeof(WCHAR));
3113     pwszDeviceValueData[cchPrinter] = L',';
3114 
3115     // Append the value data, which has the format "winspool,<Port>:"
3116     dwErrorCode = (DWORD)RegQueryValueExW(hDevicesKey, pszPrinter, NULL, NULL, (PBYTE)&pwszDeviceValueData[cchPrinter + 1], &cbPrinterValueData);
3117     if (dwErrorCode != ERROR_SUCCESS)
3118         goto Cleanup;
3119 
3120     // Open the Windows registry key.
3121     dwErrorCode = (DWORD)RegOpenKeyExW(HKEY_CURRENT_USER, wszWindowsKey, 0, KEY_SET_VALUE, &hWindowsKey);
3122     if (dwErrorCode != ERROR_SUCCESS)
3123     {
3124         ERR("RegOpenKeyExW failed with status %lu!\n", dwErrorCode);
3125         goto Cleanup;
3126     }
3127 
3128     // Store our new default printer.
3129     dwErrorCode = (DWORD)RegSetValueExW(hWindowsKey, wszDeviceValue, 0, REG_SZ, (PBYTE)pwszDeviceValueData, cbDeviceValueData);
3130     if (dwErrorCode != ERROR_SUCCESS)
3131     {
3132         ERR("RegSetValueExW failed with status %lu!\n", dwErrorCode);
3133         goto Cleanup;
3134     }
3135 
3136 Cleanup:
3137     if (hDevicesKey)
3138         RegCloseKey(hDevicesKey);
3139 
3140     if (hWindowsKey)
3141         RegCloseKey(hWindowsKey);
3142 
3143     if (pwszDeviceValueData)
3144         HeapFree(hProcessHeap, 0, pwszDeviceValueData);
3145 
3146     SetLastError(dwErrorCode);
3147     return (dwErrorCode == ERROR_SUCCESS);
3148 }
3149 
3150 BOOL WINAPI
3151 SetPrinterA(HANDLE hPrinter, DWORD Level, PBYTE pPrinter, DWORD Command)
3152 {
3153     BOOL Ret = FALSE;
3154     UNICODE_STRING usBuffer;
3155     PPRINTER_INFO_STRESS ppisa = (PPRINTER_INFO_STRESS)pPrinter;
3156     PPRINTER_INFO_STRESS ppisw = (PPRINTER_INFO_STRESS)pPrinter;
3157     PPRINTER_INFO_2A ppi2a = (PPRINTER_INFO_2A)pPrinter;
3158     PPRINTER_INFO_2W ppi2w = (PPRINTER_INFO_2W)pPrinter;
3159     PPRINTER_INFO_7A ppi7a = (PPRINTER_INFO_7A)pPrinter;
3160     PPRINTER_INFO_7W ppi7w = (PPRINTER_INFO_7W)pPrinter;
3161     PPRINTER_INFO_9A ppi9a = (PPRINTER_INFO_9A)pPrinter;
3162     PPRINTER_INFO_9W ppi9w = (PPRINTER_INFO_9W)pPrinter;
3163     PWSTR pwszPrinterName = NULL;
3164     PWSTR pwszServerName = NULL;
3165     PWSTR pwszShareName = NULL;
3166     PWSTR pwszPortName = NULL;
3167     PWSTR pwszDriverName = NULL;
3168     PWSTR pwszComment = NULL;
3169     PWSTR pwszLocation = NULL;
3170     PWSTR pwszSepFile = NULL;
3171     PWSTR pwszPrintProcessor = NULL;
3172     PWSTR pwszDatatype = NULL;
3173     PWSTR pwszParameters = NULL;
3174     PDEVMODEW pdmw = NULL;
3175 
3176     FIXME("SetPrinterA(%p, %lu, %p, %lu)\n", hPrinter, Level, pPrinter, Command);
3177 
3178     switch ( Level )
3179     {
3180         case 0:
3181             if ( Command == 0 )
3182             {
3183                 if (ppisa->pPrinterName)
3184                 {
3185                     pwszPrinterName = AsciiToUnicode(&usBuffer, (LPCSTR)ppisa->pPrinterName);
3186                     if (!(ppisw->pPrinterName = pwszPrinterName)) goto Cleanup;
3187                 }
3188                 if (ppisa->pServerName)
3189                 {
3190                     pwszServerName = AsciiToUnicode(&usBuffer, (LPCSTR)ppisa->pServerName);
3191                     if (!(ppisw->pPrinterName = pwszServerName)) goto Cleanup;
3192                 }
3193             }
3194             if ( Command == PRINTER_CONTROL_SET_STATUS )
3195             {
3196                 // Set the pPrinter parameter to a pointer to a DWORD value that specifies the new printer status.
3197                 PRINTER_INFO_6 pi6;
3198                 pi6.dwStatus = (DWORD_PTR)pPrinter;
3199                 pPrinter = (LPBYTE)&pi6;
3200                 Level = 6;
3201                 Command = 0;
3202             }
3203             break;
3204         case 2:
3205             {
3206                 if (ppi2a->pShareName)
3207                 {
3208                     pwszShareName = AsciiToUnicode(&usBuffer, ppi2a->pShareName);
3209                     if (!(ppi2w->pShareName = pwszShareName)) goto Cleanup;
3210                 }
3211                 if (ppi2a->pPortName)
3212                 {
3213                     pwszPortName = AsciiToUnicode(&usBuffer, ppi2a->pPortName);
3214                     if (!(ppi2w->pPortName = pwszPortName)) goto Cleanup;
3215                 }
3216                 if (ppi2a->pDriverName)
3217                 {
3218                     pwszDriverName = AsciiToUnicode(&usBuffer, ppi2a->pDriverName);
3219                     if (!(ppi2w->pDriverName = pwszDriverName)) goto Cleanup;
3220                 }
3221                 if (ppi2a->pComment)
3222                 {
3223                     pwszComment = AsciiToUnicode(&usBuffer, ppi2a->pComment);
3224                     if (!(ppi2w->pComment = pwszComment)) goto Cleanup;
3225                 }
3226                 if (ppi2a->pLocation)
3227                 {
3228                     pwszLocation = AsciiToUnicode(&usBuffer, ppi2a->pLocation);
3229                     if (!(ppi2w->pLocation = pwszLocation)) goto Cleanup;
3230                 }
3231                 if (ppi2a->pSepFile)
3232                 {
3233                     pwszSepFile = AsciiToUnicode(&usBuffer, ppi2a->pSepFile);
3234                     if (!(ppi2w->pSepFile = pwszSepFile)) goto Cleanup;
3235                 }
3236                 if (ppi2a->pServerName)
3237                 {
3238                     pwszPrintProcessor = AsciiToUnicode(&usBuffer, ppi2a->pPrintProcessor);
3239                     if (!(ppi2w->pPrintProcessor = pwszPrintProcessor)) goto Cleanup;
3240                 }
3241                 if (ppi2a->pDatatype)
3242                 {
3243                     pwszDatatype = AsciiToUnicode(&usBuffer, ppi2a->pDatatype);
3244                     if (!(ppi2w->pDatatype = pwszDatatype)) goto Cleanup;
3245                 }
3246                 if (ppi2a->pParameters)
3247                 {
3248                     pwszParameters = AsciiToUnicode(&usBuffer, ppi2a->pParameters);
3249                     if (!(ppi2w->pParameters = pwszParameters)) goto Cleanup;
3250                 }
3251 
3252                 if ( ppi2a->pDevMode )
3253                 {
3254                     RosConvertAnsiDevModeToUnicodeDevmode( ppi2a->pDevMode, &pdmw );
3255                     ppi2w->pDevMode = pdmw;
3256                 }
3257             }
3258         //
3259         //  These two strings are relitive and common to these three Levels.
3260         //  Fall through...
3261         //
3262         case 4:
3263         case 5:
3264             {
3265                 if (ppi2a->pServerName) // 4 & 5 : pPrinterName.
3266                 {
3267                     pwszServerName = AsciiToUnicode(&usBuffer, ppi2a->pServerName);
3268                     if (!(ppi2w->pPrinterName = pwszServerName)) goto Cleanup;
3269                 }
3270                 if (ppi2a->pPrinterName) // 4 : pServerName, 5 : pPortName.
3271                 {
3272                     pwszPrinterName = AsciiToUnicode(&usBuffer, ppi2a->pPrinterName);
3273                     if (!(ppi2w->pPrinterName = pwszPrinterName)) goto Cleanup;
3274                 }
3275             }
3276             break;
3277         case 3:
3278         case 6:
3279             break;
3280         case 7:
3281             {
3282                 if (ppi7a->pszObjectGUID)
3283                 {
3284                     pwszPrinterName = AsciiToUnicode(&usBuffer, ppi7a->pszObjectGUID);
3285                     if (!(ppi7w->pszObjectGUID = pwszPrinterName)) goto Cleanup;
3286                 }
3287             }
3288             break;
3289 
3290         case 8:
3291         /* 8 is the global default printer info and 9 already sets it instead of the per-user one */
3292         /* still, PRINTER_INFO_8W is the same as PRINTER_INFO_9W */
3293         /* fall through */
3294         case 9:
3295             {
3296                 RosConvertAnsiDevModeToUnicodeDevmode( ppi9a->pDevMode, &pdmw );
3297                 ppi9w->pDevMode = pdmw;
3298             }
3299             break;
3300 
3301         default:
3302             FIXME( "Unsupported level %d\n", Level);
3303             SetLastError( ERROR_INVALID_LEVEL );
3304     }
3305 
3306     Ret = SetPrinterW( hPrinter, Level, pPrinter, Command );
3307 
3308 Cleanup:
3309     if (pdmw) HeapFree(hProcessHeap, 0, pdmw);
3310     if (pwszPrinterName) HeapFree(hProcessHeap, 0, pwszPrinterName);
3311     if (pwszServerName) HeapFree(hProcessHeap, 0, pwszServerName);
3312     if (pwszShareName) HeapFree(hProcessHeap, 0, pwszShareName);
3313     if (pwszPortName) HeapFree(hProcessHeap, 0, pwszPortName);
3314     if (pwszDriverName) HeapFree(hProcessHeap, 0, pwszDriverName);
3315     if (pwszComment) HeapFree(hProcessHeap, 0, pwszComment);
3316     if (pwszLocation) HeapFree(hProcessHeap, 0, pwszLocation);
3317     if (pwszSepFile) HeapFree(hProcessHeap, 0, pwszSepFile);
3318     if (pwszPrintProcessor) HeapFree(hProcessHeap, 0, pwszPrintProcessor);
3319     if (pwszDatatype) HeapFree(hProcessHeap, 0, pwszDatatype);
3320     if (pwszParameters) HeapFree(hProcessHeap, 0, pwszParameters);
3321     return Ret;
3322 }
3323 
3324 BOOL WINAPI
3325 SetPrinterW(HANDLE hPrinter, DWORD Level, PBYTE pPrinter, DWORD Command)
3326 {
3327     DWORD dwErrorCode;
3328     WINSPOOL_PRINTER_CONTAINER PrinterContainer;
3329     WINSPOOL_DEVMODE_CONTAINER DevModeContainer;
3330     WINSPOOL_SECURITY_CONTAINER SecurityContainer;
3331     SECURITY_DESCRIPTOR *sd = NULL;
3332     DWORD size;
3333     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
3334 
3335     FIXME("SetPrinterW(%p, %lu, %p, %lu)\n", hPrinter, Level, pPrinter, Command);
3336 
3337     // Sanity checks
3338     if (!pHandle)
3339         return ERROR_INVALID_HANDLE;
3340 
3341     DevModeContainer.cbBuf = 0;
3342     DevModeContainer.pDevMode = NULL;
3343 
3344     SecurityContainer.cbBuf = 0;
3345     SecurityContainer.pSecurity = NULL;
3346 
3347     switch ( Level )
3348     {
3349         case 0:
3350             if ( Command == PRINTER_CONTROL_SET_STATUS )
3351             {
3352                 // Set the pPrinter parameter to a pointer to a DWORD value that specifies the new printer status.
3353                 PRINTER_INFO_6 pi6;
3354                 pi6.dwStatus = (DWORD_PTR)pPrinter;
3355                 pPrinter = (LPBYTE)&pi6;
3356                 Level = 6;
3357                 Command = 0;
3358             }
3359             break;
3360         case 2:
3361             {
3362                 PPRINTER_INFO_2W pi2w = (PPRINTER_INFO_2W)pPrinter;
3363                 if ( pi2w )
3364                 {
3365                     if ( pi2w->pDevMode )
3366                     {
3367                          if ( IsValidDevmodeNoSizeW( pi2w->pDevMode ) )
3368                          {
3369                              DevModeContainer.cbBuf = pi2w->pDevMode->dmSize + pi2w->pDevMode->dmDriverExtra;
3370                              DevModeContainer.pDevMode = (PBYTE)pi2w->pDevMode;
3371                          }
3372                     }
3373 
3374                     if ( pi2w->pSecurityDescriptor )
3375                     {
3376                         sd = get_sd( pi2w->pSecurityDescriptor, &size );
3377                         if ( sd )
3378                         {
3379                             SecurityContainer.cbBuf = size;
3380                             SecurityContainer.pSecurity = (PBYTE)sd;
3381                         }
3382                     }
3383                 }
3384                 else
3385                 {
3386                     SetLastError(ERROR_INVALID_PARAMETER);
3387                     return FALSE;
3388                 }
3389             }
3390             break;
3391         case 3:
3392             {
3393                 PPRINTER_INFO_3 pi3 = (PPRINTER_INFO_3)pPrinter;
3394                 if ( pi3 )
3395                 {
3396                     if ( pi3->pSecurityDescriptor )
3397                     {
3398                         sd = get_sd( pi3->pSecurityDescriptor, &size );
3399                         if ( sd )
3400                         {
3401                             SecurityContainer.cbBuf = size;
3402                             SecurityContainer.pSecurity = (PBYTE)sd;
3403                         }
3404                     }
3405                 }
3406                 else
3407                 {
3408                     SetLastError(ERROR_INVALID_PARAMETER);
3409                     return FALSE;
3410                 }
3411             }
3412             break;
3413 
3414         case 4:
3415         case 5:
3416         case 6:
3417         case 7:
3418             if ( pPrinter == NULL )
3419             {
3420                 SetLastError(ERROR_INVALID_PARAMETER);
3421                 return FALSE;
3422             }
3423             break;
3424 
3425         case 8:
3426         /* 8 is the global default printer info and 9 already sets it instead of the per-user one */
3427         /* still, PRINTER_INFO_8W is the same as PRINTER_INFO_9W */
3428         /* fall through */
3429         case 9:
3430             {
3431                 PPRINTER_INFO_9W pi9w = (PPRINTER_INFO_9W)pPrinter;
3432                 if ( pi9w )
3433                 {
3434                     if ( pi9w->pDevMode )
3435                     {
3436                          if ( IsValidDevmodeNoSizeW( pi9w->pDevMode ) )
3437                          {
3438                              DevModeContainer.cbBuf = pi9w->pDevMode->dmSize + pi9w->pDevMode->dmDriverExtra;
3439                              DevModeContainer.pDevMode = (LPBYTE)pi9w->pDevMode;
3440                          }
3441                     }
3442                 }
3443             }
3444             break;
3445 
3446         default:
3447             FIXME( "Unsupported level %d\n", Level );
3448             SetLastError( ERROR_INVALID_LEVEL );
3449             return FALSE;
3450     }
3451 
3452     PrinterContainer.PrinterInfo.pPrinterInfo1 = (WINSPOOL_PRINTER_INFO_1*)pPrinter;
3453     PrinterContainer.Level = Level;
3454 
3455     // Do the RPC call
3456     RpcTryExcept
3457     {
3458         dwErrorCode = _RpcSetPrinter(pHandle->hPrinter, &PrinterContainer, &DevModeContainer, &SecurityContainer, Command);
3459     }
3460     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
3461     {
3462         dwErrorCode = RpcExceptionCode();
3463     }
3464     RpcEndExcept;
3465 
3466     if ( sd ) HeapFree( GetProcessHeap(), 0, sd );
3467 
3468     SetLastError(dwErrorCode);
3469     return (dwErrorCode == ERROR_SUCCESS);
3470 }
3471 
3472 BOOL WINAPI
3473 SplDriverUnloadComplete(LPWSTR pDriverFile)
3474 {
3475     TRACE("DriverUnloadComplete(%S)\n", pDriverFile);
3476     UNIMPLEMENTED;
3477     return TRUE; // return true for now.
3478 }
3479 BOOL WINAPI
3480 
3481 SpoolerPrinterEvent( LPWSTR pPrinterName, INT DriverEvent, DWORD Flags, LPARAM lParam )
3482 {
3483     HMODULE hLibrary;
3484     HANDLE hPrinter;
3485     BOOL Ret = FALSE;
3486 
3487     if ( OpenPrinterW( pPrinterName, &hPrinter, NULL ) )
3488     {
3489         hLibrary = LoadPrinterDriver( hPrinter );
3490 
3491         if ( hLibrary )
3492         {
3493             fpPrinterEvent = (PVOID)GetProcAddress( hLibrary, "DrvPrinterEvent" );
3494 
3495             if ( fpPrinterEvent )
3496             {
3497                 Ret = fpPrinterEvent( pPrinterName, DriverEvent, Flags, lParam );
3498             }
3499 
3500             FreeLibrary(hLibrary);
3501         }
3502 
3503         ClosePrinter( hPrinter );
3504     }
3505 
3506     return Ret;
3507 }
3508 
3509 INT_PTR CALLBACK file_dlg_proc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam)
3510 {
3511     LPWSTR filename;
3512 
3513     switch(msg)
3514     {
3515     case WM_INITDIALOG:
3516         SetWindowLongPtrW(hwnd, DWLP_USER, lparam);
3517         return TRUE;
3518 
3519     case WM_COMMAND:
3520         if(HIWORD(wparam) == BN_CLICKED)
3521         {
3522             if(LOWORD(wparam) == IDOK)
3523             {
3524                 HANDLE hf;
3525                 DWORD len = SendDlgItemMessageW(hwnd, EDITBOX, WM_GETTEXTLENGTH, 0, 0);
3526                 LPWSTR *output;
3527 
3528                 filename = HeapAlloc(GetProcessHeap(), 0, (len + 1) * sizeof(WCHAR));
3529                 if (!filename)
3530                 {
3531                     ERR("Failed to allocate filename of %u bytes\n", (len + 1) * sizeof(WCHAR));
3532                     return FALSE;
3533                 }
3534                 GetDlgItemTextW(hwnd, EDITBOX, filename, len + 1);
3535 
3536                 if(GetFileAttributesW(filename) != INVALID_FILE_ATTRIBUTES)
3537                 {
3538                     WCHAR caption[200], message[200];
3539                     int mb_ret;
3540 
3541                     LoadStringW(hinstWinSpool, IDS_CAPTION, caption, ARRAYSIZE(caption));
3542                     LoadStringW(hinstWinSpool, IDS_FILE_EXISTS, message, ARRAYSIZE(message));
3543                     mb_ret = MessageBoxW(hwnd, message, caption, MB_OKCANCEL | MB_ICONEXCLAMATION);
3544                     if(mb_ret == IDCANCEL)
3545                     {
3546                         HeapFree(GetProcessHeap(), 0, filename);
3547                         return TRUE;
3548                     }
3549                 }
3550                 hf = CreateFileW(filename, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
3551                 if(hf == INVALID_HANDLE_VALUE)
3552                 {
3553                     WCHAR caption[200], message[200];
3554 
3555                     LoadStringW(hinstWinSpool, IDS_CAPTION, caption, ARRAYSIZE(caption));
3556                     LoadStringW(hinstWinSpool, IDS_CANNOT_OPEN, message, ARRAYSIZE(message));
3557                     MessageBoxW(hwnd, message, caption, MB_OK | MB_ICONEXCLAMATION);
3558                     HeapFree(GetProcessHeap(), 0, filename);
3559                     return TRUE;
3560                 }
3561                 CloseHandle(hf);
3562                 DeleteFileW(filename);
3563                 output = (LPWSTR *)GetWindowLongPtrW(hwnd, DWLP_USER);
3564                 *output = filename;
3565                 EndDialog(hwnd, IDOK);
3566                 return TRUE;
3567             }
3568             if(LOWORD(wparam) == IDCANCEL)
3569             {
3570                 EndDialog(hwnd, IDCANCEL);
3571                 return TRUE;
3572             }
3573         }
3574         return FALSE;
3575     }
3576     return FALSE;
3577 }
3578 
3579 static const WCHAR FILE_Port[] = {'F','I','L','E',':',0};
3580 
3581 LPWSTR WINAPI
3582 StartDocDlgW( HANDLE hPrinter, DOCINFOW *doc )
3583 {
3584     LPWSTR ret = NULL;
3585     DWORD len, attr, retDlg;
3586 
3587     FIXME("StartDocDlgW(%p, %p)\n", hPrinter, doc);
3588 
3589     if (doc->lpszOutput == NULL) /* Check whether default port is FILE: */
3590     {
3591         PRINTER_INFO_5W *pi5;
3592         GetPrinterW(hPrinter, 5, NULL, 0, &len);
3593         if(GetLastError() != ERROR_INSUFFICIENT_BUFFER)
3594             return NULL;
3595         pi5 = HeapAlloc(GetProcessHeap(), 0, len);
3596         if (!pi5)
3597         {
3598             ERR("Failed to allocate PRINTER_INFO_5W of %u bytes\n", len);
3599             return NULL;
3600         }
3601 
3602         GetPrinterW(hPrinter, 5, (LPBYTE)pi5, len, &len);
3603         if (!pi5->pPortName || wcsicmp(pi5->pPortName, FILE_Port))
3604         {
3605             HeapFree(GetProcessHeap(), 0, pi5);
3606             return NULL;
3607         }
3608         HeapFree(GetProcessHeap(), 0, pi5);
3609     }
3610 
3611     if (doc->lpszOutput == NULL || !wcsicmp(doc->lpszOutput, FILE_Port))
3612     {
3613         LPWSTR name;
3614 
3615         retDlg = DialogBoxParamW( hinstWinSpool,
3616                                   MAKEINTRESOURCEW(FILENAME_DIALOG),
3617                                   GetForegroundWindow(),
3618                                   file_dlg_proc,
3619                                  (LPARAM)&name );
3620 
3621         if ( retDlg == IDOK )
3622         {
3623             if (!(len = GetFullPathNameW(name, 0, NULL, NULL)))
3624             {
3625                 HeapFree(GetProcessHeap(), 0, name);
3626                 return NULL;
3627             }
3628             ret = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
3629             if (!ret)
3630             {
3631                 ERR("Failed to allocate path name buffer of %u bytes\n", len * sizeof(WCHAR));
3632                 HeapFree(GetProcessHeap(), 0, name);
3633                 return NULL;
3634             }
3635             GetFullPathNameW(name, len, ret, NULL);
3636             HeapFree(GetProcessHeap(), 0, name);
3637         }
3638         else if ( retDlg == 0 ) // FALSE, some type of error occurred.
3639         {
3640             ret = (LPWSTR)SP_ERROR;
3641         }
3642         else if ( retDlg == IDCANCEL )
3643         {
3644             SetLastError( ERROR_CANCELLED );
3645             ret = (LPWSTR)SP_APPABORT;
3646         }
3647         return ret;
3648     }
3649 
3650     if (!(len = GetFullPathNameW(doc->lpszOutput, 0, NULL, NULL)))
3651         return NULL;
3652 
3653     ret = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
3654     if (!ret)
3655     {
3656         ERR("Failed to allocate path name buffer of %u bytes\n", len * sizeof(WCHAR));
3657         return NULL;
3658     }
3659     GetFullPathNameW(doc->lpszOutput, len, ret, NULL);
3660 
3661     attr = GetFileAttributesW(ret);
3662     if (attr != INVALID_FILE_ATTRIBUTES && (attr & FILE_ATTRIBUTE_DIRECTORY))
3663     {
3664         HeapFree(GetProcessHeap(), 0, ret);
3665         ret = NULL;
3666     }
3667     return ret;
3668 }
3669 
3670 LPSTR WINAPI
3671 StartDocDlgA( HANDLE hPrinter, DOCINFOA *doc )
3672 {
3673     UNICODE_STRING usBuffer;
3674     DOCINFOW docW = { 0 };
3675     LPWSTR retW;
3676     LPWSTR docnameW = NULL, outputW = NULL, datatypeW = NULL;
3677     LPSTR ret = NULL;
3678 
3679     docW.cbSize = sizeof(docW);
3680     if (doc->lpszDocName)
3681     {
3682         docnameW = AsciiToUnicode(&usBuffer, doc->lpszDocName);
3683         if (!(docW.lpszDocName = docnameW)) goto failed;
3684     }
3685     if (doc->lpszOutput)
3686     {
3687         outputW = AsciiToUnicode(&usBuffer, doc->lpszOutput);
3688         if (!(docW.lpszOutput = outputW)) goto failed;
3689     }
3690     if (doc->lpszDatatype)
3691     {
3692         datatypeW = AsciiToUnicode(&usBuffer, doc->lpszDatatype);
3693         if (!(docW.lpszDatatype = datatypeW)) goto failed;
3694     }
3695     docW.fwType = doc->fwType;
3696 
3697     retW = StartDocDlgW(hPrinter, &docW);
3698 
3699     if (retW)
3700     {
3701         DWORD len = WideCharToMultiByte(CP_ACP, 0, retW, -1, NULL, 0, NULL, NULL);
3702         ret = HeapAlloc(GetProcessHeap(), 0, len);
3703         if (ret)
3704         {
3705             WideCharToMultiByte(CP_ACP, 0, retW, -1, ret, len, NULL, NULL);
3706         }
3707         else
3708         {
3709             ERR("Failed to allocate path name buffer of %u bytes\n", len);
3710         }
3711         HeapFree(GetProcessHeap(), 0, retW);
3712     }
3713 
3714 failed:
3715     if (datatypeW) HeapFree(GetProcessHeap(), 0, datatypeW);
3716     if (outputW) HeapFree(GetProcessHeap(), 0, outputW);
3717     if (docnameW) HeapFree(GetProcessHeap(), 0, docnameW);
3718 
3719     return ret;
3720 }
3721 
3722 DWORD WINAPI
3723 StartDocPrinterA(HANDLE hPrinter, DWORD Level, PBYTE pDocInfo)
3724 {
3725     DOC_INFO_1W wDocInfo1 = { 0 };
3726     DWORD cch;
3727     DWORD dwErrorCode;
3728     DWORD dwReturnValue = 0;
3729     PDOC_INFO_1A pDocInfo1 = (PDOC_INFO_1A)pDocInfo;
3730 
3731     TRACE("StartDocPrinterA(%p, %lu, %p)\n", hPrinter, Level, pDocInfo);
3732 
3733     // Only check the minimum required for accessing pDocInfo.
3734     // Additional sanity checks are done in StartDocPrinterW.
3735     if (!pDocInfo1)
3736     {
3737         dwErrorCode = ERROR_INVALID_PARAMETER;
3738         goto Cleanup;
3739     }
3740 
3741     if (Level != 1)
3742     {
3743         ERR("Level = %d, unsupported!\n", Level);
3744         dwErrorCode = ERROR_INVALID_LEVEL;
3745         goto Cleanup;
3746     }
3747 
3748     if (pDocInfo1->pDatatype)
3749     {
3750         // Convert pDocInfo1->pDatatype to a Unicode string wDocInfo1.pDatatype
3751         cch = strlen(pDocInfo1->pDatatype);
3752 
3753         wDocInfo1.pDatatype = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(WCHAR));
3754         if (!wDocInfo1.pDatatype)
3755         {
3756             dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
3757             ERR("HeapAlloc failed!\n");
3758             goto Cleanup;
3759         }
3760 
3761         MultiByteToWideChar(CP_ACP, 0, pDocInfo1->pDatatype, -1, wDocInfo1.pDatatype, cch + 1);
3762     }
3763 
3764     if (pDocInfo1->pDocName)
3765     {
3766         // Convert pDocInfo1->pDocName to a Unicode string wDocInfo1.pDocName
3767         cch = strlen(pDocInfo1->pDocName);
3768 
3769         wDocInfo1.pDocName = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(WCHAR));
3770         if (!wDocInfo1.pDocName)
3771         {
3772             dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
3773             ERR("HeapAlloc failed!\n");
3774             goto Cleanup;
3775         }
3776 
3777         MultiByteToWideChar(CP_ACP, 0, pDocInfo1->pDocName, -1, wDocInfo1.pDocName, cch + 1);
3778     }
3779 
3780     if (pDocInfo1->pOutputFile)
3781     {
3782         // Convert pDocInfo1->pOutputFile to a Unicode string wDocInfo1.pOutputFile
3783         cch = strlen(pDocInfo1->pOutputFile);
3784 
3785         wDocInfo1.pOutputFile = HeapAlloc(hProcessHeap, 0, (cch + 1) * sizeof(WCHAR));
3786         if (!wDocInfo1.pOutputFile)
3787         {
3788             dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
3789             ERR("HeapAlloc failed!\n");
3790             goto Cleanup;
3791         }
3792 
3793         MultiByteToWideChar(CP_ACP, 0, pDocInfo1->pOutputFile, -1, wDocInfo1.pOutputFile, cch + 1);
3794     }
3795 
3796     dwReturnValue = StartDocPrinterW(hPrinter, Level, (PBYTE)&wDocInfo1);
3797     dwErrorCode = GetLastError();
3798 
3799 Cleanup:
3800     if (wDocInfo1.pDatatype)
3801         HeapFree(hProcessHeap, 0, wDocInfo1.pDatatype);
3802 
3803     if (wDocInfo1.pDocName)
3804         HeapFree(hProcessHeap, 0, wDocInfo1.pDocName);
3805 
3806     if (wDocInfo1.pOutputFile)
3807         HeapFree(hProcessHeap, 0, wDocInfo1.pOutputFile);
3808 
3809     SetLastError(dwErrorCode);
3810     return dwReturnValue;
3811 }
3812 
3813 DWORD WINAPI
3814 StartDocPrinterW(HANDLE hPrinter, DWORD Level, PBYTE pDocInfo)
3815 {
3816     DWORD cbAddJobInfo1;
3817     DWORD cbNeeded;
3818     DWORD dwErrorCode;
3819     DWORD dwReturnValue = 0;
3820     PADDJOB_INFO_1W pAddJobInfo1 = NULL;
3821     PDOC_INFO_1W pDocInfo1 = (PDOC_INFO_1W)pDocInfo;
3822     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
3823 
3824     TRACE("StartDocPrinterW(%p, %lu, %p)\n", hPrinter, Level, pDocInfo);
3825 
3826     // Sanity checks.
3827     if (!pHandle)
3828     {
3829         dwErrorCode = ERROR_INVALID_HANDLE;
3830         goto Cleanup;
3831     }
3832 
3833     if (!pDocInfo1)
3834     {
3835         dwErrorCode = ERROR_INVALID_PARAMETER;
3836         goto Cleanup;
3837     }
3838 
3839     if (Level != 1)
3840     {
3841         ERR("Level = %d, unsupported!\n", Level);
3842         dwErrorCode = ERROR_INVALID_LEVEL;
3843         goto Cleanup;
3844     }
3845 
3846     if (pHandle->bStartedDoc)
3847     {
3848         dwErrorCode = ERROR_INVALID_PRINTER_STATE;
3849         goto Cleanup;
3850     }
3851 
3852     // Check if we want to redirect output into a file.
3853     if (pDocInfo1->pOutputFile)
3854     {
3855         // Do a StartDocPrinter RPC call in this case.
3856         dwErrorCode = _StartDocPrinterWithRPC(pHandle, pDocInfo1);
3857     }
3858     else
3859     {
3860         // Allocate memory for the ADDJOB_INFO_1W structure and a path.
3861         cbAddJobInfo1 = sizeof(ADDJOB_INFO_1W) + MAX_PATH * sizeof(WCHAR);
3862         pAddJobInfo1 = HeapAlloc(hProcessHeap, 0, cbAddJobInfo1);
3863         if (!pAddJobInfo1)
3864         {
3865             dwErrorCode = ERROR_NOT_ENOUGH_MEMORY;
3866             ERR("HeapAlloc failed!\n");
3867             goto Cleanup;
3868         }
3869 
3870         // Try to add a new job.
3871         // This only succeeds if the printer is set to do spooled printing.
3872         if (AddJobW((HANDLE)pHandle, 1, (PBYTE)pAddJobInfo1, cbAddJobInfo1, &cbNeeded))
3873         {
3874             // Do spooled printing.
3875             dwErrorCode = _StartDocPrinterSpooled(pHandle, pDocInfo1, pAddJobInfo1);
3876         }
3877         else if (GetLastError() == ERROR_INVALID_ACCESS)
3878         {
3879             // ERROR_INVALID_ACCESS is returned when the printer is set to do direct printing.
3880             // In this case, we do a StartDocPrinter RPC call.
3881             dwErrorCode = _StartDocPrinterWithRPC(pHandle, pDocInfo1);
3882         }
3883         else
3884         {
3885             dwErrorCode = GetLastError();
3886             ERR("AddJobW failed with error %lu!\n", dwErrorCode);
3887             goto Cleanup;
3888         }
3889     }
3890 
3891     if (dwErrorCode == ERROR_SUCCESS)
3892     {
3893         pHandle->bStartedDoc = TRUE;
3894         dwReturnValue = pHandle->dwJobID;
3895         if ( !pHandle->bTrayIcon )
3896         {
3897             UpdateTrayIcon( hPrinter, pHandle->dwJobID );
3898         }
3899     }
3900 
3901 Cleanup:
3902     if (pAddJobInfo1)
3903         HeapFree(hProcessHeap, 0, pAddJobInfo1);
3904 
3905     SetLastError(dwErrorCode);
3906     return dwReturnValue;
3907 }
3908 
3909 BOOL WINAPI
3910 StartPagePrinter(HANDLE hPrinter)
3911 {
3912     DWORD dwErrorCode;
3913     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
3914 
3915     TRACE("StartPagePrinter(%p)\n", hPrinter);
3916 
3917     // Sanity checks.
3918     if (!pHandle)
3919     {
3920         dwErrorCode = ERROR_INVALID_HANDLE;
3921         goto Cleanup;
3922     }
3923 
3924     // Do the RPC call
3925     RpcTryExcept
3926     {
3927         dwErrorCode = _RpcStartPagePrinter(pHandle->hPrinter);
3928     }
3929     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
3930     {
3931         dwErrorCode = RpcExceptionCode();
3932         ERR("_RpcStartPagePrinter failed with exception code %lu!\n", dwErrorCode);
3933     }
3934     RpcEndExcept;
3935 
3936 Cleanup:
3937     SetLastError(dwErrorCode);
3938     return (dwErrorCode == ERROR_SUCCESS);
3939 }
3940 
3941 BOOL WINAPI
3942 WritePrinter(HANDLE hPrinter, PVOID pBuf, DWORD cbBuf, PDWORD pcWritten)
3943 {
3944     DWORD dwErrorCode;
3945     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hPrinter;
3946 
3947     TRACE("WritePrinter(%p, %p, %lu, %p)\n", hPrinter, pBuf, cbBuf, pcWritten);
3948 
3949     // Sanity checks.
3950     if (!pHandle)
3951     {
3952         dwErrorCode = ERROR_INVALID_HANDLE;
3953         goto Cleanup;
3954     }
3955 
3956     if (!pHandle->bStartedDoc)
3957     {
3958         dwErrorCode = ERROR_SPL_NO_STARTDOC;
3959         goto Cleanup;
3960     }
3961 
3962     if (pHandle->hSPLFile != INVALID_HANDLE_VALUE)
3963     {
3964         // Write to the spool file. This doesn't need an RPC request.
3965         if (!WriteFile(pHandle->hSPLFile, pBuf, cbBuf, pcWritten, NULL))
3966         {
3967             dwErrorCode = GetLastError();
3968             ERR("WriteFile failed with error %lu!\n", dwErrorCode);
3969             goto Cleanup;
3970         }
3971 
3972         dwErrorCode = ERROR_SUCCESS;
3973     }
3974     else
3975     {
3976         // TODO: This case (for direct printing or remote printing) has bad performance if multiple small-sized WritePrinter calls are performed.
3977         // We may increase performance by writing into a buffer and only doing a single RPC call when the buffer is full.
3978 
3979         // Do the RPC call
3980         RpcTryExcept
3981         {
3982             dwErrorCode = _RpcWritePrinter(pHandle->hPrinter, pBuf, cbBuf, pcWritten);
3983         }
3984         RpcExcept(EXCEPTION_EXECUTE_HANDLER)
3985         {
3986             dwErrorCode = RpcExceptionCode();
3987             ERR("_RpcWritePrinter failed with exception code %lu!\n", dwErrorCode);
3988         }
3989         RpcEndExcept;
3990     }
3991 
3992 Cleanup:
3993     SetLastError(dwErrorCode);
3994     return (dwErrorCode == ERROR_SUCCESS);
3995 }
3996 
3997 BOOL WINAPI
3998 XcvDataW(HANDLE hXcv, PCWSTR pszDataName, PBYTE pInputData, DWORD cbInputData, PBYTE pOutputData, DWORD cbOutputData, PDWORD pcbOutputNeeded, PDWORD pdwStatus)
3999 {
4000     DWORD dwErrorCode, Bogus = 0;
4001     PSPOOLER_HANDLE pHandle = (PSPOOLER_HANDLE)hXcv;
4002 
4003     TRACE("XcvDataW(%p, %S, %p, %lu, %p, %lu, %p, %p)\n", hXcv, pszDataName, pInputData, cbInputData, pOutputData, cbOutputData, pcbOutputNeeded, pdwStatus);
4004 
4005     if ( pcbOutputNeeded == NULL )
4006     {
4007         dwErrorCode = ERROR_INVALID_PARAMETER;
4008         goto Cleanup;
4009     }
4010 
4011     // Sanity checks.
4012     if (!pHandle) // ( IntProtectHandle( hXcv, FALSE ) )
4013     {
4014         dwErrorCode = ERROR_INVALID_HANDLE;
4015         goto Cleanup;
4016     }
4017 
4018     //
4019     // Do fixups.
4020     //
4021     if ( pInputData == NULL )
4022     {
4023         if ( !cbInputData )
4024         {
4025              pInputData = (PBYTE)&Bogus;
4026         }
4027     }
4028 
4029     if ( pOutputData == NULL )
4030     {
4031         if ( !cbOutputData )
4032         {
4033             pOutputData = (PBYTE)&Bogus;
4034         }
4035     }
4036 
4037     // Do the RPC call
4038     RpcTryExcept
4039     {
4040         dwErrorCode = _RpcXcvData( pHandle->hPrinter,
4041                                    pszDataName,
4042                                    pInputData,
4043                                    cbInputData,
4044                                    pOutputData,
4045                                    cbOutputData,
4046                                    pcbOutputNeeded,
4047                                    pdwStatus );
4048     }
4049     RpcExcept(EXCEPTION_EXECUTE_HANDLER)
4050     {
4051         dwErrorCode = RpcExceptionCode();
4052         ERR("_RpcXcvData failed with exception code %lu!\n", dwErrorCode);
4053     }
4054     RpcEndExcept;
4055 
4056     //IntUnprotectHandle( hXcv );
4057 
4058 Cleanup:
4059     SetLastError(dwErrorCode);
4060     return (dwErrorCode == ERROR_SUCCESS);
4061 }
4062