22#include <freerdp/config.h>
27#include <winpr/assert.h>
28#include <winpr/string.h>
30#include <winpr/wlog.h>
31#include <winpr/path.h>
32#include <winpr/ncrypt.h>
33#include <winpr/environment.h>
34#include <winpr/timezone.h>
36#include <freerdp/freerdp.h>
37#include <freerdp/addin.h>
38#include <freerdp/settings.h>
39#include <freerdp/client.h>
40#include <freerdp/client/channels.h>
41#include <freerdp/channels/drdynvc.h>
42#include <freerdp/channels/cliprdr.h>
43#include <freerdp/channels/encomsp.h>
44#include <freerdp/channels/rdpear.h>
45#include <freerdp/channels/rdp2tcp.h>
46#include <freerdp/channels/remdesk.h>
47#include <freerdp/channels/rdpsnd.h>
48#include <freerdp/channels/disp.h>
49#include <freerdp/crypto/crypto.h>
50#include <freerdp/locale/keyboard.h>
51#include <freerdp/utils/passphrase.h>
52#include <freerdp/utils/proxy_utils.h>
53#include <freerdp/utils/string.h>
54#include <freerdp/channels/urbdrc.h>
55#include <freerdp/channels/rdpdr.h>
56#include <freerdp/locale/locale.h>
58#if defined(CHANNEL_AINPUT_CLIENT)
59#include <freerdp/channels/ainput.h>
62#include <freerdp/channels/audin.h>
63#include <freerdp/channels/echo.h>
65#include <freerdp/client/cmdline.h>
66#include <freerdp/version.h>
67#include <freerdp/client/utils/smartcard_cli.h>
69#include <openssl/tls1.h>
72#include <freerdp/log.h>
73#define TAG CLIENT_TAG("common.cmdline")
75static const char str_force[] =
"force";
77static const char* option_starts_with(
const char* what,
const char* val);
78static BOOL option_ends_with(
const char* str,
const char* ext);
79static BOOL option_equals(
const char* what,
const char* val);
81static BOOL freerdp_client_print_codepages(
const char* arg)
85 const char* filter = NULL;
90 filter = strchr(arg,
',');
96 pages = freerdp_keyboard_get_matching_codepages(column, filter, &count);
100 printf(
"%-10s %-8s %-60s %-36s %-48s\n",
"<id>",
"<locale>",
"<win langid>",
"<language>",
102 for (
size_t x = 0; x < count; x++)
105 char buffer[2048] = { 0 };
107 if (strnlen(page->subLanguageSymbol, ARRAYSIZE(page->subLanguageSymbol)) > 0)
108 (void)_snprintf(buffer,
sizeof(buffer),
"[%s|%s]", page->primaryLanguageSymbol,
109 page->subLanguageSymbol);
111 (
void)_snprintf(buffer,
sizeof(buffer),
"[%s]", page->primaryLanguageSymbol);
112 printf(
"id=0x%04" PRIx16
": [%-6s] %-60s %-36s %-48s\n", page->id, page->locale, buffer,
113 page->primaryLanguage, page->subLanguage);
115 freerdp_codepages_free(pages);
119static BOOL freerdp_path_valid(
const char* path, BOOL* special)
121 const char DynamicDrives[] =
"DynamicDrives";
128 (option_equals(
"*", path) || option_equals(DynamicDrives, path) || option_equals(
"%", path))
132 isPath = winpr_PathFileExists(path);
135 *special = isSpecial;
137 return isSpecial || isPath;
140static BOOL freerdp_sanitize_drive_name(
char* name,
const char* invalid,
const char* replacement)
142 if (!name || !invalid || !replacement)
144 if (strlen(invalid) != strlen(replacement))
147 while (*invalid !=
'\0')
149 const char what = *invalid++;
150 const char with = *replacement++;
153 while ((cur = strchr(cur, what)) != NULL)
159static char* name_from_path(
const char* path)
161 const char* name =
"NULL";
164 if (option_equals(
"%", path))
166 else if (option_equals(
"*", path))
167 name =
"hotplug-all";
168 else if (option_equals(
"DynamicDrives", path))
173 return _strdup(name);
176static BOOL freerdp_client_add_drive(rdpSettings* settings,
const char* path,
const char* name)
197 if (!skip && winpr_PathFileExists(name))
199 if (!winpr_PathFileExists(path) || (!PathIsRelativeA(name) && PathIsRelativeA(path)))
201 const char* tmp = path;
209 dname = _strdup(name);
211 dname = name_from_path(path);
213 if (freerdp_sanitize_drive_name(dname,
"\\/",
"__"))
215 const char* args[] = { dname, path };
216 device = freerdp_device_new(RDPDR_DTYP_FILESYSTEM, ARRAYSIZE(args), args);
226 BOOL isSpecial = FALSE;
227 BOOL isPath = freerdp_path_valid(path, &isSpecial);
229 if (!isPath && !isSpecial)
231 WLog_WARN(TAG,
"Invalid drive to redirect: '%s' does not exist, skipping.", path);
232 freerdp_device_free(device);
234 else if (!freerdp_device_collection_add(settings, device))
241 freerdp_device_free(device);
245static BOOL value_to_int(
const char* value, LONGLONG* result, LONGLONG min, LONGLONG max)
249 if (!value || !result)
253 rc = _strtoi64(value, NULL, 0);
258 if ((rc < min) || (rc > max))
265static BOOL value_to_uint(
const char* value, ULONGLONG* result, ULONGLONG min, ULONGLONG max)
267 unsigned long long rc = 0;
269 if (!value || !result)
273 rc = _strtoui64(value, NULL, 0);
278 if ((rc < min) || (rc > max))
285BOOL freerdp_client_print_version(
void)
287 printf(
"This is FreeRDP version %s (%s)\n", FREERDP_VERSION_FULL, FREERDP_GIT_REVISION);
291BOOL freerdp_client_print_version_ex(
int argc,
char** argv)
293 WINPR_ASSERT(argc >= 0);
294 WINPR_ASSERT(argv || (argc == 0));
295 const char* name = (argc > 0) ? argv[0] :
"argc < 1";
296 printf(
"This is FreeRDP version [%s] %s (%s)\n", name, FREERDP_VERSION_FULL,
297 FREERDP_GIT_REVISION);
301BOOL freerdp_client_print_buildconfig(
void)
303 printf(
"%s", freerdp_get_build_config());
307BOOL freerdp_client_print_buildconfig_ex(
int argc,
char** argv)
309 WINPR_ASSERT(argc >= 0);
310 WINPR_ASSERT(argv || (argc == 0));
311 const char* name = (argc > 0) ? argv[0] :
"argc < 1";
312 printf(
"[%s] %s", name, freerdp_get_build_config());
316static void freerdp_client_print_scancodes(
void)
318 printf(
"RDP scancodes and their name for use with /kbd:remap\n");
320 for (UINT32 x = 0; x < UINT16_MAX; x++)
322 const char* name = freerdp_keyboard_scancode_name(x);
324 printf(
"0x%04" PRIx32
" --> %s\n", x, name);
328static BOOL is_delimiter(
char c,
const char* delimiters)
331 while ((d = *delimiters++) !=
'\0')
339static const char* get_last(
const char* start,
size_t len,
const char* delimiters)
341 const char* last = NULL;
342 for (
size_t x = 0; x < len; x++)
345 if (is_delimiter(c, delimiters))
351static SSIZE_T next_delimiter(
const char* text,
size_t len,
size_t max,
const char* delimiters)
356 const char* last = get_last(text, max, delimiters);
360 return (SSIZE_T)(last - text);
363static SSIZE_T forced_newline_at(
const char* text,
size_t len,
size_t limit,
364 const char* force_newline)
367 while ((d = *force_newline++) !=
'\0')
369 const char* tok = strchr(text, d);
372 const size_t offset = WINPR_ASSERTING_INT_CAST(
size_t, tok - text);
373 if ((offset > len) || (offset > limit))
375 return (SSIZE_T)(offset);
381static BOOL print_align(
size_t start_offset,
size_t* current)
383 WINPR_ASSERT(current);
384 if (*current < start_offset)
386 const int rc = printf(
"%*c", (
int)(start_offset - *current),
' ');
389 *current += (size_t)rc;
394static char* print_token(
char* text,
size_t start_offset,
size_t* current,
size_t limit,
395 const char* delimiters,
const char* force_newline)
398 const size_t tlen = strnlen(text, limit);
400 const SSIZE_T force_at = forced_newline_at(text, len, limit - *current, force_newline);
401 BOOL isForce = (force_at >= 0);
404 len = MIN(len, (
size_t)force_at);
406 if (!print_align(start_offset, current))
409 const SSIZE_T delim = next_delimiter(text, len, limit - *current, delimiters);
410 const BOOL isDelim = delim > 0;
413 len = MIN(len, (
size_t)delim + 1);
416 rc = printf(
"%.*s", (
int)len, text);
420 if (isForce || isDelim)
425 const size_t offset = len + ((isForce && (force_at == 0)) ? 1 : 0);
426 return &text[offset];
429 *current += (size_t)rc;
431 if (tlen == (
size_t)rc)
433 return &text[(size_t)rc];
436static size_t print_optionals(
const char* text,
size_t start_offset,
size_t current)
438 const size_t limit = 80;
439 char* str = _strdup(text);
444 cur = print_token(cur, start_offset + 1, ¤t, limit,
"[], ",
"\r\n");
445 }
while (cur != NULL);
451static size_t print_description(
const char* text,
size_t start_offset,
size_t current)
453 const size_t limit = 80;
454 char* str = _strdup(text);
458 cur = print_token(cur, start_offset, ¤t, limit,
" ",
"\r\n");
461 const int rc = printf(
"\n");
464 const size_t src = WINPR_ASSERTING_INT_CAST(
size_t, rc);
465 WINPR_ASSERT(SIZE_MAX - src > current);
471static int cmp_cmdline_args(
const void* pva,
const void* pvb)
476 if (!a->Name && !b->Name)
482 return strcmp(a->Name, b->Name);
497 const size_t description_offset = 30 + 8;
499 if (arg->Flags & (COMMAND_LINE_VALUE_BOOL | COMMAND_LINE_VALUE_FLAG))
501 if ((arg->Flags & (uint32_t)~COMMAND_LINE_VALUE_BOOL) == 0)
502 rc = printf(
" %s%s", arg->Default ?
"-" :
"+", arg->Name);
503 else if ((arg->Flags & COMMAND_LINE_VALUE_OPTIONAL) != 0)
504 rc = printf(
" [%s|/]%s", arg->Default ?
"-" :
"+", arg->Name);
507 rc = printf(
" %s%s", arg->Default ?
"-" :
"+", arg->Name);
511 rc = printf(
" /%s", arg->Name);
517 if ((arg->Flags & COMMAND_LINE_VALUE_REQUIRED) ||
518 (arg->Flags & COMMAND_LINE_VALUE_OPTIONAL))
522 if (arg->Flags & COMMAND_LINE_VALUE_OPTIONAL)
528 pos = print_optionals(arg->Format, pos, pos);
540 pos = print_optionals(arg->Format, pos, pos);
543 if (pos > description_offset)
551 rc = printf(
"%*c", (
int)(description_offset - pos),
' ');
556 if (arg->Flags & COMMAND_LINE_VALUE_BOOL)
558 rc = printf(
"%s ", arg->Default ?
"Disable" :
"Enable");
564 print_description(arg->Text, description_offset, pos);
565 }
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
568BOOL freerdp_client_print_command_line_help(
int argc,
char** argv)
570 return freerdp_client_print_command_line_help_ex(argc, argv, NULL);
574 SSIZE_T count,
size_t* pcount)
576 WINPR_ASSERT(pcount);
581 while (cur && cur->Name)
596 while (cur && cur->Name)
598 largs[lcount++] = *cur++;
601 cur = global_cmd_args;
602 while (cur && cur->Name)
604 largs[lcount++] = *cur++;
610BOOL freerdp_client_print_command_line_help_ex(
int argc,
char** argv,
613 const char* name =
"FreeRDP";
625 printf(
"FreeRDP - A Free Remote Desktop Protocol Implementation\n");
626 printf(
"See www.freerdp.com for more information\n");
628 printf(
"Usage: %s [file] [options] [/v:<server>[:port]]\n", argv[0]);
631 printf(
" /flag (enables flag)\n");
632 printf(
" /option:<value> (specifies option with value)\n");
633 printf(
" +toggle -toggle (enables or disables toggle, where '/' is a synonym of '+')\n");
636 freerdp_client_print_command_line_args(largs, lcount);
640 printf(
"Examples:\n");
641 printf(
" %s connection.rdp /p:Pwd123! /f\n", name);
642 printf(
" %s /u:CONTOSO\\JohnDoe /p:Pwd123! /v:rdp.contoso.com\n", name);
643 printf(
" %s /u:JohnDoe /p:Pwd123! /w:1366 /h:768 /v:192.168.1.100:4489\n", name);
644 printf(
" %s /u:JohnDoe /p:Pwd123! /vmconnect:C824F53E-95D2-46C6-9A18-23A5BB403532 "
645 "/v:192.168.1.100\n",
647 printf(
" %s /u:\\AzureAD\\user@corp.example /p:pwd /v:host\n", name);
648 printf(
"Use a generic pipe as transport:");
649 printf(
" %s /v:/path/to/pipe\n", name);
650 printf(
"Use a external socket:");
651 printf(
" %s /v:|:1234\n", name);
653 printf(
"Clipboard Redirection: +clipboard\n");
655 printf(
"Drive Redirection: /drive:home,/home/user\n");
656 printf(
"Smartcard Redirection: /smartcard:<device>\n");
657 printf(
"Smartcard logon with Kerberos authentication: /smartcard-logon /sec:nla\n");
659#if defined(CHANNEL_SERIAL_CLIENT)
660 printf(
"Serial Port Redirection: /serial:<name>,<device>,[SerCx2|SerCx|Serial],[permissive]\n");
661 printf(
"Serial Port Redirection: /serial:COM1,/dev/ttyS0\n");
663#if defined(CHANNEL_PARALLEL_CLIENT)
664 printf(
"Parallel Port Redirection: /parallel:<name>,<device>\n");
666 printf(
"Printer Redirection: /printer:<device>,<driver>,[default]\n");
667 printf(
"TCP redirection: /rdp2tcp:/usr/bin/rdp2tcp\n");
669 printf(
"Audio Output Redirection: /sound:sys:oss,dev:1,format:1\n");
670 printf(
"Audio Output Redirection: /sound:sys:alsa\n");
671 printf(
"Audio Input Redirection: /microphone:sys:oss,dev:1,format:1\n");
672 printf(
"Audio Input Redirection: /microphone:sys:alsa\n");
674 printf(
"Multimedia Redirection: /video\n");
675#ifdef CHANNEL_URBDRC_CLIENT
676 printf(
"USB Device Redirection: /usb:id:054c:0268#4669:6e6b,addr:04:0c\n");
679 printf(
"For Gateways, the https_proxy environment variable is respected:\n");
681 printf(
" set HTTPS_PROXY=http://proxy.contoso.com:3128/\n");
683 printf(
" export https_proxy=http://proxy.contoso.com:3128/\n");
685 printf(
" %s /g:rdp.contoso.com ...\n", name);
687 printf(
"More documentation is coming, in the meantime consult source files\n");
692static BOOL option_is_rdp_file(
const char* option)
694 WINPR_ASSERT(option);
696 if (option_ends_with(option,
".rdp"))
698 if (option_ends_with(option,
".rdpw"))
703static BOOL option_is_incident_file(
const char* option)
705 WINPR_ASSERT(option);
707 if (option_ends_with(option,
".msrcIncident"))
712static int freerdp_client_command_line_pre_filter(
void* context,
int index,
int argc, LPSTR* argv)
717 rdpSettings* settings = NULL;
722 length = strlen(argv[index]);
726 if (option_is_rdp_file(argv[index]))
728 settings = (rdpSettings*)context;
731 return COMMAND_LINE_ERROR_MEMORY;
739 if (option_is_incident_file(argv[index]))
741 settings = (rdpSettings*)context;
744 return COMMAND_LINE_ERROR_MEMORY;
754BOOL freerdp_client_add_device_channel(rdpSettings* settings,
size_t count,
755 const char*
const* params)
757 WINPR_ASSERT(settings);
758 WINPR_ASSERT(params);
759 WINPR_ASSERT(count > 0);
761 if (option_equals(params[0],
"drive"))
770 rc = freerdp_client_add_drive(settings, params[1], NULL);
772 rc = freerdp_client_add_drive(settings, params[2], params[1]);
776 else if (option_equals(params[0],
"printer"))
788 printer = freerdp_device_new(RDPDR_DTYP_PRINT, count - 1, ¶ms[1]);
792 if (!freerdp_device_collection_add(settings, printer))
794 freerdp_device_free(printer);
800 else if (option_equals(params[0],
"smartcard"))
812 smartcard = freerdp_device_new(RDPDR_DTYP_SMARTCARD, count - 1, ¶ms[1]);
817 if (!freerdp_device_collection_add(settings, smartcard))
819 freerdp_device_free(smartcard);
825#if defined(CHANNEL_SERIAL_CLIENT)
826 else if (option_equals(params[0],
"serial"))
838 serial = freerdp_device_new(RDPDR_DTYP_SERIAL, count - 1, ¶ms[1]);
843 if (!freerdp_device_collection_add(settings, serial))
845 freerdp_device_free(serial);
852 else if (option_equals(params[0],
"parallel"))
864 parallel = freerdp_device_new(RDPDR_DTYP_PARALLEL, count - 1, ¶ms[1]);
869 if (!freerdp_device_collection_add(settings, parallel))
871 freerdp_device_free(parallel);
881BOOL freerdp_client_del_static_channel(rdpSettings* settings,
const char* name)
883 return freerdp_static_channel_collection_del(settings, name);
886BOOL freerdp_client_add_static_channel(rdpSettings* settings,
size_t count,
887 const char*
const* params)
891 if (!settings || !params || !params[0] || (count > INT_MAX))
894 if (freerdp_static_channel_collection_find(settings, params[0]))
897 _args = freerdp_addin_argv_new(count, params);
902 if (!freerdp_static_channel_collection_add(settings, _args))
907 freerdp_addin_argv_free(_args);
911BOOL freerdp_client_del_dynamic_channel(rdpSettings* settings,
const char* name)
913 return freerdp_dynamic_channel_collection_del(settings, name);
916BOOL freerdp_client_add_dynamic_channel(rdpSettings* settings,
size_t count,
917 const char*
const* params)
921 if (!settings || !params || !params[0] || (count > INT_MAX))
924 if (freerdp_dynamic_channel_collection_find(settings, params[0]))
927 _args = freerdp_addin_argv_new(count, params);
932 if (!freerdp_dynamic_channel_collection_add(settings, _args))
938 freerdp_addin_argv_free(_args);
942static BOOL read_pem_file(rdpSettings* settings, FreeRDP_Settings_Keys_String
id,
const char* file)
945 char* pem = crypto_read_pem(file, &length);
946 if (!pem || (length == 0))
960 CMDLINE_SUBOPTION_STRING,
961 CMDLINE_SUBOPTION_FILE,
962} CmdLineSubOptionType;
964typedef BOOL (*CmdLineSubOptionCb)(
const char* value, rdpSettings* settings);
968 FreeRDP_Settings_Keys_String id;
969 CmdLineSubOptionType opttype;
970 CmdLineSubOptionCb cb;
973static BOOL parseSubOptions(rdpSettings* settings,
const CmdLineSubOptions* opts,
size_t count,
978 for (
size_t xx = 0; xx < count; xx++)
980 const CmdLineSubOptions* opt = &opts[xx];
982 if (option_starts_with(opt->optname, arg))
984 const size_t optlen = strlen(opt->optname);
985 const char* val = &arg[optlen];
988 switch (opt->opttype)
990 case CMDLINE_SUBOPTION_STRING:
993 case CMDLINE_SUBOPTION_FILE:
994 status = read_pem_file(settings, opt->id, val);
997 WLog_ERR(TAG,
"invalid subOption type");
1004 if (opt->cb && !opt->cb(val, settings))
1013 WLog_ERR(TAG,
"option %s not handled", arg);
1018#define fail_at(arg, rc) fail_at_((arg), (rc), __FILE__, __func__, __LINE__)
1022 const DWORD level = WLOG_ERROR;
1023 wLog* log = WLog_Get(TAG);
1024 if (WLog_IsLevelActive(log, level))
1025 WLog_PrintMessage(log, WLOG_MESSAGE_TEXT, level, line, file, fkt,
1026 "Command line parsing failed at '%s' value '%s' [%d]", arg->Name,
1033 rdpSettings* settings = (rdpSettings*)context;
1034 int status = CHANNEL_RC_OK;
1035 BOOL enable = arg->Value ? TRUE : FALSE;
1037 CommandLineSwitchStart(arg) CommandLineSwitchCase(arg,
"a")
1040 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
1042 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1043 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1045 status = COMMAND_LINE_ERROR;
1047 CommandLineParserFree(ptr);
1049 return fail_at(arg, status);
1051 CommandLineSwitchCase(arg,
"kerberos")
1055 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"kerberos", arg->Value, &count);
1058 const CmdLineSubOptions opts[] = {
1059 {
"kdc-url:", FreeRDP_KerberosKdcUrl, CMDLINE_SUBOPTION_STRING, NULL },
1060 {
"start-time:", FreeRDP_KerberosStartTime, CMDLINE_SUBOPTION_STRING, NULL },
1061 {
"lifetime:", FreeRDP_KerberosLifeTime, CMDLINE_SUBOPTION_STRING, NULL },
1062 {
"renewable-lifetime:", FreeRDP_KerberosRenewableLifeTime,
1063 CMDLINE_SUBOPTION_STRING, NULL },
1064 {
"cache:", FreeRDP_KerberosCache, CMDLINE_SUBOPTION_STRING, NULL },
1065 {
"armor:", FreeRDP_KerberosArmor, CMDLINE_SUBOPTION_STRING, NULL },
1066 {
"pkinit-anchors:", FreeRDP_PkinitAnchors, CMDLINE_SUBOPTION_STRING, NULL },
1067 {
"pkcs11-module:", FreeRDP_Pkcs11Module, CMDLINE_SUBOPTION_STRING, NULL }
1070 for (
size_t x = 1; x < count; x++)
1072 const char* cur = ptr[x];
1073 if (!parseSubOptions(settings, opts, ARRAYSIZE(opts), cur))
1075 CommandLineParserFree(ptr);
1076 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
1080 CommandLineParserFree(ptr);
1083 CommandLineSwitchCase(arg,
"vc")
1086 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
1087 if (!freerdp_client_add_static_channel(settings, count, (
const char*
const*)ptr))
1088 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1089 CommandLineParserFree(ptr);
1091 return fail_at(arg, status);
1093 CommandLineSwitchCase(arg,
"dvc")
1096 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
1097 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1098 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1099 CommandLineParserFree(ptr);
1101 return fail_at(arg, status);
1103 CommandLineSwitchCase(arg,
"drive")
1106 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1107 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1108 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1109 CommandLineParserFree(ptr);
1111 return fail_at(arg, status);
1113#if defined(CHANNEL_SERIAL_CLIENT)
1114 CommandLineSwitchCase(arg,
"serial")
1117 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1118 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1119 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1120 CommandLineParserFree(ptr);
1122 return fail_at(arg, status);
1125#if defined(CHANNEL_PARALLEL_CLIENT)
1126 CommandLineSwitchCase(arg,
"parallel")
1129 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1130 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1131 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1132 CommandLineParserFree(ptr);
1134 return fail_at(arg, status);
1137 CommandLineSwitchCase(arg,
"smartcard")
1140 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1141 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1142 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1143 CommandLineParserFree(ptr);
1145 return fail_at(arg, status);
1147 CommandLineSwitchCase(arg,
"printer")
1150 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1151 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1152 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1153 CommandLineParserFree(ptr);
1155 return fail_at(arg, status);
1157 CommandLineSwitchCase(arg,
"usb")
1161 CommandLineParseCommaSeparatedValuesEx(URBDRC_CHANNEL_NAME, arg->Value, &count);
1162 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1163 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1164 CommandLineParserFree(ptr);
1166 return fail_at(arg, status);
1168 CommandLineSwitchCase(arg,
"multitouch")
1171 return fail_at(arg, COMMAND_LINE_ERROR);
1173 CommandLineSwitchCase(arg,
"gestures")
1176 return fail_at(arg, COMMAND_LINE_ERROR);
1178 CommandLineSwitchCase(arg,
"echo")
1181 return fail_at(arg, COMMAND_LINE_ERROR);
1183 CommandLineSwitchCase(arg,
"ssh-agent")
1186 return fail_at(arg, COMMAND_LINE_ERROR);
1188 CommandLineSwitchCase(arg,
"disp")
1191 return fail_at(arg, COMMAND_LINE_ERROR);
1193 CommandLineSwitchCase(arg,
"geometry")
1196 return fail_at(arg, COMMAND_LINE_ERROR);
1198 CommandLineSwitchCase(arg,
"video")
1202 return fail_at(arg, COMMAND_LINE_ERROR);
1204 return fail_at(arg, COMMAND_LINE_ERROR);
1206 CommandLineSwitchCase(arg,
"sound")
1210 CommandLineParseCommaSeparatedValuesEx(RDPSND_CHANNEL_NAME, arg->Value, &count);
1211 if (!freerdp_client_add_static_channel(settings, count, (
const char*
const*)ptr))
1212 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1213 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1214 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1216 CommandLineParserFree(ptr);
1218 return fail_at(arg, status);
1220 CommandLineSwitchCase(arg,
"microphone")
1223 char** ptr = CommandLineParseCommaSeparatedValuesEx(AUDIN_CHANNEL_NAME, arg->Value, &count);
1224 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1225 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1226 CommandLineParserFree(ptr);
1228 return fail_at(arg, status);
1230#if defined(CHANNEL_TSMF_CLIENT)
1231 CommandLineSwitchCase(arg,
"multimedia")
1234 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"tsmf", arg->Value, &count);
1235 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1236 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1237 CommandLineParserFree(ptr);
1239 return fail_at(arg, status);
1242 CommandLineSwitchCase(arg,
"heartbeat")
1245 return fail_at(arg, COMMAND_LINE_ERROR);
1247 CommandLineSwitchCase(arg,
"multitransport")
1250 return fail_at(arg, COMMAND_LINE_ERROR);
1255 (TRANSPORT_TYPE_UDP_FECR | TRANSPORT_TYPE_UDP_FECL | TRANSPORT_TYPE_UDP_PREFERRED);
1258 return fail_at(arg, COMMAND_LINE_ERROR);
1260 CommandLineSwitchEnd(arg)
1267 int status = freerdp_client_command_line_post_filter_int(context, arg);
1268 return status == CHANNEL_RC_OK ? 1 : -1;
1271static BOOL freerdp_parse_username_ptr(
const char* username,
const char** user,
size_t* userlen,
1272 const char** domain,
size_t* domainlen)
1275 WINPR_ASSERT(userlen);
1276 WINPR_ASSERT(domain);
1277 WINPR_ASSERT(domainlen);
1282 const char* p = strchr(username,
'\\');
1292 const size_t length = (size_t)(p - username);
1294 *userlen = strlen(*user);
1297 *domainlen = length;
1306 *userlen = strlen(username);
1312static BOOL freerdp_parse_username_settings(
const char* username, rdpSettings* settings,
1313 FreeRDP_Settings_Keys_String userID,
1314 FreeRDP_Settings_Keys_String domainID)
1316 const char* user = NULL;
1317 const char* domain = NULL;
1319 size_t domainlen = 0;
1321 const BOOL rc = freerdp_parse_username_ptr(username, &user, &userlen, &domain, &domainlen);
1329BOOL freerdp_parse_username(
const char* username,
char** puser,
char** pdomain)
1331 const char* user = NULL;
1332 const char* domain = NULL;
1334 size_t domainlen = 0;
1339 const BOOL rc = freerdp_parse_username_ptr(username, &user, &userlen, &domain, &domainlen);
1345 *puser = strndup(user, userlen);
1352 *pdomain = strndup(domain, domainlen);
1364BOOL freerdp_parse_hostname(
const char* hostname,
char** host,
int* port)
1367 p = strrchr(hostname,
':');
1371 size_t length = (size_t)(p - hostname);
1374 if (!value_to_int(p + 1, &val, 1, UINT16_MAX))
1377 *host = (
char*)calloc(length + 1UL,
sizeof(
char));
1382 CopyMemory(*host, hostname, length);
1383 (*host)[length] =
'\0';
1384 *port = (UINT16)val;
1388 *host = _strdup(hostname);
1399static BOOL freerdp_apply_connection_type(rdpSettings* settings, UINT32 type)
1401 struct network_settings
1403 FreeRDP_Settings_Keys_Bool id;
1406 const struct network_settings config[] = {
1407 { FreeRDP_DisableWallpaper, { TRUE, TRUE, TRUE, TRUE, FALSE, FALSE, FALSE } },
1408 { FreeRDP_AllowFontSmoothing, { FALSE, FALSE, FALSE, FALSE, TRUE, TRUE, TRUE } },
1409 { FreeRDP_AllowDesktopComposition, { FALSE, FALSE, TRUE, TRUE, TRUE, TRUE, TRUE } },
1410 { FreeRDP_DisableFullWindowDrag, { TRUE, TRUE, TRUE, TRUE, FALSE, FALSE, FALSE } },
1411 { FreeRDP_DisableMenuAnims, { TRUE, TRUE, TRUE, TRUE, FALSE, FALSE, FALSE } },
1412 { FreeRDP_DisableThemes, { TRUE, FALSE, FALSE, FALSE, FALSE, FALSE, FALSE } }
1417 case CONNECTION_TYPE_INVALID:
1420 case CONNECTION_TYPE_MODEM:
1421 case CONNECTION_TYPE_BROADBAND_LOW:
1422 case CONNECTION_TYPE_BROADBAND_HIGH:
1423 case CONNECTION_TYPE_SATELLITE:
1424 case CONNECTION_TYPE_WAN:
1425 case CONNECTION_TYPE_LAN:
1426 case CONNECTION_TYPE_AUTODETECT:
1429 WLog_WARN(TAG,
"Unknown ConnectionType %" PRIu32
", aborting", type);
1433 for (
size_t x = 0; x < ARRAYSIZE(config); x++)
1435 const struct network_settings* cur = &config[x];
1442BOOL freerdp_set_connection_type(rdpSettings* settings, UINT32 type)
1450 case CONNECTION_TYPE_INVALID:
1451 case CONNECTION_TYPE_MODEM:
1452 case CONNECTION_TYPE_BROADBAND_LOW:
1453 case CONNECTION_TYPE_SATELLITE:
1454 case CONNECTION_TYPE_BROADBAND_HIGH:
1455 case CONNECTION_TYPE_WAN:
1456 case CONNECTION_TYPE_LAN:
1457 if (!freerdp_apply_connection_type(settings, type))
1460 case CONNECTION_TYPE_AUTODETECT:
1461 if (!freerdp_apply_connection_type(settings, type))
1475 WLog_WARN(TAG,
"Unknown ConnectionType %" PRIu32
", aborting", type);
1482static UINT32 freerdp_get_keyboard_layout_for_type(
const char* name, WINPR_ATTR_UNUSED DWORD type)
1487 freerdp_keyboard_get_layouts(RDP_KEYBOARD_LAYOUT_TYPE_STANDARD, &count);
1489 if (!layouts || (count == 0))
1492 for (
size_t x = 0; x < count; x++)
1495 if (option_equals(layout->name, name))
1503 freerdp_keyboard_layouts_free(layouts, count);
1507static UINT32 freerdp_map_keyboard_layout_name_to_id(
const char* name)
1509 const UINT32 variants[] = { RDP_KEYBOARD_LAYOUT_TYPE_STANDARD, RDP_KEYBOARD_LAYOUT_TYPE_VARIANT,
1510 RDP_KEYBOARD_LAYOUT_TYPE_IME };
1512 for (
size_t x = 0; x < ARRAYSIZE(variants); x++)
1514 UINT32 rc = freerdp_get_keyboard_layout_for_type(name, variants[x]);
1522static int freerdp_detect_command_line_pre_filter(
void* context,
int index,
int argc, LPSTR* argv)
1525 WINPR_UNUSED(context);
1532 length = strlen(argv[index]);
1536 if (option_is_rdp_file(argv[index]))
1544 if (option_is_incident_file(argv[index]))
1554static int freerdp_detect_windows_style_command_line_syntax(
int argc,
char** argv,
size_t* count,
1559 int detect_status = 0;
1562 memcpy(largs, global_cmd_args,
sizeof(global_cmd_args));
1564 flags = COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_SILENCE_PARSER;
1565 flags |= COMMAND_LINE_SIGIL_SLASH | COMMAND_LINE_SIGIL_PLUS_MINUS;
1569 flags |= COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
1574 CommandLineClearArgumentsA(largs);
1575 status = CommandLineParseArgumentsA(argc, argv, largs, flags, NULL,
1576 freerdp_detect_command_line_pre_filter, NULL);
1585 if (!(arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
1589 }
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
1591 return detect_status;
1594static int freerdp_detect_posix_style_command_line_syntax(
int argc,
char** argv,
size_t* count,
1599 int detect_status = 0;
1602 memcpy(largs, global_cmd_args,
sizeof(global_cmd_args));
1604 flags = COMMAND_LINE_SEPARATOR_SPACE | COMMAND_LINE_SILENCE_PARSER;
1605 flags |= COMMAND_LINE_SIGIL_DASH | COMMAND_LINE_SIGIL_DOUBLE_DASH;
1606 flags |= COMMAND_LINE_SIGIL_ENABLE_DISABLE;
1610 flags |= COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
1615 CommandLineClearArgumentsA(largs);
1616 status = CommandLineParseArgumentsA(argc, argv, largs, flags, NULL,
1617 freerdp_detect_command_line_pre_filter, NULL);
1626 if (!(arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
1630 }
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
1632 return detect_status;
1635static BOOL freerdp_client_detect_command_line(
int argc,
char** argv, DWORD* flags)
1637 size_t posix_cli_count = 0;
1638 size_t windows_cli_count = 0;
1639 const BOOL ignoreUnknown = TRUE;
1640 const int windows_cli_status = freerdp_detect_windows_style_command_line_syntax(
1641 argc, argv, &windows_cli_count, ignoreUnknown);
1642 const int posix_cli_status =
1643 freerdp_detect_posix_style_command_line_syntax(argc, argv, &posix_cli_count, ignoreUnknown);
1646 *flags = COMMAND_LINE_SEPARATOR_SPACE;
1647 *flags |= COMMAND_LINE_SIGIL_DASH | COMMAND_LINE_SIGIL_DOUBLE_DASH;
1648 *flags |= COMMAND_LINE_SIGIL_ENABLE_DISABLE;
1650 if (posix_cli_status <= COMMAND_LINE_STATUS_PRINT)
1654 if ((windows_cli_count && (windows_cli_count >= posix_cli_count)) ||
1655 (windows_cli_status <= COMMAND_LINE_STATUS_PRINT))
1657 windows_cli_count = 1;
1658 *flags = COMMAND_LINE_SEPARATOR_COLON;
1659 *flags |= COMMAND_LINE_SIGIL_SLASH | COMMAND_LINE_SIGIL_PLUS_MINUS;
1662 WLog_DBG(TAG,
"windows: %d/%" PRIuz
" posix: %d/%" PRIuz
"", windows_cli_status,
1663 windows_cli_count, posix_cli_status, posix_cli_count);
1664 if ((posix_cli_count == 0) && (windows_cli_count == 0))
1666 if ((posix_cli_status == COMMAND_LINE_ERROR) && (windows_cli_status == COMMAND_LINE_ERROR))
1672int freerdp_client_settings_command_line_status_print(rdpSettings* settings,
int status,
int argc,
1675 return freerdp_client_settings_command_line_status_print_ex(settings, status, argc, argv, NULL);
1678static void freerdp_client_print_keyboard_type_list(
const char* msg, DWORD type)
1682 layouts = freerdp_keyboard_get_layouts(type, &count);
1684 printf(
"\n%s\n", msg);
1686 for (
size_t x = 0; x < count; x++)
1689 printf(
"0x%08" PRIX32
"\t%s\n", layout->code, layout->name);
1692 freerdp_keyboard_layouts_free(layouts, count);
1695static void freerdp_client_print_keyboard_list(
void)
1697 freerdp_client_print_keyboard_type_list(
"Keyboard Layouts", RDP_KEYBOARD_LAYOUT_TYPE_STANDARD);
1698 freerdp_client_print_keyboard_type_list(
"Keyboard Layout Variants",
1699 RDP_KEYBOARD_LAYOUT_TYPE_VARIANT);
1700 freerdp_client_print_keyboard_type_list(
"Keyboard Layout Variants",
1701 RDP_KEYBOARD_LAYOUT_TYPE_IME);
1704static void freerdp_client_print_timezone_list(
void)
1708 while (EnumDynamicTimeZoneInformation(index++, &info) != ERROR_NO_MORE_ITEMS)
1710 char TimeZoneKeyName[ARRAYSIZE(info.TimeZoneKeyName) + 1] = { 0 };
1712 (void)ConvertWCharNToUtf8(info.TimeZoneKeyName, ARRAYSIZE(info.TimeZoneKeyName),
1713 TimeZoneKeyName, ARRAYSIZE(TimeZoneKeyName));
1714 printf(
"%" PRIu32
": '%s'\n", index, TimeZoneKeyName);
1718static void freerdp_client_print_tune_list(
const rdpSettings* settings)
1722 for (SSIZE_T x = 0; x < FreeRDP_Settings_StableAPI_MAX; x++)
1730 case RDP_SETTINGS_TYPE_BOOL:
1731 printf(
"%" PRIdz
"\t%50s\tBOOL\t%s\n", x, name,
1736 case RDP_SETTINGS_TYPE_UINT16:
1737 printf(
"%" PRIdz
"\t%50s\tUINT16\t%" PRIu16
"\n", x, name,
1740 case RDP_SETTINGS_TYPE_INT16:
1741 printf(
"%" PRIdz
"\t%50s\tINT16\t%" PRId16
"\n", x, name,
1744 case RDP_SETTINGS_TYPE_UINT32:
1745 printf(
"%" PRIdz
"\t%50s\tUINT32\t%" PRIu32
"\n", x, name,
1748 case RDP_SETTINGS_TYPE_INT32:
1749 printf(
"%" PRIdz
"\t%50s\tINT32\t%" PRId32
"\n", x, name,
1752 case RDP_SETTINGS_TYPE_UINT64:
1753 printf(
"%" PRIdz
"\t%50s\tUINT64\t%" PRIu64
"\n", x, name,
1756 case RDP_SETTINGS_TYPE_INT64:
1757 printf(
"%" PRIdz
"\t%50s\tINT64\t%" PRId64
"\n", x, name,
1760 case RDP_SETTINGS_TYPE_STRING:
1761 printf(
"%" PRIdz
"\t%50s\tSTRING\t%s"
1766 case RDP_SETTINGS_TYPE_POINTER:
1767 printf(
"%" PRIdz
"\t%50s\tPOINTER\t%p"
1779int freerdp_client_settings_command_line_status_print_ex(rdpSettings* settings,
int status,
1780 int argc,
char** argv,
1785 memcpy(largs, global_cmd_args,
sizeof(global_cmd_args));
1787 if (status == COMMAND_LINE_STATUS_PRINT_VERSION)
1789 freerdp_client_print_version();
1793 if (status == COMMAND_LINE_STATUS_PRINT_BUILDCONFIG)
1795 freerdp_client_print_version_ex(argc, argv);
1796 freerdp_client_print_buildconfig_ex(argc, argv);
1799 else if (status == COMMAND_LINE_STATUS_PRINT)
1801 (void)CommandLineParseArgumentsA(argc, argv, largs, 0x112, NULL, NULL, NULL);
1803 arg = CommandLineFindArgumentA(largs,
"list");
1806 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
1808 if (option_equals(
"timezones", arg->Value))
1809 freerdp_client_print_timezone_list();
1810 else if (option_equals(
"tune", arg->Value))
1811 freerdp_client_print_tune_list(settings);
1812 else if (option_equals(
"kbd", arg->Value))
1813 freerdp_client_print_keyboard_list();
1814 else if (option_starts_with(
"kbd-lang", arg->Value))
1816 const char* val = NULL;
1817 if (option_starts_with(
"kbd-lang:", arg->Value))
1818 val = &arg->Value[9];
1819 else if (!option_equals(
"kbd-lang", arg->Value))
1820 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1822 if (val && strchr(val,
','))
1823 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1824 freerdp_client_print_codepages(val);
1826 else if (option_equals(
"kbd-scancode", arg->Value))
1827 freerdp_client_print_scancodes();
1828 else if (option_equals(
"monitor", arg->Value))
1831 return COMMAND_LINE_ERROR;
1833 else if (option_starts_with(
"smartcard", arg->Value))
1836 if (option_starts_with(
"smartcard:", arg->Value))
1838 else if (!option_equals(
"smartcard", arg->Value))
1839 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1843 const char* sub = strchr(arg->Value,
':') + 1;
1844 const CmdLineSubOptions options[] = {
1845 {
"pkinit-anchors:", FreeRDP_PkinitAnchors, CMDLINE_SUBOPTION_STRING,
1847 {
"pkcs11-module:", FreeRDP_Pkcs11Module, CMDLINE_SUBOPTION_STRING, NULL }
1852 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"smartcard", sub, &count);
1854 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1857 CommandLineParserFree(ptr);
1858 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1861 for (
size_t x = 1; x < count; x++)
1863 const char* cur = ptr[x];
1864 if (!parseSubOptions(settings, options, ARRAYSIZE(options), cur))
1866 CommandLineParserFree(ptr);
1867 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1871 CommandLineParserFree(ptr);
1874 freerdp_smartcard_list(settings);
1878 freerdp_client_print_command_line_help_ex(argc, argv, custom);
1879 return COMMAND_LINE_ERROR;
1882#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
1883 arg = CommandLineFindArgumentA(largs,
"tune-list");
1886 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
1888 WLog_WARN(TAG,
"Option /tune-list is deprecated, use /list:tune instead");
1889 freerdp_client_print_tune_list(settings);
1892 arg = CommandLineFindArgumentA(largs,
"kbd-lang-list");
1895 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
1897 WLog_WARN(TAG,
"Option /kbd-lang-list is deprecated, use /list:kbd-lang instead");
1898 freerdp_client_print_codepages(arg->Value);
1901 arg = CommandLineFindArgumentA(largs,
"kbd-list");
1904 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1906 WLog_WARN(TAG,
"Option /kbd-list is deprecated, use /list:kbd instead");
1907 freerdp_client_print_keyboard_list();
1910 arg = CommandLineFindArgumentA(largs,
"monitor-list");
1913 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1915 WLog_WARN(TAG,
"Option /monitor-list is deprecated, use /list:monitor instead");
1917 return COMMAND_LINE_ERROR;
1920 arg = CommandLineFindArgumentA(largs,
"smartcard-list");
1923 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1925 WLog_WARN(TAG,
"Option /smartcard-list is deprecated, use /list:smartcard instead");
1926 freerdp_smartcard_list(settings);
1929 arg = CommandLineFindArgumentA(largs,
"kbd-scancode-list");
1932 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1935 "Option /kbd-scancode-list is deprecated, use /list:kbd-scancode instead");
1936 freerdp_client_print_scancodes();
1942 else if (status < 0)
1944 freerdp_client_print_command_line_help_ex(argc, argv, custom);
1949 if (status <= COMMAND_LINE_STATUS_PRINT && status >= COMMAND_LINE_STATUS_PRINT_LAST)
1962static BOOL parseSizeValue(
const char* input,
unsigned long* v1,
unsigned long* v2)
1964 const char* xcharpos = NULL;
1965 char* endPtr = NULL;
1966 unsigned long v = 0;
1968 v = strtoul(input, &endPtr, 10);
1970 if ((v == 0 || v == ULONG_MAX) && (errno != 0))
1976 xcharpos = strchr(input,
'x');
1978 if (!xcharpos || xcharpos != endPtr)
1982 v = strtoul(xcharpos + 1, &endPtr, 10);
1984 if ((v == 0 || v == ULONG_MAX) && (errno != 0))
1987 if (*endPtr !=
'\0')
1999 const char* arguments[] = {
"network",
"gfx",
"rfx",
"bpp" };
2000 WINPR_ASSERT(settings);
2006 for (
size_t x = 0; x < ARRAYSIZE(arguments); x++)
2008 const char* arg = arguments[x];
2010 if (p && (p->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
2014 return freerdp_set_connection_type(settings, CONNECTION_TYPE_AUTODETECT);
2017static BOOL setSmartcardEmulation(WINPR_ATTR_UNUSED
const char* value, rdpSettings* settings)
2022const char* option_starts_with(
const char* what,
const char* val)
2026 const size_t wlen = strlen(what);
2028 if (_strnicmp(what, val, wlen) != 0)
2033BOOL option_ends_with(
const char* str,
const char* ext)
2037 const size_t strLen = strlen(str);
2038 const size_t extLen = strlen(ext);
2040 if (strLen < extLen)
2043 return _strnicmp(&str[strLen - extLen], ext, extLen) == 0;
2046BOOL option_equals(
const char* what,
const char* val)
2050 return _stricmp(what, val) == 0;
2059} PARSE_ON_OFF_RESULT;
2061static PARSE_ON_OFF_RESULT parse_on_off_option(
const char* value)
2063 WINPR_ASSERT(value);
2064 const char* sep = strchr(value,
':');
2067 if (option_equals(
"on", &sep[1]))
2069 if (option_equals(
"off", &sep[1]))
2078 CLIP_DIR_PARSE_LOCAL,
2079 CLIP_DIR_PARSE_REMOTE,
2081} PARSE_CLIP_DIR_RESULT;
2083static PARSE_CLIP_DIR_RESULT parse_clip_direciton_to_option(
const char* value)
2085 WINPR_ASSERT(value);
2086 const char* sep = strchr(value,
':');
2088 return CLIP_DIR_PARSE_FAIL;
2089 if (option_equals(
"all", &sep[1]))
2090 return CLIP_DIR_PARSE_ALL;
2091 if (option_equals(
"off", &sep[1]))
2092 return CLIP_DIR_PARSE_OFF;
2093 if (option_equals(
"local", &sep[1]))
2094 return CLIP_DIR_PARSE_LOCAL;
2095 if (option_equals(
"remote", &sep[1]))
2096 return CLIP_DIR_PARSE_REMOTE;
2097 return CLIP_DIR_PARSE_FAIL;
2100static int parse_tls_ciphers(rdpSettings* settings,
const char* Value)
2102 const char* ciphers = NULL;
2104 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2106 if (option_equals(Value,
"netmon"))
2108 ciphers =
"ALL:!ECDH:!ADH:!DHE";
2110 else if (option_equals(Value,
"ma"))
2112 ciphers =
"AES128-SHA";
2120 return COMMAND_LINE_ERROR_MEMORY;
2124static int parse_tls_seclevel(rdpSettings* settings,
const char* Value)
2128 if (!value_to_int(Value, &val, 0, 5))
2129 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2132 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2136static int parse_tls_secrets_file(rdpSettings* settings,
const char* Value)
2139 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2142 return COMMAND_LINE_ERROR_MEMORY;
2146static int parse_tls_enforce(rdpSettings* settings,
const char* Value)
2148 UINT16 version = TLS1_2_VERSION;
2157 const struct map_t map[] = { {
"1.0", TLS1_VERSION },
2158 {
"1.1", TLS1_1_VERSION },
2159 {
"1.2", TLS1_2_VERSION }
2160#if defined(TLS1_3_VERSION)
2162 {
"1.3", TLS1_3_VERSION }
2166 const struct map_t* found = NULL;
2167 for (
size_t x = 0; x < ARRAYSIZE(map); x++)
2169 const struct map_t* cur = &map[x];
2170 if (option_equals(cur->name, Value))
2178 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2179 version = found->version;
2184 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2190 int rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2191 CommandLineSwitchStart(arg) CommandLineSwitchCase(arg,
"tls")
2193 if (option_starts_with(
"ciphers:", arg->Value))
2194 rc = fail_at(arg, parse_tls_ciphers(settings, &arg->Value[8]));
2195 else if (option_starts_with(
"seclevel:", arg->Value))
2196 rc = fail_at(arg, parse_tls_seclevel(settings, &arg->Value[9]));
2197 else if (option_starts_with(
"secrets-file:", arg->Value))
2198 rc = fail_at(arg, parse_tls_secrets_file(settings, &arg->Value[13]));
2199 else if (option_starts_with(
"enforce:", arg->Value))
2200 rc = fail_at(arg, parse_tls_enforce(settings, &arg->Value[8]));
2203#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
2204 CommandLineSwitchCase(arg,
"tls-ciphers")
2206 WLog_WARN(TAG,
"Option /tls-ciphers is deprecated, use /tls:ciphers instead");
2207 rc = fail_at(arg, parse_tls_ciphers(settings, arg->Value));
2209 CommandLineSwitchCase(arg,
"tls-seclevel")
2211 WLog_WARN(TAG,
"Option /tls-seclevel is deprecated, use /tls:seclevel instead");
2212 rc = fail_at(arg, parse_tls_seclevel(settings, arg->Value));
2214 CommandLineSwitchCase(arg,
"tls-secrets-file")
2216 WLog_WARN(TAG,
"Option /tls-secrets-file is deprecated, use /tls:secrets-file instead");
2217 rc = fail_at(arg, parse_tls_secrets_file(settings, arg->Value));
2219 CommandLineSwitchCase(arg,
"enforce-tlsv1_2")
2221 WLog_WARN(TAG,
"Option /enforce-tlsv1_2 is deprecated, use /tls:enforce:1.2 instead");
2222 rc = fail_at(arg, parse_tls_enforce(settings,
"1.2"));
2225 CommandLineSwitchDefault(arg)
2228 CommandLineSwitchEnd(arg)
2235 WINPR_ASSERT(settings);
2239 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2240 for (
size_t x = 0; x < count; x++)
2243 larg.Value = ptr[x];
2245 int rc = parse_tls_cipher_options(settings, &larg);
2248 CommandLineParserFree(ptr);
2252 CommandLineParserFree(ptr);
2258 WINPR_ASSERT(settings);
2262 return COMMAND_LINE_ERROR;
2266 int rc = CHANNEL_RC_OK;
2268 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2269 if (!ptr || (count == 0))
2270 rc = COMMAND_LINE_ERROR;
2273 BOOL GfxH264 = FALSE;
2274 BOOL GfxAVC444 = FALSE;
2275 BOOL RemoteFxCodec = FALSE;
2276 BOOL GfxProgressive = FALSE;
2277 BOOL codecSelected = FALSE;
2279 for (
size_t x = 0; x < count; x++)
2281 const char* val = ptr[x];
2283 if (option_starts_with(
"AVC444", val))
2285 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2286 if (bval == PARSE_FAIL)
2287 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2289 GfxAVC444 = bval != PARSE_OFF;
2290 codecSelected = TRUE;
2292 else if (option_starts_with(
"AVC420", val))
2294 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2295 if (bval == PARSE_FAIL)
2296 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2298 GfxH264 = bval != PARSE_OFF;
2299 codecSelected = TRUE;
2303 if (option_starts_with(
"RFX", val))
2305 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2306 if (bval == PARSE_FAIL)
2307 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2309 RemoteFxCodec = bval != PARSE_OFF;
2310 codecSelected = TRUE;
2312 else if (option_starts_with(
"progressive", val))
2314 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2315 if (bval == PARSE_FAIL)
2316 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2318 GfxProgressive = bval != PARSE_OFF;
2319 codecSelected = TRUE;
2321 else if (option_starts_with(
"mask:", val))
2324 const char* uv = &val[5];
2325 if (!value_to_uint(uv, &v, 0, UINT32_MAX))
2326 rc = COMMAND_LINE_ERROR;
2331 rc = COMMAND_LINE_ERROR;
2334 else if (option_starts_with(
"small-cache", val))
2336 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2337 if (bval == PARSE_FAIL)
2338 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2341 rc = COMMAND_LINE_ERROR;
2343 else if (option_starts_with(
"thin-client", val))
2345 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2346 if (bval == PARSE_FAIL)
2347 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2350 rc = COMMAND_LINE_ERROR;
2351 if ((rc == CHANNEL_RC_OK) && (bval > 0))
2355 rc = COMMAND_LINE_ERROR;
2358 else if (option_starts_with(
"frame-ack", val))
2360 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2361 if (bval == PARSE_FAIL)
2362 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2365 rc = COMMAND_LINE_ERROR;
2368 rc = COMMAND_LINE_ERROR;
2371 if ((rc == CHANNEL_RC_OK) && codecSelected)
2374 rc = COMMAND_LINE_ERROR;
2376 rc = COMMAND_LINE_ERROR;
2378 rc = COMMAND_LINE_ERROR;
2380 rc = COMMAND_LINE_ERROR;
2382 rc = COMMAND_LINE_ERROR;
2385 CommandLineParserFree(ptr);
2386 if (rc != CHANNEL_RC_OK)
2389 return CHANNEL_RC_OK;
2392static int parse_kbd_layout(rdpSettings* settings,
const char* value)
2394 WINPR_ASSERT(settings);
2395 WINPR_ASSERT(value);
2399 const BOOL isInt = value_to_int(value, &ival, 1, UINT32_MAX);
2402 ival = freerdp_map_keyboard_layout_name_to_id(value);
2406 WLog_ERR(TAG,
"Could not identify keyboard layout: %s", value);
2407 WLog_ERR(TAG,
"Use /list:kbd to list available layouts");
2408 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2415 rc = COMMAND_LINE_ERROR;
2420#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
2423 WINPR_ASSERT(settings);
2427 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2429 return COMMAND_LINE_ERROR;
2431 if (option_equals(arg->Value,
"rfx"))
2434 return COMMAND_LINE_ERROR;
2436 else if (option_equals(arg->Value,
"nsc"))
2439 return COMMAND_LINE_ERROR;
2442#if defined(WITH_JPEG)
2443 else if (option_equals(arg->Value,
"jpeg"))
2446 return COMMAND_LINE_ERROR;
2451 return COMMAND_LINE_ERROR;
2460static BOOL check_kbd_remap_valid(
const char* token)
2465 WINPR_ASSERT(token);
2467 if (strlen(token) > 10)
2470 if (!freerdp_extract_key_value(token, &key, &value))
2472 WLog_WARN(TAG,
"/kbd:remap invalid entry '%s'", token);
2480 WINPR_ASSERT(settings);
2484 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2486 return COMMAND_LINE_ERROR_MEMORY;
2487 char* p = strchr(arg->Value,
'[');
2492 const char scheme[] =
"://";
2493 const char* val = strstr(arg->Value, scheme);
2495 val += strnlen(scheme,
sizeof(scheme));
2498 p = strchr(val,
':');
2505 if (!value_to_int(&p[1], &lval, 1, UINT16_MAX))
2506 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2508 length = (size_t)(p - arg->Value);
2510 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2513 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2518 return COMMAND_LINE_ERROR_MEMORY;
2524 char* p2 = strchr(arg->Value,
']');
2528 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2530 length = (size_t)(p2 - p);
2532 return COMMAND_LINE_ERROR_MEMORY;
2534 if (*(p2 + 1) ==
':')
2538 if (!value_to_int(&p2[2], &val, 0, UINT16_MAX))
2539 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2542 return COMMAND_LINE_ERROR;
2545 printf(
"hostname %s port %" PRIu32
"\n",
2554 WINPR_ASSERT(settings);
2558 char* cur = arg->Value;
2560 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2562 return COMMAND_LINE_ERROR;
2568 char* next = strchr(cur,
',');
2576 if (option_equals(
"fqdn", cur))
2578 else if (option_equals(
"ip", cur))
2580 else if (option_equals(
"netbios", cur))
2583 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2586 mask = (mask & 0x07);
2587 value |= mask << (count * 3);
2589 }
while (cur != NULL);
2592 return COMMAND_LINE_ERROR;
2595 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2599static int parse_prevent_session_lock_options(rdpSettings* settings,
2602 WINPR_ASSERT(settings);
2606 return COMMAND_LINE_ERROR_MEMORY;
2608 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2612 if (!value_to_int(arg->Value, &val, 1, UINT32_MAX))
2613 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2616 return COMMAND_LINE_ERROR_MEMORY;
2624 WINPR_ASSERT(settings);
2628 return COMMAND_LINE_ERROR;
2635 return COMMAND_LINE_ERROR;
2637 return COMMAND_LINE_ERROR;
2639 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2642 return COMMAND_LINE_ERROR;
2645 return COMMAND_LINE_ERROR_MEMORY;
2653 WINPR_ASSERT(settings);
2657 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2658 char* p = strchr(arg->Value,
'x');
2662 unsigned long w = 0;
2663 unsigned long h = 0;
2665 if (!parseSizeValue(arg->Value, &w, &h) || (w > UINT16_MAX) || (h > UINT16_MAX))
2666 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2669 return COMMAND_LINE_ERROR;
2671 return COMMAND_LINE_ERROR;
2675 char* str = _strdup(arg->Value);
2677 return COMMAND_LINE_ERROR_MEMORY;
2679 p = strchr(str,
'%');
2683 BOOL partial = FALSE;
2685 status = COMMAND_LINE_ERROR;
2712 if (!value_to_int(str, &val, 0, 100))
2714 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2734 WINPR_ASSERT(settings);
2737 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2740 UINT32* MonitorIds = NULL;
2741 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2744 return COMMAND_LINE_ERROR_MEMORY;
2751 CommandLineParserFree(ptr);
2755 MonitorIds = freerdp_settings_get_pointer_array_writable(settings, FreeRDP_MonitorIds, 0);
2756 for (UINT32 i = 0; i < count; i++)
2760 if (!value_to_int(ptr[i], &val, 0, UINT16_MAX))
2761 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2763 MonitorIds[i] = (UINT32)val;
2766 CommandLineParserFree(ptr);
2772static int parse_dynamic_resolution_options(rdpSettings* settings,
2775 WINPR_ASSERT(settings);
2778 const BOOL val = arg->Value != 0;
2782 WLog_ERR(TAG,
"Smart sizing and dynamic resolution are mutually exclusive options");
2783 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2787 return COMMAND_LINE_ERROR;
2789 return COMMAND_LINE_ERROR;
2796 WINPR_ASSERT(settings);
2801 WLog_ERR(TAG,
"Smart sizing and dynamic resolution are mutually exclusive options");
2802 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2806 return COMMAND_LINE_ERROR;
2810 unsigned long w = 0;
2811 unsigned long h = 0;
2813 if (!parseSizeValue(arg->Value, &w, &h) || (w > UINT16_MAX) || (h > UINT16_MAX))
2814 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2817 return COMMAND_LINE_ERROR;
2819 return COMMAND_LINE_ERROR;
2826 WINPR_ASSERT(settings);
2831 if (!value_to_int(arg->Value, &val, 0, UINT32_MAX))
2832 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2842 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2846 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2853 WINPR_ASSERT(settings);
2856 int rc = CHANNEL_RC_OK;
2858 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2859 if (!ptr || (count == 0))
2860 rc = COMMAND_LINE_ERROR;
2863 for (
size_t x = 0; x < count; x++)
2865 const char* val = ptr[x];
2867 if (option_starts_with(
"remap:", val))
2870 char* now = _strdup(&val[6]);
2875 if (!check_kbd_remap_valid(now))
2876 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2879 const size_t olen = strlen(old);
2880 const size_t alen = strlen(now);
2881 const size_t tlen = olen + alen + 2;
2882 char* tmp = calloc(tlen,
sizeof(
char));
2884 rc = COMMAND_LINE_ERROR_MEMORY;
2886 (
void)_snprintf(tmp, tlen,
"%s,%s", old, now);
2894 rc = COMMAND_LINE_ERROR;
2898 else if (option_starts_with(
"layout:", val))
2900 rc = parse_kbd_layout(settings, &val[7]);
2902 else if (option_starts_with(
"lang:", val))
2905 const BOOL isInt = value_to_int(&val[5], &ival, 1, UINT32_MAX);
2907 ival = freerdp_get_locale_id_from_string(&val[5]);
2910 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2913 rc = COMMAND_LINE_ERROR;
2915 else if (option_starts_with(
"type:", val))
2918 const BOOL isInt = value_to_int(&val[5], &ival, 1, UINT32_MAX);
2920 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2922 rc = COMMAND_LINE_ERROR;
2924 else if (option_starts_with(
"subtype:", val))
2927 const BOOL isInt = value_to_int(&val[8], &ival, 1, UINT32_MAX);
2929 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2932 rc = COMMAND_LINE_ERROR;
2934 else if (option_starts_with(
"fn-key:", val))
2937 const BOOL isInt = value_to_int(&val[7], &ival, 1, UINT32_MAX);
2939 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2942 rc = COMMAND_LINE_ERROR;
2944 else if (option_starts_with(
"unicode", val))
2946 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2947 if (bval == PARSE_FAIL)
2948 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2951 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2953 else if (option_starts_with(
"pipe:", val))
2956 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2958 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2960#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
2961 else if (count == 1)
2964 rc = parse_kbd_layout(settings, val);
2968 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2974 CommandLineParserFree(ptr);
2980 WINPR_ASSERT(settings);
2985 return COMMAND_LINE_ERROR_MEMORY;
2987 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2989 const char* cur = arg->Value;
2992 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2994 if (!proxy_parse_uri(settings, cur))
2995 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2999 WLog_ERR(TAG,
"Option http-proxy needs argument.");
3000 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3007 WINPR_ASSERT(settings);
3010 BOOL failed = FALSE;
3012 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3017 BOOL modernsyntax = FALSE;
3018 BOOL oldsyntax = FALSE;
3019 for (
size_t x = 0; (x < count) && !failed; x++)
3021 const char* carg = ptr[x];
3022 if (option_starts_with(
"file:", carg))
3024 const char* val = &carg[5];
3029 modernsyntax = TRUE;
3031 else if (option_equals(
"replay", carg))
3038 else if (option_equals(
"record", carg))
3045 else if (option_equals(
"nodelay", carg))
3052 modernsyntax = TRUE;
3067 if (oldsyntax && (count != 2))
3070 CommandLineParserFree(ptr);
3072 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3078 WINPR_ASSERT(settings);
3081 if (arg->Value == BoolValueTrue || arg->Value == BoolValueFalse)
3084 (arg->Value == BoolValueTrue)))
3085 return COMMAND_LINE_ERROR;
3091 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3092 for (
size_t x = 0; (x < count) && (rc == 0); x++)
3094 const char* usesel =
"use-selection:";
3096 const char* cur = ptr[x];
3097 if (option_starts_with(usesel, cur))
3099 const char* val = &cur[strlen(usesel)];
3101 rc = COMMAND_LINE_ERROR_MEMORY;
3103 return COMMAND_LINE_ERROR;
3105 else if (option_starts_with(
"direction-to", cur))
3109 (uint32_t)~(CLIPRDR_FLAG_LOCAL_TO_REMOTE | CLIPRDR_FLAG_REMOTE_TO_LOCAL);
3110 const PARSE_CLIP_DIR_RESULT bval = parse_clip_direciton_to_option(cur);
3114 case CLIP_DIR_PARSE_ALL:
3115 bflags |= CLIPRDR_FLAG_LOCAL_TO_REMOTE | CLIPRDR_FLAG_REMOTE_TO_LOCAL;
3117 case CLIP_DIR_PARSE_LOCAL:
3118 bflags |= CLIPRDR_FLAG_REMOTE_TO_LOCAL;
3120 case CLIP_DIR_PARSE_REMOTE:
3121 bflags |= CLIPRDR_FLAG_LOCAL_TO_REMOTE;
3123 case CLIP_DIR_PARSE_OFF:
3125 case CLIP_DIR_PARSE_FAIL:
3127 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3133 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3135 else if (option_starts_with(
"files-to", cur))
3139 (uint32_t)~(CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES |
3140 CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES);
3141 const PARSE_CLIP_DIR_RESULT bval = parse_clip_direciton_to_option(cur);
3145 case CLIP_DIR_PARSE_ALL:
3147 CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES | CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES;
3149 case CLIP_DIR_PARSE_LOCAL:
3150 bflags |= CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES;
3152 case CLIP_DIR_PARSE_REMOTE:
3153 bflags |= CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES;
3155 case CLIP_DIR_PARSE_OFF:
3157 case CLIP_DIR_PARSE_FAIL:
3159 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3165 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3168 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3170 CommandLineParserFree(ptr);
3180 WINPR_ASSERT(settings);
3185 if (!value_to_int(arg->Value, &val, 0, UINT32_MAX))
3186 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3190 case AUDIO_MODE_REDIRECT:
3192 return COMMAND_LINE_ERROR;
3195 case AUDIO_MODE_PLAY_ON_SERVER:
3197 return COMMAND_LINE_ERROR;
3200 case AUDIO_MODE_NONE:
3202 return COMMAND_LINE_ERROR;
3204 return COMMAND_LINE_ERROR;
3208 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3215 WINPR_ASSERT(settings);
3218 UINT32 type = CONNECTION_TYPE_INVALID;
3220 if (option_equals(arg->Value,
"invalid"))
3221 type = CONNECTION_TYPE_INVALID;
3222 else if (option_equals(arg->Value,
"modem"))
3223 type = CONNECTION_TYPE_MODEM;
3224 else if (option_equals(arg->Value,
"broadband"))
3225 type = CONNECTION_TYPE_BROADBAND_HIGH;
3226 else if (option_equals(arg->Value,
"broadband-low"))
3227 type = CONNECTION_TYPE_BROADBAND_LOW;
3228 else if (option_equals(arg->Value,
"broadband-high"))
3229 type = CONNECTION_TYPE_BROADBAND_HIGH;
3230 else if (option_equals(arg->Value,
"wan"))
3231 type = CONNECTION_TYPE_WAN;
3232 else if (option_equals(arg->Value,
"lan"))
3233 type = CONNECTION_TYPE_LAN;
3234 else if ((option_equals(arg->Value,
"autodetect")) || (option_equals(arg->Value,
"auto")) ||
3235 (option_equals(arg->Value,
"detect")))
3237 type = CONNECTION_TYPE_AUTODETECT;
3243 if (!value_to_int(arg->Value, &val, 0, 7))
3244 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3249 if (!freerdp_set_connection_type(settings, type))
3250 return COMMAND_LINE_ERROR;
3256 WINPR_ASSERT(settings);
3260 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3262 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3264 FreeRDP_Settings_Keys_Bool singleOptionWithoutOnOff = FreeRDP_BOOL_UNUSED;
3265 for (
size_t x = 0; x < count; x++)
3267 const char* cur = ptr[x];
3268 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(cur);
3269 if (bval == PARSE_FAIL)
3271 CommandLineParserFree(ptr);
3272 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3275 const BOOL val = bval != PARSE_OFF;
3276 FreeRDP_Settings_Keys_Bool
id = FreeRDP_BOOL_UNUSED;
3277 if (option_starts_with(
"rdp", cur))
3279 id = FreeRDP_RdpSecurity;
3281 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3283 else if (option_starts_with(
"tls", cur))
3284 id = FreeRDP_TlsSecurity;
3285 else if (option_starts_with(
"nla", cur))
3286 id = FreeRDP_NlaSecurity;
3287 else if (option_starts_with(
"ext", cur))
3288 id = FreeRDP_ExtSecurity;
3289 else if (option_equals(
"aad", cur))
3290 id = FreeRDP_AadSecurity;
3293 WLog_ERR(TAG,
"unknown protocol security: %s", arg->Value);
3294 CommandLineParserFree(ptr);
3295 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3298 if ((bval == PARSE_NONE) && (count == 1))
3299 singleOptionWithoutOnOff = id;
3301 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3304 if (singleOptionWithoutOnOff != FreeRDP_BOOL_UNUSED)
3306 const FreeRDP_Settings_Keys_Bool options[] = { FreeRDP_AadSecurity,
3307 FreeRDP_UseRdpSecurityLayer,
3308 FreeRDP_RdpSecurity, FreeRDP_NlaSecurity,
3309 FreeRDP_TlsSecurity };
3311 for (
size_t i = 0; i < ARRAYSIZE(options); i++)
3314 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3318 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3319 if (singleOptionWithoutOnOff == FreeRDP_RdpSecurity)
3322 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3325 CommandLineParserFree(ptr);
3329static int parse_encryption_methods_options(rdpSettings* settings,
3332 WINPR_ASSERT(settings);
3335 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
3338 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3340 UINT32 EncryptionMethods = 0;
3341 for (UINT32 i = 0; i < count; i++)
3343 if (option_equals(ptr[i],
"40"))
3344 EncryptionMethods |= ENCRYPTION_METHOD_40BIT;
3345 else if (option_equals(ptr[i],
"56"))
3346 EncryptionMethods |= ENCRYPTION_METHOD_56BIT;
3347 else if (option_equals(ptr[i],
"128"))
3348 EncryptionMethods |= ENCRYPTION_METHOD_128BIT;
3349 else if (option_equals(ptr[i],
"FIPS"))
3350 EncryptionMethods |= ENCRYPTION_METHOD_FIPS;
3352 WLog_ERR(TAG,
"unknown encryption method '%s'", ptr[i]);
3356 return COMMAND_LINE_ERROR;
3357 CommandLineParserFree(ptr);
3364 WINPR_ASSERT(settings);
3369 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3370 for (
size_t x = 0; (x < count) && (rc == 0); x++)
3372 const char deny[] =
"deny";
3373 const char ignore[] =
"ignore";
3374 const char tofu[] =
"tofu";
3375 const char name[] =
"name:";
3376 const char fingerprints[] =
"fingerprint:";
3378 const char* cur = ptr[x];
3379 if (option_equals(deny, cur))
3382 return COMMAND_LINE_ERROR;
3384 else if (option_equals(ignore, cur))
3387 return COMMAND_LINE_ERROR;
3389 else if (option_equals(tofu, cur))
3392 return COMMAND_LINE_ERROR;
3394 else if (option_starts_with(name, cur))
3396 const char* val = &cur[strnlen(name,
sizeof(name))];
3398 rc = COMMAND_LINE_ERROR_MEMORY;
3400 else if (option_starts_with(fingerprints, cur))
3402 const char* val = &cur[strnlen(fingerprints,
sizeof(fingerprints))];
3405 rc = COMMAND_LINE_ERROR_MEMORY;
3408 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3410 CommandLineParserFree(ptr);
3417 WINPR_ASSERT(settings);
3421 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"mouse", arg->Value, &count);
3425 for (
size_t x = 1; x < count; x++)
3427 const char* cur = ptr[x];
3429 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(cur);
3430 if (bval == PARSE_FAIL)
3431 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3434 const BOOL val = bval != PARSE_OFF;
3436 if (option_starts_with(
"relative", cur))
3439 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3441 else if (option_starts_with(
"grab", cur))
3444 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3452 CommandLineParserFree(ptr);
3459 WINPR_ASSERT(settings);
3463 UINT32 Floatbar = 0x0017;
3467 char* start = arg->Value;
3472 start = strchr(start,
',');
3481 if (option_starts_with(
"sticky:", cur))
3485 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(cur);
3497 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3501 else if (option_starts_with(
"default:", cur))
3503 const char* val = cur + 8;
3506 if (option_equals(
"visible", val))
3508 else if (option_equals(
"hidden", val))
3511 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3514 else if (option_starts_with(
"show:", cur))
3516 const char* val = cur + 5;
3519 if (option_equals(
"always", val))
3521 else if (option_equals(
"fullscreen", val))
3523 else if (option_equals(
"window", val))
3526 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3529 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3533 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3539 WINPR_ASSERT(settings);
3542 BYTE* base64 = NULL;
3545 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3547 crypto_base64_decode((
const char*)(arg->Value), strlen(arg->Value), &base64, &length);
3553 return COMMAND_LINE_ERROR;
3557 WLog_ERR(TAG,
"reconnect-cookie: invalid base64 '%s'", arg->Value);
3564static BOOL set_monitor_override(rdpSettings* settings, uint64_t flag)
3566 const FreeRDP_Settings_Keys_UInt64 key = FreeRDP_MonitorOverrideFlags;
3574 WINPR_ASSERT(settings);
3579 if (!value_to_int(arg->Value, &val, 100, 180))
3580 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3588 return COMMAND_LINE_ERROR;
3590 return COMMAND_LINE_ERROR;
3591 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_DESKTOP_SCALE |
3592 FREERDP_MONITOR_OVERRIDE_DEVICE_SCALE))
3593 return fail_at(arg, COMMAND_LINE_ERROR);
3597 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3604 WINPR_ASSERT(settings);
3609 if (!value_to_int(arg->Value, &val, 100, 180))
3610 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3618 return COMMAND_LINE_ERROR;
3619 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_DEVICE_SCALE))
3620 return fail_at(arg, COMMAND_LINE_ERROR);
3624 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3631 WINPR_ASSERT(settings);
3637 return COMMAND_LINE_ERROR;
3639 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"smartcard-logon", arg->Value, &count);
3642 const CmdLineSubOptions opts[] = {
3643 {
"cert:", FreeRDP_SmartcardCertificate, CMDLINE_SUBOPTION_FILE,
3644 setSmartcardEmulation },
3645 {
"key:", FreeRDP_SmartcardPrivateKey, CMDLINE_SUBOPTION_FILE, setSmartcardEmulation },
3646 {
"pin:", FreeRDP_Password, CMDLINE_SUBOPTION_STRING, NULL },
3647 {
"csp:", FreeRDP_CspName, CMDLINE_SUBOPTION_STRING, NULL },
3648 {
"reader:", FreeRDP_ReaderName, CMDLINE_SUBOPTION_STRING, NULL },
3649 {
"card:", FreeRDP_CardName, CMDLINE_SUBOPTION_STRING, NULL },
3650 {
"container:", FreeRDP_ContainerName, CMDLINE_SUBOPTION_STRING, NULL }
3653 for (
size_t x = 1; x < count; x++)
3655 const char* cur = ptr[x];
3656 if (!parseSubOptions(settings, opts, ARRAYSIZE(opts), cur))
3658 CommandLineParserFree(ptr);
3659 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3663 CommandLineParserFree(ptr);
3669 WINPR_ASSERT(settings);
3673 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"tune", arg->Value, &count);
3675 return COMMAND_LINE_ERROR;
3676 for (
size_t x = 1; x < count; x++)
3678 const char* cur = ptr[x];
3679 char* sep = strchr(cur,
':');
3682 CommandLineParserFree(ptr);
3683 return COMMAND_LINE_ERROR;
3686 if (!freerdp_settings_set_value_for_name(settings, cur, sep))
3688 CommandLineParserFree(ptr);
3689 return COMMAND_LINE_ERROR;
3693 CommandLineParserFree(ptr);
3697static int parse_app_option_program(rdpSettings* settings,
const char* cmd)
3699 const FreeRDP_Settings_Keys_Bool ids[] = { FreeRDP_RemoteApplicationMode,
3700 FreeRDP_RemoteAppLanguageBarSupported,
3701 FreeRDP_Workarea, FreeRDP_DisableWallpaper,
3702 FreeRDP_DisableFullWindowDrag };
3705 return COMMAND_LINE_ERROR_MEMORY;
3707 for (
size_t y = 0; y < ARRAYSIZE(ids); y++)
3710 return COMMAND_LINE_ERROR;
3712 return CHANNEL_RC_OK;
3717 WINPR_ASSERT(settings);
3720 int rc = CHANNEL_RC_OK;
3722 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3723 if (!ptr || (count == 0))
3724 rc = COMMAND_LINE_ERROR;
3731 int (*fkt)(rdpSettings* settings,
const char* value);
3733 const struct app_map amap[] = {
3734 {
"tenantid:", FreeRDP_GatewayAvdAadtenantid, parse_app_option_program },
3735 {
"ad:", FreeRDP_GatewayAzureActiveDirectory, NULL },
3736 {
"avd-access:", FreeRDP_GatewayAvdAccessAadFormat, NULL },
3737 {
"avd-token:", FreeRDP_GatewayAvdAccessTokenFormat, NULL },
3738 {
"avd-scope:", FreeRDP_GatewayAvdScope, NULL }
3741 for (
size_t x = 0; x < count; x++)
3743 BOOL handled = FALSE;
3744 const char* val = ptr[x];
3746 if (option_starts_with(
"use-tenantid", val))
3748 PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
3749 if (bval == PARSE_FAIL)
3751 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3759 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3765 for (
size_t y = 0; y < ARRAYSIZE(amap); y++)
3767 const struct app_map* cur = &amap[y];
3768 if (option_starts_with(cur->name, val))
3770 const char* xval = &val[strlen(cur->name)];
3772 rc = cur->fkt(settings, xval);
3776 if (!freerdp_settings_set_value_for_name(settings, name, xval))
3777 rc = COMMAND_LINE_ERROR_MEMORY;
3786 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3793 CommandLineParserFree(ptr);
3799 WINPR_ASSERT(settings);
3802 int rc = CHANNEL_RC_OK;
3804 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3805 if (!ptr || (count == 0))
3806 rc = COMMAND_LINE_ERROR;
3813 int (*fkt)(rdpSettings* settings,
const char* value);
3815 const struct app_map amap[] = { {
"program:", FreeRDP_RemoteApplicationProgram,
3816 parse_app_option_program },
3817 {
"workdir:", FreeRDP_RemoteApplicationWorkingDir, NULL },
3818 {
"name:", FreeRDP_RemoteApplicationName, NULL },
3819 {
"icon:", FreeRDP_RemoteApplicationIcon, NULL },
3820 {
"cmd:", FreeRDP_RemoteApplicationCmdLine, NULL },
3821 {
"file:", FreeRDP_RemoteApplicationFile, NULL },
3822 {
"guid:", FreeRDP_RemoteApplicationGuid, NULL },
3823 {
"hidef:", FreeRDP_HiDefRemoteApp, NULL } };
3824 for (
size_t x = 0; x < count; x++)
3826 BOOL handled = FALSE;
3827 const char* val = ptr[x];
3829 for (
size_t y = 0; y < ARRAYSIZE(amap); y++)
3831 const struct app_map* cur = &amap[y];
3832 if (option_starts_with(cur->name, val))
3834 const char* xval = &val[strlen(cur->name)];
3836 rc = cur->fkt(settings, xval);
3840 if (!freerdp_settings_set_value_for_name(settings, name, xval))
3841 rc = COMMAND_LINE_ERROR_MEMORY;
3849#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
3850 if (!handled && (count == 1))
3853 rc = parse_app_option_program(settings, val);
3858 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3865 CommandLineParserFree(ptr);
3871 WINPR_ASSERT(settings);
3874 int rc = CHANNEL_RC_OK;
3876 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3877 if (!ptr || (count == 0))
3878 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3880 for (
size_t x = 0; x < count; x++)
3882 const char* val = ptr[x];
3884 if (option_starts_with(
"codec:", val))
3887 rc = COMMAND_LINE_ERROR;
3888 else if (option_equals(arg->Value,
"rfx"))
3891 rc = COMMAND_LINE_ERROR;
3893 else if (option_equals(arg->Value,
"nsc"))
3896 rc = COMMAND_LINE_ERROR;
3899#if defined(WITH_JPEG)
3900 else if (option_equals(arg->Value,
"jpeg"))
3903 rc = COMMAND_LINE_ERROR;
3908 return COMMAND_LINE_ERROR;
3914 else if (option_starts_with(
"persist-file:", val))
3918 rc = COMMAND_LINE_ERROR_MEMORY;
3920 rc = COMMAND_LINE_ERROR;
3924 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
3925 if (bval == PARSE_FAIL)
3926 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3929 if (option_starts_with(
"bitmap", val))
3933 rc = COMMAND_LINE_ERROR;
3935 else if (option_starts_with(
"glyph", val))
3938 bval != PARSE_OFF ? GLYPH_SUPPORT_FULL
3939 : GLYPH_SUPPORT_NONE))
3940 rc = COMMAND_LINE_ERROR;
3942 else if (option_starts_with(
"persist", val))
3946 rc = COMMAND_LINE_ERROR;
3948 else if (option_starts_with(
"offscreen", val))
3952 rc = COMMAND_LINE_ERROR;
3958 CommandLineParserFree(ptr);
3962static BOOL parse_gateway_host_option(rdpSettings* settings,
const char* host)
3964 WINPR_ASSERT(settings);
3969 if (!freerdp_parse_hostname(host, &name, &port))
3988static BOOL parse_gateway_cred_option(rdpSettings* settings,
const char* value,
3989 FreeRDP_Settings_Keys_String what)
3991 WINPR_ASSERT(settings);
3992 WINPR_ASSERT(value);
3996 case FreeRDP_GatewayUsername:
3997 if (!freerdp_parse_username_settings(value, settings, FreeRDP_GatewayUsername,
3998 FreeRDP_GatewayDomain))
4010static BOOL parse_gateway_type_option(rdpSettings* settings,
const char* value)
4014 WINPR_ASSERT(settings);
4015 WINPR_ASSERT(value);
4017 if (option_equals(value,
"rpc"))
4028 if (option_equals(value,
"http"))
4036 else if (option_equals(value,
"auto"))
4044 else if (option_equals(value,
"arm"))
4057static BOOL parse_gateway_usage_option(rdpSettings* settings,
const char* value)
4061 WINPR_ASSERT(settings);
4062 WINPR_ASSERT(value);
4064 if (option_equals(value,
"none"))
4065 type = TSC_PROXY_MODE_NONE_DIRECT;
4066 else if (option_equals(value,
"direct"))
4067 type = TSC_PROXY_MODE_DIRECT;
4068 else if (option_equals(value,
"detect"))
4069 type = TSC_PROXY_MODE_DETECT;
4070 else if (option_equals(value,
"default"))
4071 type = TSC_PROXY_MODE_DEFAULT;
4076 if (!value_to_int(value, &val, TSC_PROXY_MODE_NONE_DIRECT, TSC_PROXY_MODE_NONE_DETECT))
4087 WINPR_ASSERT(settings);
4091 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
4099 BOOL allowHttpOpts = FALSE;
4100 for (
size_t x = 0; x < count; x++)
4102 BOOL validOption = FALSE;
4103 const char* argval = ptr[x];
4105 WINPR_ASSERT(argval);
4107 const char* gw = option_starts_with(
"g:", argval);
4110 if (!parse_gateway_host_option(settings, gw))
4113 allowHttpOpts = FALSE;
4116 const char* gu = option_starts_with(
"u:", argval);
4119 if (!parse_gateway_cred_option(settings, gu, FreeRDP_GatewayUsername))
4122 allowHttpOpts = FALSE;
4125 const char* gd = option_starts_with(
"d:", argval);
4128 if (!parse_gateway_cred_option(settings, gd, FreeRDP_GatewayDomain))
4131 allowHttpOpts = FALSE;
4134 const char* gp = option_starts_with(
"p:", argval);
4137 if (!parse_gateway_cred_option(settings, gp, FreeRDP_GatewayPassword))
4140 allowHttpOpts = FALSE;
4143 const char* gt = option_starts_with(
"type:", argval);
4146 if (!parse_gateway_type_option(settings, gt))
4152 const char* gat = option_starts_with(
"access-token:", argval);
4158 allowHttpOpts = FALSE;
4161 const char* bearer = option_starts_with(
"bearer:", argval);
4167 allowHttpOpts = FALSE;
4170 const char* gwurl = option_starts_with(
"url:", argval);
4178 allowHttpOpts = FALSE;
4181 const char* um = option_starts_with(
"usage-method:", argval);
4184 if (!parse_gateway_usage_option(settings, um))
4187 allowHttpOpts = FALSE;
4192 if (option_equals(argval,
"no-websockets"))
4198 else if (option_equals(argval,
"extauth-sspi-ntlm"))
4212 CommandLineParserFree(ptr);
4219 WINPR_ASSERT(value);
4225 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
4226 FillMemory(arg->Value, strlen(arg->Value),
'*');
4231 const char* credentials[] = {
4234#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
4243 for (
size_t x = 0; x < ARRAYSIZE(credentials); x++)
4245 const char* cred = credentials[x];
4246 fill_credential_string(args, cred);
4250 if (arg && ((arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT) != 0))
4252 const char* gwcreds[] = {
"p:",
"access-token:" };
4253 char* saveptr = NULL;
4254 char* tok = strtok_s(arg->Value,
",", &saveptr);
4257 for (
size_t x = 0; x < ARRAYSIZE(gwcreds); x++)
4259 const char* opt = gwcreds[x];
4260 if (option_starts_with(opt, tok))
4262 char* val = &tok[strlen(opt)];
4263 FillMemory(val, strlen(val),
'*');
4266 tok = strtok_s(NULL,
",", &saveptr);
4271static int parse_command_line_option_uint32(rdpSettings* settings,
4273 FreeRDP_Settings_Keys_UInt32 key, LONGLONG min,
4278 if (!value_to_int(arg->Value, &val, min, max))
4279 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4282 return fail_at(arg, COMMAND_LINE_ERROR);
4286#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
4291 WINPR_ASSERT(settings);
4294 BOOL enable = arg->Value ? TRUE : FALSE;
4295 CommandLineSwitchStart(arg) CommandLineSwitchCase(arg,
"gfx-thin-client")
4297 WLog_WARN(TAG,
"/gfx-thin-client is deprecated, use /gfx:thin-client[:on|off] instead");
4299 return fail_at(arg, COMMAND_LINE_ERROR);
4304 return fail_at(arg, COMMAND_LINE_ERROR);
4308 return fail_at(arg, COMMAND_LINE_ERROR);
4310 CommandLineSwitchCase(arg,
"gfx-small-cache")
4312 WLog_WARN(TAG,
"/gfx-small-cache is deprecated, use /gfx:small-cache[:on|off] instead");
4314 return fail_at(arg, COMMAND_LINE_ERROR);
4318 return fail_at(arg, COMMAND_LINE_ERROR);
4320 CommandLineSwitchCase(arg,
"gfx-progressive")
4322 WLog_WARN(TAG,
"/gfx-progressive is deprecated, use /gfx:progressive[:on|off] instead");
4324 return fail_at(arg, COMMAND_LINE_ERROR);
4326 return fail_at(arg, COMMAND_LINE_ERROR);
4331 return fail_at(arg, COMMAND_LINE_ERROR);
4335 CommandLineSwitchCase(arg,
"gfx-h264")
4337 WLog_WARN(TAG,
"/gfx-h264 is deprecated, use /gfx:avc420 instead");
4338 int rc = parse_gfx_options(settings, arg);
4340 return fail_at(arg, rc);
4343 CommandLineSwitchCase(arg,
"app-workdir")
4346 "/app-workdir:<directory> is deprecated, use /app:workdir:<directory> instead");
4348 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4350 CommandLineSwitchCase(arg,
"app-name")
4352 WLog_WARN(TAG,
"/app-name:<directory> is deprecated, use /app:name:<name> instead");
4354 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4356 CommandLineSwitchCase(arg,
"app-icon")
4358 WLog_WARN(TAG,
"/app-icon:<filename> is deprecated, use /app:icon:<filename> instead");
4360 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4362 CommandLineSwitchCase(arg,
"app-cmd")
4364 WLog_WARN(TAG,
"/app-cmd:<command> is deprecated, use /app:cmd:<command> instead");
4366 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4368 CommandLineSwitchCase(arg,
"app-file")
4370 WLog_WARN(TAG,
"/app-file:<filename> is deprecated, use /app:file:<filename> instead");
4372 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4374 CommandLineSwitchCase(arg,
"app-guid")
4376 WLog_WARN(TAG,
"/app-guid:<guid> is deprecated, use /app:guid:<guid> instead");
4378 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4380 CommandLineSwitchCase(arg,
"g")
4382 if (!parse_gateway_host_option(settings, arg->Value))
4383 return fail_at(arg, COMMAND_LINE_ERROR);
4385 CommandLineSwitchCase(arg,
"gu")
4387 if (!parse_gateway_cred_option(settings, arg->Value, FreeRDP_GatewayUsername))
4388 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4390 CommandLineSwitchCase(arg,
"gd")
4392 if (!parse_gateway_cred_option(settings, arg->Value, FreeRDP_GatewayDomain))
4393 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4395 CommandLineSwitchCase(arg,
"gp")
4397 if (!parse_gateway_cred_option(settings, arg->Value, FreeRDP_GatewayPassword))
4398 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4400 CommandLineSwitchCase(arg,
"gt")
4402 if (!parse_gateway_type_option(settings, arg->Value))
4403 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4405 CommandLineSwitchCase(arg,
"gat")
4408 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4410 CommandLineSwitchCase(arg,
"gateway-usage-method")
4412 if (!parse_gateway_usage_option(settings, arg->Value))
4413 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4415 CommandLineSwitchCase(arg,
"kbd-remap")
4417 WLog_WARN(TAG,
"/kbd-remap:<key>=<value>,<key2>=<value2> is deprecated, use "
4418 "/kbd:remap:<key>=<value>,remap:<key2>=<value2>,... instead");
4420 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4422 CommandLineSwitchCase(arg,
"kbd-lang")
4426 WLog_WARN(TAG,
"/kbd-lang:<value> is deprecated, use /kbd:lang:<value> instead");
4427 if (!value_to_int(arg->Value, &val, 1, UINT32_MAX))
4429 WLog_ERR(TAG,
"Could not identify keyboard active language %s", arg->Value);
4430 WLog_ERR(TAG,
"Use /list:kbd-lang to list available layouts");
4431 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4435 return fail_at(arg, COMMAND_LINE_ERROR);
4437 CommandLineSwitchCase(arg,
"kbd-type")
4439 WLog_WARN(TAG,
"/kbd-type:<value> is deprecated, use /kbd:type:<value> instead");
4441 parse_command_line_option_uint32(settings, arg, FreeRDP_KeyboardType, 0, UINT32_MAX);
4443 return fail_at(arg, rc);
4445 CommandLineSwitchCase(arg,
"kbd-unicode")
4447 WLog_WARN(TAG,
"/kbd-unicode is deprecated, use /kbd:unicode[:on|off] instead");
4449 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4451 CommandLineSwitchCase(arg,
"kbd-subtype")
4453 WLog_WARN(TAG,
"/kbd-subtype:<value> is deprecated, use /kbd:subtype:<value> instead");
4455 parse_command_line_option_uint32(settings, arg, FreeRDP_KeyboardSubType, 0, UINT32_MAX);
4457 return fail_at(arg, rc);
4459 CommandLineSwitchCase(arg,
"kbd-fn-key")
4461 WLog_WARN(TAG,
"/kbd-fn-key:<value> is deprecated, use /kbd:fn-key:<value> instead");
4462 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_KeyboardFunctionKey,
4465 return fail_at(arg, rc);
4467 CommandLineSwitchCase(arg,
"bitmap-cache")
4469 WLog_WARN(TAG,
"/bitmap-cache is deprecated, use /cache:bitmap[:on|off] instead");
4471 return fail_at(arg, COMMAND_LINE_ERROR);
4473 CommandLineSwitchCase(arg,
"persist-cache")
4475 WLog_WARN(TAG,
"/persist-cache is deprecated, use /cache:persist[:on|off] instead");
4477 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4479 CommandLineSwitchCase(arg,
"persist-cache-file")
4481 WLog_WARN(TAG,
"/persist-cache-file:<filename> is deprecated, use "
4482 "/cache:persist-file:<filename> instead");
4484 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4487 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4489 CommandLineSwitchCase(arg,
"offscreen-cache")
4491 WLog_WARN(TAG,
"/bitmap-cache is deprecated, use /cache:bitmap[:on|off] instead");
4493 return fail_at(arg, COMMAND_LINE_ERROR);
4495 CommandLineSwitchCase(arg,
"glyph-cache")
4497 WLog_WARN(TAG,
"/glyph-cache is deprecated, use /cache:glyph[:on|off] instead");
4499 arg->Value ? GLYPH_SUPPORT_FULL : GLYPH_SUPPORT_NONE))
4500 return fail_at(arg, COMMAND_LINE_ERROR);
4502 CommandLineSwitchCase(arg,
"codec-cache")
4504 WLog_WARN(TAG,
"/codec-cache:<option> is deprecated, use /cache:codec:<option> instead");
4505 const int rc = parse_codec_cache_options(settings, arg);
4507 return fail_at(arg, rc);
4509 CommandLineSwitchCase(arg,
"sec-rdp")
4511 WLog_WARN(TAG,
"/sec-rdp is deprecated, use /sec:rdp[:on|off] instead");
4513 return fail_at(arg, COMMAND_LINE_ERROR);
4515 CommandLineSwitchCase(arg,
"sec-tls")
4517 WLog_WARN(TAG,
"/sec-tls is deprecated, use /sec:tls[:on|off] instead");
4519 return fail_at(arg, COMMAND_LINE_ERROR);
4521 CommandLineSwitchCase(arg,
"sec-nla")
4523 WLog_WARN(TAG,
"/sec-nla is deprecated, use /sec:nla[:on|off] instead");
4525 return fail_at(arg, COMMAND_LINE_ERROR);
4527 CommandLineSwitchCase(arg,
"sec-ext")
4529 WLog_WARN(TAG,
"/sec-ext is deprecated, use /sec:ext[:on|off] instead");
4531 return fail_at(arg, COMMAND_LINE_ERROR);
4533 CommandLineSwitchCase(arg,
"tls-ciphers")
4535 WLog_WARN(TAG,
"/tls-ciphers:<cipher list> is deprecated, use "
4536 "/tls:ciphers:<cipher list> instead");
4537 int rc = parse_tls_cipher_options(settings, arg);
4539 return fail_at(arg, rc);
4541 CommandLineSwitchCase(arg,
"tls-seclevel")
4543 WLog_WARN(TAG,
"/tls-seclevel:<level> is deprecated, use /tls:sec-level:<level> instead");
4544 int rc = parse_tls_cipher_options(settings, arg);
4546 return fail_at(arg, rc);
4548 CommandLineSwitchCase(arg,
"tls-secrets-file")
4550 WLog_WARN(TAG,
"/tls-secrets-file:<filename> is deprecated, use "
4551 "/tls:secrets-file:<filename> instead");
4552 int rc = parse_tls_cipher_options(settings, arg);
4554 return fail_at(arg, rc);
4556 CommandLineSwitchCase(arg,
"enforce-tlsv1_2")
4558 WLog_WARN(TAG,
"/enforce-tlsv1_2 is deprecated, use /tls:enforce:1.2 instead");
4559 int rc = parse_tls_cipher_options(settings, arg);
4561 return fail_at(arg, rc);
4563 CommandLineSwitchCase(arg,
"cert-name")
4565 WLog_WARN(TAG,
"/cert-name is deprecated, use /cert:name instead");
4567 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4569 CommandLineSwitchCase(arg,
"cert-ignore")
4571 WLog_WARN(TAG,
"/cert-ignore is deprecated, use /cert:ignore instead");
4573 return fail_at(arg, COMMAND_LINE_ERROR);
4575 CommandLineSwitchCase(arg,
"cert-tofu")
4577 WLog_WARN(TAG,
"/cert-tofu is deprecated, use /cert:tofu instead");
4579 return fail_at(arg, COMMAND_LINE_ERROR);
4581 CommandLineSwitchCase(arg,
"cert-deny")
4583 WLog_WARN(TAG,
"/cert-deny is deprecated, use /cert:deny instead");
4585 return fail_at(arg, COMMAND_LINE_ERROR);
4587 CommandLineSwitchDefault(arg)
4591 CommandLineSwitchEnd(arg);
4596static int parse_command_line_option_timezone(rdpSettings* settings,
4602 char TimeZoneKeyName[ARRAYSIZE(info.TimeZoneKeyName) + 1] = { 0 };
4603 while (EnumDynamicTimeZoneInformation(index++, &info) != ERROR_NO_MORE_ITEMS)
4605 (void)ConvertWCharNToUtf8(info.TimeZoneKeyName, ARRAYSIZE(info.TimeZoneKeyName),
4606 TimeZoneKeyName, ARRAYSIZE(TimeZoneKeyName));
4608 WINPR_ASSERT(arg->Value);
4609 if (strncmp(TimeZoneKeyName, arg->Value, ARRAYSIZE(TimeZoneKeyName)) == 0)
4616 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4619 return fail_at(arg, COMMAND_LINE_ERROR);
4624 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4626 tz->Bias = info.Bias;
4627 tz->DaylightBias = info.DaylightBias;
4628 tz->DaylightDate = info.DaylightDate;
4629 memcpy(tz->DaylightName, info.DaylightName,
sizeof(tz->DaylightName));
4630 tz->StandardBias = info.StandardBias;
4631 tz->StandardDate = info.StandardDate;
4632 memcpy(tz->StandardName, info.StandardName,
sizeof(tz->StandardName));
4637static int parse_command_line_option_window_pos(rdpSettings* settings,
4640 WINPR_ASSERT(settings);
4643 unsigned long x = 0;
4644 unsigned long y = 0;
4647 return fail_at(arg, COMMAND_LINE_ERROR_MISSING_ARGUMENT);
4649 if (!parseSizeValue(arg->Value, &x, &y) || x > UINT16_MAX || y > UINT16_MAX)
4651 WLog_ERR(TAG,
"invalid window-position argument");
4652 return fail_at(arg, COMMAND_LINE_ERROR_MISSING_ARGUMENT);
4656 return fail_at(arg, COMMAND_LINE_ERROR);
4658 return fail_at(arg, COMMAND_LINE_ERROR);
4663 freerdp_command_line_handle_option_t handle_option,
4664 void* handle_userdata, BOOL* promptForPassword,
char** user)
4666 WINPR_ASSERT(promptForPassword);
4671 BOOL enable = arg->Value ? TRUE : FALSE;
4673 if (!(arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
4676 CommandLineSwitchStart(arg)
4678 CommandLineSwitchCase(arg,
"v")
4680 const int rc = parse_host_options(settings, arg);
4682 return fail_at(arg, rc);
4684 CommandLineSwitchCase(arg,
"spn-class")
4688 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4690 CommandLineSwitchCase(arg,
"sspi-module")
4693 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4695 CommandLineSwitchCase(arg,
"winscard-module")
4698 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4700 CommandLineSwitchCase(arg,
"redirect-prefer")
4702 const int rc = parse_redirect_prefer_options(settings, arg);
4704 return fail_at(arg, rc);
4706 CommandLineSwitchCase(arg,
"credentials-delegation")
4709 return fail_at(arg, COMMAND_LINE_ERROR);
4711 CommandLineSwitchCase(arg,
"prevent-session-lock")
4713 const int rc = parse_prevent_session_lock_options(settings, arg);
4715 return fail_at(arg, rc);
4717 CommandLineSwitchCase(arg,
"vmconnect")
4719 const int rc = parse_vmconnect_options(settings, arg);
4721 return fail_at(arg, rc);
4723 CommandLineSwitchCase(arg,
"w")
4725 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_DesktopWidth, -1,
4728 return fail_at(arg, rc);
4730 CommandLineSwitchCase(arg,
"h")
4732 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_DesktopHeight,
4735 return fail_at(arg, rc);
4737 CommandLineSwitchCase(arg,
"size")
4739 const int rc = parse_size_options(settings, arg);
4741 return fail_at(arg, rc);
4743 CommandLineSwitchCase(arg,
"f")
4746 return fail_at(arg, COMMAND_LINE_ERROR);
4748 CommandLineSwitchCase(arg,
"suppress-output")
4751 return fail_at(arg, COMMAND_LINE_ERROR);
4753 CommandLineSwitchCase(arg,
"multimon")
4756 return fail_at(arg, COMMAND_LINE_ERROR);
4758 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
4760 if (option_equals(arg->Value, str_force))
4763 return fail_at(arg, COMMAND_LINE_ERROR);
4767 CommandLineSwitchCase(arg,
"span")
4770 return fail_at(arg, COMMAND_LINE_ERROR);
4772 CommandLineSwitchCase(arg,
"workarea")
4775 return fail_at(arg, COMMAND_LINE_ERROR);
4777 CommandLineSwitchCase(arg,
"monitors")
4779 const int rc = parse_monitors_options(settings, arg);
4781 return fail_at(arg, rc);
4783 CommandLineSwitchCase(arg,
"t")
4786 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4788 CommandLineSwitchCase(arg,
"decorations")
4791 return fail_at(arg, COMMAND_LINE_ERROR);
4793 CommandLineSwitchCase(arg,
"dynamic-resolution")
4795 const int rc = parse_dynamic_resolution_options(settings, arg);
4797 return fail_at(arg, rc);
4799 CommandLineSwitchCase(arg,
"smart-sizing")
4801 const int rc = parse_smart_sizing_options(settings, arg);
4803 return fail_at(arg, rc);
4805 CommandLineSwitchCase(arg,
"bpp")
4807 const int rc = parse_bpp_options(settings, arg);
4809 return fail_at(arg, rc);
4811 CommandLineSwitchCase(arg,
"admin")
4814 return fail_at(arg, COMMAND_LINE_ERROR);
4816 CommandLineSwitchCase(arg,
"relax-order-checks")
4820 return fail_at(arg, COMMAND_LINE_ERROR);
4822 CommandLineSwitchCase(arg,
"restricted-admin")
4825 return fail_at(arg, COMMAND_LINE_ERROR);
4827 return fail_at(arg, COMMAND_LINE_ERROR);
4829 CommandLineSwitchCase(arg,
"remoteGuard")
4832 return fail_at(arg, COMMAND_LINE_ERROR);
4834 return fail_at(arg, COMMAND_LINE_ERROR);
4836 CommandLineSwitchCase(arg,
"pth")
4839 return fail_at(arg, COMMAND_LINE_ERROR);
4841 return fail_at(arg, COMMAND_LINE_ERROR);
4844 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4846 CommandLineSwitchCase(arg,
"client-hostname")
4849 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4851 CommandLineSwitchCase(arg,
"kbd")
4853 int rc = parse_kbd_options(settings, arg);
4855 return fail_at(arg, rc);
4858 CommandLineSwitchCase(arg,
"u")
4860 WINPR_ASSERT(arg->Value);
4863 CommandLineSwitchCase(arg,
"d")
4866 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4868 CommandLineSwitchCase(arg,
"p")
4871 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4873 CommandLineSwitchCase(arg,
"gateway")
4875 if (!parse_gateway_options(settings, arg))
4876 return fail_at(arg, COMMAND_LINE_ERROR);
4878 CommandLineSwitchCase(arg,
"proxy")
4880 const int rc = parse_proxy_options(settings, arg);
4882 return fail_at(arg, rc);
4885 CommandLineSwitchCase(arg,
"azure")
4887 int rc = parse_aad_options(settings, arg);
4889 return fail_at(arg, rc);
4891 CommandLineSwitchCase(arg,
"app")
4893 int rc = parse_app_options(settings, arg);
4895 return fail_at(arg, rc);
4897 CommandLineSwitchCase(arg,
"load-balance-info")
4899 WINPR_ASSERT(arg->Value);
4901 strlen(arg->Value)))
4902 return fail_at(arg, COMMAND_LINE_ERROR);
4905 CommandLineSwitchCase(arg,
"compression")
4908 return fail_at(arg, COMMAND_LINE_ERROR);
4910 CommandLineSwitchCase(arg,
"compression-level")
4912 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_CompressionLevel,
4915 return fail_at(arg, rc);
4917 CommandLineSwitchCase(arg,
"drives")
4920 return fail_at(arg, COMMAND_LINE_ERROR);
4922 CommandLineSwitchCase(arg,
"dump")
4924 const int rc = parse_dump_options(settings, arg);
4926 return fail_at(arg, rc);
4928 CommandLineSwitchCase(arg,
"disable-output")
4931 return fail_at(arg, COMMAND_LINE_ERROR);
4933 CommandLineSwitchCase(arg,
"home-drive")
4936 return fail_at(arg, COMMAND_LINE_ERROR);
4938 CommandLineSwitchCase(arg,
"ipv4")
4940 if (arg->Value != NULL && strncmp(arg->Value, str_force, ARRAYSIZE(str_force)) == 0)
4943 return fail_at(arg, COMMAND_LINE_ERROR);
4948 return fail_at(arg, COMMAND_LINE_ERROR);
4951 CommandLineSwitchCase(arg,
"ipv6")
4953 if (arg->Value != NULL && strncmp(arg->Value, str_force, ARRAYSIZE(str_force)) == 0)
4956 return fail_at(arg, COMMAND_LINE_ERROR);
4961 return fail_at(arg, COMMAND_LINE_ERROR);
4964 CommandLineSwitchCase(arg,
"clipboard")
4966 const int rc = parse_clipboard_options(settings, arg);
4968 return fail_at(arg, rc);
4970 CommandLineSwitchCase(arg,
"server-name")
4973 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4975 CommandLineSwitchCase(arg,
"shell")
4978 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4980 CommandLineSwitchCase(arg,
"shell-dir")
4983 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4985 CommandLineSwitchCase(arg,
"audio-mode")
4987 const int rc = parse_audio_mode_options(settings, arg);
4989 return fail_at(arg, rc);
4991 CommandLineSwitchCase(arg,
"network")
4993 const int rc = parse_network_options(settings, arg);
4995 return fail_at(arg, rc);
4997 CommandLineSwitchCase(arg,
"fonts")
5000 return fail_at(arg, COMMAND_LINE_ERROR);
5002 CommandLineSwitchCase(arg,
"wallpaper")
5005 return fail_at(arg, COMMAND_LINE_ERROR);
5007 CommandLineSwitchCase(arg,
"window-drag")
5010 return fail_at(arg, COMMAND_LINE_ERROR);
5012 CommandLineSwitchCase(arg,
"window-position")
5014 const int rc = parse_command_line_option_window_pos(settings, arg);
5016 return fail_at(arg, rc);
5018 CommandLineSwitchCase(arg,
"menu-anims")
5021 return fail_at(arg, COMMAND_LINE_ERROR);
5023 CommandLineSwitchCase(arg,
"themes")
5026 return fail_at(arg, COMMAND_LINE_ERROR);
5028 CommandLineSwitchCase(arg,
"timeout")
5031 parse_command_line_option_uint32(settings, arg, FreeRDP_TcpAckTimeout, 0, 600000);
5033 return fail_at(arg, rc);
5035 CommandLineSwitchCase(arg,
"timezone")
5037 const int rc = parse_command_line_option_timezone(settings, arg);
5039 return fail_at(arg, rc);
5041 CommandLineSwitchCase(arg,
"aero")
5044 return fail_at(arg, COMMAND_LINE_ERROR);
5046 CommandLineSwitchCase(arg,
"gdi")
5048 if (option_equals(arg->Value,
"sw"))
5051 return fail_at(arg, COMMAND_LINE_ERROR);
5053 else if (option_equals(arg->Value,
"hw"))
5056 return fail_at(arg, COMMAND_LINE_ERROR);
5059 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5061 CommandLineSwitchCase(arg,
"gfx")
5063 int rc = parse_gfx_options(settings, arg);
5065 return fail_at(arg, rc);
5068 CommandLineSwitchCase(arg,
"rfx")
5071 return fail_at(arg, COMMAND_LINE_ERROR);
5073 CommandLineSwitchCase(arg,
"rfx-mode")
5076 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5078 if (option_equals(arg->Value,
"video"))
5081 return fail_at(arg, COMMAND_LINE_ERROR);
5083 else if (option_equals(arg->Value,
"image"))
5086 return fail_at(arg, COMMAND_LINE_ERROR);
5088 return fail_at(arg, COMMAND_LINE_ERROR);
5091 CommandLineSwitchCase(arg,
"frame-ack")
5093 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_FrameAcknowledge,
5096 return fail_at(arg, rc);
5098 CommandLineSwitchCase(arg,
"nsc")
5101 return fail_at(arg, COMMAND_LINE_ERROR);
5103#if defined(WITH_JPEG)
5104 CommandLineSwitchCase(arg,
"jpeg")
5107 return fail_at(arg, COMMAND_LINE_ERROR);
5109 return fail_at(arg, COMMAND_LINE_ERROR);
5111 CommandLineSwitchCase(arg,
"jpeg-quality")
5115 if (!value_to_int(arg->Value, &val, 0, 100))
5116 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5119 return fail_at(arg, COMMAND_LINE_ERROR);
5122 CommandLineSwitchCase(arg,
"nego")
5125 return fail_at(arg, COMMAND_LINE_ERROR);
5127 CommandLineSwitchCase(arg,
"pcb")
5130 return fail_at(arg, COMMAND_LINE_ERROR);
5133 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5135 CommandLineSwitchCase(arg,
"pcid")
5137 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_PreconnectionId,
5140 return fail_at(arg, rc);
5142 return fail_at(arg, COMMAND_LINE_ERROR);
5145 CommandLineSwitchCase(arg,
"connect-child-session")
5159 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5162 CommandLineSwitchCase(arg,
"sec")
5164 const int rc = parse_sec_options(settings, arg);
5166 return fail_at(arg, rc);
5168 CommandLineSwitchCase(arg,
"encryption-methods")
5170 const int rc = parse_encryption_methods_options(settings, arg);
5172 return fail_at(arg, rc);
5174 CommandLineSwitchCase(arg,
"args-from")
5176 WLog_ERR(TAG,
"/args-from:%s can not be used in combination with other arguments!",
5178 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5180 CommandLineSwitchCase(arg,
"from-stdin")
5183 return fail_at(arg, COMMAND_LINE_ERROR);
5185 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
5188 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5189 *promptForPassword = (option_equals(arg->Value, str_force));
5191 if (!*promptForPassword)
5192 return fail_at(arg, COMMAND_LINE_ERROR);
5195 CommandLineSwitchCase(arg,
"log-level")
5197 wLog* root = WLog_GetRoot();
5199 if (!WLog_SetStringLogLevel(root, arg->Value))
5200 return fail_at(arg, COMMAND_LINE_ERROR);
5202 CommandLineSwitchCase(arg,
"log-filters")
5204 if (!WLog_AddStringLogFilters(arg->Value))
5205 return fail_at(arg, COMMAND_LINE_ERROR);
5207 CommandLineSwitchCase(arg,
"tls")
5209 int rc = parse_tls_options(settings, arg);
5211 return fail_at(arg, rc);
5213 CommandLineSwitchCase(arg,
"cert")
5215 const int rc = parse_cert_options(settings, arg);
5217 return fail_at(arg, rc);
5219 CommandLineSwitchCase(arg,
"authentication")
5222 return fail_at(arg, COMMAND_LINE_ERROR);
5224 CommandLineSwitchCase(arg,
"encryption")
5227 return fail_at(arg, COMMAND_LINE_ERROR);
5229 CommandLineSwitchCase(arg,
"grab-keyboard")
5232 return fail_at(arg, COMMAND_LINE_ERROR);
5234 CommandLineSwitchCase(arg,
"grab-mouse")
5237 return fail_at(arg, COMMAND_LINE_ERROR);
5239 CommandLineSwitchCase(arg,
"mouse-relative")
5242 return fail_at(arg, COMMAND_LINE_ERROR);
5244 CommandLineSwitchCase(arg,
"mouse")
5246 const int rc = parse_mouse_options(settings, arg);
5248 return fail_at(arg, rc);
5250 CommandLineSwitchCase(arg,
"unmap-buttons")
5253 return fail_at(arg, COMMAND_LINE_ERROR);
5255 CommandLineSwitchCase(arg,
"toggle-fullscreen")
5258 return fail_at(arg, COMMAND_LINE_ERROR);
5260 CommandLineSwitchCase(arg,
"force-console-callbacks")
5263 return fail_at(arg, COMMAND_LINE_ERROR);
5265 CommandLineSwitchCase(arg,
"floatbar")
5267 const int rc = parse_floatbar_options(settings, arg);
5269 return fail_at(arg, rc);
5271 CommandLineSwitchCase(arg,
"mouse-motion")
5274 return fail_at(arg, COMMAND_LINE_ERROR);
5276 CommandLineSwitchCase(arg,
"parent-window")
5280 if (!value_to_uint(arg->Value, &val, 0, UINT64_MAX))
5281 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5284 return fail_at(arg, COMMAND_LINE_ERROR);
5286 CommandLineSwitchCase(arg,
"client-build-number")
5289 parse_command_line_option_uint32(settings, arg, FreeRDP_ClientBuild, 0, UINT32_MAX);
5291 return fail_at(arg, rc);
5293 CommandLineSwitchCase(arg,
"cache")
5295 int rc = parse_cache_options(settings, arg);
5297 return fail_at(arg, rc);
5300 CommandLineSwitchCase(arg,
"max-fast-path-size")
5302 const int rc = parse_command_line_option_uint32(
5303 settings, arg, FreeRDP_MultifragMaxRequestSize, 0, UINT32_MAX);
5305 return fail_at(arg, rc);
5307 CommandLineSwitchCase(arg,
"auto-request-control")
5311 return fail_at(arg, COMMAND_LINE_ERROR);
5313 CommandLineSwitchCase(arg,
"async-update")
5316 return fail_at(arg, COMMAND_LINE_ERROR);
5318 CommandLineSwitchCase(arg,
"async-channels")
5321 return fail_at(arg, COMMAND_LINE_ERROR);
5323 CommandLineSwitchCase(arg,
"wm-class")
5326 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5328 CommandLineSwitchCase(arg,
"play-rfx")
5331 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5334 return fail_at(arg, COMMAND_LINE_ERROR);
5336 CommandLineSwitchCase(arg,
"auth-only")
5339 return fail_at(arg, COMMAND_LINE_ERROR);
5341 CommandLineSwitchCase(arg,
"auth-pkg-list")
5345 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5347 CommandLineSwitchCase(arg,
"auto-reconnect")
5350 return fail_at(arg, COMMAND_LINE_ERROR);
5352 CommandLineSwitchCase(arg,
"auto-reconnect-max-retries")
5354 const int rc = parse_command_line_option_uint32(
5355 settings, arg, FreeRDP_AutoReconnectMaxRetries, 0, 1000);
5357 return fail_at(arg, rc);
5359 CommandLineSwitchCase(arg,
"reconnect-cookie")
5361 const int rc = parse_reconnect_cookie_options(settings, arg);
5363 return fail_at(arg, rc);
5365 CommandLineSwitchCase(arg,
"print-reconnect-cookie")
5368 return fail_at(arg, COMMAND_LINE_ERROR);
5370 CommandLineSwitchCase(arg,
"pwidth")
5372 const int rc = parse_command_line_option_uint32(
5373 settings, arg, FreeRDP_DesktopPhysicalWidth, 0, UINT32_MAX);
5375 return fail_at(arg, rc);
5377 CommandLineSwitchCase(arg,
"pheight")
5379 const int rc = parse_command_line_option_uint32(
5380 settings, arg, FreeRDP_DesktopPhysicalHeight, 0, UINT32_MAX);
5382 return fail_at(arg, rc);
5384 CommandLineSwitchCase(arg,
"orientation")
5388 if (!value_to_int(arg->Value, &val, 0, UINT16_MAX))
5389 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5392 return fail_at(arg, COMMAND_LINE_ERROR);
5393 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_ORIENTATION))
5394 return fail_at(arg, COMMAND_LINE_ERROR);
5396 CommandLineSwitchCase(arg,
"old-license")
5399 return fail_at(arg, COMMAND_LINE_ERROR);
5401 CommandLineSwitchCase(arg,
"scale")
5403 const int rc = parse_scale_options(settings, arg);
5405 return fail_at(arg, rc);
5407 CommandLineSwitchCase(arg,
"scale-desktop")
5409 const int rc = parse_command_line_option_uint32(settings, arg,
5410 FreeRDP_DesktopScaleFactor, 100, 500);
5412 return fail_at(arg, rc);
5413 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_DESKTOP_SCALE))
5414 return fail_at(arg, COMMAND_LINE_ERROR);
5416 CommandLineSwitchCase(arg,
"scale-device")
5418 const int rc = parse_scale_device_options(settings, arg);
5420 return fail_at(arg, rc);
5422 CommandLineSwitchCase(arg,
"action-script")
5425 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5427 CommandLineSwitchCase(arg, RDP2TCP_DVC_CHANNEL_NAME)
5430 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5432 CommandLineSwitchCase(arg,
"fipsmode")
5435 return fail_at(arg, COMMAND_LINE_ERROR);
5437 CommandLineSwitchCase(arg,
"smartcard-logon")
5439 const int rc = parse_smartcard_logon_options(settings, arg);
5441 return fail_at(arg, rc);
5443 CommandLineSwitchCase(arg,
"tune")
5445 const int rc = parse_tune_options(settings, arg);
5447 return fail_at(arg, rc);
5449 CommandLineSwitchDefault(arg)
5451#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
5452 const int status = parse_deprecated_command_line(settings, arg);
5459 const int rc = handle_option(arg, handle_userdata);
5461 return fail_at(arg, rc);
5464 CommandLineSwitchEnd(arg)
5465 }
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
5469static int freerdp_client_settings_parse_command_line_arguments_int(
5470 rdpSettings* settings,
int argc,
char* argv[], BOOL allowUnknown,
5472 freerdp_command_line_handle_option_t handle_option,
void* handle_userdata)
5477 BOOL assist = FALSE;
5479 BOOL promptForPassword = FALSE;
5480 BOOL compatibility = FALSE;
5488 ext = option_is_rdp_file(argv[1]);
5489 assist = option_is_incident_file(argv[1]);
5492 if (!ext && !assist)
5493 compatibility = freerdp_client_detect_command_line(argc, argv, &flags);
5495 compatibility = freerdp_client_detect_command_line(argc - 1, &argv[1], &flags);
5506 WLog_WARN(TAG,
"Unsupported command line syntax!");
5507 WLog_WARN(TAG,
"FreeRDP 1.0 style syntax was dropped with version 3!");
5513 flags |= COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
5517 if (freerdp_client_settings_parse_connection_file(settings, argv[1]))
5518 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
5523 if (freerdp_client_settings_parse_assistance_file(settings, argc, argv) < 0)
5524 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
5527 CommandLineClearArgumentsA(largs);
5528 status = CommandLineParseArgumentsA(argc, argv, largs, flags, settings,
5529 freerdp_client_command_line_pre_filter,
5530 freerdp_client_command_line_post_filter);
5535 prepare_default_settings(settings, largs, ext);
5538 CommandLineFindArgumentA(largs,
"v");
5544 return COMMAND_LINE_ERROR_MEMORY;
5546 status = parse_command_line(settings, arg, handle_option, handle_userdata, &promptForPassword,
5554 return COMMAND_LINE_ERROR;
5557 return COMMAND_LINE_ERROR;
5559 if (!freerdp_parse_username_settings(user, settings, FreeRDP_Username, FreeRDP_Domain))
5560 return COMMAND_LINE_ERROR;
5565 return COMMAND_LINE_ERROR;
5569 if (promptForPassword)
5574 char buffer[512 + 1] = { 0 };
5576 if (!freerdp_passphrase_read(instance->context,
"Password: ", buffer,
5577 ARRAYSIZE(buffer) - 1, 1))
5578 return COMMAND_LINE_ERROR;
5580 return COMMAND_LINE_ERROR;
5588 char buffer[512 + 1] = { 0 };
5590 if (!freerdp_passphrase_read(instance->context,
"Gateway Password: ", buffer,
5591 ARRAYSIZE(buffer) - 1, 1))
5592 return COMMAND_LINE_ERROR;
5594 return COMMAND_LINE_ERROR;
5599 freerdp_performance_flags_make(settings);
5606 return COMMAND_LINE_ERROR;
5608 return COMMAND_LINE_ERROR;
5611 arg = CommandLineFindArgumentA(largs,
"port");
5612 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
5615 parse_command_line_option_uint32(settings, arg, FreeRDP_ServerPort, 0, UINT16_MAX);
5617 return fail_at(arg, rc);
5623 if (nego && (nego->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
5624 return fail_at(arg, COMMAND_LINE_ERROR);
5627 WLog_INFO(TAG,
"/vmconnect uses custom port %" PRIu32, port);
5630 fill_credential_strings(largs);
5635static void argv_free(
int* pargc,
char** pargv[])
5637 WINPR_ASSERT(pargc);
5638 WINPR_ASSERT(pargv);
5639 const int argc = *pargc;
5640 char** argv = *pargv;
5646 for (
int x = 0; x < argc; x++)
5651static BOOL argv_append(
int* pargc,
char** pargv[],
char* what)
5653 WINPR_ASSERT(pargc);
5654 WINPR_ASSERT(pargv);
5662 int nargc = *pargc + 1;
5663 char** tmp = (
char**)realloc((
void*)*pargv, (size_t)nargc *
sizeof(
char*));
5673static BOOL argv_append_dup(
int* pargc,
char** pargv[],
const char* what)
5677 copy = _strdup(what);
5679 const BOOL rc = argv_append(pargc, pargv, copy);
5685static BOOL args_from_fp(FILE* fp,
int* aargc,
char** aargv[],
const char* file,
const char* cmd)
5687 BOOL success = FALSE;
5689 WINPR_ASSERT(aargc);
5690 WINPR_ASSERT(aargv);
5695 WLog_ERR(TAG,
"Failed to read command line options from file '%s'", file);
5698 if (!argv_append_dup(aargc, aargv, cmd))
5704 INT64 rc = GetLine(&line, &size, fp);
5705 if ((rc < 0) || !line)
5715 const char cur = (line[rc - 1]);
5716 if ((cur ==
'\n') || (cur ==
'\r'))
5718 line[rc - 1] =
'\0';
5730 if (!argv_append(aargc, aargv, line))
5741 argv_free(aargc, aargv);
5745static BOOL args_from_env(
const char* name,
int* aargc,
char** aargv[],
const char* arg,
5748 BOOL success = FALSE;
5751 WINPR_ASSERT(aargc);
5752 WINPR_ASSERT(aargv);
5757 WLog_ERR(TAG,
"%s - environment variable name empty", arg);
5761 const DWORD size = GetEnvironmentVariableX(name, env, 0);
5764 WLog_ERR(TAG,
"%s - no environment variable '%s'", arg, name);
5767 env = calloc(size + 1,
sizeof(
char));
5770 const DWORD rc = GetEnvironmentVariableX(name, env, size);
5775 WLog_ERR(TAG,
"%s - environment variable '%s' is empty", arg);
5779 if (!argv_append_dup(aargc, aargv, cmd))
5782 char* context = NULL;
5783 char* tok = strtok_s(env,
"\n", &context);
5786 if (!argv_append_dup(aargc, aargv, tok))
5788 tok = strtok_s(NULL,
"\n", &context);
5795 argv_free(aargc, aargv);
5799int freerdp_client_settings_parse_command_line_arguments(rdpSettings* settings,
int oargc,
5800 char* oargv[], BOOL allowUnknown)
5802 return freerdp_client_settings_parse_command_line_arguments_ex(
5803 settings, oargc, oargv, allowUnknown, NULL, 0, NULL, NULL);
5806int freerdp_client_settings_parse_command_line_arguments_ex(
5807 rdpSettings* settings,
int oargc,
char** oargv, BOOL allowUnknown,
5809 void* handle_userdata)
5812 char** argv = oargv;
5815 char** aargv = NULL;
5816 if ((argc == 2) && option_starts_with(
"/args-from:", argv[1]))
5818 BOOL success = FALSE;
5819 const char* file = strchr(argv[1],
':') + 1;
5822 if (option_starts_with(
"fd:", file))
5824 ULONGLONG result = 0;
5825 const char* val = strchr(file,
':') + 1;
5826 if (!value_to_uint(val, &result, 0, INT_MAX))
5828 fp = fdopen((
int)result,
"r");
5829 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
5831 else if (strncmp(file,
"env:", 4) == 0)
5833 const char* name = strchr(file,
':') + 1;
5834 success = args_from_env(name, &aargc, &aargv, oargv[1], oargv[0]);
5836 else if (strcmp(file,
"stdin") != 0)
5838 fp = winpr_fopen(file,
"r");
5839 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
5842 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
5850 WINPR_ASSERT(count <= SSIZE_MAX);
5856 res = freerdp_client_settings_parse_command_line_arguments_int(
5857 settings, argc, argv, allowUnknown, largs, lcount, handle_option, handle_userdata);
5860 argv_free(&aargc, &aargv);
5864static BOOL freerdp_client_load_static_channel_addin(rdpChannels* channels, rdpSettings* settings,
5865 const char* name,
void* data)
5867 PVIRTUALCHANNELENTRY entry = NULL;
5868 PVIRTUALCHANNELENTRY pvce = freerdp_load_channel_addin_entry(
5869 name, NULL, NULL, FREERDP_ADDIN_CHANNEL_STATIC | FREERDP_ADDIN_CHANNEL_ENTRYEX);
5870 PVIRTUALCHANNELENTRYEX pvceex = WINPR_FUNC_PTR_CAST(pvce, PVIRTUALCHANNELENTRYEX);
5873 entry = freerdp_load_channel_addin_entry(name, NULL, NULL, FREERDP_ADDIN_CHANNEL_STATIC);
5877 if (freerdp_channels_client_load_ex(channels, settings, pvceex, data) == 0)
5879 WLog_DBG(TAG,
"loading channelEx %s", name);
5885 if (freerdp_channels_client_load(channels, settings, entry, data) == 0)
5887 WLog_DBG(TAG,
"loading channel %s", name);
5897 FreeRDP_Settings_Keys_Bool settingId;
5898 const char* channelName;
5902BOOL freerdp_client_load_addins(rdpChannels* channels, rdpSettings* settings)
5904 ChannelToLoad dynChannels[] = {
5905#if defined(CHANNEL_AINPUT_CLIENT)
5906 { FreeRDP_BOOL_UNUSED, AINPUT_CHANNEL_NAME, NULL },
5908 { FreeRDP_AudioCapture, AUDIN_CHANNEL_NAME, NULL },
5909 { FreeRDP_AudioPlayback, RDPSND_CHANNEL_NAME, NULL },
5910#ifdef CHANNEL_RDPEI_CLIENT
5911 { FreeRDP_MultiTouchInput, RDPEI_CHANNEL_NAME, NULL },
5913 { FreeRDP_SupportGraphicsPipeline, RDPGFX_CHANNEL_NAME, NULL },
5914 { FreeRDP_SupportEchoChannel, ECHO_CHANNEL_NAME, NULL },
5915 { FreeRDP_SupportSSHAgentChannel,
"sshagent", NULL },
5916 { FreeRDP_SupportDisplayControl, DISP_CHANNEL_NAME, NULL },
5917 { FreeRDP_SupportGeometryTracking, GEOMETRY_CHANNEL_NAME, NULL },
5918 { FreeRDP_SupportVideoOptimized, VIDEO_CHANNEL_NAME, NULL },
5919 { FreeRDP_RemoteCredentialGuard, RDPEAR_CHANNEL_NAME, NULL },
5922 ChannelToLoad staticChannels[] = {
5923 { FreeRDP_AudioPlayback, RDPSND_CHANNEL_NAME, NULL },
5924 { FreeRDP_RedirectClipboard, CLIPRDR_SVC_CHANNEL_NAME, NULL },
5925#if defined(CHANNEL_ENCOMSP_CLIENT)
5926 { FreeRDP_EncomspVirtualChannel, ENCOMSP_SVC_CHANNEL_NAME, settings },
5928 { FreeRDP_RemdeskVirtualChannel, REMDESK_SVC_CHANNEL_NAME, settings },
5929 { FreeRDP_RemoteApplicationMode, RAIL_SVC_CHANNEL_NAME, settings }
5935 for (
size_t i = 0; i < ARRAYSIZE(dynChannels); i++)
5937 if ((dynChannels[i].settingId == FreeRDP_BOOL_UNUSED) ||
5940 const char*
const p[] = { dynChannels[i].channelName };
5942 if (!freerdp_client_add_dynamic_channel(settings, ARRAYSIZE(p), p))
5950 if ((freerdp_static_channel_collection_find(settings, RDPSND_CHANNEL_NAME)) ||
5951 (freerdp_dynamic_channel_collection_find(settings, RDPSND_CHANNEL_NAME))
5952#
if defined(CHANNEL_TSMF_CLIENT)
5953 || (freerdp_dynamic_channel_collection_find(settings,
"tsmf"))
5963 if (freerdp_dynamic_channel_collection_find(settings, AUDIN_CHANNEL_NAME))
5979 if (DrivesToRedirect && (strlen(DrivesToRedirect) != 0))
5989 char* context = NULL;
5991 value = _strdup(DrivesToRedirect);
5995 tok = strtok_s(value,
";", &context);
5998 WLog_ERR(TAG,
"DrivesToRedirect contains invalid data: '%s'", DrivesToRedirect);
6015 const char* name = NULL;
6016 const char* drive = tok;
6017 char* subcontext = NULL;
6018 char* start = strtok_s(tok,
"(", &subcontext);
6019 char* end = strtok_s(NULL,
")", &subcontext);
6023 if (freerdp_path_valid(name, NULL) && freerdp_path_valid(drive, NULL))
6025 success = freerdp_client_add_drive(settings, name, NULL);
6027 success = freerdp_client_add_drive(settings, drive, NULL);
6030 success = freerdp_client_add_drive(settings, drive, name);
6038 tok = strtok_s(NULL,
";", &context);
6047 if (!freerdp_device_collection_find(settings,
"drive"))
6049 const char*
const params[] = {
"drive",
"media",
"*" };
6051 if (!freerdp_client_add_device_channel(settings, ARRAYSIZE(params), params))
6068 if (!freerdp_device_collection_find(settings,
"drive"))
6070 const char* params[] = {
"drive",
"home",
"%" };
6072 if (!freerdp_client_add_device_channel(settings, ARRAYSIZE(params), params))
6079 if (!freerdp_client_load_static_channel_addin(channels, settings, RDPDR_SVC_CHANNEL_NAME,
6083 if (!freerdp_static_channel_collection_find(settings, RDPSND_CHANNEL_NAME) &&
6084 !freerdp_dynamic_channel_collection_find(settings, RDPSND_CHANNEL_NAME))
6086 const char*
const params[] = { RDPSND_CHANNEL_NAME,
"sys:fake" };
6088 if (!freerdp_client_add_static_channel(settings, ARRAYSIZE(params), params))
6091 if (!freerdp_client_add_dynamic_channel(settings, ARRAYSIZE(params), params))
6098 if (!freerdp_device_collection_find_type(settings, RDPDR_DTYP_SMARTCARD))
6100 RDPDR_DEVICE* smartcard = freerdp_device_new(RDPDR_DTYP_SMARTCARD, 0, NULL);
6105 if (!freerdp_device_collection_add(settings, smartcard))
6107 freerdp_device_free(smartcard);
6115 if (!freerdp_device_collection_find_type(settings, RDPDR_DTYP_PRINT))
6117 RDPDR_DEVICE* printer = freerdp_device_new(RDPDR_DTYP_PRINT, 0, NULL);
6122 if (!freerdp_device_collection_add(settings, printer))
6124 freerdp_device_free(printer);
6151 for (
size_t i = 0; i < ARRAYSIZE(staticChannels); i++)
6153 if ((staticChannels[i].settingId == 0) ||
6156 if (staticChannels[i].args)
6158 if (!freerdp_client_load_static_channel_addin(
6159 channels, settings, staticChannels[i].channelName, staticChannels[i].args))
6164 const char*
const p[] = { staticChannels[i].channelName };
6165 if (!freerdp_client_add_static_channel(settings, ARRAYSIZE(p), p))
6175 const char*
const p[] = { RDP2TCP_DVC_CHANNEL_NAME, RDP2TCPArgs };
6176 if (!freerdp_client_add_static_channel(settings, ARRAYSIZE(p), p))
6185 freerdp_settings_get_pointer_array_writable(settings, FreeRDP_StaticChannelArray, i);
6187 if (!freerdp_client_load_static_channel_addin(channels, settings, _args->argv[0], _args))
6199 if (!freerdp_client_load_static_channel_addin(channels, settings, DRDYNVC_SVC_CHANNEL_NAME,
6207void freerdp_client_warn_unmaintained(
int argc,
char* argv[])
6209 const char* app = (argc > 0) ? argv[0] :
"INVALID_ARGV";
6210 const DWORD log_level = WLOG_WARN;
6211 wLog* log = WLog_Get(TAG);
6214 if (!WLog_IsLevelActive(log, log_level))
6217 WLog_Print_unchecked(log, log_level,
"[unmaintained] %s client is currently unmaintained!",
6219 WLog_Print_unchecked(
6221 " If problems occur please check https://github.com/FreeRDP/FreeRDP/issues for "
6223 WLog_Print_unchecked(
6225 "Be prepared to fix issues yourself though as nobody is actively working on this.");
6226 WLog_Print_unchecked(
6228 " Developers hang out in https://matrix.to/#/#FreeRDP:matrix.org?via=matrix.org "
6229 "- don't hesitate to ask some questions. (replies might take some time depending "
6230 "on your timezone) - if you intend using this component write us a message");
6233void freerdp_client_warn_experimental(
int argc,
char* argv[])
6235 const char* app = (argc > 0) ? argv[0] :
"INVALID_ARGV";
6236 const DWORD log_level = WLOG_WARN;
6237 wLog* log = WLog_Get(TAG);
6240 if (!WLog_IsLevelActive(log, log_level))
6243 WLog_Print_unchecked(log, log_level,
"[experimental] %s client is currently experimental!",
6245 WLog_Print_unchecked(
6247 " If problems occur please check https://github.com/FreeRDP/FreeRDP/issues for "
6248 "known issues or create a new one!");
6249 WLog_Print_unchecked(
6251 " Developers hang out in https://matrix.to/#/#FreeRDP:matrix.org?via=matrix.org "
6252 "- don't hesitate to ask some questions. (replies might take some time depending "
6253 "on your timezone)");
6256void freerdp_client_warn_deprecated(
int argc,
char* argv[])
6258 const char* app = (argc > 0) ? argv[0] :
"INVALID_ARGV";
6259 const DWORD log_level = WLOG_WARN;
6260 wLog* log = WLog_Get(TAG);
6263 if (!WLog_IsLevelActive(log, log_level))
6266 WLog_Print_unchecked(log, log_level,
"[deprecated] %s client has been deprecated", app);
6267 WLog_Print_unchecked(log, log_level,
"As replacement there is a SDL3 based client available.");
6268 WLog_Print_unchecked(
6270 "If you are interested in keeping %s alive get in touch with the developers", app);
6271 WLog_Print_unchecked(
6273 "The project is hosted at https://github.com/freerdp/freerdp and "
6274 " developers hang out in https://matrix.to/#/#FreeRDP:matrix.org?via=matrix.org "
6275 "- don't hesitate to ask some questions. (replies might take some time depending "
6276 "on your timezone)");
FREERDP_API UINT32 freerdp_settings_get_uint32(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id)
Returns a UINT32 settings value.
FREERDP_API BOOL freerdp_settings_set_string(rdpSettings *settings, FreeRDP_Settings_Keys_String id, const char *param)
Sets a string settings value. The param is copied.
FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.
FREERDP_API INT64 freerdp_settings_get_int64(const rdpSettings *settings, FreeRDP_Settings_Keys_Int64 id)
Returns a INT64 settings value.
FREERDP_API SSIZE_T freerdp_settings_get_type_for_key(SSIZE_T key)
Get a key type for the key index.
FREERDP_API UINT64 freerdp_settings_get_uint64(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt64 id)
Returns a UINT64 settings value.
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.
FREERDP_API UINT16 freerdp_settings_get_uint16(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt16 id)
Returns a UINT16 settings value.
FREERDP_API void * freerdp_settings_get_pointer_writable(rdpSettings *settings, FreeRDP_Settings_Keys_Pointer id)
Returns a mutable pointer settings value.
FREERDP_API BOOL freerdp_set_gateway_usage_method(rdpSettings *settings, UINT32 GatewayUsageMethod)
FREERDP_API BOOL freerdp_settings_set_uint64(rdpSettings *settings, FreeRDP_Settings_Keys_UInt64 id, UINT64 param)
Sets a UINT64 settings value.
FREERDP_API INT32 freerdp_settings_get_int32(const rdpSettings *settings, FreeRDP_Settings_Keys_Int32 id)
Returns a INT32 settings value.
FREERDP_API const void * freerdp_settings_get_pointer(const rdpSettings *settings, FreeRDP_Settings_Keys_Pointer id)
Returns a immutable pointer settings value.
FREERDP_API BOOL freerdp_settings_set_string_len(rdpSettings *settings, FreeRDP_Settings_Keys_String id, const char *param, size_t len)
Sets a string settings value. The param is copied.
FREERDP_API BOOL freerdp_settings_set_uint32(rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id, UINT32 param)
Sets a UINT32 settings value.
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...
FREERDP_API char * freerdp_settings_get_string_writable(rdpSettings *settings, FreeRDP_Settings_Keys_String id)
Returns a string settings value.
FREERDP_API const char * freerdp_settings_get_name_for_key(SSIZE_T key)
Returns the type name for a key.
FREERDP_API const char * freerdp_settings_get_string(const rdpSettings *settings, FreeRDP_Settings_Keys_String id)
Returns a immutable string settings value.
FREERDP_API BOOL freerdp_settings_set_uint16(rdpSettings *settings, FreeRDP_Settings_Keys_UInt16 id, UINT16 param)
Sets a UINT16 settings value.
FREERDP_API INT16 freerdp_settings_get_int16(const rdpSettings *settings, FreeRDP_Settings_Keys_Int16 id)
Returns a INT16 settings value.
FREERDP_API BOOL freerdp_settings_set_bool(rdpSettings *settings, FreeRDP_Settings_Keys_Bool id, BOOL param)
Sets a BOOL settings value.