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 UINT32 versionMajor = 0;
159 UINT32 versionMinor = 0;
161 WINPR_ASSERT(remdesk);
163 WINPR_ASSERT(header);
165 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
166 return ERROR_INVALID_DATA;
168 Stream_Read_UINT32(s, versionMajor);
169 Stream_Read_UINT32(s, versionMinor);
171 if ((versionMajor != 1) || (versionMinor > 2) || (versionMinor == 0))
173 WLog_ERR(TAG,
"Unsupported protocol version %" PRId32
".%" PRId32, versionMajor,
177 remdesk->Version = versionMinor;
178 return CHANNEL_RC_OK;
186static UINT remdesk_send_ctl_version_info_pdu(
remdeskPlugin* remdesk)
190 WINPR_ASSERT(remdesk);
192 UINT error = remdesk_prepare_ctl_header(&(pdu.ctlHeader), REMDESK_CTL_VERSIONINFO, 8);
196 pdu.versionMajor = 1;
197 pdu.versionMinor = 2;
198 wStream* s = Stream_New(NULL, REMDESK_CHANNEL_CTL_SIZE + pdu.ctlHeader.ch.DataLength);
202 WLog_ERR(TAG,
"Stream_New failed!");
203 return CHANNEL_RC_NO_MEMORY;
206 error = remdesk_write_ctl_header(s, &(pdu.ctlHeader));
209 Stream_Free(s, TRUE);
212 Stream_Write_UINT32(s, pdu.versionMajor);
213 Stream_Write_UINT32(s, pdu.versionMinor);
214 Stream_SealLength(s);
216 if ((error = remdesk_virtual_channel_write(remdesk, s)))
217 WLog_ERR(TAG,
"remdesk_virtual_channel_write failed with error %" PRIu32
"!", error);
233 WINPR_ASSERT(remdesk);
235 WINPR_ASSERT(header);
236 WINPR_ASSERT(pResult);
238 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
239 return ERROR_INVALID_DATA;
241 Stream_Read_UINT32(s, result);
246 case REMDESK_ERROR_HELPEESAIDNO:
247 WLog_DBG(TAG,
"remote assistance connection request was denied");
248 return ERROR_CONNECTION_REFUSED;
254 return CHANNEL_RC_OK;
262static UINT remdesk_send_ctl_authenticate_pdu(
remdeskPlugin* remdesk)
264 UINT error = ERROR_INTERNAL_ERROR;
265 size_t cbExpertBlobW = 0;
266 WCHAR* expertBlobW = NULL;
267 size_t cbRaConnectionStringW = 0;
270 WINPR_ASSERT(remdesk);
272 if ((error = remdesk_generate_expert_blob(remdesk)))
274 WLog_ERR(TAG,
"remdesk_generate_expert_blob failed with error %" PRIu32
"", error);
278 const char* expertBlob = remdesk->ExpertBlob;
279 WINPR_ASSERT(remdesk->rdpcontext);
280 rdpSettings* settings = remdesk->rdpcontext->settings;
281 WINPR_ASSERT(settings);
283 const char* raConnectionString =
285 WCHAR* raConnectionStringW =
286 ConvertUtf8ToWCharAlloc(raConnectionString, &cbRaConnectionStringW);
288 if (!raConnectionStringW || (cbRaConnectionStringW > UINT32_MAX /
sizeof(WCHAR)))
291 cbRaConnectionStringW = cbRaConnectionStringW *
sizeof(WCHAR);
293 expertBlobW = ConvertUtf8ToWCharAlloc(expertBlob, &cbExpertBlobW);
298 cbExpertBlobW = cbExpertBlobW *
sizeof(WCHAR);
299 error = remdesk_prepare_ctl_header(&(ctlHeader), REMDESK_CTL_AUTHENTICATE,
300 cbRaConnectionStringW + cbExpertBlobW);
304 wStream* s = Stream_New(NULL, REMDESK_CHANNEL_CTL_SIZE + ctlHeader.ch.DataLength);
308 WLog_ERR(TAG,
"Stream_New failed!");
309 error = CHANNEL_RC_NO_MEMORY;
313 error = remdesk_write_ctl_header(s, &ctlHeader);
316 Stream_Free(s, TRUE);
319 Stream_Write(s, raConnectionStringW, cbRaConnectionStringW);
320 Stream_Write(s, expertBlobW, cbExpertBlobW);
321 Stream_SealLength(s);
323 error = remdesk_virtual_channel_write(remdesk, s);
325 WLog_ERR(TAG,
"remdesk_virtual_channel_write failed with error %" PRIu32
"!", error);
328 free(raConnectionStringW);
339static UINT remdesk_send_ctl_remote_control_desktop_pdu(
remdeskPlugin* remdesk)
344 WINPR_ASSERT(remdesk);
345 WINPR_ASSERT(remdesk->rdpcontext);
346 rdpSettings* settings = remdesk->rdpcontext->settings;
347 WINPR_ASSERT(settings);
349 const char* raConnectionString =
351 WCHAR* raConnectionStringW = ConvertUtf8ToWCharAlloc(raConnectionString, &length);
352 size_t cbRaConnectionStringW = length *
sizeof(WCHAR);
354 if (!raConnectionStringW)
355 return ERROR_INTERNAL_ERROR;
358 error = remdesk_prepare_ctl_header(&ctlHeader, REMDESK_CTL_REMOTE_CONTROL_DESKTOP,
359 cbRaConnectionStringW);
360 if (error != CHANNEL_RC_OK)
363 wStream* s = Stream_New(NULL, REMDESK_CHANNEL_CTL_SIZE + ctlHeader.ch.DataLength);
367 WLog_ERR(TAG,
"Stream_New failed!");
368 error = CHANNEL_RC_NO_MEMORY;
372 error = remdesk_write_ctl_header(s, &ctlHeader);
375 Stream_Free(s, TRUE);
378 Stream_Write(s, raConnectionStringW, cbRaConnectionStringW);
379 Stream_SealLength(s);
381 if ((error = remdesk_virtual_channel_write(remdesk, s)))
382 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);
421 wStream* s = Stream_New(NULL, 1ULL * REMDESK_CHANNEL_CTL_SIZE + pdu.ctlHeader.ch.DataLength);
425 WLog_ERR(TAG,
"Stream_New failed!");
426 error = CHANNEL_RC_NO_MEMORY;
430 error = remdesk_write_ctl_header(s, &(pdu.ctlHeader));
433 Stream_Free(s, TRUE);
436 Stream_Write(s, expertBlobW, cbExpertBlobW);
437 Stream_SealLength(s);
439 error = remdesk_virtual_channel_write(remdesk, s);
441 WLog_ERR(TAG,
"remdesk_virtual_channel_write failed with error %" PRIu32
"!", error);
454static UINT remdesk_send_ctl_expert_on_vista_pdu(
remdeskPlugin* remdesk)
458 WINPR_ASSERT(remdesk);
460 UINT error = remdesk_generate_expert_blob(remdesk);
463 WLog_ERR(TAG,
"remdesk_generate_expert_blob failed with error %" PRIu32
"!", error);
466 if (remdesk->EncryptedPassStubSize > UINT32_MAX)
467 return ERROR_INTERNAL_ERROR;
469 pdu.EncryptedPasswordLength = (UINT32)remdesk->EncryptedPassStubSize;
470 pdu.EncryptedPassword = remdesk->EncryptedPassStub;
471 error = remdesk_prepare_ctl_header(&(pdu.ctlHeader), REMDESK_CTL_EXPERT_ON_VISTA,
472 pdu.EncryptedPasswordLength);
476 wStream* s = Stream_New(NULL, REMDESK_CHANNEL_CTL_SIZE + pdu.ctlHeader.ch.DataLength);
480 WLog_ERR(TAG,
"Stream_New failed!");
481 return CHANNEL_RC_NO_MEMORY;
484 error = remdesk_write_ctl_header(s, &(pdu.ctlHeader));
487 Stream_Free(s, TRUE);
490 Stream_Write(s, pdu.EncryptedPassword, pdu.EncryptedPasswordLength);
491 Stream_SealLength(s);
492 return remdesk_virtual_channel_write(remdesk, s);
502 UINT error = CHANNEL_RC_OK;
506 WINPR_ASSERT(remdesk);
508 WINPR_ASSERT(header);
510 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
511 return ERROR_INVALID_DATA;
513 Stream_Read_UINT32(s, msgType);
519 case REMDESK_CTL_REMOTE_CONTROL_DESKTOP:
522 case REMDESK_CTL_RESULT:
523 if ((error = remdesk_recv_ctl_result_pdu(remdesk, s, header, &result)))
524 WLog_ERR(TAG,
"remdesk_recv_ctl_result_pdu failed with error %" PRIu32
"", error);
528 case REMDESK_CTL_AUTHENTICATE:
531 case REMDESK_CTL_SERVER_ANNOUNCE:
532 if ((error = remdesk_recv_ctl_server_announce_pdu(remdesk, s, header)))
533 WLog_ERR(TAG,
"remdesk_recv_ctl_server_announce_pdu failed with error %" PRIu32
"",
538 case REMDESK_CTL_DISCONNECT:
541 case REMDESK_CTL_VERSIONINFO:
542 if ((error = remdesk_recv_ctl_version_info_pdu(remdesk, s, header)))
544 WLog_ERR(TAG,
"remdesk_recv_ctl_version_info_pdu failed with error %" PRIu32
"",
549 if (remdesk->Version == 1)
551 if ((error = remdesk_send_ctl_version_info_pdu(remdesk)))
553 WLog_ERR(TAG,
"remdesk_send_ctl_version_info_pdu failed with error %" PRIu32
"",
558 if ((error = remdesk_send_ctl_authenticate_pdu(remdesk)))
560 WLog_ERR(TAG,
"remdesk_send_ctl_authenticate_pdu failed with error %" PRIu32
"",
565 if ((error = remdesk_send_ctl_remote_control_desktop_pdu(remdesk)))
569 "remdesk_send_ctl_remote_control_desktop_pdu failed with error %" PRIu32
"",
574 else if (remdesk->Version == 2)
576 if ((error = remdesk_send_ctl_expert_on_vista_pdu(remdesk)))
579 "remdesk_send_ctl_expert_on_vista_pdu failed with error %" PRIu32
"",
584 if ((error = remdesk_send_ctl_verify_password_pdu(remdesk)))
587 "remdesk_send_ctl_verify_password_pdu failed with error %" PRIu32
"",
595 case REMDESK_CTL_ISCONNECTED:
598 case REMDESK_CTL_VERIFY_PASSWORD:
601 case REMDESK_CTL_EXPERT_ON_VISTA:
604 case REMDESK_CTL_RANOVICE_NAME:
607 case REMDESK_CTL_RAEXPERT_NAME:
610 case REMDESK_CTL_TOKEN:
614 WLog_ERR(TAG,
"unknown msgType: %" PRIu32
"", msgType);
615 error = ERROR_INVALID_DATA;
632 WINPR_ASSERT(remdesk);
635 if ((status = remdesk_read_channel_header(s, &header)))
637 WLog_ERR(TAG,
"remdesk_read_channel_header failed with error %" PRIu32
"", status);
641 if (strcmp(header.ChannelName,
"RC_CTL") == 0)
643 status = remdesk_recv_ctl_pdu(remdesk, s, &header);
645 else if (strcmp(header.ChannelName,
"70") == 0)
648 else if (strcmp(header.ChannelName,
"71") == 0)
651 else if (strcmp(header.ChannelName,
".") == 0)
654 else if (strcmp(header.ChannelName,
"1000.") == 0)
657 else if (strcmp(header.ChannelName,
"RA_FX") == 0)
667static void remdesk_process_connect(WINPR_ATTR_UNUSED
remdeskPlugin* remdesk)
669 WINPR_ASSERT(remdesk);
670 WLog_ERR(
"TODO",
"TODO: implement");
678static UINT remdesk_virtual_channel_event_data_received(
remdeskPlugin* remdesk,
const void* pData,
679 UINT32 dataLength, UINT32 totalLength,
684 WINPR_ASSERT(remdesk);
686 if ((dataFlags & CHANNEL_FLAG_SUSPEND) || (dataFlags & CHANNEL_FLAG_RESUME))
688 return CHANNEL_RC_OK;
691 if (dataFlags & CHANNEL_FLAG_FIRST)
693 if (remdesk->data_in)
694 Stream_Free(remdesk->data_in, TRUE);
696 remdesk->data_in = Stream_New(NULL, totalLength);
698 if (!remdesk->data_in)
700 WLog_ERR(TAG,
"Stream_New failed!");
701 return CHANNEL_RC_NO_MEMORY;
705 data_in = remdesk->data_in;
707 if (!Stream_EnsureRemainingCapacity(data_in, dataLength))
709 WLog_ERR(TAG,
"Stream_EnsureRemainingCapacity failed!");
710 return CHANNEL_RC_NO_MEMORY;
713 Stream_Write(data_in, pData, dataLength);
715 if (dataFlags & CHANNEL_FLAG_LAST)
717 if (Stream_Capacity(data_in) != Stream_GetPosition(data_in))
719 WLog_ERR(TAG,
"read error");
720 return ERROR_INTERNAL_ERROR;
723 remdesk->data_in = NULL;
724 Stream_SealLength(data_in);
725 Stream_SetPosition(data_in, 0);
727 if (!MessageQueue_Post(remdesk->queue, NULL, 0, (
void*)data_in, NULL))
729 WLog_ERR(TAG,
"MessageQueue_Post failed!");
730 return ERROR_INTERNAL_ERROR;
734 return CHANNEL_RC_OK;
737static VOID VCAPITYPE remdesk_virtual_channel_open_event_ex(LPVOID lpUserParam, DWORD openHandle,
738 UINT event, LPVOID pData,
739 UINT32 dataLength, UINT32 totalLength,
742 UINT error = CHANNEL_RC_OK;
747 case CHANNEL_EVENT_INITIALIZED:
750 case CHANNEL_EVENT_DATA_RECEIVED:
751 if (!remdesk || (remdesk->OpenHandle != openHandle))
753 WLog_ERR(TAG,
"error no match");
756 if ((error = remdesk_virtual_channel_event_data_received(remdesk, pData, dataLength,
757 totalLength, dataFlags)))
759 "remdesk_virtual_channel_event_data_received failed with error %" PRIu32
765 case CHANNEL_EVENT_WRITE_CANCELLED:
766 case CHANNEL_EVENT_WRITE_COMPLETE:
769 Stream_Free(s, TRUE);
773 case CHANNEL_EVENT_USER:
777 WLog_ERR(TAG,
"unhandled event %" PRIu32
"!", event);
778 error = ERROR_INTERNAL_ERROR;
782 if (error && remdesk && remdesk->rdpcontext)
783 setChannelError(remdesk->rdpcontext, error,
784 "remdesk_virtual_channel_open_event_ex reported an error");
787static DWORD WINAPI remdesk_virtual_channel_client_thread(LPVOID arg)
790 wMessage message = { 0 };
792 UINT error = CHANNEL_RC_OK;
794 WINPR_ASSERT(remdesk);
796 remdesk_process_connect(remdesk);
800 if (!MessageQueue_Wait(remdesk->queue))
802 WLog_ERR(TAG,
"MessageQueue_Wait failed!");
803 error = ERROR_INTERNAL_ERROR;
807 if (!MessageQueue_Peek(remdesk->queue, &message, TRUE))
809 WLog_ERR(TAG,
"MessageQueue_Peek failed!");
810 error = ERROR_INTERNAL_ERROR;
814 if (message.id == WMQ_QUIT)
819 data = (
wStream*)message.wParam;
821 if ((error = remdesk_process_receive(remdesk, data)))
823 WLog_ERR(TAG,
"remdesk_process_receive failed with error %" PRIu32
"!", error);
824 Stream_Free(data, TRUE);
828 Stream_Free(data, TRUE);
832 if (error && remdesk->rdpcontext)
833 setChannelError(remdesk->rdpcontext, error,
834 "remdesk_virtual_channel_client_thread reported an error");
845static UINT remdesk_virtual_channel_event_connected(
remdeskPlugin* remdesk,
846 WINPR_ATTR_UNUSED LPVOID pData,
847 WINPR_ATTR_UNUSED UINT32 dataLength)
851 WINPR_ASSERT(remdesk);
853 remdesk->queue = MessageQueue_New(NULL);
857 WLog_ERR(TAG,
"MessageQueue_New failed!");
858 error = CHANNEL_RC_NO_MEMORY;
863 CreateThread(NULL, 0, remdesk_virtual_channel_client_thread, (
void*)remdesk, 0, NULL);
865 if (!remdesk->thread)
867 WLog_ERR(TAG,
"CreateThread failed");
868 error = ERROR_INTERNAL_ERROR;
872 return remdesk->channelEntryPoints.pVirtualChannelOpenEx(
873 remdesk->InitHandle, &remdesk->OpenHandle, remdesk->channelDef.name,
874 remdesk_virtual_channel_open_event_ex);
876 MessageQueue_Free(remdesk->queue);
877 remdesk->queue = NULL;
886static UINT remdesk_virtual_channel_event_disconnected(
remdeskPlugin* remdesk)
888 UINT rc = CHANNEL_RC_OK;
890 WINPR_ASSERT(remdesk);
892 if (remdesk->queue && remdesk->thread)
894 if (MessageQueue_PostQuit(remdesk->queue, 0) &&
895 (WaitForSingleObject(remdesk->thread, INFINITE) == WAIT_FAILED))
898 WLog_ERR(TAG,
"WaitForSingleObject failed with error %" PRIu32
"", rc);
903 if (remdesk->OpenHandle != 0)
905 WINPR_ASSERT(remdesk->channelEntryPoints.pVirtualChannelCloseEx);
906 rc = remdesk->channelEntryPoints.pVirtualChannelCloseEx(remdesk->InitHandle,
907 remdesk->OpenHandle);
909 if (CHANNEL_RC_OK != rc)
911 WLog_ERR(TAG,
"pVirtualChannelCloseEx failed with %s [%08" PRIX32
"]",
912 WTSErrorToString(rc), rc);
915 remdesk->OpenHandle = 0;
917 MessageQueue_Free(remdesk->queue);
918 (void)CloseHandle(remdesk->thread);
919 Stream_Free(remdesk->data_in, TRUE);
920 remdesk->data_in = NULL;
921 remdesk->queue = NULL;
922 remdesk->thread = NULL;
926static void remdesk_virtual_channel_event_terminated(
remdeskPlugin* remdesk)
928 WINPR_ASSERT(remdesk);
930 remdesk->InitHandle = 0;
931 free(remdesk->context);
935static VOID VCAPITYPE remdesk_virtual_channel_init_event_ex(LPVOID lpUserParam, LPVOID pInitHandle,
936 UINT event, LPVOID pData,
939 UINT error = CHANNEL_RC_OK;
942 if (!remdesk || (remdesk->InitHandle != pInitHandle))
944 WLog_ERR(TAG,
"error no match");
950 case CHANNEL_EVENT_CONNECTED:
951 if ((error = remdesk_virtual_channel_event_connected(remdesk, pData, dataLength)))
953 "remdesk_virtual_channel_event_connected failed with error %" PRIu32
"",
958 case CHANNEL_EVENT_DISCONNECTED:
959 if ((error = remdesk_virtual_channel_event_disconnected(remdesk)))
961 "remdesk_virtual_channel_event_disconnected failed with error %" PRIu32
"",
966 case CHANNEL_EVENT_TERMINATED:
967 remdesk_virtual_channel_event_terminated(remdesk);
970 case CHANNEL_EVENT_ATTACHED:
971 case CHANNEL_EVENT_DETACHED:
976 if (error && remdesk->rdpcontext)
977 setChannelError(remdesk->rdpcontext, error,
978 "remdesk_virtual_channel_init_event reported an error");
982#define VirtualChannelEntryEx remdesk_VirtualChannelEntryEx
984FREERDP_ENTRY_POINT(BOOL VCAPITYPE VirtualChannelEntryEx(PCHANNEL_ENTRY_POINTS pEntryPoints,
1001 WLog_ERR(TAG,
"calloc failed!");
1005 remdesk->channelDef.options = CHANNEL_OPTION_INITIALIZED | CHANNEL_OPTION_ENCRYPT_RDP |
1006 CHANNEL_OPTION_COMPRESS_RDP | CHANNEL_OPTION_SHOW_PROTOCOL;
1007 (void)sprintf_s(remdesk->channelDef.name, ARRAYSIZE(remdesk->channelDef.name),
1008 REMDESK_SVC_CHANNEL_NAME);
1009 remdesk->Version = 2;
1013 (pEntryPointsEx->MagicNumber == FREERDP_CHANNEL_MAGIC_NUMBER))
1019 WLog_ERR(TAG,
"calloc failed!");
1023 context->handle = (
void*)remdesk;
1024 remdesk->context = context;
1025 remdesk->rdpcontext = pEntryPointsEx->context;
1028 CopyMemory(&(remdesk->channelEntryPoints), pEntryPoints,
1030 remdesk->InitHandle = pInitHandle;
1031 rc = remdesk->channelEntryPoints.pVirtualChannelInitEx(
1032 remdesk, context, pInitHandle, &remdesk->channelDef, 1, VIRTUAL_CHANNEL_VERSION_WIN2000,
1033 remdesk_virtual_channel_init_event_ex);
1035 if (CHANNEL_RC_OK != rc)
1037 WLog_ERR(TAG,
"pVirtualChannelInitEx failed with %s [%08" PRIX32
"]", WTSErrorToString(rc),
1042 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.