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/path.h>
39#include <freerdp/channels/rdpdr.h>
40#include <freerdp/crypto/crypto.h>
41#include <freerdp/freerdp.h>
43#include "../printer.h"
45#include <freerdp/client/printer.h>
47#include <freerdp/channels/log.h>
49#define TAG CHANNELS_TAG("printer.client")
63 rdpContext* rdpcontext;
76static const char* filemap[] = {
"PortDosName",
"PnPName",
"DriverName",
77 "CachedPrinterConfigData" };
79static char* get_printer_config_path(
const rdpSettings* settings,
const WCHAR* name,
size_t length)
82 char* dir = GetCombinedPath(path,
"printers");
83 char* bname = crypto_base64_encode((
const BYTE*)name, length);
84 char* config = GetCombinedPath(dir, bname);
86 if (config && !winpr_PathFileExists(config))
88 if (!winpr_PathMakePath(config, NULL))
100static BOOL printer_write_setting(
const char* path, prn_conf_t type,
const void* data,
107 const char* name = filemap[type];
108 char* abs = GetCombinedPath(path, name);
110 if (!abs || (length > INT32_MAX))
116 file = CreateFileA(abs, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
119 if (file == INVALID_HANDLE_VALUE)
124 base64 = crypto_base64_encode(data, length);
131 const size_t b64len = strnlen(base64, 2 * length);
132 rc = WriteFile(file, base64, (UINT32)b64len, &written, NULL);
134 if (b64len != written)
141 (void)CloseHandle(file);
146static BOOL printer_config_valid(
const char* path)
151 if (!winpr_PathFileExists(path))
157static BOOL printer_read_setting(
const char* path, prn_conf_t type,
void** data, UINT32* length)
164 const char* name = filemap[type];
171 WLog_DBG(TAG,
"Printer option %s ignored", name);
175 char* abs = GetCombinedPath(path, name);
180 CreateFileA(abs, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
183 if (file == INVALID_HANDLE_VALUE)
186 lowSize = GetFileSize(file, &highSize);
188 if ((lowSize == INVALID_FILE_SIZE) || (highSize != 0))
193 fdata = malloc(lowSize);
198 rc = ReadFile(file, fdata, lowSize, &read, NULL);
205 (void)CloseHandle(file);
207 if (rc && (lowSize <= INT_MAX))
210 crypto_base64_decode(fdata, lowSize, (BYTE**)data, &blen);
212 if (*data && (blen > 0))
213 *length = (UINT32)blen;
230static BOOL printer_save_to_config(
const rdpSettings* settings,
const char* PortDosName,
231 size_t PortDosNameLen,
const WCHAR* PnPName,
size_t PnPNameLen,
232 const WCHAR* DriverName,
size_t DriverNameLen,
233 const WCHAR* PrinterName,
size_t PrintNameLen,
234 const BYTE* CachedPrinterConfigData,
size_t CacheFieldsLen)
237 char* path = get_printer_config_path(settings, PrinterName, PrintNameLen);
242 if (!printer_write_setting(path, PRN_CONF_PORT, PortDosName, PortDosNameLen))
245 if (!printer_write_setting(path, PRN_CONF_PNP, PnPName, PnPNameLen))
248 if (!printer_write_setting(path, PRN_CONF_DRIVER, DriverName, DriverNameLen))
251 if (!printer_write_setting(path, PRN_CONF_DATA, CachedPrinterConfigData, CacheFieldsLen))
259static BOOL printer_update_to_config(
const rdpSettings* settings,
const WCHAR* name,
size_t length,
260 const BYTE* data,
size_t datalen)
263 char* path = get_printer_config_path(settings, name, length);
264 rc = printer_write_setting(path, PRN_CONF_DATA, data, datalen);
269static BOOL printer_remove_config(
const rdpSettings* settings,
const WCHAR* name,
size_t length)
272 char* path = get_printer_config_path(settings, name, length);
274 if (!printer_config_valid(path))
277 rc = winpr_RemoveDirectory(path);
283static BOOL printer_move_config(
const rdpSettings* settings,
const WCHAR* oldName,
size_t oldLength,
284 const WCHAR* newName,
size_t newLength)
287 char* oldPath = get_printer_config_path(settings, oldName, oldLength);
288 char* newPath = get_printer_config_path(settings, newName, newLength);
290 if (printer_config_valid(oldPath))
291 rc = winpr_MoveFile(oldPath, newPath);
298static BOOL printer_load_from_config(
const rdpSettings* settings, rdpPrinter* printer,
299 PRINTER_DEVICE* printer_dev)
306 void* DriverName = NULL;
307 UINT32 DriverNameLen = 0;
308 void* PnPName = NULL;
309 UINT32 PnPNameLen = 0;
310 void* CachedPrinterConfigData = NULL;
311 UINT32 CachedFieldsLen = 0;
312 UINT32 PrinterNameLen = 0;
314 if (!settings || !printer || !printer->name)
317 wname = ConvertUtf8ToWCharAlloc(printer->name, &wlen);
323 path = get_printer_config_path(settings, wname, wlen *
sizeof(WCHAR));
325 const size_t plen = wlen *
sizeof(WCHAR);
326 if (plen > UINT32_MAX)
328 PrinterNameLen = (UINT32)plen;
334 if (printer->is_default)
335 flags |= RDPDR_PRINTER_ANNOUNCE_FLAG_DEFAULTPRINTER;
337 if (!printer_read_setting(path, PRN_CONF_PNP, &PnPName, &PnPNameLen))
341 if (!printer_read_setting(path, PRN_CONF_DRIVER, &DriverName, &DriverNameLen))
344 DriverName = ConvertUtf8ToWCharAlloc(printer->driver, &len);
347 const size_t dlen = (len + 1) *
sizeof(WCHAR);
348 if (dlen > UINT32_MAX)
350 DriverNameLen = (UINT32)dlen;
353 if (!printer_read_setting(path, PRN_CONF_DATA, &CachedPrinterConfigData, &CachedFieldsLen))
357 Stream_SetPosition(printer_dev->device.data, 0);
359 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, 24))
362 Stream_Write_UINT32(printer_dev->device.data, flags);
363 Stream_Write_UINT32(printer_dev->device.data, 0);
364 Stream_Write_UINT32(printer_dev->device.data, PnPNameLen);
365 Stream_Write_UINT32(printer_dev->device.data, DriverNameLen);
366 Stream_Write_UINT32(printer_dev->device.data, PrinterNameLen);
367 Stream_Write_UINT32(printer_dev->device.data, CachedFieldsLen);
369 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, PnPNameLen))
373 Stream_Write(printer_dev->device.data, PnPName, PnPNameLen);
375 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, DriverNameLen))
378 Stream_Write(printer_dev->device.data, DriverName, DriverNameLen);
380 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, PrinterNameLen))
388 backslash.c[0] =
'\\';
389 backslash.c[1] =
'\0';
391 for (WCHAR* wptr = wname; (wptr = _wcschr(wptr, backslash.w));)
393 Stream_Write(printer_dev->device.data, wname, PrinterNameLen);
395 if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, CachedFieldsLen))
398 Stream_Write(printer_dev->device.data, CachedPrinterConfigData, CachedFieldsLen);
405 free(CachedPrinterConfigData);
409static BOOL printer_save_default_config(
const rdpSettings* settings, rdpPrinter* printer)
413 WCHAR* driver = NULL;
418 if (!settings || !printer || !printer->name || !printer->driver)
421 wname = ConvertUtf8ToWCharAlloc(printer->name, NULL);
426 driver = ConvertUtf8ToWCharAlloc(printer->driver, NULL);
431 wlen = _wcslen(wname) + 1;
432 dlen = _wcslen(driver) + 1;
433 path = get_printer_config_path(settings, wname, wlen *
sizeof(WCHAR));
440 if (!printer_write_setting(path, PRN_CONF_DRIVER, driver, dlen *
sizeof(WCHAR)))
457static UINT printer_process_irp_create(PRINTER_DEVICE* printer_dev, IRP* irp)
459 rdpPrintJob* printjob = NULL;
461 WINPR_ASSERT(printer_dev);
464 if (printer_dev->printer)
466 WINPR_ASSERT(printer_dev->printer->CreatePrintJob);
468 printer_dev->printer->CreatePrintJob(printer_dev->printer, irp->devman->id_sequence++);
473 Stream_Write_UINT32(irp->output, printjob->id);
477 Stream_Write_UINT32(irp->output, 0);
478 irp->IoStatus = STATUS_PRINT_QUEUE_FULL;
481 WINPR_ASSERT(irp->Complete);
482 return irp->Complete(irp);
490static UINT printer_process_irp_close(PRINTER_DEVICE* printer_dev, IRP* irp)
492 rdpPrintJob* printjob = NULL;
494 WINPR_ASSERT(printer_dev);
497 if (printer_dev->printer)
499 WINPR_ASSERT(printer_dev->printer->FindPrintJob);
500 printjob = printer_dev->printer->FindPrintJob(printer_dev->printer, irp->FileId);
505 irp->IoStatus = STATUS_UNSUCCESSFUL;
509 printjob->Close(printjob);
512 Stream_Zero(irp->output, 4);
513 WINPR_ASSERT(irp->Complete);
514 return irp->Complete(irp);
522static UINT printer_process_irp_write(PRINTER_DEVICE* printer_dev, IRP* irp)
526 rdpPrintJob* printjob = NULL;
527 UINT error = CHANNEL_RC_OK;
529 WINPR_ASSERT(printer_dev);
532 if (!Stream_CheckAndLogRequiredLength(TAG, irp->input, 32))
533 return ERROR_INVALID_DATA;
534 Stream_Read_UINT32(irp->input, Length);
535 Stream_Read_UINT64(irp->input, Offset);
538 Stream_Seek(irp->input, 20);
539 const void* ptr = Stream_ConstPointer(irp->input);
540 if (!Stream_SafeSeek(irp->input, Length))
541 return ERROR_INVALID_DATA;
542 if (printer_dev->printer)
544 WINPR_ASSERT(printer_dev->printer->FindPrintJob);
545 printjob = printer_dev->printer->FindPrintJob(printer_dev->printer, irp->FileId);
550 irp->IoStatus = STATUS_UNSUCCESSFUL;
555 error = printjob->Write(printjob, ptr, Length);
560 WLog_ERR(TAG,
"printjob->Write failed with error %" PRIu32
"!", error);
564 Stream_Write_UINT32(irp->output, Length);
565 Stream_Write_UINT8(irp->output, 0);
567 WINPR_ASSERT(irp->Complete);
568 return irp->Complete(irp);
576static UINT printer_process_irp_device_control(WINPR_ATTR_UNUSED PRINTER_DEVICE* printer_dev,
579 WINPR_ASSERT(printer_dev);
582 Stream_Write_UINT32(irp->output, 0);
584 WINPR_ASSERT(irp->Complete);
585 return irp->Complete(irp);
593static UINT printer_process_irp(PRINTER_DEVICE* printer_dev, IRP* irp)
597 WINPR_ASSERT(printer_dev);
600 switch (irp->MajorFunction)
603 if ((error = printer_process_irp_create(printer_dev, irp)))
605 WLog_ERR(TAG,
"printer_process_irp_create failed with error %" PRIu32
"!", error);
612 if ((error = printer_process_irp_close(printer_dev, irp)))
614 WLog_ERR(TAG,
"printer_process_irp_close failed with error %" PRIu32
"!", error);
621 if ((error = printer_process_irp_write(printer_dev, irp)))
623 WLog_ERR(TAG,
"printer_process_irp_write failed with error %" PRIu32
"!", error);
629 case IRP_MJ_DEVICE_CONTROL:
630 if ((error = printer_process_irp_device_control(printer_dev, irp)))
632 WLog_ERR(TAG,
"printer_process_irp_device_control failed with error %" PRIu32
"!",
640 irp->IoStatus = STATUS_NOT_SUPPORTED;
641 WINPR_ASSERT(irp->Complete);
642 return irp->Complete(irp);
645 return CHANNEL_RC_OK;
648static DWORD WINAPI printer_thread_func(LPVOID arg)
651 PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)arg;
652 UINT error = CHANNEL_RC_OK;
654 WINPR_ASSERT(printer_dev);
658 HANDLE obj[] = { printer_dev->event, printer_dev->stopEvent };
659 DWORD rc = WaitForMultipleObjects(ARRAYSIZE(obj), obj, FALSE, INFINITE);
661 if (rc == WAIT_FAILED)
663 error = GetLastError();
664 WLog_ERR(TAG,
"WaitForMultipleObjects failed with error %" PRIu32
"!", error);
668 if (rc == WAIT_OBJECT_0 + 1)
670 else if (rc != WAIT_OBJECT_0)
673 (void)ResetEvent(printer_dev->event);
674 irp = (IRP*)InterlockedPopEntrySList(printer_dev->pIrpList);
678 WLog_ERR(TAG,
"InterlockedPopEntrySList failed!");
679 error = ERROR_INTERNAL_ERROR;
683 if ((error = printer_process_irp(printer_dev, irp)))
685 WLog_ERR(TAG,
"printer_process_irp failed with error %" PRIu32
"!", error);
690 if (error && printer_dev->rdpcontext)
691 setChannelError(printer_dev->rdpcontext, error,
"printer_thread_func reported an error");
702static UINT printer_irp_request(DEVICE* device, IRP* irp)
704 PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
706 WINPR_ASSERT(printer_dev);
709 if (printer_dev->async)
711 InterlockedPushEntrySList(printer_dev->pIrpList, &(irp->ItemEntry));
712 (void)SetEvent(printer_dev->event);
716 UINT error = printer_process_irp(printer_dev, irp);
719 WLog_ERR(TAG,
"printer_process_irp failed with error %" PRIu32
"!", error);
724 return CHANNEL_RC_OK;
727static UINT printer_custom_component(DEVICE* device, UINT16 component, UINT16 packetId,
wStream* s)
730 PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
732 WINPR_ASSERT(printer_dev);
733 WINPR_ASSERT(printer_dev->rdpcontext);
735 const rdpSettings* settings = printer_dev->rdpcontext->settings;
736 WINPR_ASSERT(settings);
738 if (component != RDPDR_CTYP_PRN)
739 return ERROR_INVALID_DATA;
741 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
742 return ERROR_INVALID_DATA;
744 Stream_Read_UINT32(s, eventID);
748 case PAKID_PRN_CACHE_DATA:
751 case RDPDR_ADD_PRINTER_EVENT:
754 UINT32 PnPNameLen = 0;
755 UINT32 DriverNameLen = 0;
756 UINT32 PrintNameLen = 0;
757 UINT32 CacheFieldsLen = 0;
758 const WCHAR* PnPName = NULL;
759 const WCHAR* DriverName = NULL;
760 const WCHAR* PrinterName = NULL;
761 const BYTE* CachedPrinterConfigData = NULL;
763 if (!Stream_CheckAndLogRequiredLength(TAG, s, 24))
764 return ERROR_INVALID_DATA;
766 Stream_Read(s, PortDosName,
sizeof(PortDosName));
767 Stream_Read_UINT32(s, PnPNameLen);
768 Stream_Read_UINT32(s, DriverNameLen);
769 Stream_Read_UINT32(s, PrintNameLen);
770 Stream_Read_UINT32(s, CacheFieldsLen);
772 if (!Stream_CheckAndLogRequiredLength(TAG, s, PnPNameLen))
773 return ERROR_INVALID_DATA;
775 PnPName = Stream_ConstPointer(s);
776 Stream_Seek(s, PnPNameLen);
778 if (!Stream_CheckAndLogRequiredLength(TAG, s, DriverNameLen))
779 return ERROR_INVALID_DATA;
781 DriverName = Stream_ConstPointer(s);
782 Stream_Seek(s, DriverNameLen);
784 if (!Stream_CheckAndLogRequiredLength(TAG, s, PrintNameLen))
785 return ERROR_INVALID_DATA;
787 PrinterName = Stream_ConstPointer(s);
788 Stream_Seek(s, PrintNameLen);
790 if (!Stream_CheckAndLogRequiredLength(TAG, s, CacheFieldsLen))
791 return ERROR_INVALID_DATA;
793 CachedPrinterConfigData = Stream_ConstPointer(s);
794 Stream_Seek(s, CacheFieldsLen);
796 if (!printer_save_to_config(settings, PortDosName,
sizeof(PortDosName), PnPName,
797 PnPNameLen, DriverName, DriverNameLen, PrinterName,
798 PrintNameLen, CachedPrinterConfigData,
800 return ERROR_INTERNAL_ERROR;
804 case RDPDR_UPDATE_PRINTER_EVENT:
806 UINT32 PrinterNameLen = 0;
807 UINT32 ConfigDataLen = 0;
808 const WCHAR* PrinterName = NULL;
809 const BYTE* ConfigData = NULL;
811 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
812 return ERROR_INVALID_DATA;
814 Stream_Read_UINT32(s, PrinterNameLen);
815 Stream_Read_UINT32(s, ConfigDataLen);
817 if (!Stream_CheckAndLogRequiredLength(TAG, s, PrinterNameLen))
818 return ERROR_INVALID_DATA;
820 PrinterName = Stream_ConstPointer(s);
821 Stream_Seek(s, PrinterNameLen);
823 if (!Stream_CheckAndLogRequiredLength(TAG, s, ConfigDataLen))
824 return ERROR_INVALID_DATA;
826 ConfigData = Stream_ConstPointer(s);
827 Stream_Seek(s, ConfigDataLen);
829 if (!printer_update_to_config(settings, PrinterName, PrinterNameLen, ConfigData,
831 return ERROR_INTERNAL_ERROR;
835 case RDPDR_DELETE_PRINTER_EVENT:
837 UINT32 PrinterNameLen = 0;
838 const WCHAR* PrinterName = NULL;
840 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
841 return ERROR_INVALID_DATA;
843 Stream_Read_UINT32(s, PrinterNameLen);
845 if (!Stream_CheckAndLogRequiredLength(TAG, s, PrinterNameLen))
846 return ERROR_INVALID_DATA;
848 PrinterName = Stream_ConstPointer(s);
849 Stream_Seek(s, PrinterNameLen);
850 printer_remove_config(settings, PrinterName, PrinterNameLen);
854 case RDPDR_RENAME_PRINTER_EVENT:
856 UINT32 OldPrinterNameLen = 0;
857 UINT32 NewPrinterNameLen = 0;
858 const WCHAR* OldPrinterName = NULL;
859 const WCHAR* NewPrinterName = NULL;
861 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
862 return ERROR_INVALID_DATA;
864 Stream_Read_UINT32(s, OldPrinterNameLen);
865 Stream_Read_UINT32(s, NewPrinterNameLen);
867 if (!Stream_CheckAndLogRequiredLength(TAG, s, OldPrinterNameLen))
868 return ERROR_INVALID_DATA;
870 OldPrinterName = Stream_ConstPointer(s);
871 Stream_Seek(s, OldPrinterNameLen);
873 if (!Stream_CheckAndLogRequiredLength(TAG, s, NewPrinterNameLen))
874 return ERROR_INVALID_DATA;
876 NewPrinterName = Stream_ConstPointer(s);
877 Stream_Seek(s, NewPrinterNameLen);
879 if (!printer_move_config(settings, OldPrinterName, OldPrinterNameLen,
880 NewPrinterName, NewPrinterNameLen))
881 return ERROR_INTERNAL_ERROR;
886 WLog_ERR(TAG,
"Unknown cache data eventID: 0x%08" PRIX32
"", eventID);
887 return ERROR_INVALID_DATA;
892 case PAKID_PRN_USING_XPS:
896 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
897 return ERROR_INVALID_DATA;
899 Stream_Read_UINT32(s, flags);
901 "Ignoring unhandled message PAKID_PRN_USING_XPS [printerID=%08" PRIx32
902 ", flags=%08" PRIx32
"]",
908 WLog_ERR(TAG,
"Unknown printing component packetID: 0x%04" PRIX16
"", packetId);
909 return ERROR_INVALID_DATA;
912 return CHANNEL_RC_OK;
920static UINT printer_free(DEVICE* device)
923 PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
926 WINPR_ASSERT(printer_dev);
928 if (printer_dev->async)
930 (void)SetEvent(printer_dev->stopEvent);
932 if (WaitForSingleObject(printer_dev->thread, INFINITE) == WAIT_FAILED)
934 error = GetLastError();
935 WLog_ERR(TAG,
"WaitForSingleObject failed with error %" PRIu32
"", error);
940#ifndef __clang_analyzer__
945 while ((irp = (IRP*)InterlockedPopEntrySList(printer_dev->pIrpList)) != NULL)
947 WINPR_ASSERT(irp->Discard);
951 (void)CloseHandle(printer_dev->thread);
952 (void)CloseHandle(printer_dev->stopEvent);
953 (void)CloseHandle(printer_dev->event);
954 winpr_aligned_free(printer_dev->pIrpList);
957 if (printer_dev->printer)
959 WINPR_ASSERT(printer_dev->printer->ReleaseRef);
960 printer_dev->printer->ReleaseRef(printer_dev->printer);
963 Stream_Free(printer_dev->device.data, TRUE);
965 return CHANNEL_RC_OK;
975 PRINTER_DEVICE* printer_dev = NULL;
976 UINT error = ERROR_INTERNAL_ERROR;
978 WINPR_ASSERT(pEntryPoints);
979 WINPR_ASSERT(printer);
981 printer_dev = (PRINTER_DEVICE*)calloc(1,
sizeof(PRINTER_DEVICE));
985 WLog_ERR(TAG,
"calloc failed!");
986 return CHANNEL_RC_NO_MEMORY;
989 printer_dev->device.data = Stream_New(NULL, 1024);
991 if (!printer_dev->device.data)
994 (void)sprintf_s(printer_dev->port,
sizeof(printer_dev->port),
"PRN%" PRIuz, printer->id);
995 printer_dev->device.type = RDPDR_DTYP_PRINT;
996 printer_dev->device.name = printer_dev->port;
997 printer_dev->device.IRPRequest = printer_irp_request;
998 printer_dev->device.CustomComponentRequest = printer_custom_component;
999 printer_dev->device.Free = printer_free;
1000 printer_dev->rdpcontext = pEntryPoints->rdpcontext;
1001 printer_dev->printer = printer;
1004 FreeRDP_SynchronousStaticChannels))
1005 printer_dev->async = TRUE;
1007 if (!printer_load_from_config(pEntryPoints->rdpcontext->settings, printer, printer_dev))
1010 if (printer_dev->async)
1015 if (!printer_dev->pIrpList)
1017 WLog_ERR(TAG,
"_aligned_malloc failed!");
1018 error = CHANNEL_RC_NO_MEMORY;
1022 InitializeSListHead(printer_dev->pIrpList);
1024 printer_dev->event = CreateEvent(NULL, TRUE, FALSE, NULL);
1025 if (!printer_dev->event)
1027 WLog_ERR(TAG,
"CreateEvent failed!");
1028 error = ERROR_INTERNAL_ERROR;
1032 printer_dev->stopEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
1033 if (!printer_dev->stopEvent)
1035 WLog_ERR(TAG,
"CreateEvent failed!");
1036 error = ERROR_INTERNAL_ERROR;
1041 error = pEntryPoints->RegisterDevice(pEntryPoints->devman, &printer_dev->device);
1044 WLog_ERR(TAG,
"RegisterDevice failed with error %" PRIu32
"!", error);
1048 if (printer_dev->async)
1050 printer_dev->thread =
1051 CreateThread(NULL, 0, printer_thread_func, (
void*)printer_dev, 0, NULL);
1052 if (!printer_dev->thread)
1054 WLog_ERR(TAG,
"CreateThread failed!");
1055 error = ERROR_INTERNAL_ERROR;
1060 WINPR_ASSERT(printer->AddRef);
1061 printer->AddRef(printer);
1062 return CHANNEL_RC_OK;
1064 printer_free(&printer_dev->device);
1068static rdpPrinterDriver* printer_load_backend(
const char* backend)
1070 typedef UINT(VCAPITYPE * backend_load_t)(rdpPrinterDriver**);
1071 PVIRTUALCHANNELENTRY entry = freerdp_load_channel_addin_entry(
"printer", backend, NULL, 0);
1072 backend_load_t func = WINPR_FUNC_PTR_CAST(entry, backend_load_t);
1076 rdpPrinterDriver* printer = NULL;
1077 const UINT rc = func(&printer);
1078 if (rc != CHANNEL_RC_OK)
1093 char* driver_name = NULL;
1094 BOOL default_backend = TRUE;
1096 rdpPrinterDriver* driver = NULL;
1097 UINT error = CHANNEL_RC_OK;
1099 if (!pEntryPoints || !pEntryPoints->device)
1100 return ERROR_INVALID_PARAMETER;
1103 name = device->device.Name;
1104 driver_name = _strdup(device->DriverName);
1113 char* sep = strstr(driver_name,
":");
1116 const char* backend = sep + 1;
1118 driver = printer_load_backend(backend);
1119 default_backend = FALSE;
1123 if (!driver && default_backend)
1125 const char* backend =
1126#if defined(WITH_CUPS)
1128#elif defined(_WIN32)
1135 driver = printer_load_backend(backend);
1140 WLog_ERR(TAG,
"Could not get a printer driver!");
1141 error = CHANNEL_RC_INITIALIZATION_ERROR;
1145 if (name && name[0])
1147 WINPR_ASSERT(driver->GetPrinter);
1148 rdpPrinter* printer = driver->GetPrinter(driver, name, driver_name, device->IsDefault);
1152 WLog_ERR(TAG,
"Could not get printer %s!", name);
1153 error = CHANNEL_RC_INITIALIZATION_ERROR;
1157 WINPR_ASSERT(printer->ReleaseRef);
1158 if (!printer_save_default_config(pEntryPoints->rdpcontext->settings, printer))
1160 error = CHANNEL_RC_INITIALIZATION_ERROR;
1161 printer->ReleaseRef(printer);
1165 error = printer_register(pEntryPoints, printer);
1166 printer->ReleaseRef(printer);
1169 WLog_ERR(TAG,
"printer_register failed with error %" PRIu32
"!", error);
1175 WINPR_ASSERT(driver->EnumPrinters);
1176 rdpPrinter** printers = driver->EnumPrinters(driver);
1179 for (rdpPrinter** current = printers; *current; ++current)
1181 error = printer_register(pEntryPoints, *current);
1184 WLog_ERR(TAG,
"printer_register failed with error %" PRIu32
"!", error);
1191 WLog_ERR(TAG,
"Failed to enumerate printers!");
1192 error = CHANNEL_RC_INITIALIZATION_ERROR;
1195 WINPR_ASSERT(driver->ReleaseEnumPrinters);
1196 driver->ReleaseEnumPrinters(printers);
1203 WINPR_ASSERT(driver->ReleaseRef);
1204 driver->ReleaseRef(driver);
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.