24#include <freerdp/config.h>
26#if defined(__has_include)
27#if __has_include(<version>)
36#if defined(__cpp_lib_source_location) && (__cpp_lib_source_location >= 201907L)
37#include <source_location>
40#include <freerdp/constants.h>
41#include <freerdp/freerdp.h>
42#include <freerdp/gdi/gdi.h>
43#include <freerdp/streamdump.h>
44#include <freerdp/utils/signal.h>
46#include <freerdp/channels/channels.h>
47#include <freerdp/client/channels.h>
48#include <freerdp/client/cliprdr.h>
49#include <freerdp/client/cmdline.h>
50#include <freerdp/client/file.h>
52#include <freerdp/log.h>
53#include <winpr/assert.h>
54#include <winpr/config.h>
56#include <winpr/synch.h>
59#if !defined(__MINGW32__)
60#include <SDL3/SDL_main.h>
62#include <SDL3/SDL_hints.h>
63#include <SDL3/SDL_oldnames.h>
64#include <SDL3/SDL_video.h>
66#include <sdl_config.hpp>
68#include "dialogs/sdl_connection_dialog_hider.hpp"
69#include "dialogs/sdl_dialogs.hpp"
70#include "scoped_guard.hpp"
71#include "sdl_channels.hpp"
72#include "sdl_freerdp.hpp"
73#include "sdl_context.hpp"
74#include "sdl_monitor.hpp"
75#include "sdl_pointer.hpp"
76#include "sdl_prefs.hpp"
78#include "sdl_win32_console.hpp"
80#include "sdl_utils.hpp"
83#define SDL_TAG CLIENT_TAG("SDL")
86class ErrorMsg :
public std::exception
89 ErrorMsg(
int rc, Uint32 type,
const std::string& msg,
90 const std::string& sdlError = SDL_GetError()
91#
if defined(__cpp_lib_source_location) && (__cpp_lib_source_location >= 201907L)
93 std::source_location loc = std::source_location::current()
94#elif defined(__GNUC__)
96 const std::string& file = __builtin_FILE(),
97 const std::string& fkt = __builtin_FUNCTION(),
size_t line = __builtin_LINE(),
99 size_t column = __builtin_COLUMN()
106 [[nodiscard]]
int rc()
const;
107 [[nodiscard]]
const char* what() const noexcept override;
113 std::
string _sdlError;
121const
char* ErrorMsg::what() const noexcept
123 return _buffer.c_str();
126int ErrorMsg::rc()
const
131ErrorMsg::ErrorMsg(
int rc, Uint32 type,
const std::string& msg,
const std::string& sdlError
132#
if defined(__cpp_lib_source_location) && (__cpp_lib_source_location >= 201907L)
134 std::source_location loc
135#elif defined(__GNUC__)
137 const std::string& file,
const std::string& fkt,
size_t line,
size_t column
140 : _rc(rc), _type(type), _msg(msg), _sdlError(sdlError)
141#if defined(__cpp_lib_source_location) && (__cpp_lib_source_location >= 201907L)
143 _file(loc.file_name()), _fkt(loc.function_name()), _line(loc.line()), _column(loc.column())
144#elif defined(__GNUC__)
146 _file(file), _fkt(fkt), _line(line), _column(column)
149 std::stringstream ss;
150 ss << _msg <<
" {" << sdl::utils::toString(_type) <<
"}{SDL:" << sdlError <<
"}";
151 ss <<
"{" << _fkt <<
" [" << _file <<
":" << _line <<
"-" << _column <<
"]}";
155static void sdl_term_handler([[maybe_unused]]
int signum, [[maybe_unused]]
const char* signame,
156 [[maybe_unused]]
void* context)
158 std::ignore = sdl_push_quit();
168 while (!
sdl->shallAbort())
170 SDL_Event windowEvent = {};
171 while (!
sdl->shallAbort() && SDL_WaitEventTimeout(
nullptr, 1000))
176 const int prc = SDL_PeepEvents(&windowEvent, 1, SDL_GETEVENT, SDL_EVENT_FIRST,
177 SDL_EVENT_USER_RETRY_DIALOG);
180 if (sdl_log_error(prc,
sdl->getWLog(),
"SDL_PeepEvents"))
184 WLog_Print(
sdl->getWLog(), WLOG_TRACE,
"got event %s [0x%08" PRIx32
"]",
185 sdl::utils::toString(windowEvent.type).c_str(), windowEvent.type);
186 if (
sdl->shallAbort(
true))
189 if (
sdl->getDialog().handleEvent(windowEvent))
192 if (!
sdl->handleEvent(windowEvent))
193 throw ErrorMsg{ -1, windowEvent.type,
"sdl->handleEvent" };
195 switch (windowEvent.type)
198 std::ignore = freerdp_abort_connect_context(
sdl->context());
200 case SDL_EVENT_USER_CERT_DIALOG:
203 auto title =
static_cast<const char*
>(windowEvent.user.data1);
204 auto msg =
static_cast<const char*
>(windowEvent.user.data2);
205 if (!sdl_cert_dialog_show(title, msg))
206 throw ErrorMsg{ -1, windowEvent.type,
"sdl_cert_dialog_show" };
209 case SDL_EVENT_USER_SHOW_DIALOG:
212 auto title =
static_cast<const char*
>(windowEvent.user.data1);
213 auto msg =
static_cast<const char*
>(windowEvent.user.data2);
214 if (!sdl_message_dialog_show(title, msg, windowEvent.user.code))
215 throw ErrorMsg{ -1, windowEvent.type,
"sdl_message_dialog_show" };
218 case SDL_EVENT_USER_SCARD_DIALOG:
221 auto title =
static_cast<const char*
>(windowEvent.user.data1);
222 auto msg =
static_cast<const char**
>(windowEvent.user.data2);
223 if (!sdl_scard_dialog_show(title, windowEvent.user.code, msg))
224 throw ErrorMsg{ -1, windowEvent.type,
"sdl_scard_dialog_show" };
227 case SDL_EVENT_USER_AUTH_DIALOG:
230 if (!sdl_auth_dialog_show(
232 throw ErrorMsg{ -1, windowEvent.type,
"sdl_auth_dialog_show" };
235 case SDL_EVENT_USER_UPDATE:
237 std::vector<SDL_Rect> rectangles;
240 rectangles =
sdl->pop();
241 if (!
sdl->drawToWindows(rectangles))
242 throw ErrorMsg{ -1, windowEvent.type,
"sdl->drawToWindows" };
243 }
while (!rectangles.empty());
246 case SDL_EVENT_USER_CREATE_WINDOWS:
248 auto ctx =
static_cast<SdlContext*
>(windowEvent.user.data1);
249 if (!ctx->createWindows())
250 throw ErrorMsg{ -1, windowEvent.type,
"sdl->createWindows" };
253 case SDL_EVENT_USER_WINDOW_RESIZEABLE:
255 auto window =
static_cast<SdlWindow*
>(windowEvent.user.data1);
256 const bool use = windowEvent.user.code != 0;
258 window->resizeable(use);
261 case SDL_EVENT_USER_WINDOW_FULLSCREEN:
263 auto window =
static_cast<SdlWindow*
>(windowEvent.user.data1);
264 const bool enter = windowEvent.user.code != 0;
265 const bool forceOriginalDisplay = windowEvent.user.data2 !=
nullptr;
267 window->fullscreen(enter, forceOriginalDisplay);
270 case SDL_EVENT_USER_WINDOW_MINIMIZE:
271 if (!
sdl->minimizeAllWindows())
272 throw ErrorMsg{ -1, windowEvent.type,
"sdl->minimizeAllWindows" };
274 case SDL_EVENT_USER_POINTER_NULL:
275 if (!
sdl->setCursor(SdlContext::CURSOR_NULL))
276 throw ErrorMsg{ -1, windowEvent.type,
"sdl->setCursor" };
278 case SDL_EVENT_USER_POINTER_DEFAULT:
279 if (!
sdl->setCursor(SdlContext::CURSOR_DEFAULT))
280 throw ErrorMsg{ -1, windowEvent.type,
"sdl->setCursor" };
282 case SDL_EVENT_USER_POINTER_POSITION:
285 static_cast<INT32
>(
reinterpret_cast<uintptr_t
>(windowEvent.user.data1));
287 static_cast<INT32
>(
reinterpret_cast<uintptr_t
>(windowEvent.user.data2));
288 if (!
sdl->moveMouseTo(
289 {
static_cast<float>(x) * 1.0f,
static_cast<float>(y) * 1.0f }))
290 throw ErrorMsg{ -1, windowEvent.type,
"sdl->moveMouseTo" };
293 case SDL_EVENT_USER_POINTER_SET:
294 if (!
sdl->setCursor(
static_cast<rdpPointer*
>(windowEvent.user.data1)))
295 throw ErrorMsg{ -1, windowEvent.type,
"sdl->setCursor" };
297 case SDL_EVENT_USER_QUIT:
305 catch (ErrorMsg& msg)
307 WLog_Print(
sdl->getWLog(), WLOG_ERROR,
"[exception] %s", msg.what());
317[[nodiscard]]
static BOOL sdl_client_global_init()
321 const DWORD wVersionRequested = MAKEWORD(1, 1);
322 const int rc = WSAStartup(wVersionRequested, &wsaData);
325 WLog_ERR(SDL_TAG,
"WSAStartup failed with %s [%d]", gai_strerrorA(rc), rc);
330 return (freerdp_handle_signals() == 0);
334static void sdl_client_global_uninit()
341[[nodiscard]]
static BOOL sdl_client_new(freerdp* instance, rdpContext* context)
345 if (!instance || !context)
349 return sdl->sdl !=
nullptr;
352static void sdl_client_free([[maybe_unused]] freerdp* instance, rdpContext* context)
362[[nodiscard]]
static int sdl_client_start(rdpContext* context)
364 auto sdl = get_context(context);
369[[nodiscard]]
static int sdl_client_stop(rdpContext* context)
371 auto sdl = get_context(context);
376static void RdpClientEntry(RDP_CLIENT_ENTRY_POINTS* pEntryPoints)
378 WINPR_ASSERT(pEntryPoints);
380 ZeroMemory(pEntryPoints,
sizeof(RDP_CLIENT_ENTRY_POINTS));
381 pEntryPoints->Version = RDP_CLIENT_INTERFACE_VERSION;
382 pEntryPoints->Size =
sizeof(RDP_CLIENT_ENTRY_POINTS_V1);
383 pEntryPoints->GlobalInit = sdl_client_global_init;
384 pEntryPoints->GlobalUninit = sdl_client_global_uninit;
386 pEntryPoints->ClientNew = sdl_client_new;
387 pEntryPoints->ClientFree = sdl_client_free;
388 pEntryPoints->ClientStart = sdl_client_start;
389 pEntryPoints->ClientStop = sdl_client_stop;
395 freerdp_client_context_free(&
sdl->common.context);
398[[nodiscard]]
static const char* category2str(
int category)
402 case SDL_LOG_CATEGORY_APPLICATION:
403 return "SDL_LOG_CATEGORY_APPLICATION";
404 case SDL_LOG_CATEGORY_ERROR:
405 return "SDL_LOG_CATEGORY_ERROR";
406 case SDL_LOG_CATEGORY_ASSERT:
407 return "SDL_LOG_CATEGORY_ASSERT";
408 case SDL_LOG_CATEGORY_SYSTEM:
409 return "SDL_LOG_CATEGORY_SYSTEM";
410 case SDL_LOG_CATEGORY_AUDIO:
411 return "SDL_LOG_CATEGORY_AUDIO";
412 case SDL_LOG_CATEGORY_VIDEO:
413 return "SDL_LOG_CATEGORY_VIDEO";
414 case SDL_LOG_CATEGORY_RENDER:
415 return "SDL_LOG_CATEGORY_RENDER";
416 case SDL_LOG_CATEGORY_INPUT:
417 return "SDL_LOG_CATEGORY_INPUT";
418 case SDL_LOG_CATEGORY_TEST:
419 return "SDL_LOG_CATEGORY_TEST";
420 case SDL_LOG_CATEGORY_GPU:
421 return "SDL_LOG_CATEGORY_GPU";
422 case SDL_LOG_CATEGORY_RESERVED2:
423 return "SDL_LOG_CATEGORY_RESERVED2";
424 case SDL_LOG_CATEGORY_RESERVED3:
425 return "SDL_LOG_CATEGORY_RESERVED3";
426 case SDL_LOG_CATEGORY_RESERVED4:
427 return "SDL_LOG_CATEGORY_RESERVED4";
428 case SDL_LOG_CATEGORY_RESERVED5:
429 return "SDL_LOG_CATEGORY_RESERVED5";
430 case SDL_LOG_CATEGORY_RESERVED6:
431 return "SDL_LOG_CATEGORY_RESERVED6";
432 case SDL_LOG_CATEGORY_RESERVED7:
433 return "SDL_LOG_CATEGORY_RESERVED7";
434 case SDL_LOG_CATEGORY_RESERVED8:
435 return "SDL_LOG_CATEGORY_RESERVED8";
436 case SDL_LOG_CATEGORY_RESERVED9:
437 return "SDL_LOG_CATEGORY_RESERVED9";
438 case SDL_LOG_CATEGORY_RESERVED10:
439 return "SDL_LOG_CATEGORY_RESERVED10";
440 case SDL_LOG_CATEGORY_CUSTOM:
442 return "SDL_LOG_CATEGORY_CUSTOM";
446[[nodiscard]]
static SDL_LogPriority wloglevel2dl(DWORD level)
451 return SDL_LOG_PRIORITY_VERBOSE;
453 return SDL_LOG_PRIORITY_DEBUG;
455 return SDL_LOG_PRIORITY_INFO;
457 return SDL_LOG_PRIORITY_WARN;
459 return SDL_LOG_PRIORITY_ERROR;
461 return SDL_LOG_PRIORITY_CRITICAL;
464 return SDL_LOG_PRIORITY_VERBOSE;
468[[nodiscard]]
static DWORD sdlpriority2wlog(SDL_LogPriority priority)
470 DWORD level = WLOG_OFF;
473 case SDL_LOG_PRIORITY_VERBOSE:
476 case SDL_LOG_PRIORITY_DEBUG:
479 case SDL_LOG_PRIORITY_INFO:
482 case SDL_LOG_PRIORITY_WARN:
485 case SDL_LOG_PRIORITY_ERROR:
488 case SDL_LOG_PRIORITY_CRITICAL:
498static void SDLCALL winpr_LogOutputFunction(
void* userdata,
int category, SDL_LogPriority priority,
504 const DWORD level = sdlpriority2wlog(priority);
505 auto log =
sdl->getWLog();
506 if (!WLog_IsLevelActive(log, level))
509 WLog_PrintTextMessage(log, level, __LINE__, __FILE__, __func__,
"[%s] %s",
510 category2str(category), message);
513static void sdl_quit()
515 const auto cat = SDL_LOG_CATEGORY_APPLICATION;
517 ev.type = SDL_EVENT_QUIT;
518 if (!SDL_PushEvent(&ev))
520 SDL_LogError(cat,
"An error occurred: %s", SDL_GetError());
524static void SDLCALL rdp_file_cb(
void* userdata,
const char*
const* filelist,
525 WINPR_ATTR_UNUSED
int filter)
527 auto rdp =
static_cast<std::string*
>(userdata);
529 const auto cat = SDL_LOG_CATEGORY_APPLICATION;
532 SDL_LogError(cat,
"An error occurred: %s", SDL_GetError());
538 SDL_LogError(cat,
"The user did not select any file.");
539 SDL_LogError(cat,
"Most likely, the dialog was canceled.");
546 SDL_LogError(cat,
"Full path to selected file: '%s'", *filelist);
554[[nodiscard]]
static std::string getRdpFile()
556 const auto flags = SDL_INIT_VIDEO | SDL_INIT_EVENTS;
557 SDL_DialogFileFilter filters[] = { {
"RDP files",
"rdp;rdpw" } };
561 if (!SDL_Init(flags))
564 auto props = SDL_CreateProperties();
565 SDL_SetStringProperty(props, SDL_PROP_FILE_DIALOG_TITLE_STRING,
566 "SDL Freerdp - Open a RDP file");
567 SDL_SetBooleanProperty(props, SDL_PROP_FILE_DIALOG_MANY_BOOLEAN,
false);
568 SDL_SetPointerProperty(props, SDL_PROP_FILE_DIALOG_FILTERS_POINTER, filters);
569 SDL_SetNumberProperty(props, SDL_PROP_FILE_DIALOG_NFILTERS_NUMBER, ARRAYSIZE(filters));
570 SDL_ShowFileDialogWithProperties(SDL_FILEDIALOG_OPENFILE, rdp_file_cb, &rdp, props);
571 SDL_DestroyProperties(props);
575 SDL_Event
event = {};
576 std::ignore = SDL_PollEvent(&event);
591int main(
int argc,
char* argv[])
594 sdl::win32::release_transient_console();
598 RDP_CLIENT_ENTRY_POINTS clientEntryPoints = {};
601 RdpClientEntry(&clientEntryPoints);
603 reinterpret_cast<sdl_rdp_context*
>(freerdp_client_context_new(&clientEntryPoints)),
608 auto sdl = sdl_rdp->sdl;
610 auto settings =
sdl->context()->settings;
611 WINPR_ASSERT(settings);
613 std::string rdp_file;
614 std::vector<char*> args;
615 args.reserve(WINPR_ASSERTING_INT_CAST(
size_t, argc));
616 for (
auto x = 0; x < argc; x++)
617 args.push_back(argv[x]);
621 rdp_file = getRdpFile();
622 if (!rdp_file.empty())
624 args.push_back(rdp_file.data());
628 auto status = freerdp_client_settings_parse_command_line(
629 settings, WINPR_ASSERTING_INT_CAST(
int, args.size()), args.data(), FALSE);
630 sdl_rdp->sdl->setMetadata();
633 rc = freerdp_client_settings_command_line_status_print(settings, status, argc, argv);
636 if (!sdl_list_monitors(
sdl))
643 case COMMAND_LINE_STATUS_PRINT:
644 case COMMAND_LINE_STATUS_PRINT_VERSION:
645 case COMMAND_LINE_STATUS_PRINT_BUILDCONFIG:
647 case COMMAND_LINE_STATUS_PRINT_HELP:
648 SdlPref::print_config_file_help(3);
658 if (!SDL_Init(SDL_INIT_VIDEO | SDL_INIT_EVENTS))
661 SDL_SetHint(SDL_HINT_ALLOW_ALT_TAB_WHILE_GRABBED,
"0");
662 SDL_SetHint(SDL_HINT_VIDEO_X11_NET_WM_BYPASS_COMPOSITOR,
"0");
663 SDL_SetHint(SDL_HINT_PEN_MOUSE_EVENTS,
"0");
664 SDL_SetHint(SDL_HINT_TOUCH_MOUSE_EVENTS,
"0");
665 SDL_SetHint(SDL_HINT_PEN_TOUCH_EVENTS,
"1");
666 SDL_SetHint(SDL_HINT_TRACKPAD_IS_TOUCH_ONLY,
"1");
667#if SDL_VERSION_ATLEAST(3, 4, 0)
668 SDL_SetHint(SDL_HINT_MOUSE_DPI_SCALE_CURSORS,
"1");
672 SDL_SetLogOutputFunction(winpr_LogOutputFunction,
sdl);
673 auto level = WLog_GetLogLevel(
sdl->getWLog());
674 SDL_SetLogPriorities(wloglevel2dl(level));
676 auto backend = SDL_GetCurrentVideoDriver();
677 WLog_Print(
sdl->getWLog(), WLOG_DEBUG,
"client is using backend '%s'", backend);
685 freerdp_del_signal_cleanup_handler(
sdl->context(), sdl_term_handler);
686 sdl_dialogs_uninit();
689 if (!
sdl->detectDisplays())
693 auto context =
sdl->context();
694 WINPR_ASSERT(context);
696 if (!stream_dump_register_handlers(context, CONNECTION_STATE_MCS_CREATE_REQUEST, FALSE))
699 if (freerdp_client_start(context) != 0)
704 if (freerdp_client_stop(context) != 0)
707 if (
sdl->exitCode() != 0)
708 rc =
sdl->exitCode();
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.