24#include <freerdp/config.h>
28#include <winpr/path.h>
29#include <winpr/synch.h>
30#include <winpr/string.h>
31#include <winpr/winsock.h>
32#include <winpr/thread.h>
35#include <freerdp/freerdp.h>
36#include <freerdp/streamdump.h>
37#include <freerdp/channels/wtsvc.h>
38#include <freerdp/channels/channels.h>
39#include <freerdp/channels/drdynvc.h>
40#include <freerdp/build-config.h>
42#include <freerdp/channels/rdpdr.h>
44#include <freerdp/server/proxy/proxy_server.h>
45#include <freerdp/server/proxy/proxy_log.h>
49#include "pf_channel.h"
50#include <freerdp/server/proxy/proxy_config.h>
52#include <freerdp/server/proxy/proxy_context.h>
54#include "proxy_modules.h"
56#include "channels/pf_channel_drdynvc.h"
57#include "channels/pf_channel_rdpdr.h"
59#define TAG PROXY_TAG("server")
68static BOOL pf_server_parse_target_from_routing_token(rdpContext* context, rdpSettings* settings,
69 FreeRDP_Settings_Keys_String targetID,
70 FreeRDP_Settings_Keys_UInt32 portID)
72#define TARGET_MAX (100)
73#define ROUTING_TOKEN_PREFIX "Cookie: msts="
74 char* colon =
nullptr;
76 DWORD routing_token_length = 0;
77 const size_t prefix_len = strnlen(ROUTING_TOKEN_PREFIX,
sizeof(ROUTING_TOKEN_PREFIX));
78 const char* routing_token = freerdp_nego_get_routing_token(context, &routing_token_length);
79 pServerContext* ps = (pServerContext*)context;
84 if ((routing_token_length <= prefix_len) || (routing_token_length >= TARGET_MAX))
86 PROXY_LOG_ERR(TAG, ps,
"invalid routing token length: %" PRIu32
"", routing_token_length);
90 len = routing_token_length - prefix_len;
96 colon = strchr(target,
':');
101 unsigned long p = strtoul(colon + 1,
nullptr, 10);
114static BOOL pf_server_get_target_info(rdpContext* context, rdpSettings* settings,
115 const proxyConfig* config)
117 pServerContext* ps = (pServerContext*)context;
120 WINPR_ASSERT(settings);
122 WINPR_ASSERT(ps->pdata);
124 ev.fetch_method = config->FixedTarget ? PROXY_FETCH_TARGET_METHOD_CONFIG
125 : PROXY_FETCH_TARGET_METHOD_LOAD_BALANCE_INFO;
127 if (!pf_modules_run_filter(ps->pdata->module, FILTER_TYPE_SERVER_FETCH_TARGET_ADDR, ps->pdata,
131 switch (ev.fetch_method)
133 case PROXY_FETCH_TARGET_METHOD_DEFAULT:
134 case PROXY_FETCH_TARGET_METHOD_LOAD_BALANCE_INFO:
135 return pf_server_parse_target_from_routing_token(
136 context, settings, FreeRDP_ServerHostname, FreeRDP_ServerPort);
138 case PROXY_FETCH_TARGET_METHOD_CONFIG:
140 WINPR_ASSERT(config);
142 if (config->TargetPort > 0)
154 config->TargetTlsSecLevel))
159 PROXY_LOG_ERR(TAG, ps,
"strdup failed!");
163 if (config->TargetUser)
169 if (config->TargetDomain)
175 if (config->TargetPassword)
178 config->TargetPassword))
183 config->TargetSmartcardAuth))
186 if (config->TargetSmartcardCert)
191 config->TargetSmartcardCert,
192 config->TargetSmartcardCertLength))
196 if (config->TargetSmartcardKey)
201 config->TargetSmartcardKey,
202 config->TargetSmartcardKeyLength))
208 case PROXY_FETCH_TARGET_USE_CUSTOM_ADDR:
210 if (!ev.target_address)
214 "router: using CUSTOM_ADDR fetch method, but target_address == nullptr");
220 PROXY_LOG_ERR(TAG, ps,
"strdup failed!");
224 free(ev.target_address);
228 PROXY_LOG_ERR(TAG, ps,
"unknown target fetch method: %u", ev.fetch_method);
236static BOOL pf_server_setup_channels(freerdp_peer* peer)
239 char** accepted_channels =
nullptr;
240 size_t accepted_channels_count = 0;
241 pServerContext* ps = (pServerContext*)peer->context;
243 accepted_channels = WTSGetAcceptedChannelNames(peer, &accepted_channels_count);
244 if (!accepted_channels)
247 for (
size_t i = 0; i < accepted_channels_count; i++)
249 pServerStaticChannelContext* channelContext =
nullptr;
250 const char* cname = accepted_channels[i];
251 UINT16 channelId = WTSChannelGetId(peer, cname);
253 PROXY_LOG_INFO(TAG, ps,
"Accepted channel: %s (%" PRIu16
")", cname, channelId);
254 channelContext = StaticChannelContext_new(ps, cname, channelId);
257 PROXY_LOG_ERR(TAG, ps,
"error setting up channelContext for '%s'", cname);
261 if ((strcmp(cname, DRDYNVC_SVC_CHANNEL_NAME) == 0) &&
262 (channelContext->channelMode == PF_UTILS_CHANNEL_INTERCEPT))
264 if (!pf_channel_setup_drdynvc(ps->pdata, channelContext))
266 PROXY_LOG_ERR(TAG, ps,
"error while setting up dynamic channel");
267 StaticChannelContext_free(channelContext);
271 else if (strcmp(cname, RDPDR_SVC_CHANNEL_NAME) == 0 &&
272 (channelContext->channelMode == PF_UTILS_CHANNEL_INTERCEPT))
274 if (!pf_channel_setup_rdpdr(ps, channelContext))
276 PROXY_LOG_ERR(TAG, ps,
"error while setting up redirection channel");
277 StaticChannelContext_free(channelContext);
283 if (!pf_channel_setup_generic(channelContext))
285 PROXY_LOG_ERR(TAG, ps,
"error while setting up generic channel");
286 StaticChannelContext_free(channelContext);
291 if (!HashTable_Insert(ps->channelsByFrontId, &channelContext->front_channel_id,
294 StaticChannelContext_free(channelContext);
295 PROXY_LOG_ERR(TAG, ps,
"error inserting channelContext in byId table for '%s'", cname);
302 free((
void*)accepted_channels);
316static BOOL pf_server_post_connect(freerdp_peer* peer)
318 pServerContext* ps =
nullptr;
319 pClientContext* pc =
nullptr;
320 rdpSettings* client_settings =
nullptr;
321 proxyData* pdata =
nullptr;
322 rdpSettings* frontSettings =
nullptr;
326 ps = (pServerContext*)peer->context;
329 frontSettings = peer->context->settings;
330 WINPR_ASSERT(frontSettings);
336 PROXY_LOG_INFO(TAG, ps,
"Accepted client: %s", ClientHostname);
337 if (!pf_server_setup_channels(peer))
339 PROXY_LOG_ERR(TAG, ps,
"error setting up channels");
343 pc = pf_context_create_client_context(frontSettings);
346 PROXY_LOG_ERR(TAG, ps,
"failed to create client context!");
350 client_settings = pc->context.settings;
353 proxy_data_set_client_context(pdata, pc);
355 if (!pf_server_get_target_info(peer->context, client_settings, pdata->config))
357 PROXY_LOG_INFO(TAG, ps,
"pf_server_get_target_info failed!");
361 PROXY_LOG_INFO(TAG, ps,
"remote target is %s:%" PRIu32
"",
365 if (!pf_modules_run_hook(pdata->module, HOOK_TYPE_SERVER_POST_CONNECT, pdata, peer))
369 pdata->client_thread = CreateThread(
nullptr, 0, pf_client_start, pc, 0,
nullptr);
370 if (!pdata->client_thread)
372 PROXY_LOG_ERR(TAG, ps,
"failed to create client thread");
380static BOOL pf_server_activate(freerdp_peer* peer)
382 pServerContext* ps =
nullptr;
383 proxyData* pdata =
nullptr;
384 rdpSettings* settings =
nullptr;
388 ps = (pServerContext*)peer->context;
394 settings = peer->context->settings;
398 if (!pf_modules_run_hook(pdata->module, HOOK_TYPE_SERVER_ACTIVATE, pdata, peer))
405static BOOL pf_server_logon(freerdp_peer* peer,
const SEC_WINNT_AUTH_IDENTITY* identity,
408 pServerContext* ps =
nullptr;
409 proxyData* pdata =
nullptr;
414 ps = (pServerContext*)peer->context;
419 WINPR_ASSERT(identity);
421 info.identity = identity;
422 info.automatic = automatic;
423 return (pf_modules_run_filter(pdata->module, FILTER_TYPE_SERVER_PEER_LOGON, pdata, &info));
427static BOOL pf_server_adjust_monitor_layout(WINPR_ATTR_UNUSED freerdp_peer* peer)
435static BOOL pf_server_receive_channel_data_hook(freerdp_peer* peer, UINT16 channelId,
436 const BYTE* data,
size_t size, UINT32 flags,
439 UINT64 channelId64 = channelId;
443 pServerContext* ps = (pServerContext*)peer->context;
446 proxyData* pdata = ps->pdata;
449 pClientContext* pc = pdata->pc;
460 const pServerStaticChannelContext* channel =
461 HashTable_GetItemValue(ps->channelsByFrontId, &channelId64);
464 PROXY_LOG_ERR(TAG, ps,
"channel id=%" PRIu64
" not registered here, dropping",
469 WINPR_ASSERT(channel->onFrontData);
470 switch (channel->onFrontData(pdata, channel, data, size, flags, totalSize))
472 case PF_CHANNEL_RESULT_PASS:
476 ev.channel_id = channelId;
477 ev.channel_name = channel->channel_name;
481 ev.total_size = totalSize;
482 return IFCALLRESULT(TRUE, pc->sendChannelData, pc, &ev);
484 case PF_CHANNEL_RESULT_DROP:
486 case PF_CHANNEL_RESULT_ERROR:
492 WINPR_ASSERT(pdata->server_receive_channel_data_original);
493 return pdata->server_receive_channel_data_original(peer, channelId, data, size, flags,
498static BOOL pf_server_initialize_peer_connection(freerdp_peer* peer, proxyData* pdata)
503 pServerContext* ps = (pServerContext*)peer->context;
507 rdpSettings* settings = peer->context->settings;
508 WINPR_ASSERT(settings);
510 proxyServer* server = (proxyServer*)peer->ContextExtra;
511 WINPR_ASSERT(server);
512 proxy_data_set_server_context(pdata, ps);
514 pdata->module = server->module;
515 const proxyConfig* config = pdata->config = server->config;
517 rdpPrivateKey* key = freerdp_key_new_from_pem_enc(config->PrivateKeyPEM,
nullptr);
524 rdpCertificate* cert = freerdp_certificate_new_from_pem(config->CertificatePEM);
540 ZeroMemory(OrderSupport, 32);
543 WINPR_ASSERT(peer->context->update);
544 peer->context->update->autoCalculateBitmapData = FALSE;
551 if (pf_utils_is_passthrough(config))
557 if (config->RemoteApp)
560 RAIL_LEVEL_SUPPORTED | RAIL_LEVEL_DOCKED_LANGBAR_SUPPORTED |
561 RAIL_LEVEL_SHELL_INTEGRATION_SUPPORTED | RAIL_LEVEL_LANGUAGE_IME_SYNC_SUPPORTED |
562 RAIL_LEVEL_SERVER_TO_CLIENT_IME_SYNC_SUPPORTED |
563 RAIL_LEVEL_HIDE_MINIMIZED_APPS_SUPPORTED | RAIL_LEVEL_WINDOW_CLOAKING_SUPPORTED |
564 RAIL_LEVEL_HANDSHAKE_EX_SUPPORTED;
579 ENCRYPTION_LEVEL_CLIENT_COMPATIBLE))
600 peer->PostConnect = pf_server_post_connect;
601 peer->Activate = pf_server_activate;
602 peer->Logon = pf_server_logon;
603 peer->AdjustMonitorsLayout = pf_server_adjust_monitor_layout;
606 pdata->server_receive_channel_data_original = peer->ReceiveChannelData;
607 peer->ReceiveChannelData = pf_server_receive_channel_data_hook;
610 pf_server_register_input_callbacks(peer->context->input);
611 pf_server_register_update_callbacks(peer->context->update);
613 return (stream_dump_register_handlers(peer->context, CONNECTION_STATE_NEGO, TRUE));
622static DWORD WINAPI pf_server_handle_peer(LPVOID arg)
624 HANDLE eventHandles[MAXIMUM_WAIT_OBJECTS] = WINPR_C_ARRAY_INIT;
625 pServerContext* ps =
nullptr;
626 peer_thread_args* args = arg;
630 freerdp_peer* client = args->client;
631 WINPR_ASSERT(client);
633 proxyServer* server = (proxyServer*)client->ContextExtra;
634 WINPR_ASSERT(server);
636 ArrayList_Lock(server->peer_list);
637 size_t count = ArrayList_Count(server->peer_list);
638 ArrayList_Unlock(server->peer_list);
640 proxyData* pdata = proxy_data_new();
644 if (!pf_context_init_server_context(client))
647 if (!pf_server_initialize_peer_connection(client, pdata))
650 ps = (pServerContext*)client->context;
652 PROXY_LOG_DBG(TAG, ps,
"Added peer, %" PRIuz
" connected", count);
654 if (!pf_modules_run_hook(pdata->module, HOOK_TYPE_SERVER_SESSION_INITIALIZE, pdata, client))
657 WINPR_ASSERT(client->Initialize);
658 if (!client->Initialize(client))
661 PROXY_LOG_INFO(TAG, ps,
"new connection: proxy address: %s, client address: %s",
662 pdata->config->Host, client->hostname);
664 if (!pf_modules_run_hook(pdata->module, HOOK_TYPE_SERVER_SESSION_STARTED, pdata, client))
669 HANDLE ChannelEvent = INVALID_HANDLE_VALUE;
670 DWORD eventCount = 0;
672 WINPR_ASSERT(client->GetEventHandles);
673 const DWORD tmp = client->GetEventHandles(client, &eventHandles[eventCount],
674 ARRAYSIZE(eventHandles) - eventCount);
678 PROXY_LOG_ERR(TAG, ps,
"Failed to get FreeRDP transport event handles");
685 ChannelEvent = WTSVirtualChannelManagerGetEventHandle(ps->vcm);
687 WINPR_ASSERT(ChannelEvent && (ChannelEvent != INVALID_HANDLE_VALUE));
688 WINPR_ASSERT(pdata->abort_event && (pdata->abort_event != INVALID_HANDLE_VALUE));
689 eventHandles[eventCount++] = ChannelEvent;
690 eventHandles[eventCount++] = pdata->abort_event;
691 eventHandles[eventCount++] = server->stopEvent;
693 const DWORD status = WaitForMultipleObjects(
694 eventCount, eventHandles, FALSE, 1000);
696 if (status == WAIT_FAILED)
698 PROXY_LOG_ERR(TAG, ps,
"WaitForMultipleObjects failed (status: %" PRIu32
")", status);
702 WINPR_ASSERT(client->CheckFileDescriptor);
703 if (client->CheckFileDescriptor(client) != TRUE)
706 if (WaitForSingleObject(ChannelEvent, 0) == WAIT_OBJECT_0)
708 if (!WTSVirtualChannelManagerCheckFileDescriptor(ps->vcm))
710 PROXY_LOG_ERR(TAG, ps,
"WTSVirtualChannelManagerCheckFileDescriptor failure");
716 if (proxy_data_shall_disconnect(pdata))
718 PROXY_LOG_INFO(TAG, ps,
"abort event is set, closing connection with peer %s",
723 if (WaitForSingleObject(server->stopEvent, 0) == WAIT_OBJECT_0)
725 PROXY_LOG_INFO(TAG, ps,
"Server shutting down, terminating peer");
729 switch (WTSVirtualChannelManagerGetDrdynvcState(ps->vcm))
732 case DRDYNVC_STATE_NONE:
735 if (!WTSVirtualChannelManagerCheckFileDescriptor(ps->vcm))
737 PROXY_LOG_ERR(TAG, ps,
"Failed to initialize drdynvc channel");
743 case DRDYNVC_STATE_READY:
744 if (WaitForSingleObject(ps->dynvcReady, 0) == WAIT_TIMEOUT)
746 (void)SetEvent(ps->dynvcReady);
758 PROXY_LOG_INFO(TAG, ps,
"starting shutdown of connection");
759 PROXY_LOG_INFO(TAG, ps,
"stopping proxy's client");
762 proxy_data_abort_connect(pdata);
764 (void)pf_modules_run_hook(pdata->module, HOOK_TYPE_SERVER_SESSION_END, pdata, client);
766 PROXY_LOG_INFO(TAG, ps,
"freeing server's channels");
768 WINPR_ASSERT(client->Close);
769 client->Close(client);
771 WINPR_ASSERT(client->Disconnect);
772 client->Disconnect(client);
775 PROXY_LOG_INFO(TAG, ps,
"freeing proxy data");
777 if (pdata && pdata->client_thread)
779 PROXY_LOG_INFO(TAG, ps,
"stopping proxy RDP client thread");
780 proxy_data_abort_connect(pdata);
781 (void)WaitForSingleObject(pdata->client_thread, INFINITE);
785 PROXY_LOG_INFO(TAG, ps,
"ignore proxy RDP client thread, not started");
789 PROXY_LOG_INFO(TAG, ps,
"ignore proxy RDP client thread, not proxyData not created");
793 ArrayList_Lock(server->peer_list);
794 ArrayList_Remove(server->peer_list, args->thread);
795 count = ArrayList_Count(server->peer_list);
796 ArrayList_Unlock(server->peer_list);
798 PROXY_LOG_DBG(TAG, ps,
"Removed peer, %" PRIuz
" connected", count);
799 freerdp_peer_context_free(client);
800 freerdp_peer_free(client);
801 proxy_data_free(pdata);
803#if defined(WITH_DEBUG_EVENTS)
812static BOOL pf_server_start_peer(freerdp_peer* client)
814 HANDLE hThread =
nullptr;
815 proxyServer* server =
nullptr;
816 peer_thread_args* args = calloc(1,
sizeof(peer_thread_args));
820 WINPR_ASSERT(client);
821 args->client = client;
823 server = (proxyServer*)client->ContextExtra;
824 WINPR_ASSERT(server);
826 hThread = CreateThread(
nullptr, 0, pf_server_handle_peer, args, CREATE_SUSPENDED,
nullptr);
833 args->thread = hThread;
834 ArrayList_Lock(server->peer_list);
835 const BOOL appended = ArrayList_Append(server->peer_list, hThread);
836 ArrayList_Unlock(server->peer_list);
839 (void)CloseHandle(hThread);
844 return ResumeThread(hThread) != (DWORD)-1;
848static BOOL pf_server_peer_accepted(freerdp_listener* listener, freerdp_peer* client)
850 WINPR_ASSERT(listener);
851 WINPR_ASSERT(client);
853 client->ContextExtra = listener->info;
855 return pf_server_start_peer(client);
858BOOL pf_server_start(proxyServer* server)
862 WINPR_ASSERT(server);
864 if (!WTSRegisterWtsApiFunctionTable(FreeRDP_InitWtsApi()))
867 if (!winpr_InitializeSSL(WINPR_SSL_INIT_DEFAULT))
870 if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)
873 WINPR_ASSERT(server->config);
874 WINPR_ASSERT(server->listener);
875 WINPR_ASSERT(server->listener->Open);
876 if (!server->listener->Open(server->listener, server->config->Host, server->config->Port))
881 WLog_ERR(TAG,
"failed to start listener: address already in use!");
884 WLog_ERR(TAG,
"failed to start listener: insufficient permissions!");
887 WLog_ERR(TAG,
"failed to start listener: errno=%d", errno);
901BOOL pf_server_start_from_socket(proxyServer* server,
int socket)
905 WINPR_ASSERT(server);
907 if (!WTSRegisterWtsApiFunctionTable(FreeRDP_InitWtsApi()))
910 if (!winpr_InitializeSSL(WINPR_SSL_INIT_DEFAULT))
913 if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)
916 WINPR_ASSERT(server->listener);
917 WINPR_ASSERT(server->listener->OpenFromSocket);
918 if (!server->listener->OpenFromSocket(server->listener, socket))
923 WLog_ERR(TAG,
"failed to start listener: address already in use!");
926 WLog_ERR(TAG,
"failed to start listener: insufficient permissions!");
929 WLog_ERR(TAG,
"failed to start listener: errno=%d", errno);
943BOOL pf_server_start_with_peer_socket(proxyServer* server,
int socket)
945 struct sockaddr_storage peer_addr;
946 socklen_t len =
sizeof(peer_addr);
947 freerdp_peer* client =
nullptr;
949 WINPR_ASSERT(server);
951 if (WaitForSingleObject(server->stopEvent, 0) == WAIT_OBJECT_0)
954 client = freerdp_peer_new(socket);
958 if (getpeername(socket, (
struct sockaddr*)&peer_addr, &len) != 0)
961 if (!freerdp_peer_set_local_and_hostname(client, &peer_addr))
964 client->ContextExtra = server;
966 if (!pf_server_start_peer(client))
972 WLog_ERR(TAG,
"PeerAccepted callback failed");
973 freerdp_peer_free(client);
978static BOOL are_all_required_modules_loaded(proxyModule* module,
const proxyConfig* config)
984 if (!pf_modules_is_plugin_loaded(module, plugin_name))
986 WLog_ERR(TAG,
"Required plugin '%s' is not loaded. stopping.", plugin_name);
994static void peer_free(
void* obj)
996 HANDLE hdl = (HANDLE)obj;
997 (void)CloseHandle(hdl);
1000proxyServer* pf_server_new(
const proxyConfig* config)
1003 proxyServer* server =
nullptr;
1005 WINPR_ASSERT(config);
1007 server = calloc(1,
sizeof(proxyServer));
1014 server->module = pf_modules_new(FREERDP_PROXY_PLUGINDIR,
pf_config_modules(server->config),
1016 if (!server->module)
1018 WLog_ERR(TAG,
"failed to initialize proxy modules!");
1022 if (!pf_modules_list_loaded_plugins(server->module))
1025 if (!are_all_required_modules_loaded(server->module, server->config))
1028 server->stopEvent = CreateEvent(
nullptr, TRUE, FALSE,
nullptr);
1029 if (!server->stopEvent)
1032 server->listener = freerdp_listener_new();
1033 if (!server->listener)
1036 server->peer_list = ArrayList_New(FALSE);
1037 if (!server->peer_list)
1040 obj = ArrayList_Object(server->peer_list);
1045 server->listener->info = server;
1046 server->listener->PeerAccepted = pf_server_peer_accepted;
1048 if (!pf_modules_add(server->module,
pf_config_plugin, (
void*)server->config))
1054 WINPR_PRAGMA_DIAG_PUSH
1055 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
1056 pf_server_free(server);
1057 WINPR_PRAGMA_DIAG_POP
1061BOOL pf_server_run(proxyServer* server)
1064 HANDLE eventHandles[MAXIMUM_WAIT_OBJECTS] = WINPR_C_ARRAY_INIT;
1065 DWORD eventCount = 0;
1067 freerdp_listener* listener =
nullptr;
1069 WINPR_ASSERT(server);
1071 listener = server->listener;
1072 WINPR_ASSERT(listener);
1076 WINPR_ASSERT(listener->GetEventHandles);
1077 eventCount = listener->GetEventHandles(listener, eventHandles, ARRAYSIZE(eventHandles));
1079 if ((0 == eventCount) || (eventCount >= ARRAYSIZE(eventHandles)))
1081 WLog_ERR(TAG,
"Failed to get FreeRDP event handles");
1085 WINPR_ASSERT(server->stopEvent);
1086 eventHandles[eventCount++] = server->stopEvent;
1087 status = WaitForMultipleObjects(eventCount, eventHandles, FALSE, 1000);
1089 if (WAIT_FAILED == status)
1092 if (WaitForSingleObject(server->stopEvent, 0) == WAIT_OBJECT_0)
1095 if (WAIT_FAILED == status)
1097 WLog_ERR(TAG,
"select failed");
1102 WINPR_ASSERT(listener->CheckFileDescriptor);
1103 if (listener->CheckFileDescriptor(listener) != TRUE)
1105 WLog_ERR(TAG,
"Failed to accept new peer");
1111 WINPR_ASSERT(listener->Close);
1112 listener->Close(listener);
1116void pf_server_stop(proxyServer* server)
1123 (void)SetEvent(server->stopEvent);
1126void pf_server_free(proxyServer* server)
1131 pf_server_stop(server);
1133 if (server->peer_list)
1135 while (ArrayList_Count(server->peer_list) > 0)
1147 ArrayList_Free(server->peer_list);
1148 freerdp_listener_free(server->listener);
1150 if (server->stopEvent)
1151 (void)CloseHandle(server->stopEvent);
1154 pf_modules_free(server->module);
1157#if defined(WITH_DEBUG_EVENTS)
1162BOOL pf_server_add_module(proxyServer* server, proxyModuleEntryPoint ep,
void* userdata)
1164 WINPR_ASSERT(server);
1167 return pf_modules_add(server->module, ep, userdata);
FREERDP_API void pf_server_config_free(proxyConfig *config)
pf_server_config_free Releases all resources associated with proxyConfig
WINPR_ATTR_NODISCARD FREERDP_API const char ** pf_config_modules(const proxyConfig *config)
pf_config_modules
WINPR_ATTR_NODISCARD FREERDP_API const char * pf_config_required_plugin(const proxyConfig *config, size_t index)
pf_config_required_plugin
WINPR_ATTR_NODISCARD FREERDP_API size_t pf_config_modules_count(const proxyConfig *config)
pf_config_modules_count
WINPR_ATTR_NODISCARD FREERDP_API BOOL pf_config_clone(proxyConfig **dst, const proxyConfig *config)
pf_config_clone Create a copy of the configuration
WINPR_ATTR_NODISCARD FREERDP_API size_t pf_config_required_plugins_count(const proxyConfig *config)
pf_config_required_plugins_count
WINPR_ATTR_NODISCARD FREERDP_API BOOL pf_config_plugin(proxyPluginsManager *plugins_manager, void *userdata)
pf_config_plugin Register a proxy plugin handling event filtering defined in the configuration.
WINPR_ATTR_NODISCARD FREERDP_API const char * freerdp_settings_get_string(const rdpSettings *settings, FreeRDP_Settings_Keys_String id)
Returns a immutable string settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_set_bool(rdpSettings *settings, FreeRDP_Settings_Keys_Bool id, BOOL val)
Sets a BOOL settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_set_string_len(rdpSettings *settings, FreeRDP_Settings_Keys_String id, const char *val, size_t len)
Sets a string settings value. The val is copied.
WINPR_ATTR_NODISCARD FREERDP_API void * freerdp_settings_get_pointer_writable(rdpSettings *settings, FreeRDP_Settings_Keys_Pointer id)
Returns a mutable pointer settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_set_uint32(rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id, UINT32 val)
Sets a UINT32 settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_set_pointer_len(rdpSettings *settings, FreeRDP_Settings_Keys_Pointer id, const void *data, size_t len)
Set a pointer to value data.
WINPR_ATTR_NODISCARD FREERDP_API UINT32 freerdp_settings_get_uint32(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id)
Returns a UINT32 settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_set_string(rdpSettings *settings, FreeRDP_Settings_Keys_String id, const char *val)
Sets a string settings value. The param is copied.
This struct contains function pointer to initialize/free objects.
OBJECT_FREE_FN fnObjectFree