Blame server/proxy/pf_server.c

Packit Service b1ea74
/**
Packit Service b1ea74
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit Service b1ea74
 * FreeRDP Proxy Server
Packit Service b1ea74
 *
Packit Service b1ea74
 * Copyright 2019 Mati Shabtay <matishabtay@gmail.com>
Packit Service b1ea74
 * Copyright 2019 Kobi Mizrachi <kmizrachi18@gmail.com>
Packit Service b1ea74
 * Copyright 2019 Idan Freiberg <speidy@gmail.com>
Packit Service b1ea74
 *
Packit Service b1ea74
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit Service b1ea74
 * you may not use this file except in compliance with the License.
Packit Service b1ea74
 * You may obtain a copy of the License at
Packit Service b1ea74
 *
Packit Service b1ea74
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit Service b1ea74
 *
Packit Service b1ea74
 * Unless required by applicable law or agreed to in writing, software
Packit Service b1ea74
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit Service b1ea74
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit Service b1ea74
 * See the License for the specific language governing permissions and
Packit Service b1ea74
 * limitations under the License.
Packit Service b1ea74
 */
Packit Service b1ea74
Packit Service b1ea74
#include <winpr/crt.h>
Packit Service b1ea74
#include <winpr/ssl.h>
Packit Service b1ea74
#include <winpr/synch.h>
Packit Service b1ea74
#include <winpr/string.h>
Packit Service b1ea74
#include <winpr/winsock.h>
Packit Service b1ea74
#include <winpr/thread.h>
Packit Service b1ea74
#include <errno.h>
Packit Service b1ea74
Packit Service b1ea74
#include <freerdp/freerdp.h>
Packit Service b1ea74
#include <freerdp/channels/wtsvc.h>
Packit Service b1ea74
#include <freerdp/channels/channels.h>
Packit Service b1ea74
Packit Service b1ea74
#include "pf_server.h"
Packit Service b1ea74
#include "pf_log.h"
Packit Service b1ea74
#include "pf_config.h"
Packit Service b1ea74
#include "pf_client.h"
Packit Service b1ea74
#include "pf_context.h"
Packit Service b1ea74
#include "pf_update.h"
Packit Service b1ea74
#include "pf_rdpgfx.h"
Packit Service b1ea74
#include "pf_disp.h"
Packit Service b1ea74
#include "pf_rail.h"
Packit Service b1ea74
#include "pf_channels.h"
Packit Service b1ea74
#include "pf_modules.h"
Packit Service b1ea74
Packit Service b1ea74
#define TAG PROXY_TAG("server")
Packit Service b1ea74
Packit Service b1ea74
static psPeerReceiveChannelData server_receive_channel_data_original = NULL;
Packit Service b1ea74
Packit Service b1ea74
static BOOL pf_server_parse_target_from_routing_token(rdpContext* context, char** target,
Packit Service b1ea74
                                                      DWORD* port)
Packit Service b1ea74
{
Packit Service b1ea74
#define TARGET_MAX (100)
Packit Service b1ea74
#define ROUTING_TOKEN_PREFIX "Cookie: msts="
Packit Service b1ea74
	char* colon;
Packit Service b1ea74
	size_t len;
Packit Service b1ea74
	DWORD routing_token_length;
Packit Service b1ea74
	const size_t prefix_len = strnlen(ROUTING_TOKEN_PREFIX, sizeof(ROUTING_TOKEN_PREFIX));
Packit Service b1ea74
	const char* routing_token = freerdp_nego_get_routing_token(context, &routing_token_length);
Packit Service b1ea74
	pServerContext* ps = (pServerContext*)context;
Packit Service b1ea74
Packit Service b1ea74
	if (routing_token == NULL)
Packit Service b1ea74
	{
Packit Service b1ea74
		/* no routing token */
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	if ((routing_token_length <= prefix_len) || (routing_token_length >= TARGET_MAX))
Packit Service b1ea74
	{
Packit Service b1ea74
		LOG_ERR(TAG, ps, "invalid routing token length: %" PRIu32 "", routing_token_length);
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	len = routing_token_length - prefix_len;
Packit Service b1ea74
	*target = malloc(len + 1);
Packit Service b1ea74
Packit Service b1ea74
	if (!(*target))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	CopyMemory(*target, routing_token + prefix_len, len);
Packit Service b1ea74
	*(*target + len) = '\0';
Packit Service b1ea74
	colon = strchr(*target, ':');
Packit Service b1ea74
Packit Service b1ea74
	if (colon)
Packit Service b1ea74
	{
Packit Service b1ea74
		/* port is specified */
Packit Service b1ea74
		unsigned long p = strtoul(colon + 1, NULL, 10);
Packit Service b1ea74
Packit Service b1ea74
		if (p > USHRT_MAX)
Packit Service b1ea74
		{
Packit Service b1ea74
			free(*target);
Packit Service b1ea74
			return FALSE;
Packit Service b1ea74
		}
Packit Service b1ea74
Packit Service b1ea74
		*port = (DWORD)p;
Packit Service b1ea74
		*colon = '\0';
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL pf_server_get_target_info(rdpContext* context, rdpSettings* settings,
Packit Service b1ea74
                                      proxyConfig* config)
Packit Service b1ea74
{
Packit Service b1ea74
	pServerContext* ps = (pServerContext*)context;
Packit Service b1ea74
Packit Service b1ea74
	LOG_INFO(TAG, ps, "fetching target from %s",
Packit Service b1ea74
	         config->UseLoadBalanceInfo ? "load-balance-info" : "config");
Packit Service b1ea74
Packit Service b1ea74
	if (config->UseLoadBalanceInfo)
Packit Service b1ea74
		return pf_server_parse_target_from_routing_token(context, &settings->ServerHostname,
Packit Service b1ea74
		                                                 &settings->ServerPort);
Packit Service b1ea74
Packit Service b1ea74
	/* use hardcoded target info from configuration */
Packit Service b1ea74
	if (!(settings->ServerHostname = _strdup(config->TargetHost)))
Packit Service b1ea74
	{
Packit Service b1ea74
		LOG_ERR(TAG, ps, "strdup failed!");
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	settings->ServerPort = config->TargetPort > 0 ? 3389 : settings->ServerPort;
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
/* Event callbacks */
Packit Service b1ea74
/**
Packit Service b1ea74
 * This callback is called when the entire connection sequence is done (as
Packit Service b1ea74
 * described in MS-RDPBCGR section 1.3)
Packit Service b1ea74
 *
Packit Service b1ea74
 * The server may start sending graphics output and receiving keyboard/mouse
Packit Service b1ea74
 * input after this callback returns.
Packit Service b1ea74
 */
Packit Service b1ea74
static BOOL pf_server_post_connect(freerdp_peer* peer)
Packit Service b1ea74
{
Packit Service b1ea74
	pServerContext* ps;
Packit Service b1ea74
	pClientContext* pc;
Packit Service b1ea74
	rdpSettings* client_settings;
Packit Service b1ea74
	proxyData* pdata;
Packit Service b1ea74
	ps = (pServerContext*)peer->context;
Packit Service b1ea74
	pdata = ps->pdata;
Packit Service b1ea74
Packit Service b1ea74
	if (pdata->config->SessionCapture && !peer->settings->SupportGraphicsPipeline)
Packit Service b1ea74
	{
Packit Service b1ea74
		LOG_ERR(TAG, ps, "Session capture feature is enabled, only accepting connections with GFX");
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	pc = pf_context_create_client_context(peer->settings);
Packit Service b1ea74
	if (pc == NULL)
Packit Service b1ea74
	{
Packit Service b1ea74
		LOG_ERR(TAG, ps, "[%s]: pf_context_create_client_context failed!");
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	client_settings = pc->context.settings;
Packit Service b1ea74
Packit Service b1ea74
	/* keep both sides of the connection in pdata */
Packit Service b1ea74
	proxy_data_set_client_context(pdata, pc);
Packit Service b1ea74
Packit Service b1ea74
	if (!pf_server_get_target_info(peer->context, client_settings, pdata->config))
Packit Service b1ea74
	{
Packit Service b1ea74
Packit Service b1ea74
		LOG_INFO(TAG, ps, "pf_server_get_target_info failed!");
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	LOG_INFO(TAG, ps, "remote target is %s:%" PRIu16 "", client_settings->ServerHostname,
Packit Service b1ea74
	         client_settings->ServerPort);
Packit Service b1ea74
Packit Service b1ea74
	if (!pf_server_channels_init(ps))
Packit Service b1ea74
	{
Packit Service b1ea74
		LOG_INFO(TAG, ps, "failed to initialize server's channels!");
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	/* Start a proxy's client in it's own thread */
Packit Service b1ea74
	if (!(pdata->client_thread = CreateThread(NULL, 0, pf_client_start, pc, 0, NULL)))
Packit Service b1ea74
	{
Packit Service b1ea74
		LOG_ERR(TAG, ps, "failed to create client thread");
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	return pf_modules_run_hook(HOOK_TYPE_SERVER_POST_CONNECT, pdata);
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL pf_server_activate(freerdp_peer* peer)
Packit Service b1ea74
{
Packit Service b1ea74
	peer->settings->CompressionLevel = PACKET_COMPR_TYPE_RDP8;
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL pf_server_adjust_monitor_layout(freerdp_peer* peer)
Packit Service b1ea74
{
Packit Service b1ea74
	/* proxy as is, there's no need to do anything here */
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL pf_server_receive_channel_data_hook(freerdp_peer* peer, UINT16 channelId,
Packit Service b1ea74
                                                const BYTE* data, size_t size, UINT32 flags,
Packit Service b1ea74
                                                size_t totalSize)
Packit Service b1ea74
{
Packit Service b1ea74
	pServerContext* ps = (pServerContext*)peer->context;
Packit Service b1ea74
	pClientContext* pc = ps->pdata->pc;
Packit Service b1ea74
	proxyData* pdata = pc->pdata;
Packit Service b1ea74
	proxyConfig* config = pdata->config;
Packit Service b1ea74
	size_t i;
Packit Service b1ea74
	const char* channel_name = WTSChannelGetName(peer, channelId);
Packit Service b1ea74
Packit Service b1ea74
	for (i = 0; i < config->PassthroughCount; i++)
Packit Service b1ea74
	{
Packit Service b1ea74
		if (strncmp(channel_name, config->Passthrough[i], CHANNEL_NAME_LEN) == 0)
Packit Service b1ea74
		{
Packit Service b1ea74
			proxyChannelDataEventInfo ev;
Packit Service b1ea74
			UINT64 client_channel_id;
Packit Service b1ea74
Packit Service b1ea74
			ev.channel_id = channelId;
Packit Service b1ea74
			ev.channel_name = channel_name;
Packit Service b1ea74
			ev.data = data;
Packit Service b1ea74
			ev.data_len = size;
Packit Service b1ea74
Packit Service b1ea74
			if (!pf_modules_run_filter(FILTER_TYPE_SERVER_PASSTHROUGH_CHANNEL_DATA, pdata, &ev))
Packit Service b1ea74
				return FALSE;
Packit Service b1ea74
Packit Service b1ea74
			client_channel_id = (UINT64)HashTable_GetItemValue(pc->vc_ids, (void*)channel_name);
Packit Service b1ea74
Packit Service b1ea74
			return pc->context.instance->SendChannelData(pc->context.instance,
Packit Service b1ea74
			                                             (UINT16)client_channel_id, data, size);
Packit Service b1ea74
		}
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	return server_receive_channel_data_original(peer, channelId, data, size, flags, totalSize);
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL pf_server_initialize_peer_connection(freerdp_peer* peer)
Packit Service b1ea74
{
Packit Service b1ea74
	pServerContext* ps = (pServerContext*)peer->context;
Packit Service b1ea74
	rdpSettings* settings = peer->settings;
Packit Service b1ea74
	proxyData* pdata;
Packit Service b1ea74
	proxyConfig* config;
Packit Service b1ea74
	proxyServer* server;
Packit Service b1ea74
Packit Service b1ea74
	if (!ps)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	pdata = proxy_data_new();
Packit Service b1ea74
	if (!pdata)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	proxy_data_set_server_context(pdata, ps);
Packit Service b1ea74
	server = (proxyServer*)peer->ContextExtra;
Packit Service b1ea74
Packit Service b1ea74
	config = pdata->config = server->config;
Packit Service b1ea74
Packit Service b1ea74
	/* currently not supporting GDI orders */
Packit Service b1ea74
	ZeroMemory(settings->OrderSupport, 32);
Packit Service b1ea74
	peer->update->autoCalculateBitmapData = FALSE;
Packit Service b1ea74
Packit Service b1ea74
	settings->SupportMonitorLayoutPdu = TRUE;
Packit Service b1ea74
	settings->SupportGraphicsPipeline = config->GFX;
Packit Service b1ea74
	settings->CertificateFile = _strdup("server.crt");
Packit Service b1ea74
	settings->PrivateKeyFile = _strdup("server.key");
Packit Service b1ea74
	settings->RdpKeyFile = _strdup("server.key");
Packit Service b1ea74
Packit Service b1ea74
	if (config->RemoteApp)
Packit Service b1ea74
	{
Packit Service b1ea74
		settings->RemoteApplicationSupportLevel =
Packit Service b1ea74
		    RAIL_LEVEL_SUPPORTED | RAIL_LEVEL_DOCKED_LANGBAR_SUPPORTED |
Packit Service b1ea74
		    RAIL_LEVEL_SHELL_INTEGRATION_SUPPORTED | RAIL_LEVEL_LANGUAGE_IME_SYNC_SUPPORTED |
Packit Service b1ea74
		    RAIL_LEVEL_SERVER_TO_CLIENT_IME_SYNC_SUPPORTED |
Packit Service b1ea74
		    RAIL_LEVEL_HIDE_MINIMIZED_APPS_SUPPORTED | RAIL_LEVEL_WINDOW_CLOAKING_SUPPORTED |
Packit Service b1ea74
		    RAIL_LEVEL_HANDSHAKE_EX_SUPPORTED;
Packit Service b1ea74
		settings->RemoteAppLanguageBarSupported = TRUE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	if (!settings->CertificateFile || !settings->PrivateKeyFile || !settings->RdpKeyFile)
Packit Service b1ea74
	{
Packit Service b1ea74
		WLog_ERR(TAG, "Memory allocation failed (strdup)");
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	settings->RdpSecurity = config->ServerRdpSecurity;
Packit Service b1ea74
	settings->TlsSecurity = config->ServerTlsSecurity;
Packit Service b1ea74
	settings->NlaSecurity = FALSE; /* currently NLA is not supported in proxy server */
Packit Service b1ea74
	settings->EncryptionLevel = ENCRYPTION_LEVEL_CLIENT_COMPATIBLE;
Packit Service b1ea74
	settings->ColorDepth = 32;
Packit Service b1ea74
	settings->SuppressOutput = TRUE;
Packit Service b1ea74
	settings->RefreshRect = TRUE;
Packit Service b1ea74
	settings->DesktopResize = TRUE;
Packit Service b1ea74
Packit Service b1ea74
	peer->PostConnect = pf_server_post_connect;
Packit Service b1ea74
	peer->Activate = pf_server_activate;
Packit Service b1ea74
	peer->AdjustMonitorsLayout = pf_server_adjust_monitor_layout;
Packit Service b1ea74
	peer->settings->MultifragMaxRequestSize = 0xFFFFFF; /* FIXME */
Packit Service b1ea74
Packit Service b1ea74
	/* virtual channels receive data hook */
Packit Service b1ea74
	server_receive_channel_data_original = peer->ReceiveChannelData;
Packit Service b1ea74
	peer->ReceiveChannelData = pf_server_receive_channel_data_hook;
Packit Service b1ea74
Packit Service b1ea74
	if (ArrayList_Add(server->clients, pdata) < 0)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	CountdownEvent_AddCount(server->waitGroup, 1);
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
/**
Packit Service b1ea74
 * Handles an incoming client connection, to be run in it's own thread.
Packit Service b1ea74
 *
Packit Service b1ea74
 * arg is a pointer to a freerdp_peer representing the client.
Packit Service b1ea74
 */
Packit Service b1ea74
static DWORD WINAPI pf_server_handle_peer(LPVOID arg)
Packit Service b1ea74
{
Packit Service b1ea74
	HANDLE eventHandles[32];
Packit Service b1ea74
	HANDLE ChannelEvent;
Packit Service b1ea74
	DWORD eventCount;
Packit Service b1ea74
	DWORD tmp;
Packit Service b1ea74
	DWORD status;
Packit Service b1ea74
	pServerContext* ps;
Packit Service b1ea74
	rdpContext* pc;
Packit Service b1ea74
	proxyData* pdata;
Packit Service b1ea74
	freerdp_peer* client = (freerdp_peer*)arg;
Packit Service b1ea74
	proxyServer* server = (proxyServer*)client->ContextExtra;
Packit Service b1ea74
Packit Service b1ea74
	if (!pf_context_init_server_context(client))
Packit Service b1ea74
		goto out_free_peer;
Packit Service b1ea74
Packit Service b1ea74
	if (!pf_server_initialize_peer_connection(client))
Packit Service b1ea74
		goto out_free_peer;
Packit Service b1ea74
Packit Service b1ea74
	ps = (pServerContext*)client->context;
Packit Service b1ea74
	pdata = ps->pdata;
Packit Service b1ea74
Packit Service b1ea74
	client->Initialize(client);
Packit Service b1ea74
	LOG_INFO(TAG, ps, "peer connected: %s", client->hostname);
Packit Service b1ea74
	/* Main client event handling loop */
Packit Service b1ea74
	ChannelEvent = WTSVirtualChannelManagerGetEventHandle(ps->vcm);
Packit Service b1ea74
Packit Service b1ea74
	while (1)
Packit Service b1ea74
	{
Packit Service b1ea74
		eventCount = 0;
Packit Service b1ea74
		{
Packit Service b1ea74
			tmp = client->GetEventHandles(client, &eventHandles[eventCount], 32 - eventCount);
Packit Service b1ea74
Packit Service b1ea74
			if (tmp == 0)
Packit Service b1ea74
			{
Packit Service b1ea74
				WLog_ERR(TAG, "Failed to get FreeRDP transport event handles");
Packit Service b1ea74
				break;
Packit Service b1ea74
			}
Packit Service b1ea74
Packit Service b1ea74
			eventCount += tmp;
Packit Service b1ea74
		}
Packit Service b1ea74
		eventHandles[eventCount++] = ChannelEvent;
Packit Service b1ea74
		eventHandles[eventCount++] = pdata->abort_event;
Packit Service b1ea74
		eventHandles[eventCount++] = WTSVirtualChannelManagerGetEventHandle(ps->vcm);
Packit Service b1ea74
		status = WaitForMultipleObjects(eventCount, eventHandles, FALSE, INFINITE);
Packit Service b1ea74
Packit Service b1ea74
		if (status == WAIT_FAILED)
Packit Service b1ea74
		{
Packit Service b1ea74
			WLog_ERR(TAG, "WaitForMultipleObjects failed (status: %d)", status);
Packit Service b1ea74
			break;
Packit Service b1ea74
		}
Packit Service b1ea74
Packit Service b1ea74
		if (client->CheckFileDescriptor(client) != TRUE)
Packit Service b1ea74
			break;
Packit Service b1ea74
Packit Service b1ea74
		if (WaitForSingleObject(ChannelEvent, 0) == WAIT_OBJECT_0)
Packit Service b1ea74
		{
Packit Service b1ea74
			if (!WTSVirtualChannelManagerCheckFileDescriptor(ps->vcm))
Packit Service b1ea74
			{
Packit Service b1ea74
				WLog_ERR(TAG, "WTSVirtualChannelManagerCheckFileDescriptor failure");
Packit Service b1ea74
				goto fail;
Packit Service b1ea74
			}
Packit Service b1ea74
		}
Packit Service b1ea74
Packit Service b1ea74
		/* only disconnect after checking client's and vcm's file descriptors  */
Packit Service b1ea74
		if (proxy_data_shall_disconnect(pdata))
Packit Service b1ea74
		{
Packit Service b1ea74
			WLog_INFO(TAG, "abort event is set, closing connection with peer %s", client->hostname);
Packit Service b1ea74
			break;
Packit Service b1ea74
		}
Packit Service b1ea74
Packit Service b1ea74
		switch (WTSVirtualChannelManagerGetDrdynvcState(ps->vcm))
Packit Service b1ea74
		{
Packit Service b1ea74
			/* Dynamic channel status may have been changed after processing */
Packit Service b1ea74
			case DRDYNVC_STATE_NONE:
Packit Service b1ea74
Packit Service b1ea74
				/* Initialize drdynvc channel */
Packit Service b1ea74
				if (!WTSVirtualChannelManagerCheckFileDescriptor(ps->vcm))
Packit Service b1ea74
				{
Packit Service b1ea74
					WLog_ERR(TAG, "Failed to initialize drdynvc channel");
Packit Service b1ea74
					goto fail;
Packit Service b1ea74
				}
Packit Service b1ea74
Packit Service b1ea74
				break;
Packit Service b1ea74
Packit Service b1ea74
			case DRDYNVC_STATE_READY:
Packit Service b1ea74
				if (WaitForSingleObject(ps->dynvcReady, 0) == WAIT_TIMEOUT)
Packit Service b1ea74
				{
Packit Service b1ea74
					SetEvent(ps->dynvcReady);
Packit Service b1ea74
				}
Packit Service b1ea74
Packit Service b1ea74
				break;
Packit Service b1ea74
Packit Service b1ea74
			default:
Packit Service b1ea74
				break;
Packit Service b1ea74
		}
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
fail:
Packit Service b1ea74
Packit Service b1ea74
	pc = (rdpContext*)pdata->pc;
Packit Service b1ea74
	LOG_INFO(TAG, ps, "starting shutdown of connection");
Packit Service b1ea74
	LOG_INFO(TAG, ps, "stopping proxy's client");
Packit Service b1ea74
	freerdp_client_stop(pc);
Packit Service b1ea74
	LOG_INFO(TAG, ps, "freeing server's channels");
Packit Service b1ea74
	pf_server_channels_free(ps);
Packit Service b1ea74
	LOG_INFO(TAG, ps, "freeing proxy data");
Packit Service b1ea74
	ArrayList_Remove(server->clients, pdata);
Packit Service b1ea74
	proxy_data_free(pdata);
Packit Service b1ea74
	freerdp_client_context_free(pc);
Packit Service b1ea74
	client->Close(client);
Packit Service b1ea74
	client->Disconnect(client);
Packit Service b1ea74
out_free_peer:
Packit Service b1ea74
	freerdp_peer_context_free(client);
Packit Service b1ea74
	freerdp_peer_free(client);
Packit Service b1ea74
	CountdownEvent_Signal(server->waitGroup, 1);
Packit Service b1ea74
	ExitThread(0);
Packit Service b1ea74
	return 0;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL pf_server_peer_accepted(freerdp_listener* listener, freerdp_peer* client)
Packit Service b1ea74
{
Packit Service b1ea74
	HANDLE hThread;
Packit Service b1ea74
	client->ContextExtra = listener->info;
Packit Service b1ea74
Packit Service b1ea74
	if (!(hThread = CreateThread(NULL, 0, pf_server_handle_peer, (void*)client, 0, NULL)))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	CloseHandle(hThread);
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static DWORD WINAPI pf_server_mainloop(LPVOID arg)
Packit Service b1ea74
{
Packit Service b1ea74
	HANDLE eventHandles[32];
Packit Service b1ea74
	DWORD eventCount;
Packit Service b1ea74
	DWORD status;
Packit Service b1ea74
	proxyServer* server = (proxyServer*)arg;
Packit Service b1ea74
	freerdp_listener* listener = server->listener;
Packit Service b1ea74
Packit Service b1ea74
	while (1)
Packit Service b1ea74
	{
Packit Service b1ea74
		eventCount = listener->GetEventHandles(listener, eventHandles, 32);
Packit Service b1ea74
Packit Service b1ea74
		if (0 == eventCount)
Packit Service b1ea74
		{
Packit Service b1ea74
			WLog_ERR(TAG, "Failed to get FreeRDP event handles");
Packit Service b1ea74
			break;
Packit Service b1ea74
		}
Packit Service b1ea74
Packit Service b1ea74
		eventHandles[eventCount++] = server->stopEvent;
Packit Service b1ea74
		status = WaitForMultipleObjects(eventCount, eventHandles, FALSE, INFINITE);
Packit Service b1ea74
Packit Service b1ea74
		if (WaitForSingleObject(server->stopEvent, 0) == WAIT_OBJECT_0)
Packit Service b1ea74
			break;
Packit Service b1ea74
Packit Service b1ea74
		if (WAIT_FAILED == status)
Packit Service b1ea74
		{
Packit Service b1ea74
			WLog_ERR(TAG, "select failed");
Packit Service b1ea74
			break;
Packit Service b1ea74
		}
Packit Service b1ea74
Packit Service b1ea74
		if (listener->CheckFileDescriptor(listener) != TRUE)
Packit Service b1ea74
		{
Packit Service b1ea74
			WLog_ERR(TAG, "Failed to check FreeRDP file descriptor");
Packit Service b1ea74
			break;
Packit Service b1ea74
		}
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	listener->Close(listener);
Packit Service b1ea74
	ExitThread(0);
Packit Service b1ea74
	return 0;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
BOOL pf_server_start(proxyServer* server)
Packit Service b1ea74
{
Packit Service b1ea74
	WSADATA wsaData;
Packit Service b1ea74
	WTSRegisterWtsApiFunctionTable(FreeRDP_InitWtsApi());
Packit Service b1ea74
	winpr_InitializeSSL(WINPR_SSL_INIT_DEFAULT);
Packit Service b1ea74
Packit Service b1ea74
	if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)
Packit Service b1ea74
		goto error;
Packit Service b1ea74
Packit Service b1ea74
	if (!server->listener->Open(server->listener, server->config->Host, server->config->Port))
Packit Service b1ea74
	{
Packit Service b1ea74
		switch (errno)
Packit Service b1ea74
		{
Packit Service b1ea74
			case EADDRINUSE:
Packit Service b1ea74
				WLog_ERR(TAG, "failed to start listener: address already in use!");
Packit Service b1ea74
				break;
Packit Service b1ea74
			case EACCES:
Packit Service b1ea74
				WLog_ERR(TAG, "failed to start listener: insufficent permissions!");
Packit Service b1ea74
				break;
Packit Service b1ea74
			default:
Packit Service b1ea74
				WLog_ERR(TAG, "failed to start listener: errno=%d", errno);
Packit Service b1ea74
				break;
Packit Service b1ea74
		}
Packit Service b1ea74
Packit Service b1ea74
		goto error;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	server->thread = CreateThread(NULL, 0, pf_server_mainloop, (void*)server, 0, NULL);
Packit Service b1ea74
	if (!server->thread)
Packit Service b1ea74
		goto error;
Packit Service b1ea74
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
Packit Service b1ea74
error:
Packit Service b1ea74
	WSACleanup();
Packit Service b1ea74
	return FALSE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static void pf_server_clients_list_client_free(void* obj)
Packit Service b1ea74
{
Packit Service b1ea74
	proxyData* pdata = (proxyData*)obj;
Packit Service b1ea74
	proxy_data_abort_connect(pdata);
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
proxyServer* pf_server_new(proxyConfig* config)
Packit Service b1ea74
{
Packit Service b1ea74
	proxyServer* server;
Packit Service b1ea74
Packit Service b1ea74
	if (!config)
Packit Service b1ea74
		return NULL;
Packit Service b1ea74
Packit Service b1ea74
	server = calloc(1, sizeof(proxyServer));
Packit Service b1ea74
	if (!server)
Packit Service b1ea74
		return NULL;
Packit Service b1ea74
Packit Service b1ea74
	server->config = config;
Packit Service b1ea74
Packit Service b1ea74
	server->stopEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
Packit Service b1ea74
	if (!server->stopEvent)
Packit Service b1ea74
		goto out;
Packit Service b1ea74
Packit Service b1ea74
	server->clients = ArrayList_New(TRUE);
Packit Service b1ea74
	if (!server->clients)
Packit Service b1ea74
		goto out;
Packit Service b1ea74
Packit Service b1ea74
	server->clients->object.fnObjectFree = pf_server_clients_list_client_free;
Packit Service b1ea74
Packit Service b1ea74
	server->waitGroup = CountdownEvent_New(0);
Packit Service b1ea74
	if (!server->waitGroup)
Packit Service b1ea74
		goto out;
Packit Service b1ea74
Packit Service b1ea74
	server->listener = freerdp_listener_new();
Packit Service b1ea74
	if (!server->listener)
Packit Service b1ea74
		goto out;
Packit Service b1ea74
Packit Service b1ea74
	server->listener->info = server;
Packit Service b1ea74
	server->listener->PeerAccepted = pf_server_peer_accepted;
Packit Service b1ea74
	return server;
Packit Service b1ea74
Packit Service b1ea74
out:
Packit Service b1ea74
	pf_server_free(server);
Packit Service b1ea74
	return NULL;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
void pf_server_stop(proxyServer* server)
Packit Service b1ea74
{
Packit Service b1ea74
	HANDLE waitHandle = INVALID_HANDLE_VALUE;
Packit Service b1ea74
Packit Service b1ea74
	if (!server)
Packit Service b1ea74
		return;
Packit Service b1ea74
Packit Service b1ea74
	/* clear clients list, also disconnects every client */
Packit Service b1ea74
	ArrayList_Clear(server->clients);
Packit Service b1ea74
Packit Service b1ea74
	/* block until all clients are disconnected */
Packit Service b1ea74
	waitHandle = CountdownEvent_WaitHandle(server->waitGroup);
Packit Service b1ea74
	if (WaitForSingleObject(waitHandle, INFINITE) != WAIT_OBJECT_0)
Packit Service b1ea74
		WLog_ERR(TAG, "[%s]: WaitForSingleObject failed!", __FUNCTION__);
Packit Service b1ea74
Packit Service b1ea74
	/* signal main thread to stop and wait for the thread to exit */
Packit Service b1ea74
	SetEvent(server->stopEvent);
Packit Service b1ea74
	WaitForSingleObject(server->thread, INFINITE);
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
void pf_server_free(proxyServer* server)
Packit Service b1ea74
{
Packit Service b1ea74
	if (!server)
Packit Service b1ea74
		return;
Packit Service b1ea74
Packit Service b1ea74
	freerdp_listener_free(server->listener);
Packit Service b1ea74
	ArrayList_Free(server->clients);
Packit Service b1ea74
	CountdownEvent_Free(server->waitGroup);
Packit Service b1ea74
Packit Service b1ea74
	if (server->stopEvent)
Packit Service b1ea74
		CloseHandle(server->stopEvent);
Packit Service b1ea74
Packit Service b1ea74
	if (server->thread)
Packit Service b1ea74
		CloseHandle(server->thread);
Packit Service b1ea74
Packit Service b1ea74
	free(server);
Packit Service b1ea74
}