FreeRDP
Loading...
Searching...
No Matches
printer_main.c
1
24#include <freerdp/config.h>
25
26#include <stdio.h>
27#include <stdlib.h>
28#include <string.h>
29
30#include <winpr/crt.h>
31#include <winpr/assert.h>
32#include <winpr/string.h>
33#include <winpr/synch.h>
34#include <winpr/thread.h>
35#include <winpr/stream.h>
36#include <winpr/interlocked.h>
37#include <winpr/file.h>
38#include <winpr/path.h>
39#include <winpr/print.h>
40
41#include <freerdp/channels/rdpdr.h>
42#include <freerdp/crypto/crypto.h>
43#include <freerdp/freerdp.h>
44
45#include "../printer.h"
46
47#include <freerdp/client/printer.h>
48
49#include <freerdp/channels/log.h>
50
51#define TAG CHANNELS_TAG("printer.client")
52
53typedef struct
54{
55 DEVICE device;
56
57 rdpPrinter* printer;
58
59 WINPR_PSLIST_HEADER pIrpList;
60
61 HANDLE event;
62 HANDLE stopEvent;
63
64 HANDLE thread;
65 rdpContext* rdpcontext;
66 char port[64];
67 BOOL async;
68} PRINTER_DEVICE;
69
70typedef enum
71{
72 PRN_CONF_PORT = 0,
73 PRN_CONF_PNP = 1,
74 PRN_CONF_DRIVER = 2,
75 PRN_CONF_DATA = 3
76} prn_conf_t;
77
78static const char* filemap[] = { "PortDosName", "PnPName", "DriverName",
79 "CachedPrinterConfigData" };
80
81WINPR_ATTR_MALLOC(free, 1)
82static char* get_printer_hash(const WCHAR* name, size_t length)
83{
84 BYTE hash[WINPR_SHA256_DIGEST_LENGTH] = { 0 };
85
86 if (!winpr_Digest(WINPR_MD_SHA256, name, length, hash, sizeof(hash)))
87 return NULL;
88
89 return winpr_BinToHexString(hash, sizeof(hash), FALSE);
90}
91
92WINPR_ATTR_MALLOC(free, 1)
93static char* get_printer_config_path(const rdpSettings* settings, const WCHAR* name, size_t length)
94{
95 char* config = NULL;
96 const char* path = freerdp_settings_get_string(settings, FreeRDP_ConfigPath);
97 char* dir = GetCombinedPath(path, "printers");
98 if (!dir)
99 return NULL;
100 char* bname = get_printer_hash(name, length);
101 if (!bname)
102 goto fail;
103 config = GetCombinedPath(dir, bname);
104
105 if (config && !winpr_PathFileExists(config))
106 {
107 if (!winpr_PathMakePath(config, NULL))
108 {
109 free(config);
110 config = NULL;
111 }
112 }
113
114fail:
115 free(dir);
116 free(bname);
117 return config;
118}
119
120static BOOL printer_write_setting(const char* path, prn_conf_t type, const void* data,
121 size_t length)
122{
123 if (!path)
124 return FALSE;
125
126 DWORD written = 0;
127 BOOL rc = FALSE;
128 HANDLE file = NULL;
129 char* base64 = NULL;
130 const char* name = filemap[type];
131 char* abs = GetCombinedPath(path, name);
132
133 if (!abs || (length > INT32_MAX))
134 {
135 free(abs);
136 return FALSE;
137 }
138
139 file =
140 winpr_CreateFile(abs, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
141 free(abs);
142
143 if (file == INVALID_HANDLE_VALUE)
144 return FALSE;
145
146 if (length > 0)
147 {
148 base64 = crypto_base64_encode(data, length);
149
150 if (!base64)
151 goto fail;
152
153 /* base64 char represents 6bit -> 4*(n/3) is the length which is
154 * always smaller than 2*n */
155 const size_t b64len = strnlen(base64, 2 * length);
156 rc = WriteFile(file, base64, (UINT32)b64len, &written, NULL);
157
158 if (b64len != written)
159 rc = FALSE;
160 }
161 else
162 rc = TRUE;
163
164fail:
165 (void)CloseHandle(file);
166 free(base64);
167 return rc;
168}
169
170static BOOL printer_config_valid(const char* path)
171{
172 if (!path)
173 return FALSE;
174
175 if (!winpr_PathFileExists(path))
176 return FALSE;
177
178 return TRUE;
179}
180
181static BOOL printer_read_setting(const char* path, prn_conf_t type, void** data, UINT32* length)
182{
183 DWORD lowSize = 0;
184 DWORD highSize = 0;
185 DWORD read = 0;
186 BOOL rc = FALSE;
187 char* fdata = NULL;
188 const char* name = filemap[type];
189
190 switch (type)
191 {
192 case PRN_CONF_DATA:
193 break;
194 default:
195 WLog_DBG(TAG, "Printer option %s ignored", name);
196 return FALSE;
197 }
198
199 char* abs = GetCombinedPath(path, name);
200 if (!abs)
201 return FALSE;
202
203 HANDLE file =
204 winpr_CreateFile(abs, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
205 free(abs);
206
207 if (file == INVALID_HANDLE_VALUE)
208 return FALSE;
209
210 lowSize = GetFileSize(file, &highSize);
211
212 if ((lowSize == INVALID_FILE_SIZE) || (highSize != 0))
213 goto fail;
214
215 if (lowSize != 0)
216 {
217 fdata = malloc(lowSize);
218
219 if (!fdata)
220 goto fail;
221
222 rc = ReadFile(file, fdata, lowSize, &read, NULL);
223
224 if (lowSize != read)
225 rc = FALSE;
226 }
227
228fail:
229 (void)CloseHandle(file);
230
231 if (rc && (lowSize <= INT_MAX))
232 {
233 size_t blen = 0;
234 crypto_base64_decode(fdata, lowSize, (BYTE**)data, &blen);
235
236 if (*data && (blen > 0))
237 *length = (UINT32)blen;
238 else
239 {
240 rc = FALSE;
241 *length = 0;
242 }
243 }
244 else
245 {
246 *length = 0;
247 *data = NULL;
248 }
249
250 free(fdata);
251 return rc;
252}
253
254static BOOL printer_save_to_config(const rdpSettings* settings, const char* PortDosName,
255 size_t PortDosNameLen, const WCHAR* PnPName, size_t PnPNameLen,
256 const WCHAR* DriverName, size_t DriverNameLen,
257 const WCHAR* PrinterName, size_t PrintNameLen,
258 const BYTE* CachedPrinterConfigData, size_t CacheFieldsLen)
259{
260 BOOL rc = FALSE;
261 char* path = get_printer_config_path(settings, PrinterName, PrintNameLen);
262
263 if (!path)
264 goto fail;
265
266 if (!printer_write_setting(path, PRN_CONF_PORT, PortDosName, PortDosNameLen))
267 goto fail;
268
269 if (!printer_write_setting(path, PRN_CONF_PNP, PnPName, PnPNameLen))
270 goto fail;
271
272 if (!printer_write_setting(path, PRN_CONF_DRIVER, DriverName, DriverNameLen))
273 goto fail;
274
275 if (!printer_write_setting(path, PRN_CONF_DATA, CachedPrinterConfigData, CacheFieldsLen))
276 goto fail;
277
278fail:
279 free(path);
280 return rc;
281}
282
283static BOOL printer_update_to_config(const rdpSettings* settings, const WCHAR* name, size_t length,
284 const BYTE* data, size_t datalen)
285{
286 BOOL rc = FALSE;
287 char* path = get_printer_config_path(settings, name, length);
288 rc = printer_write_setting(path, PRN_CONF_DATA, data, datalen);
289 free(path);
290 return rc;
291}
292
293static BOOL printer_remove_config(const rdpSettings* settings, const WCHAR* name, size_t length)
294{
295 BOOL rc = FALSE;
296 char* path = get_printer_config_path(settings, name, length);
297
298 if (!printer_config_valid(path))
299 goto fail;
300
301 rc = winpr_RemoveDirectory(path);
302fail:
303 free(path);
304 return rc;
305}
306
307static BOOL printer_move_config(const rdpSettings* settings, const WCHAR* oldName, size_t oldLength,
308 const WCHAR* newName, size_t newLength)
309{
310 BOOL rc = FALSE;
311 char* oldPath = get_printer_config_path(settings, oldName, oldLength);
312 char* newPath = get_printer_config_path(settings, newName, newLength);
313
314 if (printer_config_valid(oldPath))
315 rc = winpr_MoveFile(oldPath, newPath);
316
317 free(oldPath);
318 free(newPath);
319 return rc;
320}
321
322static BOOL printer_load_from_config(const rdpSettings* settings, rdpPrinter* printer,
323 PRINTER_DEVICE* printer_dev)
324{
325 BOOL res = FALSE;
326 WCHAR* wname = NULL;
327 size_t wlen = 0;
328 char* path = NULL;
329 UINT32 flags = 0;
330 void* DriverName = NULL;
331 UINT32 DriverNameLen = 0;
332 void* PnPName = NULL;
333 UINT32 PnPNameLen = 0;
334 void* CachedPrinterConfigData = NULL;
335 UINT32 CachedFieldsLen = 0;
336 UINT32 PrinterNameLen = 0;
337
338 if (!settings || !printer || !printer->name)
339 return FALSE;
340
341 wname = ConvertUtf8ToWCharAlloc(printer->name, &wlen);
342
343 if (!wname)
344 goto fail;
345
346 wlen++;
347 path = get_printer_config_path(settings, wname, wlen * sizeof(WCHAR));
348 {
349 const size_t plen = wlen * sizeof(WCHAR);
350 if (plen > UINT32_MAX)
351 goto fail;
352 PrinterNameLen = (UINT32)plen;
353 }
354
355 if (!path)
356 goto fail;
357
358 if (printer->is_default)
359 flags |= RDPDR_PRINTER_ANNOUNCE_FLAG_DEFAULTPRINTER;
360
361 if (!printer_read_setting(path, PRN_CONF_PNP, &PnPName, &PnPNameLen))
362 {
363 }
364
365 if (!printer_read_setting(path, PRN_CONF_DRIVER, &DriverName, &DriverNameLen))
366 {
367 size_t len = 0;
368 DriverName = ConvertUtf8ToWCharAlloc(printer->driver, &len);
369 if (!DriverName)
370 goto fail;
371 const size_t dlen = (len + 1) * sizeof(WCHAR);
372 if (dlen > UINT32_MAX)
373 goto fail;
374 DriverNameLen = (UINT32)dlen;
375 }
376
377 if (!printer_read_setting(path, PRN_CONF_DATA, &CachedPrinterConfigData, &CachedFieldsLen))
378 {
379 }
380
381 Stream_SetPosition(printer_dev->device.data, 0);
382
383 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, 24))
384 goto fail;
385
386 Stream_Write_UINT32(printer_dev->device.data, flags);
387 Stream_Write_UINT32(printer_dev->device.data, 0); /* CodePage, reserved */
388 Stream_Write_UINT32(printer_dev->device.data, PnPNameLen); /* PnPNameLen */
389 Stream_Write_UINT32(printer_dev->device.data, DriverNameLen);
390 Stream_Write_UINT32(printer_dev->device.data, PrinterNameLen);
391 Stream_Write_UINT32(printer_dev->device.data, CachedFieldsLen);
392
393 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, PnPNameLen))
394 goto fail;
395
396 if (PnPNameLen > 0)
397 Stream_Write(printer_dev->device.data, PnPName, PnPNameLen);
398
399 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, DriverNameLen))
400 goto fail;
401
402 Stream_Write(printer_dev->device.data, DriverName, DriverNameLen);
403
404 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, PrinterNameLen))
405 goto fail;
406
407 union
408 {
409 char c[2];
410 WCHAR w;
411 } backslash;
412 backslash.c[0] = '\\';
413 backslash.c[1] = '\0';
414
415 for (WCHAR* wptr = wname; (wptr = _wcschr(wptr, backslash.w));)
416 *wptr = L'_';
417 Stream_Write(printer_dev->device.data, wname, PrinterNameLen);
418
419 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, CachedFieldsLen))
420 goto fail;
421
422 Stream_Write(printer_dev->device.data, CachedPrinterConfigData, CachedFieldsLen);
423 res = TRUE;
424fail:
425 free(path);
426 free(wname);
427 free(PnPName);
428 free(DriverName);
429 free(CachedPrinterConfigData);
430 return res;
431}
432
433static BOOL printer_save_default_config(const rdpSettings* settings, rdpPrinter* printer)
434{
435 BOOL res = FALSE;
436 WCHAR* wname = NULL;
437 WCHAR* driver = NULL;
438 size_t wlen = 0;
439 size_t dlen = 0;
440 char* path = NULL;
441
442 if (!settings || !printer || !printer->name || !printer->driver)
443 return FALSE;
444
445 wname = ConvertUtf8ToWCharAlloc(printer->name, NULL);
446
447 if (!wname)
448 goto fail;
449
450 driver = ConvertUtf8ToWCharAlloc(printer->driver, NULL);
451
452 if (!driver)
453 goto fail;
454
455 wlen = _wcslen(wname) + 1;
456 dlen = _wcslen(driver) + 1;
457 path = get_printer_config_path(settings, wname, wlen * sizeof(WCHAR));
458
459 if (!path)
460 goto fail;
461
462 if (dlen > 1)
463 {
464 if (!printer_write_setting(path, PRN_CONF_DRIVER, driver, dlen * sizeof(WCHAR)))
465 goto fail;
466 }
467
468 res = TRUE;
469fail:
470 free(path);
471 free(wname);
472 free(driver);
473 return res;
474}
475
481static UINT printer_process_irp_create(PRINTER_DEVICE* printer_dev, IRP* irp)
482{
483 rdpPrintJob* printjob = NULL;
484
485 WINPR_ASSERT(printer_dev);
486 WINPR_ASSERT(irp);
487
488 if (printer_dev->printer)
489 {
490 WINPR_ASSERT(printer_dev->printer->CreatePrintJob);
491 printjob =
492 printer_dev->printer->CreatePrintJob(printer_dev->printer, irp->devman->id_sequence++);
493 }
494
495 if (printjob)
496 {
497 Stream_Write_UINT32(irp->output, printjob->id); /* FileId */
498 }
499 else
500 {
501 Stream_Write_UINT32(irp->output, 0); /* FileId */
502 irp->IoStatus = STATUS_PRINT_QUEUE_FULL;
503 }
504
505 WINPR_ASSERT(irp->Complete);
506 return irp->Complete(irp);
507}
508
514static UINT printer_process_irp_close(PRINTER_DEVICE* printer_dev, IRP* irp)
515{
516 rdpPrintJob* printjob = NULL;
517
518 WINPR_ASSERT(printer_dev);
519 WINPR_ASSERT(irp);
520
521 if (printer_dev->printer)
522 {
523 WINPR_ASSERT(printer_dev->printer->FindPrintJob);
524 printjob = printer_dev->printer->FindPrintJob(printer_dev->printer, irp->FileId);
525 }
526
527 if (!printjob)
528 {
529 irp->IoStatus = STATUS_UNSUCCESSFUL;
530 }
531 else
532 {
533 printjob->Close(printjob);
534 }
535
536 Stream_Zero(irp->output, 4); /* Padding(4) */
537 WINPR_ASSERT(irp->Complete);
538 return irp->Complete(irp);
539}
540
546static UINT printer_process_irp_write(PRINTER_DEVICE* printer_dev, IRP* irp)
547{
548 UINT32 Length = 0;
549 UINT64 Offset = 0;
550 rdpPrintJob* printjob = NULL;
551 UINT error = CHANNEL_RC_OK;
552
553 WINPR_ASSERT(printer_dev);
554 WINPR_ASSERT(irp);
555
556 if (!Stream_CheckAndLogRequiredLength(TAG, irp->input, 32))
557 return ERROR_INVALID_DATA;
558 Stream_Read_UINT32(irp->input, Length);
559 Stream_Read_UINT64(irp->input, Offset);
560 (void)Offset; /* [MS-RDPEPC] 2.2.2.9 Server Printer Write Request (DR_PRN_WRITE_REQ)
561 * reserved for future use, ignore */
562 Stream_Seek(irp->input, 20); /* Padding */
563 const void* ptr = Stream_ConstPointer(irp->input);
564 if (!Stream_SafeSeek(irp->input, Length))
565 return ERROR_INVALID_DATA;
566 if (printer_dev->printer)
567 {
568 WINPR_ASSERT(printer_dev->printer->FindPrintJob);
569 printjob = printer_dev->printer->FindPrintJob(printer_dev->printer, irp->FileId);
570 }
571
572 if (!printjob)
573 {
574 irp->IoStatus = STATUS_UNSUCCESSFUL;
575 Length = 0;
576 }
577 else
578 {
579 error = printjob->Write(printjob, ptr, Length);
580 }
581
582 if (error)
583 {
584 WLog_ERR(TAG, "printjob->Write failed with error %" PRIu32 "!", error);
585 return error;
586 }
587
588 Stream_Write_UINT32(irp->output, Length);
589 Stream_Write_UINT8(irp->output, 0); /* Padding */
590
591 WINPR_ASSERT(irp->Complete);
592 return irp->Complete(irp);
593}
594
600static UINT printer_process_irp_device_control(WINPR_ATTR_UNUSED PRINTER_DEVICE* printer_dev,
601 IRP* irp)
602{
603 WINPR_ASSERT(printer_dev);
604 WINPR_ASSERT(irp);
605
606 Stream_Write_UINT32(irp->output, 0); /* OutputBufferLength */
607
608 WINPR_ASSERT(irp->Complete);
609 return irp->Complete(irp);
610}
611
617static UINT printer_process_irp(PRINTER_DEVICE* printer_dev, IRP* irp)
618{
619 UINT error = 0;
620
621 WINPR_ASSERT(printer_dev);
622 WINPR_ASSERT(irp);
623
624 switch (irp->MajorFunction)
625 {
626 case IRP_MJ_CREATE:
627 if ((error = printer_process_irp_create(printer_dev, irp)))
628 {
629 WLog_ERR(TAG, "printer_process_irp_create failed with error %" PRIu32 "!", error);
630 return error;
631 }
632
633 break;
634
635 case IRP_MJ_CLOSE:
636 if ((error = printer_process_irp_close(printer_dev, irp)))
637 {
638 WLog_ERR(TAG, "printer_process_irp_close failed with error %" PRIu32 "!", error);
639 return error;
640 }
641
642 break;
643
644 case IRP_MJ_WRITE:
645 if ((error = printer_process_irp_write(printer_dev, irp)))
646 {
647 WLog_ERR(TAG, "printer_process_irp_write failed with error %" PRIu32 "!", error);
648 return error;
649 }
650
651 break;
652
653 case IRP_MJ_DEVICE_CONTROL:
654 if ((error = printer_process_irp_device_control(printer_dev, irp)))
655 {
656 WLog_ERR(TAG, "printer_process_irp_device_control failed with error %" PRIu32 "!",
657 error);
658 return error;
659 }
660
661 break;
662
663 default:
664 irp->IoStatus = STATUS_NOT_SUPPORTED;
665 WINPR_ASSERT(irp->Complete);
666 return irp->Complete(irp);
667 }
668
669 return CHANNEL_RC_OK;
670}
671
672static DWORD WINAPI printer_thread_func(LPVOID arg)
673{
674 IRP* irp = NULL;
675 PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)arg;
676 UINT error = CHANNEL_RC_OK;
677
678 WINPR_ASSERT(printer_dev);
679
680 while (1)
681 {
682 HANDLE obj[] = { printer_dev->event, printer_dev->stopEvent };
683 DWORD rc = WaitForMultipleObjects(ARRAYSIZE(obj), obj, FALSE, INFINITE);
684
685 if (rc == WAIT_FAILED)
686 {
687 error = GetLastError();
688 WLog_ERR(TAG, "WaitForMultipleObjects failed with error %" PRIu32 "!", error);
689 break;
690 }
691
692 if (rc == WAIT_OBJECT_0 + 1)
693 break;
694 else if (rc != WAIT_OBJECT_0)
695 continue;
696
697 (void)ResetEvent(printer_dev->event);
698 irp = (IRP*)InterlockedPopEntrySList(printer_dev->pIrpList);
699
700 if (irp == NULL)
701 {
702 WLog_ERR(TAG, "InterlockedPopEntrySList failed!");
703 error = ERROR_INTERNAL_ERROR;
704 break;
705 }
706
707 if ((error = printer_process_irp(printer_dev, irp)))
708 {
709 WLog_ERR(TAG, "printer_process_irp failed with error %" PRIu32 "!", error);
710 break;
711 }
712 }
713
714 if (error && printer_dev->rdpcontext)
715 setChannelError(printer_dev->rdpcontext, error, "printer_thread_func reported an error");
716
717 ExitThread(error);
718 return error;
719}
720
726static UINT printer_irp_request(DEVICE* device, IRP* irp)
727{
728 PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
729
730 WINPR_ASSERT(printer_dev);
731 WINPR_ASSERT(irp);
732
733 if (printer_dev->async)
734 {
735 InterlockedPushEntrySList(printer_dev->pIrpList, &(irp->ItemEntry));
736 (void)SetEvent(printer_dev->event);
737 }
738 else
739 {
740 UINT error = printer_process_irp(printer_dev, irp);
741 if (error)
742 {
743 WLog_ERR(TAG, "printer_process_irp failed with error %" PRIu32 "!", error);
744 return error;
745 }
746 }
747
748 return CHANNEL_RC_OK;
749}
750
751static UINT printer_custom_component(DEVICE* device, UINT16 component, UINT16 packetId, wStream* s)
752{
753 UINT32 eventID = 0;
754 PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
755
756 WINPR_ASSERT(printer_dev);
757 WINPR_ASSERT(printer_dev->rdpcontext);
758
759 const rdpSettings* settings = printer_dev->rdpcontext->settings;
760 WINPR_ASSERT(settings);
761
762 if (component != RDPDR_CTYP_PRN)
763 return ERROR_INVALID_DATA;
764
765 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
766 return ERROR_INVALID_DATA;
767
768 Stream_Read_UINT32(s, eventID);
769
770 switch (packetId)
771 {
772 case PAKID_PRN_CACHE_DATA:
773 switch (eventID)
774 {
775 case RDPDR_ADD_PRINTER_EVENT:
776 {
777 char PortDosName[8];
778 UINT32 PnPNameLen = 0;
779 UINT32 DriverNameLen = 0;
780 UINT32 PrintNameLen = 0;
781 UINT32 CacheFieldsLen = 0;
782 const WCHAR* PnPName = NULL;
783 const WCHAR* DriverName = NULL;
784 const WCHAR* PrinterName = NULL;
785 const BYTE* CachedPrinterConfigData = NULL;
786
787 if (!Stream_CheckAndLogRequiredLength(TAG, s, 24))
788 return ERROR_INVALID_DATA;
789
790 Stream_Read(s, PortDosName, sizeof(PortDosName));
791 Stream_Read_UINT32(s, PnPNameLen);
792 Stream_Read_UINT32(s, DriverNameLen);
793 Stream_Read_UINT32(s, PrintNameLen);
794 Stream_Read_UINT32(s, CacheFieldsLen);
795
796 if (!Stream_CheckAndLogRequiredLength(TAG, s, PnPNameLen))
797 return ERROR_INVALID_DATA;
798
799 PnPName = Stream_ConstPointer(s);
800 Stream_Seek(s, PnPNameLen);
801
802 if (!Stream_CheckAndLogRequiredLength(TAG, s, DriverNameLen))
803 return ERROR_INVALID_DATA;
804
805 DriverName = Stream_ConstPointer(s);
806 Stream_Seek(s, DriverNameLen);
807
808 if (!Stream_CheckAndLogRequiredLength(TAG, s, PrintNameLen))
809 return ERROR_INVALID_DATA;
810
811 PrinterName = Stream_ConstPointer(s);
812 Stream_Seek(s, PrintNameLen);
813
814 if (!Stream_CheckAndLogRequiredLength(TAG, s, CacheFieldsLen))
815 return ERROR_INVALID_DATA;
816
817 CachedPrinterConfigData = Stream_ConstPointer(s);
818 Stream_Seek(s, CacheFieldsLen);
819
820 if (!printer_save_to_config(settings, PortDosName, sizeof(PortDosName), PnPName,
821 PnPNameLen, DriverName, DriverNameLen, PrinterName,
822 PrintNameLen, CachedPrinterConfigData,
823 CacheFieldsLen))
824 return ERROR_INTERNAL_ERROR;
825 }
826 break;
827
828 case RDPDR_UPDATE_PRINTER_EVENT:
829 {
830 UINT32 PrinterNameLen = 0;
831 UINT32 ConfigDataLen = 0;
832 const WCHAR* PrinterName = NULL;
833 const BYTE* ConfigData = NULL;
834
835 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
836 return ERROR_INVALID_DATA;
837
838 Stream_Read_UINT32(s, PrinterNameLen);
839 Stream_Read_UINT32(s, ConfigDataLen);
840
841 if (!Stream_CheckAndLogRequiredLength(TAG, s, PrinterNameLen))
842 return ERROR_INVALID_DATA;
843
844 PrinterName = Stream_ConstPointer(s);
845 Stream_Seek(s, PrinterNameLen);
846
847 if (!Stream_CheckAndLogRequiredLength(TAG, s, ConfigDataLen))
848 return ERROR_INVALID_DATA;
849
850 ConfigData = Stream_ConstPointer(s);
851 Stream_Seek(s, ConfigDataLen);
852
853 if (!printer_update_to_config(settings, PrinterName, PrinterNameLen, ConfigData,
854 ConfigDataLen))
855 return ERROR_INTERNAL_ERROR;
856 }
857 break;
858
859 case RDPDR_DELETE_PRINTER_EVENT:
860 {
861 UINT32 PrinterNameLen = 0;
862 const WCHAR* PrinterName = NULL;
863
864 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
865 return ERROR_INVALID_DATA;
866
867 Stream_Read_UINT32(s, PrinterNameLen);
868
869 if (!Stream_CheckAndLogRequiredLength(TAG, s, PrinterNameLen))
870 return ERROR_INVALID_DATA;
871
872 PrinterName = Stream_ConstPointer(s);
873 Stream_Seek(s, PrinterNameLen);
874 printer_remove_config(settings, PrinterName, PrinterNameLen);
875 }
876 break;
877
878 case RDPDR_RENAME_PRINTER_EVENT:
879 {
880 UINT32 OldPrinterNameLen = 0;
881 UINT32 NewPrinterNameLen = 0;
882 const WCHAR* OldPrinterName = NULL;
883 const WCHAR* NewPrinterName = NULL;
884
885 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
886 return ERROR_INVALID_DATA;
887
888 Stream_Read_UINT32(s, OldPrinterNameLen);
889 Stream_Read_UINT32(s, NewPrinterNameLen);
890
891 if (!Stream_CheckAndLogRequiredLength(TAG, s, OldPrinterNameLen))
892 return ERROR_INVALID_DATA;
893
894 OldPrinterName = Stream_ConstPointer(s);
895 Stream_Seek(s, OldPrinterNameLen);
896
897 if (!Stream_CheckAndLogRequiredLength(TAG, s, NewPrinterNameLen))
898 return ERROR_INVALID_DATA;
899
900 NewPrinterName = Stream_ConstPointer(s);
901 Stream_Seek(s, NewPrinterNameLen);
902
903 if (!printer_move_config(settings, OldPrinterName, OldPrinterNameLen,
904 NewPrinterName, NewPrinterNameLen))
905 return ERROR_INTERNAL_ERROR;
906 }
907 break;
908
909 default:
910 WLog_ERR(TAG, "Unknown cache data eventID: 0x%08" PRIX32 "", eventID);
911 return ERROR_INVALID_DATA;
912 }
913
914 break;
915
916 case PAKID_PRN_USING_XPS:
917 {
918 UINT32 flags = 0;
919
920 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
921 return ERROR_INVALID_DATA;
922
923 Stream_Read_UINT32(s, flags);
924 WLog_ERR(TAG,
925 "Ignoring unhandled message PAKID_PRN_USING_XPS [printerID=%08" PRIx32
926 ", flags=%08" PRIx32 "]",
927 eventID, flags);
928 }
929 break;
930
931 default:
932 WLog_ERR(TAG, "Unknown printing component packetID: 0x%04" PRIX16 "", packetId);
933 return ERROR_INVALID_DATA;
934 }
935
936 return CHANNEL_RC_OK;
937}
938
944static UINT printer_free(DEVICE* device)
945{
946 IRP* irp = NULL;
947 PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
948 UINT error = 0;
949
950 WINPR_ASSERT(printer_dev);
951
952 if (printer_dev->async)
953 {
954 (void)SetEvent(printer_dev->stopEvent);
955
956 if (WaitForSingleObject(printer_dev->thread, INFINITE) == WAIT_FAILED)
957 {
958 error = GetLastError();
959 WLog_ERR(TAG, "WaitForSingleObject failed with error %" PRIu32 "", error);
960
961 /* The analyzer is confused by this premature return value.
962 * Since this case can not be handled gracefully silence the
963 * analyzer here. */
964#ifndef __clang_analyzer__
965 return error;
966#endif
967 }
968
969 while ((irp = (IRP*)InterlockedPopEntrySList(printer_dev->pIrpList)) != NULL)
970 {
971 WINPR_ASSERT(irp->Discard);
972 irp->Discard(irp);
973 }
974
975 (void)CloseHandle(printer_dev->thread);
976 (void)CloseHandle(printer_dev->stopEvent);
977 (void)CloseHandle(printer_dev->event);
978 winpr_aligned_free(printer_dev->pIrpList);
979 }
980
981 if (printer_dev->printer)
982 {
983 WINPR_ASSERT(printer_dev->printer->ReleaseRef);
984 printer_dev->printer->ReleaseRef(printer_dev->printer);
985 }
986
987 Stream_Free(printer_dev->device.data, TRUE);
988 free(printer_dev);
989 return CHANNEL_RC_OK;
990}
991
997static UINT printer_register(PDEVICE_SERVICE_ENTRY_POINTS pEntryPoints, rdpPrinter* printer)
998{
999 PRINTER_DEVICE* printer_dev = NULL;
1000 UINT error = ERROR_INTERNAL_ERROR;
1001
1002 WINPR_ASSERT(pEntryPoints);
1003 WINPR_ASSERT(printer);
1004
1005 printer_dev = (PRINTER_DEVICE*)calloc(1, sizeof(PRINTER_DEVICE));
1006
1007 if (!printer_dev)
1008 {
1009 WLog_ERR(TAG, "calloc failed!");
1010 return CHANNEL_RC_NO_MEMORY;
1011 }
1012
1013 printer_dev->device.data = Stream_New(NULL, 1024);
1014
1015 if (!printer_dev->device.data)
1016 goto error_out;
1017
1018 (void)sprintf_s(printer_dev->port, sizeof(printer_dev->port), "PRN%" PRIuz, printer->id);
1019 printer_dev->device.type = RDPDR_DTYP_PRINT;
1020 printer_dev->device.name = printer_dev->port;
1021 printer_dev->device.IRPRequest = printer_irp_request;
1022 printer_dev->device.CustomComponentRequest = printer_custom_component;
1023 printer_dev->device.Free = printer_free;
1024 printer_dev->rdpcontext = pEntryPoints->rdpcontext;
1025 printer_dev->printer = printer;
1026
1027 if (!freerdp_settings_get_bool(pEntryPoints->rdpcontext->settings,
1028 FreeRDP_SynchronousStaticChannels))
1029 printer_dev->async = TRUE;
1030
1031 if (!printer_load_from_config(pEntryPoints->rdpcontext->settings, printer, printer_dev))
1032 goto error_out;
1033
1034 if (printer_dev->async)
1035 {
1036 printer_dev->pIrpList = (WINPR_PSLIST_HEADER)winpr_aligned_malloc(
1037 sizeof(WINPR_SLIST_HEADER), MEMORY_ALLOCATION_ALIGNMENT);
1038
1039 if (!printer_dev->pIrpList)
1040 {
1041 WLog_ERR(TAG, "_aligned_malloc failed!");
1042 error = CHANNEL_RC_NO_MEMORY;
1043 goto error_out;
1044 }
1045
1046 InitializeSListHead(printer_dev->pIrpList);
1047
1048 printer_dev->event = CreateEvent(NULL, TRUE, FALSE, NULL);
1049 if (!printer_dev->event)
1050 {
1051 WLog_ERR(TAG, "CreateEvent failed!");
1052 error = ERROR_INTERNAL_ERROR;
1053 goto error_out;
1054 }
1055
1056 printer_dev->stopEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
1057 if (!printer_dev->stopEvent)
1058 {
1059 WLog_ERR(TAG, "CreateEvent failed!");
1060 error = ERROR_INTERNAL_ERROR;
1061 goto error_out;
1062 }
1063 }
1064
1065 error = pEntryPoints->RegisterDevice(pEntryPoints->devman, &printer_dev->device);
1066 if (error)
1067 {
1068 WLog_ERR(TAG, "RegisterDevice failed with error %" PRIu32 "!", error);
1069 goto error_out;
1070 }
1071
1072 if (printer_dev->async)
1073 {
1074 printer_dev->thread =
1075 CreateThread(NULL, 0, printer_thread_func, (void*)printer_dev, 0, NULL);
1076 if (!printer_dev->thread)
1077 {
1078 WLog_ERR(TAG, "CreateThread failed!");
1079 error = ERROR_INTERNAL_ERROR;
1080 goto error_out;
1081 }
1082 }
1083
1084 WINPR_ASSERT(printer->AddRef);
1085 printer->AddRef(printer);
1086 return CHANNEL_RC_OK;
1087error_out:
1088 printer_free(&printer_dev->device);
1089 return error;
1090}
1091
1092static rdpPrinterDriver* printer_load_backend(const char* backend)
1093{
1094 typedef UINT(VCAPITYPE * backend_load_t)(rdpPrinterDriver**);
1095 PVIRTUALCHANNELENTRY entry = freerdp_load_channel_addin_entry("printer", backend, NULL, 0);
1096 backend_load_t func = WINPR_FUNC_PTR_CAST(entry, backend_load_t);
1097 if (!func)
1098 return NULL;
1099
1100 rdpPrinterDriver* printer = NULL;
1101 const UINT rc = func(&printer);
1102 if (rc != CHANNEL_RC_OK)
1103 return NULL;
1104
1105 return printer;
1106}
1107
1113FREERDP_ENTRY_POINT(
1114 UINT VCAPITYPE printer_DeviceServiceEntry(PDEVICE_SERVICE_ENTRY_POINTS pEntryPoints))
1115{
1116 char* name = NULL;
1117 char* driver_name = NULL;
1118 BOOL default_backend = TRUE;
1119 RDPDR_PRINTER* device = NULL;
1120 rdpPrinterDriver* driver = NULL;
1121 UINT error = CHANNEL_RC_OK;
1122
1123 if (!pEntryPoints || !pEntryPoints->device)
1124 return ERROR_INVALID_PARAMETER;
1125
1126 device = (RDPDR_PRINTER*)pEntryPoints->device;
1127 name = device->device.Name;
1128 driver_name = _strdup(device->DriverName);
1129
1130 /* Secondary argument is one of the following:
1131 *
1132 * <driver_name> ... name of a printer driver
1133 * <driver_name>:<backend_name> ... name of a printer driver and local printer backend to use
1134 */
1135 if (driver_name)
1136 {
1137 char* sep = strstr(driver_name, ":");
1138 if (sep)
1139 {
1140 const char* backend = sep + 1;
1141 *sep = '\0';
1142 driver = printer_load_backend(backend);
1143 default_backend = FALSE;
1144 }
1145 }
1146
1147 if (!driver && default_backend)
1148 {
1149 const char* backend =
1150#if defined(WITH_CUPS)
1151 "cups"
1152#elif defined(_WIN32)
1153 "win"
1154#else
1155 ""
1156#endif
1157 ;
1158
1159 driver = printer_load_backend(backend);
1160 }
1161
1162 if (!driver)
1163 {
1164 WLog_ERR(TAG, "Could not get a printer driver!");
1165 error = CHANNEL_RC_INITIALIZATION_ERROR;
1166 goto fail;
1167 }
1168
1169 if (name && name[0])
1170 {
1171 WINPR_ASSERT(driver->GetPrinter);
1172 rdpPrinter* printer = driver->GetPrinter(driver, name, driver_name, device->IsDefault);
1173
1174 if (!printer)
1175 {
1176 WLog_ERR(TAG, "Could not get printer %s!", name);
1177 error = CHANNEL_RC_INITIALIZATION_ERROR;
1178 goto fail;
1179 }
1180
1181 WINPR_ASSERT(printer->ReleaseRef);
1182 if (!printer_save_default_config(pEntryPoints->rdpcontext->settings, printer))
1183 {
1184 error = CHANNEL_RC_INITIALIZATION_ERROR;
1185 printer->ReleaseRef(printer);
1186 goto fail;
1187 }
1188
1189 error = printer_register(pEntryPoints, printer);
1190 printer->ReleaseRef(printer);
1191 if (error)
1192 {
1193 WLog_ERR(TAG, "printer_register failed with error %" PRIu32 "!", error);
1194 goto fail;
1195 }
1196 }
1197 else
1198 {
1199 WINPR_ASSERT(driver->EnumPrinters);
1200 rdpPrinter** printers = driver->EnumPrinters(driver);
1201 if (printers)
1202 {
1203 for (rdpPrinter** current = printers; *current; ++current)
1204 {
1205 error = printer_register(pEntryPoints, *current);
1206 if (error)
1207 {
1208 WLog_ERR(TAG, "printer_register failed with error %" PRIu32 "!", error);
1209 break;
1210 }
1211 }
1212 }
1213 else
1214 {
1215 WLog_ERR(TAG, "Failed to enumerate printers!");
1216 error = CHANNEL_RC_INITIALIZATION_ERROR;
1217 }
1218
1219 WINPR_ASSERT(driver->ReleaseEnumPrinters);
1220 driver->ReleaseEnumPrinters(printers);
1221 }
1222
1223fail:
1224 free(driver_name);
1225 if (driver)
1226 {
1227 WINPR_ASSERT(driver->ReleaseRef);
1228 driver->ReleaseRef(driver);
1229 }
1230
1231 return error;
1232}
FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.
FREERDP_API const char * freerdp_settings_get_string(const rdpSettings *settings, FreeRDP_Settings_Keys_String id)
Returns a immutable string settings value.