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* credential_args[] = {
"p",
"smartcard-logon",
78#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
81 "pth",
"reconnect-cookie",
84static const char* option_starts_with(
const char* what,
const char* val);
85static BOOL option_ends_with(
const char* str,
const char* ext);
86static BOOL option_equals(
const char* what,
const char* val);
88static BOOL freerdp_client_print_codepages(
const char* arg)
92 const char* filter = NULL;
97 filter = strchr(arg,
',');
103 pages = freerdp_keyboard_get_matching_codepages(column, filter, &count);
107 printf(
"%-10s %-8s %-60s %-36s %-48s\n",
"<id>",
"<locale>",
"<win langid>",
"<language>",
109 for (
size_t x = 0; x < count; x++)
112 char buffer[2048] = { 0 };
114 if (strnlen(page->subLanguageSymbol, ARRAYSIZE(page->subLanguageSymbol)) > 0)
115 (void)_snprintf(buffer,
sizeof(buffer),
"[%s|%s]", page->primaryLanguageSymbol,
116 page->subLanguageSymbol);
118 (
void)_snprintf(buffer,
sizeof(buffer),
"[%s]", page->primaryLanguageSymbol);
119 printf(
"id=0x%04" PRIx16
": [%-6s] %-60s %-36s %-48s\n", page->id, page->locale, buffer,
120 page->primaryLanguage, page->subLanguage);
122 freerdp_codepages_free(pages);
126static BOOL freerdp_path_valid(
const char* path, BOOL* special)
128 const char DynamicDrives[] =
"DynamicDrives";
135 (option_equals(
"*", path) || option_equals(DynamicDrives, path) || option_equals(
"%", path))
139 isPath = winpr_PathFileExists(path);
142 *special = isSpecial;
144 return isSpecial || isPath;
147static BOOL freerdp_sanitize_drive_name(
char* name,
const char* invalid,
const char* replacement)
149 if (!name || !invalid || !replacement)
151 if (strlen(invalid) != strlen(replacement))
154 while (*invalid !=
'\0')
156 const char what = *invalid++;
157 const char with = *replacement++;
160 while ((cur = strchr(cur, what)) != NULL)
166static char* name_from_path(
const char* path)
168 const char* name =
"NULL";
171 if (option_equals(
"%", path))
173 else if (option_equals(
"*", path))
174 name =
"hotplug-all";
175 else if (option_equals(
"DynamicDrives", path))
180 return _strdup(name);
183static BOOL freerdp_client_add_drive(rdpSettings* settings,
const char* path,
const char* name)
204 if (!skip && winpr_PathFileExists(name))
206 if (!winpr_PathFileExists(path) || (!PathIsRelativeA(name) && PathIsRelativeA(path)))
208 const char* tmp = path;
216 dname = _strdup(name);
218 dname = name_from_path(path);
220 if (freerdp_sanitize_drive_name(dname,
"\\/",
"__"))
222 const char* args[] = { dname, path };
223 device = freerdp_device_new(RDPDR_DTYP_FILESYSTEM, ARRAYSIZE(args), args);
233 BOOL isSpecial = FALSE;
234 BOOL isPath = freerdp_path_valid(path, &isSpecial);
236 if (!isPath && !isSpecial)
238 WLog_WARN(TAG,
"Invalid drive to redirect: '%s' does not exist, skipping.", path);
239 freerdp_device_free(device);
241 else if (!freerdp_device_collection_add(settings, device))
248 freerdp_device_free(device);
252static BOOL value_to_int(
const char* value, LONGLONG* result, LONGLONG min, LONGLONG max)
256 if (!value || !result)
260 rc = _strtoi64(value, NULL, 0);
265 if ((rc < min) || (rc > max))
272static BOOL value_to_uint(
const char* value, ULONGLONG* result, ULONGLONG min, ULONGLONG max)
274 unsigned long long rc = 0;
276 if (!value || !result)
280 rc = _strtoui64(value, NULL, 0);
285 if ((rc < min) || (rc > max))
292BOOL freerdp_client_print_version(
void)
294 printf(
"This is FreeRDP version %s (%s)\n", FREERDP_VERSION_FULL, FREERDP_GIT_REVISION);
298BOOL freerdp_client_print_version_ex(
int argc,
char** argv)
300 WINPR_ASSERT(argc >= 0);
301 WINPR_ASSERT(argv || (argc == 0));
302 const char* name = (argc > 0) ? argv[0] :
"argc < 1";
303 printf(
"This is FreeRDP version [%s] %s (%s)\n", name, FREERDP_VERSION_FULL,
304 FREERDP_GIT_REVISION);
308BOOL freerdp_client_print_buildconfig(
void)
310 printf(
"%s", freerdp_get_build_config());
314BOOL freerdp_client_print_buildconfig_ex(
int argc,
char** argv)
316 WINPR_ASSERT(argc >= 0);
317 WINPR_ASSERT(argv || (argc == 0));
318 const char* name = (argc > 0) ? argv[0] :
"argc < 1";
319 printf(
"[%s] %s", name, freerdp_get_build_config());
323static void freerdp_client_print_scancodes(
void)
325 printf(
"RDP scancodes and their name for use with /kbd:remap\n");
327 for (UINT32 x = 0; x < UINT16_MAX; x++)
329 const char* name = freerdp_keyboard_scancode_name(x);
331 printf(
"0x%04" PRIx32
" --> %s\n", x, name);
335static BOOL is_delimiter(
char c,
const char* delimiters)
338 while ((d = *delimiters++) !=
'\0')
346static const char* get_last(
const char* start,
size_t len,
const char* delimiters)
348 const char* last = NULL;
349 for (
size_t x = 0; x < len; x++)
352 if (is_delimiter(c, delimiters))
358static SSIZE_T next_delimiter(
const char* text,
size_t len,
size_t max,
const char* delimiters)
363 const char* last = get_last(text, max, delimiters);
367 return (SSIZE_T)(last - text);
370static SSIZE_T forced_newline_at(
const char* text,
size_t len,
size_t limit,
371 const char* force_newline)
374 while ((d = *force_newline++) !=
'\0')
376 const char* tok = strchr(text, d);
379 const size_t offset = WINPR_ASSERTING_INT_CAST(
size_t, tok - text);
380 if ((offset > len) || (offset > limit))
382 return (SSIZE_T)(offset);
388static BOOL print_align(
size_t start_offset,
size_t* current)
390 WINPR_ASSERT(current);
391 if (*current < start_offset)
393 const int rc = printf(
"%*c", (
int)(start_offset - *current),
' ');
396 *current += (size_t)rc;
401static char* print_token(
char* text,
size_t start_offset,
size_t* current,
size_t limit,
402 const char* delimiters,
const char* force_newline)
405 const size_t tlen = strnlen(text, limit);
407 const SSIZE_T force_at = forced_newline_at(text, len, limit - *current, force_newline);
408 BOOL isForce = (force_at >= 0);
411 len = MIN(len, (
size_t)force_at);
413 if (!print_align(start_offset, current))
416 const SSIZE_T delim = next_delimiter(text, len, limit - *current, delimiters);
417 const BOOL isDelim = delim > 0;
420 len = MIN(len, (
size_t)delim + 1);
423 rc = printf(
"%.*s", (
int)len, text);
427 if (isForce || isDelim)
432 const size_t offset = len + ((isForce && (force_at == 0)) ? 1 : 0);
433 return &text[offset];
436 *current += (size_t)rc;
438 if (tlen == (
size_t)rc)
440 return &text[(size_t)rc];
443static size_t print_optionals(
const char* text,
size_t start_offset,
size_t current)
445 const size_t limit = 80;
446 char* str = _strdup(text);
451 cur = print_token(cur, start_offset + 1, ¤t, limit,
"[], ",
"\r\n");
452 }
while (cur != NULL);
458static size_t print_description(
const char* text,
size_t start_offset,
size_t current)
460 const size_t limit = 80;
461 char* str = _strdup(text);
465 cur = print_token(cur, start_offset, ¤t, limit,
" ",
"\r\n");
468 const int rc = printf(
"\n");
471 const size_t src = WINPR_ASSERTING_INT_CAST(
size_t, rc);
472 WINPR_ASSERT(SIZE_MAX - src > current);
478static int cmp_cmdline_args(
const void* pva,
const void* pvb)
483 if (!a->Name && !b->Name)
489 return strcmp(a->Name, b->Name);
504 const size_t description_offset = 30 + 8;
506 if (arg->Flags & (COMMAND_LINE_VALUE_BOOL | COMMAND_LINE_VALUE_FLAG))
508 if ((arg->Flags & (uint32_t)~COMMAND_LINE_VALUE_BOOL) == 0)
509 rc = printf(
" %s%s", arg->Default ?
"-" :
"+", arg->Name);
510 else if ((arg->Flags & COMMAND_LINE_VALUE_OPTIONAL) != 0)
511 rc = printf(
" [%s|/]%s", arg->Default ?
"-" :
"+", arg->Name);
514 rc = printf(
" %s%s", arg->Default ?
"-" :
"+", arg->Name);
518 rc = printf(
" /%s", arg->Name);
524 if ((arg->Flags & COMMAND_LINE_VALUE_REQUIRED) ||
525 (arg->Flags & COMMAND_LINE_VALUE_OPTIONAL))
529 if (arg->Flags & COMMAND_LINE_VALUE_OPTIONAL)
535 pos = print_optionals(arg->Format, pos, pos);
547 pos = print_optionals(arg->Format, pos, pos);
550 if (pos > description_offset)
558 rc = printf(
"%*c", (
int)(description_offset - pos),
' ');
563 if (arg->Flags & COMMAND_LINE_VALUE_BOOL)
565 rc = printf(
"%s ", arg->Default ?
"Disable" :
"Enable");
571 print_description(arg->Text, description_offset, pos);
572 }
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
575BOOL freerdp_client_print_command_line_help(
int argc,
char** argv)
577 return freerdp_client_print_command_line_help_ex(argc, argv, NULL);
581 SSIZE_T count,
size_t* pcount)
583 WINPR_ASSERT(pcount);
588 while (cur && cur->Name)
603 while (cur && cur->Name)
605 largs[lcount++] = *cur++;
608 cur = global_cmd_args;
609 while (cur && cur->Name)
611 largs[lcount++] = *cur++;
617BOOL freerdp_client_print_command_line_help_ex(
int argc,
char** argv,
620 const char* name =
"FreeRDP";
632 printf(
"FreeRDP - A Free Remote Desktop Protocol Implementation\n");
633 printf(
"See www.freerdp.com for more information\n");
635 printf(
"Usage: %s [file] [options] [/v:<server>[:port]]\n", argv[0]);
638 printf(
" /flag (enables flag)\n");
639 printf(
" /option:<value> (specifies option with value)\n");
640 printf(
" +toggle -toggle (enables or disables toggle, where '/' is a synonym of '+')\n");
643 freerdp_client_print_command_line_args(largs, lcount);
647 printf(
"Examples:\n");
648 printf(
" %s connection.rdp /p:Pwd123! /f\n", name);
649 printf(
" %s /u:CONTOSO\\JohnDoe /p:Pwd123! /v:rdp.contoso.com\n", name);
650 printf(
" %s /u:JohnDoe /p:Pwd123! /w:1366 /h:768 /v:192.168.1.100:4489\n", name);
651 printf(
" %s /u:JohnDoe /p:Pwd123! /vmconnect:C824F53E-95D2-46C6-9A18-23A5BB403532 "
652 "/v:192.168.1.100\n",
654 printf(
" %s /u:\\AzureAD\\user@corp.example /p:pwd /v:host\n", name);
655 printf(
"Use a generic pipe as transport:");
656 printf(
" %s /v:/path/to/pipe\n", name);
657 printf(
"Use a external socket:");
658 printf(
" %s /v:|:1234\n", name);
660 printf(
"Clipboard Redirection: +clipboard\n");
662 printf(
"Drive Redirection: /drive:home,/home/user\n");
663 printf(
"Smartcard Redirection: /smartcard:<device>\n");
664 printf(
"Smartcard logon with Kerberos authentication: /smartcard-logon /sec:nla\n");
666#if defined(CHANNEL_SERIAL_CLIENT)
667 printf(
"Serial Port Redirection: /serial:<name>,<device>,[SerCx2|SerCx|Serial],[permissive]\n");
668 printf(
"Serial Port Redirection: /serial:COM1,/dev/ttyS0\n");
670#if defined(CHANNEL_PARALLEL_CLIENT)
671 printf(
"Parallel Port Redirection: /parallel:<name>,<device>\n");
673 printf(
"Printer Redirection: /printer:<device>,<driver>,[default]\n");
674 printf(
"TCP redirection: /rdp2tcp:/usr/bin/rdp2tcp\n");
676 printf(
"Audio Output Redirection: /sound:sys:oss,dev:1,format:1\n");
677 printf(
"Audio Output Redirection: /sound:sys:alsa\n");
678 printf(
"Audio Input Redirection: /microphone:sys:oss,dev:1,format:1\n");
679 printf(
"Audio Input Redirection: /microphone:sys:alsa\n");
681 printf(
"Multimedia Redirection: /video\n");
682#ifdef CHANNEL_URBDRC_CLIENT
683 printf(
"USB Device Redirection: /usb:id:054c:0268#4669:6e6b,addr:04:0c\n");
686 printf(
"For Gateways, the https_proxy environment variable is respected:\n");
688 printf(
" set HTTPS_PROXY=http://proxy.contoso.com:3128/\n");
690 printf(
" export https_proxy=http://proxy.contoso.com:3128/\n");
692 printf(
" %s /g:rdp.contoso.com ...\n", name);
694 printf(
"More documentation is coming, in the meantime consult source files\n");
699static BOOL option_is_rdp_file(
const char* option)
701 WINPR_ASSERT(option);
703 if (option_ends_with(option,
".rdp"))
705 if (option_ends_with(option,
".rdpw"))
710static BOOL option_is_incident_file(
const char* option)
712 WINPR_ASSERT(option);
714 if (option_ends_with(option,
".msrcIncident"))
719static int freerdp_client_command_line_pre_filter(
void* context,
int index,
int argc, LPSTR* argv)
724 rdpSettings* settings = NULL;
729 length = strlen(argv[index]);
733 if (option_is_rdp_file(argv[index]))
735 settings = (rdpSettings*)context;
738 return COMMAND_LINE_ERROR_MEMORY;
746 if (option_is_incident_file(argv[index]))
748 settings = (rdpSettings*)context;
751 return COMMAND_LINE_ERROR_MEMORY;
761BOOL freerdp_client_add_device_channel(rdpSettings* settings,
size_t count,
762 const char*
const* params)
764 WINPR_ASSERT(settings);
765 WINPR_ASSERT(params);
766 WINPR_ASSERT(count > 0);
768 if (option_equals(params[0],
"drive"))
777 rc = freerdp_client_add_drive(settings, params[1], NULL);
779 rc = freerdp_client_add_drive(settings, params[2], params[1]);
783 else if (option_equals(params[0],
"printer"))
795 printer = freerdp_device_new(RDPDR_DTYP_PRINT, count - 1, ¶ms[1]);
799 if (!freerdp_device_collection_add(settings, printer))
801 freerdp_device_free(printer);
807 else if (option_equals(params[0],
"smartcard"))
819 smartcard = freerdp_device_new(RDPDR_DTYP_SMARTCARD, count - 1, ¶ms[1]);
824 if (!freerdp_device_collection_add(settings, smartcard))
826 freerdp_device_free(smartcard);
832#if defined(CHANNEL_SERIAL_CLIENT)
833 else if (option_equals(params[0],
"serial"))
845 serial = freerdp_device_new(RDPDR_DTYP_SERIAL, count - 1, ¶ms[1]);
850 if (!freerdp_device_collection_add(settings, serial))
852 freerdp_device_free(serial);
859 else if (option_equals(params[0],
"parallel"))
871 parallel = freerdp_device_new(RDPDR_DTYP_PARALLEL, count - 1, ¶ms[1]);
876 if (!freerdp_device_collection_add(settings, parallel))
878 freerdp_device_free(parallel);
888BOOL freerdp_client_del_static_channel(rdpSettings* settings,
const char* name)
890 return freerdp_static_channel_collection_del(settings, name);
893BOOL freerdp_client_add_static_channel(rdpSettings* settings,
size_t count,
894 const char*
const* params)
898 if (!settings || !params || !params[0] || (count > INT_MAX))
901 if (freerdp_static_channel_collection_find(settings, params[0]))
904 _args = freerdp_addin_argv_new(count, params);
909 if (!freerdp_static_channel_collection_add(settings, _args))
914 freerdp_addin_argv_free(_args);
918BOOL freerdp_client_del_dynamic_channel(rdpSettings* settings,
const char* name)
920 return freerdp_dynamic_channel_collection_del(settings, name);
923BOOL freerdp_client_add_dynamic_channel(rdpSettings* settings,
size_t count,
924 const char*
const* params)
928 if (!settings || !params || !params[0] || (count > INT_MAX))
931 if (freerdp_dynamic_channel_collection_find(settings, params[0]))
934 _args = freerdp_addin_argv_new(count, params);
939 if (!freerdp_dynamic_channel_collection_add(settings, _args))
945 freerdp_addin_argv_free(_args);
949static BOOL read_pem_file(rdpSettings* settings, FreeRDP_Settings_Keys_String
id,
const char* file)
952 char* pem = crypto_read_pem(file, &length);
953 if (!pem || (length == 0))
967 CMDLINE_SUBOPTION_STRING,
968 CMDLINE_SUBOPTION_FILE,
969} CmdLineSubOptionType;
971typedef BOOL (*CmdLineSubOptionCb)(
const char* value, rdpSettings* settings);
975 FreeRDP_Settings_Keys_String id;
976 CmdLineSubOptionType opttype;
977 CmdLineSubOptionCb cb;
980static BOOL parseSubOptions(rdpSettings* settings,
const CmdLineSubOptions* opts,
size_t count,
985 for (
size_t xx = 0; xx < count; xx++)
987 const CmdLineSubOptions* opt = &opts[xx];
989 if (option_starts_with(opt->optname, arg))
991 const size_t optlen = strlen(opt->optname);
992 const char* val = &arg[optlen];
995 switch (opt->opttype)
997 case CMDLINE_SUBOPTION_STRING:
1000 case CMDLINE_SUBOPTION_FILE:
1001 status = read_pem_file(settings, opt->id, val);
1004 WLog_ERR(TAG,
"invalid subOption type");
1011 if (opt->cb && !opt->cb(val, settings))
1020 WLog_ERR(TAG,
"option %s not handled", arg);
1025#define fail_at(arg, rc) fail_at_((arg), (rc), __FILE__, __func__, __LINE__)
1032 const DWORD level = WLOG_ERROR;
1033 wLog* log = WLog_Get(TAG);
1034 if (WLog_IsLevelActive(log, level))
1035 WLog_PrintTextMessage(log, level, line, file, fkt,
1036 "Command line parsing failed at '%s' value '%s' [%d]", arg->Name,
1043 rdpSettings* settings = (rdpSettings*)context;
1044 int status = CHANNEL_RC_OK;
1045 BOOL enable = arg->Value ? TRUE : FALSE;
1047 CommandLineSwitchStart(arg) CommandLineSwitchCase(arg,
"a")
1050 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
1052 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1053 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1055 status = COMMAND_LINE_ERROR;
1057 CommandLineParserFree(ptr);
1059 return fail_at(arg, status);
1061 CommandLineSwitchCase(arg,
"kerberos")
1065 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"kerberos", arg->Value, &count);
1068 const CmdLineSubOptions opts[] = {
1069 {
"kdc-url:", FreeRDP_KerberosKdcUrl, CMDLINE_SUBOPTION_STRING, NULL },
1070 {
"start-time:", FreeRDP_KerberosStartTime, CMDLINE_SUBOPTION_STRING, NULL },
1071 {
"lifetime:", FreeRDP_KerberosLifeTime, CMDLINE_SUBOPTION_STRING, NULL },
1072 {
"renewable-lifetime:", FreeRDP_KerberosRenewableLifeTime,
1073 CMDLINE_SUBOPTION_STRING, NULL },
1074 {
"cache:", FreeRDP_KerberosCache, CMDLINE_SUBOPTION_STRING, NULL },
1075 {
"armor:", FreeRDP_KerberosArmor, CMDLINE_SUBOPTION_STRING, NULL },
1076 {
"pkinit-anchors:", FreeRDP_PkinitAnchors, CMDLINE_SUBOPTION_STRING, NULL },
1077 {
"pkcs11-module:", FreeRDP_Pkcs11Module, CMDLINE_SUBOPTION_STRING, NULL }
1080 for (
size_t x = 1; x < count; x++)
1082 const char* cur = ptr[x];
1083 if (!parseSubOptions(settings, opts, ARRAYSIZE(opts), cur))
1085 CommandLineParserFree(ptr);
1086 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
1090 CommandLineParserFree(ptr);
1093 CommandLineSwitchCase(arg,
"vc")
1096 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
1097 if (!freerdp_client_add_static_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,
"dvc")
1106 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
1107 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1108 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1109 CommandLineParserFree(ptr);
1111 return fail_at(arg, status);
1113 CommandLineSwitchCase(arg,
"drive")
1116 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1117 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1118 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1119 CommandLineParserFree(ptr);
1121 return fail_at(arg, status);
1123#if defined(CHANNEL_SERIAL_CLIENT)
1124 CommandLineSwitchCase(arg,
"serial")
1127 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1128 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1129 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1130 CommandLineParserFree(ptr);
1132 return fail_at(arg, status);
1135#if defined(CHANNEL_PARALLEL_CLIENT)
1136 CommandLineSwitchCase(arg,
"parallel")
1139 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1140 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1141 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1142 CommandLineParserFree(ptr);
1144 return fail_at(arg, status);
1147 CommandLineSwitchCase(arg,
"smartcard")
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,
"printer")
1160 char** ptr = CommandLineParseCommaSeparatedValuesEx(arg->Name, arg->Value, &count);
1161 if (!freerdp_client_add_device_channel(settings, count, (
const char*
const*)ptr))
1162 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1163 CommandLineParserFree(ptr);
1165 return fail_at(arg, status);
1167 CommandLineSwitchCase(arg,
"usb")
1171 CommandLineParseCommaSeparatedValuesEx(URBDRC_CHANNEL_NAME, arg->Value, &count);
1172 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1173 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1174 CommandLineParserFree(ptr);
1176 return fail_at(arg, status);
1178 CommandLineSwitchCase(arg,
"multitouch")
1181 return fail_at(arg, COMMAND_LINE_ERROR);
1183 CommandLineSwitchCase(arg,
"gestures")
1186 return fail_at(arg, COMMAND_LINE_ERROR);
1188 CommandLineSwitchCase(arg,
"echo")
1191 return fail_at(arg, COMMAND_LINE_ERROR);
1193 CommandLineSwitchCase(arg,
"ssh-agent")
1196 return fail_at(arg, COMMAND_LINE_ERROR);
1198 CommandLineSwitchCase(arg,
"disp")
1201 return fail_at(arg, COMMAND_LINE_ERROR);
1203 CommandLineSwitchCase(arg,
"geometry")
1206 return fail_at(arg, COMMAND_LINE_ERROR);
1208 CommandLineSwitchCase(arg,
"video")
1212 return fail_at(arg, COMMAND_LINE_ERROR);
1214 return fail_at(arg, COMMAND_LINE_ERROR);
1216 CommandLineSwitchCase(arg,
"sound")
1220 CommandLineParseCommaSeparatedValuesEx(RDPSND_CHANNEL_NAME, arg->Value, &count);
1221 if (!freerdp_client_add_static_channel(settings, count, (
const char*
const*)ptr))
1222 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1223 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1224 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1226 CommandLineParserFree(ptr);
1228 return fail_at(arg, status);
1230 CommandLineSwitchCase(arg,
"microphone")
1233 char** ptr = CommandLineParseCommaSeparatedValuesEx(AUDIN_CHANNEL_NAME, arg->Value, &count);
1234 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1235 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1236 CommandLineParserFree(ptr);
1238 return fail_at(arg, status);
1240#if defined(CHANNEL_TSMF_CLIENT)
1241 CommandLineSwitchCase(arg,
"multimedia")
1244 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"tsmf", arg->Value, &count);
1245 if (!freerdp_client_add_dynamic_channel(settings, count, (
const char*
const*)ptr))
1246 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1247 CommandLineParserFree(ptr);
1249 return fail_at(arg, status);
1252 CommandLineSwitchCase(arg,
"heartbeat")
1255 return fail_at(arg, COMMAND_LINE_ERROR);
1257 CommandLineSwitchCase(arg,
"multitransport")
1260 return fail_at(arg, COMMAND_LINE_ERROR);
1265 (TRANSPORT_TYPE_UDP_FECR | TRANSPORT_TYPE_UDP_FECL | TRANSPORT_TYPE_UDP_PREFERRED);
1268 return fail_at(arg, COMMAND_LINE_ERROR);
1270 CommandLineSwitchEnd(arg)
1277 int status = freerdp_client_command_line_post_filter_int(context, arg);
1278 return status == CHANNEL_RC_OK ? 1 : -1;
1281static BOOL freerdp_parse_username_ptr(
const char* username,
const char** user,
size_t* userlen,
1282 const char** domain,
size_t* domainlen)
1285 WINPR_ASSERT(userlen);
1286 WINPR_ASSERT(domain);
1287 WINPR_ASSERT(domainlen);
1292 const char* p = strchr(username,
'\\');
1302 const size_t length = (size_t)(p - username);
1304 *userlen = strlen(*user);
1307 *domainlen = length;
1316 *userlen = strlen(username);
1322static BOOL freerdp_parse_username_settings(
const char* username, rdpSettings* settings,
1323 FreeRDP_Settings_Keys_String userID,
1324 FreeRDP_Settings_Keys_String domainID)
1326 const char* user = NULL;
1327 const char* domain = NULL;
1329 size_t domainlen = 0;
1331 const BOOL rc = freerdp_parse_username_ptr(username, &user, &userlen, &domain, &domainlen);
1339BOOL freerdp_parse_username(
const char* username,
char** puser,
char** pdomain)
1341 const char* user = NULL;
1342 const char* domain = NULL;
1344 size_t domainlen = 0;
1349 const BOOL rc = freerdp_parse_username_ptr(username, &user, &userlen, &domain, &domainlen);
1355 *puser = strndup(user, userlen);
1362 *pdomain = strndup(domain, domainlen);
1374BOOL freerdp_parse_hostname(
const char* hostname,
char** host,
int* port)
1377 p = strrchr(hostname,
':');
1381 size_t length = (size_t)(p - hostname);
1384 if (!value_to_int(p + 1, &val, 1, UINT16_MAX))
1387 *host = (
char*)calloc(length + 1UL,
sizeof(
char));
1392 CopyMemory(*host, hostname, length);
1393 (*host)[length] =
'\0';
1394 *port = (UINT16)val;
1398 *host = _strdup(hostname);
1409static BOOL freerdp_apply_connection_type(rdpSettings* settings, UINT32 type)
1411 struct network_settings
1413 FreeRDP_Settings_Keys_Bool id;
1416 const struct network_settings config[] = {
1417 { FreeRDP_DisableWallpaper, { TRUE, TRUE, TRUE, TRUE, FALSE, FALSE, FALSE } },
1418 { FreeRDP_AllowFontSmoothing, { FALSE, FALSE, FALSE, FALSE, TRUE, TRUE, TRUE } },
1419 { FreeRDP_AllowDesktopComposition, { FALSE, FALSE, TRUE, TRUE, TRUE, TRUE, TRUE } },
1420 { FreeRDP_DisableFullWindowDrag, { TRUE, TRUE, TRUE, TRUE, FALSE, FALSE, FALSE } },
1421 { FreeRDP_DisableMenuAnims, { TRUE, TRUE, TRUE, TRUE, FALSE, FALSE, FALSE } },
1422 { FreeRDP_DisableThemes, { TRUE, FALSE, FALSE, FALSE, FALSE, FALSE, FALSE } }
1427 case CONNECTION_TYPE_INVALID:
1430 case CONNECTION_TYPE_MODEM:
1431 case CONNECTION_TYPE_BROADBAND_LOW:
1432 case CONNECTION_TYPE_BROADBAND_HIGH:
1433 case CONNECTION_TYPE_SATELLITE:
1434 case CONNECTION_TYPE_WAN:
1435 case CONNECTION_TYPE_LAN:
1436 case CONNECTION_TYPE_AUTODETECT:
1439 WLog_WARN(TAG,
"Unknown ConnectionType %" PRIu32
", aborting", type);
1443 for (
size_t x = 0; x < ARRAYSIZE(config); x++)
1445 const struct network_settings* cur = &config[x];
1452BOOL freerdp_set_connection_type(rdpSettings* settings, UINT32 type)
1460 case CONNECTION_TYPE_INVALID:
1461 case CONNECTION_TYPE_MODEM:
1462 case CONNECTION_TYPE_BROADBAND_LOW:
1463 case CONNECTION_TYPE_SATELLITE:
1464 case CONNECTION_TYPE_BROADBAND_HIGH:
1465 case CONNECTION_TYPE_WAN:
1466 case CONNECTION_TYPE_LAN:
1467 if (!freerdp_apply_connection_type(settings, type))
1470 case CONNECTION_TYPE_AUTODETECT:
1471 if (!freerdp_apply_connection_type(settings, type))
1485 WLog_WARN(TAG,
"Unknown ConnectionType %" PRIu32
", aborting", type);
1492static UINT32 freerdp_get_keyboard_layout_for_type(
const char* name, WINPR_ATTR_UNUSED DWORD type)
1497 freerdp_keyboard_get_layouts(RDP_KEYBOARD_LAYOUT_TYPE_STANDARD, &count);
1499 if (!layouts || (count == 0))
1502 for (
size_t x = 0; x < count; x++)
1505 if (option_equals(layout->name, name))
1513 freerdp_keyboard_layouts_free(layouts, count);
1517static UINT32 freerdp_map_keyboard_layout_name_to_id(
const char* name)
1519 const UINT32 variants[] = { RDP_KEYBOARD_LAYOUT_TYPE_STANDARD, RDP_KEYBOARD_LAYOUT_TYPE_VARIANT,
1520 RDP_KEYBOARD_LAYOUT_TYPE_IME };
1522 for (
size_t x = 0; x < ARRAYSIZE(variants); x++)
1524 UINT32 rc = freerdp_get_keyboard_layout_for_type(name, variants[x]);
1532static int freerdp_detect_command_line_pre_filter(
void* context,
int index,
int argc, LPSTR* argv)
1535 WINPR_UNUSED(context);
1542 length = strlen(argv[index]);
1546 if (option_is_rdp_file(argv[index]))
1554 if (option_is_incident_file(argv[index]))
1564static int freerdp_detect_windows_style_command_line_syntax(
int argc,
char** argv,
size_t* count,
1569 int detect_status = 0;
1572 memcpy(largs, global_cmd_args,
sizeof(global_cmd_args));
1574 flags = COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_SILENCE_PARSER;
1575 flags |= COMMAND_LINE_SIGIL_SLASH | COMMAND_LINE_SIGIL_PLUS_MINUS;
1579 flags |= COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
1584 CommandLineClearArgumentsA(largs);
1585 status = CommandLineParseArgumentsA(argc, argv, largs, flags, NULL,
1586 freerdp_detect_command_line_pre_filter, NULL);
1595 if (!(arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
1599 }
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
1601 return detect_status;
1604static int freerdp_detect_posix_style_command_line_syntax(
int argc,
char** argv,
size_t* count,
1609 int detect_status = 0;
1612 memcpy(largs, global_cmd_args,
sizeof(global_cmd_args));
1614 flags = COMMAND_LINE_SEPARATOR_SPACE | COMMAND_LINE_SILENCE_PARSER;
1615 flags |= COMMAND_LINE_SIGIL_DASH | COMMAND_LINE_SIGIL_DOUBLE_DASH;
1616 flags |= COMMAND_LINE_SIGIL_ENABLE_DISABLE;
1620 flags |= COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
1625 CommandLineClearArgumentsA(largs);
1626 status = CommandLineParseArgumentsA(argc, argv, largs, flags, NULL,
1627 freerdp_detect_command_line_pre_filter, NULL);
1636 if (!(arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
1640 }
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
1642 return detect_status;
1645static BOOL freerdp_client_detect_command_line(
int argc,
char** argv, DWORD* flags)
1647 size_t posix_cli_count = 0;
1648 size_t windows_cli_count = 0;
1649 const BOOL ignoreUnknown = TRUE;
1650 const int windows_cli_status = freerdp_detect_windows_style_command_line_syntax(
1651 argc, argv, &windows_cli_count, ignoreUnknown);
1652 const int posix_cli_status =
1653 freerdp_detect_posix_style_command_line_syntax(argc, argv, &posix_cli_count, ignoreUnknown);
1656 *flags = COMMAND_LINE_SEPARATOR_SPACE;
1657 *flags |= COMMAND_LINE_SIGIL_DASH | COMMAND_LINE_SIGIL_DOUBLE_DASH;
1658 *flags |= COMMAND_LINE_SIGIL_ENABLE_DISABLE;
1660 if (posix_cli_status <= COMMAND_LINE_STATUS_PRINT)
1664 if ((windows_cli_count && (windows_cli_count >= posix_cli_count)) ||
1665 (windows_cli_status <= COMMAND_LINE_STATUS_PRINT))
1667 windows_cli_count = 1;
1668 *flags = COMMAND_LINE_SEPARATOR_COLON;
1669 *flags |= COMMAND_LINE_SIGIL_SLASH | COMMAND_LINE_SIGIL_PLUS_MINUS;
1672 WLog_DBG(TAG,
"windows: %d/%" PRIuz
" posix: %d/%" PRIuz
"", windows_cli_status,
1673 windows_cli_count, posix_cli_status, posix_cli_count);
1674 if ((posix_cli_count == 0) && (windows_cli_count == 0))
1676 if ((posix_cli_status == COMMAND_LINE_ERROR) && (windows_cli_status == COMMAND_LINE_ERROR))
1682int freerdp_client_settings_command_line_status_print(rdpSettings* settings,
int status,
int argc,
1685 return freerdp_client_settings_command_line_status_print_ex(settings, status, argc, argv, NULL);
1688static void freerdp_client_print_keyboard_type_list(
const char* msg, DWORD type)
1692 layouts = freerdp_keyboard_get_layouts(type, &count);
1694 printf(
"\n%s\n", msg);
1696 for (
size_t x = 0; x < count; x++)
1699 printf(
"0x%08" PRIX32
"\t%s\n", layout->code, layout->name);
1702 freerdp_keyboard_layouts_free(layouts, count);
1705static void freerdp_client_print_keyboard_list(
void)
1707 freerdp_client_print_keyboard_type_list(
"Keyboard Layouts", RDP_KEYBOARD_LAYOUT_TYPE_STANDARD);
1708 freerdp_client_print_keyboard_type_list(
"Keyboard Layout Variants",
1709 RDP_KEYBOARD_LAYOUT_TYPE_VARIANT);
1710 freerdp_client_print_keyboard_type_list(
"Keyboard Layout Variants",
1711 RDP_KEYBOARD_LAYOUT_TYPE_IME);
1714static void freerdp_client_print_timezone_list(
void)
1718 while (EnumDynamicTimeZoneInformation(index++, &info) != ERROR_NO_MORE_ITEMS)
1720 char TimeZoneKeyName[ARRAYSIZE(info.TimeZoneKeyName) + 1] = { 0 };
1722 (void)ConvertWCharNToUtf8(info.TimeZoneKeyName, ARRAYSIZE(info.TimeZoneKeyName),
1723 TimeZoneKeyName, ARRAYSIZE(TimeZoneKeyName));
1724 printf(
"%" PRIu32
": '%s'\n", index, TimeZoneKeyName);
1728static void freerdp_client_print_tune_list(
const rdpSettings* settings)
1732 for (SSIZE_T x = 0; x < FreeRDP_Settings_StableAPI_MAX; x++)
1740 case RDP_SETTINGS_TYPE_BOOL:
1741 printf(
"%" PRIdz
"\t%50s\tBOOL\t%s\n", x, name,
1746 case RDP_SETTINGS_TYPE_UINT16:
1747 printf(
"%" PRIdz
"\t%50s\tUINT16\t%" PRIu16
"\n", x, name,
1750 case RDP_SETTINGS_TYPE_INT16:
1751 printf(
"%" PRIdz
"\t%50s\tINT16\t%" PRId16
"\n", x, name,
1754 case RDP_SETTINGS_TYPE_UINT32:
1755 printf(
"%" PRIdz
"\t%50s\tUINT32\t%" PRIu32
"\n", x, name,
1758 case RDP_SETTINGS_TYPE_INT32:
1759 printf(
"%" PRIdz
"\t%50s\tINT32\t%" PRId32
"\n", x, name,
1762 case RDP_SETTINGS_TYPE_UINT64:
1763 printf(
"%" PRIdz
"\t%50s\tUINT64\t%" PRIu64
"\n", x, name,
1766 case RDP_SETTINGS_TYPE_INT64:
1767 printf(
"%" PRIdz
"\t%50s\tINT64\t%" PRId64
"\n", x, name,
1770 case RDP_SETTINGS_TYPE_STRING:
1771 printf(
"%" PRIdz
"\t%50s\tSTRING\t%s"
1776 case RDP_SETTINGS_TYPE_POINTER:
1777 printf(
"%" PRIdz
"\t%50s\tPOINTER\t%p"
1789int freerdp_client_settings_command_line_status_print_ex(rdpSettings* settings,
int status,
1790 int argc,
char** argv,
1795 memcpy(largs, global_cmd_args,
sizeof(global_cmd_args));
1797 if (status == COMMAND_LINE_STATUS_PRINT_VERSION)
1799 freerdp_client_print_version();
1803 if (status == COMMAND_LINE_STATUS_PRINT_BUILDCONFIG)
1805 freerdp_client_print_version_ex(argc, argv);
1806 freerdp_client_print_buildconfig_ex(argc, argv);
1809 else if (status == COMMAND_LINE_STATUS_PRINT)
1811 (void)CommandLineParseArgumentsA(argc, argv, largs, 0x112, NULL, NULL, NULL);
1813 arg = CommandLineFindArgumentA(largs,
"list");
1816 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
1818 if (option_equals(
"timezones", arg->Value))
1819 freerdp_client_print_timezone_list();
1820 else if (option_equals(
"tune", arg->Value))
1821 freerdp_client_print_tune_list(settings);
1822 else if (option_equals(
"kbd", arg->Value))
1823 freerdp_client_print_keyboard_list();
1824 else if (option_starts_with(
"kbd-lang", arg->Value))
1826 const char* val = NULL;
1827 if (option_starts_with(
"kbd-lang:", arg->Value))
1828 val = &arg->Value[9];
1829 else if (!option_equals(
"kbd-lang", arg->Value))
1830 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1832 if (val && strchr(val,
','))
1833 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1834 freerdp_client_print_codepages(val);
1836 else if (option_equals(
"kbd-scancode", arg->Value))
1837 freerdp_client_print_scancodes();
1838 else if (option_equals(
"monitor", arg->Value))
1841 return COMMAND_LINE_ERROR;
1843 else if (option_starts_with(
"smartcard", arg->Value))
1846 if (option_starts_with(
"smartcard:", arg->Value))
1848 else if (!option_equals(
"smartcard", arg->Value))
1849 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1853 const char* sub = strchr(arg->Value,
':') + 1;
1854 const CmdLineSubOptions options[] = {
1855 {
"pkinit-anchors:", FreeRDP_PkinitAnchors, CMDLINE_SUBOPTION_STRING,
1857 {
"pkcs11-module:", FreeRDP_Pkcs11Module, CMDLINE_SUBOPTION_STRING, NULL }
1862 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"smartcard", sub, &count);
1864 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1867 CommandLineParserFree(ptr);
1868 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1871 for (
size_t x = 1; x < count; x++)
1873 const char* cur = ptr[x];
1874 if (!parseSubOptions(settings, options, ARRAYSIZE(options), cur))
1876 CommandLineParserFree(ptr);
1877 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1881 CommandLineParserFree(ptr);
1884 freerdp_smartcard_list(settings);
1888 freerdp_client_print_command_line_help_ex(argc, argv, custom);
1889 return COMMAND_LINE_ERROR;
1892#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
1893 arg = CommandLineFindArgumentA(largs,
"tune-list");
1896 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
1898 WLog_WARN(TAG,
"Option /tune-list is deprecated, use /list:tune instead");
1899 freerdp_client_print_tune_list(settings);
1902 arg = CommandLineFindArgumentA(largs,
"kbd-lang-list");
1905 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
1907 WLog_WARN(TAG,
"Option /kbd-lang-list is deprecated, use /list:kbd-lang instead");
1908 freerdp_client_print_codepages(arg->Value);
1911 arg = CommandLineFindArgumentA(largs,
"kbd-list");
1914 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1916 WLog_WARN(TAG,
"Option /kbd-list is deprecated, use /list:kbd instead");
1917 freerdp_client_print_keyboard_list();
1920 arg = CommandLineFindArgumentA(largs,
"monitor-list");
1923 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1925 WLog_WARN(TAG,
"Option /monitor-list is deprecated, use /list:monitor instead");
1927 return COMMAND_LINE_ERROR;
1930 arg = CommandLineFindArgumentA(largs,
"smartcard-list");
1933 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1935 WLog_WARN(TAG,
"Option /smartcard-list is deprecated, use /list:smartcard instead");
1936 freerdp_smartcard_list(settings);
1939 arg = CommandLineFindArgumentA(largs,
"kbd-scancode-list");
1942 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1945 "Option /kbd-scancode-list is deprecated, use /list:kbd-scancode instead");
1946 freerdp_client_print_scancodes();
1952 else if (status < 0)
1954 freerdp_client_print_command_line_help_ex(argc, argv, custom);
1959 if (status <= COMMAND_LINE_STATUS_PRINT && status >= COMMAND_LINE_STATUS_PRINT_LAST)
1972static BOOL parseSizeValue(
const char* input,
unsigned long* v1,
unsigned long* v2)
1974 const char* xcharpos = NULL;
1975 char* endPtr = NULL;
1976 unsigned long v = 0;
1978 v = strtoul(input, &endPtr, 10);
1980 if ((v == 0 || v == ULONG_MAX) && (errno != 0))
1986 xcharpos = strchr(input,
'x');
1988 if (!xcharpos || xcharpos != endPtr)
1992 v = strtoul(xcharpos + 1, &endPtr, 10);
1994 if ((v == 0 || v == ULONG_MAX) && (errno != 0))
1997 if (*endPtr !=
'\0')
2009 const char* arguments[] = {
"network",
"gfx",
"rfx",
"bpp" };
2010 WINPR_ASSERT(settings);
2016 for (
size_t x = 0; x < ARRAYSIZE(arguments); x++)
2018 const char* arg = arguments[x];
2020 if (p && (p->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
2024 return freerdp_set_connection_type(settings, CONNECTION_TYPE_AUTODETECT);
2027static BOOL setSmartcardEmulation(WINPR_ATTR_UNUSED
const char* value, rdpSettings* settings)
2032const char* option_starts_with(
const char* what,
const char* val)
2036 const size_t wlen = strlen(what);
2038 if (_strnicmp(what, val, wlen) != 0)
2043BOOL option_ends_with(
const char* str,
const char* ext)
2047 const size_t strLen = strlen(str);
2048 const size_t extLen = strlen(ext);
2050 if (strLen < extLen)
2053 return _strnicmp(&str[strLen - extLen], ext, extLen) == 0;
2056BOOL option_equals(
const char* what,
const char* val)
2060 return _stricmp(what, val) == 0;
2069} PARSE_ON_OFF_RESULT;
2071static PARSE_ON_OFF_RESULT parse_on_off_option(
const char* value)
2073 WINPR_ASSERT(value);
2074 const char* sep = strchr(value,
':');
2077 if (option_equals(
"on", &sep[1]))
2079 if (option_equals(
"off", &sep[1]))
2088 CLIP_DIR_PARSE_LOCAL,
2089 CLIP_DIR_PARSE_REMOTE,
2091} PARSE_CLIP_DIR_RESULT;
2093static PARSE_CLIP_DIR_RESULT parse_clip_direciton_to_option(
const char* value)
2095 WINPR_ASSERT(value);
2096 const char* sep = strchr(value,
':');
2098 return CLIP_DIR_PARSE_FAIL;
2099 if (option_equals(
"all", &sep[1]))
2100 return CLIP_DIR_PARSE_ALL;
2101 if (option_equals(
"off", &sep[1]))
2102 return CLIP_DIR_PARSE_OFF;
2103 if (option_equals(
"local", &sep[1]))
2104 return CLIP_DIR_PARSE_LOCAL;
2105 if (option_equals(
"remote", &sep[1]))
2106 return CLIP_DIR_PARSE_REMOTE;
2107 return CLIP_DIR_PARSE_FAIL;
2110static int parse_tls_ciphers(rdpSettings* settings,
const char* Value)
2112 const char* ciphers = NULL;
2114 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2116 if (option_equals(Value,
"netmon"))
2118 ciphers =
"ALL:!ECDH:!ADH:!DHE";
2120 else if (option_equals(Value,
"ma"))
2122 ciphers =
"AES128-SHA";
2130 return COMMAND_LINE_ERROR_MEMORY;
2134static int parse_tls_seclevel(rdpSettings* settings,
const char* Value)
2138 if (!value_to_int(Value, &val, 0, 5))
2139 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2142 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2146static int parse_tls_secrets_file(rdpSettings* settings,
const char* Value)
2149 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2152 return COMMAND_LINE_ERROR_MEMORY;
2156static int parse_tls_enforce(rdpSettings* settings,
const char* Value)
2158 UINT16 version = TLS1_2_VERSION;
2167 const struct map_t map[] = { {
"1.0", TLS1_VERSION },
2168 {
"1.1", TLS1_1_VERSION },
2169 {
"1.2", TLS1_2_VERSION }
2170#if defined(TLS1_3_VERSION)
2172 {
"1.3", TLS1_3_VERSION }
2176 const struct map_t* found = NULL;
2177 for (
size_t x = 0; x < ARRAYSIZE(map); x++)
2179 const struct map_t* cur = &map[x];
2180 if (option_equals(cur->name, Value))
2188 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2189 version = found->version;
2194 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2200 int rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2201 CommandLineSwitchStart(arg) CommandLineSwitchCase(arg,
"tls")
2203 if (option_starts_with(
"ciphers:", arg->Value))
2204 rc = fail_at(arg, parse_tls_ciphers(settings, &arg->Value[8]));
2205 else if (option_starts_with(
"seclevel:", arg->Value))
2206 rc = fail_at(arg, parse_tls_seclevel(settings, &arg->Value[9]));
2207 else if (option_starts_with(
"secrets-file:", arg->Value))
2208 rc = fail_at(arg, parse_tls_secrets_file(settings, &arg->Value[13]));
2209 else if (option_starts_with(
"enforce:", arg->Value))
2210 rc = fail_at(arg, parse_tls_enforce(settings, &arg->Value[8]));
2213#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
2214 CommandLineSwitchCase(arg,
"tls-ciphers")
2216 WLog_WARN(TAG,
"Option /tls-ciphers is deprecated, use /tls:ciphers instead");
2217 rc = fail_at(arg, parse_tls_ciphers(settings, arg->Value));
2219 CommandLineSwitchCase(arg,
"tls-seclevel")
2221 WLog_WARN(TAG,
"Option /tls-seclevel is deprecated, use /tls:seclevel instead");
2222 rc = fail_at(arg, parse_tls_seclevel(settings, arg->Value));
2224 CommandLineSwitchCase(arg,
"tls-secrets-file")
2226 WLog_WARN(TAG,
"Option /tls-secrets-file is deprecated, use /tls:secrets-file instead");
2227 rc = fail_at(arg, parse_tls_secrets_file(settings, arg->Value));
2229 CommandLineSwitchCase(arg,
"enforce-tlsv1_2")
2231 WLog_WARN(TAG,
"Option /enforce-tlsv1_2 is deprecated, use /tls:enforce:1.2 instead");
2232 rc = fail_at(arg, parse_tls_enforce(settings,
"1.2"));
2235 CommandLineSwitchDefault(arg)
2238 CommandLineSwitchEnd(arg)
2245 WINPR_ASSERT(settings);
2249 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2250 for (
size_t x = 0; x < count; x++)
2253 larg.Value = ptr[x];
2255 int rc = parse_tls_cipher_options(settings, &larg);
2258 CommandLineParserFree(ptr);
2262 CommandLineParserFree(ptr);
2268 WINPR_ASSERT(settings);
2272 return COMMAND_LINE_ERROR;
2276 int rc = CHANNEL_RC_OK;
2278 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2279 if (!ptr || (count == 0))
2280 rc = COMMAND_LINE_ERROR;
2283 BOOL GfxH264 = FALSE;
2284 BOOL GfxAVC444 = FALSE;
2285 BOOL RemoteFxCodec = FALSE;
2286 BOOL GfxProgressive = FALSE;
2287 BOOL codecSelected = FALSE;
2289 for (
size_t x = 0; x < count; x++)
2291 const char* val = ptr[x];
2293 if (option_starts_with(
"AVC444", val))
2295 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2296 if (bval == PARSE_FAIL)
2297 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2299 GfxAVC444 = bval != PARSE_OFF;
2300 codecSelected = TRUE;
2302 else if (option_starts_with(
"AVC420", val))
2304 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2305 if (bval == PARSE_FAIL)
2306 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2308 GfxH264 = bval != PARSE_OFF;
2309 codecSelected = TRUE;
2313 if (option_starts_with(
"RFX", val))
2315 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2316 if (bval == PARSE_FAIL)
2317 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2319 RemoteFxCodec = bval != PARSE_OFF;
2320 codecSelected = TRUE;
2322 else if (option_starts_with(
"progressive", val))
2324 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2325 if (bval == PARSE_FAIL)
2326 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2328 GfxProgressive = bval != PARSE_OFF;
2329 codecSelected = TRUE;
2331 else if (option_starts_with(
"mask:", val))
2334 const char* uv = &val[5];
2335 if (!value_to_uint(uv, &v, 0, UINT32_MAX))
2336 rc = COMMAND_LINE_ERROR;
2341 rc = COMMAND_LINE_ERROR;
2344 else if (option_starts_with(
"small-cache", val))
2346 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2347 if (bval == PARSE_FAIL)
2348 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2351 rc = COMMAND_LINE_ERROR;
2353 else if (option_starts_with(
"thin-client", val))
2355 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2356 if (bval == PARSE_FAIL)
2357 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2360 rc = COMMAND_LINE_ERROR;
2361 if ((rc == CHANNEL_RC_OK) && (bval > 0))
2365 rc = COMMAND_LINE_ERROR;
2368 else if (option_starts_with(
"frame-ack", val))
2370 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2371 if (bval == PARSE_FAIL)
2372 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2375 rc = COMMAND_LINE_ERROR;
2378 rc = COMMAND_LINE_ERROR;
2381 if ((rc == CHANNEL_RC_OK) && codecSelected)
2384 rc = COMMAND_LINE_ERROR;
2386 rc = COMMAND_LINE_ERROR;
2388 rc = COMMAND_LINE_ERROR;
2390 rc = COMMAND_LINE_ERROR;
2392 rc = COMMAND_LINE_ERROR;
2395 CommandLineParserFree(ptr);
2396 if (rc != CHANNEL_RC_OK)
2399 return CHANNEL_RC_OK;
2402static int parse_kbd_layout(rdpSettings* settings,
const char* value)
2404 WINPR_ASSERT(settings);
2405 WINPR_ASSERT(value);
2409 const BOOL isInt = value_to_int(value, &ival, 1, UINT32_MAX);
2412 ival = freerdp_map_keyboard_layout_name_to_id(value);
2416 WLog_ERR(TAG,
"Could not identify keyboard layout: %s", value);
2417 WLog_ERR(TAG,
"Use /list:kbd to list available layouts");
2418 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2425 rc = COMMAND_LINE_ERROR;
2430#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
2433 WINPR_ASSERT(settings);
2437 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2439 return COMMAND_LINE_ERROR;
2441 if (option_equals(arg->Value,
"rfx"))
2444 return COMMAND_LINE_ERROR;
2446 else if (option_equals(arg->Value,
"nsc"))
2449 return COMMAND_LINE_ERROR;
2452#if defined(WITH_JPEG)
2453 else if (option_equals(arg->Value,
"jpeg"))
2456 return COMMAND_LINE_ERROR;
2461 return COMMAND_LINE_ERROR;
2470static BOOL check_kbd_remap_valid(
const char* token)
2475 WINPR_ASSERT(token);
2477 if (strlen(token) > 10)
2480 if (!freerdp_extract_key_value(token, &key, &value))
2482 WLog_WARN(TAG,
"/kbd:remap invalid entry '%s'", token);
2490 WINPR_ASSERT(settings);
2494 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2496 return COMMAND_LINE_ERROR_MEMORY;
2497 char* p = strchr(arg->Value,
'[');
2502 const char scheme[] =
"://";
2503 const char* val = strstr(arg->Value, scheme);
2505 val += strnlen(scheme,
sizeof(scheme));
2508 p = strchr(val,
':');
2515 if (!value_to_int(&p[1], &lval, 1, UINT16_MAX))
2516 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2518 length = (size_t)(p - arg->Value);
2520 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2523 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2528 return COMMAND_LINE_ERROR_MEMORY;
2534 char* p2 = strchr(arg->Value,
']');
2538 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2540 length = (size_t)(p2 - p);
2542 return COMMAND_LINE_ERROR_MEMORY;
2544 if (*(p2 + 1) ==
':')
2548 if (!value_to_int(&p2[2], &val, 0, UINT16_MAX))
2549 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2552 return COMMAND_LINE_ERROR;
2555 printf(
"hostname %s port %" PRIu32
"\n",
2564 WINPR_ASSERT(settings);
2568 char* cur = arg->Value;
2570 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2572 return COMMAND_LINE_ERROR;
2578 char* next = strchr(cur,
',');
2586 if (option_equals(
"fqdn", cur))
2588 else if (option_equals(
"ip", cur))
2590 else if (option_equals(
"netbios", cur))
2593 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2596 mask = (mask & 0x07);
2597 value |= mask << (count * 3);
2599 }
while (cur != NULL);
2602 return COMMAND_LINE_ERROR;
2605 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2609static int parse_prevent_session_lock_options(rdpSettings* settings,
2612 WINPR_ASSERT(settings);
2616 return COMMAND_LINE_ERROR_MEMORY;
2618 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2622 if (!value_to_int(arg->Value, &val, 1, UINT32_MAX))
2623 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2626 return COMMAND_LINE_ERROR_MEMORY;
2634 WINPR_ASSERT(settings);
2638 return COMMAND_LINE_ERROR;
2645 return COMMAND_LINE_ERROR;
2647 return COMMAND_LINE_ERROR;
2649 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2652 return COMMAND_LINE_ERROR;
2655 return COMMAND_LINE_ERROR_MEMORY;
2663 WINPR_ASSERT(settings);
2667 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2668 char* p = strchr(arg->Value,
'x');
2672 unsigned long w = 0;
2673 unsigned long h = 0;
2675 if (!parseSizeValue(arg->Value, &w, &h) || (w > UINT16_MAX) || (h > UINT16_MAX))
2676 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2679 return COMMAND_LINE_ERROR;
2681 return COMMAND_LINE_ERROR;
2685 char* str = _strdup(arg->Value);
2687 return COMMAND_LINE_ERROR_MEMORY;
2689 p = strchr(str,
'%');
2693 BOOL partial = FALSE;
2695 status = COMMAND_LINE_ERROR;
2722 if (!value_to_int(str, &val, 0, 100))
2724 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2744 WINPR_ASSERT(settings);
2747 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2750 UINT32* MonitorIds = NULL;
2751 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2754 return COMMAND_LINE_ERROR_MEMORY;
2761 CommandLineParserFree(ptr);
2765 MonitorIds = freerdp_settings_get_pointer_array_writable(settings, FreeRDP_MonitorIds, 0);
2766 for (UINT32 i = 0; i < count; i++)
2770 if (!value_to_int(ptr[i], &val, 0, UINT16_MAX))
2771 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2773 MonitorIds[i] = (UINT32)val;
2776 CommandLineParserFree(ptr);
2782static int parse_dynamic_resolution_options(rdpSettings* settings,
2785 WINPR_ASSERT(settings);
2788 const BOOL val = arg->Value != 0;
2792 WLog_ERR(TAG,
"Smart sizing and dynamic resolution are mutually exclusive options");
2793 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2797 return COMMAND_LINE_ERROR;
2799 return COMMAND_LINE_ERROR;
2806 WINPR_ASSERT(settings);
2811 WLog_ERR(TAG,
"Smart sizing and dynamic resolution are mutually exclusive options");
2812 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2816 return COMMAND_LINE_ERROR;
2820 unsigned long w = 0;
2821 unsigned long h = 0;
2823 if (!parseSizeValue(arg->Value, &w, &h) || (w > UINT16_MAX) || (h > UINT16_MAX))
2824 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2827 return COMMAND_LINE_ERROR;
2829 return COMMAND_LINE_ERROR;
2836 WINPR_ASSERT(settings);
2841 if (!value_to_int(arg->Value, &val, 0, UINT32_MAX))
2842 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2852 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2856 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2863 WINPR_ASSERT(settings);
2866 int rc = CHANNEL_RC_OK;
2868 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2869 if (!ptr || (count == 0))
2870 rc = COMMAND_LINE_ERROR;
2873 for (
size_t x = 0; x < count; x++)
2875 const char* val = ptr[x];
2877 if (option_starts_with(
"remap:", val))
2880 char* now = _strdup(&val[6]);
2885 if (!check_kbd_remap_valid(now))
2886 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2889 const size_t olen = strlen(old);
2890 const size_t alen = strlen(now);
2891 const size_t tlen = olen + alen + 2;
2892 char* tmp = calloc(tlen,
sizeof(
char));
2894 rc = COMMAND_LINE_ERROR_MEMORY;
2896 (
void)_snprintf(tmp, tlen,
"%s,%s", old, now);
2904 rc = COMMAND_LINE_ERROR;
2908 else if (option_starts_with(
"layout:", val))
2910 rc = parse_kbd_layout(settings, &val[7]);
2912 else if (option_starts_with(
"lang:", val))
2915 const BOOL isInt = value_to_int(&val[5], &ival, 1, UINT32_MAX);
2917 ival = freerdp_get_locale_id_from_string(&val[5]);
2920 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2923 rc = COMMAND_LINE_ERROR;
2925 else if (option_starts_with(
"type:", val))
2928 const BOOL isInt = value_to_int(&val[5], &ival, 1, UINT32_MAX);
2930 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2932 rc = COMMAND_LINE_ERROR;
2934 else if (option_starts_with(
"subtype:", val))
2937 const BOOL isInt = value_to_int(&val[8], &ival, 1, UINT32_MAX);
2939 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2942 rc = COMMAND_LINE_ERROR;
2944 else if (option_starts_with(
"fn-key:", val))
2947 const BOOL isInt = value_to_int(&val[7], &ival, 1, UINT32_MAX);
2949 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2952 rc = COMMAND_LINE_ERROR;
2954 else if (option_starts_with(
"unicode", val))
2956 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2957 if (bval == PARSE_FAIL)
2958 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2961 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2963 else if (option_starts_with(
"pipe:", val))
2966 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2968 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2970#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
2971 else if (count == 1)
2974 rc = parse_kbd_layout(settings, val);
2978 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2984 CommandLineParserFree(ptr);
2990 WINPR_ASSERT(settings);
2995 return COMMAND_LINE_ERROR_MEMORY;
2997 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2999 const char* cur = arg->Value;
3002 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3004 if (!proxy_parse_uri(settings, cur))
3005 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3009 WLog_ERR(TAG,
"Option http-proxy needs argument.");
3010 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3017 WINPR_ASSERT(settings);
3020 BOOL failed = FALSE;
3022 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3027 BOOL modernsyntax = FALSE;
3028 BOOL oldsyntax = FALSE;
3029 for (
size_t x = 0; (x < count) && !failed; x++)
3031 const char* carg = ptr[x];
3032 if (option_starts_with(
"file:", carg))
3034 const char* val = &carg[5];
3039 modernsyntax = TRUE;
3041 else if (option_equals(
"replay", carg))
3048 else if (option_equals(
"record", carg))
3055 else if (option_equals(
"nodelay", carg))
3062 modernsyntax = TRUE;
3077 if (oldsyntax && (count != 2))
3080 CommandLineParserFree(ptr);
3082 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3088 WINPR_ASSERT(settings);
3091 if (arg->Value == BoolValueTrue || arg->Value == BoolValueFalse)
3094 (arg->Value == BoolValueTrue)))
3095 return COMMAND_LINE_ERROR;
3101 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3102 for (
size_t x = 0; (x < count) && (rc == 0); x++)
3104 const char* usesel =
"use-selection:";
3106 const char* cur = ptr[x];
3107 if (option_starts_with(usesel, cur))
3109 const char* val = &cur[strlen(usesel)];
3111 rc = COMMAND_LINE_ERROR_MEMORY;
3113 return COMMAND_LINE_ERROR;
3115 else if (option_starts_with(
"direction-to", cur))
3119 (uint32_t)~(CLIPRDR_FLAG_LOCAL_TO_REMOTE | CLIPRDR_FLAG_REMOTE_TO_LOCAL);
3120 const PARSE_CLIP_DIR_RESULT bval = parse_clip_direciton_to_option(cur);
3124 case CLIP_DIR_PARSE_ALL:
3125 bflags |= CLIPRDR_FLAG_LOCAL_TO_REMOTE | CLIPRDR_FLAG_REMOTE_TO_LOCAL;
3127 case CLIP_DIR_PARSE_LOCAL:
3128 bflags |= CLIPRDR_FLAG_REMOTE_TO_LOCAL;
3130 case CLIP_DIR_PARSE_REMOTE:
3131 bflags |= CLIPRDR_FLAG_LOCAL_TO_REMOTE;
3133 case CLIP_DIR_PARSE_OFF:
3135 case CLIP_DIR_PARSE_FAIL:
3137 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3143 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3145 else if (option_starts_with(
"files-to", cur))
3149 (uint32_t)~(CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES |
3150 CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES);
3151 const PARSE_CLIP_DIR_RESULT bval = parse_clip_direciton_to_option(cur);
3155 case CLIP_DIR_PARSE_ALL:
3157 CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES | CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES;
3159 case CLIP_DIR_PARSE_LOCAL:
3160 bflags |= CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES;
3162 case CLIP_DIR_PARSE_REMOTE:
3163 bflags |= CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES;
3165 case CLIP_DIR_PARSE_OFF:
3167 case CLIP_DIR_PARSE_FAIL:
3169 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3175 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3178 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3180 CommandLineParserFree(ptr);
3190 WINPR_ASSERT(settings);
3195 if (!value_to_int(arg->Value, &val, 0, UINT32_MAX))
3196 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3200 case AUDIO_MODE_REDIRECT:
3202 return COMMAND_LINE_ERROR;
3205 case AUDIO_MODE_PLAY_ON_SERVER:
3207 return COMMAND_LINE_ERROR;
3210 case AUDIO_MODE_NONE:
3212 return COMMAND_LINE_ERROR;
3214 return COMMAND_LINE_ERROR;
3218 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3225 WINPR_ASSERT(settings);
3228 UINT32 type = CONNECTION_TYPE_INVALID;
3230 if (option_equals(arg->Value,
"invalid"))
3231 type = CONNECTION_TYPE_INVALID;
3232 else if (option_equals(arg->Value,
"modem"))
3233 type = CONNECTION_TYPE_MODEM;
3234 else if (option_equals(arg->Value,
"broadband"))
3235 type = CONNECTION_TYPE_BROADBAND_HIGH;
3236 else if (option_equals(arg->Value,
"broadband-low"))
3237 type = CONNECTION_TYPE_BROADBAND_LOW;
3238 else if (option_equals(arg->Value,
"broadband-high"))
3239 type = CONNECTION_TYPE_BROADBAND_HIGH;
3240 else if (option_equals(arg->Value,
"wan"))
3241 type = CONNECTION_TYPE_WAN;
3242 else if (option_equals(arg->Value,
"lan"))
3243 type = CONNECTION_TYPE_LAN;
3244 else if ((option_equals(arg->Value,
"autodetect")) || (option_equals(arg->Value,
"auto")) ||
3245 (option_equals(arg->Value,
"detect")))
3247 type = CONNECTION_TYPE_AUTODETECT;
3253 if (!value_to_int(arg->Value, &val, 0, 7))
3254 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3259 if (!freerdp_set_connection_type(settings, type))
3260 return COMMAND_LINE_ERROR;
3266 WINPR_ASSERT(settings);
3270 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3272 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3274 FreeRDP_Settings_Keys_Bool singleOptionWithoutOnOff = FreeRDP_BOOL_UNUSED;
3275 for (
size_t x = 0; x < count; x++)
3277 const char* cur = ptr[x];
3278 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(cur);
3279 if (bval == PARSE_FAIL)
3281 CommandLineParserFree(ptr);
3282 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3285 const BOOL val = bval != PARSE_OFF;
3286 FreeRDP_Settings_Keys_Bool
id = FreeRDP_BOOL_UNUSED;
3287 if (option_starts_with(
"rdp", cur))
3289 id = FreeRDP_RdpSecurity;
3291 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3293 else if (option_starts_with(
"tls", cur))
3294 id = FreeRDP_TlsSecurity;
3295 else if (option_starts_with(
"nla", cur))
3296 id = FreeRDP_NlaSecurity;
3297 else if (option_starts_with(
"ext", cur))
3298 id = FreeRDP_ExtSecurity;
3299 else if (option_equals(
"aad", cur))
3300 id = FreeRDP_AadSecurity;
3303 WLog_ERR(TAG,
"unknown protocol security: %s", arg->Value);
3304 CommandLineParserFree(ptr);
3305 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3308 if ((bval == PARSE_NONE) && (count == 1))
3309 singleOptionWithoutOnOff = id;
3311 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3314 if (singleOptionWithoutOnOff != FreeRDP_BOOL_UNUSED)
3316 const FreeRDP_Settings_Keys_Bool options[] = { FreeRDP_AadSecurity,
3317 FreeRDP_UseRdpSecurityLayer,
3318 FreeRDP_RdpSecurity, FreeRDP_NlaSecurity,
3319 FreeRDP_TlsSecurity };
3321 for (
size_t i = 0; i < ARRAYSIZE(options); i++)
3324 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3328 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3329 if (singleOptionWithoutOnOff == FreeRDP_RdpSecurity)
3332 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3335 CommandLineParserFree(ptr);
3339static int parse_encryption_methods_options(rdpSettings* settings,
3342 WINPR_ASSERT(settings);
3345 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
3348 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3350 UINT32 EncryptionMethods = 0;
3351 for (UINT32 i = 0; i < count; i++)
3353 if (option_equals(ptr[i],
"40"))
3354 EncryptionMethods |= ENCRYPTION_METHOD_40BIT;
3355 else if (option_equals(ptr[i],
"56"))
3356 EncryptionMethods |= ENCRYPTION_METHOD_56BIT;
3357 else if (option_equals(ptr[i],
"128"))
3358 EncryptionMethods |= ENCRYPTION_METHOD_128BIT;
3359 else if (option_equals(ptr[i],
"FIPS"))
3360 EncryptionMethods |= ENCRYPTION_METHOD_FIPS;
3362 WLog_ERR(TAG,
"unknown encryption method '%s'", ptr[i]);
3366 return COMMAND_LINE_ERROR;
3367 CommandLineParserFree(ptr);
3374 WINPR_ASSERT(settings);
3379 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3380 for (
size_t x = 0; (x < count) && (rc == 0); x++)
3382 const char deny[] =
"deny";
3383 const char ignore[] =
"ignore";
3384 const char tofu[] =
"tofu";
3385 const char name[] =
"name:";
3386 const char fingerprints[] =
"fingerprint:";
3388 const char* cur = ptr[x];
3389 if (option_equals(deny, cur))
3392 return COMMAND_LINE_ERROR;
3394 else if (option_equals(ignore, cur))
3397 return COMMAND_LINE_ERROR;
3399 else if (option_equals(tofu, cur))
3402 return COMMAND_LINE_ERROR;
3404 else if (option_starts_with(name, cur))
3406 const char* val = &cur[strnlen(name,
sizeof(name))];
3408 rc = COMMAND_LINE_ERROR_MEMORY;
3410 else if (option_starts_with(fingerprints, cur))
3412 const char* val = &cur[strnlen(fingerprints,
sizeof(fingerprints))];
3415 rc = COMMAND_LINE_ERROR_MEMORY;
3418 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3420 CommandLineParserFree(ptr);
3427 WINPR_ASSERT(settings);
3431 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"mouse", arg->Value, &count);
3435 for (
size_t x = 1; x < count; x++)
3437 const char* cur = ptr[x];
3439 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(cur);
3440 if (bval == PARSE_FAIL)
3441 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3444 const BOOL val = bval != PARSE_OFF;
3446 if (option_starts_with(
"relative", cur))
3449 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3451 else if (option_starts_with(
"grab", cur))
3454 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3462 CommandLineParserFree(ptr);
3469 WINPR_ASSERT(settings);
3473 UINT32 Floatbar = 0x0017;
3477 char* start = arg->Value;
3482 start = strchr(start,
',');
3491 if (option_starts_with(
"sticky:", cur))
3495 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(cur);
3507 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3511 else if (option_starts_with(
"default:", cur))
3513 const char* val = cur + 8;
3516 if (option_equals(
"visible", val))
3518 else if (option_equals(
"hidden", val))
3521 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3524 else if (option_starts_with(
"show:", cur))
3526 const char* val = cur + 5;
3529 if (option_equals(
"always", val))
3531 else if (option_equals(
"fullscreen", val))
3533 else if (option_equals(
"window", val))
3536 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3539 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3543 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3549 WINPR_ASSERT(settings);
3552 BYTE* base64 = NULL;
3555 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3557 crypto_base64_decode((
const char*)(arg->Value), strlen(arg->Value), &base64, &length);
3563 return COMMAND_LINE_ERROR;
3567 WLog_ERR(TAG,
"reconnect-cookie: invalid base64 '%s'", arg->Value);
3574static BOOL set_monitor_override(rdpSettings* settings, uint64_t flag)
3576 const FreeRDP_Settings_Keys_UInt64 key = FreeRDP_MonitorOverrideFlags;
3584 WINPR_ASSERT(settings);
3589 if (!value_to_int(arg->Value, &val, 100, 180))
3590 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3598 return COMMAND_LINE_ERROR;
3600 return COMMAND_LINE_ERROR;
3601 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_DESKTOP_SCALE |
3602 FREERDP_MONITOR_OVERRIDE_DEVICE_SCALE))
3603 return fail_at(arg, COMMAND_LINE_ERROR);
3607 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3614 WINPR_ASSERT(settings);
3619 if (!value_to_int(arg->Value, &val, 100, 180))
3620 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3628 return COMMAND_LINE_ERROR;
3629 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_DEVICE_SCALE))
3630 return fail_at(arg, COMMAND_LINE_ERROR);
3634 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3641 WINPR_ASSERT(settings);
3647 return COMMAND_LINE_ERROR;
3649 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"smartcard-logon", arg->Value, &count);
3652 const CmdLineSubOptions opts[] = {
3653 {
"cert:", FreeRDP_SmartcardCertificate, CMDLINE_SUBOPTION_FILE,
3654 setSmartcardEmulation },
3655 {
"key:", FreeRDP_SmartcardPrivateKey, CMDLINE_SUBOPTION_FILE, setSmartcardEmulation },
3656 {
"pin:", FreeRDP_Password, CMDLINE_SUBOPTION_STRING, NULL },
3657 {
"csp:", FreeRDP_CspName, CMDLINE_SUBOPTION_STRING, NULL },
3658 {
"reader:", FreeRDP_ReaderName, CMDLINE_SUBOPTION_STRING, NULL },
3659 {
"card:", FreeRDP_CardName, CMDLINE_SUBOPTION_STRING, NULL },
3660 {
"container:", FreeRDP_ContainerName, CMDLINE_SUBOPTION_STRING, NULL }
3663 for (
size_t x = 1; x < count; x++)
3665 const char* cur = ptr[x];
3666 if (!parseSubOptions(settings, opts, ARRAYSIZE(opts), cur))
3668 CommandLineParserFree(ptr);
3669 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3673 CommandLineParserFree(ptr);
3679 WINPR_ASSERT(settings);
3683 char** ptr = CommandLineParseCommaSeparatedValuesEx(
"tune", arg->Value, &count);
3685 return COMMAND_LINE_ERROR;
3686 for (
size_t x = 1; x < count; x++)
3688 const char* cur = ptr[x];
3689 char* sep = strchr(cur,
':');
3692 CommandLineParserFree(ptr);
3693 return COMMAND_LINE_ERROR;
3696 if (!freerdp_settings_set_value_for_name(settings, cur, sep))
3698 CommandLineParserFree(ptr);
3699 return COMMAND_LINE_ERROR;
3703 CommandLineParserFree(ptr);
3707static int parse_app_option_program(rdpSettings* settings,
const char* cmd)
3709 const FreeRDP_Settings_Keys_Bool ids[] = { FreeRDP_RemoteApplicationMode,
3710 FreeRDP_RemoteAppLanguageBarSupported,
3711 FreeRDP_Workarea, FreeRDP_DisableWallpaper,
3712 FreeRDP_DisableFullWindowDrag };
3715 return COMMAND_LINE_ERROR_MEMORY;
3717 for (
size_t y = 0; y < ARRAYSIZE(ids); y++)
3720 return COMMAND_LINE_ERROR;
3722 return CHANNEL_RC_OK;
3727 WINPR_ASSERT(settings);
3730 int rc = CHANNEL_RC_OK;
3732 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3733 if (!ptr || (count == 0))
3734 rc = COMMAND_LINE_ERROR;
3741 int (*fkt)(rdpSettings* settings,
const char* value);
3743 const struct app_map amap[] = { {
"tenantid:", FreeRDP_GatewayAvdAadtenantid, NULL },
3744 {
"ad:", FreeRDP_GatewayAzureActiveDirectory, NULL },
3745 {
"avd-access:", FreeRDP_GatewayAvdAccessAadFormat, NULL },
3746 {
"avd-token:", FreeRDP_GatewayAvdAccessTokenFormat, NULL },
3747 {
"avd-scope:", FreeRDP_GatewayAvdScope, NULL }
3750 for (
size_t x = 0; x < count; x++)
3752 BOOL handled = FALSE;
3753 const char* val = ptr[x];
3755 if (option_starts_with(
"use-tenantid", val))
3757 PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
3758 if (bval == PARSE_FAIL)
3760 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3768 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3774 for (
size_t y = 0; y < ARRAYSIZE(amap); y++)
3776 const struct app_map* cur = &amap[y];
3777 if (option_starts_with(cur->name, val))
3779 const char* xval = &val[strlen(cur->name)];
3781 rc = cur->fkt(settings, xval);
3785 if (!freerdp_settings_set_value_for_name(settings, name, xval))
3786 rc = COMMAND_LINE_ERROR_MEMORY;
3795 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3802 CommandLineParserFree(ptr);
3808 WINPR_ASSERT(settings);
3811 int rc = CHANNEL_RC_OK;
3813 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3814 if (!ptr || (count == 0))
3815 rc = COMMAND_LINE_ERROR;
3822 int (*fkt)(rdpSettings* settings,
const char* value);
3824 const struct app_map amap[] = { {
"program:", FreeRDP_RemoteApplicationProgram,
3825 parse_app_option_program },
3826 {
"workdir:", FreeRDP_RemoteApplicationWorkingDir, NULL },
3827 {
"name:", FreeRDP_RemoteApplicationName, NULL },
3828 {
"icon:", FreeRDP_RemoteApplicationIcon, NULL },
3829 {
"cmd:", FreeRDP_RemoteApplicationCmdLine, NULL },
3830 {
"file:", FreeRDP_RemoteApplicationFile, NULL },
3831 {
"guid:", FreeRDP_RemoteApplicationGuid, NULL },
3832 {
"hidef:", FreeRDP_HiDefRemoteApp, NULL } };
3833 for (
size_t x = 0; x < count; x++)
3835 BOOL handled = FALSE;
3836 const char* val = ptr[x];
3838 for (
size_t y = 0; y < ARRAYSIZE(amap); y++)
3840 const struct app_map* cur = &amap[y];
3841 if (option_starts_with(cur->name, val))
3843 const char* xval = &val[strlen(cur->name)];
3845 rc = cur->fkt(settings, xval);
3849 if (!freerdp_settings_set_value_for_name(settings, name, xval))
3850 rc = COMMAND_LINE_ERROR_MEMORY;
3858#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
3859 if (!handled && (count == 1))
3862 rc = parse_app_option_program(settings, val);
3867 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3874 CommandLineParserFree(ptr);
3880 WINPR_ASSERT(settings);
3883 int rc = CHANNEL_RC_OK;
3885 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3886 if (!ptr || (count == 0))
3887 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3889 for (
size_t x = 0; x < count; x++)
3891 const char* val = ptr[x];
3893 if (option_starts_with(
"codec:", val))
3896 rc = COMMAND_LINE_ERROR;
3897 else if (option_equals(arg->Value,
"rfx"))
3900 rc = COMMAND_LINE_ERROR;
3902 else if (option_equals(arg->Value,
"nsc"))
3905 rc = COMMAND_LINE_ERROR;
3908#if defined(WITH_JPEG)
3909 else if (option_equals(arg->Value,
"jpeg"))
3912 rc = COMMAND_LINE_ERROR;
3917 return COMMAND_LINE_ERROR;
3923 else if (option_starts_with(
"persist-file:", val))
3927 rc = COMMAND_LINE_ERROR_MEMORY;
3929 rc = COMMAND_LINE_ERROR;
3933 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
3934 if (bval == PARSE_FAIL)
3935 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3938 if (option_starts_with(
"bitmap", val))
3942 rc = COMMAND_LINE_ERROR;
3944 else if (option_starts_with(
"glyph", val))
3947 bval != PARSE_OFF ? GLYPH_SUPPORT_FULL
3948 : GLYPH_SUPPORT_NONE))
3949 rc = COMMAND_LINE_ERROR;
3951 else if (option_starts_with(
"persist", val))
3955 rc = COMMAND_LINE_ERROR;
3957 else if (option_starts_with(
"offscreen", val))
3961 rc = COMMAND_LINE_ERROR;
3967 CommandLineParserFree(ptr);
3971static BOOL parse_gateway_host_option(rdpSettings* settings,
const char* host)
3973 WINPR_ASSERT(settings);
3978 if (!freerdp_parse_hostname(host, &name, &port))
3997static BOOL parse_gateway_cred_option(rdpSettings* settings,
const char* value,
3998 FreeRDP_Settings_Keys_String what)
4000 WINPR_ASSERT(settings);
4001 WINPR_ASSERT(value);
4005 case FreeRDP_GatewayUsername:
4006 if (!freerdp_parse_username_settings(value, settings, FreeRDP_GatewayUsername,
4007 FreeRDP_GatewayDomain))
4019static BOOL parse_gateway_type_option(rdpSettings* settings,
const char* value)
4023 WINPR_ASSERT(settings);
4024 WINPR_ASSERT(value);
4026 if (option_equals(value,
"rpc"))
4037 if (option_equals(value,
"http"))
4045 else if (option_equals(value,
"auto"))
4053 else if (option_equals(value,
"arm"))
4066static BOOL parse_gateway_usage_option(rdpSettings* settings,
const char* value)
4070 WINPR_ASSERT(settings);
4071 WINPR_ASSERT(value);
4073 if (option_equals(value,
"none"))
4074 type = TSC_PROXY_MODE_NONE_DIRECT;
4075 else if (option_equals(value,
"direct"))
4076 type = TSC_PROXY_MODE_DIRECT;
4077 else if (option_equals(value,
"detect"))
4078 type = TSC_PROXY_MODE_DETECT;
4079 else if (option_equals(value,
"default"))
4080 type = TSC_PROXY_MODE_DEFAULT;
4085 if (!value_to_int(value, &val, TSC_PROXY_MODE_NONE_DIRECT, TSC_PROXY_MODE_NONE_DETECT))
4092static char* unescape(
const char* str)
4094 char* copy = strdup(str);
4098 bool escaped =
false;
4100 while (*str !=
'\0')
4128 char* argval = NULL;
4131 WINPR_ASSERT(settings);
4135 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
4143 BOOL allowHttpOpts = FALSE;
4144 for (
size_t x = 0; x < count; x++)
4146 BOOL validOption = FALSE;
4148 argval = unescape(ptr[x]);
4152 const char* gw = option_starts_with(
"g:", argval);
4155 if (!parse_gateway_host_option(settings, gw))
4158 allowHttpOpts = FALSE;
4161 const char* gu = option_starts_with(
"u:", argval);
4164 if (!parse_gateway_cred_option(settings, gu, FreeRDP_GatewayUsername))
4167 allowHttpOpts = FALSE;
4170 const char* gd = option_starts_with(
"d:", argval);
4173 if (!parse_gateway_cred_option(settings, gd, FreeRDP_GatewayDomain))
4176 allowHttpOpts = FALSE;
4179 const char* gp = option_starts_with(
"p:", argval);
4182 if (!parse_gateway_cred_option(settings, gp, FreeRDP_GatewayPassword))
4185 allowHttpOpts = FALSE;
4188 const char* gt = option_starts_with(
"type:", argval);
4191 if (!parse_gateway_type_option(settings, gt))
4197 const char* gat = option_starts_with(
"access-token:", argval);
4203 allowHttpOpts = FALSE;
4206 const char* bearer = option_starts_with(
"bearer:", argval);
4212 allowHttpOpts = FALSE;
4215 const char* gwurl = option_starts_with(
"url:", argval);
4223 allowHttpOpts = FALSE;
4226 const char* um = option_starts_with(
"usage-method:", argval);
4229 if (!parse_gateway_usage_option(settings, um))
4232 allowHttpOpts = FALSE;
4237 if (option_equals(argval,
"no-websockets"))
4243 else if (option_equals(argval,
"extauth-sspi-ntlm"))
4258 CommandLineParserFree(ptr);
4265 WINPR_ASSERT(value);
4271 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
4272 FillMemory(arg->Value, strlen(arg->Value),
'*');
4277 for (
size_t x = 0; x < ARRAYSIZE(credential_args); x++)
4279 const char* cred = credential_args[x];
4280 fill_credential_string(args, cred);
4284 if (arg && ((arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT) != 0))
4286 const char* gwcreds[] = {
"p:",
"access-token:" };
4287 char* saveptr = NULL;
4288 char* tok = strtok_s(arg->Value,
",", &saveptr);
4291 for (
size_t x = 0; x < ARRAYSIZE(gwcreds); x++)
4293 const char* opt = gwcreds[x];
4294 if (option_starts_with(opt, tok))
4296 char* val = &tok[strlen(opt)];
4297 FillMemory(val, strlen(val),
'*');
4300 tok = strtok_s(NULL,
",", &saveptr);
4305static int parse_command_line_option_uint32(rdpSettings* settings,
4307 FreeRDP_Settings_Keys_UInt32 key, LONGLONG min,
4312 if (!value_to_int(arg->Value, &val, min, max))
4313 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4316 return fail_at(arg, COMMAND_LINE_ERROR);
4320#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
4325 WINPR_ASSERT(settings);
4328 BOOL enable = arg->Value ? TRUE : FALSE;
4329 CommandLineSwitchStart(arg) CommandLineSwitchCase(arg,
"gfx-thin-client")
4331 WLog_WARN(TAG,
"/gfx-thin-client is deprecated, use /gfx:thin-client[:on|off] instead");
4333 return fail_at(arg, COMMAND_LINE_ERROR);
4338 return fail_at(arg, COMMAND_LINE_ERROR);
4342 return fail_at(arg, COMMAND_LINE_ERROR);
4344 CommandLineSwitchCase(arg,
"gfx-small-cache")
4346 WLog_WARN(TAG,
"/gfx-small-cache is deprecated, use /gfx:small-cache[:on|off] instead");
4348 return fail_at(arg, COMMAND_LINE_ERROR);
4352 return fail_at(arg, COMMAND_LINE_ERROR);
4354 CommandLineSwitchCase(arg,
"gfx-progressive")
4356 WLog_WARN(TAG,
"/gfx-progressive is deprecated, use /gfx:progressive[:on|off] instead");
4358 return fail_at(arg, COMMAND_LINE_ERROR);
4360 return fail_at(arg, COMMAND_LINE_ERROR);
4365 return fail_at(arg, COMMAND_LINE_ERROR);
4369 CommandLineSwitchCase(arg,
"gfx-h264")
4371 WLog_WARN(TAG,
"/gfx-h264 is deprecated, use /gfx:avc420 instead");
4372 int rc = parse_gfx_options(settings, arg);
4374 return fail_at(arg, rc);
4377 CommandLineSwitchCase(arg,
"app-workdir")
4380 "/app-workdir:<directory> is deprecated, use /app:workdir:<directory> instead");
4382 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4384 CommandLineSwitchCase(arg,
"app-name")
4386 WLog_WARN(TAG,
"/app-name:<directory> is deprecated, use /app:name:<name> instead");
4388 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4390 CommandLineSwitchCase(arg,
"app-icon")
4392 WLog_WARN(TAG,
"/app-icon:<filename> is deprecated, use /app:icon:<filename> instead");
4394 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4396 CommandLineSwitchCase(arg,
"app-cmd")
4398 WLog_WARN(TAG,
"/app-cmd:<command> is deprecated, use /app:cmd:<command> instead");
4400 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4402 CommandLineSwitchCase(arg,
"app-file")
4404 WLog_WARN(TAG,
"/app-file:<filename> is deprecated, use /app:file:<filename> instead");
4406 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4408 CommandLineSwitchCase(arg,
"app-guid")
4410 WLog_WARN(TAG,
"/app-guid:<guid> is deprecated, use /app:guid:<guid> instead");
4412 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4414 CommandLineSwitchCase(arg,
"g")
4416 if (!parse_gateway_host_option(settings, arg->Value))
4417 return fail_at(arg, COMMAND_LINE_ERROR);
4419 CommandLineSwitchCase(arg,
"gu")
4421 if (!parse_gateway_cred_option(settings, arg->Value, FreeRDP_GatewayUsername))
4422 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4424 CommandLineSwitchCase(arg,
"gd")
4426 if (!parse_gateway_cred_option(settings, arg->Value, FreeRDP_GatewayDomain))
4427 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4429 CommandLineSwitchCase(arg,
"gp")
4431 if (!parse_gateway_cred_option(settings, arg->Value, FreeRDP_GatewayPassword))
4432 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4434 CommandLineSwitchCase(arg,
"gt")
4436 if (!parse_gateway_type_option(settings, arg->Value))
4437 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4439 CommandLineSwitchCase(arg,
"gat")
4442 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4444 CommandLineSwitchCase(arg,
"gateway-usage-method")
4446 if (!parse_gateway_usage_option(settings, arg->Value))
4447 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4449 CommandLineSwitchCase(arg,
"kbd-remap")
4451 WLog_WARN(TAG,
"/kbd-remap:<key>=<value>,<key2>=<value2> is deprecated, use "
4452 "/kbd:remap:<key>=<value>,remap:<key2>=<value2>,... instead");
4454 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4456 CommandLineSwitchCase(arg,
"kbd-lang")
4460 WLog_WARN(TAG,
"/kbd-lang:<value> is deprecated, use /kbd:lang:<value> instead");
4461 if (!value_to_int(arg->Value, &val, 1, UINT32_MAX))
4463 WLog_ERR(TAG,
"Could not identify keyboard active language %s", arg->Value);
4464 WLog_ERR(TAG,
"Use /list:kbd-lang to list available layouts");
4465 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4469 return fail_at(arg, COMMAND_LINE_ERROR);
4471 CommandLineSwitchCase(arg,
"kbd-type")
4473 WLog_WARN(TAG,
"/kbd-type:<value> is deprecated, use /kbd:type:<value> instead");
4475 parse_command_line_option_uint32(settings, arg, FreeRDP_KeyboardType, 0, UINT32_MAX);
4477 return fail_at(arg, rc);
4479 CommandLineSwitchCase(arg,
"kbd-unicode")
4481 WLog_WARN(TAG,
"/kbd-unicode is deprecated, use /kbd:unicode[:on|off] instead");
4483 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4485 CommandLineSwitchCase(arg,
"kbd-subtype")
4487 WLog_WARN(TAG,
"/kbd-subtype:<value> is deprecated, use /kbd:subtype:<value> instead");
4489 parse_command_line_option_uint32(settings, arg, FreeRDP_KeyboardSubType, 0, UINT32_MAX);
4491 return fail_at(arg, rc);
4493 CommandLineSwitchCase(arg,
"kbd-fn-key")
4495 WLog_WARN(TAG,
"/kbd-fn-key:<value> is deprecated, use /kbd:fn-key:<value> instead");
4496 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_KeyboardFunctionKey,
4499 return fail_at(arg, rc);
4501 CommandLineSwitchCase(arg,
"bitmap-cache")
4503 WLog_WARN(TAG,
"/bitmap-cache is deprecated, use /cache:bitmap[:on|off] instead");
4505 return fail_at(arg, COMMAND_LINE_ERROR);
4507 CommandLineSwitchCase(arg,
"persist-cache")
4509 WLog_WARN(TAG,
"/persist-cache is deprecated, use /cache:persist[:on|off] instead");
4511 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4513 CommandLineSwitchCase(arg,
"persist-cache-file")
4515 WLog_WARN(TAG,
"/persist-cache-file:<filename> is deprecated, use "
4516 "/cache:persist-file:<filename> instead");
4518 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4521 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4523 CommandLineSwitchCase(arg,
"offscreen-cache")
4525 WLog_WARN(TAG,
"/bitmap-cache is deprecated, use /cache:bitmap[:on|off] instead");
4527 return fail_at(arg, COMMAND_LINE_ERROR);
4529 CommandLineSwitchCase(arg,
"glyph-cache")
4531 WLog_WARN(TAG,
"/glyph-cache is deprecated, use /cache:glyph[:on|off] instead");
4533 arg->Value ? GLYPH_SUPPORT_FULL : GLYPH_SUPPORT_NONE))
4534 return fail_at(arg, COMMAND_LINE_ERROR);
4536 CommandLineSwitchCase(arg,
"codec-cache")
4538 WLog_WARN(TAG,
"/codec-cache:<option> is deprecated, use /cache:codec:<option> instead");
4539 const int rc = parse_codec_cache_options(settings, arg);
4541 return fail_at(arg, rc);
4543 CommandLineSwitchCase(arg,
"sec-rdp")
4545 WLog_WARN(TAG,
"/sec-rdp is deprecated, use /sec:rdp[:on|off] instead");
4547 return fail_at(arg, COMMAND_LINE_ERROR);
4549 CommandLineSwitchCase(arg,
"sec-tls")
4551 WLog_WARN(TAG,
"/sec-tls is deprecated, use /sec:tls[:on|off] instead");
4553 return fail_at(arg, COMMAND_LINE_ERROR);
4555 CommandLineSwitchCase(arg,
"sec-nla")
4557 WLog_WARN(TAG,
"/sec-nla is deprecated, use /sec:nla[:on|off] instead");
4559 return fail_at(arg, COMMAND_LINE_ERROR);
4561 CommandLineSwitchCase(arg,
"sec-ext")
4563 WLog_WARN(TAG,
"/sec-ext is deprecated, use /sec:ext[:on|off] instead");
4565 return fail_at(arg, COMMAND_LINE_ERROR);
4567 CommandLineSwitchCase(arg,
"tls-ciphers")
4569 WLog_WARN(TAG,
"/tls-ciphers:<cipher list> is deprecated, use "
4570 "/tls:ciphers:<cipher list> instead");
4571 int rc = parse_tls_cipher_options(settings, arg);
4573 return fail_at(arg, rc);
4575 CommandLineSwitchCase(arg,
"tls-seclevel")
4577 WLog_WARN(TAG,
"/tls-seclevel:<level> is deprecated, use /tls:sec-level:<level> instead");
4578 int rc = parse_tls_cipher_options(settings, arg);
4580 return fail_at(arg, rc);
4582 CommandLineSwitchCase(arg,
"tls-secrets-file")
4584 WLog_WARN(TAG,
"/tls-secrets-file:<filename> is deprecated, use "
4585 "/tls:secrets-file:<filename> instead");
4586 int rc = parse_tls_cipher_options(settings, arg);
4588 return fail_at(arg, rc);
4590 CommandLineSwitchCase(arg,
"enforce-tlsv1_2")
4592 WLog_WARN(TAG,
"/enforce-tlsv1_2 is deprecated, use /tls:enforce:1.2 instead");
4593 int rc = parse_tls_cipher_options(settings, arg);
4595 return fail_at(arg, rc);
4597 CommandLineSwitchCase(arg,
"cert-name")
4599 WLog_WARN(TAG,
"/cert-name is deprecated, use /cert:name instead");
4601 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4603 CommandLineSwitchCase(arg,
"cert-ignore")
4605 WLog_WARN(TAG,
"/cert-ignore is deprecated, use /cert:ignore instead");
4607 return fail_at(arg, COMMAND_LINE_ERROR);
4609 CommandLineSwitchCase(arg,
"cert-tofu")
4611 WLog_WARN(TAG,
"/cert-tofu is deprecated, use /cert:tofu instead");
4613 return fail_at(arg, COMMAND_LINE_ERROR);
4615 CommandLineSwitchCase(arg,
"cert-deny")
4617 WLog_WARN(TAG,
"/cert-deny is deprecated, use /cert:deny instead");
4619 return fail_at(arg, COMMAND_LINE_ERROR);
4621 CommandLineSwitchDefault(arg)
4625 CommandLineSwitchEnd(arg);
4630static int parse_command_line_option_timezone(rdpSettings* settings,
4636 char TimeZoneKeyName[ARRAYSIZE(info.TimeZoneKeyName) + 1] = { 0 };
4637 while (EnumDynamicTimeZoneInformation(index++, &info) != ERROR_NO_MORE_ITEMS)
4639 (void)ConvertWCharNToUtf8(info.TimeZoneKeyName, ARRAYSIZE(info.TimeZoneKeyName),
4640 TimeZoneKeyName, ARRAYSIZE(TimeZoneKeyName));
4642 WINPR_ASSERT(arg->Value);
4643 if (strncmp(TimeZoneKeyName, arg->Value, ARRAYSIZE(TimeZoneKeyName)) == 0)
4650 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4653 return fail_at(arg, COMMAND_LINE_ERROR);
4658 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4660 tz->Bias = info.Bias;
4661 tz->DaylightBias = info.DaylightBias;
4662 tz->DaylightDate = info.DaylightDate;
4663 memcpy(tz->DaylightName, info.DaylightName,
sizeof(tz->DaylightName));
4664 tz->StandardBias = info.StandardBias;
4665 tz->StandardDate = info.StandardDate;
4666 memcpy(tz->StandardName, info.StandardName,
sizeof(tz->StandardName));
4671static int parse_command_line_option_window_pos(rdpSettings* settings,
4674 WINPR_ASSERT(settings);
4677 unsigned long x = 0;
4678 unsigned long y = 0;
4681 return fail_at(arg, COMMAND_LINE_ERROR_MISSING_ARGUMENT);
4683 if (!parseSizeValue(arg->Value, &x, &y) || x > UINT16_MAX || y > UINT16_MAX)
4685 WLog_ERR(TAG,
"invalid window-position argument");
4686 return fail_at(arg, COMMAND_LINE_ERROR_MISSING_ARGUMENT);
4690 return fail_at(arg, COMMAND_LINE_ERROR);
4692 return fail_at(arg, COMMAND_LINE_ERROR);
4697 freerdp_command_line_handle_option_t handle_option,
4698 void* handle_userdata, BOOL* promptForPassword,
char** user)
4700 WINPR_ASSERT(promptForPassword);
4705 BOOL enable = arg->Value ? TRUE : FALSE;
4707 if (!(arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
4710 CommandLineSwitchStart(arg)
4712 CommandLineSwitchCase(arg,
"v")
4714 const int rc = parse_host_options(settings, arg);
4716 return fail_at(arg, rc);
4718 CommandLineSwitchCase(arg,
"spn-class")
4722 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4724 CommandLineSwitchCase(arg,
"sspi-module")
4727 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4729 CommandLineSwitchCase(arg,
"winscard-module")
4732 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4734 CommandLineSwitchCase(arg,
"redirect-prefer")
4736 const int rc = parse_redirect_prefer_options(settings, arg);
4738 return fail_at(arg, rc);
4740 CommandLineSwitchCase(arg,
"credentials-delegation")
4743 return fail_at(arg, COMMAND_LINE_ERROR);
4745 CommandLineSwitchCase(arg,
"prevent-session-lock")
4747 const int rc = parse_prevent_session_lock_options(settings, arg);
4749 return fail_at(arg, rc);
4751 CommandLineSwitchCase(arg,
"vmconnect")
4753 const int rc = parse_vmconnect_options(settings, arg);
4755 return fail_at(arg, rc);
4757 CommandLineSwitchCase(arg,
"w")
4759 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_DesktopWidth, -1,
4762 return fail_at(arg, rc);
4764 CommandLineSwitchCase(arg,
"h")
4766 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_DesktopHeight,
4769 return fail_at(arg, rc);
4771 CommandLineSwitchCase(arg,
"size")
4773 const int rc = parse_size_options(settings, arg);
4775 return fail_at(arg, rc);
4777 CommandLineSwitchCase(arg,
"f")
4780 return fail_at(arg, COMMAND_LINE_ERROR);
4782 CommandLineSwitchCase(arg,
"suppress-output")
4785 return fail_at(arg, COMMAND_LINE_ERROR);
4787 CommandLineSwitchCase(arg,
"multimon")
4790 return fail_at(arg, COMMAND_LINE_ERROR);
4792 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
4794 if (option_equals(arg->Value, str_force))
4797 return fail_at(arg, COMMAND_LINE_ERROR);
4801 CommandLineSwitchCase(arg,
"span")
4804 return fail_at(arg, COMMAND_LINE_ERROR);
4806 CommandLineSwitchCase(arg,
"workarea")
4809 return fail_at(arg, COMMAND_LINE_ERROR);
4811 CommandLineSwitchCase(arg,
"monitors")
4813 const int rc = parse_monitors_options(settings, arg);
4815 return fail_at(arg, rc);
4817 CommandLineSwitchCase(arg,
"t")
4820 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4822 CommandLineSwitchCase(arg,
"decorations")
4825 return fail_at(arg, COMMAND_LINE_ERROR);
4827 CommandLineSwitchCase(arg,
"dynamic-resolution")
4829 const int rc = parse_dynamic_resolution_options(settings, arg);
4831 return fail_at(arg, rc);
4833 CommandLineSwitchCase(arg,
"smart-sizing")
4835 const int rc = parse_smart_sizing_options(settings, arg);
4837 return fail_at(arg, rc);
4839 CommandLineSwitchCase(arg,
"bpp")
4841 const int rc = parse_bpp_options(settings, arg);
4843 return fail_at(arg, rc);
4845 CommandLineSwitchCase(arg,
"admin")
4848 return fail_at(arg, COMMAND_LINE_ERROR);
4850 CommandLineSwitchCase(arg,
"relax-order-checks")
4854 return fail_at(arg, COMMAND_LINE_ERROR);
4856 CommandLineSwitchCase(arg,
"restricted-admin")
4859 return fail_at(arg, COMMAND_LINE_ERROR);
4861 return fail_at(arg, COMMAND_LINE_ERROR);
4863 CommandLineSwitchCase(arg,
"remoteGuard")
4866 return fail_at(arg, COMMAND_LINE_ERROR);
4868 return fail_at(arg, COMMAND_LINE_ERROR);
4870 CommandLineSwitchCase(arg,
"pth")
4873 return fail_at(arg, COMMAND_LINE_ERROR);
4875 return fail_at(arg, COMMAND_LINE_ERROR);
4878 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4880 CommandLineSwitchCase(arg,
"client-hostname")
4883 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4885 CommandLineSwitchCase(arg,
"kbd")
4887 int rc = parse_kbd_options(settings, arg);
4889 return fail_at(arg, rc);
4892 CommandLineSwitchCase(arg,
"u")
4894 WINPR_ASSERT(arg->Value);
4897 CommandLineSwitchCase(arg,
"d")
4900 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4902 CommandLineSwitchCase(arg,
"p")
4905 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4907 CommandLineSwitchCase(arg,
"gateway")
4909 if (!parse_gateway_options(settings, arg))
4910 return fail_at(arg, COMMAND_LINE_ERROR);
4912 CommandLineSwitchCase(arg,
"proxy")
4914 const int rc = parse_proxy_options(settings, arg);
4916 return fail_at(arg, rc);
4919 CommandLineSwitchCase(arg,
"azure")
4921 int rc = parse_aad_options(settings, arg);
4923 return fail_at(arg, rc);
4925 CommandLineSwitchCase(arg,
"app")
4927 int rc = parse_app_options(settings, arg);
4929 return fail_at(arg, rc);
4931 CommandLineSwitchCase(arg,
"load-balance-info")
4933 WINPR_ASSERT(arg->Value);
4935 strlen(arg->Value)))
4936 return fail_at(arg, COMMAND_LINE_ERROR);
4939 CommandLineSwitchCase(arg,
"compression")
4942 return fail_at(arg, COMMAND_LINE_ERROR);
4944 CommandLineSwitchCase(arg,
"compression-level")
4946 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_CompressionLevel,
4949 return fail_at(arg, rc);
4951 CommandLineSwitchCase(arg,
"drives")
4954 return fail_at(arg, COMMAND_LINE_ERROR);
4956 CommandLineSwitchCase(arg,
"dump")
4958 const int rc = parse_dump_options(settings, arg);
4960 return fail_at(arg, rc);
4962 CommandLineSwitchCase(arg,
"disable-output")
4965 return fail_at(arg, COMMAND_LINE_ERROR);
4967 CommandLineSwitchCase(arg,
"home-drive")
4970 return fail_at(arg, COMMAND_LINE_ERROR);
4972 CommandLineSwitchCase(arg,
"ipv4")
4974 if (arg->Value != NULL && strncmp(arg->Value, str_force, ARRAYSIZE(str_force)) == 0)
4977 return fail_at(arg, COMMAND_LINE_ERROR);
4982 return fail_at(arg, COMMAND_LINE_ERROR);
4985 CommandLineSwitchCase(arg,
"ipv6")
4987 if (arg->Value != NULL && strncmp(arg->Value, str_force, ARRAYSIZE(str_force)) == 0)
4990 return fail_at(arg, COMMAND_LINE_ERROR);
4995 return fail_at(arg, COMMAND_LINE_ERROR);
4998 CommandLineSwitchCase(arg,
"clipboard")
5000 const int rc = parse_clipboard_options(settings, arg);
5002 return fail_at(arg, rc);
5004 CommandLineSwitchCase(arg,
"server-name")
5007 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5009 CommandLineSwitchCase(arg,
"shell")
5012 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5014 CommandLineSwitchCase(arg,
"shell-dir")
5017 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5019 CommandLineSwitchCase(arg,
"audio-mode")
5021 const int rc = parse_audio_mode_options(settings, arg);
5023 return fail_at(arg, rc);
5025 CommandLineSwitchCase(arg,
"network")
5027 const int rc = parse_network_options(settings, arg);
5029 return fail_at(arg, rc);
5031 CommandLineSwitchCase(arg,
"fonts")
5034 return fail_at(arg, COMMAND_LINE_ERROR);
5036 CommandLineSwitchCase(arg,
"wallpaper")
5039 return fail_at(arg, COMMAND_LINE_ERROR);
5041 CommandLineSwitchCase(arg,
"window-drag")
5044 return fail_at(arg, COMMAND_LINE_ERROR);
5046 CommandLineSwitchCase(arg,
"window-position")
5048 const int rc = parse_command_line_option_window_pos(settings, arg);
5050 return fail_at(arg, rc);
5052 CommandLineSwitchCase(arg,
"menu-anims")
5055 return fail_at(arg, COMMAND_LINE_ERROR);
5057 CommandLineSwitchCase(arg,
"themes")
5060 return fail_at(arg, COMMAND_LINE_ERROR);
5062 CommandLineSwitchCase(arg,
"timeout")
5065 parse_command_line_option_uint32(settings, arg, FreeRDP_TcpAckTimeout, 0, 600000);
5067 return fail_at(arg, rc);
5069 CommandLineSwitchCase(arg,
"timezone")
5071 const int rc = parse_command_line_option_timezone(settings, arg);
5073 return fail_at(arg, rc);
5075 CommandLineSwitchCase(arg,
"aero")
5078 return fail_at(arg, COMMAND_LINE_ERROR);
5080 CommandLineSwitchCase(arg,
"gdi")
5082 if (option_equals(arg->Value,
"sw"))
5085 return fail_at(arg, COMMAND_LINE_ERROR);
5087 else if (option_equals(arg->Value,
"hw"))
5090 return fail_at(arg, COMMAND_LINE_ERROR);
5093 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5095 CommandLineSwitchCase(arg,
"gfx")
5097 int rc = parse_gfx_options(settings, arg);
5099 return fail_at(arg, rc);
5102 CommandLineSwitchCase(arg,
"rfx")
5105 return fail_at(arg, COMMAND_LINE_ERROR);
5107 CommandLineSwitchCase(arg,
"rfx-mode")
5110 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5112 if (option_equals(arg->Value,
"video"))
5115 return fail_at(arg, COMMAND_LINE_ERROR);
5117 else if (option_equals(arg->Value,
"image"))
5120 return fail_at(arg, COMMAND_LINE_ERROR);
5122 return fail_at(arg, COMMAND_LINE_ERROR);
5125 CommandLineSwitchCase(arg,
"frame-ack")
5127 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_FrameAcknowledge,
5130 return fail_at(arg, rc);
5132 CommandLineSwitchCase(arg,
"nsc")
5135 return fail_at(arg, COMMAND_LINE_ERROR);
5137#if defined(WITH_JPEG)
5138 CommandLineSwitchCase(arg,
"jpeg")
5141 return fail_at(arg, COMMAND_LINE_ERROR);
5143 return fail_at(arg, COMMAND_LINE_ERROR);
5145 CommandLineSwitchCase(arg,
"jpeg-quality")
5149 if (!value_to_int(arg->Value, &val, 0, 100))
5150 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5153 return fail_at(arg, COMMAND_LINE_ERROR);
5156 CommandLineSwitchCase(arg,
"nego")
5159 return fail_at(arg, COMMAND_LINE_ERROR);
5161 CommandLineSwitchCase(arg,
"pcb")
5164 return fail_at(arg, COMMAND_LINE_ERROR);
5167 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5169 CommandLineSwitchCase(arg,
"pcid")
5171 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_PreconnectionId,
5174 return fail_at(arg, rc);
5176 return fail_at(arg, COMMAND_LINE_ERROR);
5179 CommandLineSwitchCase(arg,
"connect-child-session")
5193 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5196 CommandLineSwitchCase(arg,
"sec")
5198 const int rc = parse_sec_options(settings, arg);
5200 return fail_at(arg, rc);
5202 CommandLineSwitchCase(arg,
"encryption-methods")
5204 const int rc = parse_encryption_methods_options(settings, arg);
5206 return fail_at(arg, rc);
5208 CommandLineSwitchCase(arg,
"args-from")
5210 WLog_ERR(TAG,
"/args-from:%s can not be used in combination with other arguments!",
5212 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5214 CommandLineSwitchCase(arg,
"from-stdin")
5217 return fail_at(arg, COMMAND_LINE_ERROR);
5219 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
5222 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5223 *promptForPassword = (option_equals(arg->Value, str_force));
5225 if (!*promptForPassword)
5226 return fail_at(arg, COMMAND_LINE_ERROR);
5229 CommandLineSwitchCase(arg,
"log-level")
5231 wLog* root = WLog_GetRoot();
5233 if (!WLog_SetStringLogLevel(root, arg->Value))
5234 return fail_at(arg, COMMAND_LINE_ERROR);
5236 CommandLineSwitchCase(arg,
"log-filters")
5238 if (!WLog_AddStringLogFilters(arg->Value))
5239 return fail_at(arg, COMMAND_LINE_ERROR);
5241 CommandLineSwitchCase(arg,
"tls")
5243 int rc = parse_tls_options(settings, arg);
5245 return fail_at(arg, rc);
5247 CommandLineSwitchCase(arg,
"cert")
5249 const int rc = parse_cert_options(settings, arg);
5251 return fail_at(arg, rc);
5253 CommandLineSwitchCase(arg,
"authentication")
5256 return fail_at(arg, COMMAND_LINE_ERROR);
5258 CommandLineSwitchCase(arg,
"encryption")
5261 return fail_at(arg, COMMAND_LINE_ERROR);
5263 CommandLineSwitchCase(arg,
"grab-keyboard")
5266 return fail_at(arg, COMMAND_LINE_ERROR);
5268 CommandLineSwitchCase(arg,
"grab-mouse")
5271 return fail_at(arg, COMMAND_LINE_ERROR);
5273 CommandLineSwitchCase(arg,
"mouse-relative")
5276 return fail_at(arg, COMMAND_LINE_ERROR);
5278 CommandLineSwitchCase(arg,
"mouse")
5280 const int rc = parse_mouse_options(settings, arg);
5282 return fail_at(arg, rc);
5284 CommandLineSwitchCase(arg,
"unmap-buttons")
5287 return fail_at(arg, COMMAND_LINE_ERROR);
5289 CommandLineSwitchCase(arg,
"toggle-fullscreen")
5292 return fail_at(arg, COMMAND_LINE_ERROR);
5294 CommandLineSwitchCase(arg,
"force-console-callbacks")
5297 return fail_at(arg, COMMAND_LINE_ERROR);
5299 CommandLineSwitchCase(arg,
"floatbar")
5301 const int rc = parse_floatbar_options(settings, arg);
5303 return fail_at(arg, rc);
5305 CommandLineSwitchCase(arg,
"mouse-motion")
5308 return fail_at(arg, COMMAND_LINE_ERROR);
5310 CommandLineSwitchCase(arg,
"parent-window")
5314 if (!value_to_uint(arg->Value, &val, 0, UINT64_MAX))
5315 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5318 return fail_at(arg, COMMAND_LINE_ERROR);
5320 CommandLineSwitchCase(arg,
"client-build-number")
5323 parse_command_line_option_uint32(settings, arg, FreeRDP_ClientBuild, 0, UINT32_MAX);
5325 return fail_at(arg, rc);
5327 CommandLineSwitchCase(arg,
"cache")
5329 int rc = parse_cache_options(settings, arg);
5331 return fail_at(arg, rc);
5334 CommandLineSwitchCase(arg,
"max-fast-path-size")
5336 const int rc = parse_command_line_option_uint32(
5337 settings, arg, FreeRDP_MultifragMaxRequestSize, 0, UINT32_MAX);
5339 return fail_at(arg, rc);
5341 CommandLineSwitchCase(arg,
"auto-request-control")
5345 return fail_at(arg, COMMAND_LINE_ERROR);
5347 CommandLineSwitchCase(arg,
"async-update")
5350 return fail_at(arg, COMMAND_LINE_ERROR);
5352 CommandLineSwitchCase(arg,
"async-channels")
5355 return fail_at(arg, COMMAND_LINE_ERROR);
5357 CommandLineSwitchCase(arg,
"wm-class")
5360 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5362 CommandLineSwitchCase(arg,
"play-rfx")
5365 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5368 return fail_at(arg, COMMAND_LINE_ERROR);
5370 CommandLineSwitchCase(arg,
"auth-only")
5373 return fail_at(arg, COMMAND_LINE_ERROR);
5375 CommandLineSwitchCase(arg,
"auth-pkg-list")
5379 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5381 CommandLineSwitchCase(arg,
"auto-reconnect")
5384 return fail_at(arg, COMMAND_LINE_ERROR);
5386 CommandLineSwitchCase(arg,
"auto-reconnect-max-retries")
5388 const int rc = parse_command_line_option_uint32(
5389 settings, arg, FreeRDP_AutoReconnectMaxRetries, 0, 1000);
5391 return fail_at(arg, rc);
5393 CommandLineSwitchCase(arg,
"reconnect-cookie")
5395 const int rc = parse_reconnect_cookie_options(settings, arg);
5397 return fail_at(arg, rc);
5399 CommandLineSwitchCase(arg,
"print-reconnect-cookie")
5402 return fail_at(arg, COMMAND_LINE_ERROR);
5404 CommandLineSwitchCase(arg,
"pwidth")
5406 const int rc = parse_command_line_option_uint32(
5407 settings, arg, FreeRDP_DesktopPhysicalWidth, 0, UINT32_MAX);
5409 return fail_at(arg, rc);
5411 CommandLineSwitchCase(arg,
"pheight")
5413 const int rc = parse_command_line_option_uint32(
5414 settings, arg, FreeRDP_DesktopPhysicalHeight, 0, UINT32_MAX);
5416 return fail_at(arg, rc);
5418 CommandLineSwitchCase(arg,
"orientation")
5422 if (!value_to_int(arg->Value, &val, 0, UINT16_MAX))
5423 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5426 return fail_at(arg, COMMAND_LINE_ERROR);
5427 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_ORIENTATION))
5428 return fail_at(arg, COMMAND_LINE_ERROR);
5430 CommandLineSwitchCase(arg,
"old-license")
5433 return fail_at(arg, COMMAND_LINE_ERROR);
5435 CommandLineSwitchCase(arg,
"scale")
5437 const int rc = parse_scale_options(settings, arg);
5439 return fail_at(arg, rc);
5441 CommandLineSwitchCase(arg,
"scale-desktop")
5443 const int rc = parse_command_line_option_uint32(settings, arg,
5444 FreeRDP_DesktopScaleFactor, 100, 500);
5446 return fail_at(arg, rc);
5447 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_DESKTOP_SCALE))
5448 return fail_at(arg, COMMAND_LINE_ERROR);
5450 CommandLineSwitchCase(arg,
"scale-device")
5452 const int rc = parse_scale_device_options(settings, arg);
5454 return fail_at(arg, rc);
5456 CommandLineSwitchCase(arg,
"action-script")
5459 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5461 CommandLineSwitchCase(arg, RDP2TCP_DVC_CHANNEL_NAME)
5464 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5466 CommandLineSwitchCase(arg,
"fipsmode")
5469 return fail_at(arg, COMMAND_LINE_ERROR);
5471 CommandLineSwitchCase(arg,
"smartcard-logon")
5473 const int rc = parse_smartcard_logon_options(settings, arg);
5475 return fail_at(arg, rc);
5477 CommandLineSwitchCase(arg,
"tune")
5479 const int rc = parse_tune_options(settings, arg);
5481 return fail_at(arg, rc);
5483 CommandLineSwitchDefault(arg)
5485#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
5486 const int status = parse_deprecated_command_line(settings, arg);
5493 const int rc = handle_option(arg, handle_userdata);
5495 return fail_at(arg, rc);
5498 CommandLineSwitchEnd(arg)
5499 }
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
5503static void warn_credential_args(
const COMMAND_LINE_ARGUMENT* args)
5506 bool insecureArgFound =
false;
5507 for (
size_t x = 0; x < ARRAYSIZE(credential_args); x++)
5509 const char* cred = credential_args[x];
5510 const COMMAND_LINE_ARGUMENT* arg = CommandLineFindArgumentA(args, cred);
5513 if ((arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT) == 0)
5516 WLog_WARN(TAG,
"Using /%s is insecure", arg->Name);
5517 insecureArgFound =
true;
5520 if (insecureArgFound)
5522 WLog_WARN(TAG,
"Passing credentials or secrets via command line might expose these in the "
5524 WLog_WARN(TAG,
"Consider using one of the following (more secure) alternatives:");
5525 WLog_WARN(TAG,
" - /args-from: pipe in arguments from stdin, file or file descriptor");
5526 WLog_WARN(TAG,
" - /from-stdin pass the credential via stdin");
5527 WLog_WARN(TAG,
" - set environment variable FREERDP_ASKPASS to have a gui tool query for "
5532static int freerdp_client_settings_parse_command_line_arguments_int(
5533 rdpSettings* settings,
int argc,
char* argv[], BOOL allowUnknown,
5535 freerdp_command_line_handle_option_t handle_option,
void* handle_userdata,
bool isArgsFrom)
5540 BOOL assist = FALSE;
5542 BOOL promptForPassword = FALSE;
5543 BOOL compatibility = FALSE;
5551 ext = option_is_rdp_file(argv[1]);
5552 assist = option_is_incident_file(argv[1]);
5555 if (!ext && !assist)
5556 compatibility = freerdp_client_detect_command_line(argc, argv, &flags);
5558 compatibility = freerdp_client_detect_command_line(argc - 1, &argv[1], &flags);
5569 WLog_WARN(TAG,
"Unsupported command line syntax!");
5570 WLog_WARN(TAG,
"FreeRDP 1.0 style syntax was dropped with version 3!");
5575 flags |= COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
5579 if (freerdp_client_settings_parse_connection_file(settings, argv[1]))
5580 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
5585 if (freerdp_client_settings_parse_assistance_file(settings, argc, argv) < 0)
5586 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
5589 CommandLineClearArgumentsA(largs);
5590 status = CommandLineParseArgumentsA(argc, argv, largs, flags, settings,
5591 freerdp_client_command_line_pre_filter,
5592 freerdp_client_command_line_post_filter);
5597 prepare_default_settings(settings, largs, ext);
5599 warn_credential_args(largs);
5601 CommandLineFindArgumentA(largs,
"v");
5607 return COMMAND_LINE_ERROR_MEMORY;
5609 status = parse_command_line(settings, arg, handle_option, handle_userdata, &promptForPassword,
5617 return COMMAND_LINE_ERROR;
5620 return COMMAND_LINE_ERROR;
5622 if (!freerdp_parse_username_settings(user, settings, FreeRDP_Username, FreeRDP_Domain))
5623 return COMMAND_LINE_ERROR;
5628 return COMMAND_LINE_ERROR;
5632 if (promptForPassword)
5637 char buffer[512 + 1] = { 0 };
5639 if (!freerdp_passphrase_read(instance->context,
"Password: ", buffer,
5640 ARRAYSIZE(buffer) - 1, 1))
5641 return COMMAND_LINE_ERROR;
5643 return COMMAND_LINE_ERROR;
5651 char buffer[512 + 1] = { 0 };
5653 if (!freerdp_passphrase_read(instance->context,
"Gateway Password: ", buffer,
5654 ARRAYSIZE(buffer) - 1, 1))
5655 return COMMAND_LINE_ERROR;
5657 return COMMAND_LINE_ERROR;
5662 freerdp_performance_flags_make(settings);
5669 return COMMAND_LINE_ERROR;
5671 return COMMAND_LINE_ERROR;
5674 arg = CommandLineFindArgumentA(largs,
"port");
5675 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
5678 parse_command_line_option_uint32(settings, arg, FreeRDP_ServerPort, 0, UINT16_MAX);
5680 return fail_at(arg, rc);
5686 if (nego && (nego->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
5687 return fail_at(arg, COMMAND_LINE_ERROR);
5690 WLog_INFO(TAG,
"/vmconnect uses custom port %" PRIu32, port);
5693 fill_credential_strings(largs);
5698static void argv_free(
int* pargc,
char** pargv[])
5700 WINPR_ASSERT(pargc);
5701 WINPR_ASSERT(pargv);
5702 const int argc = *pargc;
5703 char** argv = *pargv;
5709 for (
int x = 0; x < argc; x++)
5714static BOOL argv_append(
int* pargc,
char** pargv[],
char* what)
5716 WINPR_ASSERT(pargc);
5717 WINPR_ASSERT(pargv);
5725 int nargc = *pargc + 1;
5726 char** tmp = (
char**)realloc((
void*)*pargv, (size_t)nargc *
sizeof(
char*));
5736static BOOL argv_append_dup(
int* pargc,
char** pargv[],
const char* what)
5740 copy = _strdup(what);
5742 const BOOL rc = argv_append(pargc, pargv, copy);
5748static BOOL args_from_fp(FILE* fp,
int* aargc,
char** aargv[],
const char* file,
const char* cmd)
5750 BOOL success = FALSE;
5752 WINPR_ASSERT(aargc);
5753 WINPR_ASSERT(aargv);
5758 WLog_ERR(TAG,
"Failed to read command line options from file '%s'", file);
5761 if (!argv_append_dup(aargc, aargv, cmd))
5767 INT64 rc = GetLine(&line, &size, fp);
5768 if ((rc < 0) || !line)
5778 const char cur = (line[rc - 1]);
5779 if ((cur ==
'\n') || (cur ==
'\r'))
5781 line[rc - 1] =
'\0';
5793 if (!argv_append(aargc, aargv, line))
5804 argv_free(aargc, aargv);
5808static BOOL args_from_env(
const char* name,
int* aargc,
char** aargv[],
const char* arg,
5811 BOOL success = FALSE;
5814 WINPR_ASSERT(aargc);
5815 WINPR_ASSERT(aargv);
5820 WLog_ERR(TAG,
"%s - environment variable name empty", arg);
5824 const DWORD size = GetEnvironmentVariableX(name, env, 0);
5827 WLog_ERR(TAG,
"%s - no environment variable '%s'", arg, name);
5830 env = calloc(size + 1,
sizeof(
char));
5833 const DWORD rc = GetEnvironmentVariableX(name, env, size);
5838 WLog_ERR(TAG,
"%s - environment variable '%s' is empty", arg);
5842 if (!argv_append_dup(aargc, aargv, cmd))
5845 char* context = NULL;
5846 char* tok = strtok_s(env,
"\n", &context);
5849 if (!argv_append_dup(aargc, aargv, tok))
5851 tok = strtok_s(NULL,
"\n", &context);
5858 argv_free(aargc, aargv);
5862int freerdp_client_settings_parse_command_line_arguments(rdpSettings* settings,
int oargc,
5863 char* oargv[], BOOL allowUnknown)
5865 return freerdp_client_settings_parse_command_line_arguments_ex(
5866 settings, oargc, oargv, allowUnknown, NULL, 0, NULL, NULL);
5869int freerdp_client_settings_parse_command_line_arguments_ex(
5870 rdpSettings* settings,
int oargc,
char** oargv, BOOL allowUnknown,
5872 void* handle_userdata)
5875 char** argv = oargv;
5878 char** aargv = NULL;
5880 bool isArgsFrom =
false;
5881 if ((argc == 2) && option_starts_with(
"/args-from:", argv[1]))
5884 BOOL success = FALSE;
5885 const char* file = strchr(argv[1],
':') + 1;
5888 if (option_starts_with(
"fd:", file))
5890 ULONGLONG result = 0;
5891 const char* val = strchr(file,
':') + 1;
5892 if (!value_to_uint(val, &result, 0, INT_MAX))
5894 fp = fdopen((
int)result,
"r");
5895 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
5897 else if (strncmp(file,
"env:", 4) == 0)
5899 const char* name = strchr(file,
':') + 1;
5900 success = args_from_env(name, &aargc, &aargv, oargv[1], oargv[0]);
5902 else if (strcmp(file,
"stdin") != 0)
5904 fp = winpr_fopen(file,
"r");
5905 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
5908 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
5916 WINPR_ASSERT(count <= SSIZE_MAX);
5922 res = freerdp_client_settings_parse_command_line_arguments_int(
5923 settings, argc, argv, allowUnknown, largs, lcount, handle_option, handle_userdata,
5927 argv_free(&aargc, &aargv);
5931static BOOL freerdp_client_load_static_channel_addin(rdpChannels* channels, rdpSettings* settings,
5932 const char* name,
void* data)
5934 PVIRTUALCHANNELENTRY entry = NULL;
5935 PVIRTUALCHANNELENTRY pvce = freerdp_load_channel_addin_entry(
5936 name, NULL, NULL, FREERDP_ADDIN_CHANNEL_STATIC | FREERDP_ADDIN_CHANNEL_ENTRYEX);
5937 PVIRTUALCHANNELENTRYEX pvceex = WINPR_FUNC_PTR_CAST(pvce, PVIRTUALCHANNELENTRYEX);
5940 entry = freerdp_load_channel_addin_entry(name, NULL, NULL, FREERDP_ADDIN_CHANNEL_STATIC);
5944 if (freerdp_channels_client_load_ex(channels, settings, pvceex, data) == 0)
5946 WLog_DBG(TAG,
"loading channelEx %s", name);
5952 if (freerdp_channels_client_load(channels, settings, entry, data) == 0)
5954 WLog_DBG(TAG,
"loading channel %s", name);
5964 FreeRDP_Settings_Keys_Bool settingId;
5965 const char* channelName;
5969BOOL freerdp_client_load_addins(rdpChannels* channels, rdpSettings* settings)
5971 ChannelToLoad dynChannels[] = {
5972#if defined(CHANNEL_AINPUT_CLIENT)
5973 { FreeRDP_BOOL_UNUSED, AINPUT_CHANNEL_NAME, NULL },
5975 { FreeRDP_AudioCapture, AUDIN_CHANNEL_NAME, NULL },
5976 { FreeRDP_AudioPlayback, RDPSND_CHANNEL_NAME, NULL },
5977#ifdef CHANNEL_RDPEI_CLIENT
5978 { FreeRDP_MultiTouchInput, RDPEI_CHANNEL_NAME, NULL },
5980 { FreeRDP_SupportGraphicsPipeline, RDPGFX_CHANNEL_NAME, NULL },
5981 { FreeRDP_SupportEchoChannel, ECHO_CHANNEL_NAME, NULL },
5982 { FreeRDP_SupportSSHAgentChannel,
"sshagent", NULL },
5983 { FreeRDP_SupportDisplayControl, DISP_CHANNEL_NAME, NULL },
5984 { FreeRDP_SupportGeometryTracking, GEOMETRY_CHANNEL_NAME, NULL },
5985 { FreeRDP_SupportVideoOptimized, VIDEO_CHANNEL_NAME, NULL },
5986 { FreeRDP_RemoteCredentialGuard, RDPEAR_CHANNEL_NAME, NULL },
5989 ChannelToLoad staticChannels[] = {
5990 { FreeRDP_AudioPlayback, RDPSND_CHANNEL_NAME, NULL },
5991 { FreeRDP_RedirectClipboard, CLIPRDR_SVC_CHANNEL_NAME, NULL },
5992#if defined(CHANNEL_ENCOMSP_CLIENT)
5993 { FreeRDP_EncomspVirtualChannel, ENCOMSP_SVC_CHANNEL_NAME, settings },
5995 { FreeRDP_RemdeskVirtualChannel, REMDESK_SVC_CHANNEL_NAME, settings },
5996 { FreeRDP_RemoteApplicationMode, RAIL_SVC_CHANNEL_NAME, settings }
6002 for (
size_t i = 0; i < ARRAYSIZE(dynChannels); i++)
6004 if ((dynChannels[i].settingId == FreeRDP_BOOL_UNUSED) ||
6007 const char*
const p[] = { dynChannels[i].channelName };
6009 if (!freerdp_client_add_dynamic_channel(settings, ARRAYSIZE(p), p))
6017 if ((freerdp_static_channel_collection_find(settings, RDPSND_CHANNEL_NAME)) ||
6018 (freerdp_dynamic_channel_collection_find(settings, RDPSND_CHANNEL_NAME))
6019#
if defined(CHANNEL_TSMF_CLIENT)
6020 || (freerdp_dynamic_channel_collection_find(settings,
"tsmf"))
6030 if (freerdp_dynamic_channel_collection_find(settings, AUDIN_CHANNEL_NAME))
6046 if (DrivesToRedirect && (strlen(DrivesToRedirect) != 0))
6056 char* context = NULL;
6058 value = _strdup(DrivesToRedirect);
6062 tok = strtok_s(value,
";", &context);
6065 WLog_ERR(TAG,
"DrivesToRedirect contains invalid data: '%s'", DrivesToRedirect);
6082 const char* name = NULL;
6083 const char* drive = tok;
6084 char* subcontext = NULL;
6085 char* start = strtok_s(tok,
"(", &subcontext);
6086 char* end = strtok_s(NULL,
")", &subcontext);
6090 if (freerdp_path_valid(name, NULL) && freerdp_path_valid(drive, NULL))
6092 success = freerdp_client_add_drive(settings, name, NULL);
6094 success = freerdp_client_add_drive(settings, drive, NULL);
6097 success = freerdp_client_add_drive(settings, drive, name);
6105 tok = strtok_s(NULL,
";", &context);
6114 if (!freerdp_device_collection_find(settings,
"drive"))
6116 const char*
const params[] = {
"drive",
"media",
"*" };
6118 if (!freerdp_client_add_device_channel(settings, ARRAYSIZE(params), params))
6135 if (!freerdp_device_collection_find(settings,
"drive"))
6137 const char* params[] = {
"drive",
"home",
"%" };
6139 if (!freerdp_client_add_device_channel(settings, ARRAYSIZE(params), params))
6146 if (!freerdp_client_load_static_channel_addin(channels, settings, RDPDR_SVC_CHANNEL_NAME,
6150 if (!freerdp_static_channel_collection_find(settings, RDPSND_CHANNEL_NAME) &&
6151 !freerdp_dynamic_channel_collection_find(settings, RDPSND_CHANNEL_NAME))
6153 const char*
const params[] = { RDPSND_CHANNEL_NAME,
"sys:fake" };
6155 if (!freerdp_client_add_static_channel(settings, ARRAYSIZE(params), params))
6158 if (!freerdp_client_add_dynamic_channel(settings, ARRAYSIZE(params), params))
6165 if (!freerdp_device_collection_find_type(settings, RDPDR_DTYP_SMARTCARD))
6167 RDPDR_DEVICE* smartcard = freerdp_device_new(RDPDR_DTYP_SMARTCARD, 0, NULL);
6172 if (!freerdp_device_collection_add(settings, smartcard))
6174 freerdp_device_free(smartcard);
6182 if (!freerdp_device_collection_find_type(settings, RDPDR_DTYP_PRINT))
6184 RDPDR_DEVICE* printer = freerdp_device_new(RDPDR_DTYP_PRINT, 0, NULL);
6189 if (!freerdp_device_collection_add(settings, printer))
6191 freerdp_device_free(printer);
6218 for (
size_t i = 0; i < ARRAYSIZE(staticChannels); i++)
6220 if ((staticChannels[i].settingId == 0) ||
6223 if (staticChannels[i].args)
6225 if (!freerdp_client_load_static_channel_addin(
6226 channels, settings, staticChannels[i].channelName, staticChannels[i].args))
6231 const char*
const p[] = { staticChannels[i].channelName };
6232 if (!freerdp_client_add_static_channel(settings, ARRAYSIZE(p), p))
6242 const char*
const p[] = { RDP2TCP_DVC_CHANNEL_NAME, RDP2TCPArgs };
6243 if (!freerdp_client_add_static_channel(settings, ARRAYSIZE(p), p))
6252 freerdp_settings_get_pointer_array_writable(settings, FreeRDP_StaticChannelArray, i);
6254 if (!freerdp_client_load_static_channel_addin(channels, settings, _args->argv[0], _args))
6266 if (!freerdp_client_load_static_channel_addin(channels, settings, DRDYNVC_SVC_CHANNEL_NAME,
6274void freerdp_client_warn_unmaintained(
int argc,
char* argv[])
6276 const char* app = (argc > 0) ? argv[0] :
"INVALID_ARGV";
6277 const DWORD log_level = WLOG_WARN;
6278 wLog* log = WLog_Get(TAG);
6281 if (!WLog_IsLevelActive(log, log_level))
6284 WLog_Print_unchecked(log, log_level,
"[unmaintained] %s client is currently unmaintained!",
6286 WLog_Print_unchecked(
6288 " If problems occur please check https://github.com/FreeRDP/FreeRDP/issues for "
6290 WLog_Print_unchecked(
6292 "Be prepared to fix issues yourself though as nobody is actively working on this.");
6293 WLog_Print_unchecked(
6295 " Developers hang out in https://matrix.to/#/#FreeRDP:matrix.org?via=matrix.org "
6296 "- don't hesitate to ask some questions. (replies might take some time depending "
6297 "on your timezone) - if you intend using this component write us a message");
6300void freerdp_client_warn_experimental(
int argc,
char* argv[])
6302 const char* app = (argc > 0) ? argv[0] :
"INVALID_ARGV";
6303 const DWORD log_level = WLOG_WARN;
6304 wLog* log = WLog_Get(TAG);
6307 if (!WLog_IsLevelActive(log, log_level))
6310 WLog_Print_unchecked(log, log_level,
"[experimental] %s client is currently experimental!",
6312 WLog_Print_unchecked(
6314 " If problems occur please check https://github.com/FreeRDP/FreeRDP/issues for "
6315 "known issues or create a new one!");
6316 WLog_Print_unchecked(
6318 " Developers hang out in https://matrix.to/#/#FreeRDP:matrix.org?via=matrix.org "
6319 "- don't hesitate to ask some questions. (replies might take some time depending "
6320 "on your timezone)");
6323void freerdp_client_warn_deprecated(
int argc,
char* argv[])
6325 const char* app = (argc > 0) ? argv[0] :
"INVALID_ARGV";
6326 const DWORD log_level = WLOG_WARN;
6327 wLog* log = WLog_Get(TAG);
6330 if (!WLog_IsLevelActive(log, log_level))
6333 WLog_Print_unchecked(log, log_level,
"[deprecated] %s client has been deprecated", app);
6334 WLog_Print_unchecked(log, log_level,
"As replacement there is a SDL3 based client available.");
6335 WLog_Print_unchecked(
6337 "If you are interested in keeping %s alive get in touch with the developers", app);
6338 WLog_Print_unchecked(
6340 "The project is hosted at https://github.com/freerdp/freerdp and "
6341 " developers hang out in https://matrix.to/#/#FreeRDP:matrix.org?via=matrix.org "
6342 "- don't hesitate to ask some questions. (replies might take some time depending "
6343 "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.