22#include <freerdp/config.h>
35#include "connection.h"
37#include <freerdp/buildflags.h>
38#include "gateway/rpc_fault.h"
40#include <winpr/assert.h>
43#include <winpr/string.h>
44#include <winpr/stream.h>
45#include <winpr/wtsapi.h>
47#include <winpr/debug.h>
49#include <freerdp/freerdp.h>
50#include <freerdp/streamdump.h>
51#include <freerdp/error.h>
52#include <freerdp/event.h>
53#include <freerdp/locale/keyboard.h>
54#include <freerdp/locale/locale.h>
55#include <freerdp/channels/channels.h>
56#include <freerdp/version.h>
57#include <freerdp/log.h>
58#include <freerdp/utils/signal.h>
60#include "../cache/pointer.h"
63#define TAG FREERDP_TAG("core")
65static void sig_abort_connect(
int signum,
const char* signame,
void* ctx)
67 rdpContext* context = (rdpContext*)ctx;
69 WLog_INFO(TAG,
"Signal %s [%d], terminating session %p", signame, signum,
70 WINPR_CXX_COMPAT_CAST(
const void*, context));
72 freerdp_abort_connect_context(context);
86static int freerdp_connect_begin(freerdp* instance)
89 rdpRdp* rdp =
nullptr;
91 rdpSettings* settings =
nullptr;
96 WINPR_ASSERT(instance->context);
99 instance->ConnectionCallbackState = CLIENT_STATE_INITIAL;
100 freerdp_set_last_error_log(instance->context, FREERDP_ERROR_SUCCESS);
101 clearChannelError(instance->context);
102 if (!utils_reset_abort(instance->context->rdp))
105 rdp = instance->context->rdp;
108 settings = instance->context->settings;
109 WINPR_ASSERT(settings);
111 freerdp_channels_register_instance(instance->context->channels, instance);
113 if (!freerdp_settings_set_default_order_support(settings))
116 if (!freerdp_add_signal_cleanup_handler(instance->context, sig_abort_connect))
119 IFCALLRET(instance->PreConnect, status, instance);
120 instance->ConnectionCallbackState = CLIENT_STATE_PRECONNECT_PASSED;
122 freerdp_settings_print_warnings(settings);
124 status = freerdp_settings_enforce_monitor_exists(settings);
127 status = freerdp_settings_enforce_consistency(settings);
130 status = freerdp_settings_check_client_after_preconnect(settings);
133 status = rdp_set_backup_settings(rdp);
135 status = utils_reload_channels(instance->context);
138 int64_t KeyboardLayout = freerdp_get_keyboard_default_layout_for_locale(cp);
139 if (KeyboardLayout == 0)
142 switch (KeyboardLayout)
145 case KBD_JAPANESE_INPUT_SYSTEM_MS_IME2002:
148 WINPR_KBD_TYPE_JAPANESE))
162 rdpContext* context = instance->context;
163 WINPR_ASSERT(context);
164 freerdp_set_last_error_if_not(context, FREERDP_ERROR_PRE_CONNECT_FAILED);
166 WLog_Print(context->log, WLOG_ERROR,
"freerdp_pre_connect failed: %s",
167 rdp_client_connection_state_string(instance->ConnectionCallbackState));
171 rc = rdp_client_connect(rdp);
176 rdpContext* context = rdp->context;
177 WINPR_ASSERT(context);
178 WLog_Print(context->log, WLOG_ERROR,
"Authentication only, exit status %" PRId32
"", rc);
185BOOL freerdp_connect(freerdp* instance)
188 ConnectionResultEventArgs e = WINPR_C_ARRAY_INIT;
189 const int rc = freerdp_connect_begin(instance);
190 rdpRdp* rdp =
nullptr;
191 UINT status2 = ERROR_INTERNAL_ERROR;
193 WINPR_ASSERT(instance);
194 WINPR_ASSERT(instance->context);
196 rdp = instance->context->rdp;
198 WINPR_ASSERT(rdp->settings);
207 up->pcap_rfx = pcap_open(
214 pointer_cache_register_callbacks(instance->context->update);
215 status = IFCALLRESULT(TRUE, instance->PostConnect, instance);
216 instance->ConnectionCallbackState = CLIENT_STATE_POSTCONNECT_PASSED;
219 status2 = freerdp_channels_post_connect(instance->context->channels, instance);
223 status2 = CHANNEL_RC_OK;
224 if (freerdp_get_last_error(instance->context) == FREERDP_ERROR_CONNECT_TRANSPORT_FAILED)
225 status = freerdp_reconnect(instance);
227 goto freerdp_connect_finally;
230 if (!status || (status2 != CHANNEL_RC_OK) || !update_post_connect(instance->context->update))
232 rdpContext* context = instance->context;
233 WINPR_ASSERT(context);
234 WLog_Print(context->log, WLOG_ERROR,
"freerdp_post_connect failed");
236 freerdp_set_last_error_if_not(context, FREERDP_ERROR_POST_CONNECT_FAILED);
239 goto freerdp_connect_finally;
246 pcap_record record = WINPR_C_ARRAY_INIT;
248 WINPR_ASSERT(update);
249 const char* PlayRemoteFxFile =
251 update->pcap_rfx = pcap_open(PlayRemoteFxFile, FALSE);
254 if (!update->pcap_rfx)
255 goto freerdp_connect_finally;
257 update->play_rfx = TRUE;
261 while (pcap_has_next_record(update->pcap_rfx) && status)
263 if (!pcap_get_next_record_header(update->pcap_rfx, &record))
266 s = transport_take_from_pool(rdp->transport, record.length);
270 record.data = Stream_Buffer(s);
271 if (!pcap_get_next_record_content(update->pcap_rfx, &record))
273 if (!Stream_SetLength(s, record.length))
278 Stream_ResetPosition(s);
280 if (!update_begin_paint(&update->common))
284 if (!update_recv_surfcmds(&update->common, s))
287 if (!update_end_paint(&update->common))
294 pcap_close(update->pcap_rfx);
295 update->pcap_rfx =
nullptr;
296 goto freerdp_connect_finally;
299 if (rdp->errorInfo == ERRINFO_SERVER_INSUFFICIENT_PRIVILEGES)
300 freerdp_set_last_error_log(instance->context, FREERDP_ERROR_INSUFFICIENT_PRIVILEGES);
303 status = transport_set_connected_event(rdp->transport);
305freerdp_connect_finally:
306 EventArgsInit(&e,
"freerdp");
307 e.result = status ? 0 : -1;
308 if (PubSub_OnConnectionResult(rdp->pubSub, instance->context, &e) < 0)
312 freerdp_disconnect(instance);
317#if !defined(WITHOUT_FREERDP_3x_DEPRECATED)
318BOOL freerdp_abort_connect(freerdp* instance)
323 return freerdp_abort_connect_context(instance->context);
327BOOL freerdp_abort_connect_context(rdpContext* context)
332 freerdp_set_last_error_if_not(context, FREERDP_ERROR_CONNECT_CANCELLED);
333 return utils_abort_connect(context->rdp);
336#if defined(WITH_FREERDP_DEPRECATED)
337BOOL freerdp_get_fds(freerdp* instance,
void** rfds,
int* rcount, WINPR_ATTR_UNUSED
void** wfds,
338 WINPR_ATTR_UNUSED
int* wcount)
340 rdpRdp* rdp =
nullptr;
342 WINPR_ASSERT(instance);
343 WINPR_ASSERT(instance->context);
345 rdp = instance->context->rdp;
348 transport_get_fds(rdp->transport, rfds, rcount);
353BOOL freerdp_check_fds(freerdp* instance)
356 rdpRdp* rdp =
nullptr;
361 if (!instance->context)
364 if (!instance->context->rdp)
367 rdp = instance->context->rdp;
368 status = rdp_check_fds(rdp);
372 TerminateEventArgs e;
373 rdpContext* context = instance->context;
374 WINPR_ASSERT(context);
376 WLog_Print(context->log, WLOG_DEBUG,
"rdp_check_fds() - %i", status);
377 EventArgsInit(&e,
"freerdp");
379 if (PubSub_OnTerminate(rdp->pubSub, context, &e) < 0)
387DWORD freerdp_get_event_handles(rdpContext* context, HANDLE* events, DWORD count)
391 WINPR_ASSERT(context);
392 WINPR_ASSERT(context->rdp);
393 WINPR_ASSERT(events || (count == 0));
395 const size_t rrc = rdp_get_event_handles(context->rdp, &events[nCount], count - nCount);
399 nCount += WINPR_ASSERTING_INT_CAST(uint32_t, rrc);
401 if (events && (nCount < count + 2))
403 events[nCount++] = freerdp_channels_get_event_handle(context->instance);
404 events[nCount++] = getChannelErrorEventHandle(context);
409 const SSIZE_T rc = freerdp_client_channel_get_registered_event_handles(
410 context->channels, &events[nCount], count - nCount);
413 return nCount + (DWORD)rc;
417static BOOL freerdp_prevent_session_lock(rdpContext* context)
419 WINPR_ASSERT(context);
420 WINPR_ASSERT(context->input);
424 UINT32 FakeMouseMotionInterval =
426 if (FakeMouseMotionInterval && in->lastInputTimestamp)
428 const time_t now = time(
nullptr);
429 if (WINPR_ASSERTING_INT_CAST(
size_t, now) - in->lastInputTimestamp >
430 FakeMouseMotionInterval)
432 WLog_Print(context->log, WLOG_DEBUG,
433 "fake mouse move: x=%d y=%d lastInputTimestamp=%" PRIu64
" "
434 "FakeMouseMotionInterval=%" PRIu32,
435 in->lastX, in->lastY, in->lastInputTimestamp, FakeMouseMotionInterval);
437 BOOL status = freerdp_input_send_mouse_event(context->input, PTR_FLAGS_MOVE, in->lastX,
441 if (freerdp_get_last_error(context) == FREERDP_ERROR_SUCCESS)
442 WLog_Print(context->log, WLOG_ERROR,
443 "freerdp_prevent_session_lock() failed - %" PRIi32
"", status);
455BOOL freerdp_check_event_handles(rdpContext* context)
457 WINPR_ASSERT(context);
459 BOOL status = freerdp_check_fds(context->instance);
463 if (freerdp_get_last_error(context) == FREERDP_ERROR_SUCCESS)
464 WLog_Print(context->log, WLOG_ERROR,
"freerdp_check_fds() failed - %" PRIi32
"",
470 status = freerdp_channels_check_fds(context->channels, context->instance);
474 if (freerdp_get_last_error(context) == FREERDP_ERROR_SUCCESS)
475 WLog_Print(context->log, WLOG_ERROR,
476 "freerdp_channels_check_fds() failed - %" PRIi32
"", status);
481 status = checkChannelErrorEvent(context);
485 if (freerdp_get_last_error(context) == FREERDP_ERROR_SUCCESS)
486 WLog_Print(context->log, WLOG_ERROR,
"checkChannelErrorEvent() failed - %" PRIi32
"",
492 status = freerdp_prevent_session_lock(context);
497wMessageQueue* freerdp_get_message_queue(freerdp* instance, DWORD
id)
499 wMessageQueue* queue =
nullptr;
501 WINPR_ASSERT(instance);
503 rdpContext* context = instance->context;
504 WINPR_ASSERT(context);
508 case FREERDP_UPDATE_MESSAGE_QUEUE:
511 queue = update->queue;
515 case FREERDP_INPUT_MESSAGE_QUEUE:
518 queue = input->queue;
528HANDLE freerdp_get_message_queue_event_handle(freerdp* instance, DWORD
id)
530 HANDLE
event =
nullptr;
531 wMessageQueue* queue = freerdp_get_message_queue(instance,
id);
534 event = MessageQueue_Event(queue);
539int freerdp_message_queue_process_message(freerdp* instance, DWORD
id, wMessage* message)
542 rdpContext* context =
nullptr;
544 WINPR_ASSERT(instance);
546 context = instance->context;
547 WINPR_ASSERT(context);
551 case FREERDP_UPDATE_MESSAGE_QUEUE:
552 status = update_message_queue_process_message(context->update, message);
555 case FREERDP_INPUT_MESSAGE_QUEUE:
556 status = input_message_queue_process_message(context->input, message);
565int freerdp_message_queue_process_pending_messages(freerdp* instance, DWORD
id)
568 rdpContext* context =
nullptr;
570 WINPR_ASSERT(instance);
572 context = instance->context;
573 WINPR_ASSERT(context);
577 case FREERDP_UPDATE_MESSAGE_QUEUE:
578 status = update_message_queue_process_pending_messages(context->update);
581 case FREERDP_INPUT_MESSAGE_QUEUE:
582 status = input_message_queue_process_pending_messages(context->input);
591static BOOL freerdp_send_channel_data(freerdp* instance, UINT16 channelId,
const BYTE* data,
594 WINPR_ASSERT(instance);
595 WINPR_ASSERT(instance->context);
596 WINPR_ASSERT(instance->context->rdp);
597 return rdp_send_channel_data(instance->context->rdp, channelId, data, size);
600static BOOL freerdp_send_channel_packet(freerdp* instance, UINT16 channelId,
size_t totalSize,
601 UINT32 flags,
const BYTE* data,
size_t chunkSize)
603 WINPR_ASSERT(instance);
604 WINPR_ASSERT(instance->context);
605 WINPR_ASSERT(instance->context->rdp);
606 return rdp_channel_send_packet(instance->context->rdp, channelId, totalSize, flags, data,
610BOOL freerdp_disconnect(freerdp* instance)
614 if (!instance || !instance->context)
617 rdpRdp* rdp = instance->context->rdp;
622 if (freerdp_get_last_error(instance->context) == FREERDP_ERROR_CONNECT_CANCELLED)
624 (void)mcs_send_disconnect_provider_ultimatum(rdp->mcs,
625 Disconnect_Ultimatum_user_requested);
629 utils_abort_connect(rdp);
631 if (!rdp_client_disconnect(rdp))
635 if (rdp && rdp->update)
637 up = update_cast(rdp->update);
639 update_post_disconnect(rdp->update);
642 IFCALL(instance->PostDisconnect, instance);
646 update_dump_stats(rdp->update);
649 up->dump_rfx = FALSE;
650 pcap_close(up->pcap_rfx);
651 up->pcap_rfx =
nullptr;
655 if (instance->context->channels)
656 freerdp_channels_close(instance->context->channels, instance);
658 IFCALL(instance->PostFinalDisconnect, instance);
660 freerdp_del_signal_cleanup_handler(instance->context, sig_abort_connect);
664#if !defined(WITHOUT_FREERDP_3x_DEPRECATED)
665BOOL freerdp_disconnect_before_reconnect(freerdp* instance)
667 WINPR_ASSERT(instance);
668 return freerdp_disconnect_before_reconnect_context(instance->context);
672BOOL freerdp_disconnect_before_reconnect_context(rdpContext* context)
674 rdpRdp* rdp =
nullptr;
676 WINPR_ASSERT(context);
679 return rdp_client_disconnect_and_clear(rdp);
682BOOL freerdp_reconnect(freerdp* instance)
684 rdpRdp* rdp =
nullptr;
686 WINPR_ASSERT(instance);
687 WINPR_ASSERT(instance->context);
689 if (freerdp_get_last_error(instance->context) == FREERDP_ERROR_CONNECT_CANCELLED)
692 rdp = instance->context->rdp;
694 if (!utils_reset_abort(instance->context->rdp))
696 return rdp_client_reconnect(rdp);
699#if !defined(WITHOUT_FREERDP_3x_DEPRECATED)
700BOOL freerdp_shall_disconnect(
const freerdp* instance)
705 return freerdp_shall_disconnect_context(instance->context);
709BOOL freerdp_shall_disconnect_context(
const rdpContext* context)
714 return utils_abort_event_is_set(context->rdp);
717BOOL freerdp_focus_required(freerdp* instance)
719 rdpRdp* rdp =
nullptr;
720 BOOL bRetCode = FALSE;
722 WINPR_ASSERT(instance);
723 WINPR_ASSERT(instance->context);
725 rdp = instance->context->rdp;
728 if (rdp->resendFocus)
731 rdp->resendFocus = FALSE;
737void freerdp_set_focus(freerdp* instance)
739 rdpRdp* rdp =
nullptr;
741 WINPR_ASSERT(instance);
742 WINPR_ASSERT(instance->context);
744 rdp = instance->context->rdp;
747 rdp->resendFocus = TRUE;
750void freerdp_get_version(
int* major,
int* minor,
int* revision)
752 if (major !=
nullptr)
753 *major = FREERDP_VERSION_MAJOR;
755 if (minor !=
nullptr)
756 *minor = FREERDP_VERSION_MINOR;
758 if (revision !=
nullptr)
759 *revision = FREERDP_VERSION_REVISION;
762const char* freerdp_get_version_string(
void)
764 return FREERDP_VERSION_FULL;
767const char* freerdp_get_build_config(
void)
769 WINPR_PRAGMA_DIAG_PUSH
770 WINPR_PRAGMA_DIAG_IGNORED_OVERLENGTH_STRINGS
771 static const char build_config[] =
772 "Build configuration: " FREERDP_BUILD_CONFIG
"\n"
773 "Build type: " FREERDP_BUILD_TYPE
"\n"
774 "CFLAGS: " FREERDP_CFLAGS
"\n"
775 "Compiler: " FREERDP_COMPILER_ID
", " FREERDP_COMPILER_VERSION
"\n"
776 "Target architecture: " FREERDP_TARGET_ARCH
"\n";
777 WINPR_PRAGMA_DIAG_POP
781const char* freerdp_get_build_revision(
void)
783 return FREERDP_GIT_REVISION;
787 DEFINE_EVENT_ENTRY(WindowStateChange), DEFINE_EVENT_ENTRY(ResizeWindow),
788 DEFINE_EVENT_ENTRY(LocalResizeWindow), DEFINE_EVENT_ENTRY(EmbedWindow),
789 DEFINE_EVENT_ENTRY(PanningChange), DEFINE_EVENT_ENTRY(ZoomingChange),
790 DEFINE_EVENT_ENTRY(ErrorInfo), DEFINE_EVENT_ENTRY(Terminate),
791 DEFINE_EVENT_ENTRY(ConnectionResult), DEFINE_EVENT_ENTRY(ChannelConnected),
792 DEFINE_EVENT_ENTRY(ChannelDisconnected), DEFINE_EVENT_ENTRY(MouseEvent),
793 DEFINE_EVENT_ENTRY(Activated), DEFINE_EVENT_ENTRY(Timer),
794 DEFINE_EVENT_ENTRY(GraphicsReset), DEFINE_EVENT_ENTRY(UserNotification),
795 DEFINE_EVENT_ENTRY(ChannelInitialized), DEFINE_EVENT_ENTRY(ChannelTerminated),
796 DEFINE_EVENT_ENTRY(StateChanged)
809BOOL freerdp_context_new(freerdp* instance)
811 return freerdp_context_new_ex(instance,
nullptr);
814static BOOL freerdp_common_context(rdpContext* context, AccessTokenType tokenType,
char** token,
819 WINPR_ASSERT(context);
820 if (!context->instance || !context->instance->GetAccessToken)
823 va_list ap = WINPR_C_ARRAY_INIT;
827 case ACCESS_TOKEN_TYPE_AAD:
831 "ACCESS_TOKEN_TYPE_AAD expected 2 additional arguments, but got %" PRIuz
837 const char* scope = va_arg(ap,
const char*);
838 const char* req_cnf = va_arg(ap,
const char*);
839 rc = context->instance->GetAccessToken(context->instance, tokenType, token, count,
843 case ACCESS_TOKEN_TYPE_AVD:
847 "ACCESS_TOKEN_TYPE_AVD expected 0 additional arguments, but got %" PRIuz
853 rc = context->instance->GetAccessToken(context->instance, tokenType, token, count);
862 freerdp_set_last_error_if_not(context, FREERDP_ERROR_CONNECT_ACCESS_DENIED);
867BOOL freerdp_context_new_ex(freerdp* instance, rdpSettings* settings)
869 rdpRdp* rdp =
nullptr;
870 rdpContext* context =
nullptr;
873 WINPR_ASSERT(instance);
875 instance->context = context = (rdpContext*)calloc(1, instance->ContextSize);
880 context->log = WLog_Get(TAG);
885 context->settings = settings;
886 context->instance = instance;
887 context->ServerMode = FALSE;
888 context->disconnectUltimatum = 0;
890 context->metrics = metrics_new(context);
892 if (!context->metrics)
895 rdp = rdp_new(context);
901 context->pubSub = rdp->pubSub;
903 if (!context->pubSub)
906 PubSub_AddEventTypes(rdp->pubSub, FreeRDP_Events, ARRAYSIZE(FreeRDP_Events));
908#if defined(WITH_FREERDP_DEPRECATED)
909 instance->input = rdp->input;
910 instance->update = rdp->update;
911 instance->settings = rdp->settings;
912 instance->autodetect = rdp->autodetect;
915 instance->heartbeat = rdp->heartbeat;
916 context->graphics = graphics_new(context);
918 if (!context->graphics)
921 context->input = rdp->input;
922 context->update = rdp->update;
923 context->settings = rdp->settings;
924 context->autodetect = rdp->autodetect;
926 if (!(context->errorDescription = calloc(1, 500)))
928 WLog_Print(context->log, WLOG_ERROR,
"calloc failed!");
932 if (!(context->channelErrorEvent = CreateEvent(
nullptr, TRUE, FALSE,
nullptr)))
934 WLog_Print(context->log, WLOG_ERROR,
"CreateEvent failed!");
938 update_register_client_callbacks(rdp->update);
940 if (!(context->channels = freerdp_channels_new(instance)))
943 context->dump = stream_dump_new();
951 if (!freerdp_set_common_access_token(context, freerdp_common_context))
954 IFCALLRET(instance->ContextNew, ret, instance, context);
962 freerdp_context_free(instance);
966BOOL freerdp_context_reset(freerdp* instance)
971 WINPR_ASSERT(instance->context);
972 rdpRdp* rdp = instance->context->rdp;
974 return rdp_reset_runtime_settings(rdp);
985void freerdp_context_free(freerdp* instance)
987 rdpContext* ctx =
nullptr;
992 if (!instance->context)
995 ctx = instance->context;
997 IFCALL(instance->ContextFree, instance, ctx);
1000 ctx->settings =
nullptr;
1002 graphics_free(ctx->graphics);
1003 ctx->graphics =
nullptr;
1005 metrics_free(ctx->metrics);
1006 ctx->metrics =
nullptr;
1008 if (ctx->channelErrorEvent)
1009 (void)CloseHandle(ctx->channelErrorEvent);
1010 ctx->channelErrorEvent =
nullptr;
1012 free(ctx->errorDescription);
1013 ctx->errorDescription =
nullptr;
1015 freerdp_channels_free(ctx->channels);
1016 ctx->channels =
nullptr;
1018 freerdp_client_codecs_free(ctx->codecs);
1019 ctx->codecs =
nullptr;
1021 stream_dump_free(ctx->dump);
1022 ctx->dump =
nullptr;
1024 ctx->input =
nullptr;
1025 ctx->update =
nullptr;
1026 ctx->settings =
nullptr;
1027 ctx->autodetect =
nullptr;
1030 instance->context =
nullptr;
1031#if defined(WITH_FREERDP_DEPRECATED)
1032 instance->input =
nullptr;
1033 instance->update =
nullptr;
1034 instance->settings =
nullptr;
1035 instance->autodetect =
nullptr;
1037 instance->heartbeat =
nullptr;
1040int freerdp_get_disconnect_ultimatum(
const rdpContext* context)
1042 WINPR_ASSERT(context);
1043 return context->disconnectUltimatum;
1046UINT32 freerdp_error_info(
const freerdp* instance)
1048 WINPR_ASSERT(instance);
1049 WINPR_ASSERT(instance->context);
1050 WINPR_ASSERT(instance->context->rdp);
1051 return instance->context->rdp->errorInfo;
1054void freerdp_set_error_info(rdpRdp* rdp, UINT32 error)
1059 rdp_set_error_info(rdp, error);
1062BOOL freerdp_send_error_info(rdpRdp* rdp)
1067 return rdp_send_error_info(rdp);
1070UINT32 freerdp_get_last_error(
const rdpContext* context)
1072 WINPR_ASSERT(context);
1073 return context->LastError;
1076const char* freerdp_get_last_error_name(UINT32 code)
1078 const char* name =
nullptr;
1079 const UINT32 cls = GET_FREERDP_ERROR_CLASS(code);
1080 const UINT32 type = GET_FREERDP_ERROR_TYPE(code);
1084 case FREERDP_ERROR_ERRBASE_CLASS:
1085 name = freerdp_get_error_base_name(type);
1088 case FREERDP_ERROR_ERRINFO_CLASS:
1089 name = freerdp_get_error_info_name(type);
1092 case FREERDP_ERROR_CONNECT_CLASS:
1093 name = freerdp_get_error_connect_name(type);
1097 name = rpc_error_to_string(code);
1104const char* freerdp_get_last_error_string(UINT32 code)
1106 const char*
string =
nullptr;
1107 const UINT32 cls = GET_FREERDP_ERROR_CLASS(code);
1108 const UINT32 type = GET_FREERDP_ERROR_TYPE(code);
1112 case FREERDP_ERROR_ERRBASE_CLASS:
1113 string = freerdp_get_error_base_string(type);
1116 case FREERDP_ERROR_ERRINFO_CLASS:
1117 string = freerdp_get_error_info_string(type);
1120 case FREERDP_ERROR_CONNECT_CLASS:
1121 string = freerdp_get_error_connect_string(type);
1125 string = rpc_error_to_string(code);
1132const char* freerdp_get_last_error_category(UINT32 code)
1134 const char*
string =
nullptr;
1135 const UINT32 cls = GET_FREERDP_ERROR_CLASS(code);
1136 const UINT32 type = GET_FREERDP_ERROR_TYPE(code);
1140 case FREERDP_ERROR_ERRBASE_CLASS:
1141 string = freerdp_get_error_base_category(type);
1144 case FREERDP_ERROR_ERRINFO_CLASS:
1145 string = freerdp_get_error_info_category(type);
1148 case FREERDP_ERROR_CONNECT_CLASS:
1149 string = freerdp_get_error_connect_category(type);
1153 string = rpc_error_to_category(code);
1160void freerdp_set_last_error_ex(rdpContext* context, UINT32 lastError,
const char* fkt,
1161 const char* file,
int line)
1163 WINPR_ASSERT(context);
1164 WINPR_ASSERT(line >= 0);
1168 if (WLog_IsLevelActive(context->log, WLOG_ERROR))
1170 WLog_PrintTextMessage(context->log, WLOG_ERROR, (
size_t)line, file, fkt,
1171 "%s [0x%08" PRIX32
"]", freerdp_get_last_error_name(lastError),
1176 if (lastError == FREERDP_ERROR_SUCCESS)
1178 if (WLog_IsLevelActive(context->log, WLOG_DEBUG))
1179 WLog_PrintTextMessage(context->log, WLOG_DEBUG, (
size_t)line, file, fkt,
1180 "resetting error state");
1182 else if (context->LastError != FREERDP_ERROR_SUCCESS)
1184 if (WLog_IsLevelActive(context->log, WLOG_ERROR))
1186 WLog_PrintTextMessage(context->log, WLOG_ERROR, (
size_t)line, file, fkt,
1187 "TODO: Trying to set error code %s, but %s already set!",
1188 freerdp_get_last_error_name(lastError),
1189 freerdp_get_last_error_name(context->LastError));
1192 context->LastError = lastError;
1195const char* freerdp_get_logon_error_info_type_ex(UINT32 type,
char* buffer,
size_t size)
1197 const char* str = freerdp_get_logon_error_info_type(type);
1198 (void)_snprintf(buffer, size,
"%s(0x%04" PRIx32
")", str, type);
1202const char* freerdp_get_logon_error_info_type(UINT32 type)
1204#define CASE_ENTRY(x) \
1209 CASE_ENTRY(LOGON_MSG_SESSION_BUSY_OPTIONS);
1210 CASE_ENTRY(LOGON_MSG_DISCONNECT_REFUSED);
1211 CASE_ENTRY(LOGON_MSG_NO_PERMISSION);
1212 CASE_ENTRY(LOGON_MSG_BUMP_OPTIONS);
1213 CASE_ENTRY(LOGON_MSG_RECONNECT_OPTIONS);
1214 CASE_ENTRY(LOGON_MSG_SESSION_TERMINATE);
1215 CASE_ENTRY(LOGON_MSG_SESSION_CONTINUE);
1216 CASE_ENTRY(ERROR_CODE_ACCESS_DENIED);
1224const char* freerdp_get_logon_error_info_data(UINT32 data)
1228 case LOGON_FAILED_BAD_PASSWORD:
1229 return "LOGON_FAILED_BAD_PASSWORD";
1231 case LOGON_FAILED_UPDATE_PASSWORD:
1232 return "LOGON_FAILED_UPDATE_PASSWORD";
1234 case LOGON_FAILED_OTHER:
1235 return "LOGON_FAILED_OTHER";
1238 return "LOGON_WARNING";
1241 return "SESSION_ID";
1245const char* freerdp_get_logon_error_info_data_ex(UINT32 data,
char* buffer,
size_t size)
1247 const char* str = freerdp_get_logon_error_info_data(data);
1248 (void)_snprintf(buffer, size,
"%s(0x%04" PRIx32
")", str, data);
1255freerdp* freerdp_new(
void)
1257 freerdp* instance =
nullptr;
1258 instance = (freerdp*)calloc(1,
sizeof(freerdp));
1263 instance->ContextSize =
sizeof(rdpContext);
1264 instance->SendChannelData = freerdp_send_channel_data;
1265 instance->SendChannelPacket = freerdp_send_channel_packet;
1266 instance->ReceiveChannelData = freerdp_channels_data;
1274void freerdp_free(freerdp* instance)
1279ULONG freerdp_get_transport_sent(
const rdpContext* context, BOOL resetCount)
1281 WINPR_ASSERT(context);
1282 WINPR_ASSERT(context->rdp);
1283 UINT64 rc = transport_get_bytes_sent(context->rdp->transport, resetCount);
1284 return WINPR_CXX_COMPAT_CAST(ULONG, MIN(rc, UINT32_MAX));
1287BOOL freerdp_nla_impersonate(rdpContext* context)
1289 rdpNla* nla =
nullptr;
1297 if (!context->rdp->transport)
1300 nla = transport_get_nla(context->rdp->transport);
1301 return nla_impersonate(nla);
1304BOOL freerdp_nla_revert_to_self(rdpContext* context)
1306 rdpNla* nla =
nullptr;
1314 if (!context->rdp->transport)
1317 nla = transport_get_nla(context->rdp->transport);
1318 return nla_revert_to_self(nla);
1321UINT32 freerdp_get_nla_sspi_error(
const rdpContext* context)
1323 WINPR_ASSERT(context);
1324 WINPR_ASSERT(context->rdp);
1325 WINPR_ASSERT(context->rdp->transport);
1327 rdpNla* nla = context->rdp->nla;
1329 nla = transport_get_nla(context->rdp->transport);
1330 return (UINT32)nla_get_sspi_error(nla);
1333BOOL freerdp_nla_encrypt(rdpContext* context,
const SecBuffer* inBuffer,
SecBuffer* outBuffer)
1335 WINPR_ASSERT(context);
1336 WINPR_ASSERT(context->rdp);
1338 rdpNla* nla = context->rdp->nla;
1339 return nla_encrypt(nla, inBuffer, outBuffer);
1342BOOL freerdp_nla_decrypt(rdpContext* context,
const SecBuffer* inBuffer,
SecBuffer* outBuffer)
1344 WINPR_ASSERT(context);
1345 WINPR_ASSERT(context->rdp);
1347 rdpNla* nla = context->rdp->nla;
1348 return nla_decrypt(nla, inBuffer, outBuffer);
1351SECURITY_STATUS freerdp_nla_QueryContextAttributes(rdpContext* context, DWORD ulAttr, PVOID pBuffer)
1353 WINPR_ASSERT(context);
1354 WINPR_ASSERT(context->rdp);
1356 rdpNla* nla = context->rdp->nla;
1358 nla = transport_get_nla(context->rdp->transport);
1362 return nla_QueryContextAttributes(nla, ulAttr, pBuffer);
1365SECURITY_STATUS freerdp_nla_FreeContextBuffer(rdpContext* context, PVOID pBuffer)
1367 WINPR_ASSERT(context);
1368 WINPR_ASSERT(context->rdp);
1370 rdpNla* nla = context->rdp->nla;
1372 nla = transport_get_nla(context->rdp->transport);
1376 return nla_FreeContextBuffer(nla, pBuffer);
1379HANDLE getChannelErrorEventHandle(rdpContext* context)
1381 WINPR_ASSERT(context);
1382 return context->channelErrorEvent;
1385BOOL checkChannelErrorEvent(rdpContext* context)
1387 WINPR_ASSERT(context);
1389 if (WaitForSingleObject(context->channelErrorEvent, 0) == WAIT_OBJECT_0)
1391 WLog_Print(context->log, WLOG_ERROR,
"%s. Error was %" PRIu32
"", context->errorDescription,
1392 context->channelErrorNum);
1404UINT getChannelError(
const rdpContext* context)
1406 WINPR_ASSERT(context);
1407 return context->channelErrorNum;
1410const char* getChannelErrorDescription(
const rdpContext* context)
1412 WINPR_ASSERT(context);
1413 return context->errorDescription;
1416void clearChannelError(rdpContext* context)
1418 WINPR_ASSERT(context);
1419 context->channelErrorNum = 0;
1420 memset(context->errorDescription, 0, 500);
1421 (void)ResetEvent(context->channelErrorEvent);
1424WINPR_ATTR_FORMAT_ARG(3, 4)
1425void setChannelError(rdpContext* context, UINT errorNum, WINPR_FORMAT_ARG const
char* format, ...)
1427 va_list ap = WINPR_C_ARRAY_INIT;
1428 va_start(ap, format);
1430 WINPR_ASSERT(context);
1432 context->channelErrorNum = errorNum;
1433 (void)vsnprintf(context->errorDescription, 499, format, ap);
1435 (void)SetEvent(context->channelErrorEvent);
1438const char* freerdp_nego_get_routing_token(
const rdpContext* context, DWORD* length)
1440 if (!context || !context->rdp)
1443 return (
const char*)nego_get_routing_token(context->rdp->nego, length);
1446BOOL freerdp_io_callback_set_event(rdpContext* context, BOOL set)
1448 WINPR_ASSERT(context);
1449 return rdp_io_callback_set_event(context->rdp, set);
1452const rdpTransportIo* freerdp_get_io_callbacks(rdpContext* context)
1454 WINPR_ASSERT(context);
1455 return rdp_get_io_callbacks(context->rdp);
1458BOOL freerdp_set_io_callbacks(rdpContext* context,
const rdpTransportIo* io_callbacks)
1460 WINPR_ASSERT(context);
1461 return rdp_set_io_callbacks(context->rdp, io_callbacks);
1464BOOL freerdp_set_io_callback_context(rdpContext* context,
void* usercontext)
1466 WINPR_ASSERT(context);
1467 return rdp_set_io_callback_context(context->rdp, usercontext);
1470void* freerdp_get_io_callback_context(rdpContext* context)
1472 WINPR_ASSERT(context);
1473 return rdp_get_io_callback_context(context->rdp);
1476CONNECTION_STATE freerdp_get_state(
const rdpContext* context)
1478 WINPR_ASSERT(context);
1479 return rdp_get_state(context->rdp);
1482const char* freerdp_state_string(CONNECTION_STATE state)
1484 return rdp_state_string(state);
1487BOOL freerdp_is_active_state(
const rdpContext* context)
1489 WINPR_ASSERT(context);
1490 return rdp_is_active_state(context->rdp);
1493BOOL freerdp_channels_from_mcs(rdpSettings* settings,
const rdpContext* context)
1495 WINPR_ASSERT(context);
1496 return rdp_channels_from_mcs(settings, context->rdp);
1499HANDLE freerdp_abort_event(rdpContext* context)
1501 WINPR_ASSERT(context);
1502 return utils_get_abort_event(context->rdp);
1505static void test_mcs_free(rdpMcs* mcs)
1512 rdpSettings* settings = mcs->context->settings;
1520static rdpMcs* test_mcs_new(
void)
1523 rdpContext* context = calloc(1,
sizeof(rdpContext));
1532 context->settings = settings;
1533 return mcs_new(context);
1542BOOL freerdp_is_valid_mcs_create_request(
const BYTE* data,
size_t size)
1545 wStream sbuffer = WINPR_C_ARRAY_INIT;
1546 wStream* s = Stream_StaticConstInit(&sbuffer, data, size);
1548 WINPR_ASSERT(data || (size == 0));
1551 rdpMcs* mcs = test_mcs_new();
1554 BOOL result = mcs_recv_connect_initial(mcs, s);
1559BOOL freerdp_is_valid_mcs_create_response(
const BYTE* data,
size_t size)
1562 wStream sbuffer = WINPR_C_ARRAY_INIT;
1563 wStream* s = Stream_StaticConstInit(&sbuffer, data, size);
1565 WINPR_ASSERT(data || (size == 0));
1568 rdpMcs* mcs = test_mcs_new();
1571 BOOL result = mcs_recv_connect_response(mcs, s);
1576BOOL freerdp_persist_credentials(rdpContext* context)
1580 WINPR_ASSERT(context->rdp);
1581 return utils_persist_credentials(context->rdp->originalSettings, context->rdp->settings);
1584const char* freerdp_disconnect_reason_string(
int reason)
1588 case Disconnect_Ultimatum_domain_disconnected:
1589 return "rn-domain-disconnected";
1590 case Disconnect_Ultimatum_provider_initiated:
1591 return "rn-provider-initiated";
1592 case Disconnect_Ultimatum_token_purged:
1593 return "rn-token-purged";
1594 case Disconnect_Ultimatum_user_requested:
1595 return "rn-user-requested";
1596 case Disconnect_Ultimatum_channel_purged:
1597 return "rn-channel-purged";
1599 return "rn-unknown";
1603BOOL freerdp_set_common_access_token(rdpContext* context,
1604 pGetCommonAccessToken GetCommonAccessToken)
1606 WINPR_ASSERT(context);
1607 WINPR_ASSERT(context->rdp);
1608 context->rdp->GetCommonAccessToken = GetCommonAccessToken;
1612pGetCommonAccessToken freerdp_get_common_access_token(
const rdpContext* context)
1614 WINPR_ASSERT(context);
1615 WINPR_ASSERT(context->rdp);
1616 return context->rdp->GetCommonAccessToken;
FREERDP_API rdpSettings * freerdp_settings_new(DWORD flags)
creates a new setting struct
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_uint32(rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id, UINT32 val)
Sets a UINT32 settings value.
FREERDP_API void freerdp_settings_free(rdpSettings *settings)
Free a settings struct with all data in it.
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_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.