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> 
   40#include <freerdp/channels/rdpdr.h> 
   41#include <freerdp/crypto/crypto.h> 
   42#include <freerdp/freerdp.h> 
   44#include "../printer.h" 
   46#include <freerdp/client/printer.h> 
   48#include <freerdp/channels/log.h> 
   50#define TAG CHANNELS_TAG("printer.client") 
   64  rdpContext* rdpcontext;
 
   77static const char* filemap[] = { 
"PortDosName", 
"PnPName", 
"DriverName",
 
   78                               "CachedPrinterConfigData" };
 
   80static char* get_printer_config_path(
const rdpSettings* settings, 
const WCHAR* name, 
size_t length)
 
   83  char* dir = GetCombinedPath(path, 
"printers");
 
   84  char* bname = crypto_base64_encode((
const BYTE*)name, length);
 
   85  char* config = GetCombinedPath(dir, bname);
 
   87  if (config && !winpr_PathFileExists(config))
 
   89    if (!winpr_PathMakePath(config, NULL))
 
  101static BOOL printer_write_setting(
const char* path, prn_conf_t type, 
const void* data,
 
  108  const char* name = filemap[type];
 
  109  char* abs = GetCombinedPath(path, name);
 
  111  if (!abs || (length > INT32_MAX))
 
  118      winpr_CreateFile(abs, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
 
  121  if (file == INVALID_HANDLE_VALUE)
 
  126    base64 = crypto_base64_encode(data, length);
 
  133    const size_t b64len = strnlen(base64, 2 * length);
 
  134    rc = WriteFile(file, base64, (UINT32)b64len, &written, NULL);
 
  136    if (b64len != written)
 
  143  (void)CloseHandle(file);
 
  148static BOOL printer_config_valid(
const char* path)
 
  153  if (!winpr_PathFileExists(path))
 
  159static BOOL printer_read_setting(
const char* path, prn_conf_t type, 
void** data, UINT32* length)
 
  166  const char* name = filemap[type];
 
  173      WLog_DBG(TAG, 
"Printer option %s ignored", name);
 
  177  char* abs = GetCombinedPath(path, name);
 
  182      winpr_CreateFile(abs, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
 
  185  if (file == INVALID_HANDLE_VALUE)
 
  188  lowSize = GetFileSize(file, &highSize);
 
  190  if ((lowSize == INVALID_FILE_SIZE) || (highSize != 0))
 
  195    fdata = malloc(lowSize);
 
  200    rc = ReadFile(file, fdata, lowSize, &read, NULL);
 
  207  (void)CloseHandle(file);
 
  209  if (rc && (lowSize <= INT_MAX))
 
  212    crypto_base64_decode(fdata, lowSize, (BYTE**)data, &blen);
 
  214    if (*data && (blen > 0))
 
  215      *length = (UINT32)blen;
 
  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)
 
  239  char* path = get_printer_config_path(settings, PrinterName, PrintNameLen);
 
  244  if (!printer_write_setting(path, PRN_CONF_PORT, PortDosName, PortDosNameLen))
 
  247  if (!printer_write_setting(path, PRN_CONF_PNP, PnPName, PnPNameLen))
 
  250  if (!printer_write_setting(path, PRN_CONF_DRIVER, DriverName, DriverNameLen))
 
  253  if (!printer_write_setting(path, PRN_CONF_DATA, CachedPrinterConfigData, CacheFieldsLen))
 
  261static BOOL printer_update_to_config(
const rdpSettings* settings, 
const WCHAR* name, 
size_t length,
 
  262                                     const BYTE* data, 
size_t datalen)
 
  265  char* path = get_printer_config_path(settings, name, length);
 
  266  rc = printer_write_setting(path, PRN_CONF_DATA, data, datalen);
 
  271static BOOL printer_remove_config(
const rdpSettings* settings, 
const WCHAR* name, 
size_t length)
 
  274  char* path = get_printer_config_path(settings, name, length);
 
  276  if (!printer_config_valid(path))
 
  279  rc = winpr_RemoveDirectory(path);
 
  285static BOOL printer_move_config(
const rdpSettings* settings, 
const WCHAR* oldName, 
size_t oldLength,
 
  286                                const WCHAR* newName, 
size_t newLength)
 
  289  char* oldPath = get_printer_config_path(settings, oldName, oldLength);
 
  290  char* newPath = get_printer_config_path(settings, newName, newLength);
 
  292  if (printer_config_valid(oldPath))
 
  293    rc = winpr_MoveFile(oldPath, newPath);
 
  300static BOOL printer_load_from_config(
const rdpSettings* settings, rdpPrinter* printer,
 
  301                                     PRINTER_DEVICE* printer_dev)
 
  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;
 
  316  if (!settings || !printer || !printer->name)
 
  319  wname = ConvertUtf8ToWCharAlloc(printer->name, &wlen);
 
  325  path = get_printer_config_path(settings, wname, wlen * 
sizeof(WCHAR));
 
  327    const size_t plen = wlen * 
sizeof(WCHAR);
 
  328    if (plen > UINT32_MAX)
 
  330    PrinterNameLen = (UINT32)plen;
 
  336  if (printer->is_default)
 
  337    flags |= RDPDR_PRINTER_ANNOUNCE_FLAG_DEFAULTPRINTER;
 
  339  if (!printer_read_setting(path, PRN_CONF_PNP, &PnPName, &PnPNameLen))
 
  343  if (!printer_read_setting(path, PRN_CONF_DRIVER, &DriverName, &DriverNameLen))
 
  346    DriverName = ConvertUtf8ToWCharAlloc(printer->driver, &len);
 
  349    const size_t dlen = (len + 1) * 
sizeof(WCHAR);
 
  350    if (dlen > UINT32_MAX)
 
  352    DriverNameLen = (UINT32)dlen;
 
  355  if (!printer_read_setting(path, PRN_CONF_DATA, &CachedPrinterConfigData, &CachedFieldsLen))
 
  359  Stream_SetPosition(printer_dev->device.data, 0);
 
  361  if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, 24))
 
  364  Stream_Write_UINT32(printer_dev->device.data, flags);
 
  365  Stream_Write_UINT32(printer_dev->device.data, 0);          
 
  366  Stream_Write_UINT32(printer_dev->device.data, 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);
 
  371  if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, PnPNameLen))
 
  375    Stream_Write(printer_dev->device.data, PnPName, PnPNameLen);
 
  377  if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, DriverNameLen))
 
  380  Stream_Write(printer_dev->device.data, DriverName, DriverNameLen);
 
  382  if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, PrinterNameLen))
 
  390  backslash.c[0] = 
'\\';
 
  391  backslash.c[1] = 
'\0';
 
  393  for (WCHAR* wptr = wname; (wptr = _wcschr(wptr, backslash.w));)
 
  395  Stream_Write(printer_dev->device.data, wname, PrinterNameLen);
 
  397  if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, CachedFieldsLen))
 
  400  Stream_Write(printer_dev->device.data, CachedPrinterConfigData, CachedFieldsLen);
 
  407  free(CachedPrinterConfigData);
 
  411static BOOL printer_save_default_config(
const rdpSettings* settings, rdpPrinter* printer)
 
  415  WCHAR* driver = NULL;
 
  420  if (!settings || !printer || !printer->name || !printer->driver)
 
  423  wname = ConvertUtf8ToWCharAlloc(printer->name, NULL);
 
  428  driver = ConvertUtf8ToWCharAlloc(printer->driver, NULL);
 
  433  wlen = _wcslen(wname) + 1;
 
  434  dlen = _wcslen(driver) + 1;
 
  435  path = get_printer_config_path(settings, wname, wlen * 
sizeof(WCHAR));
 
  442    if (!printer_write_setting(path, PRN_CONF_DRIVER, driver, dlen * 
sizeof(WCHAR)))
 
  459static UINT printer_process_irp_create(PRINTER_DEVICE* printer_dev, IRP* irp)
 
  461  rdpPrintJob* printjob = NULL;
 
  463  WINPR_ASSERT(printer_dev);
 
  466  if (printer_dev->printer)
 
  468    WINPR_ASSERT(printer_dev->printer->CreatePrintJob);
 
  470        printer_dev->printer->CreatePrintJob(printer_dev->printer, irp->devman->id_sequence++);
 
  475    Stream_Write_UINT32(irp->output, printjob->id); 
 
  479    Stream_Write_UINT32(irp->output, 0); 
 
  480    irp->IoStatus = STATUS_PRINT_QUEUE_FULL;
 
  483  WINPR_ASSERT(irp->Complete);
 
  484  return irp->Complete(irp);
 
  492static UINT printer_process_irp_close(PRINTER_DEVICE* printer_dev, IRP* irp)
 
  494  rdpPrintJob* printjob = NULL;
 
  496  WINPR_ASSERT(printer_dev);
 
  499  if (printer_dev->printer)
 
  501    WINPR_ASSERT(printer_dev->printer->FindPrintJob);
 
  502    printjob = printer_dev->printer->FindPrintJob(printer_dev->printer, irp->FileId);
 
  507    irp->IoStatus = STATUS_UNSUCCESSFUL;
 
  511    printjob->Close(printjob);
 
  514  Stream_Zero(irp->output, 4); 
 
  515  WINPR_ASSERT(irp->Complete);
 
  516  return irp->Complete(irp);
 
  524static UINT printer_process_irp_write(PRINTER_DEVICE* printer_dev, IRP* irp)
 
  528  rdpPrintJob* printjob = NULL;
 
  529  UINT error = CHANNEL_RC_OK;
 
  531  WINPR_ASSERT(printer_dev);
 
  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);
 
  540  Stream_Seek(irp->input, 20); 
 
  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)
 
  546    WINPR_ASSERT(printer_dev->printer->FindPrintJob);
 
  547    printjob = printer_dev->printer->FindPrintJob(printer_dev->printer, irp->FileId);
 
  552    irp->IoStatus = STATUS_UNSUCCESSFUL;
 
  557    error = printjob->Write(printjob, ptr, Length);
 
  562    WLog_ERR(TAG, 
"printjob->Write failed with error %" PRIu32 
"!", error);
 
  566  Stream_Write_UINT32(irp->output, Length);
 
  567  Stream_Write_UINT8(irp->output, 0); 
 
  569  WINPR_ASSERT(irp->Complete);
 
  570  return irp->Complete(irp);
 
  578static UINT printer_process_irp_device_control(WINPR_ATTR_UNUSED PRINTER_DEVICE* printer_dev,
 
  581  WINPR_ASSERT(printer_dev);
 
  584  Stream_Write_UINT32(irp->output, 0); 
 
  586  WINPR_ASSERT(irp->Complete);
 
  587  return irp->Complete(irp);
 
  595static UINT printer_process_irp(PRINTER_DEVICE* printer_dev, IRP* irp)
 
  599  WINPR_ASSERT(printer_dev);
 
  602  switch (irp->MajorFunction)
 
  605      if ((error = printer_process_irp_create(printer_dev, irp)))
 
  607        WLog_ERR(TAG, 
"printer_process_irp_create failed with error %" PRIu32 
"!", error);
 
  614      if ((error = printer_process_irp_close(printer_dev, irp)))
 
  616        WLog_ERR(TAG, 
"printer_process_irp_close failed with error %" PRIu32 
"!", error);
 
  623      if ((error = printer_process_irp_write(printer_dev, irp)))
 
  625        WLog_ERR(TAG, 
"printer_process_irp_write failed with error %" PRIu32 
"!", error);
 
  631    case IRP_MJ_DEVICE_CONTROL:
 
  632      if ((error = printer_process_irp_device_control(printer_dev, irp)))
 
  634        WLog_ERR(TAG, 
"printer_process_irp_device_control failed with error %" PRIu32 
"!",
 
  642      irp->IoStatus = STATUS_NOT_SUPPORTED;
 
  643      WINPR_ASSERT(irp->Complete);
 
  644      return irp->Complete(irp);
 
  647  return CHANNEL_RC_OK;
 
  650static DWORD WINAPI printer_thread_func(LPVOID arg)
 
  653  PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)arg;
 
  654  UINT error = CHANNEL_RC_OK;
 
  656  WINPR_ASSERT(printer_dev);
 
  660    HANDLE obj[] = { printer_dev->event, printer_dev->stopEvent };
 
  661    DWORD rc = WaitForMultipleObjects(ARRAYSIZE(obj), obj, FALSE, INFINITE);
 
  663    if (rc == WAIT_FAILED)
 
  665      error = GetLastError();
 
  666      WLog_ERR(TAG, 
"WaitForMultipleObjects failed with error %" PRIu32 
"!", error);
 
  670    if (rc == WAIT_OBJECT_0 + 1)
 
  672    else if (rc != WAIT_OBJECT_0)
 
  675    (void)ResetEvent(printer_dev->event);
 
  676    irp = (IRP*)InterlockedPopEntrySList(printer_dev->pIrpList);
 
  680      WLog_ERR(TAG, 
"InterlockedPopEntrySList failed!");
 
  681      error = ERROR_INTERNAL_ERROR;
 
  685    if ((error = printer_process_irp(printer_dev, irp)))
 
  687      WLog_ERR(TAG, 
"printer_process_irp failed with error %" PRIu32 
"!", error);
 
  692  if (error && printer_dev->rdpcontext)
 
  693    setChannelError(printer_dev->rdpcontext, error, 
"printer_thread_func reported an error");
 
  704static UINT printer_irp_request(DEVICE* device, IRP* irp)
 
  706  PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
 
  708  WINPR_ASSERT(printer_dev);
 
  711  if (printer_dev->async)
 
  713    InterlockedPushEntrySList(printer_dev->pIrpList, &(irp->ItemEntry));
 
  714    (void)SetEvent(printer_dev->event);
 
  718    UINT error = printer_process_irp(printer_dev, irp);
 
  721      WLog_ERR(TAG, 
"printer_process_irp failed with error %" PRIu32 
"!", error);
 
  726  return CHANNEL_RC_OK;
 
  729static UINT printer_custom_component(DEVICE* device, UINT16 component, UINT16 packetId, 
wStream* s)
 
  732  PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
 
  734  WINPR_ASSERT(printer_dev);
 
  735  WINPR_ASSERT(printer_dev->rdpcontext);
 
  737  const rdpSettings* settings = printer_dev->rdpcontext->settings;
 
  738  WINPR_ASSERT(settings);
 
  740  if (component != RDPDR_CTYP_PRN)
 
  741    return ERROR_INVALID_DATA;
 
  743  if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
 
  744    return ERROR_INVALID_DATA;
 
  746  Stream_Read_UINT32(s, eventID);
 
  750    case PAKID_PRN_CACHE_DATA:
 
  753        case RDPDR_ADD_PRINTER_EVENT:
 
  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;
 
  765          if (!Stream_CheckAndLogRequiredLength(TAG, s, 24))
 
  766            return ERROR_INVALID_DATA;
 
  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);
 
  774          if (!Stream_CheckAndLogRequiredLength(TAG, s, PnPNameLen))
 
  775            return ERROR_INVALID_DATA;
 
  777          PnPName = Stream_ConstPointer(s);
 
  778          Stream_Seek(s, PnPNameLen);
 
  780          if (!Stream_CheckAndLogRequiredLength(TAG, s, DriverNameLen))
 
  781            return ERROR_INVALID_DATA;
 
  783          DriverName = Stream_ConstPointer(s);
 
  784          Stream_Seek(s, DriverNameLen);
 
  786          if (!Stream_CheckAndLogRequiredLength(TAG, s, PrintNameLen))
 
  787            return ERROR_INVALID_DATA;
 
  789          PrinterName = Stream_ConstPointer(s);
 
  790          Stream_Seek(s, PrintNameLen);
 
  792          if (!Stream_CheckAndLogRequiredLength(TAG, s, CacheFieldsLen))
 
  793            return ERROR_INVALID_DATA;
 
  795          CachedPrinterConfigData = Stream_ConstPointer(s);
 
  796          Stream_Seek(s, CacheFieldsLen);
 
  798          if (!printer_save_to_config(settings, PortDosName, 
sizeof(PortDosName), PnPName,
 
  799                                      PnPNameLen, DriverName, DriverNameLen, PrinterName,
 
  800                                      PrintNameLen, CachedPrinterConfigData,
 
  802            return ERROR_INTERNAL_ERROR;
 
  806        case RDPDR_UPDATE_PRINTER_EVENT:
 
  808          UINT32 PrinterNameLen = 0;
 
  809          UINT32 ConfigDataLen = 0;
 
  810          const WCHAR* PrinterName = NULL;
 
  811          const BYTE* ConfigData = NULL;
 
  813          if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
 
  814            return ERROR_INVALID_DATA;
 
  816          Stream_Read_UINT32(s, PrinterNameLen);
 
  817          Stream_Read_UINT32(s, ConfigDataLen);
 
  819          if (!Stream_CheckAndLogRequiredLength(TAG, s, PrinterNameLen))
 
  820            return ERROR_INVALID_DATA;
 
  822          PrinterName = Stream_ConstPointer(s);
 
  823          Stream_Seek(s, PrinterNameLen);
 
  825          if (!Stream_CheckAndLogRequiredLength(TAG, s, ConfigDataLen))
 
  826            return ERROR_INVALID_DATA;
 
  828          ConfigData = Stream_ConstPointer(s);
 
  829          Stream_Seek(s, ConfigDataLen);
 
  831          if (!printer_update_to_config(settings, PrinterName, PrinterNameLen, ConfigData,
 
  833            return ERROR_INTERNAL_ERROR;
 
  837        case RDPDR_DELETE_PRINTER_EVENT:
 
  839          UINT32 PrinterNameLen = 0;
 
  840          const WCHAR* PrinterName = NULL;
 
  842          if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
 
  843            return ERROR_INVALID_DATA;
 
  845          Stream_Read_UINT32(s, PrinterNameLen);
 
  847          if (!Stream_CheckAndLogRequiredLength(TAG, s, PrinterNameLen))
 
  848            return ERROR_INVALID_DATA;
 
  850          PrinterName = Stream_ConstPointer(s);
 
  851          Stream_Seek(s, PrinterNameLen);
 
  852          printer_remove_config(settings, PrinterName, PrinterNameLen);
 
  856        case RDPDR_RENAME_PRINTER_EVENT:
 
  858          UINT32 OldPrinterNameLen = 0;
 
  859          UINT32 NewPrinterNameLen = 0;
 
  860          const WCHAR* OldPrinterName = NULL;
 
  861          const WCHAR* NewPrinterName = NULL;
 
  863          if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
 
  864            return ERROR_INVALID_DATA;
 
  866          Stream_Read_UINT32(s, OldPrinterNameLen);
 
  867          Stream_Read_UINT32(s, NewPrinterNameLen);
 
  869          if (!Stream_CheckAndLogRequiredLength(TAG, s, OldPrinterNameLen))
 
  870            return ERROR_INVALID_DATA;
 
  872          OldPrinterName = Stream_ConstPointer(s);
 
  873          Stream_Seek(s, OldPrinterNameLen);
 
  875          if (!Stream_CheckAndLogRequiredLength(TAG, s, NewPrinterNameLen))
 
  876            return ERROR_INVALID_DATA;
 
  878          NewPrinterName = Stream_ConstPointer(s);
 
  879          Stream_Seek(s, NewPrinterNameLen);
 
  881          if (!printer_move_config(settings, OldPrinterName, OldPrinterNameLen,
 
  882                                   NewPrinterName, NewPrinterNameLen))
 
  883            return ERROR_INTERNAL_ERROR;
 
  888          WLog_ERR(TAG, 
"Unknown cache data eventID: 0x%08" PRIX32 
"", eventID);
 
  889          return ERROR_INVALID_DATA;
 
  894    case PAKID_PRN_USING_XPS:
 
  898      if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
 
  899        return ERROR_INVALID_DATA;
 
  901      Stream_Read_UINT32(s, flags);
 
  903               "Ignoring unhandled message PAKID_PRN_USING_XPS [printerID=%08" PRIx32
 
  904               ", flags=%08" PRIx32 
"]",
 
  910      WLog_ERR(TAG, 
"Unknown printing component packetID: 0x%04" PRIX16 
"", packetId);
 
  911      return ERROR_INVALID_DATA;
 
  914  return CHANNEL_RC_OK;
 
  922static UINT printer_free(DEVICE* device)
 
  925  PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
 
  928  WINPR_ASSERT(printer_dev);
 
  930  if (printer_dev->async)
 
  932    (void)SetEvent(printer_dev->stopEvent);
 
  934    if (WaitForSingleObject(printer_dev->thread, INFINITE) == WAIT_FAILED)
 
  936      error = GetLastError();
 
  937      WLog_ERR(TAG, 
"WaitForSingleObject failed with error %" PRIu32 
"", error);
 
  942#ifndef __clang_analyzer__ 
  947    while ((irp = (IRP*)InterlockedPopEntrySList(printer_dev->pIrpList)) != NULL)
 
  949      WINPR_ASSERT(irp->Discard);
 
  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);
 
  959  if (printer_dev->printer)
 
  961    WINPR_ASSERT(printer_dev->printer->ReleaseRef);
 
  962    printer_dev->printer->ReleaseRef(printer_dev->printer);
 
  965  Stream_Free(printer_dev->device.data, TRUE);
 
  967  return CHANNEL_RC_OK;
 
  977  PRINTER_DEVICE* printer_dev = NULL;
 
  978  UINT error = ERROR_INTERNAL_ERROR;
 
  980  WINPR_ASSERT(pEntryPoints);
 
  981  WINPR_ASSERT(printer);
 
  983  printer_dev = (PRINTER_DEVICE*)calloc(1, 
sizeof(PRINTER_DEVICE));
 
  987    WLog_ERR(TAG, 
"calloc failed!");
 
  988    return CHANNEL_RC_NO_MEMORY;
 
  991  printer_dev->device.data = Stream_New(NULL, 1024);
 
  993  if (!printer_dev->device.data)
 
  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;
 
 1006                                 FreeRDP_SynchronousStaticChannels))
 
 1007    printer_dev->async = TRUE;
 
 1009  if (!printer_load_from_config(pEntryPoints->rdpcontext->settings, printer, printer_dev))
 
 1012  if (printer_dev->async)
 
 1017    if (!printer_dev->pIrpList)
 
 1019      WLog_ERR(TAG, 
"_aligned_malloc failed!");
 
 1020      error = CHANNEL_RC_NO_MEMORY;
 
 1024    InitializeSListHead(printer_dev->pIrpList);
 
 1026    printer_dev->event = CreateEvent(NULL, TRUE, FALSE, NULL);
 
 1027    if (!printer_dev->event)
 
 1029      WLog_ERR(TAG, 
"CreateEvent failed!");
 
 1030      error = ERROR_INTERNAL_ERROR;
 
 1034    printer_dev->stopEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
 
 1035    if (!printer_dev->stopEvent)
 
 1037      WLog_ERR(TAG, 
"CreateEvent failed!");
 
 1038      error = ERROR_INTERNAL_ERROR;
 
 1043  error = pEntryPoints->RegisterDevice(pEntryPoints->devman, &printer_dev->device);
 
 1046    WLog_ERR(TAG, 
"RegisterDevice failed with error %" PRIu32 
"!", error);
 
 1050  if (printer_dev->async)
 
 1052    printer_dev->thread =
 
 1053        CreateThread(NULL, 0, printer_thread_func, (
void*)printer_dev, 0, NULL);
 
 1054    if (!printer_dev->thread)
 
 1056      WLog_ERR(TAG, 
"CreateThread failed!");
 
 1057      error = ERROR_INTERNAL_ERROR;
 
 1062  WINPR_ASSERT(printer->AddRef);
 
 1063  printer->AddRef(printer);
 
 1064  return CHANNEL_RC_OK;
 
 1066  printer_free(&printer_dev->device);
 
 1070static rdpPrinterDriver* printer_load_backend(
const char* backend)
 
 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);
 
 1078  rdpPrinterDriver* printer = NULL;
 
 1079  const UINT rc = func(&printer);
 
 1080  if (rc != CHANNEL_RC_OK)
 
 1095  char* driver_name = NULL;
 
 1096  BOOL default_backend = TRUE;
 
 1098  rdpPrinterDriver* driver = NULL;
 
 1099  UINT error = CHANNEL_RC_OK;
 
 1101  if (!pEntryPoints || !pEntryPoints->device)
 
 1102    return ERROR_INVALID_PARAMETER;
 
 1105  name = device->device.Name;
 
 1106  driver_name = _strdup(device->DriverName);
 
 1115    char* sep = strstr(driver_name, 
":");
 
 1118      const char* backend = sep + 1;
 
 1120      driver = printer_load_backend(backend);
 
 1121      default_backend = FALSE;
 
 1125  if (!driver && default_backend)
 
 1127    const char* backend =
 
 1128#if defined(WITH_CUPS) 
 1130#elif defined(_WIN32) 
 1137    driver = printer_load_backend(backend);
 
 1142    WLog_ERR(TAG, 
"Could not get a printer driver!");
 
 1143    error = CHANNEL_RC_INITIALIZATION_ERROR;
 
 1147  if (name && name[0])
 
 1149    WINPR_ASSERT(driver->GetPrinter);
 
 1150    rdpPrinter* printer = driver->GetPrinter(driver, name, driver_name, device->IsDefault);
 
 1154      WLog_ERR(TAG, 
"Could not get printer %s!", name);
 
 1155      error = CHANNEL_RC_INITIALIZATION_ERROR;
 
 1159    WINPR_ASSERT(printer->ReleaseRef);
 
 1160    if (!printer_save_default_config(pEntryPoints->rdpcontext->settings, printer))
 
 1162      error = CHANNEL_RC_INITIALIZATION_ERROR;
 
 1163      printer->ReleaseRef(printer);
 
 1167    error = printer_register(pEntryPoints, printer);
 
 1168    printer->ReleaseRef(printer);
 
 1171      WLog_ERR(TAG, 
"printer_register failed with error %" PRIu32 
"!", error);
 
 1177    WINPR_ASSERT(driver->EnumPrinters);
 
 1178    rdpPrinter** printers = driver->EnumPrinters(driver);
 
 1181      for (rdpPrinter** current = printers; *current; ++current)
 
 1183        error = printer_register(pEntryPoints, *current);
 
 1186          WLog_ERR(TAG, 
"printer_register failed with error %" PRIu32 
"!", error);
 
 1193      WLog_ERR(TAG, 
"Failed to enumerate printers!");
 
 1194      error = CHANNEL_RC_INITIALIZATION_ERROR;
 
 1197    WINPR_ASSERT(driver->ReleaseEnumPrinters);
 
 1198    driver->ReleaseEnumPrinters(printers);
 
 1205    WINPR_ASSERT(driver->ReleaseRef);
 
 1206    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.