Blame channels/rdpdr/client/devman.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 * Device Redirection Virtual Channel
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2010-2011 Vic Lee
Packit 1fb8d4
 * Copyright 2010-2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit 1fb8d4
 * Copyright 2015 Thincast Technologies GmbH
Packit 1fb8d4
 * Copyright 2015 DI (FH) Martin Haimberger <martin.haimberger@thincast.com>
Packit 1fb8d4
 * Copyright 2016 Armin Novak <armin.novak@gmail.com>
Packit 1fb8d4
 *
Packit 1fb8d4
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit 1fb8d4
 * you may not use this file except in compliance with the License.
Packit 1fb8d4
 * You may obtain a copy of the License at
Packit 1fb8d4
 *
Packit 1fb8d4
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit 1fb8d4
 *
Packit 1fb8d4
 * Unless required by applicable law or agreed to in writing, software
Packit 1fb8d4
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 1fb8d4
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 1fb8d4
 * See the License for the specific language governing permissions and
Packit 1fb8d4
 * limitations under the License.
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_CONFIG_H
Packit 1fb8d4
#include "config.h"
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include <stdio.h>
Packit 1fb8d4
#include <stdlib.h>
Packit 1fb8d4
#include <string.h>
Packit 1fb8d4
Packit 1fb8d4
#include <winpr/crt.h>
Packit 1fb8d4
#include <winpr/stream.h>
Packit 1fb8d4
Packit 1fb8d4
#include <freerdp/types.h>
Packit 1fb8d4
#include <freerdp/addin.h>
Packit 1fb8d4
#include <freerdp/client/channels.h>
Packit 1fb8d4
#include <freerdp/channels/log.h>
Packit 1fb8d4
Packit 1fb8d4
#include "rdpdr_main.h"
Packit 1fb8d4
Packit 1fb8d4
#include "devman.h"
Packit 1fb8d4
Packit 1fb8d4
static void devman_device_free(void* obj)
Packit 1fb8d4
{
Packit Service 5a9772
	DEVICE* device = (DEVICE*)obj;
Packit 1fb8d4
Packit 1fb8d4
	if (!device)
Packit 1fb8d4
		return;
Packit 1fb8d4
Packit 1fb8d4
	IFCALL(device->Free, device);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
DEVMAN* devman_new(rdpdrPlugin* rdpdr)
Packit 1fb8d4
{
Packit 1fb8d4
	DEVMAN* devman;
Packit 1fb8d4
Packit 1fb8d4
	if (!rdpdr)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit Service 5a9772
	devman = (DEVMAN*)calloc(1, sizeof(DEVMAN));
Packit 1fb8d4
Packit 1fb8d4
	if (!devman)
Packit 1fb8d4
	{
Packit Service 5a9772
		WLog_INFO(TAG, "calloc failed!");
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit Service 5a9772
	devman->plugin = (void*)rdpdr;
Packit 1fb8d4
	devman->id_sequence = 1;
Packit 1fb8d4
	devman->devices = ListDictionary_New(TRUE);
Packit Service 5a9772
Packit 1fb8d4
	if (!devman->devices)
Packit 1fb8d4
	{
Packit Service 5a9772
		WLog_INFO(TAG, "ListDictionary_New failed!");
Packit 1fb8d4
		free(devman);
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	ListDictionary_ValueObject(devman->devices)->fnObjectFree = devman_device_free;
Packit 1fb8d4
	return devman;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void devman_free(DEVMAN* devman)
Packit 1fb8d4
{
Packit 1fb8d4
	ListDictionary_Free(devman->devices);
Packit 1fb8d4
	free(devman);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void devman_unregister_device(DEVMAN* devman, void* key)
Packit 1fb8d4
{
Packit 1fb8d4
	DEVICE* device;
Packit 1fb8d4
Packit 1fb8d4
	if (!devman || !key)
Packit 1fb8d4
		return;
Packit 1fb8d4
Packit Service 5a9772
	device = (DEVICE*)ListDictionary_Remove(devman->devices, key);
Packit 1fb8d4
Packit 1fb8d4
	if (device)
Packit 1fb8d4
		devman_device_free(device);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * Function description
Packit 1fb8d4
 *
Packit 1fb8d4
 * @return 0 on success, otherwise a Win32 error code
Packit 1fb8d4
 */
Packit 1fb8d4
static UINT devman_register_device(DEVMAN* devman, DEVICE* device)
Packit 1fb8d4
{
Packit 1fb8d4
	void* key = NULL;
Packit 1fb8d4
Packit 1fb8d4
	if (!devman || !device)
Packit 1fb8d4
		return ERROR_INVALID_PARAMETER;
Packit 1fb8d4
Packit 1fb8d4
	device->id = devman->id_sequence++;
Packit Service 5a9772
	key = (void*)(size_t)device->id;
Packit 1fb8d4
Packit 1fb8d4
	if (!ListDictionary_Add(devman->devices, key, device))
Packit 1fb8d4
	{
Packit Service 5a9772
		WLog_INFO(TAG, "ListDictionary_Add failed!");
Packit 1fb8d4
		return ERROR_INTERNAL_ERROR;
Packit 1fb8d4
	}
Packit Service 5a9772
Packit 1fb8d4
	return CHANNEL_RC_OK;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
DEVICE* devman_get_device_by_id(DEVMAN* devman, UINT32 id)
Packit 1fb8d4
{
Packit 1fb8d4
	DEVICE* device = NULL;
Packit Service 5a9772
	void* key = (void*)(size_t)id;
Packit 1fb8d4
Packit 1fb8d4
	if (!devman)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit Service 5a9772
	device = (DEVICE*)ListDictionary_GetItemValue(devman->devices, key);
Packit Service 5a9772
	return device;
Packit Service 5a9772
}
Packit 1fb8d4
Packit Service 5a9772
DEVICE* devman_get_device_by_type(DEVMAN* devman, UINT32 type)
Packit Service 5a9772
{
Packit Service 5a9772
	DEVICE* device = NULL;
Packit Service 5a9772
	ULONG_PTR* keys;
Packit Service 5a9772
	int count, x;
Packit Service 5a9772
Packit Service 5a9772
	if (!devman)
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
Packit Service 5a9772
	ListDictionary_Lock(devman->devices);
Packit Service 5a9772
	count = ListDictionary_GetKeys(devman->devices, &keys);
Packit Service 5a9772
Packit Service 5a9772
	for (x = 0; x < count; x++)
Packit Service 5a9772
	{
Packit Service 5a9772
		DEVICE* cur = (DEVICE*)ListDictionary_GetItemValue(devman->devices, (void*)keys[x]);
Packit Service 5a9772
Packit Service 5a9772
		if (!cur)
Packit Service 5a9772
			continue;
Packit Service 5a9772
Packit Service 5a9772
		if (cur->type != type)
Packit Service 5a9772
			continue;
Packit Service 5a9772
Packit Service 5a9772
		device = cur;
Packit Service 5a9772
		break;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	free(keys);
Packit Service 5a9772
	ListDictionary_Unlock(devman->devices);
Packit 1fb8d4
	return device;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static const char DRIVE_SERVICE_NAME[] = "drive";
Packit Service 5a9772
static const char PRINTER_SERVICE_NAME[] = "printer";
Packit Service 5a9772
static const char SMARTCARD_SERVICE_NAME[] = "smartcard";
Packit Service 5a9772
static const char SERIAL_SERVICE_NAME[] = "serial";
Packit Service 5a9772
static const char PARALLEL_SERVICE_NAME[] = "parallel";
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * Function description
Packit 1fb8d4
 *
Packit 1fb8d4
 * @return 0 on success, otherwise a Win32 error code
Packit 1fb8d4
 */
Packit Service 5a9772
UINT devman_load_device_service(DEVMAN* devman, const RDPDR_DEVICE* device, rdpContext* rdpcontext)
Packit 1fb8d4
{
Packit Service 5a9772
	const char* ServiceName = NULL;
Packit 1fb8d4
	DEVICE_SERVICE_ENTRY_POINTS ep;
Packit 1fb8d4
	PDEVICE_SERVICE_ENTRY entry = NULL;
Packit Service 5a9772
	union {
Packit Service 5a9772
		const RDPDR_DEVICE* cdp;
Packit Service 5a9772
		RDPDR_DEVICE* dp;
Packit Service 5a9772
	} devconv;
Packit 1fb8d4
Packit Service 5a9772
	devconv.cdp = device;
Packit 1fb8d4
	if (!devman || !device || !rdpcontext)
Packit 1fb8d4
		return ERROR_INVALID_PARAMETER;
Packit 1fb8d4
Packit 1fb8d4
	if (device->Type == RDPDR_DTYP_FILESYSTEM)
Packit 1fb8d4
		ServiceName = DRIVE_SERVICE_NAME;
Packit 1fb8d4
	else if (device->Type == RDPDR_DTYP_PRINT)
Packit 1fb8d4
		ServiceName = PRINTER_SERVICE_NAME;
Packit 1fb8d4
	else if (device->Type == RDPDR_DTYP_SMARTCARD)
Packit 1fb8d4
		ServiceName = SMARTCARD_SERVICE_NAME;
Packit 1fb8d4
	else if (device->Type == RDPDR_DTYP_SERIAL)
Packit 1fb8d4
		ServiceName = SERIAL_SERVICE_NAME;
Packit 1fb8d4
	else if (device->Type == RDPDR_DTYP_PARALLEL)
Packit 1fb8d4
		ServiceName = PARALLEL_SERVICE_NAME;
Packit 1fb8d4
Packit 1fb8d4
	if (!ServiceName)
Packit 1fb8d4
	{
Packit Service 5a9772
		WLog_INFO(TAG, "ServiceName %s did not match!", ServiceName);
Packit 1fb8d4
		return ERROR_INVALID_NAME;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (device->Name)
Packit Service 5a9772
		WLog_INFO(TAG, "Loading device service %s [%s] (static)", ServiceName, device->Name);
Packit 1fb8d4
	else
Packit Service 5a9772
		WLog_INFO(TAG, "Loading device service %s (static)", ServiceName);
Packit Service 5a9772
Packit Service 5a9772
	entry = (PDEVICE_SERVICE_ENTRY)freerdp_load_channel_addin_entry(ServiceName, NULL,
Packit Service 5a9772
	                                                                "DeviceServiceEntry", 0);
Packit 1fb8d4
Packit 1fb8d4
	if (!entry)
Packit 1fb8d4
	{
Packit Service 5a9772
		WLog_INFO(TAG, "freerdp_load_channel_addin_entry failed!");
Packit 1fb8d4
		return ERROR_INTERNAL_ERROR;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	ep.devman = devman;
Packit 1fb8d4
	ep.RegisterDevice = devman_register_device;
Packit Service 5a9772
	ep.device = devconv.dp;
Packit 1fb8d4
	ep.rdpcontext = rdpcontext;
Packit 1fb8d4
	return entry(&ep);
Packit 1fb8d4
}