• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "printing/backend/win_helper.h"
6 
7 #include <algorithm>
8 
9 #include "base/file_version_info.h"
10 #include "base/files/file_path.h"
11 #include "base/logging.h"
12 #include "base/memory/scoped_ptr.h"
13 #include "base/strings/utf_string_conversions.h"
14 #include "printing/backend/print_backend.h"
15 #include "printing/backend/print_backend_consts.h"
16 #include "printing/backend/printing_info_win.h"
17 
18 namespace {
19 
20 typedef HRESULT (WINAPI* PTOpenProviderProc)(PCWSTR printer_name,
21                                              DWORD version,
22                                              HPTPROVIDER* provider);
23 
24 typedef HRESULT (WINAPI* PTGetPrintCapabilitiesProc)(HPTPROVIDER provider,
25                                                      IStream* print_ticket,
26                                                      IStream* capabilities,
27                                                      BSTR* error_message);
28 
29 typedef HRESULT (WINAPI* PTConvertDevModeToPrintTicketProc)(
30     HPTPROVIDER provider,
31     ULONG devmode_size_in_bytes,
32     PDEVMODE devmode,
33     EPrintTicketScope scope,
34     IStream* print_ticket);
35 
36 typedef HRESULT (WINAPI* PTConvertPrintTicketToDevModeProc)(
37     HPTPROVIDER provider,
38     IStream* print_ticket,
39     EDefaultDevmodeType base_devmode_type,
40     EPrintTicketScope scope,
41     ULONG* devmode_byte_count,
42     PDEVMODE* devmode,
43     BSTR* error_message);
44 
45 typedef HRESULT (WINAPI* PTMergeAndValidatePrintTicketProc)(
46     HPTPROVIDER provider,
47     IStream* base_ticket,
48     IStream* delta_ticket,
49     EPrintTicketScope scope,
50     IStream* result_ticket,
51     BSTR* error_message);
52 
53 typedef HRESULT (WINAPI* PTReleaseMemoryProc)(PVOID buffer);
54 
55 typedef HRESULT (WINAPI* PTCloseProviderProc)(HPTPROVIDER provider);
56 
57 typedef HRESULT (WINAPI* StartXpsPrintJobProc)(
58     const LPCWSTR printer_name,
59     const LPCWSTR job_name,
60     const LPCWSTR output_file_name,
61     HANDLE progress_event,
62     HANDLE completion_event,
63     UINT8* printable_pages_on,
64     UINT32 printable_pages_on_count,
65     IXpsPrintJob** xps_print_job,
66     IXpsPrintJobStream** document_stream,
67     IXpsPrintJobStream** print_ticket_stream);
68 
69 PTOpenProviderProc g_open_provider_proc = NULL;
70 PTGetPrintCapabilitiesProc g_get_print_capabilities_proc = NULL;
71 PTConvertDevModeToPrintTicketProc g_convert_devmode_to_print_ticket_proc = NULL;
72 PTConvertPrintTicketToDevModeProc g_convert_print_ticket_to_devmode_proc = NULL;
73 PTMergeAndValidatePrintTicketProc g_merge_and_validate_print_ticket_proc = NULL;
74 PTReleaseMemoryProc g_release_memory_proc = NULL;
75 PTCloseProviderProc g_close_provider_proc = NULL;
76 StartXpsPrintJobProc g_start_xps_print_job_proc = NULL;
77 
78 }  // namespace
79 
80 
81 namespace printing {
82 
Init()83 bool XPSModule::Init() {
84   static bool initialized = InitImpl();
85   return initialized;
86 }
87 
InitImpl()88 bool XPSModule::InitImpl() {
89   HMODULE prntvpt_module = LoadLibrary(L"prntvpt.dll");
90   if (prntvpt_module == NULL)
91     return false;
92   g_open_provider_proc = reinterpret_cast<PTOpenProviderProc>(
93       GetProcAddress(prntvpt_module, "PTOpenProvider"));
94   if (!g_open_provider_proc) {
95     NOTREACHED();
96     return false;
97   }
98   g_get_print_capabilities_proc = reinterpret_cast<PTGetPrintCapabilitiesProc>(
99       GetProcAddress(prntvpt_module, "PTGetPrintCapabilities"));
100   if (!g_get_print_capabilities_proc) {
101     NOTREACHED();
102     return false;
103   }
104   g_convert_devmode_to_print_ticket_proc =
105       reinterpret_cast<PTConvertDevModeToPrintTicketProc>(
106           GetProcAddress(prntvpt_module, "PTConvertDevModeToPrintTicket"));
107   if (!g_convert_devmode_to_print_ticket_proc) {
108     NOTREACHED();
109     return false;
110   }
111   g_convert_print_ticket_to_devmode_proc =
112       reinterpret_cast<PTConvertPrintTicketToDevModeProc>(
113           GetProcAddress(prntvpt_module, "PTConvertPrintTicketToDevMode"));
114   if (!g_convert_print_ticket_to_devmode_proc) {
115     NOTREACHED();
116     return false;
117   }
118   g_merge_and_validate_print_ticket_proc =
119       reinterpret_cast<PTMergeAndValidatePrintTicketProc>(
120           GetProcAddress(prntvpt_module, "PTMergeAndValidatePrintTicket"));
121   if (!g_merge_and_validate_print_ticket_proc) {
122     NOTREACHED();
123     return false;
124   }
125   g_release_memory_proc =
126       reinterpret_cast<PTReleaseMemoryProc>(
127           GetProcAddress(prntvpt_module, "PTReleaseMemory"));
128   if (!g_release_memory_proc) {
129     NOTREACHED();
130     return false;
131   }
132   g_close_provider_proc =
133       reinterpret_cast<PTCloseProviderProc>(
134           GetProcAddress(prntvpt_module, "PTCloseProvider"));
135   if (!g_close_provider_proc) {
136     NOTREACHED();
137     return false;
138   }
139   return true;
140 }
141 
OpenProvider(const base::string16 & printer_name,DWORD version,HPTPROVIDER * provider)142 HRESULT XPSModule::OpenProvider(const base::string16& printer_name,
143                                 DWORD version,
144                                 HPTPROVIDER* provider) {
145   return g_open_provider_proc(printer_name.c_str(), version, provider);
146 }
147 
GetPrintCapabilities(HPTPROVIDER provider,IStream * print_ticket,IStream * capabilities,BSTR * error_message)148 HRESULT XPSModule::GetPrintCapabilities(HPTPROVIDER provider,
149                                         IStream* print_ticket,
150                                         IStream* capabilities,
151                                         BSTR* error_message) {
152   return g_get_print_capabilities_proc(provider,
153                                        print_ticket,
154                                        capabilities,
155                                        error_message);
156 }
157 
ConvertDevModeToPrintTicket(HPTPROVIDER provider,ULONG devmode_size_in_bytes,PDEVMODE devmode,EPrintTicketScope scope,IStream * print_ticket)158 HRESULT XPSModule::ConvertDevModeToPrintTicket(HPTPROVIDER provider,
159                                                ULONG devmode_size_in_bytes,
160                                                PDEVMODE devmode,
161                                                EPrintTicketScope scope,
162                                                IStream* print_ticket) {
163   return g_convert_devmode_to_print_ticket_proc(provider,
164                                                 devmode_size_in_bytes,
165                                                 devmode,
166                                                 scope,
167                                                 print_ticket);
168 }
169 
ConvertPrintTicketToDevMode(HPTPROVIDER provider,IStream * print_ticket,EDefaultDevmodeType base_devmode_type,EPrintTicketScope scope,ULONG * devmode_byte_count,PDEVMODE * devmode,BSTR * error_message)170 HRESULT XPSModule::ConvertPrintTicketToDevMode(
171     HPTPROVIDER provider,
172     IStream* print_ticket,
173     EDefaultDevmodeType base_devmode_type,
174     EPrintTicketScope scope,
175     ULONG* devmode_byte_count,
176     PDEVMODE* devmode,
177     BSTR* error_message) {
178   return g_convert_print_ticket_to_devmode_proc(provider,
179                                                 print_ticket,
180                                                 base_devmode_type,
181                                                 scope,
182                                                 devmode_byte_count,
183                                                 devmode,
184                                                 error_message);
185 }
186 
MergeAndValidatePrintTicket(HPTPROVIDER provider,IStream * base_ticket,IStream * delta_ticket,EPrintTicketScope scope,IStream * result_ticket,BSTR * error_message)187 HRESULT XPSModule::MergeAndValidatePrintTicket(HPTPROVIDER provider,
188                                                IStream* base_ticket,
189                                                IStream* delta_ticket,
190                                                EPrintTicketScope scope,
191                                                IStream* result_ticket,
192                                                BSTR* error_message) {
193   return g_merge_and_validate_print_ticket_proc(provider,
194                                                 base_ticket,
195                                                 delta_ticket,
196                                                 scope,
197                                                 result_ticket,
198                                                 error_message);
199 }
200 
ReleaseMemory(PVOID buffer)201 HRESULT XPSModule::ReleaseMemory(PVOID buffer) {
202   return g_release_memory_proc(buffer);
203 }
204 
CloseProvider(HPTPROVIDER provider)205 HRESULT XPSModule::CloseProvider(HPTPROVIDER provider) {
206   return g_close_provider_proc(provider);
207 }
208 
ScopedXPSInitializer()209 ScopedXPSInitializer::ScopedXPSInitializer() : initialized_(false) {
210   if (!XPSModule::Init())
211     return;
212   // Calls to XPS APIs typically require the XPS provider to be opened with
213   // PTOpenProvider. PTOpenProvider calls CoInitializeEx with
214   // COINIT_MULTITHREADED. We have seen certain buggy HP printer driver DLLs
215   // that call CoInitializeEx with COINIT_APARTMENTTHREADED in the context of
216   // PTGetPrintCapabilities. This call fails but the printer driver calls
217   // CoUninitialize anyway. This results in the apartment being torn down too
218   // early and the msxml DLL being unloaded which in turn causes code in
219   // unidrvui.dll to have a dangling pointer to an XML document which causes a
220   // crash. To protect ourselves from such drivers we make sure we always have
221   // an extra CoInitialize (calls to CoInitialize/CoUninitialize are
222   // refcounted).
223   HRESULT hr = CoInitializeEx(NULL, COINIT_MULTITHREADED);
224   // If this succeeded we are done because the PTOpenProvider call will provide
225   // the extra refcount on the apartment. If it failed because someone already
226   // called CoInitializeEx with COINIT_APARTMENTTHREADED, we try the other model
227   // to provide the additional refcount (since we don't know which model buggy
228   // printer drivers will use).
229   if (!SUCCEEDED(hr))
230     hr = CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
231   DCHECK(SUCCEEDED(hr));
232   initialized_ = true;
233 }
234 
~ScopedXPSInitializer()235 ScopedXPSInitializer::~ScopedXPSInitializer() {
236   if (initialized_)
237     CoUninitialize();
238   initialized_ = false;
239 }
240 
Init()241 bool XPSPrintModule::Init() {
242   static bool initialized = InitImpl();
243   return initialized;
244 }
245 
InitImpl()246 bool XPSPrintModule::InitImpl() {
247   HMODULE xpsprint_module = LoadLibrary(L"xpsprint.dll");
248   if (xpsprint_module == NULL)
249     return false;
250   g_start_xps_print_job_proc = reinterpret_cast<StartXpsPrintJobProc>(
251       GetProcAddress(xpsprint_module, "StartXpsPrintJob"));
252   if (!g_start_xps_print_job_proc) {
253     NOTREACHED();
254     return false;
255   }
256   return true;
257 }
258 
StartXpsPrintJob(const LPCWSTR printer_name,const LPCWSTR job_name,const LPCWSTR output_file_name,HANDLE progress_event,HANDLE completion_event,UINT8 * printable_pages_on,UINT32 printable_pages_on_count,IXpsPrintJob ** xps_print_job,IXpsPrintJobStream ** document_stream,IXpsPrintJobStream ** print_ticket_stream)259 HRESULT XPSPrintModule::StartXpsPrintJob(
260     const LPCWSTR printer_name,
261     const LPCWSTR job_name,
262     const LPCWSTR output_file_name,
263     HANDLE progress_event,
264     HANDLE completion_event,
265     UINT8* printable_pages_on,
266     UINT32 printable_pages_on_count,
267     IXpsPrintJob** xps_print_job,
268     IXpsPrintJobStream** document_stream,
269     IXpsPrintJobStream** print_ticket_stream) {
270   return g_start_xps_print_job_proc(printer_name,
271                                     job_name,
272                                     output_file_name,
273                                     progress_event,
274                                     completion_event,
275                                     printable_pages_on,
276                                     printable_pages_on_count,
277                                     xps_print_job,
278                                     document_stream,
279                                     print_ticket_stream);
280 }
281 
InitBasicPrinterInfo(HANDLE printer,PrinterBasicInfo * printer_info)282 bool InitBasicPrinterInfo(HANDLE printer, PrinterBasicInfo* printer_info) {
283   DCHECK(printer);
284   DCHECK(printer_info);
285   if (!printer)
286     return false;
287 
288   PrinterInfo2 info_2;
289   if (!info_2.Init(printer))
290     return false;
291 
292   printer_info->printer_name = WideToUTF8(info_2.get()->pPrinterName);
293   if (info_2.get()->pComment)
294     printer_info->printer_description = WideToUTF8(info_2.get()->pComment);
295   if (info_2.get()->pLocation)
296     printer_info->options[kLocationTagName] =
297         WideToUTF8(info_2.get()->pLocation);
298   if (info_2.get()->pDriverName)
299     printer_info->options[kDriverNameTagName] =
300         WideToUTF8(info_2.get()->pDriverName);
301   printer_info->printer_status = info_2.get()->Status;
302 
303   std::string driver_info = GetDriverInfo(printer);
304   if (!driver_info.empty())
305     printer_info->options[kDriverInfoTagName] = driver_info;
306   return true;
307 }
308 
GetDriverInfo(HANDLE printer)309 std::string GetDriverInfo(HANDLE printer) {
310   DCHECK(printer);
311   std::string driver_info;
312 
313   if (!printer)
314     return driver_info;
315 
316   DriverInfo6 info_6;
317   if (!info_6.Init(printer))
318     return driver_info;
319 
320   std::string info[4];
321   if (info_6.get()->pName)
322     info[0] = WideToUTF8(info_6.get()->pName);
323 
324   if (info_6.get()->pDriverPath) {
325     scoped_ptr<FileVersionInfo> version_info(
326         FileVersionInfo::CreateFileVersionInfo(
327             base::FilePath(info_6.get()->pDriverPath)));
328     if (version_info.get()) {
329       info[1] = WideToUTF8(version_info->file_version());
330       info[2] = WideToUTF8(version_info->product_name());
331       info[3] = WideToUTF8(version_info->product_version());
332     }
333   }
334 
335   for (size_t i = 0; i < arraysize(info); ++i) {
336     std::replace(info[i].begin(), info[i].end(), ';', ',');
337     driver_info.append(info[i]);
338     if (i < arraysize(info) - 1)
339       driver_info.append(";");
340   }
341   return driver_info;
342 }
343 
344 }  // namespace printing
345