22#include <freerdp/config.h>
23#include <freerdp/utils/helpers.h>
28#include <winpr/assert.h>
29#include <winpr/string.h>
31#include <winpr/wlog.h>
32#include <winpr/path.h>
33#include <winpr/ncrypt.h>
34#include <winpr/environment.h>
35#include <winpr/timezone.h>
37#include <freerdp/freerdp.h>
38#include <freerdp/addin.h>
39#include <freerdp/settings.h>
40#include <freerdp/client.h>
41#include <freerdp/client/channels.h>
42#include <freerdp/channels/drdynvc.h>
43#include <freerdp/channels/cliprdr.h>
44#include <freerdp/channels/encomsp.h>
45#include <freerdp/channels/rdpear.h>
46#include <freerdp/channels/rdpewa.h>
47#include <freerdp/channels/rdp2tcp.h>
48#include <freerdp/channels/remdesk.h>
49#include <freerdp/channels/rdpsnd.h>
50#include <freerdp/channels/disp.h>
51#include <freerdp/crypto/crypto.h>
52#include <freerdp/locale/keyboard.h>
53#include <freerdp/utils/passphrase.h>
54#include <freerdp/utils/proxy_utils.h>
55#include <freerdp/utils/string.h>
56#include <freerdp/channels/urbdrc.h>
57#include <freerdp/channels/rdpdr.h>
58#include <freerdp/locale/locale.h>
60#include <freerdp/utils/warnings.h>
62#if defined(CHANNEL_AINPUT_CLIENT)
63#include <freerdp/channels/ainput.h>
66#include <freerdp/channels/audin.h>
67#include <freerdp/channels/echo.h>
69#include <freerdp/client/cmdline.h>
70#include <freerdp/version.h>
71#include <freerdp/client/utils/smartcard_cli.h>
73#include <openssl/tls1.h>
76#include <freerdp/log.h>
77#define TAG CLIENT_TAG("common.cmdline")
79static const char str_force[] =
"force";
81static const char* credential_args[] = {
"p",
"smartcard-logon",
82#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
85 "pth",
"reconnect-cookie",
88static const char* option_starts_with(
const char* what,
const char* val);
89static BOOL option_ends_with(
const char* str,
const char* ext);
90static BOOL option_equals(
const char* what,
const char* val);
92static BOOL freerdp_client_print_codepages(
const char* arg)
96 const char* filter =
nullptr;
101 filter = strchr(arg,
',');
107 pages = freerdp_keyboard_get_matching_codepages(column, filter, &count);
111 printf(
"%-10s %-8s %-60s %-36s %-48s\n",
"<id>",
"<locale>",
"<win langid>",
"<language>",
113 for (
size_t x = 0; x < count; x++)
116 char buffer[2048] = WINPR_C_ARRAY_INIT;
118 if (strnlen(page->subLanguageSymbol, ARRAYSIZE(page->subLanguageSymbol)) > 0)
119 (void)_snprintf(buffer,
sizeof(buffer),
"[%s|%s]", page->primaryLanguageSymbol,
120 page->subLanguageSymbol);
122 (
void)_snprintf(buffer,
sizeof(buffer),
"[%s]", page->primaryLanguageSymbol);
123 printf(
"id=0x%04" PRIx16
": [%-6s] %-60s %-36s %-48s\n", page->id, page->locale, buffer,
124 page->primaryLanguage, page->subLanguage);
126 freerdp_codepages_free(pages);
130static BOOL freerdp_path_valid(
const char* path, BOOL* special)
132 const char DynamicDrives[] =
"DynamicDrives";
138 isSpecial = (option_equals(
"*", path) || option_equals(DynamicDrives, path) ||
139 option_equals(
"%", path));
141 isPath = winpr_PathFileExists(path);
144 *special = isSpecial;
146 return isSpecial || isPath;
149static BOOL freerdp_sanitize_drive_name(
char* name,
const char* invalid,
const char* replacement)
151 if (!name || !invalid || !replacement)
153 if (strlen(invalid) != strlen(replacement))
156 while (*invalid !=
'\0')
158 const char what = *invalid++;
159 const char with = *replacement++;
162 while ((cur = strchr(cur, what)) !=
nullptr)
168static char* name_from_path(
const char* path)
170 const char* name =
"nullptr";
173 if (option_equals(
"%", path))
175 else if (option_equals(
"*", path))
176 name =
"hotplug-all";
177 else if (option_equals(
"DynamicDrives", path))
182 return _strdup(name);
185static BOOL freerdp_client_add_drive(rdpSettings* settings,
const char* path,
const char* name)
187 char* dname =
nullptr;
206 if (!skip && winpr_PathFileExists(name))
208 if (!winpr_PathFileExists(path) || (!PathIsRelativeA(name) && PathIsRelativeA(path)))
210 const char* tmp = path;
218 dname = _strdup(name);
220 dname = name_from_path(path);
222 if (freerdp_sanitize_drive_name(dname,
"\\/",
"__"))
224 const char* args[] = { dname, path };
225 device = freerdp_device_new(RDPDR_DTYP_FILESYSTEM, ARRAYSIZE(args), args);
235 BOOL isSpecial = FALSE;
236 BOOL isPath = freerdp_path_valid(path, &isSpecial);
238 if (!isPath && !isSpecial)
240 WLog_WARN(TAG,
"Invalid drive to redirect: '%s' does not exist, skipping.", path);
241 freerdp_device_free(device);
243 else if (!freerdp_device_collection_add(settings, device))
250 freerdp_device_free(device);
254static BOOL value_to_int(
const char* value, LONGLONG* result, LONGLONG min, LONGLONG max)
258 if (!value || !result)
262 rc = _strtoi64(value,
nullptr, 0);
267 if ((rc < min) || (rc > max))
274static BOOL value_to_uint(
const char* value, ULONGLONG* result, ULONGLONG min, ULONGLONG max)
276 unsigned long long rc = 0;
278 if (!value || !result)
282 rc = _strtoui64(value,
nullptr, 0);
287 if ((rc < min) || (rc > max))
294BOOL freerdp_client_print_version(
void)
296 printf(
"This is FreeRDP version %s (%s)\n", FREERDP_VERSION_FULL, FREERDP_GIT_REVISION);
300BOOL freerdp_client_print_version_ex(
int argc,
char** argv)
302 WINPR_ASSERT(argc >= 0);
303 WINPR_ASSERT(argv || (argc == 0));
304 const char* name = (argc > 0) ? argv[0] :
"argc < 1";
305 printf(
"This is FreeRDP version [%s] %s (%s)\n", name, FREERDP_VERSION_FULL,
306 FREERDP_GIT_REVISION);
310BOOL freerdp_client_print_buildconfig(
void)
312 printf(
"%s", freerdp_get_build_config());
316BOOL freerdp_client_print_buildconfig_ex(
int argc,
char** argv)
318 WINPR_ASSERT(argc >= 0);
319 WINPR_ASSERT(argv || (argc == 0));
320 const char* name = (argc > 0) ? argv[0] :
"argc < 1";
321 printf(
"[%s] %s", name, freerdp_get_build_config());
325static void freerdp_client_print_scancodes(
void)
327 printf(
"RDP scancodes and their name for use with /kbd:remap\n");
329 for (UINT32 x = 0; x < UINT16_MAX; x++)
331 const char* name = freerdp_keyboard_scancode_name(x);
333 printf(
"0x%04" PRIx32
" --> %s\n", x, name);
337static BOOL is_delimiter(
char c,
const char* delimiters)
340 while ((d = *delimiters++) !=
'\0')
348static const char* get_last(
const char* start,
size_t len,
const char* delimiters)
350 const char* last =
nullptr;
351 for (
size_t x = 0; x < len; x++)
354 if (is_delimiter(c, delimiters))
360static SSIZE_T next_delimiter(
const char* text,
size_t len,
size_t max,
const char* delimiters)
365 const char* last = get_last(text, max, delimiters);
369 return (SSIZE_T)(last - text);
372static SSIZE_T forced_newline_at(
const char* text,
size_t len,
size_t limit,
373 const char* force_newline)
376 while ((d = *force_newline++) !=
'\0')
378 const char* tok = strchr(text, d);
381 const size_t offset = WINPR_ASSERTING_INT_CAST(
size_t, tok - text);
382 if ((offset > len) || (offset > limit))
384 return (SSIZE_T)(offset);
390static BOOL print_align(
size_t start_offset,
size_t* current)
392 WINPR_ASSERT(current);
393 if (*current < start_offset)
395 const int rc = printf(
"%*c", (
int)(start_offset - *current),
' ');
398 *current += (size_t)rc;
403static char* print_token(
char* text,
size_t start_offset,
size_t* current,
size_t limit,
404 const char* delimiters,
const char* force_newline)
407 const size_t tlen = strnlen(text, limit);
409 const SSIZE_T force_at = forced_newline_at(text, len, limit - *current, force_newline);
410 BOOL isForce = (force_at >= 0);
413 len = MIN(len, (
size_t)force_at);
415 if (!print_align(start_offset, current))
418 const SSIZE_T delim = next_delimiter(text, len, limit - *current, delimiters);
419 const BOOL isDelim = delim > 0;
422 len = MIN(len, (
size_t)delim + 1);
425 rc = printf(
"%.*s", (
int)len, text);
429 if (isForce || isDelim)
434 const size_t offset = len + ((isForce && (force_at == 0)) ? 1 : 0);
435 return &text[offset];
438 *current += (size_t)rc;
440 if (tlen == (
size_t)rc)
442 return &text[(size_t)rc];
445static size_t print_optionals(
const char* text,
size_t start_offset,
size_t current)
447 const size_t limit = 80;
448 char* str = _strdup(text);
453 cur = print_token(cur, start_offset + 1, ¤t, limit,
"[], ",
"\r\n");
454 }
while (cur !=
nullptr);
460static size_t print_description(
const char* text,
size_t start_offset,
size_t current)
462 const size_t limit = 80;
463 char* str = _strdup(text);
466 while (cur !=
nullptr)
467 cur = print_token(cur, start_offset, ¤t, limit,
" ",
"\r\n");
470 const int rc = printf(
"\n");
473 const size_t src = WINPR_ASSERTING_INT_CAST(
size_t, rc);
474 WINPR_ASSERT(SIZE_MAX - src > current);
480static int cmp_cmdline_args(
const void* pva,
const void* pvb)
485 if (!a->Name && !b->Name)
491 return strcmp(a->Name, b->Name);
506 const size_t description_offset = 30 + 8;
508 if (arg->Flags & (COMMAND_LINE_VALUE_BOOL | COMMAND_LINE_VALUE_FLAG))
510 if ((arg->Flags & (uint32_t)~COMMAND_LINE_VALUE_BOOL) == 0)
511 rc = printf(
" %s%s", arg->Default ?
"-" :
"+", arg->Name);
512 else if ((arg->Flags & COMMAND_LINE_VALUE_OPTIONAL) != 0)
513 rc = printf(
" [%s|/]%s", arg->Default ?
"-" :
"+", arg->Name);
516 rc = printf(
" %s%s", arg->Default ?
"-" :
"+", arg->Name);
520 rc = printf(
" /%s", arg->Name);
526 if ((arg->Flags & COMMAND_LINE_VALUE_REQUIRED) ||
527 (arg->Flags & COMMAND_LINE_VALUE_OPTIONAL))
531 if (arg->Flags & COMMAND_LINE_VALUE_OPTIONAL)
537 pos = print_optionals(arg->Format, pos, pos);
549 pos = print_optionals(arg->Format, pos, pos);
552 if (pos > description_offset)
560 rc = printf(
"%*c", (
int)(description_offset - pos),
' ');
565 if (arg->Flags & COMMAND_LINE_VALUE_BOOL)
567 rc = printf(
"%s ", arg->Default ?
"Disable" :
"Enable");
573 print_description(arg->Text, description_offset, pos);
574 }
while ((arg = CommandLineFindNextArgumentA(arg)) !=
nullptr);
577BOOL freerdp_client_print_command_line_help(
int argc,
char** argv)
579 return freerdp_client_print_command_line_help_ex(argc, argv,
nullptr);
583 SSIZE_T count,
size_t* pcount)
585 WINPR_ASSERT(pcount);
590 while (cur && cur->Name)
605 while (cur && cur->Name)
607 largs[lcount++] = *cur++;
610 cur = global_cmd_args;
611 while (cur && cur->Name)
613 largs[lcount++] = *cur++;
619static void freerdp_client_print_command_line_usage(
int argc,
char** argv)
621 WINPR_ASSERT(argv || (argc < 1));
623 const char* name = freerdp_getApplicationDetailsString();
627 printf(
"%s - A Free Remote Desktop Protocol Implementation\n", name);
628 printf(
"To show full command line help type\n");
629 printf(
"%s /?\n", name);
633BOOL freerdp_client_print_command_line_help_ex(
int argc,
char** argv,
636 const char* name = freerdp_getApplicationDetailsString();
648 printf(
"%s - A Free Remote Desktop Protocol Implementation\n", name);
649 printf(
"See www.freerdp.com for more information\n");
651 printf(
"Usage: %s [file] [options] [/v:<server>[:port]]\n", argv[0]);
654 printf(
" /flag (enables flag)\n");
655 printf(
" /option:<value> (specifies option with value)\n");
656 printf(
" +toggle -toggle (enables or disables toggle, where '/' is a synonym of '+')\n");
659 freerdp_client_print_command_line_args(largs, lcount);
663 printf(
"Examples:\n");
664 printf(
" %s connection.rdp /p:Pwd123! /f\n", name);
665 printf(
" %s /u:CONTOSO\\JohnDoe /p:Pwd123! /v:rdp.contoso.com\n", name);
666 printf(
" %s /u:JohnDoe /p:Pwd123! /w:1366 /h:768 /v:192.168.1.100:4489\n", name);
667 printf(
" %s /u:JohnDoe /p:Pwd123! /vmconnect:C824F53E-95D2-46C6-9A18-23A5BB403532 "
668 "/v:192.168.1.100\n",
670 printf(
" %s /u:\\AzureAD\\user@corp.example /p:pwd /v:host\n", name);
671 printf(
"Use a generic pipe as transport:");
672 printf(
" %s /v:/path/to/pipe\n", name);
673 printf(
"Use a external socket:");
674 printf(
" %s /v:|:1234\n", name);
676 printf(
"Connect to a system with TLS security and open the greeter:");
677 printf(
"NOTE: Needs a server configured to not require NLA or it will fail!");
679 printf(
" %s /sec:tls /p /v:rdp.contoso.com\n", name);
681 printf(
"Disable clipboard redirection: -clipboard\n");
683 printf(
"Drive Redirection: /drive:home,/home/user\n");
684 printf(
"Smartcard Redirection: /smartcard:<device>\n");
685 printf(
"Smartcard logon with Kerberos authentication: /smartcard-logon /sec:nla\n");
687#if defined(CHANNEL_SERIAL_CLIENT)
688 printf(
"Serial Port Redirection: /serial:<name>,<device>,[SerCx2|SerCx|Serial],[permissive]\n");
689 printf(
"Serial Port Redirection: /serial:COM1,/dev/ttyS0\n");
691#if defined(CHANNEL_PARALLEL_CLIENT)
692 printf(
"Parallel Port Redirection: /parallel:<name>,<device>\n");
694 printf(
"Printer Redirection: /printer:<device>,<driver>,[default]\n");
695 printf(
"TCP redirection: /rdp2tcp:/usr/bin/rdp2tcp\n");
697 printf(
"Audio Output Redirection: /sound:sys:oss,dev:1,format:1\n");
698 printf(
"Audio Output Redirection: /sound:sys:alsa\n");
699 printf(
"Audio Input Redirection: /microphone:sys:oss,dev:1,format:1\n");
700 printf(
"Audio Input Redirection: /microphone:sys:alsa\n");
702 printf(
"Multimedia Redirection: /video\n");
703#ifdef CHANNEL_URBDRC_CLIENT
704 printf(
"USB Device Redirection: /usb:id:054c:0268#4669:6e6b,addr:04:0c\n");
707 printf(
"For Gateways, the https_proxy environment variable is respected:\n");
709 printf(
" set HTTPS_PROXY=http://proxy.contoso.com:3128/\n");
711 printf(
" export https_proxy=http://proxy.contoso.com:3128/\n");
713 printf(
" %s /gateway:g:rdp.contoso.com ...\n", name);
715 printf(
"More documentation is coming, in the meantime consult source files\n");
720static BOOL option_is_rdp_file(
const char* option)
722 WINPR_ASSERT(option);
724 if (option_ends_with(option,
".rdp"))
726 if (option_ends_with(option,
".rdpw"))
731static BOOL option_is_incident_file(
const char* option)
733 WINPR_ASSERT(option);
735 return (option_ends_with(option,
".msrcIncident"));
738static int freerdp_client_command_line_pre_filter(
void* context,
int index,
int argc, LPSTR* argv)
743 rdpSettings* settings =
nullptr;
748 length = strlen(argv[index]);
752 if (option_is_rdp_file(argv[index]))
754 settings = (rdpSettings*)context;
757 return COMMAND_LINE_ERROR_MEMORY;
765 if (option_is_incident_file(argv[index]))
767 settings = (rdpSettings*)context;
770 return COMMAND_LINE_ERROR_MEMORY;
780BOOL freerdp_client_add_device_channel(rdpSettings* settings,
size_t count,
781 const char*
const* params)
783 WINPR_ASSERT(settings);
784 WINPR_ASSERT(params);
785 WINPR_ASSERT(count > 0);
787 if (option_equals(params[0],
"drive"))
796 rc = freerdp_client_add_drive(settings, params[1],
nullptr);
798 rc = freerdp_client_add_drive(settings, params[2], params[1]);
802 else if (option_equals(params[0],
"printer"))
814 printer = freerdp_device_new(RDPDR_DTYP_PRINT, count - 1, ¶ms[1]);
818 if (!freerdp_device_collection_add(settings, printer))
820 freerdp_device_free(printer);
826 else if (option_equals(params[0],
"smartcard"))
838 smartcard = freerdp_device_new(RDPDR_DTYP_SMARTCARD, count - 1, ¶ms[1]);
843 if (!freerdp_device_collection_add(settings, smartcard))
845 freerdp_device_free(smartcard);
851#if defined(CHANNEL_SERIAL_CLIENT)
852 else if (option_equals(params[0],
"serial"))
864 serial = freerdp_device_new(RDPDR_DTYP_SERIAL, count - 1, ¶ms[1]);
869 if (!freerdp_device_collection_add(settings, serial))
871 freerdp_device_free(serial);
878 else if (option_equals(params[0],
"parallel"))
890 parallel = freerdp_device_new(RDPDR_DTYP_PARALLEL, count - 1, ¶ms[1]);
895 if (!freerdp_device_collection_add(settings, parallel))
897 freerdp_device_free(parallel);
907BOOL freerdp_client_del_static_channel(rdpSettings* settings,
const char* name)
909 return freerdp_static_channel_collection_del(settings, name);
912BOOL freerdp_client_add_static_channel(rdpSettings* settings,
size_t count,
913 const char*
const* params)
917 if (!settings || !params || !params[0] || (count > INT_MAX))
920 if (freerdp_static_channel_collection_find(settings, params[0]))
923 _args = freerdp_addin_argv_new(count, params);
928 if (!freerdp_static_channel_collection_add(settings, _args))
933 freerdp_addin_argv_free(_args);
937BOOL freerdp_client_del_dynamic_channel(rdpSettings* settings,
const char* name)
939 return freerdp_dynamic_channel_collection_del(settings, name);
942BOOL freerdp_client_add_dynamic_channel(rdpSettings* settings,
size_t count,
943 const char*
const* params)
947 if (!settings || !params || !params[0] || (count > INT_MAX))
950 if (freerdp_dynamic_channel_collection_find(settings, params[0]))
953 _args = freerdp_addin_argv_new(count, params);
958 if (!freerdp_dynamic_channel_collection_add(settings, _args))
964 freerdp_addin_argv_free(_args);
968static BOOL read_pem_file(rdpSettings* settings, FreeRDP_Settings_Keys_String
id,
const char* file)
971 char* pem = crypto_read_pem(file, &length);
972 if (!pem || (length == 0))
986 CMDLINE_SUBOPTION_STRING,
987 CMDLINE_SUBOPTION_FILE,
988} CmdLineSubOptionType;
990typedef BOOL (*CmdLineSubOptionCb)(
const char* value, rdpSettings* settings);
994 FreeRDP_Settings_Keys_String id;
995 CmdLineSubOptionType opttype;
996 WINPR_ATTR_NODISCARD CmdLineSubOptionCb cb;
999static BOOL parseSubOptions(rdpSettings* settings,
const CmdLineSubOptions* opts,
size_t count,
1004 for (
size_t xx = 0; xx < count; xx++)
1006 const CmdLineSubOptions* opt = &opts[xx];
1008 if (option_starts_with(opt->optname, arg))
1010 const size_t optlen = strlen(opt->optname);
1011 const char* val = &arg[optlen];
1014 switch (opt->opttype)
1016 case CMDLINE_SUBOPTION_STRING:
1019 case CMDLINE_SUBOPTION_FILE:
1020 status = read_pem_file(settings, opt->id, val);
1023 WLog_ERR(TAG,
"invalid subOption type");
1030 if (opt->cb && !opt->cb(val, settings))
1039 WLog_ERR(TAG,
"option %s not handled", arg);
1044#define fail_at(arg, rc) fail_at_((arg), (rc), __FILE__, __func__, __LINE__)
1051 const DWORD level = WLOG_ERROR;
1052 wLog* log = WLog_Get(TAG);
1053 if (WLog_IsLevelActive(log, level))
1055 const char* val = arg->Value;
1056 if ((arg->Flags & COMMAND_LINE_VALUE_FLAG) != 0)
1057 val = arg->Value ==
nullptr ?
"Disable" :
"Enable";
1058 if ((arg->Flags & COMMAND_LINE_VALUE_BOOL) != 0)
1059 val = arg->Value ==
nullptr ?
"Disable" :
"Enable";
1061 WLog_PrintTextMessage(log, level, line, file, fkt,
1062 "Command line parsing failed at '%s' value '%s' [%d]", arg->Name, val,
1070 rdpSettings* settings = (rdpSettings*)context;
1071 int status = CHANNEL_RC_OK;
1072 BOOL enable = (arg->Value !=
nullptr);
1074 CommandLineSwitchStart(arg) CommandLineSwitchCase(arg,
"a")
1077 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
1079 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1080 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1082 status = COMMAND_LINE_ERROR;
1084 CommandLineParserFree(ptr);
1086 return fail_at(arg, status);
1088 CommandLineSwitchCase(arg,
"kerberos")
1092 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"kerberos", arg->Value, &count);
1095 const CmdLineSubOptions opts[] = {
1096 {
"kdc-url:", FreeRDP_KerberosKdcUrl, CMDLINE_SUBOPTION_STRING,
nullptr },
1097 {
"start-time:", FreeRDP_KerberosStartTime, CMDLINE_SUBOPTION_STRING,
nullptr },
1098 {
"lifetime:", FreeRDP_KerberosLifeTime, CMDLINE_SUBOPTION_STRING,
nullptr },
1099 {
"renewable-lifetime:", FreeRDP_KerberosRenewableLifeTime,
1100 CMDLINE_SUBOPTION_STRING,
nullptr },
1101 {
"cache:", FreeRDP_KerberosCache, CMDLINE_SUBOPTION_STRING,
nullptr },
1102 {
"armor:", FreeRDP_KerberosArmor, CMDLINE_SUBOPTION_STRING,
nullptr },
1103 {
"pkinit-anchors:", FreeRDP_PkinitAnchors, CMDLINE_SUBOPTION_STRING,
nullptr },
1104 {
"pkcs11-module:", FreeRDP_Pkcs11Module, CMDLINE_SUBOPTION_STRING,
nullptr }
1107 for (
size_t x = 1; x < count; x++)
1109 const char* cur = ptr[x];
1110 if (!parseSubOptions(settings, opts, ARRAYSIZE(opts), cur))
1112 CommandLineParserFree(ptr);
1113 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
1117 CommandLineParserFree(ptr);
1120 CommandLineSwitchCase(arg,
"vc")
1123 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
1124 if (!freerdp_client_add_static_channel(settings, count, (
const char*
const*)ptr))
1125 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1126 CommandLineParserFree(ptr);
1128 return fail_at(arg, status);
1130 CommandLineSwitchCase(arg,
"dvc")
1133 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
1134 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1135 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1136 CommandLineParserFree(ptr);
1138 return fail_at(arg, status);
1140 CommandLineSwitchCase(arg,
"drive")
1143 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1144 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1145 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1146 CommandLineParserFree(ptr);
1148 return fail_at(arg, status);
1150#if defined(CHANNEL_SERIAL_CLIENT)
1151 CommandLineSwitchCase(arg,
"serial")
1154 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1155 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1156 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1157 CommandLineParserFree(ptr);
1159 return fail_at(arg, status);
1162#if defined(CHANNEL_PARALLEL_CLIENT)
1163 CommandLineSwitchCase(arg,
"parallel")
1166 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1167 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1168 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1169 CommandLineParserFree(ptr);
1171 return fail_at(arg, status);
1174 CommandLineSwitchCase(arg,
"smartcard")
1177 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1178 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1179 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1180 CommandLineParserFree(ptr);
1182 return fail_at(arg, status);
1184 CommandLineSwitchCase(arg,
"printer")
1187 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1188 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1189 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1190 CommandLineParserFree(ptr);
1192 return fail_at(arg, status);
1194 CommandLineSwitchCase(arg,
"usb")
1198 CommandLineParseCommaSeparatedValuesEx(URBDRC_CHANNEL_NAME, arg->Value, &count);
1199 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1200 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1201 CommandLineParserFree(ptr);
1203 return fail_at(arg, status);
1205 CommandLineSwitchCase(arg,
"multitouch")
1208 return fail_at(arg, COMMAND_LINE_ERROR);
1210 CommandLineSwitchCase(arg,
"gestures")
1213 return fail_at(arg, COMMAND_LINE_ERROR);
1215 CommandLineSwitchCase(arg,
"echo")
1218 return fail_at(arg, COMMAND_LINE_ERROR);
1220 CommandLineSwitchCase(arg,
"ssh-agent")
1223 return fail_at(arg, COMMAND_LINE_ERROR);
1225 CommandLineSwitchCase(arg,
"disp")
1228 return fail_at(arg, COMMAND_LINE_ERROR);
1230 CommandLineSwitchCase(arg,
"geometry")
1233 return fail_at(arg, COMMAND_LINE_ERROR);
1235 CommandLineSwitchCase(arg,
"video")
1239 return fail_at(arg, COMMAND_LINE_ERROR);
1241 return fail_at(arg, COMMAND_LINE_ERROR);
1243 CommandLineSwitchCase(arg,
"sound")
1247 CommandLineParseCommaSeparatedValuesEx(RDPSND_CHANNEL_NAME, arg->Value, &count);
1248 if (!freerdp_client_add_static_channel(settings, count, (
const char*
const*)ptr))
1249 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1250 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1251 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1253 CommandLineParserFree(ptr);
1255 return fail_at(arg, status);
1257 CommandLineSwitchCase(arg,
"microphone")
1260 char** ptr = CommandLineParseCommaSeparatedValuesEx(AUDIN_CHANNEL_NAME, arg->Value, &count);
1261 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1262 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1263 CommandLineParserFree(ptr);
1265 return fail_at(arg, status);
1267#if defined(CHANNEL_TSMF_CLIENT)
1268 CommandLineSwitchCase(arg,
"multimedia")
1271 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"tsmf", arg->Value, &count);
1272 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1273 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1274 CommandLineParserFree(ptr);
1276 return fail_at(arg, status);
1279 CommandLineSwitchCase(arg,
"heartbeat")
1282 return fail_at(arg, COMMAND_LINE_ERROR);
1284 CommandLineSwitchCase(arg,
"multitransport")
1287 return fail_at(arg, COMMAND_LINE_ERROR);
1292 (TRANSPORT_TYPE_UDP_FECR | TRANSPORT_TYPE_UDP_FECL | TRANSPORT_TYPE_UDP_PREFERRED);
1295 return fail_at(arg, COMMAND_LINE_ERROR);
1297 CommandLineSwitchEnd(arg)
1304 int status = freerdp_client_command_line_post_filter_int(context, arg);
1305 return status == CHANNEL_RC_OK ? 1 : -1;
1308static BOOL freerdp_parse_username_ptr(
const char* username,
const char** user,
size_t* userlen,
1309 const char** domain,
size_t* domainlen)
1312 WINPR_ASSERT(userlen);
1313 WINPR_ASSERT(domain);
1314 WINPR_ASSERT(domainlen);
1319 const char* p = strchr(username,
'\\');
1329 const size_t length = (size_t)(p - username);
1331 *userlen = strlen(*user);
1334 *domainlen = length;
1343 *userlen = strlen(username);
1349static BOOL freerdp_parse_username_settings(
const char* username, rdpSettings* settings,
1350 FreeRDP_Settings_Keys_String userID,
1351 FreeRDP_Settings_Keys_String domainID)
1353 const char* user =
nullptr;
1354 const char* domain =
nullptr;
1356 size_t domainlen = 0;
1358 const BOOL rc = freerdp_parse_username_ptr(username, &user, &userlen, &domain, &domainlen);
1366BOOL freerdp_parse_username(
const char* username,
char** puser,
char** pdomain)
1368 const char* user =
nullptr;
1369 const char* domain =
nullptr;
1371 size_t domainlen = 0;
1376 const BOOL rc = freerdp_parse_username_ptr(username, &user, &userlen, &domain, &domainlen);
1382 *puser = strndup(user, userlen);
1389 *pdomain = strndup(domain, domainlen);
1401BOOL freerdp_parse_hostname(
const char* hostname,
char** host,
int* port)
1404 p = strrchr(hostname,
':');
1408 size_t length = (size_t)(p - hostname);
1411 if (!value_to_int(p + 1, &val, 1, UINT16_MAX))
1414 *host = (
char*)calloc(length + 1UL,
sizeof(
char));
1419 CopyMemory(*host, hostname, length);
1420 (*host)[length] =
'\0';
1421 *port = (UINT16)val;
1425 *host = _strdup(hostname);
1436static BOOL freerdp_apply_connection_type(rdpSettings* settings, UINT32 type)
1438 struct network_settings
1440 FreeRDP_Settings_Keys_Bool id;
1443 const struct network_settings config[] = {
1444 { FreeRDP_DisableWallpaper, { TRUE, TRUE, TRUE, TRUE, FALSE, FALSE, FALSE } },
1445 { FreeRDP_AllowFontSmoothing, { FALSE, FALSE, FALSE, FALSE, TRUE, TRUE, TRUE } },
1446 { FreeRDP_AllowDesktopComposition, { FALSE, FALSE, TRUE, TRUE, TRUE, TRUE, TRUE } },
1447 { FreeRDP_DisableFullWindowDrag, { TRUE, TRUE, TRUE, TRUE, FALSE, FALSE, FALSE } },
1448 { FreeRDP_DisableMenuAnims, { TRUE, TRUE, TRUE, TRUE, FALSE, FALSE, FALSE } },
1449 { FreeRDP_DisableThemes, { TRUE, FALSE, FALSE, FALSE, FALSE, FALSE, FALSE } }
1454 case CONNECTION_TYPE_INVALID:
1457 case CONNECTION_TYPE_MODEM:
1458 case CONNECTION_TYPE_BROADBAND_LOW:
1459 case CONNECTION_TYPE_BROADBAND_HIGH:
1460 case CONNECTION_TYPE_SATELLITE:
1461 case CONNECTION_TYPE_WAN:
1462 case CONNECTION_TYPE_LAN:
1463 case CONNECTION_TYPE_AUTODETECT:
1466 WLog_WARN(TAG,
"Unknown ConnectionType %" PRIu32
", aborting", type);
1470 for (
size_t x = 0; x < ARRAYSIZE(config); x++)
1472 const struct network_settings* cur = &config[x];
1479BOOL freerdp_set_connection_type(rdpSettings* settings, UINT32 type)
1487 case CONNECTION_TYPE_INVALID:
1488 case CONNECTION_TYPE_MODEM:
1489 case CONNECTION_TYPE_BROADBAND_LOW:
1490 case CONNECTION_TYPE_SATELLITE:
1491 case CONNECTION_TYPE_BROADBAND_HIGH:
1492 case CONNECTION_TYPE_WAN:
1493 case CONNECTION_TYPE_LAN:
1494 if (!freerdp_apply_connection_type(settings, type))
1497 case CONNECTION_TYPE_AUTODETECT:
1498 if (!freerdp_apply_connection_type(settings, type))
1512 WLog_WARN(TAG,
"Unknown ConnectionType %" PRIu32
", aborting", type);
1519static UINT32 freerdp_get_keyboard_layout_for_type(
const char* name, WINPR_ATTR_UNUSED DWORD type)
1524 freerdp_keyboard_get_layouts(RDP_KEYBOARD_LAYOUT_TYPE_STANDARD, &count);
1526 if (!layouts || (count == 0))
1529 for (
size_t x = 0; x < count; x++)
1532 if (option_equals(layout->name, name))
1540 freerdp_keyboard_layouts_free(layouts, count);
1544static UINT32 freerdp_map_keyboard_layout_name_to_id(
const char* name)
1546 const UINT32 variants[] = { RDP_KEYBOARD_LAYOUT_TYPE_STANDARD, RDP_KEYBOARD_LAYOUT_TYPE_VARIANT,
1547 RDP_KEYBOARD_LAYOUT_TYPE_IME };
1549 for (
size_t x = 0; x < ARRAYSIZE(variants); x++)
1551 UINT32 rc = freerdp_get_keyboard_layout_for_type(name, variants[x]);
1559static int freerdp_detect_command_line_pre_filter(
void* context,
int index,
int argc, LPSTR* argv)
1562 WINPR_UNUSED(context);
1569 length = strlen(argv[index]);
1573 if (option_is_rdp_file(argv[index]))
1581 if (option_is_incident_file(argv[index]))
1591static int freerdp_detect_windows_style_command_line_syntax(
int argc,
char** argv,
size_t* count,
1596 int detect_status = 0;
1599 memcpy(largs, global_cmd_args,
sizeof(global_cmd_args));
1601 flags = COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_SILENCE_PARSER;
1602 flags |= COMMAND_LINE_SIGIL_SLASH | COMMAND_LINE_SIGIL_PLUS_MINUS;
1606 flags |= COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
1611 CommandLineClearArgumentsA(largs);
1612 status = CommandLineParseArgumentsA(argc, argv, largs, flags,
nullptr,
1613 freerdp_detect_command_line_pre_filter,
nullptr);
1622 if (!(arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
1626 }
while ((arg = CommandLineFindNextArgumentA(arg)) !=
nullptr);
1628 return detect_status;
1631static int freerdp_detect_posix_style_command_line_syntax(
int argc,
char** argv,
size_t* count,
1636 int detect_status = 0;
1639 memcpy(largs, global_cmd_args,
sizeof(global_cmd_args));
1641 flags = COMMAND_LINE_SEPARATOR_SPACE | COMMAND_LINE_SILENCE_PARSER;
1642 flags |= COMMAND_LINE_SIGIL_DASH | COMMAND_LINE_SIGIL_DOUBLE_DASH;
1643 flags |= COMMAND_LINE_SIGIL_ENABLE_DISABLE;
1647 flags |= COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
1652 CommandLineClearArgumentsA(largs);
1653 status = CommandLineParseArgumentsA(argc, argv, largs, flags,
nullptr,
1654 freerdp_detect_command_line_pre_filter,
nullptr);
1663 if (!(arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
1667 }
while ((arg = CommandLineFindNextArgumentA(arg)) !=
nullptr);
1669 return detect_status;
1672static BOOL freerdp_client_detect_command_line(
int argc,
char** argv, DWORD* flags)
1674 size_t posix_cli_count = 0;
1675 size_t windows_cli_count = 0;
1676 const BOOL ignoreUnknown = TRUE;
1677 const int windows_cli_status = freerdp_detect_windows_style_command_line_syntax(
1678 argc, argv, &windows_cli_count, ignoreUnknown);
1679 const int posix_cli_status =
1680 freerdp_detect_posix_style_command_line_syntax(argc, argv, &posix_cli_count, ignoreUnknown);
1683 *flags = COMMAND_LINE_SEPARATOR_SPACE;
1684 *flags |= COMMAND_LINE_SIGIL_DASH | COMMAND_LINE_SIGIL_DOUBLE_DASH;
1685 *flags |= COMMAND_LINE_SIGIL_ENABLE_DISABLE;
1687 if (posix_cli_status <= COMMAND_LINE_STATUS_PRINT)
1691 if ((windows_cli_count && (windows_cli_count >= posix_cli_count)) ||
1692 (windows_cli_status <= COMMAND_LINE_STATUS_PRINT))
1694 windows_cli_count = 1;
1695 *flags = COMMAND_LINE_SEPARATOR_COLON;
1696 *flags |= COMMAND_LINE_SIGIL_SLASH | COMMAND_LINE_SIGIL_PLUS_MINUS;
1699 WLog_DBG(TAG,
"windows: %d/%" PRIuz
" posix: %d/%" PRIuz
"", windows_cli_status,
1700 windows_cli_count, posix_cli_status, posix_cli_count);
1701 if ((posix_cli_count == 0) && (windows_cli_count == 0))
1703 if ((posix_cli_status == COMMAND_LINE_ERROR) && (windows_cli_status == COMMAND_LINE_ERROR))
1709int freerdp_client_settings_command_line_status_print(rdpSettings* settings,
int status,
int argc,
1712 return freerdp_client_settings_command_line_status_print_ex(settings, status, argc, argv,
1716static void freerdp_client_print_keyboard_type_list(
const char* msg, DWORD type)
1720 layouts = freerdp_keyboard_get_layouts(type, &count);
1722 printf(
"\n%s\n", msg);
1724 for (
size_t x = 0; x < count; x++)
1727 printf(
"0x%08" PRIX32
"\t%s\n", layout->code, layout->name);
1730 freerdp_keyboard_layouts_free(layouts, count);
1733static void freerdp_client_print_keyboard_list(
void)
1735 freerdp_client_print_keyboard_type_list(
"Keyboard Layouts", RDP_KEYBOARD_LAYOUT_TYPE_STANDARD);
1736 freerdp_client_print_keyboard_type_list(
"Keyboard Layout Variants",
1737 RDP_KEYBOARD_LAYOUT_TYPE_VARIANT);
1738 freerdp_client_print_keyboard_type_list(
"Keyboard Layout Variants",
1739 RDP_KEYBOARD_LAYOUT_TYPE_IME);
1742static void freerdp_client_print_timezone_list(
void)
1746 while (EnumDynamicTimeZoneInformation(index++, &info) != ERROR_NO_MORE_ITEMS)
1748 char TimeZoneKeyName[ARRAYSIZE(info.TimeZoneKeyName) + 1] = WINPR_C_ARRAY_INIT;
1750 (void)ConvertWCharNToUtf8(info.TimeZoneKeyName, ARRAYSIZE(info.TimeZoneKeyName),
1751 TimeZoneKeyName, ARRAYSIZE(TimeZoneKeyName));
1752 printf(
"%" PRIu32
": '%s'\n", index, TimeZoneKeyName);
1756static void freerdp_client_print_tune_list(
const rdpSettings* settings)
1760 for (SSIZE_T x = 0; x < FreeRDP_Settings_StableAPI_MAX; x++)
1768 case RDP_SETTINGS_TYPE_BOOL:
1769 printf(
"%" PRIdz
"\t%50s\tBOOL\t%s\n", x, name,
1774 case RDP_SETTINGS_TYPE_UINT16:
1775 printf(
"%" PRIdz
"\t%50s\tUINT16\t%" PRIu16
"\n", x, name,
1778 case RDP_SETTINGS_TYPE_INT16:
1779 printf(
"%" PRIdz
"\t%50s\tINT16\t%" PRId16
"\n", x, name,
1782 case RDP_SETTINGS_TYPE_UINT32:
1783 printf(
"%" PRIdz
"\t%50s\tUINT32\t%" PRIu32
"\n", x, name,
1786 case RDP_SETTINGS_TYPE_INT32:
1787 printf(
"%" PRIdz
"\t%50s\tINT32\t%" PRId32
"\n", x, name,
1790 case RDP_SETTINGS_TYPE_UINT64:
1791 printf(
"%" PRIdz
"\t%50s\tUINT64\t%" PRIu64
"\n", x, name,
1794 case RDP_SETTINGS_TYPE_INT64:
1795 printf(
"%" PRIdz
"\t%50s\tINT64\t%" PRId64
"\n", x, name,
1798 case RDP_SETTINGS_TYPE_STRING:
1799 printf(
"%" PRIdz
"\t%50s\tSTRING\t%s"
1804 case RDP_SETTINGS_TYPE_POINTER:
1805 printf(
"%" PRIdz
"\t%50s\tPOINTER\t%p"
1817static int evaluate_result(
int argc,
char* argv[],
int rc, rdpSettings* settings,
1820 WINPR_ASSERT(settings);
1821 WINPR_ASSERT(largs);
1823 if (rc != COMMAND_LINE_STATUS_PRINT)
1825 freerdp_client_print_command_line_usage(argc, argv);
1832 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
1834 if (option_equals(
"timezones", arg->Value))
1835 freerdp_client_print_timezone_list();
1836 else if (option_equals(
"tune", arg->Value))
1837 freerdp_client_print_tune_list(settings);
1838 else if (option_equals(
"kbd", arg->Value))
1839 freerdp_client_print_keyboard_list();
1840 else if (option_starts_with(
"kbd-lang", arg->Value))
1842 const char* val =
nullptr;
1843 if (option_starts_with(
"kbd-lang:", arg->Value))
1844 val = &arg->Value[9];
1845 else if (!option_equals(
"kbd-lang", arg->Value))
1846 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1848 if (val && strchr(val,
','))
1849 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1850 freerdp_client_print_codepages(val);
1852 else if (option_equals(
"kbd-scancode", arg->Value))
1853 freerdp_client_print_scancodes();
1854 else if (option_equals(
"monitor", arg->Value))
1857 return COMMAND_LINE_ERROR;
1859 else if (option_starts_with(
"smartcard", arg->Value))
1862 if (option_starts_with(
"smartcard:", arg->Value))
1864 else if (!option_equals(
"smartcard", arg->Value))
1865 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1869 const char* sub = strchr(arg->Value,
':') + 1;
1870 const CmdLineSubOptions options[] = {
1871 {
"pkinit-anchors:", FreeRDP_PkinitAnchors, CMDLINE_SUBOPTION_STRING,
nullptr },
1872 {
"pkcs11-module:", FreeRDP_Pkcs11Module, CMDLINE_SUBOPTION_STRING,
nullptr }
1877 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"smartcard", sub, &count);
1879 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1882 CommandLineParserFree(ptr);
1883 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1886 for (
size_t x = 1; x < count; x++)
1888 const char* cur = ptr[x];
1889 if (!parseSubOptions(settings, options, ARRAYSIZE(options), cur))
1891 CommandLineParserFree(ptr);
1892 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1896 CommandLineParserFree(ptr);
1899 freerdp_smartcard_list(settings);
1903 freerdp_client_print_command_line_usage(argc, argv);
1904 return COMMAND_LINE_ERROR;
1907#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
1908 arg = CommandLineFindArgumentA(largs,
"tune-list");
1911 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
1913 WLog_WARN(TAG,
"Option /tune-list is deprecated, use /list:tune instead");
1914 freerdp_client_print_tune_list(settings);
1917 arg = CommandLineFindArgumentA(largs,
"kbd-lang-list");
1920 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
1922 WLog_WARN(TAG,
"Option /kbd-lang-list is deprecated, use /list:kbd-lang instead");
1923 freerdp_client_print_codepages(arg->Value);
1926 arg = CommandLineFindArgumentA(largs,
"kbd-list");
1929 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1931 WLog_WARN(TAG,
"Option /kbd-list is deprecated, use /list:kbd instead");
1932 freerdp_client_print_keyboard_list();
1935 arg = CommandLineFindArgumentA(largs,
"monitor-list");
1938 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1940 WLog_WARN(TAG,
"Option /monitor-list is deprecated, use /list:monitor instead");
1942 return COMMAND_LINE_ERROR;
1945 arg = CommandLineFindArgumentA(largs,
"smartcard-list");
1948 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1950 WLog_WARN(TAG,
"Option /smartcard-list is deprecated, use /list:smartcard instead");
1951 freerdp_smartcard_list(settings);
1954 arg = CommandLineFindArgumentA(largs,
"kbd-scancode-list");
1957 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1959 WLog_WARN(TAG,
"Option /kbd-scancode-list is deprecated, use /list:kbd-scancode instead");
1960 freerdp_client_print_scancodes();
1961 return COMMAND_LINE_STATUS_PRINT;
1964 return COMMAND_LINE_STATUS_PRINT;
1967int freerdp_client_settings_command_line_status_print_ex(rdpSettings* settings,
int status,
1968 int argc,
char** argv,
1971 if (status == COMMAND_LINE_STATUS_PRINT_VERSION)
1973 freerdp_client_print_version();
1977 if (status == COMMAND_LINE_STATUS_PRINT_BUILDCONFIG)
1979 freerdp_client_print_version_ex(argc, argv);
1980 freerdp_client_print_buildconfig_ex(argc, argv);
1983 else if (status == COMMAND_LINE_STATUS_PRINT)
1986 COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_SIGIL_PLUS_MINUS | COMMAND_LINE_SIGIL_SLASH;
1988 size_t customcount = 0;
1991 while (cur && cur->Name)
1997 size_t globalcount = 0;
2000 while (cur && cur->Name)
2016 CommandLineParseArgumentsA(argc, argv, largs, flags,
nullptr,
nullptr,
nullptr);
2017 status = evaluate_result(argc, argv, rc, settings, largs);
2021 else if (status == COMMAND_LINE_STATUS_PRINT_HELP)
2023 freerdp_client_print_command_line_help_ex(argc, argv, custom);
2026 else if (status < 0)
2028 freerdp_client_print_command_line_usage(argc, argv);
2033 if (status <= COMMAND_LINE_STATUS_PRINT && status >= COMMAND_LINE_STATUS_PRINT_LAST)
2046static BOOL parseSizeValue(
const char* input,
unsigned long* v1,
unsigned long* v2)
2048 const char* xcharpos =
nullptr;
2049 char* endPtr =
nullptr;
2050 unsigned long v = 0;
2052 v = strtoul(input, &endPtr, 10);
2054 if ((v == 0 || v == ULONG_MAX) && (errno != 0))
2060 xcharpos = strchr(input,
'x');
2062 if (!xcharpos || xcharpos != endPtr)
2066 v = strtoul(xcharpos + 1, &endPtr, 10);
2068 if ((v == 0 || v == ULONG_MAX) && (errno != 0))
2071 if (*endPtr !=
'\0')
2083 const char* arguments[] = {
"network",
"gfx",
"rfx",
"bpp" };
2084 WINPR_ASSERT(settings);
2090 for (
size_t x = 0; x < ARRAYSIZE(arguments); x++)
2092 const char* arg = arguments[x];
2094 if (p && (p->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
2098 return freerdp_set_connection_type(settings, CONNECTION_TYPE_AUTODETECT);
2101static BOOL setSmartcardEmulation(WINPR_ATTR_UNUSED
const char* value, rdpSettings* settings)
2106const char* option_starts_with(
const char* what,
const char* val)
2110 const size_t wlen = strlen(what);
2112 if (_strnicmp(what, val, wlen) != 0)
2117BOOL option_ends_with(
const char* str,
const char* ext)
2121 const size_t strLen = strlen(str);
2122 const size_t extLen = strlen(ext);
2124 if (strLen < extLen)
2127 return _strnicmp(&str[strLen - extLen], ext, extLen) == 0;
2130BOOL option_equals(
const char* what,
const char* val)
2134 return _stricmp(what, val) == 0;
2143} PARSE_ON_OFF_RESULT;
2145static PARSE_ON_OFF_RESULT parse_on_off_option(
const char* value)
2147 WINPR_ASSERT(value);
2148 const char* sep = strchr(value,
':');
2151 if (option_equals(
"on", &sep[1]))
2153 if (option_equals(
"off", &sep[1]))
2162 CLIP_DIR_PARSE_LOCAL,
2163 CLIP_DIR_PARSE_REMOTE,
2165} PARSE_CLIP_DIR_RESULT;
2167static PARSE_CLIP_DIR_RESULT parse_clip_direciton_to_option(
const char* value)
2169 WINPR_ASSERT(value);
2170 const char* sep = strchr(value,
':');
2172 return CLIP_DIR_PARSE_FAIL;
2173 if (option_equals(
"all", &sep[1]))
2174 return CLIP_DIR_PARSE_ALL;
2175 if (option_equals(
"off", &sep[1]))
2176 return CLIP_DIR_PARSE_OFF;
2177 if (option_equals(
"local", &sep[1]))
2178 return CLIP_DIR_PARSE_LOCAL;
2179 if (option_equals(
"remote", &sep[1]))
2180 return CLIP_DIR_PARSE_REMOTE;
2181 return CLIP_DIR_PARSE_FAIL;
2184static int parse_tls_ciphers(rdpSettings* settings,
const char* Value)
2186 const char* ciphers =
nullptr;
2188 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2190 if (option_equals(Value,
"netmon"))
2192 ciphers =
"ALL:!ECDH:!ADH:!DHE";
2194 else if (option_equals(Value,
"ma"))
2196 ciphers =
"AES128-SHA";
2204 return COMMAND_LINE_ERROR_MEMORY;
2208static int parse_tls_seclevel(rdpSettings* settings,
const char* Value)
2212 if (!value_to_int(Value, &val, 0, 5))
2213 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2216 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2220static int parse_tls_secrets_file(rdpSettings* settings,
const char* Value)
2223 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2226 return COMMAND_LINE_ERROR_MEMORY;
2230static int parse_tls_enforce(rdpSettings* settings,
const char* Value)
2232 UINT16 version = TLS1_2_VERSION;
2241 const struct map_t map[] = { {
"1.0", TLS1_VERSION },
2242 {
"1.1", TLS1_1_VERSION },
2243 {
"1.2", TLS1_2_VERSION }
2244#if defined(TLS1_3_VERSION)
2246 {
"1.3", TLS1_3_VERSION }
2250 const struct map_t* found =
nullptr;
2251 for (
size_t x = 0; x < ARRAYSIZE(map); x++)
2253 const struct map_t* cur = &map[x];
2254 if (option_equals(cur->name, Value))
2262 version = found->version;
2266 const long v = strtol(Value,
nullptr, 0);
2268 if ((v < -1) || ((v == LONG_MAX) && (errno != 0)) || (v > UINT16_MAX))
2269 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2271 version = WINPR_ASSERTING_INT_CAST(UINT16, v);
2277 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2283 int rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2284 CommandLineSwitchStart(arg) CommandLineSwitchCase(arg,
"tls")
2286 if (option_starts_with(
"ciphers:", arg->Value))
2287 rc = fail_at(arg, parse_tls_ciphers(settings, &arg->Value[8]));
2288 else if (option_starts_with(
"seclevel:", arg->Value))
2289 rc = fail_at(arg, parse_tls_seclevel(settings, &arg->Value[9]));
2290 else if (option_starts_with(
"secrets-file:", arg->Value))
2291 rc = fail_at(arg, parse_tls_secrets_file(settings, &arg->Value[13]));
2292 else if (option_starts_with(
"enforce:", arg->Value))
2293 rc = fail_at(arg, parse_tls_enforce(settings, &arg->Value[8]));
2296#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
2297 CommandLineSwitchCase(arg,
"tls-ciphers")
2299 WLog_WARN(TAG,
"Option /tls-ciphers is deprecated, use /tls:ciphers instead");
2300 rc = fail_at(arg, parse_tls_ciphers(settings, arg->Value));
2302 CommandLineSwitchCase(arg,
"tls-seclevel")
2304 WLog_WARN(TAG,
"Option /tls-seclevel is deprecated, use /tls:seclevel instead");
2305 rc = fail_at(arg, parse_tls_seclevel(settings, arg->Value));
2307 CommandLineSwitchCase(arg,
"tls-secrets-file")
2309 WLog_WARN(TAG,
"Option /tls-secrets-file is deprecated, use /tls:secrets-file instead");
2310 rc = fail_at(arg, parse_tls_secrets_file(settings, arg->Value));
2312 CommandLineSwitchCase(arg,
"enforce-tlsv1_2")
2314 WLog_WARN(TAG,
"Option /enforce-tlsv1_2 is deprecated, use /tls:enforce:1.2 instead");
2315 rc = fail_at(arg, parse_tls_enforce(settings,
"1.2"));
2318 CommandLineSwitchDefault(arg)
2321 CommandLineSwitchEnd(arg)
2328 WINPR_ASSERT(settings);
2332 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2333 for (
size_t x = 0; x < count; x++)
2336 larg.Value = ptr[x];
2338 int rc = parse_tls_cipher_options(settings, &larg);
2341 CommandLineParserFree(ptr);
2345 CommandLineParserFree(ptr);
2351 WINPR_ASSERT(settings);
2355 return COMMAND_LINE_ERROR;
2359 int rc = CHANNEL_RC_OK;
2361 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2362 if (!ptr || (count == 0))
2363 rc = COMMAND_LINE_ERROR;
2366 BOOL GfxH264 = FALSE;
2367 BOOL GfxAVC444 = FALSE;
2368 BOOL RemoteFxCodec = FALSE;
2369 BOOL GfxProgressive = FALSE;
2370 BOOL codecSelected = FALSE;
2372 for (
size_t x = 0; x < count; x++)
2374 const char* val = ptr[x];
2376 if (option_starts_with(
"AVC444", val))
2378 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2379 if (bval == PARSE_FAIL)
2380 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2382 GfxAVC444 = bval != PARSE_OFF;
2383 codecSelected = TRUE;
2385 else if (option_starts_with(
"AVC420", val))
2387 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2388 if (bval == PARSE_FAIL)
2389 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2391 GfxH264 = bval != PARSE_OFF;
2392 codecSelected = TRUE;
2396 if (option_starts_with(
"RFX", val))
2398 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2399 if (bval == PARSE_FAIL)
2400 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2402 RemoteFxCodec = bval != PARSE_OFF;
2403 codecSelected = TRUE;
2405 else if (option_starts_with(
"progressive", val))
2407 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2408 if (bval == PARSE_FAIL)
2409 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2411 GfxProgressive = bval != PARSE_OFF;
2412 codecSelected = TRUE;
2414 else if (option_starts_with(
"mask:", val))
2417 const char* uv = &val[5];
2418 if (!value_to_uint(uv, &v, 0, UINT32_MAX))
2419 rc = COMMAND_LINE_ERROR;
2424 rc = COMMAND_LINE_ERROR;
2427 else if (option_starts_with(
"small-cache", val))
2429 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2430 if (bval == PARSE_FAIL)
2431 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2434 rc = COMMAND_LINE_ERROR;
2436 else if (option_starts_with(
"thin-client", val))
2438 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2439 if (bval == PARSE_FAIL)
2440 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2443 rc = COMMAND_LINE_ERROR;
2444 if ((rc == CHANNEL_RC_OK) && (bval > 0))
2448 rc = COMMAND_LINE_ERROR;
2451 else if (option_starts_with(
"frame-ack", val))
2453 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2454 if (bval == PARSE_FAIL)
2455 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2458 rc = COMMAND_LINE_ERROR;
2460#if defined(WITH_GFX_AV1)
2461 else if (option_starts_with(
"AV1", val))
2463 uint32_t profile = 1;
2464 BOOL enabled = FALSE;
2465 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2466 if (bval == PARSE_FAIL)
2468 if (_stricmp(
"av1:i420", val) == 0)
2473 else if (_stricmp(
"av1:i444", val) == 0)
2479 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2482 enabled = bval == PARSE_ON;
2484 if (enabled || (bval != PARSE_FAIL))
2487 rc = COMMAND_LINE_ERROR;
2490 rc = COMMAND_LINE_ERROR;
2495 rc = COMMAND_LINE_ERROR;
2498 if ((rc == CHANNEL_RC_OK) && codecSelected)
2501 rc = COMMAND_LINE_ERROR;
2503 rc = COMMAND_LINE_ERROR;
2505 rc = COMMAND_LINE_ERROR;
2507 rc = COMMAND_LINE_ERROR;
2509 rc = COMMAND_LINE_ERROR;
2512 CommandLineParserFree(ptr);
2513 if (rc != CHANNEL_RC_OK)
2516 return CHANNEL_RC_OK;
2519static int parse_kbd_layout(rdpSettings* settings,
const char* value)
2521 WINPR_ASSERT(settings);
2522 WINPR_ASSERT(value);
2526 const BOOL isInt = value_to_int(value, &ival, 1, UINT32_MAX);
2529 ival = freerdp_map_keyboard_layout_name_to_id(value);
2533 WLog_ERR(TAG,
"Could not identify keyboard layout: %s", value);
2534 WLog_ERR(TAG,
"Use /list:kbd to list available layouts");
2535 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2542 rc = COMMAND_LINE_ERROR;
2547#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
2550 WINPR_ASSERT(settings);
2554 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2556 return COMMAND_LINE_ERROR;
2558 if (option_equals(arg->Value,
"rfx"))
2561 return COMMAND_LINE_ERROR;
2563 else if (option_equals(arg->Value,
"nsc"))
2566 return COMMAND_LINE_ERROR;
2569#if defined(WITH_JPEG)
2570 else if (option_equals(arg->Value,
"jpeg"))
2573 return COMMAND_LINE_ERROR;
2578 return COMMAND_LINE_ERROR;
2587static BOOL check_kbd_remap_valid(
const char* token)
2592 WINPR_ASSERT(token);
2594 if (strlen(token) > 10)
2597 if (!freerdp_extract_key_value(token, &key, &value))
2599 WLog_WARN(TAG,
"/kbd:remap invalid entry '%s'", token);
2607 WINPR_ASSERT(settings);
2611 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2613 return COMMAND_LINE_ERROR_MEMORY;
2614 char* p = strchr(arg->Value,
'[');
2619 const char scheme[] =
"://";
2620 const char* val = strstr(arg->Value, scheme);
2622 val += strnlen(scheme,
sizeof(scheme));
2625 p = strchr(val,
':');
2632 if (!value_to_int(&p[1], &lval, 1, UINT16_MAX))
2633 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2635 length = (size_t)(p - arg->Value);
2637 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2640 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2645 return COMMAND_LINE_ERROR_MEMORY;
2651 char* p2 = strchr(arg->Value,
']');
2655 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2657 length = (size_t)(p2 - p);
2659 return COMMAND_LINE_ERROR_MEMORY;
2661 if (*(p2 + 1) ==
':')
2665 if (!value_to_int(&p2[2], &val, 0, UINT16_MAX))
2666 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2669 return COMMAND_LINE_ERROR;
2672 printf(
"hostname %s port %" PRIu32
"\n",
2681 WINPR_ASSERT(settings);
2685 char* cur = arg->Value;
2687 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2693 char* next = strchr(cur,
',');
2701 if (option_equals(
"fqdn", cur))
2703 else if (option_equals(
"ip", cur))
2705 else if (option_equals(
"netbios", cur))
2708 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2711 mask = (mask & 0x07);
2712 value |= mask << (count * 3);
2714 }
while (cur !=
nullptr);
2717 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2720 return COMMAND_LINE_ERROR;
2725static int parse_prevent_session_lock_options(rdpSettings* settings,
2728 WINPR_ASSERT(settings);
2732 return COMMAND_LINE_ERROR_MEMORY;
2734 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2738 if (!value_to_int(arg->Value, &val, 1, UINT32_MAX))
2739 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2742 return COMMAND_LINE_ERROR_MEMORY;
2750 WINPR_ASSERT(settings);
2754 return COMMAND_LINE_ERROR;
2761 return COMMAND_LINE_ERROR;
2763 return COMMAND_LINE_ERROR;
2765 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2768 return COMMAND_LINE_ERROR;
2771 return COMMAND_LINE_ERROR_MEMORY;
2779 WINPR_ASSERT(settings);
2783 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2784 char* p = strchr(arg->Value,
'x');
2788 unsigned long w = 0;
2789 unsigned long h = 0;
2791 if (!parseSizeValue(arg->Value, &w, &h) || (w > UINT16_MAX) || (h > UINT16_MAX))
2792 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2795 return COMMAND_LINE_ERROR;
2797 return COMMAND_LINE_ERROR;
2801 char* str = _strdup(arg->Value);
2803 return COMMAND_LINE_ERROR_MEMORY;
2805 p = strchr(str,
'%');
2809 BOOL partial = FALSE;
2811 status = COMMAND_LINE_ERROR;
2838 if (!value_to_int(str, &val, 0, 100))
2840 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2860 WINPR_ASSERT(settings);
2863 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2866 UINT32* MonitorIds =
nullptr;
2867 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2870 return COMMAND_LINE_ERROR_MEMORY;
2877 CommandLineParserFree(ptr);
2878 return COMMAND_LINE_ERROR;
2881 MonitorIds = freerdp_settings_get_pointer_array_writable(settings, FreeRDP_MonitorIds, 0);
2882 for (UINT32 i = 0; i < count; i++)
2886 if (!value_to_int(ptr[i], &val, 0, UINT16_MAX))
2887 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2889 MonitorIds[i] = (UINT32)val;
2892 CommandLineParserFree(ptr);
2898static int parse_dynamic_resolution_options(rdpSettings* settings,
2901 WINPR_ASSERT(settings);
2904 const BOOL val = arg->Value !=
nullptr;
2908 WLog_ERR(TAG,
"Smart sizing and dynamic resolution are mutually exclusive options");
2909 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2913 return COMMAND_LINE_ERROR;
2915 return COMMAND_LINE_ERROR;
2922 WINPR_ASSERT(settings);
2927 WLog_ERR(TAG,
"Smart sizing and dynamic resolution are mutually exclusive options");
2928 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2932 return COMMAND_LINE_ERROR;
2936 unsigned long w = 0;
2937 unsigned long h = 0;
2939 if (!parseSizeValue(arg->Value, &w, &h) || (w > UINT16_MAX) || (h > UINT16_MAX))
2940 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2943 return COMMAND_LINE_ERROR;
2945 return COMMAND_LINE_ERROR;
2952 WINPR_ASSERT(settings);
2957 if (!value_to_int(arg->Value, &val, 0, UINT32_MAX))
2958 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2968 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2972 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2979 WINPR_ASSERT(settings);
2982 int rc = CHANNEL_RC_OK;
2984 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2985 if (!ptr || (count == 0))
2986 rc = COMMAND_LINE_ERROR;
2989 for (
size_t x = 0; x < count; x++)
2991 const char* val = ptr[x];
2993 if (option_starts_with(
"remap:", val))
2996 char* now = _strdup(&val[6]);
3001 if (!check_kbd_remap_valid(now))
3002 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3005 const size_t olen = strlen(old);
3006 const size_t alen = strlen(now);
3007 const size_t tlen = olen + alen + 2;
3008 char* tmp = calloc(tlen,
sizeof(
char));
3010 rc = COMMAND_LINE_ERROR_MEMORY;
3012 (
void)_snprintf(tmp, tlen,
"%s,%s", old, now);
3020 rc = COMMAND_LINE_ERROR;
3024 else if (option_starts_with(
"layout:", val))
3026 rc = parse_kbd_layout(settings, &val[7]);
3028 else if (option_starts_with(
"lang:", val))
3031 const BOOL isInt = value_to_int(&val[5], &ival, 1, UINT32_MAX);
3033 ival = freerdp_get_locale_id_from_string(&val[5]);
3036 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3039 rc = COMMAND_LINE_ERROR;
3041 else if (option_starts_with(
"type:", val))
3044 const BOOL isInt = value_to_int(&val[5], &ival, 1, UINT32_MAX);
3046 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3048 rc = COMMAND_LINE_ERROR;
3050 else if (option_starts_with(
"subtype:", val))
3053 const BOOL isInt = value_to_int(&val[8], &ival, 1, UINT32_MAX);
3055 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3058 rc = COMMAND_LINE_ERROR;
3060 else if (option_starts_with(
"fn-key:", val))
3063 const BOOL isInt = value_to_int(&val[7], &ival, 1, UINT32_MAX);
3065 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3068 rc = COMMAND_LINE_ERROR;
3070 else if (option_starts_with(
"unicode", val))
3072 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
3073 if (bval == PARSE_FAIL)
3074 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3077 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3079 else if (option_starts_with(
"pipe:", val))
3082 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3084 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3086#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
3087 else if (count == 1)
3090 rc = parse_kbd_layout(settings, val);
3094 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3100 CommandLineParserFree(ptr);
3106 WINPR_ASSERT(settings);
3111 return COMMAND_LINE_ERROR_MEMORY;
3113 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
3115 const char* cur = arg->Value;
3118 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3120 if (!proxy_parse_uri(settings, cur))
3121 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3125 WLog_ERR(TAG,
"Option http-proxy needs argument.");
3126 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3133 WINPR_ASSERT(settings);
3136 BOOL failed = FALSE;
3138 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3143 BOOL modernsyntax = FALSE;
3144 BOOL oldsyntax = FALSE;
3145 for (
size_t x = 0; (x < count) && !failed; x++)
3147 const char* carg = ptr[x];
3148 if (option_starts_with(
"file:", carg))
3150 const char* val = &carg[5];
3155 modernsyntax = TRUE;
3157 else if (option_equals(
"replay", carg))
3164 else if (option_equals(
"record", carg))
3171 else if (option_equals(
"nodelay", carg))
3178 modernsyntax = TRUE;
3193 if (oldsyntax && (count != 2))
3196 CommandLineParserFree(ptr);
3198 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3204 WINPR_ASSERT(settings);
3207 if (arg->Value == BoolValueTrue || arg->Value == BoolValueFalse)
3210 (arg->Value == BoolValueTrue)))
3211 return COMMAND_LINE_ERROR;
3217 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3218 for (
size_t x = 0; (x < count) && (rc == 0); x++)
3220 const char* usesel =
"use-selection:";
3222 const char* cur = ptr[x];
3223 if (option_starts_with(usesel, cur))
3225 const char* val = &cur[strlen(usesel)];
3227 rc = COMMAND_LINE_ERROR_MEMORY;
3229 return COMMAND_LINE_ERROR;
3231 else if (option_starts_with(
"direction-to", cur))
3235 (uint32_t)~(CLIPRDR_FLAG_LOCAL_TO_REMOTE | CLIPRDR_FLAG_REMOTE_TO_LOCAL);
3236 const PARSE_CLIP_DIR_RESULT bval = parse_clip_direciton_to_option(cur);
3240 case CLIP_DIR_PARSE_ALL:
3241 bflags |= CLIPRDR_FLAG_LOCAL_TO_REMOTE | CLIPRDR_FLAG_REMOTE_TO_LOCAL;
3243 case CLIP_DIR_PARSE_LOCAL:
3244 bflags |= CLIPRDR_FLAG_REMOTE_TO_LOCAL;
3246 case CLIP_DIR_PARSE_REMOTE:
3247 bflags |= CLIPRDR_FLAG_LOCAL_TO_REMOTE;
3249 case CLIP_DIR_PARSE_OFF:
3251 case CLIP_DIR_PARSE_FAIL:
3253 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3259 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3261 else if (option_starts_with(
"files-to", cur))
3265 (uint32_t)~(CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES |
3266 CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES);
3267 const PARSE_CLIP_DIR_RESULT bval = parse_clip_direciton_to_option(cur);
3271 case CLIP_DIR_PARSE_ALL:
3273 CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES | CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES;
3275 case CLIP_DIR_PARSE_LOCAL:
3276 bflags |= CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES;
3278 case CLIP_DIR_PARSE_REMOTE:
3279 bflags |= CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES;
3281 case CLIP_DIR_PARSE_OFF:
3283 case CLIP_DIR_PARSE_FAIL:
3285 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3291 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3294 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3296 CommandLineParserFree(ptr);
3306 WINPR_ASSERT(settings);
3311 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3313 if (strcmp(arg->Value,
"none") == 0)
3314 val = AUDIO_MODE_NONE;
3315 else if (strcmp(arg->Value,
"redirect") == 0)
3316 val = AUDIO_MODE_REDIRECT;
3317 else if (strcmp(arg->Value,
"server") == 0)
3318 val = AUDIO_MODE_PLAY_ON_SERVER;
3319 else if (!value_to_int(arg->Value, &val, 0, UINT32_MAX))
3320 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3324 case AUDIO_MODE_REDIRECT:
3326 return COMMAND_LINE_ERROR;
3329 case AUDIO_MODE_PLAY_ON_SERVER:
3331 return COMMAND_LINE_ERROR;
3334 case AUDIO_MODE_NONE:
3336 return COMMAND_LINE_ERROR;
3338 return COMMAND_LINE_ERROR;
3342 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3349 WINPR_ASSERT(settings);
3352 UINT32 type = CONNECTION_TYPE_INVALID;
3354 if (option_equals(arg->Value,
"invalid"))
3355 type = CONNECTION_TYPE_INVALID;
3356 else if (option_equals(arg->Value,
"modem"))
3357 type = CONNECTION_TYPE_MODEM;
3358 else if (option_equals(arg->Value,
"broadband"))
3359 type = CONNECTION_TYPE_BROADBAND_HIGH;
3360 else if (option_equals(arg->Value,
"broadband-low"))
3361 type = CONNECTION_TYPE_BROADBAND_LOW;
3362 else if (option_equals(arg->Value,
"broadband-high"))
3363 type = CONNECTION_TYPE_BROADBAND_HIGH;
3364 else if (option_equals(arg->Value,
"wan"))
3365 type = CONNECTION_TYPE_WAN;
3366 else if (option_equals(arg->Value,
"lan"))
3367 type = CONNECTION_TYPE_LAN;
3368 else if ((option_equals(arg->Value,
"autodetect")) || (option_equals(arg->Value,
"auto")) ||
3369 (option_equals(arg->Value,
"detect")))
3371 type = CONNECTION_TYPE_AUTODETECT;
3377 if (!value_to_int(arg->Value, &val, 0, 7))
3378 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3383 if (!freerdp_set_connection_type(settings, type))
3384 return COMMAND_LINE_ERROR;
3390 WINPR_ASSERT(settings);
3394 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3396 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3398 FreeRDP_Settings_Keys_Bool singleOptionWithoutOnOff = FreeRDP_BOOL_UNUSED;
3399 for (
size_t x = 0; x < count; x++)
3401 const char* cur = ptr[x];
3402 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(cur);
3403 if (bval == PARSE_FAIL)
3405 CommandLineParserFree(ptr);
3406 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3409 const BOOL val = bval != PARSE_OFF;
3410 FreeRDP_Settings_Keys_Bool
id = FreeRDP_BOOL_UNUSED;
3411 if (option_starts_with(
"rdp", cur))
3413 id = FreeRDP_RdpSecurity;
3415 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3417 else if (option_starts_with(
"tls", cur))
3418 id = FreeRDP_TlsSecurity;
3419 else if (option_starts_with(
"nla", cur))
3420 id = FreeRDP_NlaSecurity;
3421 else if (option_starts_with(
"ext", cur))
3422 id = FreeRDP_ExtSecurity;
3423 else if (option_equals(
"aad", cur))
3424 id = FreeRDP_AadSecurity;
3427 WLog_ERR(TAG,
"unknown protocol security: %s", arg->Value);
3428 CommandLineParserFree(ptr);
3429 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3432 if ((bval == PARSE_NONE) && (count == 1))
3433 singleOptionWithoutOnOff = id;
3435 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3438 if (singleOptionWithoutOnOff != FreeRDP_BOOL_UNUSED)
3440 const FreeRDP_Settings_Keys_Bool options[] = { FreeRDP_AadSecurity,
3441 FreeRDP_UseRdpSecurityLayer,
3442 FreeRDP_RdpSecurity, FreeRDP_NlaSecurity,
3443 FreeRDP_TlsSecurity };
3445 for (
size_t i = 0; i < ARRAYSIZE(options); i++)
3448 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3452 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3453 if (singleOptionWithoutOnOff == FreeRDP_RdpSecurity)
3456 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3459 CommandLineParserFree(ptr);
3463static int parse_encryption_methods_options(rdpSettings* settings,
3466 WINPR_ASSERT(settings);
3469 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
3472 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3474 UINT32 EncryptionMethods = 0;
3475 for (UINT32 i = 0; i < count; i++)
3477 if (option_equals(ptr[i],
"40"))
3478 EncryptionMethods |= ENCRYPTION_METHOD_40BIT;
3479 else if (option_equals(ptr[i],
"56"))
3480 EncryptionMethods |= ENCRYPTION_METHOD_56BIT;
3481 else if (option_equals(ptr[i],
"128"))
3482 EncryptionMethods |= ENCRYPTION_METHOD_128BIT;
3483 else if (option_equals(ptr[i],
"FIPS"))
3484 EncryptionMethods |= ENCRYPTION_METHOD_FIPS;
3486 WLog_ERR(TAG,
"unknown encryption method '%s'", ptr[i]);
3490 return COMMAND_LINE_ERROR;
3491 CommandLineParserFree(ptr);
3498 WINPR_ASSERT(settings);
3503 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3504 for (
size_t x = 0; (x < count) && (rc == 0); x++)
3506 const char deny[] =
"deny";
3507 const char ignore[] =
"ignore";
3508 const char tofu[] =
"tofu";
3509 const char name[] =
"name:";
3510 const char fingerprints[] =
"fingerprint:";
3512 const char* cur = ptr[x];
3513 if (option_equals(deny, cur))
3516 return COMMAND_LINE_ERROR;
3518 else if (option_equals(ignore, cur))
3521 return COMMAND_LINE_ERROR;
3523 else if (option_equals(tofu, cur))
3526 return COMMAND_LINE_ERROR;
3528 else if (option_starts_with(name, cur))
3530 const char* val = &cur[strnlen(name,
sizeof(name))];
3532 rc = COMMAND_LINE_ERROR_MEMORY;
3534 else if (option_starts_with(fingerprints, cur))
3536 const char* val = &cur[strnlen(fingerprints,
sizeof(fingerprints))];
3539 rc = COMMAND_LINE_ERROR_MEMORY;
3542 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3544 CommandLineParserFree(ptr);
3551 WINPR_ASSERT(settings);
3555 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"mouse", arg->Value, &count);
3559 for (
size_t x = 1; x < count; x++)
3561 const char* cur = ptr[x];
3563 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(cur);
3564 if (bval == PARSE_FAIL)
3565 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3568 const BOOL val = bval != PARSE_OFF;
3570 if (option_starts_with(
"relative", cur))
3573 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3575 else if (option_starts_with(
"grab", cur))
3578 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3586 CommandLineParserFree(ptr);
3593 WINPR_ASSERT(settings);
3597 UINT32 Floatbar = 0x0017;
3601 char* start = arg->Value;
3606 start = strchr(start,
',');
3615 if (option_starts_with(
"sticky:", cur))
3619 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(cur);
3631 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3635 else if (option_starts_with(
"default:", cur))
3637 const char* val = cur + 8;
3640 if (option_equals(
"visible", val))
3642 else if (option_equals(
"hidden", val))
3645 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3648 else if (option_starts_with(
"show:", cur))
3650 const char* val = cur + 5;
3653 if (option_equals(
"always", val))
3655 else if (option_equals(
"fullscreen", val))
3657 else if (option_equals(
"window", val))
3660 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3663 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3667 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3673 WINPR_ASSERT(settings);
3676 BYTE* base64 =
nullptr;
3679 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3681 crypto_base64_decode((
const char*)(arg->Value), strlen(arg->Value), &base64, &length);
3687 return COMMAND_LINE_ERROR;
3691 WLog_ERR(TAG,
"reconnect-cookie: invalid base64 '%s'", arg->Value);
3698static BOOL set_monitor_override(rdpSettings* settings, uint64_t flag)
3700 const FreeRDP_Settings_Keys_UInt64 key = FreeRDP_MonitorOverrideFlags;
3708 WINPR_ASSERT(settings);
3713 if (!value_to_int(arg->Value, &val, 100, 180))
3714 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3722 return COMMAND_LINE_ERROR;
3724 return COMMAND_LINE_ERROR;
3725 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_DESKTOP_SCALE |
3726 FREERDP_MONITOR_OVERRIDE_DEVICE_SCALE))
3727 return fail_at(arg, COMMAND_LINE_ERROR);
3731 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3738 WINPR_ASSERT(settings);
3743 if (!value_to_int(arg->Value, &val, 100, 180))
3744 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3752 return COMMAND_LINE_ERROR;
3753 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_DEVICE_SCALE))
3754 return fail_at(arg, COMMAND_LINE_ERROR);
3758 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3765 WINPR_ASSERT(settings);
3771 return COMMAND_LINE_ERROR;
3773 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"smartcard-logon", arg->Value, &count);
3776 const CmdLineSubOptions opts[] = {
3777 {
"cert:", FreeRDP_SmartcardCertificate, CMDLINE_SUBOPTION_FILE,
3778 setSmartcardEmulation },
3779 {
"key:", FreeRDP_SmartcardPrivateKey, CMDLINE_SUBOPTION_FILE, setSmartcardEmulation },
3780 {
"pin:", FreeRDP_Password, CMDLINE_SUBOPTION_STRING,
nullptr },
3781 {
"csp:", FreeRDP_CspName, CMDLINE_SUBOPTION_STRING,
nullptr },
3782 {
"reader:", FreeRDP_ReaderName, CMDLINE_SUBOPTION_STRING,
nullptr },
3783 {
"card:", FreeRDP_CardName, CMDLINE_SUBOPTION_STRING,
nullptr },
3784 {
"container:", FreeRDP_ContainerName, CMDLINE_SUBOPTION_STRING,
nullptr }
3787 for (
size_t x = 1; x < count; x++)
3789 const char* cur = ptr[x];
3790 if (!parseSubOptions(settings, opts, ARRAYSIZE(opts), cur))
3792 CommandLineParserFree(ptr);
3793 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3797 CommandLineParserFree(ptr);
3803 WINPR_ASSERT(settings);
3807 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"tune", arg->Value, &count);
3809 return COMMAND_LINE_ERROR;
3810 for (
size_t x = 1; x < count; x++)
3812 const char* cur = ptr[x];
3813 char* sep = strchr(cur,
':');
3816 CommandLineParserFree(ptr);
3817 return COMMAND_LINE_ERROR;
3820 if (!freerdp_settings_set_value_for_name(settings, cur, sep))
3822 CommandLineParserFree(ptr);
3823 return COMMAND_LINE_ERROR;
3827 CommandLineParserFree(ptr);
3831static int parse_app_option_program(rdpSettings* settings,
const char* cmd)
3833 const FreeRDP_Settings_Keys_Bool ids[] = { FreeRDP_RemoteApplicationMode,
3834 FreeRDP_RemoteAppLanguageBarSupported,
3835 FreeRDP_Workarea, FreeRDP_DisableWallpaper,
3836 FreeRDP_DisableFullWindowDrag };
3839 return COMMAND_LINE_ERROR_MEMORY;
3841 for (
size_t y = 0; y < ARRAYSIZE(ids); y++)
3844 return COMMAND_LINE_ERROR;
3846 return CHANNEL_RC_OK;
3851 WINPR_ASSERT(settings);
3854 int rc = CHANNEL_RC_OK;
3856 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3857 if (!ptr || (count == 0))
3858 rc = COMMAND_LINE_ERROR;
3865 int (*fkt)(rdpSettings* settings,
const char* value);
3867 const struct app_map amap[] = {
3868 {
"tenantid:", FreeRDP_GatewayAvdAadtenantid,
nullptr },
3869 {
"ad:", FreeRDP_GatewayAzureActiveDirectory,
nullptr },
3870 {
"avd-access:", FreeRDP_GatewayAvdAccessAadFormat,
nullptr },
3871 {
"avd-token:", FreeRDP_GatewayAvdAccessTokenFormat,
nullptr },
3872 {
"avd-scope:", FreeRDP_GatewayAvdScope,
nullptr }
3875 for (
size_t x = 0; x < count; x++)
3877 BOOL handled = FALSE;
3878 const char* val = ptr[x];
3880 if (option_starts_with(
"use-tenantid", val))
3882 PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
3883 if (bval == PARSE_FAIL)
3885 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3893 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3899 for (
size_t y = 0; y < ARRAYSIZE(amap); y++)
3901 const struct app_map* cur = &amap[y];
3902 if (option_starts_with(cur->name, val))
3904 const char* xval = &val[strlen(cur->name)];
3906 rc = cur->fkt(settings, xval);
3910 if (!freerdp_settings_set_value_for_name(settings, name, xval))
3911 rc = COMMAND_LINE_ERROR_MEMORY;
3920 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3927 CommandLineParserFree(ptr);
3933 WINPR_ASSERT(settings);
3936 int rc = CHANNEL_RC_OK;
3938 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3939 if (!ptr || (count == 0))
3940 rc = COMMAND_LINE_ERROR;
3947 int (*fkt)(rdpSettings* settings,
const char* value);
3949 const struct app_map amap[] = {
3950 {
"program:", FreeRDP_RemoteApplicationProgram, parse_app_option_program },
3951 {
"workdir:", FreeRDP_RemoteApplicationWorkingDir,
nullptr },
3952 {
"name:", FreeRDP_RemoteApplicationName,
nullptr },
3953 {
"icon:", FreeRDP_RemoteApplicationIcon,
nullptr },
3954 {
"cmd:", FreeRDP_RemoteApplicationCmdLine,
nullptr },
3955 {
"file:", FreeRDP_RemoteApplicationFile,
nullptr },
3956 {
"guid:", FreeRDP_RemoteApplicationGuid,
nullptr },
3957 {
"hidef:", FreeRDP_HiDefRemoteApp,
nullptr }
3959 for (
size_t x = 0; x < count; x++)
3961 BOOL handled = FALSE;
3962 const char* val = ptr[x];
3964 for (
size_t y = 0; y < ARRAYSIZE(amap); y++)
3966 const struct app_map* cur = &amap[y];
3967 if (option_starts_with(cur->name, val))
3969 const char* xval = &val[strlen(cur->name)];
3971 rc = cur->fkt(settings, xval);
3975 if (!freerdp_settings_set_value_for_name(settings, name, xval))
3976 rc = COMMAND_LINE_ERROR_MEMORY;
3984#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
3985 if (!handled && (count == 1))
3988 rc = parse_app_option_program(settings, val);
3993 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
4000 CommandLineParserFree(ptr);
4006 WINPR_ASSERT(settings);
4009 int rc = CHANNEL_RC_OK;
4011 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
4012 if (!ptr || (count == 0))
4013 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
4015 for (
size_t x = 0; x < count; x++)
4017 const char* val = ptr[x];
4019 if (option_starts_with(
"codec:", val))
4022 rc = COMMAND_LINE_ERROR;
4023 else if (option_equals(arg->Value,
"rfx"))
4026 rc = COMMAND_LINE_ERROR;
4028 else if (option_equals(arg->Value,
"nsc"))
4031 rc = COMMAND_LINE_ERROR;
4034#if defined(WITH_JPEG)
4035 else if (option_equals(arg->Value,
"jpeg"))
4038 rc = COMMAND_LINE_ERROR;
4043 return COMMAND_LINE_ERROR;
4049 else if (option_starts_with(
"persist-file:", val))
4053 rc = COMMAND_LINE_ERROR_MEMORY;
4055 rc = COMMAND_LINE_ERROR;
4059 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
4060 if (bval == PARSE_FAIL)
4061 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
4064 if (option_starts_with(
"bitmap", val))
4068 rc = COMMAND_LINE_ERROR;
4070 else if (option_starts_with(
"glyph", val))
4073 bval != PARSE_OFF ? GLYPH_SUPPORT_FULL
4074 : GLYPH_SUPPORT_NONE))
4075 rc = COMMAND_LINE_ERROR;
4077 else if (option_starts_with(
"persist", val))
4081 rc = COMMAND_LINE_ERROR;
4083 else if (option_starts_with(
"offscreen", val))
4087 rc = COMMAND_LINE_ERROR;
4093 CommandLineParserFree(ptr);
4097static BOOL parse_gateway_host_option(rdpSettings* settings,
const char* host)
4099 WINPR_ASSERT(settings);
4102 char* name =
nullptr;
4104 if (!freerdp_parse_hostname(host, &name, &port))
4123static BOOL parse_gateway_cred_option(rdpSettings* settings,
const char* value,
4124 FreeRDP_Settings_Keys_String what)
4126 WINPR_ASSERT(settings);
4127 WINPR_ASSERT(value);
4131 case FreeRDP_GatewayUsername:
4132 if (!freerdp_parse_username_settings(value, settings, FreeRDP_GatewayUsername,
4133 FreeRDP_GatewayDomain))
4145static BOOL parse_gateway_type_option(rdpSettings* settings,
const char* value)
4149 WINPR_ASSERT(settings);
4150 WINPR_ASSERT(value);
4152 if (option_equals(value,
"rpc"))
4163 if (option_equals(value,
"http"))
4171 else if (option_equals(value,
"auto"))
4179 else if (option_equals(value,
"arm"))
4192static BOOL parse_gateway_usage_option(rdpSettings* settings,
const char* value)
4196 WINPR_ASSERT(settings);
4197 WINPR_ASSERT(value);
4199 if (option_equals(value,
"none"))
4200 type = TSC_PROXY_MODE_NONE_DIRECT;
4201 else if (option_equals(value,
"direct"))
4202 type = TSC_PROXY_MODE_DIRECT;
4203 else if (option_equals(value,
"detect"))
4204 type = TSC_PROXY_MODE_DETECT;
4205 else if (option_equals(value,
"default"))
4206 type = TSC_PROXY_MODE_DEFAULT;
4211 if (!value_to_int(value, &val, TSC_PROXY_MODE_NONE_DIRECT, TSC_PROXY_MODE_NONE_DETECT))
4213 type = WINPR_ASSERTING_INT_CAST(UINT32, val);
4219static char* unescape(
const char* str)
4221 char* copy = _strdup(str);
4225 bool escaped =
false;
4227 while (*str !=
'\0')
4255 char* argval =
nullptr;
4258 WINPR_ASSERT(settings);
4262 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
4271 BOOL allowHttpOpts = FALSE;
4272 for (
size_t x = 0; x < count; x++)
4274 BOOL validOption = FALSE;
4276 argval = unescape(ptr[x]);
4280 const char* gw = option_starts_with(
"g:", argval);
4283 if (!parse_gateway_host_option(settings, gw))
4286 allowHttpOpts = FALSE;
4289 const char* gu = option_starts_with(
"u:", argval);
4292 if (!parse_gateway_cred_option(settings, gu, FreeRDP_GatewayUsername))
4295 allowHttpOpts = FALSE;
4298 const char* gd = option_starts_with(
"d:", argval);
4301 if (!parse_gateway_cred_option(settings, gd, FreeRDP_GatewayDomain))
4304 allowHttpOpts = FALSE;
4307 const char* gp = option_starts_with(
"p:", argval);
4310 if (!parse_gateway_cred_option(settings, gp, FreeRDP_GatewayPassword))
4313 allowHttpOpts = FALSE;
4316 const char* gt = option_starts_with(
"type:", argval);
4319 if (!parse_gateway_type_option(settings, gt))
4325 const char* gat = option_starts_with(
"access-token:", argval);
4331 allowHttpOpts = FALSE;
4334 const char* bearer = option_starts_with(
"bearer:", argval);
4341 allowHttpOpts = FALSE;
4344 const char* gwurl = option_starts_with(
"url:", argval);
4352 allowHttpOpts = FALSE;
4355 const char* um = option_starts_with(
"usage-method:", argval);
4358 if (!parse_gateway_usage_option(settings, um))
4361 allowHttpOpts = FALSE;
4366 if (option_equals(argval,
"no-websockets"))
4373 else if (option_equals(argval,
"extauth-sspi-ntlm"))
4390 CommandLineParserFree(ptr);
4397 WINPR_ASSERT(value);
4403 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
4404 FillMemory(arg->Value, strlen(arg->Value),
'*');
4409 for (
size_t x = 0; x < ARRAYSIZE(credential_args); x++)
4411 const char* cred = credential_args[x];
4412 fill_credential_string(args, cred);
4416 if (arg && ((arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT) != 0))
4418 const char* gwcreds[] = {
"p:",
"access-token:" };
4419 char* saveptr =
nullptr;
4420 char* tok = strtok_s(arg->Value,
",", &saveptr);
4423 for (
size_t x = 0; x < ARRAYSIZE(gwcreds); x++)
4425 const char* opt = gwcreds[x];
4426 if (option_starts_with(opt, tok))
4428 char* val = &tok[strlen(opt)];
4429 FillMemory(val, strlen(val),
'*');
4432 tok = strtok_s(
nullptr,
",", &saveptr);
4437static int parse_command_line_option_uint32(rdpSettings* settings,
4439 FreeRDP_Settings_Keys_UInt32 key, LONGLONG min,
4444 if (!value_to_int(arg->Value, &val, min, max))
4445 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4448 return fail_at(arg, COMMAND_LINE_ERROR);
4452#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
4457 WINPR_ASSERT(settings);
4460 BOOL enable = arg->Value ? TRUE : FALSE;
4461 CommandLineSwitchStart(arg) CommandLineSwitchCase(arg,
"gfx-thin-client")
4463 WLog_WARN(TAG,
"/gfx-thin-client is deprecated, use /gfx:thin-client[:on|off] instead");
4465 return fail_at(arg, COMMAND_LINE_ERROR);
4470 return fail_at(arg, COMMAND_LINE_ERROR);
4474 return fail_at(arg, COMMAND_LINE_ERROR);
4476 CommandLineSwitchCase(arg,
"gfx-small-cache")
4478 WLog_WARN(TAG,
"/gfx-small-cache is deprecated, use /gfx:small-cache[:on|off] instead");
4480 return fail_at(arg, COMMAND_LINE_ERROR);
4484 return fail_at(arg, COMMAND_LINE_ERROR);
4486 CommandLineSwitchCase(arg,
"gfx-progressive")
4488 WLog_WARN(TAG,
"/gfx-progressive is deprecated, use /gfx:progressive[:on|off] instead");
4490 return fail_at(arg, COMMAND_LINE_ERROR);
4492 return fail_at(arg, COMMAND_LINE_ERROR);
4497 return fail_at(arg, COMMAND_LINE_ERROR);
4501 CommandLineSwitchCase(arg,
"gfx-h264")
4503 WLog_WARN(TAG,
"/gfx-h264 is deprecated, use /gfx:avc420 instead");
4504 int rc = parse_gfx_options(settings, arg);
4506 return fail_at(arg, rc);
4509 CommandLineSwitchCase(arg,
"app-workdir")
4512 "/app-workdir:<directory> is deprecated, use /app:workdir:<directory> instead");
4514 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4516 CommandLineSwitchCase(arg,
"app-name")
4518 WLog_WARN(TAG,
"/app-name:<directory> is deprecated, use /app:name:<name> instead");
4520 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4522 CommandLineSwitchCase(arg,
"app-icon")
4524 WLog_WARN(TAG,
"/app-icon:<filename> is deprecated, use /app:icon:<filename> instead");
4526 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4528 CommandLineSwitchCase(arg,
"app-cmd")
4530 WLog_WARN(TAG,
"/app-cmd:<command> is deprecated, use /app:cmd:<command> instead");
4532 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4534 CommandLineSwitchCase(arg,
"app-file")
4536 WLog_WARN(TAG,
"/app-file:<filename> is deprecated, use /app:file:<filename> instead");
4538 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4540 CommandLineSwitchCase(arg,
"app-guid")
4542 WLog_WARN(TAG,
"/app-guid:<guid> is deprecated, use /app:guid:<guid> instead");
4544 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4546 CommandLineSwitchCase(arg,
"g")
4548 if (!parse_gateway_host_option(settings, arg->Value))
4549 return fail_at(arg, COMMAND_LINE_ERROR);
4551 CommandLineSwitchCase(arg,
"gu")
4553 if (!parse_gateway_cred_option(settings, arg->Value, FreeRDP_GatewayUsername))
4554 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4556 CommandLineSwitchCase(arg,
"gd")
4558 if (!parse_gateway_cred_option(settings, arg->Value, FreeRDP_GatewayDomain))
4559 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4561 CommandLineSwitchCase(arg,
"gp")
4563 if (!parse_gateway_cred_option(settings, arg->Value, FreeRDP_GatewayPassword))
4564 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4566 CommandLineSwitchCase(arg,
"gt")
4568 if (!parse_gateway_type_option(settings, arg->Value))
4569 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4571 CommandLineSwitchCase(arg,
"gat")
4574 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4576 CommandLineSwitchCase(arg,
"gateway-usage-method")
4578 if (!parse_gateway_usage_option(settings, arg->Value))
4579 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4581 CommandLineSwitchCase(arg,
"kbd-remap")
4583 WLog_WARN(TAG,
"/kbd-remap:<key>=<value>,<key2>=<value2> is deprecated, use "
4584 "/kbd:remap:<key>=<value>,remap:<key2>=<value2>,... instead");
4586 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4588 CommandLineSwitchCase(arg,
"kbd-lang")
4592 WLog_WARN(TAG,
"/kbd-lang:<value> is deprecated, use /kbd:lang:<value> instead");
4593 if (!value_to_int(arg->Value, &val, 1, UINT32_MAX))
4595 WLog_ERR(TAG,
"Could not identify keyboard active language %s", arg->Value);
4596 WLog_ERR(TAG,
"Use /list:kbd-lang to list available layouts");
4597 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4601 return fail_at(arg, COMMAND_LINE_ERROR);
4603 CommandLineSwitchCase(arg,
"kbd-type")
4605 WLog_WARN(TAG,
"/kbd-type:<value> is deprecated, use /kbd:type:<value> instead");
4607 parse_command_line_option_uint32(settings, arg, FreeRDP_KeyboardType, 0, UINT32_MAX);
4609 return fail_at(arg, rc);
4611 CommandLineSwitchCase(arg,
"kbd-unicode")
4613 WLog_WARN(TAG,
"/kbd-unicode is deprecated, use /kbd:unicode[:on|off] instead");
4615 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4617 CommandLineSwitchCase(arg,
"kbd-subtype")
4619 WLog_WARN(TAG,
"/kbd-subtype:<value> is deprecated, use /kbd:subtype:<value> instead");
4621 parse_command_line_option_uint32(settings, arg, FreeRDP_KeyboardSubType, 0, UINT32_MAX);
4623 return fail_at(arg, rc);
4625 CommandLineSwitchCase(arg,
"kbd-fn-key")
4627 WLog_WARN(TAG,
"/kbd-fn-key:<value> is deprecated, use /kbd:fn-key:<value> instead");
4628 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_KeyboardFunctionKey,
4631 return fail_at(arg, rc);
4633 CommandLineSwitchCase(arg,
"bitmap-cache")
4635 WLog_WARN(TAG,
"/bitmap-cache is deprecated, use /cache:bitmap[:on|off] instead");
4637 return fail_at(arg, COMMAND_LINE_ERROR);
4639 CommandLineSwitchCase(arg,
"persist-cache")
4641 WLog_WARN(TAG,
"/persist-cache is deprecated, use /cache:persist[:on|off] instead");
4643 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4645 CommandLineSwitchCase(arg,
"persist-cache-file")
4647 WLog_WARN(TAG,
"/persist-cache-file:<filename> is deprecated, use "
4648 "/cache:persist-file:<filename> instead");
4650 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4653 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4655 CommandLineSwitchCase(arg,
"offscreen-cache")
4657 WLog_WARN(TAG,
"/bitmap-cache is deprecated, use /cache:bitmap[:on|off] instead");
4659 return fail_at(arg, COMMAND_LINE_ERROR);
4661 CommandLineSwitchCase(arg,
"glyph-cache")
4663 WLog_WARN(TAG,
"/glyph-cache is deprecated, use /cache:glyph[:on|off] instead");
4665 arg->Value ? GLYPH_SUPPORT_FULL : GLYPH_SUPPORT_NONE))
4666 return fail_at(arg, COMMAND_LINE_ERROR);
4668 CommandLineSwitchCase(arg,
"codec-cache")
4670 WLog_WARN(TAG,
"/codec-cache:<option> is deprecated, use /cache:codec:<option> instead");
4671 const int rc = parse_codec_cache_options(settings, arg);
4673 return fail_at(arg, rc);
4675 CommandLineSwitchCase(arg,
"sec-rdp")
4677 WLog_WARN(TAG,
"/sec-rdp is deprecated, use /sec:rdp[:on|off] instead");
4679 return fail_at(arg, COMMAND_LINE_ERROR);
4681 CommandLineSwitchCase(arg,
"sec-tls")
4683 WLog_WARN(TAG,
"/sec-tls is deprecated, use /sec:tls[:on|off] instead");
4685 return fail_at(arg, COMMAND_LINE_ERROR);
4687 CommandLineSwitchCase(arg,
"sec-nla")
4689 WLog_WARN(TAG,
"/sec-nla is deprecated, use /sec:nla[:on|off] instead");
4691 return fail_at(arg, COMMAND_LINE_ERROR);
4693 CommandLineSwitchCase(arg,
"sec-ext")
4695 WLog_WARN(TAG,
"/sec-ext is deprecated, use /sec:ext[:on|off] instead");
4697 return fail_at(arg, COMMAND_LINE_ERROR);
4699 CommandLineSwitchCase(arg,
"tls-ciphers")
4701 WLog_WARN(TAG,
"/tls-ciphers:<cipher list> is deprecated, use "
4702 "/tls:ciphers:<cipher list> instead");
4703 int rc = parse_tls_cipher_options(settings, arg);
4705 return fail_at(arg, rc);
4707 CommandLineSwitchCase(arg,
"tls-seclevel")
4709 WLog_WARN(TAG,
"/tls-seclevel:<level> is deprecated, use /tls:sec-level:<level> instead");
4710 int rc = parse_tls_cipher_options(settings, arg);
4712 return fail_at(arg, rc);
4714 CommandLineSwitchCase(arg,
"tls-secrets-file")
4716 WLog_WARN(TAG,
"/tls-secrets-file:<filename> is deprecated, use "
4717 "/tls:secrets-file:<filename> instead");
4718 int rc = parse_tls_cipher_options(settings, arg);
4720 return fail_at(arg, rc);
4722 CommandLineSwitchCase(arg,
"enforce-tlsv1_2")
4724 WLog_WARN(TAG,
"/enforce-tlsv1_2 is deprecated, use /tls:enforce:1.2 instead");
4725 int rc = parse_tls_cipher_options(settings, arg);
4727 return fail_at(arg, rc);
4729 CommandLineSwitchCase(arg,
"cert-name")
4731 WLog_WARN(TAG,
"/cert-name is deprecated, use /cert:name instead");
4733 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4735 CommandLineSwitchCase(arg,
"cert-ignore")
4737 WLog_WARN(TAG,
"/cert-ignore is deprecated, use /cert:ignore instead");
4739 return fail_at(arg, COMMAND_LINE_ERROR);
4741 CommandLineSwitchCase(arg,
"cert-tofu")
4743 WLog_WARN(TAG,
"/cert-tofu is deprecated, use /cert:tofu instead");
4745 return fail_at(arg, COMMAND_LINE_ERROR);
4747 CommandLineSwitchCase(arg,
"cert-deny")
4749 WLog_WARN(TAG,
"/cert-deny is deprecated, use /cert:deny instead");
4751 return fail_at(arg, COMMAND_LINE_ERROR);
4753 CommandLineSwitchDefault(arg)
4757 CommandLineSwitchEnd(arg);
4762static int parse_command_line_option_timezone(rdpSettings* settings,
4768 char TimeZoneKeyName[ARRAYSIZE(info.TimeZoneKeyName) + 1] = WINPR_C_ARRAY_INIT;
4769 while (EnumDynamicTimeZoneInformation(index++, &info) != ERROR_NO_MORE_ITEMS)
4771 (void)ConvertWCharNToUtf8(info.TimeZoneKeyName, ARRAYSIZE(info.TimeZoneKeyName),
4772 TimeZoneKeyName, ARRAYSIZE(TimeZoneKeyName));
4774 WINPR_ASSERT(arg->Value);
4775 if (strncmp(TimeZoneKeyName, arg->Value, ARRAYSIZE(TimeZoneKeyName)) == 0)
4782 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4785 return fail_at(arg, COMMAND_LINE_ERROR);
4790 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4792 tz->Bias = info.Bias;
4793 tz->DaylightBias = info.DaylightBias;
4794 tz->DaylightDate = info.DaylightDate;
4795 memcpy(tz->DaylightName, info.DaylightName,
sizeof(tz->DaylightName));
4796 tz->StandardBias = info.StandardBias;
4797 tz->StandardDate = info.StandardDate;
4798 memcpy(tz->StandardName, info.StandardName,
sizeof(tz->StandardName));
4803static int parse_command_line_option_window_pos(rdpSettings* settings,
4806 WINPR_ASSERT(settings);
4809 unsigned long x = 0;
4810 unsigned long y = 0;
4813 return fail_at(arg, COMMAND_LINE_ERROR_MISSING_ARGUMENT);
4815 if (!parseSizeValue(arg->Value, &x, &y) || x > UINT16_MAX || y > UINT16_MAX)
4817 WLog_ERR(TAG,
"invalid window-position argument");
4818 return fail_at(arg, COMMAND_LINE_ERROR_MISSING_ARGUMENT);
4822 return fail_at(arg, COMMAND_LINE_ERROR);
4824 return fail_at(arg, COMMAND_LINE_ERROR);
4829 freerdp_command_line_handle_option_t handle_option,
4830 void* handle_userdata, BOOL* promptForPassword,
char** user)
4832 WINPR_ASSERT(promptForPassword);
4837 BOOL enable = (arg->Value !=
nullptr);
4839 if (!(arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
4842 CommandLineSwitchStart(arg)
4844 CommandLineSwitchCase(arg,
"v")
4846 const int rc = parse_host_options(settings, arg);
4848 return fail_at(arg, rc);
4850 CommandLineSwitchCase(arg,
"spn-class")
4854 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4856 CommandLineSwitchCase(arg,
"sspi-module")
4859 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4861 CommandLineSwitchCase(arg,
"winscard-module")
4864 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4866 CommandLineSwitchCase(arg,
"redirect-prefer")
4868 const int rc = parse_redirect_prefer_options(settings, arg);
4870 return fail_at(arg, rc);
4872 CommandLineSwitchCase(arg,
"credentials-delegation")
4875 return fail_at(arg, COMMAND_LINE_ERROR);
4877 CommandLineSwitchCase(arg,
"prevent-session-lock")
4879 const int rc = parse_prevent_session_lock_options(settings, arg);
4881 return fail_at(arg, rc);
4883 CommandLineSwitchCase(arg,
"vmconnect")
4885 const int rc = parse_vmconnect_options(settings, arg);
4887 return fail_at(arg, rc);
4889 CommandLineSwitchCase(arg,
"w")
4891 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_DesktopWidth, -1,
4894 return fail_at(arg, rc);
4896 CommandLineSwitchCase(arg,
"h")
4898 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_DesktopHeight,
4901 return fail_at(arg, rc);
4903 CommandLineSwitchCase(arg,
"size")
4905 const int rc = parse_size_options(settings, arg);
4907 return fail_at(arg, rc);
4909 CommandLineSwitchCase(arg,
"f")
4912 return fail_at(arg, COMMAND_LINE_ERROR);
4914 CommandLineSwitchCase(arg,
"suppress-output")
4917 return fail_at(arg, COMMAND_LINE_ERROR);
4919 CommandLineSwitchCase(arg,
"multimon")
4922 return fail_at(arg, COMMAND_LINE_ERROR);
4924 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
4926 if (option_equals(arg->Value, str_force))
4929 return fail_at(arg, COMMAND_LINE_ERROR);
4933 CommandLineSwitchCase(arg,
"span")
4936 return fail_at(arg, COMMAND_LINE_ERROR);
4938 CommandLineSwitchCase(arg,
"workarea")
4941 return fail_at(arg, COMMAND_LINE_ERROR);
4943 CommandLineSwitchCase(arg,
"monitors")
4945 const int rc = parse_monitors_options(settings, arg);
4947 return fail_at(arg, rc);
4949 CommandLineSwitchCase(arg,
"t")
4952 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4954 CommandLineSwitchCase(arg,
"decorations")
4957 return fail_at(arg, COMMAND_LINE_ERROR);
4959 CommandLineSwitchCase(arg,
"dynamic-resolution")
4961 const int rc = parse_dynamic_resolution_options(settings, arg);
4963 return fail_at(arg, rc);
4965 CommandLineSwitchCase(arg,
"smart-sizing")
4967 const int rc = parse_smart_sizing_options(settings, arg);
4969 return fail_at(arg, rc);
4971 CommandLineSwitchCase(arg,
"bpp")
4973 const int rc = parse_bpp_options(settings, arg);
4975 return fail_at(arg, rc);
4977 CommandLineSwitchCase(arg,
"admin")
4980 return fail_at(arg, COMMAND_LINE_ERROR);
4982 CommandLineSwitchCase(arg,
"relax-order-checks")
4986 return fail_at(arg, COMMAND_LINE_ERROR);
4988 CommandLineSwitchCase(arg,
"restricted-admin")
4991 return fail_at(arg, COMMAND_LINE_ERROR);
4993 return fail_at(arg, COMMAND_LINE_ERROR);
4995#ifdef CHANNEL_RDPEAR_CLIENT
4996 CommandLineSwitchCase(arg,
"remoteGuard")
4999 return fail_at(arg, COMMAND_LINE_ERROR);
5001 return fail_at(arg, COMMAND_LINE_ERROR);
5004 CommandLineSwitchCase(arg,
"pth")
5007 return fail_at(arg, COMMAND_LINE_ERROR);
5009 return fail_at(arg, COMMAND_LINE_ERROR);
5012 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5014 CommandLineSwitchCase(arg,
"client-hostname")
5017 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5019 CommandLineSwitchCase(arg,
"kbd")
5021 int rc = parse_kbd_options(settings, arg);
5023 return fail_at(arg, rc);
5026 CommandLineSwitchCase(arg,
"u")
5028 WINPR_ASSERT(arg->Value);
5031 CommandLineSwitchCase(arg,
"d")
5034 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5036 CommandLineSwitchCase(arg,
"p")
5040 const char* val = arg->Value;
5045 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5047 CommandLineSwitchCase(arg,
"gateway")
5049 if (!parse_gateway_options(settings, arg))
5050 return fail_at(arg, COMMAND_LINE_ERROR);
5052 CommandLineSwitchCase(arg,
"proxy")
5054 const int rc = parse_proxy_options(settings, arg);
5056 return fail_at(arg, rc);
5059 CommandLineSwitchCase(arg,
"azure")
5061 int rc = parse_aad_options(settings, arg);
5063 return fail_at(arg, rc);
5065 CommandLineSwitchCase(arg,
"app")
5067 int rc = parse_app_options(settings, arg);
5069 return fail_at(arg, rc);
5071 CommandLineSwitchCase(arg,
"load-balance-info")
5073 WINPR_ASSERT(arg->Value);
5075 strlen(arg->Value)))
5076 return fail_at(arg, COMMAND_LINE_ERROR);
5079 CommandLineSwitchCase(arg,
"compression")
5082 return fail_at(arg, COMMAND_LINE_ERROR);
5084 CommandLineSwitchCase(arg,
"compression-level")
5086 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_CompressionLevel,
5089 return fail_at(arg, rc);
5091 CommandLineSwitchCase(arg,
"drives")
5094 return fail_at(arg, COMMAND_LINE_ERROR);
5096 CommandLineSwitchCase(arg,
"dump")
5098 const int rc = parse_dump_options(settings, arg);
5100 return fail_at(arg, rc);
5102 CommandLineSwitchCase(arg,
"disable-output")
5105 return fail_at(arg, COMMAND_LINE_ERROR);
5107 CommandLineSwitchCase(arg,
"home-drive")
5110 return fail_at(arg, COMMAND_LINE_ERROR);
5112 CommandLineSwitchCase(arg,
"ipv4")
5114 if (arg->Value !=
nullptr && strncmp(arg->Value, str_force, ARRAYSIZE(str_force)) == 0)
5117 return fail_at(arg, COMMAND_LINE_ERROR);
5122 return fail_at(arg, COMMAND_LINE_ERROR);
5125 CommandLineSwitchCase(arg,
"ipv6")
5127 if (arg->Value !=
nullptr && strncmp(arg->Value, str_force, ARRAYSIZE(str_force)) == 0)
5130 return fail_at(arg, COMMAND_LINE_ERROR);
5135 return fail_at(arg, COMMAND_LINE_ERROR);
5138 CommandLineSwitchCase(arg,
"clipboard")
5140 const int rc = parse_clipboard_options(settings, arg);
5142 return fail_at(arg, rc);
5144 CommandLineSwitchCase(arg,
"server-name")
5147 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5149 CommandLineSwitchCase(arg,
"shell")
5152 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5154 CommandLineSwitchCase(arg,
"shell-dir")
5157 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5159 CommandLineSwitchCase(arg,
"audio-mode")
5161 const int rc = parse_audio_mode_options(settings, arg);
5163 return fail_at(arg, rc);
5165 CommandLineSwitchCase(arg,
"network")
5167 const int rc = parse_network_options(settings, arg);
5169 return fail_at(arg, rc);
5171 CommandLineSwitchCase(arg,
"fonts")
5174 return fail_at(arg, COMMAND_LINE_ERROR);
5176 CommandLineSwitchCase(arg,
"wallpaper")
5179 return fail_at(arg, COMMAND_LINE_ERROR);
5181 CommandLineSwitchCase(arg,
"window-drag")
5184 return fail_at(arg, COMMAND_LINE_ERROR);
5186 CommandLineSwitchCase(arg,
"window-position")
5188 const int rc = parse_command_line_option_window_pos(settings, arg);
5190 return fail_at(arg, rc);
5192 CommandLineSwitchCase(arg,
"menu-anims")
5195 return fail_at(arg, COMMAND_LINE_ERROR);
5197 CommandLineSwitchCase(arg,
"themes")
5200 return fail_at(arg, COMMAND_LINE_ERROR);
5202 CommandLineSwitchCase(arg,
"timeout")
5205 parse_command_line_option_uint32(settings, arg, FreeRDP_TcpAckTimeout, 0, 600000);
5207 return fail_at(arg, rc);
5209 CommandLineSwitchCase(arg,
"timezone")
5211 const int rc = parse_command_line_option_timezone(settings, arg);
5213 return fail_at(arg, rc);
5215 CommandLineSwitchCase(arg,
"aero")
5218 return fail_at(arg, COMMAND_LINE_ERROR);
5220 CommandLineSwitchCase(arg,
"gdi")
5222 if (option_equals(arg->Value,
"sw"))
5225 return fail_at(arg, COMMAND_LINE_ERROR);
5227 else if (option_equals(arg->Value,
"hw"))
5230 return fail_at(arg, COMMAND_LINE_ERROR);
5233 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5235 CommandLineSwitchCase(arg,
"gfx")
5237 int rc = parse_gfx_options(settings, arg);
5239 return fail_at(arg, rc);
5242 CommandLineSwitchCase(arg,
"rfx")
5245 return fail_at(arg, COMMAND_LINE_ERROR);
5247 CommandLineSwitchCase(arg,
"rfx-mode")
5250 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5252 if (option_equals(arg->Value,
"video"))
5255 return fail_at(arg, COMMAND_LINE_ERROR);
5257 else if (option_equals(arg->Value,
"image"))
5260 return fail_at(arg, COMMAND_LINE_ERROR);
5262 return fail_at(arg, COMMAND_LINE_ERROR);
5265 CommandLineSwitchCase(arg,
"frame-ack")
5267 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_FrameAcknowledge,
5270 return fail_at(arg, rc);
5272 CommandLineSwitchCase(arg,
"nsc")
5275 return fail_at(arg, COMMAND_LINE_ERROR);
5277#if defined(WITH_JPEG)
5278 CommandLineSwitchCase(arg,
"jpeg")
5281 return fail_at(arg, COMMAND_LINE_ERROR);
5283 return fail_at(arg, COMMAND_LINE_ERROR);
5285 CommandLineSwitchCase(arg,
"jpeg-quality")
5289 if (!value_to_int(arg->Value, &val, 0, 100))
5290 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5293 return fail_at(arg, COMMAND_LINE_ERROR);
5296 CommandLineSwitchCase(arg,
"nego")
5299 return fail_at(arg, COMMAND_LINE_ERROR);
5301 CommandLineSwitchCase(arg,
"pcb")
5304 return fail_at(arg, COMMAND_LINE_ERROR);
5307 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5309 CommandLineSwitchCase(arg,
"pcid")
5311 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_PreconnectionId,
5314 return fail_at(arg, rc);
5316 return fail_at(arg, COMMAND_LINE_ERROR);
5319 CommandLineSwitchCase(arg,
"connect-child-session")
5333 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5336 CommandLineSwitchCase(arg,
"sec")
5338 const int rc = parse_sec_options(settings, arg);
5340 return fail_at(arg, rc);
5342 CommandLineSwitchCase(arg,
"encryption-methods")
5344 const int rc = parse_encryption_methods_options(settings, arg);
5346 return fail_at(arg, rc);
5348 CommandLineSwitchCase(arg,
"args-from")
5350 WLog_ERR(TAG,
"/args-from:%s can not be used in combination with other arguments!",
5352 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5354 CommandLineSwitchCase(arg,
"from-stdin")
5357 return fail_at(arg, COMMAND_LINE_ERROR);
5359 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
5362 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5363 *promptForPassword = (option_equals(arg->Value, str_force));
5365 if (!*promptForPassword)
5366 return fail_at(arg, COMMAND_LINE_ERROR);
5369 CommandLineSwitchCase(arg,
"log-level")
5371 wLog* root = WLog_GetRoot();
5373 if (!WLog_SetStringLogLevel(root, arg->Value))
5374 return fail_at(arg, COMMAND_LINE_ERROR);
5376 CommandLineSwitchCase(arg,
"log-filters")
5378 if (!WLog_AddStringLogFilters(arg->Value))
5379 return fail_at(arg, COMMAND_LINE_ERROR);
5381 CommandLineSwitchCase(arg,
"tls")
5383 int rc = parse_tls_options(settings, arg);
5385 return fail_at(arg, rc);
5387 CommandLineSwitchCase(arg,
"cert")
5389 const int rc = parse_cert_options(settings, arg);
5391 return fail_at(arg, rc);
5393 CommandLineSwitchCase(arg,
"authentication")
5396 return fail_at(arg, COMMAND_LINE_ERROR);
5398 CommandLineSwitchCase(arg,
"encryption")
5401 return fail_at(arg, COMMAND_LINE_ERROR);
5403 CommandLineSwitchCase(arg,
"grab-keyboard")
5406 return fail_at(arg, COMMAND_LINE_ERROR);
5408 CommandLineSwitchCase(arg,
"grab-mouse")
5411 return fail_at(arg, COMMAND_LINE_ERROR);
5413 CommandLineSwitchCase(arg,
"mouse-relative")
5416 return fail_at(arg, COMMAND_LINE_ERROR);
5418 CommandLineSwitchCase(arg,
"mouse")
5420 const int rc = parse_mouse_options(settings, arg);
5422 return fail_at(arg, rc);
5424 CommandLineSwitchCase(arg,
"unmap-buttons")
5427 return fail_at(arg, COMMAND_LINE_ERROR);
5429 CommandLineSwitchCase(arg,
"toggle-fullscreen")
5432 return fail_at(arg, COMMAND_LINE_ERROR);
5434 CommandLineSwitchCase(arg,
"force-console-callbacks")
5437 return fail_at(arg, COMMAND_LINE_ERROR);
5439 CommandLineSwitchCase(arg,
"floatbar")
5441 const int rc = parse_floatbar_options(settings, arg);
5443 return fail_at(arg, rc);
5445 CommandLineSwitchCase(arg,
"mouse-motion")
5448 return fail_at(arg, COMMAND_LINE_ERROR);
5450 CommandLineSwitchCase(arg,
"parent-window")
5454 if (!value_to_uint(arg->Value, &val, 0, UINT64_MAX))
5455 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5458 return fail_at(arg, COMMAND_LINE_ERROR);
5460 CommandLineSwitchCase(arg,
"client-build-number")
5463 parse_command_line_option_uint32(settings, arg, FreeRDP_ClientBuild, 0, UINT32_MAX);
5465 return fail_at(arg, rc);
5467 CommandLineSwitchCase(arg,
"cache")
5469 int rc = parse_cache_options(settings, arg);
5471 return fail_at(arg, rc);
5474 CommandLineSwitchCase(arg,
"max-fast-path-size")
5476 const int rc = parse_command_line_option_uint32(
5477 settings, arg, FreeRDP_MultifragMaxRequestSize, 0, UINT32_MAX);
5479 return fail_at(arg, rc);
5481 CommandLineSwitchCase(arg,
"auto-request-control")
5485 return fail_at(arg, COMMAND_LINE_ERROR);
5487 CommandLineSwitchCase(arg,
"async-update")
5490 return fail_at(arg, COMMAND_LINE_ERROR);
5492 CommandLineSwitchCase(arg,
"async-channels")
5495 return fail_at(arg, COMMAND_LINE_ERROR);
5497 CommandLineSwitchCase(arg,
"wm-class")
5500 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5502 CommandLineSwitchCase(arg,
"play-rfx")
5505 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5508 return fail_at(arg, COMMAND_LINE_ERROR);
5510 CommandLineSwitchCase(arg,
"auth-only")
5513 return fail_at(arg, COMMAND_LINE_ERROR);
5515 CommandLineSwitchCase(arg,
"auth-pkg-list")
5519 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5521 CommandLineSwitchCase(arg,
"auto-reconnect")
5524 return fail_at(arg, COMMAND_LINE_ERROR);
5526 CommandLineSwitchCase(arg,
"auto-reconnect-max-retries")
5528 const int rc = parse_command_line_option_uint32(
5529 settings, arg, FreeRDP_AutoReconnectMaxRetries, 0, 1000);
5531 return fail_at(arg, rc);
5533 CommandLineSwitchCase(arg,
"reconnect-cookie")
5535 const int rc = parse_reconnect_cookie_options(settings, arg);
5537 return fail_at(arg, rc);
5539 CommandLineSwitchCase(arg,
"print-reconnect-cookie")
5542 return fail_at(arg, COMMAND_LINE_ERROR);
5544 CommandLineSwitchCase(arg,
"pwidth")
5546 const int rc = parse_command_line_option_uint32(
5547 settings, arg, FreeRDP_DesktopPhysicalWidth, 0, UINT32_MAX);
5549 return fail_at(arg, rc);
5551 CommandLineSwitchCase(arg,
"pheight")
5553 const int rc = parse_command_line_option_uint32(
5554 settings, arg, FreeRDP_DesktopPhysicalHeight, 0, UINT32_MAX);
5556 return fail_at(arg, rc);
5558 CommandLineSwitchCase(arg,
"orientation")
5562 if (!value_to_int(arg->Value, &val, 0, UINT16_MAX))
5563 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5566 return fail_at(arg, COMMAND_LINE_ERROR);
5567 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_ORIENTATION))
5568 return fail_at(arg, COMMAND_LINE_ERROR);
5570 CommandLineSwitchCase(arg,
"old-license")
5573 return fail_at(arg, COMMAND_LINE_ERROR);
5575 CommandLineSwitchCase(arg,
"scale")
5577 const int rc = parse_scale_options(settings, arg);
5579 return fail_at(arg, rc);
5581 CommandLineSwitchCase(arg,
"scale-desktop")
5583 const int rc = parse_command_line_option_uint32(settings, arg,
5584 FreeRDP_DesktopScaleFactor, 100, 500);
5586 return fail_at(arg, rc);
5587 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_DESKTOP_SCALE))
5588 return fail_at(arg, COMMAND_LINE_ERROR);
5590 CommandLineSwitchCase(arg,
"scale-device")
5592 const int rc = parse_scale_device_options(settings, arg);
5594 return fail_at(arg, rc);
5596 CommandLineSwitchCase(arg,
"action-script")
5599 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5601#if !defined(WITHOUT_FREERDP_3x_DEPRECATED)
5602 CommandLineSwitchCase(arg, RDP2TCP_DVC_CHANNEL_NAME)
5605 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5608 CommandLineSwitchCase(arg,
"fipsmode")
5611 return fail_at(arg, COMMAND_LINE_ERROR);
5613 CommandLineSwitchCase(arg,
"smartcard-logon")
5615 const int rc = parse_smartcard_logon_options(settings, arg);
5617 return fail_at(arg, rc);
5619 CommandLineSwitchCase(arg,
"tune")
5621 const int rc = parse_tune_options(settings, arg);
5623 return fail_at(arg, rc);
5625 CommandLineSwitchDefault(arg)
5627#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
5628 const int status = parse_deprecated_command_line(settings, arg);
5635 const int rc = handle_option(arg, handle_userdata);
5637 return fail_at(arg, rc);
5640 CommandLineSwitchEnd(arg)
5641 }
while ((arg = CommandLineFindNextArgumentA(arg)) !=
nullptr);
5648 bool insecureArgFound =
false;
5649 for (
size_t x = 0; x < ARRAYSIZE(credential_args); x++)
5651 const char* cred = credential_args[x];
5655 if ((arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT) == 0)
5658 WLog_WARN(TAG,
"Using /%s is insecure", arg->Name);
5659 insecureArgFound =
true;
5662 if (insecureArgFound)
5664 WLog_WARN(TAG,
"Passing credentials or secrets via command line might expose these in the "
5666 WLog_WARN(TAG,
"Consider using one of the following (more secure) alternatives:");
5667 WLog_WARN(TAG,
" - /args-from: pipe in arguments from stdin, file, file descriptor or "
5668 "environment variable");
5669 WLog_WARN(TAG,
" - /from-stdin pass the credential via stdin");
5670 WLog_WARN(TAG,
" - set environment variable FREERDP_ASKPASS to have a gui tool query for "
5675static int freerdp_client_settings_parse_command_line_arguments_int(
5676 rdpSettings* settings,
int argc,
char* argv[], BOOL allowUnknown,
5678 freerdp_command_line_handle_option_t handle_option,
void* handle_userdata, UINT32 cmdflags)
5680 char* user =
nullptr;
5683 BOOL assist = FALSE;
5685 BOOL promptForPassword = FALSE;
5686 BOOL compatibility = FALSE;
5694 ext = option_is_rdp_file(argv[1]);
5695 assist = option_is_incident_file(argv[1]);
5698 if (!ext && !assist)
5699 compatibility = freerdp_client_detect_command_line(argc, argv, &flags);
5701 compatibility = freerdp_client_detect_command_line(argc - 1, &argv[1], &flags);
5712 WLog_WARN(TAG,
"Unsupported command line syntax!");
5713 WLog_WARN(TAG,
"%s 1.0 style syntax was dropped with version 3!",
5714 freerdp_getApplicationDetailsString());
5719 flags |= COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
5723 if (freerdp_client_settings_parse_connection_file(settings, argv[1]))
5724 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
5729 if (freerdp_client_settings_parse_assistance_file(settings, argc, argv) < 0)
5730 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
5733 CommandLineClearArgumentsA(largs);
5734 status = CommandLineParseArgumentsA(argc, argv, largs, flags, settings,
5735 freerdp_client_command_line_pre_filter,
5736 freerdp_client_command_line_post_filter);
5741 prepare_default_settings(settings, largs, ext);
5742 if ((cmdflags & FREERDP_SETTINGS_CMD_PARSE_SUPPRESS_WARNINGS) == 0)
5743 warn_credential_args(largs);
5750 return COMMAND_LINE_ERROR_MEMORY;
5752 status = parse_command_line(settings, arg, handle_option, handle_userdata, &promptForPassword,
5760 return COMMAND_LINE_ERROR;
5763 return COMMAND_LINE_ERROR;
5765 if (!freerdp_parse_username_settings(user, settings, FreeRDP_Username, FreeRDP_Domain))
5766 return COMMAND_LINE_ERROR;
5771 return COMMAND_LINE_ERROR;
5775 if (promptForPassword)
5780 char buffer[512 + 1] = WINPR_C_ARRAY_INIT;
5782 if (!freerdp_passphrase_read(instance->context,
"Password: ", buffer,
5783 ARRAYSIZE(buffer) - 1, 1))
5784 return COMMAND_LINE_ERROR;
5786 return COMMAND_LINE_ERROR;
5794 char buffer[512 + 1] = WINPR_C_ARRAY_INIT;
5796 if (!freerdp_passphrase_read(instance->context,
"Gateway Password: ", buffer,
5797 ARRAYSIZE(buffer) - 1, 1))
5798 return COMMAND_LINE_ERROR;
5800 return COMMAND_LINE_ERROR;
5805 freerdp_performance_flags_make(settings);
5812 return COMMAND_LINE_ERROR;
5814 return COMMAND_LINE_ERROR;
5817 arg = CommandLineFindArgumentA(largs,
"port");
5818 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
5821 parse_command_line_option_uint32(settings, arg, FreeRDP_ServerPort, 0, UINT16_MAX);
5823 return fail_at(arg, rc);
5829 if (nego && (nego->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
5830 return fail_at(arg, COMMAND_LINE_ERROR);
5833 WLog_INFO(TAG,
"/vmconnect uses custom port %" PRIu32, port);
5836 fill_credential_strings(largs);
5841static void argv_free(
int* pargc,
char** pargv[])
5843 WINPR_ASSERT(pargc);
5844 WINPR_ASSERT(pargv);
5845 const int argc = *pargc;
5846 char** argv = *pargv;
5852 for (
int x = 0; x < argc; x++)
5857static BOOL argv_append(
int* pargc,
char** pargv[],
char* what)
5859 WINPR_ASSERT(pargc);
5860 WINPR_ASSERT(pargv);
5868 int nargc = *pargc + 1;
5869 char** tmp = (
char**)realloc((
void*)*pargv, (size_t)nargc *
sizeof(
char*));
5879static BOOL argv_append_dup(
int* pargc,
char** pargv[],
const char* what)
5881 char* copy =
nullptr;
5883 copy = _strdup(what);
5885 const BOOL rc = argv_append(pargc, pargv, copy);
5891static BOOL args_from_fp(FILE* fp,
int* aargc,
char** aargv[],
const char* file,
const char* cmd)
5893 BOOL success = FALSE;
5895 WINPR_ASSERT(aargc);
5896 WINPR_ASSERT(aargv);
5901 WLog_ERR(TAG,
"Failed to read command line options from file '%s'", file);
5904 if (!argv_append_dup(aargc, aargv, cmd))
5908 char* line =
nullptr;
5910 INT64 rc = GetLine(&line, &size, fp);
5911 if ((rc < 0) || !line)
5921 const char cur = (line[rc - 1]);
5922 if ((cur ==
'\n') || (cur ==
'\r'))
5924 line[rc - 1] =
'\0';
5936 if (!argv_append(aargc, aargv, line))
5947 argv_free(aargc, aargv);
5951static BOOL args_from_env(
const char* name,
int* aargc,
char** aargv[],
const char* arg,
5954 BOOL success = FALSE;
5955 char* env =
nullptr;
5957 WINPR_ASSERT(aargc);
5958 WINPR_ASSERT(aargv);
5963 WLog_ERR(TAG,
"%s - environment variable name empty", arg);
5968 const DWORD size = GetEnvironmentVariableX(name, env, 0);
5971 WLog_ERR(TAG,
"%s - no environment variable '%s'", arg, name);
5974 env = calloc(size + 1,
sizeof(
char));
5979 const DWORD rc = GetEnvironmentVariableX(name, env, size);
5984 WLog_ERR(TAG,
"environment variable '%s' is empty", arg);
5990 if (!argv_append_dup(aargc, aargv, cmd))
5994 char* context =
nullptr;
5995 char* tok = strtok_s(env,
"\n", &context);
5998 if (!argv_append_dup(aargc, aargv, tok))
6000 tok = strtok_s(
nullptr,
"\n", &context);
6008 argv_free(aargc, aargv);
6012int freerdp_client_settings_parse_command_line_arguments(rdpSettings* settings,
int oargc,
6013 char* oargv[], BOOL allowUnknown)
6015 return freerdp_client_settings_parse_command_line_arguments_ex(
6016 settings, oargc, oargv, allowUnknown,
nullptr, 0,
nullptr,
nullptr);
6019int freerdp_client_settings_parse_command_line_arguments_ex(
6021 size_t count, freerdp_command_line_handle_option_t handle_option,
void* handle_userdata)
6023 return freerdp_client_settings_parse_command_line_arguments_with_flags(
6024 settings, argc, argv, allowUnknown, args, count, handle_option, handle_userdata, 0);
6027int freerdp_client_settings_parse_command_line_arguments_with_flags(
6028 rdpSettings* settings,
int oargc,
char** oargv, BOOL allowUnknown,
6030 void* handle_userdata, UINT32 flags, ...)
6033 char** argv = oargv;
6036 char** aargv =
nullptr;
6038 if ((argc == 2) && option_starts_with(
"/args-from:", argv[1]))
6040 flags |= FREERDP_SETTINGS_CMD_PARSE_SUPPRESS_WARNINGS;
6041 BOOL success = FALSE;
6042 const char* file = strchr(argv[1],
':') + 1;
6045 if (option_starts_with(
"fd:", file))
6047 ULONGLONG result = 0;
6048 const char* val = strchr(file,
':') + 1;
6049 if (!value_to_uint(val, &result, 0, INT_MAX))
6051 fp = fdopen((
int)result,
"r");
6052 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
6054 else if (strncmp(file,
"env:", 4) == 0)
6056 const char* name = strchr(file,
':') + 1;
6057 success = args_from_env(name, &aargc, &aargv, oargv[1], oargv[0]);
6059 else if (strncmp(file,
"file:", 5) == 0)
6061 file = strchr(file,
':') + 1;
6062 fp = winpr_fopen(file,
"r");
6063 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
6065#if !defined(WITHOUT_FREERDP_3x_DEPRECATED)
6066 else if (strcmp(file,
"stdin") != 0)
6068 fp = winpr_fopen(file,
"r");
6069 WLog_WARN(TAG,
"/args-from:%s is deprecated, use /args-from:file:%s instead", file,
6071 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
6074 else if (strcmp(file,
"stdin") == 0)
6075 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
6083 WINPR_ASSERT(count <= SSIZE_MAX);
6089 res = freerdp_client_settings_parse_command_line_arguments_int(
6090 settings, argc, argv, allowUnknown, largs, lcount, handle_option, handle_userdata, flags);
6093 argv_free(&aargc, &aargv);
6097static BOOL freerdp_client_load_static_channel_addin(rdpChannels* channels, rdpSettings* settings,
6098 const char* name,
void* data)
6100 PVIRTUALCHANNELENTRY entry =
nullptr;
6101 PVIRTUALCHANNELENTRY pvce = freerdp_load_channel_addin_entry(
6102 name,
nullptr,
nullptr, FREERDP_ADDIN_CHANNEL_STATIC | FREERDP_ADDIN_CHANNEL_ENTRYEX);
6103 PVIRTUALCHANNELENTRYEX pvceex = WINPR_FUNC_PTR_CAST(pvce, PVIRTUALCHANNELENTRYEX);
6107 freerdp_load_channel_addin_entry(name,
nullptr,
nullptr, FREERDP_ADDIN_CHANNEL_STATIC);
6111 if (freerdp_channels_client_load_ex(channels, settings, pvceex, data) == 0)
6113 WLog_DBG(TAG,
"loading channelEx %s", name);
6119 if (freerdp_channels_client_load(channels, settings, entry, data) == 0)
6121 WLog_DBG(TAG,
"loading channel %s", name);
6131 FreeRDP_Settings_Keys_Bool settingId;
6132 const char* channelName;
6136BOOL freerdp_client_load_addins(rdpChannels* channels, rdpSettings* settings)
6138 ChannelToLoad dynChannels[] = {
6139#if defined(CHANNEL_AINPUT_CLIENT)
6140 { FreeRDP_BOOL_UNUSED, AINPUT_CHANNEL_NAME,
nullptr },
6142#ifdef CHANNEL_AUDIN_CLIENT
6143 { FreeRDP_AudioCapture, AUDIN_CHANNEL_NAME,
nullptr },
6145#ifdef CHANNEL_RDPSND_CLIENT
6146 { FreeRDP_AudioPlayback, RDPSND_CHANNEL_NAME,
nullptr },
6148#ifdef CHANNEL_RDPEI_CLIENT
6149 { FreeRDP_MultiTouchInput, RDPEI_CHANNEL_NAME,
nullptr },
6151#ifdef CHANNEL_RDPGFX_CLIENT
6152 { FreeRDP_SupportGraphicsPipeline, RDPGFX_CHANNEL_NAME,
nullptr },
6154#ifdef CHANNEL_ECHO_CLIENT
6155 { FreeRDP_SupportEchoChannel, ECHO_CHANNEL_NAME,
nullptr },
6157#ifdef CHANNEL_SSHAGENT_CLIENT
6158 { FreeRDP_SupportSSHAgentChannel,
"sshagent",
nullptr },
6160#ifdef CHANNEL_DISP_CLIENT
6161 { FreeRDP_SupportDisplayControl, DISP_CHANNEL_NAME,
nullptr },
6163#ifdef CHANNEL_GEOMETRY_CLIENT
6164 { FreeRDP_SupportGeometryTracking, GEOMETRY_CHANNEL_NAME,
nullptr },
6166#ifdef CHANNEL_VIDEO_CLIENT
6167 { FreeRDP_SupportVideoOptimized, VIDEO_CHANNEL_NAME,
nullptr },
6169#ifdef CHANNEL_RDPEAR_CLIENT
6170 { FreeRDP_RemoteCredentialGuard, RDPEAR_CHANNEL_NAME,
nullptr },
6172#ifdef CHANNEL_RDPEWA_CLIENT
6173 { FreeRDP_RedirectWebAuthN, RDPEWA_CHANNEL_NAME,
nullptr },
6177 ChannelToLoad staticChannels[] = {
6178#if defined(CHANNEL_RDPSND_CLIENT)
6179 { FreeRDP_AudioPlayback, RDPSND_CHANNEL_NAME,
nullptr },
6181#if defined(CHANNEL_CLIPRDR_CLIENT)
6182 { FreeRDP_RedirectClipboard, CLIPRDR_SVC_CHANNEL_NAME,
nullptr },
6184#if defined(CHANNEL_ENCOMSP_CLIENT)
6185 { FreeRDP_EncomspVirtualChannel, ENCOMSP_SVC_CHANNEL_NAME, settings },
6187#if defined(CHANNEL_REMDESK_CLIENT)
6188 { FreeRDP_RemdeskVirtualChannel, REMDESK_SVC_CHANNEL_NAME, settings },
6190#if defined(CHANNEL_RAIL_CLIENT)
6191 { FreeRDP_RemoteApplicationMode, RAIL_SVC_CHANNEL_NAME, settings }
6198 for (
size_t i = 0; i < ARRAYSIZE(dynChannels); i++)
6200 if ((dynChannels[i].settingId == FreeRDP_BOOL_UNUSED) ||
6203 const char*
const p[] = { dynChannels[i].channelName };
6205 if (!freerdp_client_add_dynamic_channel(settings, ARRAYSIZE(p), p))
6213 if ((freerdp_static_channel_collection_find(settings, RDPSND_CHANNEL_NAME)) ||
6214 (freerdp_dynamic_channel_collection_find(settings, RDPSND_CHANNEL_NAME))
6215#
if defined(CHANNEL_TSMF_CLIENT)
6216 || (freerdp_dynamic_channel_collection_find(settings,
"tsmf"))
6226 if (freerdp_dynamic_channel_collection_find(settings, AUDIN_CHANNEL_NAME))
6242 if (DrivesToRedirect && (strlen(DrivesToRedirect) != 0))
6250 char* value =
nullptr;
6251 char* tok =
nullptr;
6252 char* context =
nullptr;
6254 value = _strdup(DrivesToRedirect);
6258 tok = strtok_s(value,
";", &context);
6261 WLog_ERR(TAG,
"DrivesToRedirect contains invalid data: '%s'", DrivesToRedirect);
6278 const char* name =
nullptr;
6279 const char* drive = tok;
6280 char* subcontext =
nullptr;
6281 char* start = strtok_s(tok,
"(", &subcontext);
6282 char* end = strtok_s(
nullptr,
")", &subcontext);
6286 if (freerdp_path_valid(name,
nullptr) && freerdp_path_valid(drive,
nullptr))
6288 success = freerdp_client_add_drive(settings, name,
nullptr);
6290 success = freerdp_client_add_drive(settings, drive,
nullptr);
6293 success = freerdp_client_add_drive(settings, drive, name);
6301 tok = strtok_s(
nullptr,
";", &context);
6310 if (!freerdp_device_collection_find(settings,
"drive"))
6312 const char*
const params[] = {
"drive",
"media",
"*" };
6314 if (!freerdp_client_add_device_channel(settings, ARRAYSIZE(params), params))
6331 if (!freerdp_device_collection_find(settings,
"drive"))
6333 const char* params[] = {
"drive",
"home",
"%" };
6335 if (!freerdp_client_add_device_channel(settings, ARRAYSIZE(params), params))
6342 if (!freerdp_client_load_static_channel_addin(channels, settings, RDPDR_SVC_CHANNEL_NAME,
6346 if (!freerdp_static_channel_collection_find(settings, RDPSND_CHANNEL_NAME) &&
6347 !freerdp_dynamic_channel_collection_find(settings, RDPSND_CHANNEL_NAME))
6349 const char*
const params[] = { RDPSND_CHANNEL_NAME,
"sys:fake" };
6351 if (!freerdp_client_add_static_channel(settings, ARRAYSIZE(params), params))
6354 if (!freerdp_client_add_dynamic_channel(settings, ARRAYSIZE(params), params))
6361 if (!freerdp_device_collection_find_type(settings, RDPDR_DTYP_SMARTCARD))
6363 RDPDR_DEVICE* smartcard = freerdp_device_new(RDPDR_DTYP_SMARTCARD, 0,
nullptr);
6368 if (!freerdp_device_collection_add(settings, smartcard))
6370 freerdp_device_free(smartcard);
6378 if (!freerdp_device_collection_find_type(settings, RDPDR_DTYP_PRINT))
6380 RDPDR_DEVICE* printer = freerdp_device_new(RDPDR_DTYP_PRINT, 0,
nullptr);
6385 if (!freerdp_device_collection_add(settings, printer))
6387 freerdp_device_free(printer);
6414 for (
size_t i = 0; i < ARRAYSIZE(staticChannels); i++)
6416 if ((staticChannels[i].settingId == 0) ||
6419 if (staticChannels[i].args)
6421 if (!freerdp_client_load_static_channel_addin(
6422 channels, settings, staticChannels[i].channelName, staticChannels[i].args))
6427 const char*
const p[] = { staticChannels[i].channelName };
6428 if (!freerdp_client_add_static_channel(settings, ARRAYSIZE(p), p))
6434#if !defined(WITHOUT_FREERDP_3x_DEPRECATED)
6439 const char*
const p[] = { RDP2TCP_DVC_CHANNEL_NAME, RDP2TCPArgs };
6440 if (!freerdp_client_add_static_channel(settings, ARRAYSIZE(p), p))
6450 freerdp_settings_get_pointer_array_writable(settings, FreeRDP_StaticChannelArray, i);
6452 if (!freerdp_client_load_static_channel_addin(channels, settings, _args->argv[0], _args))
6464 if (!freerdp_client_load_static_channel_addin(channels, settings, DRDYNVC_SVC_CHANNEL_NAME,
6472void freerdp_client_warn_unmaintained(
int argc,
char* argv[])
6474 const char* app = (argc > 0) ? argv[0] :
"INVALID_ARGV";
6475 wLog* log = WLog_Get(TAG);
6478 freerdp_warn_unmaintained(log,
"%s client", app);
6481void freerdp_client_warn_experimental(
int argc,
char* argv[])
6483 const char* app = (argc > 0) ? argv[0] :
"INVALID_ARGV";
6484 wLog* log = WLog_Get(TAG);
6487 freerdp_warn_experimental(log,
"%s client", app);
6490void freerdp_client_warn_deprecated(
int argc,
char* argv[])
6492 const char* app = (argc > 0) ? argv[0] :
"INVALID_ARGV";
6493 wLog* log = WLog_Get(TAG);
6496 freerdp_warn_deprecated(log,
"%s client",
6497 "As replacement there is a SDL3 based client available.", app);
WINPR_ATTR_NODISCARD FREERDP_API const void * freerdp_settings_get_pointer(const rdpSettings *settings, FreeRDP_Settings_Keys_Pointer id)
Returns a immutable pointer settings value.
WINPR_ATTR_NODISCARD FREERDP_API char * freerdp_settings_get_string_writable(rdpSettings *settings, FreeRDP_Settings_Keys_String id)
Returns a string settings value.
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 INT32 freerdp_settings_get_int32(const rdpSettings *settings, FreeRDP_Settings_Keys_Int32 id)
Returns a INT32 settings value.
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 UINT16 freerdp_settings_get_uint16(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt16 id)
Returns a UINT16 settings value.
WINPR_ATTR_NODISCARD FREERDP_API UINT64 freerdp_settings_get_uint64(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt64 id)
Returns a UINT64 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 BOOL freerdp_settings_set_uint64(rdpSettings *settings, FreeRDP_Settings_Keys_UInt64 id, UINT64 val)
Sets a UINT64 settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_append_string(rdpSettings *settings, FreeRDP_Settings_Keys_String id, const char *separator, const char *param)
appends a string to a settings value. The param is copied. If the initial value of the setting was no...
WINPR_ATTR_NODISCARD FREERDP_API INT16 freerdp_settings_get_int16(const rdpSettings *settings, FreeRDP_Settings_Keys_Int16 id)
Returns a INT16 settings value.
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 INT64 freerdp_settings_get_int64(const rdpSettings *settings, FreeRDP_Settings_Keys_Int64 id)
Returns a INT64 settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_set_uint16(rdpSettings *settings, FreeRDP_Settings_Keys_UInt16 id, UINT16 val)
Sets a UINT16 settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_set_gateway_usage_method(rdpSettings *settings, UINT32 GatewayUsageMethod)
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.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.
WINPR_ATTR_NODISCARD FREERDP_API const char * freerdp_settings_get_name_for_key(SSIZE_T key)
Returns the type name for a key.
WINPR_ATTR_NODISCARD FREERDP_API SSIZE_T freerdp_settings_get_type_for_key(SSIZE_T key)
Get a key type for the key index.