Blame channels/printer/client/printer_main.c

Packit Service fa4841
/**
Packit Service fa4841
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit Service fa4841
 * Print Virtual Channel
Packit Service fa4841
 *
Packit Service fa4841
 * Copyright 2010-2011 Vic Lee
Packit Service fa4841
 * Copyright 2015 Thincast Technologies GmbH
Packit Service fa4841
 * Copyright 2015 DI (FH) Martin Haimberger <martin.haimberger@thincast.com>
Packit Service fa4841
 * Copyright 2016 Armin Novak <armin.novak@gmail.com>
Packit Service fa4841
 * Copyright 2016 David PHAM-VAN <d.phamvan@inuvika.com>
Packit Service fa4841
 *
Packit Service fa4841
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit Service fa4841
 * you may not use this file except in compliance with the License.
Packit Service fa4841
 * You may obtain a copy of the License at
Packit Service fa4841
 *
Packit Service fa4841
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit Service fa4841
 *
Packit Service fa4841
 * Unless required by applicable law or agreed to in writing, software
Packit Service fa4841
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit Service fa4841
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit Service fa4841
 * See the License for the specific language governing permissions and
Packit Service fa4841
 * limitations under the License.
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
#ifdef HAVE_CONFIG_H
Packit Service fa4841
#include "config.h"
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#include <stdio.h>
Packit Service fa4841
#include <stdlib.h>
Packit Service fa4841
#include <string.h>
Packit Service fa4841
Packit Service fa4841
#include <winpr/crt.h>
Packit Service fa4841
#include <winpr/string.h>
Packit Service fa4841
#include <winpr/synch.h>
Packit Service fa4841
#include <winpr/thread.h>
Packit Service fa4841
#include <winpr/stream.h>
Packit Service fa4841
#include <winpr/interlocked.h>
Packit Service b1ea74
#include <winpr/path.h>
Packit Service fa4841
Packit Service fa4841
#include <freerdp/channels/rdpdr.h>
Packit Service b1ea74
#include <freerdp/crypto/crypto.h>
Packit Service fa4841
Packit Service fa4841
#include "../printer.h"
Packit Service fa4841
Packit Service b1ea74
#include <freerdp/client/printer.h>
Packit Service fa4841
Packit Service fa4841
#include <freerdp/channels/log.h>
Packit Service fa4841
Packit Service fa4841
#define TAG CHANNELS_TAG("printer.client")
Packit Service fa4841
Packit Service fa4841
typedef struct _PRINTER_DEVICE PRINTER_DEVICE;
Packit Service fa4841
struct _PRINTER_DEVICE
Packit Service fa4841
{
Packit Service fa4841
	DEVICE device;
Packit Service fa4841
Packit Service fa4841
	rdpPrinter* printer;
Packit Service fa4841
Packit Service fa4841
	WINPR_PSLIST_HEADER pIrpList;
Packit Service fa4841
Packit Service fa4841
	HANDLE event;
Packit Service fa4841
	HANDLE stopEvent;
Packit Service fa4841
Packit Service fa4841
	HANDLE thread;
Packit Service fa4841
	rdpContext* rdpcontext;
Packit Service fa4841
	char port[64];
Packit Service fa4841
};
Packit Service fa4841
Packit Service b1ea74
typedef enum
Packit Service b1ea74
{
Packit Service b1ea74
	PRN_CONF_PORT = 0,
Packit Service b1ea74
	PRN_CONF_PNP = 1,
Packit Service b1ea74
	PRN_CONF_DRIVER = 2,
Packit Service b1ea74
	PRN_CONF_DATA = 3
Packit Service b1ea74
} prn_conf_t;
Packit Service b1ea74
Packit Service b1ea74
static const char* filemap[] = { "PortDosName", "PnPName", "DriverName",
Packit Service b1ea74
	                             "CachedPrinterConfigData" };
Packit Service b1ea74
Packit Service b1ea74
static char* get_printer_config_path(const rdpSettings* settings, const WCHAR* name, size_t length)
Packit Service b1ea74
{
Packit Service b1ea74
	char* dir = GetCombinedPath(settings->ConfigPath, "printers");
Packit Service b1ea74
	char* bname = crypto_base64_encode((const BYTE*)name, (int)length);
Packit Service b1ea74
	char* config = GetCombinedPath(dir, bname);
Packit Service b1ea74
Packit Service b1ea74
	if (config && !PathFileExistsA(config))
Packit Service b1ea74
	{
Packit Service b1ea74
		if (!PathMakePathA(config, NULL))
Packit Service b1ea74
		{
Packit Service b1ea74
			free(config);
Packit Service b1ea74
			config = NULL;
Packit Service b1ea74
		}
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	free(dir);
Packit Service b1ea74
	free(bname);
Packit Service b1ea74
	return config;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL printer_write_setting(const char* path, prn_conf_t type, const void* data,
Packit Service b1ea74
                                  size_t length)
Packit Service b1ea74
{
Packit Service b1ea74
	DWORD written = 0;
Packit Service b1ea74
	BOOL rc = FALSE;
Packit Service b1ea74
	HANDLE file;
Packit Service b1ea74
	size_t b64len;
Packit Service b1ea74
	char* base64 = NULL;
Packit Service b1ea74
	const char* name = filemap[type];
Packit Service b1ea74
	char* abs = GetCombinedPath(path, name);
Packit Service b1ea74
Packit Service b1ea74
	if (!abs || (length > INT32_MAX))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	file = CreateFileA(abs, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
Packit Service b1ea74
	free(abs);
Packit Service b1ea74
Packit Service b1ea74
	if (file == INVALID_HANDLE_VALUE)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	if (length > 0)
Packit Service b1ea74
	{
Packit Service b1ea74
		base64 = crypto_base64_encode(data, length);
Packit Service b1ea74
Packit Service b1ea74
		if (!base64)
Packit Service b1ea74
			goto fail;
Packit Service b1ea74
Packit Service b1ea74
		/* base64 char represents 6bit -> 4*(n/3) is the length which is
Packit Service b1ea74
		 * always smaller than 2*n */
Packit Service b1ea74
		b64len = strnlen(base64, 2 * length);
Packit Service b1ea74
		rc = WriteFile(file, base64, b64len, &written, NULL);
Packit Service b1ea74
Packit Service b1ea74
		if (b64len != written)
Packit Service b1ea74
			rc = FALSE;
Packit Service b1ea74
	}
Packit Service b1ea74
	else
Packit Service b1ea74
		rc = TRUE;
Packit Service b1ea74
Packit Service b1ea74
fail:
Packit Service b1ea74
	CloseHandle(file);
Packit Service b1ea74
	free(base64);
Packit Service b1ea74
	return rc;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL printer_config_valid(const char* path)
Packit Service b1ea74
{
Packit Service b1ea74
	if (!path)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	if (!PathFileExistsA(path))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL printer_read_setting(const char* path, prn_conf_t type, void** data, UINT32* length)
Packit Service b1ea74
{
Packit Service b1ea74
	DWORD lowSize, highSize;
Packit Service b1ea74
	DWORD read = 0;
Packit Service b1ea74
	BOOL rc = FALSE;
Packit Service b1ea74
	HANDLE file;
Packit Service b1ea74
	char* fdata = NULL;
Packit Service b1ea74
	const char* name = filemap[type];
Packit Service b1ea74
	char* abs = GetCombinedPath(path, name);
Packit Service b1ea74
Packit Service b1ea74
	if (!abs)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	file = CreateFileA(abs, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
Packit Service b1ea74
	free(abs);
Packit Service b1ea74
Packit Service b1ea74
	if (file == INVALID_HANDLE_VALUE)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	lowSize = GetFileSize(file, &highSize);
Packit Service b1ea74
Packit Service b1ea74
	if ((lowSize == INVALID_FILE_SIZE) || (highSize != 0))
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	if (lowSize != 0)
Packit Service b1ea74
	{
Packit Service b1ea74
		fdata = malloc(lowSize);
Packit Service b1ea74
Packit Service b1ea74
		if (!fdata)
Packit Service b1ea74
			goto fail;
Packit Service b1ea74
Packit Service b1ea74
		rc = ReadFile(file, fdata, lowSize, &read, NULL);
Packit Service b1ea74
Packit Service b1ea74
		if (lowSize != read)
Packit Service b1ea74
			rc = FALSE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
fail:
Packit Service b1ea74
	CloseHandle(file);
Packit Service b1ea74
Packit Service b1ea74
	if (rc && (lowSize <= INT_MAX))
Packit Service b1ea74
	{
Packit Service b1ea74
		int blen = 0;
Packit Service b1ea74
		crypto_base64_decode(fdata, (int)lowSize, (BYTE**)data, &blen);
Packit Service b1ea74
Packit Service b1ea74
		if (*data && (blen > 0))
Packit Service b1ea74
			*length = (UINT32)blen;
Packit Service b1ea74
		else
Packit Service b1ea74
		{
Packit Service b1ea74
			rc = FALSE;
Packit Service b1ea74
			*length = 0;
Packit Service b1ea74
		}
Packit Service b1ea74
	}
Packit Service b1ea74
	else
Packit Service b1ea74
	{
Packit Service b1ea74
		*length = 0;
Packit Service b1ea74
		*data = NULL;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	free(fdata);
Packit Service b1ea74
	return rc;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL printer_save_to_config(const rdpSettings* settings, const char* PortDosName,
Packit Service b1ea74
                                   size_t PortDosNameLen, const WCHAR* PnPName, size_t PnPNameLen,
Packit Service b1ea74
                                   const WCHAR* DriverName, size_t DriverNameLen,
Packit Service b1ea74
                                   const WCHAR* PrinterName, size_t PrintNameLen,
Packit Service b1ea74
                                   const BYTE* CachedPrinterConfigData, size_t CacheFieldsLen)
Packit Service b1ea74
{
Packit Service b1ea74
	BOOL rc = FALSE;
Packit Service b1ea74
	char* path = get_printer_config_path(settings, PrinterName, PrintNameLen);
Packit Service b1ea74
Packit Service b1ea74
	if (!path)
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	if (!printer_write_setting(path, PRN_CONF_PORT, PortDosName, PortDosNameLen))
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	if (!printer_write_setting(path, PRN_CONF_PNP, PnPName, PnPNameLen))
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	if (!printer_write_setting(path, PRN_CONF_DRIVER, DriverName, DriverNameLen))
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	if (!printer_write_setting(path, PRN_CONF_DATA, CachedPrinterConfigData, CacheFieldsLen))
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
fail:
Packit Service b1ea74
	free(path);
Packit Service b1ea74
	return rc;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL printer_update_to_config(const rdpSettings* settings, const WCHAR* name, size_t length,
Packit Service b1ea74
                                     const BYTE* data, size_t datalen)
Packit Service b1ea74
{
Packit Service b1ea74
	BOOL rc = FALSE;
Packit Service b1ea74
	char* path = get_printer_config_path(settings, name, length);
Packit Service b1ea74
	rc = printer_write_setting(path, PRN_CONF_DATA, data, datalen);
Packit Service b1ea74
	free(path);
Packit Service b1ea74
	return rc;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL printer_remove_config(const rdpSettings* settings, const WCHAR* name, size_t length)
Packit Service b1ea74
{
Packit Service b1ea74
	BOOL rc = FALSE;
Packit Service b1ea74
	char* path = get_printer_config_path(settings, name, length);
Packit Service b1ea74
Packit Service b1ea74
	if (!printer_config_valid(path))
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	rc = RemoveDirectoryA(path);
Packit Service b1ea74
fail:
Packit Service b1ea74
	free(path);
Packit Service b1ea74
	return rc;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL printer_move_config(const rdpSettings* settings, const WCHAR* oldName, size_t oldLength,
Packit Service b1ea74
                                const WCHAR* newName, size_t newLength)
Packit Service b1ea74
{
Packit Service b1ea74
	BOOL rc = FALSE;
Packit Service b1ea74
	char* oldPath = get_printer_config_path(settings, oldName, oldLength);
Packit Service b1ea74
	char* newPath = get_printer_config_path(settings, newName, newLength);
Packit Service b1ea74
Packit Service b1ea74
	if (printer_config_valid(oldPath))
Packit Service b1ea74
		rc = MoveFileA(oldPath, newPath);
Packit Service b1ea74
Packit Service b1ea74
	free(oldPath);
Packit Service b1ea74
	free(newPath);
Packit Service b1ea74
	return rc;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL printer_load_from_config(const rdpSettings* settings, rdpPrinter* printer,
Packit Service b1ea74
                                     PRINTER_DEVICE* printer_dev)
Packit Service b1ea74
{
Packit Service b1ea74
	BOOL res = FALSE;
Packit Service b1ea74
	WCHAR* wname = NULL;
Packit Service b1ea74
	size_t wlen;
Packit Service b1ea74
	char* path = NULL;
Packit Service b1ea74
	int rc;
Packit Service b1ea74
	UINT32 flags = 0;
Packit Service b1ea74
	void* DriverName = NULL;
Packit Service b1ea74
	UINT32 DriverNameLen = 0;
Packit Service b1ea74
	void* PnPName = NULL;
Packit Service b1ea74
	UINT32 PnPNameLen = 0;
Packit Service b1ea74
	void* CachedPrinterConfigData = NULL;
Packit Service b1ea74
	UINT32 CachedFieldsLen = 0;
Packit Service b1ea74
	UINT32 PrinterNameLen = 0;
Packit Service b1ea74
Packit Service b1ea74
	if (!settings || !printer)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	rc = ConvertToUnicode(CP_UTF8, 0, printer->name, -1, &wname, 0);
Packit Service b1ea74
Packit Service b1ea74
	if (rc <= 0)
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	wlen = _wcslen(wname) + 1;
Packit Service b1ea74
	path = get_printer_config_path(settings, wname, wlen * sizeof(WCHAR));
Packit Service b1ea74
	PrinterNameLen = (wlen + 1) * sizeof(WCHAR);
Packit Service b1ea74
Packit Service b1ea74
	if (!path)
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	if (printer->is_default)
Packit Service b1ea74
		flags |= RDPDR_PRINTER_ANNOUNCE_FLAG_DEFAULTPRINTER;
Packit Service b1ea74
Packit Service b1ea74
	if (!printer_read_setting(path, PRN_CONF_PNP, &PnPName, &PnPNameLen))
Packit Service b1ea74
	{
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	if (!printer_read_setting(path, PRN_CONF_DRIVER, &DriverName, &DriverNameLen))
Packit Service b1ea74
	{
Packit Service b1ea74
		DriverNameLen =
Packit Service b1ea74
		    ConvertToUnicode(CP_UTF8, 0, printer->driver, -1, (LPWSTR*)&DriverName, 0) * 2 + 1;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	if (!printer_read_setting(path, PRN_CONF_DATA, &CachedPrinterConfigData, &CachedFieldsLen))
Packit Service b1ea74
	{
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	Stream_SetPosition(printer_dev->device.data, 0);
Packit Service b1ea74
Packit Service b1ea74
	if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, 24))
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	Stream_Write_UINT32(printer_dev->device.data, flags);
Packit Service b1ea74
	Stream_Write_UINT32(printer_dev->device.data, 0);          /* CodePage, reserved */
Packit Service b1ea74
	Stream_Write_UINT32(printer_dev->device.data, PnPNameLen); /* PnPNameLen */
Packit Service b1ea74
	Stream_Write_UINT32(printer_dev->device.data, DriverNameLen);
Packit Service b1ea74
	Stream_Write_UINT32(printer_dev->device.data, PrinterNameLen);
Packit Service b1ea74
	Stream_Write_UINT32(printer_dev->device.data, CachedFieldsLen);
Packit Service b1ea74
Packit Service b1ea74
	if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, PnPNameLen))
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	if (PnPNameLen > 0)
Packit Service b1ea74
		Stream_Write(printer_dev->device.data, PnPName, PnPNameLen);
Packit Service b1ea74
Packit Service b1ea74
	if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, DriverNameLen))
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	Stream_Write(printer_dev->device.data, DriverName, DriverNameLen);
Packit Service b1ea74
Packit Service b1ea74
	if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, PrinterNameLen))
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	Stream_Write(printer_dev->device.data, wname, PrinterNameLen);
Packit Service b1ea74
Packit Service b1ea74
	if (!Stream_EnsureRemainingCapacity(printer_dev->device.data, CachedFieldsLen))
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	Stream_Write(printer_dev->device.data, CachedPrinterConfigData, CachedFieldsLen);
Packit Service b1ea74
	res = TRUE;
Packit Service b1ea74
fail:
Packit Service b1ea74
	free(path);
Packit Service b1ea74
	free(wname);
Packit Service b1ea74
	free(PnPName);
Packit Service b1ea74
	free(DriverName);
Packit Service b1ea74
	free(CachedPrinterConfigData);
Packit Service b1ea74
	return res;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL printer_save_default_config(const rdpSettings* settings, rdpPrinter* printer)
Packit Service b1ea74
{
Packit Service b1ea74
	BOOL res = FALSE;
Packit Service b1ea74
	WCHAR* wname = NULL;
Packit Service b1ea74
	WCHAR* driver = NULL;
Packit Service b1ea74
	size_t wlen, dlen;
Packit Service b1ea74
	char* path = NULL;
Packit Service b1ea74
	int rc;
Packit Service b1ea74
Packit Service b1ea74
	if (!settings || !printer)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	rc = ConvertToUnicode(CP_UTF8, 0, printer->name, -1, &wname, 0);
Packit Service b1ea74
Packit Service b1ea74
	if (rc <= 0)
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	rc = ConvertToUnicode(CP_UTF8, 0, printer->driver, -1, &driver, 0);
Packit Service b1ea74
Packit Service b1ea74
	if (rc <= 0)
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	wlen = _wcslen(wname) + 1;
Packit Service b1ea74
	dlen = _wcslen(driver) + 1;
Packit Service b1ea74
	path = get_printer_config_path(settings, wname, wlen * sizeof(WCHAR));
Packit Service b1ea74
Packit Service b1ea74
	if (!path)
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	if (dlen > 1)
Packit Service b1ea74
	{
Packit Service b1ea74
		if (!printer_write_setting(path, PRN_CONF_DRIVER, driver, dlen * sizeof(WCHAR)))
Packit Service b1ea74
			goto fail;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	res = TRUE;
Packit Service b1ea74
fail:
Packit Service b1ea74
	free(path);
Packit Service b1ea74
	free(wname);
Packit Service b1ea74
	free(driver);
Packit Service b1ea74
	return res;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT printer_process_irp_create(PRINTER_DEVICE* printer_dev, IRP* irp)
Packit Service fa4841
{
Packit Service fa4841
	rdpPrintJob* printjob = NULL;
Packit Service fa4841
Packit Service fa4841
	if (printer_dev->printer)
Packit Service b1ea74
		printjob =
Packit Service b1ea74
		    printer_dev->printer->CreatePrintJob(printer_dev->printer, irp->devman->id_sequence++);
Packit Service fa4841
Packit Service fa4841
	if (printjob)
Packit Service fa4841
	{
Packit Service fa4841
		Stream_Write_UINT32(irp->output, printjob->id); /* FileId */
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		Stream_Write_UINT32(irp->output, 0); /* FileId */
Packit Service fa4841
		irp->IoStatus = STATUS_PRINT_QUEUE_FULL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return irp->Complete(irp);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT printer_process_irp_close(PRINTER_DEVICE* printer_dev, IRP* irp)
Packit Service fa4841
{
Packit Service fa4841
	rdpPrintJob* printjob = NULL;
Packit Service fa4841
Packit Service fa4841
	if (printer_dev->printer)
Packit Service b1ea74
		printjob = printer_dev->printer->FindPrintJob(printer_dev->printer, irp->FileId);
Packit Service fa4841
Packit Service fa4841
	if (!printjob)
Packit Service fa4841
	{
Packit Service fa4841
		irp->IoStatus = STATUS_UNSUCCESSFUL;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		printjob->Close(printjob);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	Stream_Zero(irp->output, 4); /* Padding(4) */
Packit Service fa4841
	return irp->Complete(irp);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT printer_process_irp_write(PRINTER_DEVICE* printer_dev, IRP* irp)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 Length;
Packit Service fa4841
	UINT64 Offset;
Packit Service fa4841
	rdpPrintJob* printjob = NULL;
Packit Service fa4841
	UINT error = CHANNEL_RC_OK;
Packit Service b1ea74
	void* ptr;
Packit Service b1ea74
Packit Service b1ea74
	if (Stream_GetRemainingLength(irp->input) < 32)
Packit Service b1ea74
		return ERROR_INVALID_DATA;
Packit Service fa4841
	Stream_Read_UINT32(irp->input, Length);
Packit Service fa4841
	Stream_Read_UINT64(irp->input, Offset);
Packit Service fa4841
	Stream_Seek(irp->input, 20); /* Padding */
Packit Service b1ea74
	ptr = Stream_Pointer(irp->input);
Packit Service b1ea74
	if (!Stream_SafeSeek(irp->input, Length))
Packit Service b1ea74
		return ERROR_INVALID_DATA;
Packit Service fa4841
	if (printer_dev->printer)
Packit Service b1ea74
		printjob = printer_dev->printer->FindPrintJob(printer_dev->printer, irp->FileId);
Packit Service fa4841
Packit Service fa4841
	if (!printjob)
Packit Service fa4841
	{
Packit Service fa4841
		irp->IoStatus = STATUS_UNSUCCESSFUL;
Packit Service fa4841
		Length = 0;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service b1ea74
		error = printjob->Write(printjob, ptr, Length);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (error)
Packit Service fa4841
	{
Packit Service b1ea74
		WLog_ERR(TAG, "printjob->Write failed with error %" PRIu32 "!", error);
Packit Service fa4841
		return error;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	Stream_Write_UINT32(irp->output, Length);
Packit Service fa4841
	Stream_Write_UINT8(irp->output, 0); /* Padding */
Packit Service fa4841
	return irp->Complete(irp);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service b1ea74
static UINT printer_process_irp_device_control(PRINTER_DEVICE* printer_dev, IRP* irp)
Packit Service fa4841
{
Packit Service fa4841
	Stream_Write_UINT32(irp->output, 0); /* OutputBufferLength */
Packit Service fa4841
	return irp->Complete(irp);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT printer_process_irp(PRINTER_DEVICE* printer_dev, IRP* irp)
Packit Service fa4841
{
Packit Service fa4841
	UINT error;
Packit Service fa4841
Packit Service fa4841
	switch (irp->MajorFunction)
Packit Service fa4841
	{
Packit Service fa4841
		case IRP_MJ_CREATE:
Packit Service fa4841
			if ((error = printer_process_irp_create(printer_dev, irp)))
Packit Service fa4841
			{
Packit Service b1ea74
				WLog_ERR(TAG, "printer_process_irp_create failed with error %" PRIu32 "!", error);
Packit Service fa4841
				return error;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case IRP_MJ_CLOSE:
Packit Service fa4841
			if ((error = printer_process_irp_close(printer_dev, irp)))
Packit Service fa4841
			{
Packit Service b1ea74
				WLog_ERR(TAG, "printer_process_irp_close failed with error %" PRIu32 "!", error);
Packit Service fa4841
				return error;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case IRP_MJ_WRITE:
Packit Service fa4841
			if ((error = printer_process_irp_write(printer_dev, irp)))
Packit Service fa4841
			{
Packit Service b1ea74
				WLog_ERR(TAG, "printer_process_irp_write failed with error %" PRIu32 "!", error);
Packit Service fa4841
				return error;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case IRP_MJ_DEVICE_CONTROL:
Packit Service fa4841
			if ((error = printer_process_irp_device_control(printer_dev, irp)))
Packit Service fa4841
			{
Packit Service b1ea74
				WLog_ERR(TAG, "printer_process_irp_device_control failed with error %" PRIu32 "!",
Packit Service fa4841
				         error);
Packit Service fa4841
				return error;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		default:
Packit Service fa4841
			irp->IoStatus = STATUS_NOT_SUPPORTED;
Packit Service fa4841
			return irp->Complete(irp);
Packit Service fa4841
			break;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return CHANNEL_RC_OK;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static DWORD WINAPI printer_thread_func(LPVOID arg)
Packit Service fa4841
{
Packit Service fa4841
	IRP* irp;
Packit Service b1ea74
	PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)arg;
Packit Service b1ea74
	HANDLE obj[] = { printer_dev->event, printer_dev->stopEvent };
Packit Service fa4841
	UINT error = CHANNEL_RC_OK;
Packit Service fa4841
Packit Service fa4841
	while (1)
Packit Service fa4841
	{
Packit Service fa4841
		DWORD rc = WaitForMultipleObjects(2, obj, FALSE, INFINITE);
Packit Service fa4841
Packit Service fa4841
		if (rc == WAIT_FAILED)
Packit Service fa4841
		{
Packit Service fa4841
			error = GetLastError();
Packit Service b1ea74
			WLog_ERR(TAG, "WaitForMultipleObjects failed with error %" PRIu32 "!", error);
Packit Service fa4841
			break;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		if (rc == WAIT_OBJECT_0 + 1)
Packit Service fa4841
			break;
Packit Service fa4841
		else if (rc != WAIT_OBJECT_0)
Packit Service fa4841
			continue;
Packit Service fa4841
Packit Service fa4841
		ResetEvent(printer_dev->event);
Packit Service b1ea74
		irp = (IRP*)InterlockedPopEntrySList(printer_dev->pIrpList);
Packit Service fa4841
Packit Service fa4841
		if (irp == NULL)
Packit Service fa4841
		{
Packit Service fa4841
			WLog_ERR(TAG, "InterlockedPopEntrySList failed!");
Packit Service fa4841
			error = ERROR_INTERNAL_ERROR;
Packit Service fa4841
			break;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		if ((error = printer_process_irp(printer_dev, irp)))
Packit Service fa4841
		{
Packit Service b1ea74
			WLog_ERR(TAG, "printer_process_irp failed with error %" PRIu32 "!", error);
Packit Service fa4841
			break;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (error && printer_dev->rdpcontext)
Packit Service b1ea74
		setChannelError(printer_dev->rdpcontext, error, "printer_thread_func reported an error");
Packit Service fa4841
Packit Service fa4841
	ExitThread(error);
Packit Service fa4841
	return error;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT printer_irp_request(DEVICE* device, IRP* irp)
Packit Service fa4841
{
Packit Service b1ea74
	PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
Packit Service fa4841
	InterlockedPushEntrySList(printer_dev->pIrpList, &(irp->ItemEntry));
Packit Service fa4841
	SetEvent(printer_dev->event);
Packit Service fa4841
	return CHANNEL_RC_OK;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static UINT printer_custom_component(DEVICE* device, UINT16 component, UINT16 packetId, wStream* s)
Packit Service b1ea74
{
Packit Service b1ea74
	UINT32 eventID;
Packit Service b1ea74
	PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
Packit Service b1ea74
	const rdpSettings* settings = printer_dev->rdpcontext->settings;
Packit Service b1ea74
Packit Service b1ea74
	if (component != RDPDR_CTYP_PRN)
Packit Service b1ea74
		return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
	if (Stream_GetRemainingLength(s) < 4)
Packit Service b1ea74
		return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
	Stream_Read_UINT32(s, eventID);
Packit Service b1ea74
Packit Service b1ea74
	switch (packetId)
Packit Service b1ea74
	{
Packit Service b1ea74
		case PAKID_PRN_CACHE_DATA:
Packit Service b1ea74
			switch (eventID)
Packit Service b1ea74
			{
Packit Service b1ea74
				case RDPDR_ADD_PRINTER_EVENT:
Packit Service b1ea74
				{
Packit Service b1ea74
					char PortDosName[8];
Packit Service b1ea74
					UINT32 PnPNameLen, DriverNameLen, PrintNameLen, CacheFieldsLen;
Packit Service b1ea74
					const WCHAR *PnPName, *DriverName, *PrinterName;
Packit Service b1ea74
					const BYTE* CachedPrinterConfigData;
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < 24)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					Stream_Read(s, PortDosName, sizeof(PortDosName));
Packit Service b1ea74
					Stream_Read_UINT32(s, PnPNameLen);
Packit Service b1ea74
					Stream_Read_UINT32(s, DriverNameLen);
Packit Service b1ea74
					Stream_Read_UINT32(s, PrintNameLen);
Packit Service b1ea74
					Stream_Read_UINT32(s, CacheFieldsLen);
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < PnPNameLen)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					PnPName = (const WCHAR*)Stream_Pointer(s);
Packit Service b1ea74
					Stream_Seek(s, PnPNameLen);
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < DriverNameLen)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					DriverName = (const WCHAR*)Stream_Pointer(s);
Packit Service b1ea74
					Stream_Seek(s, DriverNameLen);
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < PrintNameLen)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					PrinterName = (const WCHAR*)Stream_Pointer(s);
Packit Service b1ea74
					Stream_Seek(s, PrintNameLen);
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < CacheFieldsLen)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					CachedPrinterConfigData = Stream_Pointer(s);
Packit Service b1ea74
					Stream_Seek(s, CacheFieldsLen);
Packit Service b1ea74
Packit Service b1ea74
					if (!printer_save_to_config(settings, PortDosName, sizeof(PortDosName), PnPName,
Packit Service b1ea74
					                            PnPNameLen, DriverName, DriverNameLen, PrinterName,
Packit Service b1ea74
					                            PrintNameLen, CachedPrinterConfigData,
Packit Service b1ea74
					                            CacheFieldsLen))
Packit Service b1ea74
						return ERROR_INTERNAL_ERROR;
Packit Service b1ea74
				}
Packit Service b1ea74
				break;
Packit Service b1ea74
Packit Service b1ea74
				case RDPDR_UPDATE_PRINTER_EVENT:
Packit Service b1ea74
				{
Packit Service b1ea74
					UINT32 PrinterNameLen, ConfigDataLen;
Packit Service b1ea74
					const WCHAR* PrinterName;
Packit Service b1ea74
					const BYTE* ConfigData;
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < 8)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					Stream_Read_UINT32(s, PrinterNameLen);
Packit Service b1ea74
					Stream_Read_UINT32(s, ConfigDataLen);
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < PrinterNameLen)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					PrinterName = (const WCHAR*)Stream_Pointer(s);
Packit Service b1ea74
					Stream_Seek(s, PrinterNameLen);
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < ConfigDataLen)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					ConfigData = Stream_Pointer(s);
Packit Service b1ea74
					Stream_Seek(s, ConfigDataLen);
Packit Service b1ea74
Packit Service b1ea74
					if (!printer_update_to_config(settings, PrinterName, PrinterNameLen, ConfigData,
Packit Service b1ea74
					                              ConfigDataLen))
Packit Service b1ea74
						return ERROR_INTERNAL_ERROR;
Packit Service b1ea74
				}
Packit Service b1ea74
				break;
Packit Service b1ea74
Packit Service b1ea74
				case RDPDR_DELETE_PRINTER_EVENT:
Packit Service b1ea74
				{
Packit Service b1ea74
					UINT32 PrinterNameLen;
Packit Service b1ea74
					const WCHAR* PrinterName;
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < 4)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					Stream_Read_UINT32(s, PrinterNameLen);
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < PrinterNameLen)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					PrinterName = (const WCHAR*)Stream_Pointer(s);
Packit Service b1ea74
					Stream_Seek(s, PrinterNameLen);
Packit Service b1ea74
					printer_remove_config(settings, PrinterName, PrinterNameLen);
Packit Service b1ea74
				}
Packit Service b1ea74
				break;
Packit Service b1ea74
Packit Service b1ea74
				case RDPDR_RENAME_PRINTER_EVENT:
Packit Service b1ea74
				{
Packit Service b1ea74
					UINT32 OldPrinterNameLen, NewPrinterNameLen;
Packit Service b1ea74
					const WCHAR* OldPrinterName;
Packit Service b1ea74
					const WCHAR* NewPrinterName;
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < 8)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					Stream_Read_UINT32(s, OldPrinterNameLen);
Packit Service b1ea74
					Stream_Read_UINT32(s, NewPrinterNameLen);
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < OldPrinterNameLen)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					OldPrinterName = (const WCHAR*)Stream_Pointer(s);
Packit Service b1ea74
					Stream_Seek(s, OldPrinterNameLen);
Packit Service b1ea74
Packit Service b1ea74
					if (Stream_GetRemainingLength(s) < NewPrinterNameLen)
Packit Service b1ea74
						return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
					NewPrinterName = (const WCHAR*)Stream_Pointer(s);
Packit Service b1ea74
					Stream_Seek(s, NewPrinterNameLen);
Packit Service b1ea74
Packit Service b1ea74
					if (!printer_move_config(settings, OldPrinterName, OldPrinterNameLen,
Packit Service b1ea74
					                         NewPrinterName, NewPrinterNameLen))
Packit Service b1ea74
						return ERROR_INTERNAL_ERROR;
Packit Service b1ea74
				}
Packit Service b1ea74
				break;
Packit Service b1ea74
Packit Service b1ea74
				default:
Packit Service b1ea74
					WLog_ERR(TAG, "Unknown cache data eventID: 0x%08" PRIX32 "", eventID);
Packit Service b1ea74
					return ERROR_INVALID_DATA;
Packit Service b1ea74
			}
Packit Service b1ea74
Packit Service b1ea74
			break;
Packit Service b1ea74
Packit Service b1ea74
		case PAKID_PRN_USING_XPS:
Packit Service b1ea74
		{
Packit Service b1ea74
			UINT32 flags;
Packit Service b1ea74
Packit Service b1ea74
			if (Stream_GetRemainingLength(s) < 4)
Packit Service b1ea74
				return ERROR_INVALID_DATA;
Packit Service b1ea74
Packit Service b1ea74
			Stream_Read_UINT32(s, flags);
Packit Service b1ea74
			WLog_ERR(TAG,
Packit Service b1ea74
			         "Ignoring unhandled message PAKID_PRN_USING_XPS [printerID=%08" PRIx32
Packit Service b1ea74
			         ", flags=%08" PRIx32 "]",
Packit Service b1ea74
			         eventID, flags);
Packit Service b1ea74
		}
Packit Service b1ea74
		break;
Packit Service b1ea74
Packit Service b1ea74
		default:
Packit Service b1ea74
			WLog_ERR(TAG, "Unknown printing component packetID: 0x%04" PRIX16 "", packetId);
Packit Service b1ea74
			return ERROR_INVALID_DATA;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	return CHANNEL_RC_OK;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service fa4841
static UINT printer_free(DEVICE* device)
Packit Service fa4841
{
Packit Service fa4841
	IRP* irp;
Packit Service b1ea74
	PRINTER_DEVICE* printer_dev = (PRINTER_DEVICE*)device;
Packit Service fa4841
	UINT error;
Packit Service fa4841
	SetEvent(printer_dev->stopEvent);
Packit Service fa4841
Packit Service fa4841
	if (WaitForSingleObject(printer_dev->thread, INFINITE) == WAIT_FAILED)
Packit Service fa4841
	{
Packit Service fa4841
		error = GetLastError();
Packit Service b1ea74
		WLog_ERR(TAG, "WaitForSingleObject failed with error %" PRIu32 "", error);
Packit Service b1ea74
Packit Service b1ea74
		/* The analyzer is confused by this premature return value.
Packit Service b1ea74
		 * Since this case can not be handled gracefully silence the
Packit Service b1ea74
		 * analyzer here. */
Packit Service b1ea74
#ifndef __clang_analyzer__
Packit Service fa4841
		return error;
Packit Service b1ea74
#endif
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	while ((irp = (IRP*)InterlockedPopEntrySList(printer_dev->pIrpList)) != NULL)
Packit Service fa4841
		irp->Discard(irp);
Packit Service fa4841
Packit Service fa4841
	CloseHandle(printer_dev->thread);
Packit Service fa4841
	CloseHandle(printer_dev->stopEvent);
Packit Service fa4841
	CloseHandle(printer_dev->event);
Packit Service fa4841
	_aligned_free(printer_dev->pIrpList);
Packit Service fa4841
Packit Service fa4841
	if (printer_dev->printer)
Packit Service b1ea74
		printer_dev->printer->ReleaseRef(printer_dev->printer);
Packit Service fa4841
Packit Service fa4841
	Stream_Free(printer_dev->device.data, TRUE);
Packit Service fa4841
	free(printer_dev);
Packit Service fa4841
	return CHANNEL_RC_OK;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service b1ea74
static UINT printer_register(PDEVICE_SERVICE_ENTRY_POINTS pEntryPoints, rdpPrinter* printer)
Packit Service fa4841
{
Packit Service fa4841
	PRINTER_DEVICE* printer_dev;
Packit Service b1ea74
	UINT error = ERROR_INTERNAL_ERROR;
Packit Service b1ea74
	printer_dev = (PRINTER_DEVICE*)calloc(1, sizeof(PRINTER_DEVICE));
Packit Service fa4841
Packit Service fa4841
	if (!printer_dev)
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "calloc failed!");
Packit Service fa4841
		return CHANNEL_RC_NO_MEMORY;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	printer_dev->device.data = Stream_New(NULL, 1024);
Packit Service b1ea74
Packit Service b1ea74
	if (!printer_dev->device.data)
Packit Service b1ea74
		goto error_out;
Packit Service b1ea74
Packit Service b1ea74
	sprintf_s(printer_dev->port, sizeof(printer_dev->port), "PRN%" PRIdz, printer->id);
Packit Service fa4841
	printer_dev->device.type = RDPDR_DTYP_PRINT;
Packit Service fa4841
	printer_dev->device.name = printer_dev->port;
Packit Service fa4841
	printer_dev->device.IRPRequest = printer_irp_request;
Packit Service b1ea74
	printer_dev->device.CustomComponentRequest = printer_custom_component;
Packit Service fa4841
	printer_dev->device.Free = printer_free;
Packit Service fa4841
	printer_dev->rdpcontext = pEntryPoints->rdpcontext;
Packit Service fa4841
	printer_dev->printer = printer;
Packit Service b1ea74
	printer_dev->pIrpList = (WINPR_PSLIST_HEADER)_aligned_malloc(sizeof(WINPR_SLIST_HEADER),
Packit Service b1ea74
	                                                             MEMORY_ALLOCATION_ALIGNMENT);
Packit Service bb5c11
Packit Service bb5c11
	if (!printer_dev->pIrpList)
Packit Service bb5c11
	{
Packit Service bb5c11
		WLog_ERR(TAG, "_aligned_malloc failed!");
Packit Service bb5c11
		error = CHANNEL_RC_NO_MEMORY;
Packit Service fa4841
		goto error_out;
Packit Service bb5c11
	}
Packit Service fa4841
Packit Service b1ea74
	if (!printer_load_from_config(pEntryPoints->rdpcontext->settings, printer, printer_dev))
Packit Service b1ea74
		goto error_out;
Packit Service b1ea74
Packit Service fa4841
	InitializeSListHead(printer_dev->pIrpList);
Packit Service fa4841
Packit Service fa4841
	if (!(printer_dev->event = CreateEvent(NULL, TRUE, FALSE, NULL)))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "CreateEvent failed!");
Packit Service fa4841
		error = ERROR_INTERNAL_ERROR;
Packit Service fa4841
		goto error_out;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (!(printer_dev->stopEvent = CreateEvent(NULL, TRUE, FALSE, NULL)))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "CreateEvent failed!");
Packit Service fa4841
		error = ERROR_INTERNAL_ERROR;
Packit Service fa4841
		goto error_out;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	if ((error = pEntryPoints->RegisterDevice(pEntryPoints->devman, (DEVICE*)printer_dev)))
Packit Service fa4841
	{
Packit Service b1ea74
		WLog_ERR(TAG, "RegisterDevice failed with error %" PRIu32 "!", error);
Packit Service fa4841
		goto error_out;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	if (!(printer_dev->thread =
Packit Service b1ea74
	          CreateThread(NULL, 0, printer_thread_func, (void*)printer_dev, 0, NULL)))
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "CreateThread failed!");
Packit Service fa4841
		error = ERROR_INTERNAL_ERROR;
Packit Service fa4841
		goto error_out;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	printer->AddRef(printer);
Packit Service fa4841
	return CHANNEL_RC_OK;
Packit Service fa4841
error_out:
Packit Service fa4841
	printer_free(&printer_dev->device);
Packit Service fa4841
	return error;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static rdpPrinterDriver* printer_load_backend(const char* backend)
Packit Service b1ea74
{
Packit Service b1ea74
	typedef rdpPrinterDriver* (*backend_load_t)(void);
Packit Service b1ea74
	union {
Packit Service b1ea74
		PVIRTUALCHANNELENTRY entry;
Packit Service b1ea74
		backend_load_t backend;
Packit Service b1ea74
	} fktconv;
Packit Service b1ea74
Packit Service b1ea74
	fktconv.entry = freerdp_load_channel_addin_entry("printer", backend, NULL, 0);
Packit Service b1ea74
	if (!fktconv.entry)
Packit Service b1ea74
		return NULL;
Packit Service b1ea74
Packit Service b1ea74
	return fktconv.backend();
Packit Service b1ea74
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Function description
Packit Service fa4841
 *
Packit Service fa4841
 * @return 0 on success, otherwise a Win32 error code
Packit Service fa4841
 */
Packit Service b1ea74
UINT
Packit Service b1ea74
#ifdef BUILTIN_CHANNELS
Packit Service b1ea74
printer_DeviceServiceEntry
Packit Service b1ea74
#else
Packit Service b1ea74
    FREERDP_API
Packit Service b1ea74
    DeviceServiceEntry
Packit Service b1ea74
#endif
Packit Service b1ea74
    (PDEVICE_SERVICE_ENTRY_POINTS pEntryPoints)
Packit Service fa4841
{
Packit Service fa4841
	int i;
Packit Service fa4841
	char* name;
Packit Service fa4841
	char* driver_name;
Packit Service b1ea74
	BOOL default_backend = TRUE;
Packit Service b1ea74
	RDPDR_PRINTER* device = NULL;
Packit Service fa4841
	rdpPrinterDriver* driver = NULL;
Packit Service b1ea74
	UINT error = CHANNEL_RC_OK;
Packit Service b1ea74
Packit Service b1ea74
	if (!pEntryPoints || !pEntryPoints->device)
Packit Service b1ea74
		return ERROR_INVALID_PARAMETER;
Packit Service b1ea74
Packit Service b1ea74
	device = (RDPDR_PRINTER*)pEntryPoints->device;
Packit Service b1ea74
	name = device->Name;
Packit Service b1ea74
	driver_name = device->DriverName;
Packit Service b1ea74
Packit Service b1ea74
	/* Secondary argument is one of the following:
Packit Service b1ea74
	 *
Packit Service b1ea74
	 * <driver_name>                ... name of a printer driver
Packit Service b1ea74
	 * <driver_name>:<backend_name> ... name of a printer driver and local printer backend to use
Packit Service b1ea74
	 */
Packit Service b1ea74
	if (driver_name)
Packit Service b1ea74
	{
Packit Service b1ea74
		char* sep = strstr(driver_name, ":");
Packit Service b1ea74
		if (sep)
Packit Service b1ea74
		{
Packit Service b1ea74
			const char* backend = sep + 1;
Packit Service b1ea74
			*sep = '\0';
Packit Service b1ea74
			driver = printer_load_backend(backend);
Packit Service b1ea74
			default_backend = FALSE;
Packit Service b1ea74
		}
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	if (!driver && default_backend)
Packit Service b1ea74
	{
Packit Service b1ea74
		const char* backend =
Packit Service b1ea74
#if defined(WITH_CUPS)
Packit Service b1ea74
		    "cups"
Packit Service b1ea74
#elif defined(_WIN32)
Packit Service b1ea74
		    "win"
Packit Service b1ea74
#else
Packit Service b1ea74
		    ""
Packit Service fa4841
#endif
Packit Service b1ea74
		    ;
Packit Service b1ea74
Packit Service b1ea74
		driver = printer_load_backend(backend);
Packit Service b1ea74
	}
Packit Service fa4841
Packit Service fa4841
	if (!driver)
Packit Service fa4841
	{
Packit Service fa4841
		WLog_ERR(TAG, "Could not get a printer driver!");
Packit Service fa4841
		return CHANNEL_RC_INITIALIZATION_ERROR;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (name && name[0])
Packit Service fa4841
	{
Packit Service b1ea74
		rdpPrinter* printer = driver->GetPrinter(driver, name, driver_name);
Packit Service fa4841
Packit Service fa4841
		if (!printer)
Packit Service fa4841
		{
Packit Service fa4841
			WLog_ERR(TAG, "Could not get printer %s!", name);
Packit Service b1ea74
			error = CHANNEL_RC_INITIALIZATION_ERROR;
Packit Service b1ea74
			goto fail;
Packit Service b1ea74
		}
Packit Service b1ea74
Packit Service b1ea74
		if (!printer_save_default_config(pEntryPoints->rdpcontext->settings, printer))
Packit Service b1ea74
		{
Packit Service b1ea74
			error = CHANNEL_RC_INITIALIZATION_ERROR;
Packit Service b1ea74
			printer->ReleaseRef(printer);
Packit Service b1ea74
			goto fail;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		if ((error = printer_register(pEntryPoints, printer)))
Packit Service fa4841
		{
Packit Service b1ea74
			WLog_ERR(TAG, "printer_register failed with error %" PRIu32 "!", error);
Packit Service b1ea74
			printer->ReleaseRef(printer);
Packit Service b1ea74
			goto fail;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service b1ea74
		rdpPrinter** printers = driver->EnumPrinters(driver);
Packit Service b1ea74
		rdpPrinter** current = printers;
Packit Service fa4841
Packit Service b1ea74
		for (i = 0; current[i]; i++)
Packit Service fa4841
		{
Packit Service b1ea74
			rdpPrinter* printer = current[i];
Packit Service fa4841
Packit Service fa4841
			if ((error = printer_register(pEntryPoints, printer)))
Packit Service fa4841
			{
Packit Service b1ea74
				WLog_ERR(TAG, "printer_register failed with error %" PRIu32 "!", error);
Packit Service b1ea74
				break;
Packit Service fa4841
			}
Packit Service fa4841
		}
Packit Service fa4841
Packit Service b1ea74
		driver->ReleaseEnumPrinters(printers);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
fail:
Packit Service b1ea74
	driver->ReleaseRef(driver);
Packit Service b1ea74
Packit Service b1ea74
	return error;
Packit Service fa4841
}