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