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