24#include <freerdp/config.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>
41#include <freerdp/channels/rdpdr.h>
42#include <freerdp/crypto/crypto.h>
43#include <freerdp/freerdp.h>
45#include "../printer.h"
47#include <freerdp/client/printer.h>
48#include <freerdp/utils/rdpdr_utils.h>
49#include <freerdp/channels/log.h>
51#define TAG CHANNELS_TAG("printer.client")
65 rdpContext* rdpcontext;
78static const char* filemap[] = {
"PortDosName",
"PnPName",
"DriverName",
79 "CachedPrinterConfigData" };
81WINPR_ATTR_MALLOC(free, 1)
83static
char* get_printer_hash(const WCHAR* name,
size_t length)
85 BYTE hash[WINPR_SHA256_DIGEST_LENGTH] = WINPR_C_ARRAY_INIT;
87 if (!winpr_Digest(WINPR_MD_SHA256, name, length, hash,
sizeof(hash)))
90 return winpr_BinToHexString(hash,
sizeof(hash), FALSE);
93WINPR_ATTR_MALLOC(free, 1)
95static
char* get_printer_config_path(const rdpSettings* settings, const WCHAR* name,
size_t length)
97 char* config =
nullptr;
99 char* dir = GetCombinedPath(path,
"printers");
102 char* bname = get_printer_hash(name, length);
105 config = GetCombinedPath(dir, bname);
107 if (config && !winpr_PathFileExists(config))
109 if (!winpr_PathMakePath(config,
nullptr))
122static BOOL printer_write_setting(
const char* path, prn_conf_t type,
const void* data,
130 HANDLE file =
nullptr;
131 char* base64 =
nullptr;
132 const char* name = filemap[type];
133 char* abs = GetCombinedPath(path, name);
135 if (!abs || (length > INT32_MAX))
141 file = winpr_CreateFile(abs, GENERIC_WRITE, 0,
nullptr, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL,
145 if (file == INVALID_HANDLE_VALUE)
150 base64 = crypto_base64_encode(data, length);
157 const size_t b64len = strnlen(base64, 2 * length);
158 rc = WriteFile(file, base64, (UINT32)b64len, &written,
nullptr);
160 if (b64len != written)
167 (void)CloseHandle(file);
172static BOOL printer_config_valid(
const char* path)
177 if (!winpr_PathFileExists(path))
183static BOOL printer_read_setting(
const char* path, prn_conf_t type,
void** data, UINT32* length)
189 char* fdata =
nullptr;
190 const char* name = filemap[type];
197 WLog_DBG(TAG,
"Printer option %s ignored", name);
201 char* abs = GetCombinedPath(path, name);
205 HANDLE file = winpr_CreateFile(abs, GENERIC_READ, 0,
nullptr, OPEN_EXISTING,
206 FILE_ATTRIBUTE_NORMAL,
nullptr);
209 if (file == INVALID_HANDLE_VALUE)
212 lowSize = GetFileSize(file, &highSize);
214 if ((lowSize == INVALID_FILE_SIZE) || (highSize != 0))
219 fdata = malloc(lowSize);
224 rc = ReadFile(file, fdata, lowSize, &read,
nullptr);
231 (void)CloseHandle(file);
233 if (rc && (lowSize <= INT_MAX))
236 crypto_base64_decode(fdata, lowSize, (BYTE**)data, &blen);
238 if (*data && (blen > 0))
239 *length = (UINT32)blen;
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)
263 char* path = get_printer_config_path(settings, PrinterName, PrintNameLen);
268 if (!printer_write_setting(path, PRN_CONF_PORT, PortDosName, PortDosNameLen))
271 if (!printer_write_setting(path, PRN_CONF_PNP, PnPName, PnPNameLen))
274 if (!printer_write_setting(path, PRN_CONF_DRIVER, DriverName, DriverNameLen))
277 if (!printer_write_setting(path, PRN_CONF_DATA, CachedPrinterConfigData, CacheFieldsLen))
285static BOOL printer_update_to_config(
const rdpSettings* settings,
const WCHAR* name,
size_t length,
286 const BYTE* data,
size_t datalen)
289 char* path = get_printer_config_path(settings, name, length);
290 rc = printer_write_setting(path, PRN_CONF_DATA, data, datalen);
295static BOOL printer_remove_config(
const rdpSettings* settings,
const WCHAR* name,
size_t length)
298 char* path = get_printer_config_path(settings, name, length);
300 if (!printer_config_valid(path))
303 rc = winpr_RemoveDirectory(path);
309static BOOL printer_move_config(
const rdpSettings* settings,
const WCHAR* oldName,
size_t oldLength,
310 const WCHAR* newName,
size_t newLength)
313 char* oldPath = get_printer_config_path(settings, oldName, oldLength);
314 char* newPath = get_printer_config_path(settings, newName, newLength);
316 if (printer_config_valid(oldPath))
317 rc = winpr_MoveFile(oldPath, newPath);
324static BOOL printer_load_from_config(
const rdpSettings* settings, rdpPrinter* printer,
325 PRINTER_DEVICE* printer_dev)
328 WCHAR* wname =
nullptr;
330 char* path =
nullptr;
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;
340 if (!settings || !printer || !printer->name)
343 wname = ConvertUtf8ToWCharAlloc(printer->name, &wlen);
349 path = get_printer_config_path(settings, wname, wlen *
sizeof(WCHAR));
351 const size_t plen = wlen *
sizeof(WCHAR);
352 if (plen > UINT32_MAX)
354 PrinterNameLen = (UINT32)plen;
360 if (printer->is_default)
361 flags |= RDPDR_PRINTER_ANNOUNCE_FLAG_DEFAULTPRINTER;
363 if (!printer_read_setting(path, PRN_CONF_PNP, &PnPName, &PnPNameLen))
367 if (!printer_read_setting(path, PRN_CONF_DRIVER, &DriverName, &DriverNameLen))
370 DriverName = ConvertUtf8ToWCharAlloc(printer->driver, &len);
373 const size_t dlen = (len + 1) *
sizeof(WCHAR);
374 if (dlen > UINT32_MAX)
376 DriverNameLen = (UINT32)dlen;
379 if (!printer_read_setting(path, PRN_CONF_DATA, &CachedPrinterConfigData, &CachedFieldsLen))
383 Stream_ResetPosition(printer_dev->device.data);
385 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, 24))
388 Stream_Write_UINT32(printer_dev->device.data, flags);
389 Stream_Write_UINT32(printer_dev->device.data, 0);
390 Stream_Write_UINT32(printer_dev->device.data, 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);
395 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, PnPNameLen))
399 Stream_Write(printer_dev->device.data, PnPName, PnPNameLen);
401 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, DriverNameLen))
404 Stream_Write(printer_dev->device.data, DriverName, DriverNameLen);
406 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, PrinterNameLen))
414 backslash.c[0] =
'\\';
415 backslash.c[1] =
'\0';
417 for (WCHAR* wptr = wname; (wptr = _wcschr(wptr, backslash.w));)
419 Stream_Write(printer_dev->device.data, wname, PrinterNameLen);
421 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, CachedFieldsLen))
424 Stream_Write(printer_dev->device.data, CachedPrinterConfigData, CachedFieldsLen);
431 free(CachedPrinterConfigData);
435static BOOL printer_save_default_config(
const rdpSettings* settings, rdpPrinter* printer)
438 WCHAR* wname =
nullptr;
439 WCHAR* driver =
nullptr;
442 char* path =
nullptr;
444 if (!settings || !printer || !printer->name || !printer->driver)
447 wname = ConvertUtf8ToWCharAlloc(printer->name,
nullptr);
452 driver = ConvertUtf8ToWCharAlloc(printer->driver,
nullptr);
457 wlen = _wcslen(wname) + 1;
458 dlen = _wcslen(driver) + 1;
459 path = get_printer_config_path(settings, wname, wlen *
sizeof(WCHAR));
466 if (!printer_write_setting(path, PRN_CONF_DRIVER, driver, dlen *
sizeof(WCHAR)))
483static UINT printer_process_irp_create(PRINTER_DEVICE* printer_dev, IRP* irp)
485 rdpPrintJob* printjob =
nullptr;
487 WINPR_ASSERT(printer_dev);
490 if (printer_dev->printer)
492 WINPR_ASSERT(printer_dev->printer->CreatePrintJob);
494 printer_dev->printer->CreatePrintJob(printer_dev->printer, irp->devman->id_sequence++);
499 Stream_Write_UINT32(irp->output, printjob->id);
503 Stream_Write_UINT32(irp->output, 0);
504 irp->IoStatus = STATUS_PRINT_QUEUE_FULL;
507 return CHANNEL_RC_OK;
515static UINT printer_process_irp_close(PRINTER_DEVICE* printer_dev, IRP* irp)
517 rdpPrintJob* printjob =
nullptr;
519 WINPR_ASSERT(printer_dev);
522 if (printer_dev->printer)
524 WINPR_ASSERT(printer_dev->printer->FindPrintJob);
525 printjob = printer_dev->printer->FindPrintJob(printer_dev->printer, irp->FileId);
530 irp->IoStatus = STATUS_UNSUCCESSFUL;
534 printjob->Close(printjob);
537 Stream_Zero(irp->output, 4);
538 return CHANNEL_RC_OK;
546static UINT printer_process_irp_write(PRINTER_DEVICE* printer_dev, IRP* irp)
550 rdpPrintJob* printjob =
nullptr;
551 UINT error = CHANNEL_RC_OK;
553 WINPR_ASSERT(printer_dev);
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);
562 Stream_Seek(irp->input, 20);
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)
568 WINPR_ASSERT(printer_dev->printer->FindPrintJob);
569 printjob = printer_dev->printer->FindPrintJob(printer_dev->printer, irp->FileId);
574 irp->IoStatus = STATUS_UNSUCCESSFUL;
579 error = printjob->Write(printjob, ptr, Length);
584 WLog_ERR(TAG,
"printjob->Write failed with error %" PRIu32
"!", error);
588 Stream_Write_UINT32(irp->output, Length);
589 Stream_Write_UINT8(irp->output, 0);
590 return CHANNEL_RC_OK;
598static UINT printer_process_irp_device_control(WINPR_ATTR_UNUSED PRINTER_DEVICE* printer_dev,
601 WINPR_ASSERT(printer_dev);
604 Stream_Write_UINT32(irp->output, 0);
605 return CHANNEL_RC_OK;
608static UINT printer_evaluate(UINT error, IRP* irp)
611 if (error == CHANNEL_RC_OK)
613 WINPR_ASSERT(irp->Complete);
614 return irp->Complete(irp);
617 WLog_ERR(TAG,
"IRP %s failed with %" PRIu32, rdpdr_irp_string(irp->MajorFunction), error);
618 WINPR_ASSERT(irp->Discard);
627static UINT printer_process_irp(PRINTER_DEVICE* printer_dev, IRP* irp)
629 UINT error = CHANNEL_RC_OK;
631 WINPR_ASSERT(printer_dev);
634 switch (irp->MajorFunction)
637 error = printer_process_irp_create(printer_dev, irp);
641 error = printer_process_irp_close(printer_dev, irp);
645 error = printer_process_irp_write(printer_dev, irp);
648 case IRP_MJ_DEVICE_CONTROL:
649 error = printer_process_irp_device_control(printer_dev, irp);
653 irp->IoStatus = STATUS_NOT_SUPPORTED;
657 return printer_evaluate(error, irp);
660static DWORD WINAPI printer_thread_func(LPVOID arg)
662 PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)arg;
663 UINT error = CHANNEL_RC_OK;
665 WINPR_ASSERT(printer_dev);
669 HANDLE obj[] = { printer_dev->event, printer_dev->stopEvent };
670 DWORD rc = WaitForMultipleObjects(ARRAYSIZE(obj), obj, FALSE, INFINITE);
672 if (rc == WAIT_FAILED)
674 error = GetLastError();
675 WLog_ERR(TAG,
"WaitForMultipleObjects failed with error %" PRIu32
"!", error);
679 if (rc == WAIT_OBJECT_0 + 1)
681 else if (rc != WAIT_OBJECT_0)
684 (void)ResetEvent(printer_dev->event);
685 IRP* irp = (IRP*)Queue_Dequeue(printer_dev->pIrpList);
689 WLog_ERR(TAG,
"InterlockedPopEntrySList failed!");
690 error = ERROR_INTERNAL_ERROR;
694 if ((error = printer_process_irp(printer_dev, irp)))
696 WLog_ERR(TAG,
"printer_process_irp failed with error %" PRIu32
"!", error);
701 if (error && printer_dev->rdpcontext)
702 setChannelError(printer_dev->rdpcontext, error,
"printer_thread_func reported an error");
713static UINT printer_irp_request(DEVICE* device, IRP* irp)
715 PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
717 WINPR_ASSERT(printer_dev);
720 if (printer_dev->async)
722 if (!Queue_Enqueue(printer_dev->pIrpList, irp))
723 return ERROR_INTERNAL_ERROR;
725 (void)SetEvent(printer_dev->event);
729 UINT error = printer_process_irp(printer_dev, irp);
732 WLog_ERR(TAG,
"printer_process_irp failed with error %" PRIu32
"!", error);
737 return CHANNEL_RC_OK;
740static UINT printer_custom_component(DEVICE* device, UINT16 component, UINT16 packetId,
wStream* s)
743 PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
745 WINPR_ASSERT(printer_dev);
746 WINPR_ASSERT(printer_dev->rdpcontext);
748 const rdpSettings* settings = printer_dev->rdpcontext->settings;
749 WINPR_ASSERT(settings);
751 if (component != RDPDR_CTYP_PRN)
752 return ERROR_INVALID_DATA;
754 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
755 return ERROR_INVALID_DATA;
757 Stream_Read_UINT32(s, eventID);
761 case PAKID_PRN_CACHE_DATA:
764 case RDPDR_ADD_PRINTER_EVENT:
767 UINT32 PnPNameLen = 0;
768 UINT32 DriverNameLen = 0;
769 UINT32 PrintNameLen = 0;
770 UINT32 CacheFieldsLen = 0;
771 const WCHAR* PnPName =
nullptr;
772 const WCHAR* DriverName =
nullptr;
773 const WCHAR* PrinterName =
nullptr;
774 const BYTE* CachedPrinterConfigData =
nullptr;
776 if (!Stream_CheckAndLogRequiredLength(TAG, s, 24))
777 return ERROR_INVALID_DATA;
779 Stream_Read(s, PortDosName,
sizeof(PortDosName));
780 Stream_Read_UINT32(s, PnPNameLen);
781 Stream_Read_UINT32(s, DriverNameLen);
782 Stream_Read_UINT32(s, PrintNameLen);
783 Stream_Read_UINT32(s, CacheFieldsLen);
785 if (!Stream_CheckAndLogRequiredLength(TAG, s, PnPNameLen))
786 return ERROR_INVALID_DATA;
788 PnPName = Stream_ConstPointer(s);
789 Stream_Seek(s, PnPNameLen);
791 if (!Stream_CheckAndLogRequiredLength(TAG, s, DriverNameLen))
792 return ERROR_INVALID_DATA;
794 DriverName = Stream_ConstPointer(s);
795 Stream_Seek(s, DriverNameLen);
797 if (!Stream_CheckAndLogRequiredLength(TAG, s, PrintNameLen))
798 return ERROR_INVALID_DATA;
800 PrinterName = Stream_ConstPointer(s);
801 Stream_Seek(s, PrintNameLen);
803 if (!Stream_CheckAndLogRequiredLength(TAG, s, CacheFieldsLen))
804 return ERROR_INVALID_DATA;
806 CachedPrinterConfigData = Stream_ConstPointer(s);
807 Stream_Seek(s, CacheFieldsLen);
809 if (!printer_save_to_config(settings, PortDosName,
sizeof(PortDosName), PnPName,
810 PnPNameLen, DriverName, DriverNameLen, PrinterName,
811 PrintNameLen, CachedPrinterConfigData,
813 return ERROR_INTERNAL_ERROR;
817 case RDPDR_UPDATE_PRINTER_EVENT:
819 UINT32 PrinterNameLen = 0;
820 UINT32 ConfigDataLen = 0;
821 const WCHAR* PrinterName =
nullptr;
822 const BYTE* ConfigData =
nullptr;
824 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
825 return ERROR_INVALID_DATA;
827 Stream_Read_UINT32(s, PrinterNameLen);
828 Stream_Read_UINT32(s, ConfigDataLen);
830 if (!Stream_CheckAndLogRequiredLength(TAG, s, PrinterNameLen))
831 return ERROR_INVALID_DATA;
833 PrinterName = Stream_ConstPointer(s);
834 Stream_Seek(s, PrinterNameLen);
836 if (!Stream_CheckAndLogRequiredLength(TAG, s, ConfigDataLen))
837 return ERROR_INVALID_DATA;
839 ConfigData = Stream_ConstPointer(s);
840 Stream_Seek(s, ConfigDataLen);
842 if (!printer_update_to_config(settings, PrinterName, PrinterNameLen, ConfigData,
844 return ERROR_INTERNAL_ERROR;
848 case RDPDR_DELETE_PRINTER_EVENT:
850 UINT32 PrinterNameLen = 0;
851 const WCHAR* PrinterName =
nullptr;
853 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
854 return ERROR_INVALID_DATA;
856 Stream_Read_UINT32(s, PrinterNameLen);
858 if (!Stream_CheckAndLogRequiredLength(TAG, s, PrinterNameLen))
859 return ERROR_INVALID_DATA;
861 PrinterName = Stream_ConstPointer(s);
862 Stream_Seek(s, PrinterNameLen);
863 printer_remove_config(settings, PrinterName, PrinterNameLen);
867 case RDPDR_RENAME_PRINTER_EVENT:
869 UINT32 OldPrinterNameLen = 0;
870 UINT32 NewPrinterNameLen = 0;
871 const WCHAR* OldPrinterName =
nullptr;
872 const WCHAR* NewPrinterName =
nullptr;
874 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
875 return ERROR_INVALID_DATA;
877 Stream_Read_UINT32(s, OldPrinterNameLen);
878 Stream_Read_UINT32(s, NewPrinterNameLen);
880 if (!Stream_CheckAndLogRequiredLength(TAG, s, OldPrinterNameLen))
881 return ERROR_INVALID_DATA;
883 OldPrinterName = Stream_ConstPointer(s);
884 Stream_Seek(s, OldPrinterNameLen);
886 if (!Stream_CheckAndLogRequiredLength(TAG, s, NewPrinterNameLen))
887 return ERROR_INVALID_DATA;
889 NewPrinterName = Stream_ConstPointer(s);
890 Stream_Seek(s, NewPrinterNameLen);
892 if (!printer_move_config(settings, OldPrinterName, OldPrinterNameLen,
893 NewPrinterName, NewPrinterNameLen))
894 return ERROR_INTERNAL_ERROR;
899 WLog_ERR(TAG,
"Unknown cache data eventID: 0x%08" PRIX32
"", eventID);
900 return ERROR_INVALID_DATA;
905 case PAKID_PRN_USING_XPS:
909 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
910 return ERROR_INVALID_DATA;
912 Stream_Read_UINT32(s, flags);
914 "Ignoring unhandled message PAKID_PRN_USING_XPS [printerID=%08" PRIx32
915 ", flags=%08" PRIx32
"]",
921 WLog_ERR(TAG,
"Unknown printing component packetID: 0x%04" PRIX16
"", packetId);
922 return ERROR_INVALID_DATA;
925 return CHANNEL_RC_OK;
933static UINT printer_free(DEVICE* device)
935 PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
938 WINPR_ASSERT(printer_dev);
940 if (printer_dev->async)
942 (void)SetEvent(printer_dev->stopEvent);
944 if (WaitForSingleObject(printer_dev->thread, INFINITE) == WAIT_FAILED)
946 error = GetLastError();
947 WLog_ERR(TAG,
"WaitForSingleObject failed with error %" PRIu32
"", error);
952#ifndef __clang_analyzer__
957 while (Queue_Count(printer_dev->pIrpList) > 0)
959 IRP* irp = Queue_Dequeue(printer_dev->pIrpList);
961 WINPR_ASSERT(irp->Discard);
965 (void)CloseHandle(printer_dev->thread);
966 (void)CloseHandle(printer_dev->stopEvent);
967 (void)CloseHandle(printer_dev->event);
968 Queue_Free(printer_dev->pIrpList);
971 if (printer_dev->printer)
973 WINPR_ASSERT(printer_dev->printer->ReleaseRef);
974 printer_dev->printer->ReleaseRef(printer_dev->printer);
977 Stream_Free(printer_dev->device.data, TRUE);
979 return CHANNEL_RC_OK;
989 PRINTER_DEVICE* printer_dev =
nullptr;
990 UINT error = ERROR_INTERNAL_ERROR;
992 WINPR_ASSERT(pEntryPoints);
993 WINPR_ASSERT(printer);
995 printer_dev = (PRINTER_DEVICE*)calloc(1,
sizeof(PRINTER_DEVICE));
999 WLog_ERR(TAG,
"calloc failed!");
1000 return CHANNEL_RC_NO_MEMORY;
1003 printer_dev->device.data = Stream_New(
nullptr, 1024);
1005 if (!printer_dev->device.data)
1008 (void)sprintf_s(printer_dev->port,
sizeof(printer_dev->port),
"PRN%" PRIuz, printer->id);
1009 printer_dev->device.type = RDPDR_DTYP_PRINT;
1010 printer_dev->device.name = printer_dev->port;
1011 printer_dev->device.IRPRequest = printer_irp_request;
1012 printer_dev->device.CustomComponentRequest = printer_custom_component;
1013 printer_dev->device.Free = printer_free;
1014 printer_dev->rdpcontext = pEntryPoints->rdpcontext;
1015 printer_dev->printer = printer;
1018 FreeRDP_SynchronousStaticChannels))
1019 printer_dev->async = TRUE;
1021 if (!printer_load_from_config(pEntryPoints->rdpcontext->settings, printer, printer_dev))
1024 if (printer_dev->async)
1026 printer_dev->pIrpList = Queue_New(TRUE, -1, -1);
1028 if (!printer_dev->pIrpList)
1030 error = CHANNEL_RC_NO_MEMORY;
1034 printer_dev->event = CreateEvent(
nullptr, TRUE, FALSE,
nullptr);
1035 if (!printer_dev->event)
1037 WLog_ERR(TAG,
"CreateEvent failed!");
1038 error = ERROR_INTERNAL_ERROR;
1042 printer_dev->stopEvent = CreateEvent(
nullptr, TRUE, FALSE,
nullptr);
1043 if (!printer_dev->stopEvent)
1045 WLog_ERR(TAG,
"CreateEvent failed!");
1046 error = ERROR_INTERNAL_ERROR;
1051 error = pEntryPoints->RegisterDevice(pEntryPoints->devman, &printer_dev->device);
1054 WLog_ERR(TAG,
"RegisterDevice failed with error %" PRIu32
"!", error);
1058 if (printer_dev->async)
1060 printer_dev->thread =
1061 CreateThread(
nullptr, 0, printer_thread_func, (
void*)printer_dev, 0,
nullptr);
1062 if (!printer_dev->thread)
1064 WLog_ERR(TAG,
"CreateThread failed!");
1065 error = ERROR_INTERNAL_ERROR;
1070 WINPR_ASSERT(printer->AddRef);
1071 printer->AddRef(printer);
1072 return CHANNEL_RC_OK;
1074 printer_free(&printer_dev->device);
1078static rdpPrinterDriver* printer_load_backend(
const char* backend)
1080 typedef UINT(VCAPITYPE * backend_load_t)(rdpPrinterDriver**);
1081 PVIRTUALCHANNELENTRY entry = freerdp_load_channel_addin_entry(
"printer", backend,
nullptr, 0);
1082 backend_load_t func = WINPR_FUNC_PTR_CAST(entry, backend_load_t);
1086 rdpPrinterDriver* printer =
nullptr;
1087 const UINT rc = func(&printer);
1088 if (rc != CHANNEL_RC_OK)
1102 char* name =
nullptr;
1103 char* driver_name =
nullptr;
1104 BOOL default_backend = TRUE;
1106 rdpPrinterDriver* driver =
nullptr;
1107 UINT error = CHANNEL_RC_OK;
1109 if (!pEntryPoints || !pEntryPoints->device)
1110 return ERROR_INVALID_PARAMETER;
1113 name = device->device.Name;
1114 driver_name = _strdup(device->DriverName);
1123 char* sep = strstr(driver_name,
":");
1126 const char* backend = sep + 1;
1128 driver = printer_load_backend(backend);
1129 default_backend = FALSE;
1133 if (!driver && default_backend)
1135 const char* backend =
1136#if defined(WITH_CUPS)
1138#elif defined(_WIN32)
1145 driver = printer_load_backend(backend);
1150 WLog_ERR(TAG,
"Could not get a printer driver!");
1151 error = CHANNEL_RC_INITIALIZATION_ERROR;
1155 if (driver->SetDeviceContext)
1156 driver->SetDeviceContext(driver, device);
1158 if (name && name[0])
1160 WINPR_ASSERT(driver->GetPrinter);
1161 rdpPrinter* printer = driver->GetPrinter(driver, name, driver_name, device->IsDefault);
1165 WLog_ERR(TAG,
"Could not get printer %s!", name);
1166 error = CHANNEL_RC_INITIALIZATION_ERROR;
1170 WINPR_ASSERT(printer->ReleaseRef);
1171 if (!printer_save_default_config(pEntryPoints->rdpcontext->settings, printer))
1173 error = CHANNEL_RC_INITIALIZATION_ERROR;
1174 printer->ReleaseRef(printer);
1178 error = printer_register(pEntryPoints, printer);
1179 printer->ReleaseRef(printer);
1182 WLog_ERR(TAG,
"printer_register failed with error %" PRIu32
"!", error);
1188 WINPR_ASSERT(driver->EnumPrinters);
1189 rdpPrinter** printers = driver->EnumPrinters(driver);
1192 for (rdpPrinter** current = printers; *current; ++current)
1194 error = printer_register(pEntryPoints, *current);
1197 WLog_ERR(TAG,
"printer_register failed with error %" PRIu32
"!", error);
1204 WLog_ERR(TAG,
"Failed to enumerate printers!");
1205 error = CHANNEL_RC_INITIALIZATION_ERROR;
1208 WINPR_ASSERT(driver->ReleaseEnumPrinters);
1209 driver->ReleaseEnumPrinters(printers);
1216 WINPR_ASSERT(driver->ReleaseRef);
1217 driver->ReleaseRef(driver);
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.