22#include <freerdp/config.h>
25#include <winpr/assert.h>
26#include <winpr/print.h>
28#include <freerdp/freerdp.h>
29#include <freerdp/assistance.h>
31#include <freerdp/channels/log.h>
32#include <freerdp/client/remdesk.h>
34#include "remdesk_main.h"
35#include "remdesk_common.h"
48 WLog_ERR(TAG,
"remdesk was null!");
50 return CHANNEL_RC_INVALID_INSTANCE;
53 WINPR_ASSERT(remdesk->channelEntryPoints.pVirtualChannelWriteEx);
54 status = remdesk->channelEntryPoints.pVirtualChannelWriteEx(
55 remdesk->InitHandle, remdesk->OpenHandle, Stream_Buffer(s), (UINT32)Stream_Length(s), s);
57 if (status != CHANNEL_RC_OK)
60 WLog_ERR(TAG,
"pVirtualChannelWriteEx failed with %s [%08" PRIX32
"]",
61 WTSErrorToString(status), status);
71static UINT remdesk_generate_expert_blob(
remdeskPlugin* remdesk)
73 const char* name = NULL;
75 const char* password = NULL;
76 rdpSettings* settings = NULL;
78 WINPR_ASSERT(remdesk);
80 WINPR_ASSERT(remdesk->rdpcontext);
81 settings = remdesk->rdpcontext->settings;
82 WINPR_ASSERT(settings);
84 if (remdesk->ExpertBlob)
93 WLog_ERR(TAG,
"password was not set!");
94 return ERROR_INTERNAL_ERROR;
103 remdesk->EncryptedPassStub =
104 freerdp_assistance_encrypt_pass_stub(password, stub, &(remdesk->EncryptedPassStubSize));
106 if (!remdesk->EncryptedPassStub)
108 WLog_ERR(TAG,
"freerdp_assistance_encrypt_pass_stub failed!");
109 return ERROR_INTERNAL_ERROR;
112 pass = freerdp_assistance_bin_to_hex_string(remdesk->EncryptedPassStub,
113 remdesk->EncryptedPassStubSize);
117 WLog_ERR(TAG,
"freerdp_assistance_bin_to_hex_string failed!");
118 return ERROR_INTERNAL_ERROR;
121 remdesk->ExpertBlob = freerdp_assistance_construct_expert_blob(name, pass);
124 if (!remdesk->ExpertBlob)
126 WLog_ERR(TAG,
"freerdp_assistance_construct_expert_blob failed!");
127 return ERROR_INTERNAL_ERROR;
130 return CHANNEL_RC_OK;
138static UINT remdesk_recv_ctl_server_announce_pdu(WINPR_ATTR_UNUSED
remdeskPlugin* remdesk,
142 WINPR_ASSERT(remdesk);
144 WINPR_ASSERT(header);
146 WLog_ERR(
"TODO",
"TODO: implement");
147 return CHANNEL_RC_OK;
158 WINPR_ASSERT(remdesk);
160 WINPR_ASSERT(header);
162 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
163 return ERROR_INVALID_DATA;
165 const UINT32 versionMajor = Stream_Get_UINT32(s);
166 const UINT32 versionMinor = Stream_Get_UINT32(s);
168 if ((versionMajor != 1) || (versionMinor > 2) || (versionMinor == 0))
170 WLog_ERR(TAG,
"Unsupported protocol version %" PRIu32
".%" PRIu32, versionMajor,
174 remdesk->Version = versionMinor;
175 return CHANNEL_RC_OK;
183static UINT remdesk_send_ctl_version_info_pdu(
remdeskPlugin* remdesk)
187 WINPR_ASSERT(remdesk);
189 UINT error = remdesk_prepare_ctl_header(&(pdu.ctlHeader), REMDESK_CTL_VERSIONINFO, 8);
193 pdu.versionMajor = 1;
194 pdu.versionMinor = 2;
195 wStream* s = Stream_New(NULL, REMDESK_CHANNEL_CTL_SIZE + pdu.ctlHeader.ch.DataLength);
199 WLog_ERR(TAG,
"Stream_New failed!");
200 return CHANNEL_RC_NO_MEMORY;
203 error = remdesk_write_ctl_header(s, &(pdu.ctlHeader));
206 Stream_Free(s, TRUE);
209 Stream_Write_UINT32(s, pdu.versionMajor);
210 Stream_Write_UINT32(s, pdu.versionMinor);
211 Stream_SealLength(s);
213 if ((error = remdesk_virtual_channel_write(remdesk, s)))
214 WLog_ERR(TAG,
"remdesk_virtual_channel_write failed with error %" PRIu32
"!", error);
230 WINPR_ASSERT(remdesk);
232 WINPR_ASSERT(header);
233 WINPR_ASSERT(pResult);
235 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
236 return ERROR_INVALID_DATA;
238 Stream_Read_UINT32(s, result);
243 case REMDESK_ERROR_HELPEESAIDNO:
244 WLog_DBG(TAG,
"remote assistance connection request was denied");
245 return ERROR_CONNECTION_REFUSED;
251 return CHANNEL_RC_OK;
259static UINT remdesk_send_ctl_authenticate_pdu(
remdeskPlugin* remdesk)
261 UINT error = ERROR_INTERNAL_ERROR;
262 size_t cbExpertBlobW = 0;
263 WCHAR* expertBlobW = NULL;
264 size_t cbRaConnectionStringW = 0;
267 WINPR_ASSERT(remdesk);
269 if ((error = remdesk_generate_expert_blob(remdesk)))
271 WLog_ERR(TAG,
"remdesk_generate_expert_blob failed with error %" PRIu32
"", error);
275 const char* expertBlob = remdesk->ExpertBlob;
276 WINPR_ASSERT(remdesk->rdpcontext);
277 rdpSettings* settings = remdesk->rdpcontext->settings;
278 WINPR_ASSERT(settings);
280 const char* raConnectionString =
282 WCHAR* raConnectionStringW =
283 ConvertUtf8ToWCharAlloc(raConnectionString, &cbRaConnectionStringW);
285 if (!raConnectionStringW || (cbRaConnectionStringW > UINT32_MAX /
sizeof(WCHAR)))
288 cbRaConnectionStringW = cbRaConnectionStringW *
sizeof(WCHAR);
290 expertBlobW = ConvertUtf8ToWCharAlloc(expertBlob, &cbExpertBlobW);
295 cbExpertBlobW = cbExpertBlobW *
sizeof(WCHAR);
296 error = remdesk_prepare_ctl_header(&(ctlHeader), REMDESK_CTL_AUTHENTICATE,
297 cbRaConnectionStringW + cbExpertBlobW);
302 wStream* s = Stream_New(NULL, REMDESK_CHANNEL_CTL_SIZE + ctlHeader.ch.DataLength);
305 WLog_ERR(TAG,
"Stream_New failed!");
306 error = CHANNEL_RC_NO_MEMORY;
310 error = remdesk_write_ctl_header(s, &ctlHeader);
313 Stream_Free(s, TRUE);
316 Stream_Write(s, raConnectionStringW, cbRaConnectionStringW);
317 Stream_Write(s, expertBlobW, cbExpertBlobW);
318 Stream_SealLength(s);
320 error = remdesk_virtual_channel_write(remdesk, s);
323 WLog_ERR(TAG,
"remdesk_virtual_channel_write failed with error %" PRIu32
"!", error);
326 free(raConnectionStringW);
337static UINT remdesk_send_ctl_remote_control_desktop_pdu(
remdeskPlugin* remdesk)
342 WINPR_ASSERT(remdesk);
343 WINPR_ASSERT(remdesk->rdpcontext);
344 rdpSettings* settings = remdesk->rdpcontext->settings;
345 WINPR_ASSERT(settings);
347 const char* raConnectionString =
349 WCHAR* raConnectionStringW = ConvertUtf8ToWCharAlloc(raConnectionString, &length);
350 size_t cbRaConnectionStringW = length *
sizeof(WCHAR);
352 if (!raConnectionStringW)
353 return ERROR_INTERNAL_ERROR;
356 error = remdesk_prepare_ctl_header(&ctlHeader, REMDESK_CTL_REMOTE_CONTROL_DESKTOP,
357 cbRaConnectionStringW);
358 if (error != CHANNEL_RC_OK)
362 wStream* s = Stream_New(NULL, REMDESK_CHANNEL_CTL_SIZE + ctlHeader.ch.DataLength);
366 WLog_ERR(TAG,
"Stream_New failed!");
367 error = CHANNEL_RC_NO_MEMORY;
371 error = remdesk_write_ctl_header(s, &ctlHeader);
374 Stream_Free(s, TRUE);
377 Stream_Write(s, raConnectionStringW, cbRaConnectionStringW);
378 Stream_SealLength(s);
380 if ((error = remdesk_virtual_channel_write(remdesk, s)))
381 WLog_ERR(TAG,
"remdesk_virtual_channel_write failed with error %" PRIu32
"!", error);
385 free(raConnectionStringW);
395static UINT remdesk_send_ctl_verify_password_pdu(
remdeskPlugin* remdesk)
397 size_t cbExpertBlobW = 0;
400 WINPR_ASSERT(remdesk);
402 UINT error = remdesk_generate_expert_blob(remdesk);
405 WLog_ERR(TAG,
"remdesk_generate_expert_blob failed with error %" PRIu32
"!", error);
409 pdu.expertBlob = remdesk->ExpertBlob;
410 WCHAR* expertBlobW = ConvertUtf8ToWCharAlloc(pdu.expertBlob, &cbExpertBlobW);
415 cbExpertBlobW = cbExpertBlobW *
sizeof(WCHAR);
417 remdesk_prepare_ctl_header(&(pdu.ctlHeader), REMDESK_CTL_VERIFY_PASSWORD, cbExpertBlobW);
423 Stream_New(NULL, 1ULL * REMDESK_CHANNEL_CTL_SIZE + pdu.ctlHeader.ch.DataLength);
427 WLog_ERR(TAG,
"Stream_New failed!");
428 error = CHANNEL_RC_NO_MEMORY;
432 error = remdesk_write_ctl_header(s, &(pdu.ctlHeader));
435 Stream_Free(s, TRUE);
438 Stream_Write(s, expertBlobW, cbExpertBlobW);
439 Stream_SealLength(s);
441 error = remdesk_virtual_channel_write(remdesk, s);
444 WLog_ERR(TAG,
"remdesk_virtual_channel_write failed with error %" PRIu32
"!", error);
457static UINT remdesk_send_ctl_expert_on_vista_pdu(
remdeskPlugin* remdesk)
461 WINPR_ASSERT(remdesk);
463 UINT error = remdesk_generate_expert_blob(remdesk);
466 WLog_ERR(TAG,
"remdesk_generate_expert_blob failed with error %" PRIu32
"!", error);
469 if (remdesk->EncryptedPassStubSize > UINT32_MAX)
470 return ERROR_INTERNAL_ERROR;
472 pdu.EncryptedPasswordLength = (UINT32)remdesk->EncryptedPassStubSize;
473 pdu.EncryptedPassword = remdesk->EncryptedPassStub;
474 error = remdesk_prepare_ctl_header(&(pdu.ctlHeader), REMDESK_CTL_EXPERT_ON_VISTA,
475 pdu.EncryptedPasswordLength);
479 wStream* s = Stream_New(NULL, REMDESK_CHANNEL_CTL_SIZE + pdu.ctlHeader.ch.DataLength);
483 WLog_ERR(TAG,
"Stream_New failed!");
484 return CHANNEL_RC_NO_MEMORY;
487 error = remdesk_write_ctl_header(s, &(pdu.ctlHeader));
490 Stream_Free(s, TRUE);
493 Stream_Write(s, pdu.EncryptedPassword, pdu.EncryptedPasswordLength);
494 Stream_SealLength(s);
495 return remdesk_virtual_channel_write(remdesk, s);
505 UINT error = CHANNEL_RC_OK;
509 WINPR_ASSERT(remdesk);
511 WINPR_ASSERT(header);
513 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
514 return ERROR_INVALID_DATA;
516 Stream_Read_UINT32(s, msgType);
522 case REMDESK_CTL_REMOTE_CONTROL_DESKTOP:
525 case REMDESK_CTL_RESULT:
526 if ((error = remdesk_recv_ctl_result_pdu(remdesk, s, header, &result)))
527 WLog_ERR(TAG,
"remdesk_recv_ctl_result_pdu failed with error %" PRIu32
"", error);
531 case REMDESK_CTL_AUTHENTICATE:
534 case REMDESK_CTL_SERVER_ANNOUNCE:
535 if ((error = remdesk_recv_ctl_server_announce_pdu(remdesk, s, header)))
536 WLog_ERR(TAG,
"remdesk_recv_ctl_server_announce_pdu failed with error %" PRIu32
"",
541 case REMDESK_CTL_DISCONNECT:
544 case REMDESK_CTL_VERSIONINFO:
545 if ((error = remdesk_recv_ctl_version_info_pdu(remdesk, s, header)))
547 WLog_ERR(TAG,
"remdesk_recv_ctl_version_info_pdu failed with error %" PRIu32
"",
552 if (remdesk->Version == 1)
554 if ((error = remdesk_send_ctl_version_info_pdu(remdesk)))
556 WLog_ERR(TAG,
"remdesk_send_ctl_version_info_pdu failed with error %" PRIu32
"",
561 if ((error = remdesk_send_ctl_authenticate_pdu(remdesk)))
563 WLog_ERR(TAG,
"remdesk_send_ctl_authenticate_pdu failed with error %" PRIu32
"",
568 if ((error = remdesk_send_ctl_remote_control_desktop_pdu(remdesk)))
572 "remdesk_send_ctl_remote_control_desktop_pdu failed with error %" PRIu32
"",
577 else if (remdesk->Version == 2)
579 if ((error = remdesk_send_ctl_expert_on_vista_pdu(remdesk)))
582 "remdesk_send_ctl_expert_on_vista_pdu failed with error %" PRIu32
"",
587 if ((error = remdesk_send_ctl_verify_password_pdu(remdesk)))
590 "remdesk_send_ctl_verify_password_pdu failed with error %" PRIu32
"",
598 case REMDESK_CTL_ISCONNECTED:
601 case REMDESK_CTL_VERIFY_PASSWORD:
604 case REMDESK_CTL_EXPERT_ON_VISTA:
607 case REMDESK_CTL_RANOVICE_NAME:
610 case REMDESK_CTL_RAEXPERT_NAME:
613 case REMDESK_CTL_TOKEN:
617 WLog_ERR(TAG,
"unknown msgType: %" PRIu32
"", msgType);
618 error = ERROR_INVALID_DATA;
635 WINPR_ASSERT(remdesk);
638 if ((status = remdesk_read_channel_header(s, &header)))
640 WLog_ERR(TAG,
"remdesk_read_channel_header failed with error %" PRIu32
"", status);
644 if (strcmp(header.ChannelName,
"RC_CTL") == 0)
646 status = remdesk_recv_ctl_pdu(remdesk, s, &header);
648 else if (strcmp(header.ChannelName,
"70") == 0)
651 else if (strcmp(header.ChannelName,
"71") == 0)
654 else if (strcmp(header.ChannelName,
".") == 0)
657 else if (strcmp(header.ChannelName,
"1000.") == 0)
660 else if (strcmp(header.ChannelName,
"RA_FX") == 0)
670static void remdesk_process_connect(WINPR_ATTR_UNUSED
remdeskPlugin* remdesk)
672 WINPR_ASSERT(remdesk);
673 WLog_ERR(
"TODO",
"TODO: implement");
681static UINT remdesk_virtual_channel_event_data_received(
remdeskPlugin* remdesk,
const void* pData,
682 UINT32 dataLength, UINT32 totalLength,
687 WINPR_ASSERT(remdesk);
689 if ((dataFlags & CHANNEL_FLAG_SUSPEND) || (dataFlags & CHANNEL_FLAG_RESUME))
691 return CHANNEL_RC_OK;
694 if (dataFlags & CHANNEL_FLAG_FIRST)
696 if (remdesk->data_in)
697 Stream_Free(remdesk->data_in, TRUE);
699 remdesk->data_in = Stream_New(NULL, totalLength);
701 if (!remdesk->data_in)
703 WLog_ERR(TAG,
"Stream_New failed!");
704 return CHANNEL_RC_NO_MEMORY;
708 data_in = remdesk->data_in;
710 if (!Stream_EnsureRemainingCapacity(data_in, dataLength))
712 WLog_ERR(TAG,
"Stream_EnsureRemainingCapacity failed!");
713 return CHANNEL_RC_NO_MEMORY;
716 Stream_Write(data_in, pData, dataLength);
718 if (dataFlags & CHANNEL_FLAG_LAST)
720 if (Stream_Capacity(data_in) != Stream_GetPosition(data_in))
722 WLog_ERR(TAG,
"read error");
723 return ERROR_INTERNAL_ERROR;
726 remdesk->data_in = NULL;
727 Stream_SealLength(data_in);
728 Stream_SetPosition(data_in, 0);
730 if (!MessageQueue_Post(remdesk->queue, NULL, 0, (
void*)data_in, NULL))
732 WLog_ERR(TAG,
"MessageQueue_Post failed!");
733 return ERROR_INTERNAL_ERROR;
737 return CHANNEL_RC_OK;
740static VOID VCAPITYPE remdesk_virtual_channel_open_event_ex(LPVOID lpUserParam, DWORD openHandle,
741 UINT event, LPVOID pData,
742 UINT32 dataLength, UINT32 totalLength,
745 UINT error = CHANNEL_RC_OK;
750 case CHANNEL_EVENT_INITIALIZED:
753 case CHANNEL_EVENT_DATA_RECEIVED:
754 if (!remdesk || (remdesk->OpenHandle != openHandle))
756 WLog_ERR(TAG,
"error no match");
759 if ((error = remdesk_virtual_channel_event_data_received(remdesk, pData, dataLength,
760 totalLength, dataFlags)))
762 "remdesk_virtual_channel_event_data_received failed with error %" PRIu32
768 case CHANNEL_EVENT_WRITE_CANCELLED:
769 case CHANNEL_EVENT_WRITE_COMPLETE:
772 Stream_Free(s, TRUE);
776 case CHANNEL_EVENT_USER:
780 WLog_ERR(TAG,
"unhandled event %" PRIu32
"!", event);
781 error = ERROR_INTERNAL_ERROR;
785 if (error && remdesk && remdesk->rdpcontext)
786 setChannelError(remdesk->rdpcontext, error,
787 "remdesk_virtual_channel_open_event_ex reported an error");
790static DWORD WINAPI remdesk_virtual_channel_client_thread(LPVOID arg)
793 wMessage message = { 0 };
795 UINT error = CHANNEL_RC_OK;
797 WINPR_ASSERT(remdesk);
799 remdesk_process_connect(remdesk);
803 if (!MessageQueue_Wait(remdesk->queue))
805 WLog_ERR(TAG,
"MessageQueue_Wait failed!");
806 error = ERROR_INTERNAL_ERROR;
810 if (!MessageQueue_Peek(remdesk->queue, &message, TRUE))
812 WLog_ERR(TAG,
"MessageQueue_Peek failed!");
813 error = ERROR_INTERNAL_ERROR;
817 if (message.id == WMQ_QUIT)
822 data = (
wStream*)message.wParam;
824 if ((error = remdesk_process_receive(remdesk, data)))
826 WLog_ERR(TAG,
"remdesk_process_receive failed with error %" PRIu32
"!", error);
827 Stream_Free(data, TRUE);
831 Stream_Free(data, TRUE);
835 if (error && remdesk->rdpcontext)
836 setChannelError(remdesk->rdpcontext, error,
837 "remdesk_virtual_channel_client_thread reported an error");
848static UINT remdesk_virtual_channel_event_connected(
remdeskPlugin* remdesk,
849 WINPR_ATTR_UNUSED LPVOID pData,
850 WINPR_ATTR_UNUSED UINT32 dataLength)
854 WINPR_ASSERT(remdesk);
856 remdesk->queue = MessageQueue_New(NULL);
860 WLog_ERR(TAG,
"MessageQueue_New failed!");
861 error = CHANNEL_RC_NO_MEMORY;
866 CreateThread(NULL, 0, remdesk_virtual_channel_client_thread, (
void*)remdesk, 0, NULL);
868 if (!remdesk->thread)
870 WLog_ERR(TAG,
"CreateThread failed");
871 error = ERROR_INTERNAL_ERROR;
875 return remdesk->channelEntryPoints.pVirtualChannelOpenEx(
876 remdesk->InitHandle, &remdesk->OpenHandle, remdesk->channelDef.name,
877 remdesk_virtual_channel_open_event_ex);
879 MessageQueue_Free(remdesk->queue);
880 remdesk->queue = NULL;
889static UINT remdesk_virtual_channel_event_disconnected(
remdeskPlugin* remdesk)
891 UINT rc = CHANNEL_RC_OK;
893 WINPR_ASSERT(remdesk);
895 if (remdesk->queue && remdesk->thread)
897 if (MessageQueue_PostQuit(remdesk->queue, 0) &&
898 (WaitForSingleObject(remdesk->thread, INFINITE) == WAIT_FAILED))
901 WLog_ERR(TAG,
"WaitForSingleObject failed with error %" PRIu32
"", rc);
906 if (remdesk->OpenHandle != 0)
908 WINPR_ASSERT(remdesk->channelEntryPoints.pVirtualChannelCloseEx);
909 rc = remdesk->channelEntryPoints.pVirtualChannelCloseEx(remdesk->InitHandle,
910 remdesk->OpenHandle);
912 if (CHANNEL_RC_OK != rc)
914 WLog_ERR(TAG,
"pVirtualChannelCloseEx failed with %s [%08" PRIX32
"]",
915 WTSErrorToString(rc), rc);
918 remdesk->OpenHandle = 0;
920 MessageQueue_Free(remdesk->queue);
921 (void)CloseHandle(remdesk->thread);
922 Stream_Free(remdesk->data_in, TRUE);
923 remdesk->data_in = NULL;
924 remdesk->queue = NULL;
925 remdesk->thread = NULL;
929static void remdesk_virtual_channel_event_terminated(
remdeskPlugin* remdesk)
931 WINPR_ASSERT(remdesk);
933 remdesk->InitHandle = 0;
934 free(remdesk->context);
938static VOID VCAPITYPE remdesk_virtual_channel_init_event_ex(LPVOID lpUserParam, LPVOID pInitHandle,
939 UINT event, LPVOID pData,
942 UINT error = CHANNEL_RC_OK;
945 if (!remdesk || (remdesk->InitHandle != pInitHandle))
947 WLog_ERR(TAG,
"error no match");
953 case CHANNEL_EVENT_CONNECTED:
954 if ((error = remdesk_virtual_channel_event_connected(remdesk, pData, dataLength)))
956 "remdesk_virtual_channel_event_connected failed with error %" PRIu32
"",
961 case CHANNEL_EVENT_DISCONNECTED:
962 if ((error = remdesk_virtual_channel_event_disconnected(remdesk)))
964 "remdesk_virtual_channel_event_disconnected failed with error %" PRIu32
"",
969 case CHANNEL_EVENT_TERMINATED:
970 remdesk_virtual_channel_event_terminated(remdesk);
973 case CHANNEL_EVENT_ATTACHED:
974 case CHANNEL_EVENT_DETACHED:
979 if (error && remdesk->rdpcontext)
980 setChannelError(remdesk->rdpcontext, error,
981 "remdesk_virtual_channel_init_event reported an error");
985#define VirtualChannelEntryEx remdesk_VirtualChannelEntryEx
987FREERDP_ENTRY_POINT(BOOL VCAPITYPE VirtualChannelEntryEx(PCHANNEL_ENTRY_POINTS_EX pEntryPoints,
1004 WLog_ERR(TAG,
"calloc failed!");
1008 remdesk->channelDef.options = CHANNEL_OPTION_INITIALIZED | CHANNEL_OPTION_ENCRYPT_RDP |
1009 CHANNEL_OPTION_COMPRESS_RDP | CHANNEL_OPTION_SHOW_PROTOCOL;
1010 (void)sprintf_s(remdesk->channelDef.name, ARRAYSIZE(remdesk->channelDef.name),
1011 REMDESK_SVC_CHANNEL_NAME);
1012 remdesk->Version = 2;
1016 (pEntryPointsEx->MagicNumber == FREERDP_CHANNEL_MAGIC_NUMBER))
1022 WLog_ERR(TAG,
"calloc failed!");
1026 context->handle = (
void*)remdesk;
1027 remdesk->context = context;
1028 remdesk->rdpcontext = pEntryPointsEx->context;
1031 CopyMemory(&(remdesk->channelEntryPoints), pEntryPoints,
1033 remdesk->InitHandle = pInitHandle;
1034 rc = remdesk->channelEntryPoints.pVirtualChannelInitEx(
1035 remdesk, context, pInitHandle, &remdesk->channelDef, 1, VIRTUAL_CHANNEL_VERSION_WIN2000,
1036 remdesk_virtual_channel_init_event_ex);
1038 if (CHANNEL_RC_OK != rc)
1040 WLog_ERR(TAG,
"pVirtualChannelInitEx failed with %s [%08" PRIX32
"]", WTSErrorToString(rc),
1045 remdesk->channelEntryPoints.pInterface = context;
FREERDP_API const char * freerdp_settings_get_string(const rdpSettings *settings, FreeRDP_Settings_Keys_String id)
Returns a immutable string settings value.