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