FreeRDP
Loading...
Searching...
No Matches
client/common/cmdline.c
1
22#include <freerdp/config.h>
23
24#include <ctype.h>
25#include <errno.h>
26
27#include <winpr/assert.h>
28#include <winpr/string.h>
29#include <winpr/crt.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>
35
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>
57
58#if defined(CHANNEL_AINPUT_CLIENT)
59#include <freerdp/channels/ainput.h>
60#endif
61
62#include <freerdp/channels/audin.h>
63#include <freerdp/channels/echo.h>
64
65#include <freerdp/client/cmdline.h>
66#include <freerdp/version.h>
67#include <freerdp/client/utils/smartcard_cli.h>
68
69#include <openssl/tls1.h>
70#include "cmdline.h"
71
72#include <freerdp/log.h>
73#define TAG CLIENT_TAG("common.cmdline")
74
75static const char str_force[] = "force";
76
77static const char* credential_args[] = { "p", "smartcard-logon",
78#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
79 "gp", "gat",
80#endif
81 "pth", "reconnect-cookie",
82 "assistance" };
83
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);
87
88static BOOL freerdp_client_print_codepages(const char* arg)
89{
90 size_t count = 0;
91 DWORD column = 2;
92 const char* filter = NULL;
93 RDP_CODEPAGE* pages = NULL;
94
95 if (arg)
96 {
97 filter = strchr(arg, ',');
98 if (!filter)
99 filter = arg;
100 else
101 filter++;
102 }
103 pages = freerdp_keyboard_get_matching_codepages(column, filter, &count);
104 if (!pages)
105 return TRUE;
106
107 printf("%-10s %-8s %-60s %-36s %-48s\n", "<id>", "<locale>", "<win langid>", "<language>",
108 "<country>");
109 for (size_t x = 0; x < count; x++)
110 {
111 const RDP_CODEPAGE* page = &pages[x];
112 char buffer[2048] = { 0 };
113
114 if (strnlen(page->subLanguageSymbol, ARRAYSIZE(page->subLanguageSymbol)) > 0)
115 (void)_snprintf(buffer, sizeof(buffer), "[%s|%s]", page->primaryLanguageSymbol,
116 page->subLanguageSymbol);
117 else
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);
121 }
122 freerdp_codepages_free(pages);
123 return TRUE;
124}
125
126static BOOL freerdp_path_valid(const char* path, BOOL* special)
127{
128 const char DynamicDrives[] = "DynamicDrives";
129 BOOL isPath = FALSE;
130 BOOL isSpecial = 0;
131 if (!path)
132 return FALSE;
133
134 isSpecial =
135 (option_equals("*", path) || option_equals(DynamicDrives, path) || option_equals("%", path))
136 ? TRUE
137 : FALSE;
138 if (!isSpecial)
139 isPath = winpr_PathFileExists(path);
140
141 if (special)
142 *special = isSpecial;
143
144 return isSpecial || isPath;
145}
146
147static BOOL freerdp_sanitize_drive_name(char* name, const char* invalid, const char* replacement)
148{
149 if (!name || !invalid || !replacement)
150 return FALSE;
151 if (strlen(invalid) != strlen(replacement))
152 return FALSE;
153
154 while (*invalid != '\0')
155 {
156 const char what = *invalid++;
157 const char with = *replacement++;
158
159 char* cur = name;
160 while ((cur = strchr(cur, what)) != NULL)
161 *cur = with;
162 }
163 return TRUE;
164}
165
166static char* name_from_path(const char* path)
167{
168 const char* name = "NULL";
169 if (path)
170 {
171 if (option_equals("%", path))
172 name = "home";
173 else if (option_equals("*", path))
174 name = "hotplug-all";
175 else if (option_equals("DynamicDrives", path))
176 name = "hotplug";
177 else
178 name = path;
179 }
180 return _strdup(name);
181}
182
183static BOOL freerdp_client_add_drive(rdpSettings* settings, const char* path, const char* name)
184{
185 char* dname = NULL;
186 RDPDR_DEVICE* device = NULL;
187
188 if (name)
189 {
190 BOOL skip = FALSE;
191 if (path)
192 {
193 switch (path[0])
194 {
195 case '*':
196 case '%':
197 skip = TRUE;
198 break;
199 default:
200 break;
201 }
202 }
203 /* Path was entered as secondary argument, swap */
204 if (!skip && winpr_PathFileExists(name))
205 {
206 if (!winpr_PathFileExists(path) || (!PathIsRelativeA(name) && PathIsRelativeA(path)))
207 {
208 const char* tmp = path;
209 path = name;
210 name = tmp;
211 }
212 }
213 }
214
215 if (name)
216 dname = _strdup(name);
217 else /* We need a name to send to the server. */
218 dname = name_from_path(path);
219
220 if (freerdp_sanitize_drive_name(dname, "\\/", "__"))
221 {
222 const char* args[] = { dname, path };
223 device = freerdp_device_new(RDPDR_DTYP_FILESYSTEM, ARRAYSIZE(args), args);
224 }
225 free(dname);
226 if (!device)
227 goto fail;
228
229 if (!path)
230 goto fail;
231
232 {
233 BOOL isSpecial = FALSE;
234 BOOL isPath = freerdp_path_valid(path, &isSpecial);
235
236 if (!isPath && !isSpecial)
237 {
238 WLog_WARN(TAG, "Invalid drive to redirect: '%s' does not exist, skipping.", path);
239 freerdp_device_free(device);
240 }
241 else if (!freerdp_device_collection_add(settings, device))
242 goto fail;
243 }
244
245 return TRUE;
246
247fail:
248 freerdp_device_free(device);
249 return FALSE;
250}
251
252static BOOL value_to_int(const char* value, LONGLONG* result, LONGLONG min, LONGLONG max)
253{
254 long long rc = 0;
255
256 if (!value || !result)
257 return FALSE;
258
259 errno = 0;
260 rc = _strtoi64(value, NULL, 0);
261
262 if (errno != 0)
263 return FALSE;
264
265 if ((rc < min) || (rc > max))
266 return FALSE;
267
268 *result = rc;
269 return TRUE;
270}
271
272static BOOL value_to_uint(const char* value, ULONGLONG* result, ULONGLONG min, ULONGLONG max)
273{
274 unsigned long long rc = 0;
275
276 if (!value || !result)
277 return FALSE;
278
279 errno = 0;
280 rc = _strtoui64(value, NULL, 0);
281
282 if (errno != 0)
283 return FALSE;
284
285 if ((rc < min) || (rc > max))
286 return FALSE;
287
288 *result = rc;
289 return TRUE;
290}
291
292BOOL freerdp_client_print_version(void)
293{
294 printf("This is FreeRDP version %s (%s)\n", FREERDP_VERSION_FULL, FREERDP_GIT_REVISION);
295 return TRUE;
296}
297
298BOOL freerdp_client_print_version_ex(int argc, char** argv)
299{
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);
305 return TRUE;
306}
307
308BOOL freerdp_client_print_buildconfig(void)
309{
310 printf("%s", freerdp_get_build_config());
311 return TRUE;
312}
313
314BOOL freerdp_client_print_buildconfig_ex(int argc, char** argv)
315{
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());
320 return TRUE;
321}
322
323static void freerdp_client_print_scancodes(void)
324{
325 printf("RDP scancodes and their name for use with /kbd:remap\n");
326
327 for (UINT32 x = 0; x < UINT16_MAX; x++)
328 {
329 const char* name = freerdp_keyboard_scancode_name(x);
330 if (name)
331 printf("0x%04" PRIx32 " --> %s\n", x, name);
332 }
333}
334
335static BOOL is_delimiter(char c, const char* delimiters)
336{
337 char d = 0;
338 while ((d = *delimiters++) != '\0')
339 {
340 if (c == d)
341 return TRUE;
342 }
343 return FALSE;
344}
345
346static const char* get_last(const char* start, size_t len, const char* delimiters)
347{
348 const char* last = NULL;
349 for (size_t x = 0; x < len; x++)
350 {
351 char c = start[x];
352 if (is_delimiter(c, delimiters))
353 last = &start[x];
354 }
355 return last;
356}
357
358static SSIZE_T next_delimiter(const char* text, size_t len, size_t max, const char* delimiters)
359{
360 if (len < max)
361 return -1;
362
363 const char* last = get_last(text, max, delimiters);
364 if (!last)
365 return -1;
366
367 return (SSIZE_T)(last - text);
368}
369
370static SSIZE_T forced_newline_at(const char* text, size_t len, size_t limit,
371 const char* force_newline)
372{
373 char d = 0;
374 while ((d = *force_newline++) != '\0')
375 {
376 const char* tok = strchr(text, d);
377 if (tok)
378 {
379 const size_t offset = WINPR_ASSERTING_INT_CAST(size_t, tok - text);
380 if ((offset > len) || (offset > limit))
381 continue;
382 return (SSIZE_T)(offset);
383 }
384 }
385 return -1;
386}
387
388static BOOL print_align(size_t start_offset, size_t* current)
389{
390 WINPR_ASSERT(current);
391 if (*current < start_offset)
392 {
393 const int rc = printf("%*c", (int)(start_offset - *current), ' ');
394 if (rc < 0)
395 return FALSE;
396 *current += (size_t)rc;
397 }
398 return TRUE;
399}
400
401static char* print_token(char* text, size_t start_offset, size_t* current, size_t limit,
402 const char* delimiters, const char* force_newline)
403{
404 int rc = 0;
405 const size_t tlen = strnlen(text, limit);
406 size_t len = tlen;
407 const SSIZE_T force_at = forced_newline_at(text, len, limit - *current, force_newline);
408 BOOL isForce = (force_at >= 0);
409
410 if (isForce)
411 len = MIN(len, (size_t)force_at);
412
413 if (!print_align(start_offset, current))
414 return NULL;
415
416 const SSIZE_T delim = next_delimiter(text, len, limit - *current, delimiters);
417 const BOOL isDelim = delim > 0;
418 if (isDelim)
419 {
420 len = MIN(len, (size_t)delim + 1);
421 }
422
423 rc = printf("%.*s", (int)len, text);
424 if (rc < 0)
425 return NULL;
426
427 if (isForce || isDelim)
428 {
429 printf("\n");
430 *current = 0;
431
432 const size_t offset = len + ((isForce && (force_at == 0)) ? 1 : 0);
433 return &text[offset];
434 }
435
436 *current += (size_t)rc;
437
438 if (tlen == (size_t)rc)
439 return NULL;
440 return &text[(size_t)rc];
441}
442
443static size_t print_optionals(const char* text, size_t start_offset, size_t current)
444{
445 const size_t limit = 80;
446 char* str = _strdup(text);
447 char* cur = str;
448
449 do
450 {
451 cur = print_token(cur, start_offset + 1, &current, limit, "[], ", "\r\n");
452 } while (cur != NULL);
453
454 free(str);
455 return current;
456}
457
458static size_t print_description(const char* text, size_t start_offset, size_t current)
459{
460 const size_t limit = 80;
461 char* str = _strdup(text);
462 char* cur = str;
463
464 while (cur != NULL)
465 cur = print_token(cur, start_offset, &current, limit, " ", "\r\n");
466
467 free(str);
468 const int rc = printf("\n");
469 if (rc >= 0)
470 {
471 const size_t src = WINPR_ASSERTING_INT_CAST(size_t, rc);
472 WINPR_ASSERT(SIZE_MAX - src > current);
473 current += src;
474 }
475 return current;
476}
477
478static int cmp_cmdline_args(const void* pva, const void* pvb)
479{
482
483 if (!a->Name && !b->Name)
484 return 0;
485 if (!a->Name)
486 return 1;
487 if (!b->Name)
488 return -1;
489 return strcmp(a->Name, b->Name);
490}
491
492static void freerdp_client_print_command_line_args(COMMAND_LINE_ARGUMENT_A* parg, size_t count)
493{
494 if (!parg)
495 return;
496
497 qsort(parg, count, sizeof(COMMAND_LINE_ARGUMENT_A), cmp_cmdline_args);
498
499 const COMMAND_LINE_ARGUMENT_A* arg = parg;
500 do
501 {
502 int rc = 0;
503 size_t pos = 0;
504 const size_t description_offset = 30 + 8;
505
506 if (arg->Flags & (COMMAND_LINE_VALUE_BOOL | COMMAND_LINE_VALUE_FLAG))
507 {
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);
512 else
513 {
514 rc = printf(" %s%s", arg->Default ? "-" : "+", arg->Name);
515 }
516 }
517 else
518 rc = printf(" /%s", arg->Name);
519
520 if (rc < 0)
521 return;
522 pos += (size_t)rc;
523
524 if ((arg->Flags & COMMAND_LINE_VALUE_REQUIRED) ||
525 (arg->Flags & COMMAND_LINE_VALUE_OPTIONAL))
526 {
527 if (arg->Format)
528 {
529 if (arg->Flags & COMMAND_LINE_VALUE_OPTIONAL)
530 {
531 rc = printf("[:");
532 if (rc < 0)
533 return;
534 pos += (size_t)rc;
535 pos = print_optionals(arg->Format, pos, pos);
536 rc = printf("]");
537 if (rc < 0)
538 return;
539 pos += (size_t)rc;
540 }
541 else
542 {
543 rc = printf(":");
544 if (rc < 0)
545 return;
546 pos += (size_t)rc;
547 pos = print_optionals(arg->Format, pos, pos);
548 }
549
550 if (pos > description_offset)
551 {
552 printf("\n");
553 pos = 0;
554 }
555 }
556 }
557
558 rc = printf("%*c", (int)(description_offset - pos), ' ');
559 if (rc < 0)
560 return;
561 pos += (size_t)rc;
562
563 if (arg->Flags & COMMAND_LINE_VALUE_BOOL)
564 {
565 rc = printf("%s ", arg->Default ? "Disable" : "Enable");
566 if (rc < 0)
567 return;
568 pos += (size_t)rc;
569 }
570
571 print_description(arg->Text, description_offset, pos);
572 } while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
573}
574
575BOOL freerdp_client_print_command_line_help(int argc, char** argv)
576{
577 return freerdp_client_print_command_line_help_ex(argc, argv, NULL);
578}
579
580static COMMAND_LINE_ARGUMENT_A* create_merged_args(const COMMAND_LINE_ARGUMENT_A* custom,
581 SSIZE_T count, size_t* pcount)
582{
583 WINPR_ASSERT(pcount);
584 if (count < 0)
585 {
586 const COMMAND_LINE_ARGUMENT_A* cur = custom;
587 count = 0;
588 while (cur && cur->Name)
589 {
590 count++;
591 cur++;
592 }
593 }
594
596 calloc((size_t)count + ARRAYSIZE(global_cmd_args), sizeof(COMMAND_LINE_ARGUMENT_A));
597 *pcount = 0;
598 if (!largs)
599 return NULL;
600
601 size_t lcount = 0;
602 const COMMAND_LINE_ARGUMENT_A* cur = custom;
603 while (cur && cur->Name)
604 {
605 largs[lcount++] = *cur++;
606 }
607
608 cur = global_cmd_args;
609 while (cur && cur->Name)
610 {
611 largs[lcount++] = *cur++;
612 }
613 *pcount = lcount;
614 return largs;
615}
616
617BOOL freerdp_client_print_command_line_help_ex(int argc, char** argv,
618 const COMMAND_LINE_ARGUMENT_A* custom)
619{
620 const char* name = "FreeRDP";
621
622 /* allocate a merged copy of implementation defined and default arguments */
623 size_t lcount = 0;
624 COMMAND_LINE_ARGUMENT_A* largs = create_merged_args(custom, -1, &lcount);
625 if (!largs)
626 return FALSE;
627
628 if (argc > 0)
629 name = argv[0];
630
631 printf("\n");
632 printf("FreeRDP - A Free Remote Desktop Protocol Implementation\n");
633 printf("See www.freerdp.com for more information\n");
634 printf("\n");
635 printf("Usage: %s [file] [options] [/v:<server>[:port]]\n", argv[0]);
636 printf("\n");
637 printf("Syntax:\n");
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");
641 printf("\n");
642
643 freerdp_client_print_command_line_args(largs, lcount);
644 free(largs);
645
646 printf("\n");
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",
653 name);
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);
659 printf("\n");
660 printf("Clipboard Redirection: +clipboard\n");
661 printf("\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");
665
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");
669#endif
670#if defined(CHANNEL_PARALLEL_CLIENT)
671 printf("Parallel Port Redirection: /parallel:<name>,<device>\n");
672#endif
673 printf("Printer Redirection: /printer:<device>,<driver>,[default]\n");
674 printf("TCP redirection: /rdp2tcp:/usr/bin/rdp2tcp\n");
675 printf("\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");
680 printf("\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");
684#endif
685 printf("\n");
686 printf("For Gateways, the https_proxy environment variable is respected:\n");
687#ifdef _WIN32
688 printf(" set HTTPS_PROXY=http://proxy.contoso.com:3128/\n");
689#else
690 printf(" export https_proxy=http://proxy.contoso.com:3128/\n");
691#endif
692 printf(" %s /gateway:g:rdp.contoso.com ...\n", name);
693 printf("\n");
694 printf("More documentation is coming, in the meantime consult source files\n");
695 printf("\n");
696 return TRUE;
697}
698
699static BOOL option_is_rdp_file(const char* option)
700{
701 WINPR_ASSERT(option);
702
703 if (option_ends_with(option, ".rdp"))
704 return TRUE;
705 if (option_ends_with(option, ".rdpw"))
706 return TRUE;
707 return FALSE;
708}
709
710static BOOL option_is_incident_file(const char* option)
711{
712 WINPR_ASSERT(option);
713
714 if (option_ends_with(option, ".msrcIncident"))
715 return TRUE;
716 return FALSE;
717}
718
719static int freerdp_client_command_line_pre_filter(void* context, int index, int argc, LPSTR* argv)
720{
721 if (index == 1)
722 {
723 size_t length = 0;
724 rdpSettings* settings = NULL;
725
726 if (argc <= index)
727 return -1;
728
729 length = strlen(argv[index]);
730
731 if (length > 4)
732 {
733 if (option_is_rdp_file(argv[index]))
734 {
735 settings = (rdpSettings*)context;
736
737 if (!freerdp_settings_set_string(settings, FreeRDP_ConnectionFile, argv[index]))
738 return COMMAND_LINE_ERROR_MEMORY;
739
740 return 1;
741 }
742 }
743
744 if (length > 13)
745 {
746 if (option_is_incident_file(argv[index]))
747 {
748 settings = (rdpSettings*)context;
749
750 if (!freerdp_settings_set_string(settings, FreeRDP_AssistanceFile, argv[index]))
751 return COMMAND_LINE_ERROR_MEMORY;
752
753 return 1;
754 }
755 }
756 }
757
758 return 0;
759}
760
761BOOL freerdp_client_add_device_channel(rdpSettings* settings, size_t count,
762 const char* const* params)
763{
764 WINPR_ASSERT(settings);
765 WINPR_ASSERT(params);
766 WINPR_ASSERT(count > 0);
767
768 if (option_equals(params[0], "drive"))
769 {
770 BOOL rc = 0;
771 if (count < 2)
772 return FALSE;
773
774 if (!freerdp_settings_set_bool(settings, FreeRDP_DeviceRedirection, TRUE))
775 return FALSE;
776 if (count < 3)
777 rc = freerdp_client_add_drive(settings, params[1], NULL);
778 else
779 rc = freerdp_client_add_drive(settings, params[2], params[1]);
780
781 return rc;
782 }
783 else if (option_equals(params[0], "printer"))
784 {
785 RDPDR_DEVICE* printer = NULL;
786
787 if (count < 1)
788 return FALSE;
789
790 if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectPrinters, TRUE))
791 return FALSE;
792 if (!freerdp_settings_set_bool(settings, FreeRDP_DeviceRedirection, TRUE))
793 return FALSE;
794
795 printer = freerdp_device_new(RDPDR_DTYP_PRINT, count - 1, &params[1]);
796 if (!printer)
797 return FALSE;
798
799 if (!freerdp_device_collection_add(settings, printer))
800 {
801 freerdp_device_free(printer);
802 return FALSE;
803 }
804
805 return TRUE;
806 }
807 else if (option_equals(params[0], "smartcard"))
808 {
809 RDPDR_DEVICE* smartcard = NULL;
810
811 if (count < 1)
812 return FALSE;
813
814 if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectSmartCards, TRUE))
815 return FALSE;
816 if (!freerdp_settings_set_bool(settings, FreeRDP_DeviceRedirection, TRUE))
817 return FALSE;
818
819 smartcard = freerdp_device_new(RDPDR_DTYP_SMARTCARD, count - 1, &params[1]);
820
821 if (!smartcard)
822 return FALSE;
823
824 if (!freerdp_device_collection_add(settings, smartcard))
825 {
826 freerdp_device_free(smartcard);
827 return FALSE;
828 }
829
830 return TRUE;
831 }
832#if defined(CHANNEL_SERIAL_CLIENT)
833 else if (option_equals(params[0], "serial"))
834 {
835 RDPDR_DEVICE* serial = NULL;
836
837 if (count < 1)
838 return FALSE;
839
840 if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectSerialPorts, TRUE))
841 return FALSE;
842 if (!freerdp_settings_set_bool(settings, FreeRDP_DeviceRedirection, TRUE))
843 return FALSE;
844
845 serial = freerdp_device_new(RDPDR_DTYP_SERIAL, count - 1, &params[1]);
846
847 if (!serial)
848 return FALSE;
849
850 if (!freerdp_device_collection_add(settings, serial))
851 {
852 freerdp_device_free(serial);
853 return FALSE;
854 }
855
856 return TRUE;
857 }
858#endif
859 else if (option_equals(params[0], "parallel"))
860 {
861 RDPDR_DEVICE* parallel = NULL;
862
863 if (count < 1)
864 return FALSE;
865
866 if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectParallelPorts, TRUE))
867 return FALSE;
868 if (!freerdp_settings_set_bool(settings, FreeRDP_DeviceRedirection, TRUE))
869 return FALSE;
870
871 parallel = freerdp_device_new(RDPDR_DTYP_PARALLEL, count - 1, &params[1]);
872
873 if (!parallel)
874 return FALSE;
875
876 if (!freerdp_device_collection_add(settings, parallel))
877 {
878 freerdp_device_free(parallel);
879 return FALSE;
880 }
881
882 return TRUE;
883 }
884
885 return FALSE;
886}
887
888BOOL freerdp_client_del_static_channel(rdpSettings* settings, const char* name)
889{
890 return freerdp_static_channel_collection_del(settings, name);
891}
892
893BOOL freerdp_client_add_static_channel(rdpSettings* settings, size_t count,
894 const char* const* params)
895{
896 ADDIN_ARGV* _args = NULL;
897
898 if (!settings || !params || !params[0] || (count > INT_MAX))
899 return FALSE;
900
901 if (freerdp_static_channel_collection_find(settings, params[0]))
902 return TRUE;
903
904 _args = freerdp_addin_argv_new(count, params);
905
906 if (!_args)
907 return FALSE;
908
909 if (!freerdp_static_channel_collection_add(settings, _args))
910 goto fail;
911
912 return TRUE;
913fail:
914 freerdp_addin_argv_free(_args);
915 return FALSE;
916}
917
918BOOL freerdp_client_del_dynamic_channel(rdpSettings* settings, const char* name)
919{
920 return freerdp_dynamic_channel_collection_del(settings, name);
921}
922
923BOOL freerdp_client_add_dynamic_channel(rdpSettings* settings, size_t count,
924 const char* const* params)
925{
926 ADDIN_ARGV* _args = NULL;
927
928 if (!settings || !params || !params[0] || (count > INT_MAX))
929 return FALSE;
930
931 if (freerdp_dynamic_channel_collection_find(settings, params[0]))
932 return TRUE;
933
934 _args = freerdp_addin_argv_new(count, params);
935
936 if (!_args)
937 return FALSE;
938
939 if (!freerdp_dynamic_channel_collection_add(settings, _args))
940 goto fail;
941
942 return TRUE;
943
944fail:
945 freerdp_addin_argv_free(_args);
946 return FALSE;
947}
948
949static BOOL read_pem_file(rdpSettings* settings, FreeRDP_Settings_Keys_String id, const char* file)
950{
951 size_t length = 0;
952 char* pem = crypto_read_pem(file, &length);
953 if (!pem || (length == 0))
954 {
955 free(pem);
956 return FALSE;
957 }
958
959 BOOL rc = freerdp_settings_set_string_len(settings, id, pem, length);
960 free(pem);
961 return rc;
962}
963
965typedef enum
966{
967 CMDLINE_SUBOPTION_STRING,
968 CMDLINE_SUBOPTION_FILE,
969} CmdLineSubOptionType;
970
971typedef BOOL (*CmdLineSubOptionCb)(const char* value, rdpSettings* settings);
972typedef struct
973{
974 const char* optname;
975 FreeRDP_Settings_Keys_String id;
976 CmdLineSubOptionType opttype;
977 CmdLineSubOptionCb cb;
978} CmdLineSubOptions;
979
980static BOOL parseSubOptions(rdpSettings* settings, const CmdLineSubOptions* opts, size_t count,
981 const char* arg)
982{
983 BOOL found = FALSE;
984
985 for (size_t xx = 0; xx < count; xx++)
986 {
987 const CmdLineSubOptions* opt = &opts[xx];
988
989 if (option_starts_with(opt->optname, arg))
990 {
991 const size_t optlen = strlen(opt->optname);
992 const char* val = &arg[optlen];
993 BOOL status = 0;
994
995 switch (opt->opttype)
996 {
997 case CMDLINE_SUBOPTION_STRING:
998 status = freerdp_settings_set_string(settings, opt->id, val);
999 break;
1000 case CMDLINE_SUBOPTION_FILE:
1001 status = read_pem_file(settings, opt->id, val);
1002 break;
1003 default:
1004 WLog_ERR(TAG, "invalid subOption type");
1005 return FALSE;
1006 }
1007
1008 if (!status)
1009 return FALSE;
1010
1011 if (opt->cb && !opt->cb(val, settings))
1012 return FALSE;
1013
1014 found = TRUE;
1015 break;
1016 }
1017 }
1018
1019 if (!found)
1020 WLog_ERR(TAG, "option %s not handled", arg);
1021
1022 return found;
1023}
1024
1025#define fail_at(arg, rc) fail_at_((arg), (rc), __FILE__, __func__, __LINE__)
1026static int fail_at_(const COMMAND_LINE_ARGUMENT_A* arg, int rc, const char* file, const char* fkt,
1027 size_t line)
1028{
1029 if (rc == 0)
1030 return rc;
1031
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,
1037 arg->Value, rc);
1038 return rc;
1039}
1040
1041static int freerdp_client_command_line_post_filter_int(void* context, COMMAND_LINE_ARGUMENT_A* arg)
1042{
1043 rdpSettings* settings = (rdpSettings*)context;
1044 int status = CHANNEL_RC_OK;
1045 BOOL enable = arg->Value ? TRUE : FALSE;
1046
1047 CommandLineSwitchStart(arg) CommandLineSwitchCase(arg, "a")
1048 {
1049 size_t count = 0;
1050 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
1051
1052 if (!freerdp_client_add_device_channel(settings, count, (const char* const*)ptr))
1053 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1054 if (!freerdp_settings_set_bool(settings, FreeRDP_DeviceRedirection, TRUE))
1055 status = COMMAND_LINE_ERROR;
1056
1057 CommandLineParserFree(ptr);
1058 if (status)
1059 return fail_at(arg, status);
1060 }
1061 CommandLineSwitchCase(arg, "kerberos")
1062 {
1063 size_t count = 0;
1064
1065 char** ptr = CommandLineParseCommaSeparatedValuesEx("kerberos", arg->Value, &count);
1066 if (ptr)
1067 {
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 }
1078 };
1079
1080 for (size_t x = 1; x < count; x++)
1081 {
1082 const char* cur = ptr[x];
1083 if (!parseSubOptions(settings, opts, ARRAYSIZE(opts), cur))
1084 {
1085 CommandLineParserFree(ptr);
1086 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
1087 }
1088 }
1089 }
1090 CommandLineParserFree(ptr);
1091 }
1092
1093 CommandLineSwitchCase(arg, "vc")
1094 {
1095 size_t count = 0;
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);
1100 if (status)
1101 return fail_at(arg, status);
1102 }
1103 CommandLineSwitchCase(arg, "dvc")
1104 {
1105 size_t count = 0;
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);
1110 if (status)
1111 return fail_at(arg, status);
1112 }
1113 CommandLineSwitchCase(arg, "drive")
1114 {
1115 size_t count = 0;
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);
1120 if (status)
1121 return fail_at(arg, status);
1122 }
1123#if defined(CHANNEL_SERIAL_CLIENT)
1124 CommandLineSwitchCase(arg, "serial")
1125 {
1126 size_t count = 0;
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);
1131 if (status)
1132 return fail_at(arg, status);
1133 }
1134#endif
1135#if defined(CHANNEL_PARALLEL_CLIENT)
1136 CommandLineSwitchCase(arg, "parallel")
1137 {
1138 size_t count = 0;
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);
1143 if (status)
1144 return fail_at(arg, status);
1145 }
1146#endif
1147 CommandLineSwitchCase(arg, "smartcard")
1148 {
1149 size_t count = 0;
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);
1154 if (status)
1155 return fail_at(arg, status);
1156 }
1157 CommandLineSwitchCase(arg, "printer")
1158 {
1159 size_t count = 0;
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);
1164 if (status)
1165 return fail_at(arg, status);
1166 }
1167 CommandLineSwitchCase(arg, "usb")
1168 {
1169 size_t count = 0;
1170 char** ptr =
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);
1175 if (status)
1176 return fail_at(arg, status);
1177 }
1178 CommandLineSwitchCase(arg, "multitouch")
1179 {
1180 if (!freerdp_settings_set_bool(settings, FreeRDP_MultiTouchInput, enable))
1181 return fail_at(arg, COMMAND_LINE_ERROR);
1182 }
1183 CommandLineSwitchCase(arg, "gestures")
1184 {
1185 if (!freerdp_settings_set_bool(settings, FreeRDP_MultiTouchGestures, enable))
1186 return fail_at(arg, COMMAND_LINE_ERROR);
1187 }
1188 CommandLineSwitchCase(arg, "echo")
1189 {
1190 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportEchoChannel, enable))
1191 return fail_at(arg, COMMAND_LINE_ERROR);
1192 }
1193 CommandLineSwitchCase(arg, "ssh-agent")
1194 {
1195 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportSSHAgentChannel, enable))
1196 return fail_at(arg, COMMAND_LINE_ERROR);
1197 }
1198 CommandLineSwitchCase(arg, "disp")
1199 {
1200 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportDisplayControl, enable))
1201 return fail_at(arg, COMMAND_LINE_ERROR);
1202 }
1203 CommandLineSwitchCase(arg, "geometry")
1204 {
1205 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportGeometryTracking, enable))
1206 return fail_at(arg, COMMAND_LINE_ERROR);
1207 }
1208 CommandLineSwitchCase(arg, "video")
1209 {
1210 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportGeometryTracking,
1211 enable)) /* this requires geometry tracking */
1212 return fail_at(arg, COMMAND_LINE_ERROR);
1213 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportVideoOptimized, enable))
1214 return fail_at(arg, COMMAND_LINE_ERROR);
1215 }
1216 CommandLineSwitchCase(arg, "sound")
1217 {
1218 size_t count = 0;
1219 char** ptr =
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;
1225
1226 CommandLineParserFree(ptr);
1227 if (status)
1228 return fail_at(arg, status);
1229 }
1230 CommandLineSwitchCase(arg, "microphone")
1231 {
1232 size_t count = 0;
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);
1237 if (status)
1238 return fail_at(arg, status);
1239 }
1240#if defined(CHANNEL_TSMF_CLIENT)
1241 CommandLineSwitchCase(arg, "multimedia")
1242 {
1243 size_t count = 0;
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);
1248 if (status)
1249 return fail_at(arg, status);
1250 }
1251#endif
1252 CommandLineSwitchCase(arg, "heartbeat")
1253 {
1254 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportHeartbeatPdu, enable))
1255 return fail_at(arg, COMMAND_LINE_ERROR);
1256 }
1257 CommandLineSwitchCase(arg, "multitransport")
1258 {
1259 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportMultitransport, enable))
1260 return fail_at(arg, COMMAND_LINE_ERROR);
1261
1262 UINT32 flags = 0;
1263 if (freerdp_settings_get_bool(settings, FreeRDP_SupportMultitransport))
1264 flags =
1265 (TRANSPORT_TYPE_UDP_FECR | TRANSPORT_TYPE_UDP_FECL | TRANSPORT_TYPE_UDP_PREFERRED);
1266
1267 if (!freerdp_settings_set_uint32(settings, FreeRDP_MultitransportFlags, flags))
1268 return fail_at(arg, COMMAND_LINE_ERROR);
1269 }
1270 CommandLineSwitchEnd(arg)
1271
1272 return status;
1273}
1274
1275static int freerdp_client_command_line_post_filter(void* context, COMMAND_LINE_ARGUMENT_A* arg)
1276{
1277 int status = freerdp_client_command_line_post_filter_int(context, arg);
1278 return status == CHANNEL_RC_OK ? 1 : -1;
1279}
1280
1281static BOOL freerdp_parse_username_ptr(const char* username, const char** user, size_t* userlen,
1282 const char** domain, size_t* domainlen)
1283{
1284 WINPR_ASSERT(user);
1285 WINPR_ASSERT(userlen);
1286 WINPR_ASSERT(domain);
1287 WINPR_ASSERT(domainlen);
1288
1289 if (!username)
1290 return FALSE;
1291
1292 const char* p = strchr(username, '\\');
1293
1294 *user = NULL;
1295 *userlen = 0;
1296
1297 *domain = NULL;
1298 *domainlen = 0;
1299
1300 if (p)
1301 {
1302 const size_t length = (size_t)(p - username);
1303 *user = &p[1];
1304 *userlen = strlen(*user);
1305
1306 *domain = username;
1307 *domainlen = length;
1308 }
1309 else
1310 {
1311 /* Do not break up the name for '@'; both credSSP and the
1312 * ClientInfo PDU expect 'user@corp.net' to be transmitted
1313 * as username 'user@corp.net', domain empty (not NULL!).
1314 */
1315 *user = username;
1316 *userlen = strlen(username);
1317 }
1318
1319 return TRUE;
1320}
1321
1322static BOOL freerdp_parse_username_settings(const char* username, rdpSettings* settings,
1323 FreeRDP_Settings_Keys_String userID,
1324 FreeRDP_Settings_Keys_String domainID)
1325{
1326 const char* user = NULL;
1327 const char* domain = NULL;
1328 size_t userlen = 0;
1329 size_t domainlen = 0;
1330
1331 const BOOL rc = freerdp_parse_username_ptr(username, &user, &userlen, &domain, &domainlen);
1332 if (!rc)
1333 return FALSE;
1334 if (!freerdp_settings_set_string_len(settings, userID, user, userlen))
1335 return FALSE;
1336 return freerdp_settings_set_string_len(settings, domainID, domain, domainlen);
1337}
1338
1339BOOL freerdp_parse_username(const char* username, char** puser, char** pdomain)
1340{
1341 const char* user = NULL;
1342 const char* domain = NULL;
1343 size_t userlen = 0;
1344 size_t domainlen = 0;
1345
1346 *puser = NULL;
1347 *pdomain = NULL;
1348
1349 const BOOL rc = freerdp_parse_username_ptr(username, &user, &userlen, &domain, &domainlen);
1350 if (!rc)
1351 return FALSE;
1352
1353 if (userlen > 0)
1354 {
1355 *puser = strndup(user, userlen);
1356 if (!*puser)
1357 return FALSE;
1358 }
1359
1360 if (domainlen > 0)
1361 {
1362 *pdomain = strndup(domain, domainlen);
1363 if (!*pdomain)
1364 {
1365 free(*puser);
1366 *puser = NULL;
1367 return FALSE;
1368 }
1369 }
1370
1371 return TRUE;
1372}
1373
1374BOOL freerdp_parse_hostname(const char* hostname, char** host, int* port)
1375{
1376 char* p = NULL;
1377 p = strrchr(hostname, ':');
1378
1379 if (p)
1380 {
1381 size_t length = (size_t)(p - hostname);
1382 LONGLONG val = 0;
1383
1384 if (!value_to_int(p + 1, &val, 1, UINT16_MAX))
1385 return FALSE;
1386
1387 *host = (char*)calloc(length + 1UL, sizeof(char));
1388
1389 if (!(*host))
1390 return FALSE;
1391
1392 CopyMemory(*host, hostname, length);
1393 (*host)[length] = '\0';
1394 *port = (UINT16)val;
1395 }
1396 else
1397 {
1398 *host = _strdup(hostname);
1399
1400 if (!(*host))
1401 return FALSE;
1402
1403 *port = -1;
1404 }
1405
1406 return TRUE;
1407}
1408
1409static BOOL freerdp_apply_connection_type(rdpSettings* settings, UINT32 type)
1410{
1411 struct network_settings
1412 {
1413 FreeRDP_Settings_Keys_Bool id;
1414 BOOL value[7];
1415 };
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 } }
1423 };
1424
1425 switch (type)
1426 {
1427 case CONNECTION_TYPE_INVALID:
1428 return TRUE;
1429
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:
1437 break;
1438 default:
1439 WLog_WARN(TAG, "Unknown ConnectionType %" PRIu32 ", aborting", type);
1440 return FALSE;
1441 }
1442
1443 for (size_t x = 0; x < ARRAYSIZE(config); x++)
1444 {
1445 const struct network_settings* cur = &config[x];
1446 if (!freerdp_settings_set_bool(settings, cur->id, cur->value[type - 1]))
1447 return FALSE;
1448 }
1449 return TRUE;
1450}
1451
1452BOOL freerdp_set_connection_type(rdpSettings* settings, UINT32 type)
1453{
1454
1455 if (!freerdp_settings_set_uint32(settings, FreeRDP_ConnectionType, type))
1456 return FALSE;
1457
1458 switch (type)
1459 {
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))
1468 return FALSE;
1469 break;
1470 case CONNECTION_TYPE_AUTODETECT:
1471 if (!freerdp_apply_connection_type(settings, type))
1472 return FALSE;
1473 /* Automatically activate GFX and RFX codec support */
1474#ifdef WITH_GFX_H264
1475 if (!freerdp_settings_set_bool(settings, FreeRDP_GfxAVC444v2, TRUE) ||
1476 !freerdp_settings_set_bool(settings, FreeRDP_GfxAVC444, TRUE) ||
1477 !freerdp_settings_set_bool(settings, FreeRDP_GfxH264, TRUE))
1478 return FALSE;
1479#endif
1480 if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteFxCodec, TRUE) ||
1481 !freerdp_settings_set_bool(settings, FreeRDP_SupportGraphicsPipeline, TRUE))
1482 return FALSE;
1483 break;
1484 default:
1485 WLog_WARN(TAG, "Unknown ConnectionType %" PRIu32 ", aborting", type);
1486 return FALSE;
1487 }
1488
1489 return TRUE;
1490}
1491
1492static UINT32 freerdp_get_keyboard_layout_for_type(const char* name, WINPR_ATTR_UNUSED DWORD type)
1493{
1494 UINT32 res = 0;
1495 size_t count = 0;
1496 RDP_KEYBOARD_LAYOUT* layouts =
1497 freerdp_keyboard_get_layouts(RDP_KEYBOARD_LAYOUT_TYPE_STANDARD, &count);
1498
1499 if (!layouts || (count == 0))
1500 goto fail;
1501
1502 for (size_t x = 0; x < count; x++)
1503 {
1504 const RDP_KEYBOARD_LAYOUT* layout = &layouts[x];
1505 if (option_equals(layout->name, name))
1506 {
1507 res = layout->code;
1508 break;
1509 }
1510 }
1511
1512fail:
1513 freerdp_keyboard_layouts_free(layouts, count);
1514 return res;
1515}
1516
1517static UINT32 freerdp_map_keyboard_layout_name_to_id(const char* name)
1518{
1519 const UINT32 variants[] = { RDP_KEYBOARD_LAYOUT_TYPE_STANDARD, RDP_KEYBOARD_LAYOUT_TYPE_VARIANT,
1520 RDP_KEYBOARD_LAYOUT_TYPE_IME };
1521
1522 for (size_t x = 0; x < ARRAYSIZE(variants); x++)
1523 {
1524 UINT32 rc = freerdp_get_keyboard_layout_for_type(name, variants[x]);
1525 if (rc > 0)
1526 return rc;
1527 }
1528
1529 return 0;
1530}
1531
1532static int freerdp_detect_command_line_pre_filter(void* context, int index, int argc, LPSTR* argv)
1533{
1534 size_t length = 0;
1535 WINPR_UNUSED(context);
1536
1537 if (index == 1)
1538 {
1539 if (argc < index)
1540 return -1;
1541
1542 length = strlen(argv[index]);
1543
1544 if (length > 4)
1545 {
1546 if (option_is_rdp_file(argv[index]))
1547 {
1548 return 1;
1549 }
1550 }
1551
1552 if (length > 13)
1553 {
1554 if (option_is_incident_file(argv[index]))
1555 {
1556 return 1;
1557 }
1558 }
1559 }
1560
1561 return 0;
1562}
1563
1564static int freerdp_detect_windows_style_command_line_syntax(int argc, char** argv, size_t* count,
1565 BOOL ignoreUnknown)
1566{
1567 int status = 0;
1568 DWORD flags = 0;
1569 int detect_status = 0;
1570 const COMMAND_LINE_ARGUMENT_A* arg = NULL;
1571 COMMAND_LINE_ARGUMENT_A largs[ARRAYSIZE(global_cmd_args)];
1572 memcpy(largs, global_cmd_args, sizeof(global_cmd_args));
1573
1574 flags = COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_SILENCE_PARSER;
1575 flags |= COMMAND_LINE_SIGIL_SLASH | COMMAND_LINE_SIGIL_PLUS_MINUS;
1576
1577 if (ignoreUnknown)
1578 {
1579 flags |= COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
1580 }
1581
1582 *count = 0;
1583 detect_status = 0;
1584 CommandLineClearArgumentsA(largs);
1585 status = CommandLineParseArgumentsA(argc, argv, largs, flags, NULL,
1586 freerdp_detect_command_line_pre_filter, NULL);
1587
1588 if (status < 0)
1589 return status;
1590
1591 arg = largs;
1592
1593 do
1594 {
1595 if (!(arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
1596 continue;
1597
1598 (*count)++;
1599 } while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
1600
1601 return detect_status;
1602}
1603
1604static int freerdp_detect_posix_style_command_line_syntax(int argc, char** argv, size_t* count,
1605 BOOL ignoreUnknown)
1606{
1607 int status = 0;
1608 DWORD flags = 0;
1609 int detect_status = 0;
1610 const COMMAND_LINE_ARGUMENT_A* arg = NULL;
1611 COMMAND_LINE_ARGUMENT_A largs[ARRAYSIZE(global_cmd_args)];
1612 memcpy(largs, global_cmd_args, sizeof(global_cmd_args));
1613
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;
1617
1618 if (ignoreUnknown)
1619 {
1620 flags |= COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
1621 }
1622
1623 *count = 0;
1624 detect_status = 0;
1625 CommandLineClearArgumentsA(largs);
1626 status = CommandLineParseArgumentsA(argc, argv, largs, flags, NULL,
1627 freerdp_detect_command_line_pre_filter, NULL);
1628
1629 if (status < 0)
1630 return status;
1631
1632 arg = largs;
1633
1634 do
1635 {
1636 if (!(arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
1637 continue;
1638
1639 (*count)++;
1640 } while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
1641
1642 return detect_status;
1643}
1644
1645static BOOL freerdp_client_detect_command_line(int argc, char** argv, DWORD* flags)
1646{
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);
1654
1655 /* Default is POSIX syntax */
1656 *flags = COMMAND_LINE_SEPARATOR_SPACE;
1657 *flags |= COMMAND_LINE_SIGIL_DASH | COMMAND_LINE_SIGIL_DOUBLE_DASH;
1658 *flags |= COMMAND_LINE_SIGIL_ENABLE_DISABLE;
1659
1660 if (posix_cli_status <= COMMAND_LINE_STATUS_PRINT)
1661 return FALSE;
1662
1663 /* Check, if this may be windows style syntax... */
1664 if ((windows_cli_count && (windows_cli_count >= posix_cli_count)) ||
1665 (windows_cli_status <= COMMAND_LINE_STATUS_PRINT))
1666 {
1667 windows_cli_count = 1;
1668 *flags = COMMAND_LINE_SEPARATOR_COLON;
1669 *flags |= COMMAND_LINE_SIGIL_SLASH | COMMAND_LINE_SIGIL_PLUS_MINUS;
1670 }
1671
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))
1675 {
1676 if ((posix_cli_status == COMMAND_LINE_ERROR) && (windows_cli_status == COMMAND_LINE_ERROR))
1677 return TRUE;
1678 }
1679 return FALSE;
1680}
1681
1682int freerdp_client_settings_command_line_status_print(rdpSettings* settings, int status, int argc,
1683 char** argv)
1684{
1685 return freerdp_client_settings_command_line_status_print_ex(settings, status, argc, argv, NULL);
1686}
1687
1688static void freerdp_client_print_keyboard_type_list(const char* msg, DWORD type)
1689{
1690 size_t count = 0;
1691 RDP_KEYBOARD_LAYOUT* layouts = NULL;
1692 layouts = freerdp_keyboard_get_layouts(type, &count);
1693
1694 printf("\n%s\n", msg);
1695
1696 for (size_t x = 0; x < count; x++)
1697 {
1698 const RDP_KEYBOARD_LAYOUT* layout = &layouts[x];
1699 printf("0x%08" PRIX32 "\t%s\n", layout->code, layout->name);
1700 }
1701
1702 freerdp_keyboard_layouts_free(layouts, count);
1703}
1704
1705static void freerdp_client_print_keyboard_list(void)
1706{
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);
1712}
1713
1714static void freerdp_client_print_timezone_list(void)
1715{
1716 DWORD index = 0;
1717 DYNAMIC_TIME_ZONE_INFORMATION info = { 0 };
1718 while (EnumDynamicTimeZoneInformation(index++, &info) != ERROR_NO_MORE_ITEMS)
1719 {
1720 char TimeZoneKeyName[ARRAYSIZE(info.TimeZoneKeyName) + 1] = { 0 };
1721
1722 (void)ConvertWCharNToUtf8(info.TimeZoneKeyName, ARRAYSIZE(info.TimeZoneKeyName),
1723 TimeZoneKeyName, ARRAYSIZE(TimeZoneKeyName));
1724 printf("%" PRIu32 ": '%s'\n", index, TimeZoneKeyName);
1725 }
1726}
1727
1728static void freerdp_client_print_tune_list(const rdpSettings* settings)
1729{
1730 SSIZE_T type = 0;
1731
1732 for (SSIZE_T x = 0; x < FreeRDP_Settings_StableAPI_MAX; x++)
1733 {
1734 const char* name = freerdp_settings_get_name_for_key(x);
1736
1737 // NOLINTBEGIN(clang-analyzer-optin.core.EnumCastOutOfRange)
1738 switch (type)
1739 {
1740 case RDP_SETTINGS_TYPE_BOOL:
1741 printf("%" PRIdz "\t%50s\tBOOL\t%s\n", x, name,
1742 freerdp_settings_get_bool(settings, (FreeRDP_Settings_Keys_Bool)x)
1743 ? "TRUE"
1744 : "FALSE");
1745 break;
1746 case RDP_SETTINGS_TYPE_UINT16:
1747 printf("%" PRIdz "\t%50s\tUINT16\t%" PRIu16 "\n", x, name,
1748 freerdp_settings_get_uint16(settings, (FreeRDP_Settings_Keys_UInt16)x));
1749 break;
1750 case RDP_SETTINGS_TYPE_INT16:
1751 printf("%" PRIdz "\t%50s\tINT16\t%" PRId16 "\n", x, name,
1752 freerdp_settings_get_int16(settings, (FreeRDP_Settings_Keys_Int16)x));
1753 break;
1754 case RDP_SETTINGS_TYPE_UINT32:
1755 printf("%" PRIdz "\t%50s\tUINT32\t%" PRIu32 "\n", x, name,
1756 freerdp_settings_get_uint32(settings, (FreeRDP_Settings_Keys_UInt32)x));
1757 break;
1758 case RDP_SETTINGS_TYPE_INT32:
1759 printf("%" PRIdz "\t%50s\tINT32\t%" PRId32 "\n", x, name,
1760 freerdp_settings_get_int32(settings, (FreeRDP_Settings_Keys_Int32)x));
1761 break;
1762 case RDP_SETTINGS_TYPE_UINT64:
1763 printf("%" PRIdz "\t%50s\tUINT64\t%" PRIu64 "\n", x, name,
1764 freerdp_settings_get_uint64(settings, (FreeRDP_Settings_Keys_UInt64)x));
1765 break;
1766 case RDP_SETTINGS_TYPE_INT64:
1767 printf("%" PRIdz "\t%50s\tINT64\t%" PRId64 "\n", x, name,
1768 freerdp_settings_get_int64(settings, (FreeRDP_Settings_Keys_Int64)x));
1769 break;
1770 case RDP_SETTINGS_TYPE_STRING:
1771 printf("%" PRIdz "\t%50s\tSTRING\t%s"
1772 "\n",
1773 x, name,
1774 freerdp_settings_get_string(settings, (FreeRDP_Settings_Keys_String)x));
1775 break;
1776 case RDP_SETTINGS_TYPE_POINTER:
1777 printf("%" PRIdz "\t%50s\tPOINTER\t%p"
1778 "\n",
1779 x, name,
1780 freerdp_settings_get_pointer(settings, (FreeRDP_Settings_Keys_Pointer)x));
1781 break;
1782 default:
1783 break;
1784 }
1785 // NOLINTEND(clang-analyzer-optin.core.EnumCastOutOfRange)
1786 }
1787}
1788
1789int freerdp_client_settings_command_line_status_print_ex(rdpSettings* settings, int status,
1790 int argc, char** argv,
1791 const COMMAND_LINE_ARGUMENT_A* custom)
1792{
1793 const COMMAND_LINE_ARGUMENT_A* arg = NULL;
1794 COMMAND_LINE_ARGUMENT_A largs[ARRAYSIZE(global_cmd_args)];
1795 memcpy(largs, global_cmd_args, sizeof(global_cmd_args));
1796
1797 if (status == COMMAND_LINE_STATUS_PRINT_VERSION)
1798 {
1799 freerdp_client_print_version();
1800 goto out;
1801 }
1802
1803 if (status == COMMAND_LINE_STATUS_PRINT_BUILDCONFIG)
1804 {
1805 freerdp_client_print_version_ex(argc, argv);
1806 freerdp_client_print_buildconfig_ex(argc, argv);
1807 goto out;
1808 }
1809 else if (status == COMMAND_LINE_STATUS_PRINT)
1810 {
1811 (void)CommandLineParseArgumentsA(argc, argv, largs, 0x112, NULL, NULL, NULL);
1812
1813 arg = CommandLineFindArgumentA(largs, "list");
1814 WINPR_ASSERT(arg);
1815
1816 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
1817 {
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))
1825 {
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;
1831
1832 if (val && strchr(val, ','))
1833 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1834 freerdp_client_print_codepages(val);
1835 }
1836 else if (option_equals("kbd-scancode", arg->Value))
1837 freerdp_client_print_scancodes();
1838 else if (option_equals("monitor", arg->Value))
1839 {
1840 if (!freerdp_settings_set_bool(settings, FreeRDP_ListMonitors, TRUE))
1841 return COMMAND_LINE_ERROR;
1842 }
1843 else if (option_starts_with("smartcard", arg->Value))
1844 {
1845 BOOL opts = FALSE;
1846 if (option_starts_with("smartcard:", arg->Value))
1847 opts = TRUE;
1848 else if (!option_equals("smartcard", arg->Value))
1849 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1850
1851 if (opts)
1852 {
1853 const char* sub = strchr(arg->Value, ':') + 1;
1854 const CmdLineSubOptions options[] = {
1855 { "pkinit-anchors:", FreeRDP_PkinitAnchors, CMDLINE_SUBOPTION_STRING,
1856 NULL },
1857 { "pkcs11-module:", FreeRDP_Pkcs11Module, CMDLINE_SUBOPTION_STRING, NULL }
1858 };
1859
1860 size_t count = 0;
1861
1862 char** ptr = CommandLineParseCommaSeparatedValuesEx("smartcard", sub, &count);
1863 if (!ptr)
1864 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1865 if (count < 2)
1866 {
1867 CommandLineParserFree(ptr);
1868 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1869 }
1870
1871 for (size_t x = 1; x < count; x++)
1872 {
1873 const char* cur = ptr[x];
1874 if (!parseSubOptions(settings, options, ARRAYSIZE(options), cur))
1875 {
1876 CommandLineParserFree(ptr);
1877 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
1878 }
1879 }
1880
1881 CommandLineParserFree(ptr);
1882 }
1883
1884 freerdp_smartcard_list(settings);
1885 }
1886 else
1887 {
1888 freerdp_client_print_command_line_help_ex(argc, argv, custom);
1889 return COMMAND_LINE_ERROR;
1890 }
1891 }
1892#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
1893 arg = CommandLineFindArgumentA(largs, "tune-list");
1894 WINPR_ASSERT(arg);
1895
1896 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
1897 {
1898 WLog_WARN(TAG, "Option /tune-list is deprecated, use /list:tune instead");
1899 freerdp_client_print_tune_list(settings);
1900 }
1901
1902 arg = CommandLineFindArgumentA(largs, "kbd-lang-list");
1903 WINPR_ASSERT(arg);
1904
1905 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
1906 {
1907 WLog_WARN(TAG, "Option /kbd-lang-list is deprecated, use /list:kbd-lang instead");
1908 freerdp_client_print_codepages(arg->Value);
1909 }
1910
1911 arg = CommandLineFindArgumentA(largs, "kbd-list");
1912 WINPR_ASSERT(arg);
1913
1914 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1915 {
1916 WLog_WARN(TAG, "Option /kbd-list is deprecated, use /list:kbd instead");
1917 freerdp_client_print_keyboard_list();
1918 }
1919
1920 arg = CommandLineFindArgumentA(largs, "monitor-list");
1921 WINPR_ASSERT(arg);
1922
1923 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1924 {
1925 WLog_WARN(TAG, "Option /monitor-list is deprecated, use /list:monitor instead");
1926 if (!freerdp_settings_set_bool(settings, FreeRDP_ListMonitors, TRUE))
1927 return COMMAND_LINE_ERROR;
1928 }
1929
1930 arg = CommandLineFindArgumentA(largs, "smartcard-list");
1931 WINPR_ASSERT(arg);
1932
1933 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1934 {
1935 WLog_WARN(TAG, "Option /smartcard-list is deprecated, use /list:smartcard instead");
1936 freerdp_smartcard_list(settings);
1937 }
1938
1939 arg = CommandLineFindArgumentA(largs, "kbd-scancode-list");
1940 WINPR_ASSERT(arg);
1941
1942 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
1943 {
1944 WLog_WARN(TAG,
1945 "Option /kbd-scancode-list is deprecated, use /list:kbd-scancode instead");
1946 freerdp_client_print_scancodes();
1947 goto out;
1948 }
1949#endif
1950 goto out;
1951 }
1952 else if (status < 0)
1953 {
1954 freerdp_client_print_command_line_help_ex(argc, argv, custom);
1955 goto out;
1956 }
1957
1958out:
1959 if (status <= COMMAND_LINE_STATUS_PRINT && status >= COMMAND_LINE_STATUS_PRINT_LAST)
1960 return 0;
1961 return status;
1962}
1963
1972static BOOL parseSizeValue(const char* input, unsigned long* v1, unsigned long* v2)
1973{
1974 const char* xcharpos = NULL;
1975 char* endPtr = NULL;
1976 unsigned long v = 0;
1977 errno = 0;
1978 v = strtoul(input, &endPtr, 10);
1979
1980 if ((v == 0 || v == ULONG_MAX) && (errno != 0))
1981 return FALSE;
1982
1983 if (v1)
1984 *v1 = v;
1985
1986 xcharpos = strchr(input, 'x');
1987
1988 if (!xcharpos || xcharpos != endPtr)
1989 return FALSE;
1990
1991 errno = 0;
1992 v = strtoul(xcharpos + 1, &endPtr, 10);
1993
1994 if ((v == 0 || v == ULONG_MAX) && (errno != 0))
1995 return FALSE;
1996
1997 if (*endPtr != '\0')
1998 return FALSE;
1999
2000 if (v2)
2001 *v2 = v;
2002
2003 return TRUE;
2004}
2005
2006static BOOL prepare_default_settings(rdpSettings* settings, COMMAND_LINE_ARGUMENT_A* args,
2007 BOOL rdp_file)
2008{
2009 const char* arguments[] = { "network", "gfx", "rfx", "bpp" };
2010 WINPR_ASSERT(settings);
2011 WINPR_ASSERT(args);
2012
2013 if (rdp_file)
2014 return FALSE;
2015
2016 for (size_t x = 0; x < ARRAYSIZE(arguments); x++)
2017 {
2018 const char* arg = arguments[x];
2019 const COMMAND_LINE_ARGUMENT_A* p = CommandLineFindArgumentA(args, arg);
2020 if (p && (p->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
2021 return FALSE;
2022 }
2023
2024 return freerdp_set_connection_type(settings, CONNECTION_TYPE_AUTODETECT);
2025}
2026
2027static BOOL setSmartcardEmulation(WINPR_ATTR_UNUSED const char* value, rdpSettings* settings)
2028{
2029 return freerdp_settings_set_bool(settings, FreeRDP_SmartcardEmulation, TRUE);
2030}
2031
2032const char* option_starts_with(const char* what, const char* val)
2033{
2034 WINPR_ASSERT(what);
2035 WINPR_ASSERT(val);
2036 const size_t wlen = strlen(what);
2037
2038 if (_strnicmp(what, val, wlen) != 0)
2039 return NULL;
2040 return &val[wlen];
2041}
2042
2043BOOL option_ends_with(const char* str, const char* ext)
2044{
2045 WINPR_ASSERT(str);
2046 WINPR_ASSERT(ext);
2047 const size_t strLen = strlen(str);
2048 const size_t extLen = strlen(ext);
2049
2050 if (strLen < extLen)
2051 return FALSE;
2052
2053 return _strnicmp(&str[strLen - extLen], ext, extLen) == 0;
2054}
2055
2056BOOL option_equals(const char* what, const char* val)
2057{
2058 WINPR_ASSERT(what);
2059 WINPR_ASSERT(val);
2060 return _stricmp(what, val) == 0;
2061}
2062
2063typedef enum
2064{
2065 PARSE_ON,
2066 PARSE_OFF,
2067 PARSE_NONE,
2068 PARSE_FAIL
2069} PARSE_ON_OFF_RESULT;
2070
2071static PARSE_ON_OFF_RESULT parse_on_off_option(const char* value)
2072{
2073 WINPR_ASSERT(value);
2074 const char* sep = strchr(value, ':');
2075 if (!sep)
2076 return PARSE_NONE;
2077 if (option_equals("on", &sep[1]))
2078 return PARSE_ON;
2079 if (option_equals("off", &sep[1]))
2080 return PARSE_OFF;
2081 return PARSE_FAIL;
2082}
2083
2084typedef enum
2085{
2086 CLIP_DIR_PARSE_ALL,
2087 CLIP_DIR_PARSE_OFF,
2088 CLIP_DIR_PARSE_LOCAL,
2089 CLIP_DIR_PARSE_REMOTE,
2090 CLIP_DIR_PARSE_FAIL
2091} PARSE_CLIP_DIR_RESULT;
2092
2093static PARSE_CLIP_DIR_RESULT parse_clip_direciton_to_option(const char* value)
2094{
2095 WINPR_ASSERT(value);
2096 const char* sep = strchr(value, ':');
2097 if (!sep)
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;
2108}
2109
2110static int parse_tls_ciphers(rdpSettings* settings, const char* Value)
2111{
2112 const char* ciphers = NULL;
2113 if (!Value)
2114 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2115
2116 if (option_equals(Value, "netmon"))
2117 {
2118 ciphers = "ALL:!ECDH:!ADH:!DHE";
2119 }
2120 else if (option_equals(Value, "ma"))
2121 {
2122 ciphers = "AES128-SHA";
2123 }
2124 else
2125 {
2126 ciphers = Value;
2127 }
2128
2129 if (!freerdp_settings_set_string(settings, FreeRDP_AllowedTlsCiphers, ciphers))
2130 return COMMAND_LINE_ERROR_MEMORY;
2131 return 0;
2132}
2133
2134static int parse_tls_seclevel(rdpSettings* settings, const char* Value)
2135{
2136 LONGLONG val = 0;
2137
2138 if (!value_to_int(Value, &val, 0, 5))
2139 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2140
2141 if (!freerdp_settings_set_uint32(settings, FreeRDP_TlsSecLevel, (UINT32)val))
2142 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2143 return 0;
2144}
2145
2146static int parse_tls_secrets_file(rdpSettings* settings, const char* Value)
2147{
2148 if (!Value)
2149 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2150
2151 if (!freerdp_settings_set_string(settings, FreeRDP_TlsSecretsFile, Value))
2152 return COMMAND_LINE_ERROR_MEMORY;
2153 return 0;
2154}
2155
2156static int parse_tls_enforce(rdpSettings* settings, const char* Value)
2157{
2158 UINT16 version = TLS1_2_VERSION;
2159
2160 if (Value)
2161 {
2162 struct map_t
2163 {
2164 const char* name;
2165 UINT16 version;
2166 };
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)
2171 ,
2172 { "1.3", TLS1_3_VERSION }
2173#endif
2174 };
2175
2176 const struct map_t* found = NULL;
2177 for (size_t x = 0; x < ARRAYSIZE(map); x++)
2178 {
2179 const struct map_t* cur = &map[x];
2180 if (option_equals(cur->name, Value))
2181 {
2182 found = cur;
2183 break;
2184 }
2185 }
2186
2187 if (!found)
2188 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2189 version = found->version;
2190 }
2191
2192 if (!(freerdp_settings_set_uint16(settings, FreeRDP_TLSMinVersion, version) &&
2193 freerdp_settings_set_uint16(settings, FreeRDP_TLSMaxVersion, version)))
2194 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2195 return 0;
2196}
2197
2198static int parse_tls_cipher_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2199{
2200 int rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2201 CommandLineSwitchStart(arg) CommandLineSwitchCase(arg, "tls")
2202 {
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]));
2211 }
2212
2213#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
2214 CommandLineSwitchCase(arg, "tls-ciphers")
2215 {
2216 WLog_WARN(TAG, "Option /tls-ciphers is deprecated, use /tls:ciphers instead");
2217 rc = fail_at(arg, parse_tls_ciphers(settings, arg->Value));
2218 }
2219 CommandLineSwitchCase(arg, "tls-seclevel")
2220 {
2221 WLog_WARN(TAG, "Option /tls-seclevel is deprecated, use /tls:seclevel instead");
2222 rc = fail_at(arg, parse_tls_seclevel(settings, arg->Value));
2223 }
2224 CommandLineSwitchCase(arg, "tls-secrets-file")
2225 {
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));
2228 }
2229 CommandLineSwitchCase(arg, "enforce-tlsv1_2")
2230 {
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"));
2233 }
2234#endif
2235 CommandLineSwitchDefault(arg)
2236 {
2237 }
2238 CommandLineSwitchEnd(arg)
2239
2240 return rc;
2241}
2242
2243static int parse_tls_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2244{
2245 WINPR_ASSERT(settings);
2246 WINPR_ASSERT(arg);
2247
2248 size_t count = 0;
2249 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2250 for (size_t x = 0; x < count; x++)
2251 {
2252 COMMAND_LINE_ARGUMENT_A larg = *arg;
2253 larg.Value = ptr[x];
2254
2255 int rc = parse_tls_cipher_options(settings, &larg);
2256 if (rc != 0)
2257 {
2258 CommandLineParserFree(ptr);
2259 return rc;
2260 }
2261 }
2262 CommandLineParserFree(ptr);
2263 return 0;
2264}
2265
2266static int parse_gfx_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2267{
2268 WINPR_ASSERT(settings);
2269 WINPR_ASSERT(arg);
2270
2271 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportGraphicsPipeline, TRUE))
2272 return COMMAND_LINE_ERROR;
2273
2274 if (arg->Value)
2275 {
2276 int rc = CHANNEL_RC_OK;
2277 size_t count = 0;
2278 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2279 if (!ptr || (count == 0))
2280 rc = COMMAND_LINE_ERROR;
2281 else
2282 {
2283 BOOL GfxH264 = FALSE;
2284 BOOL GfxAVC444 = FALSE;
2285 BOOL RemoteFxCodec = FALSE;
2286 BOOL GfxProgressive = FALSE;
2287 BOOL codecSelected = FALSE;
2288
2289 for (size_t x = 0; x < count; x++)
2290 {
2291 const char* val = ptr[x];
2292#ifdef WITH_GFX_H264
2293 if (option_starts_with("AVC444", val))
2294 {
2295 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2296 if (bval == PARSE_FAIL)
2297 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2298 else
2299 GfxAVC444 = bval != PARSE_OFF;
2300 codecSelected = TRUE;
2301 }
2302 else if (option_starts_with("AVC420", val))
2303 {
2304 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2305 if (bval == PARSE_FAIL)
2306 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2307 else
2308 GfxH264 = bval != PARSE_OFF;
2309 codecSelected = TRUE;
2310 }
2311 else
2312#endif
2313 if (option_starts_with("RFX", val))
2314 {
2315 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2316 if (bval == PARSE_FAIL)
2317 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2318 else
2319 RemoteFxCodec = bval != PARSE_OFF;
2320 codecSelected = TRUE;
2321 }
2322 else if (option_starts_with("progressive", val))
2323 {
2324 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2325 if (bval == PARSE_FAIL)
2326 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2327 else
2328 GfxProgressive = bval != PARSE_OFF;
2329 codecSelected = TRUE;
2330 }
2331 else if (option_starts_with("mask:", val))
2332 {
2333 ULONGLONG v = 0;
2334 const char* uv = &val[5];
2335 if (!value_to_uint(uv, &v, 0, UINT32_MAX))
2336 rc = COMMAND_LINE_ERROR;
2337 else
2338 {
2339 if (!freerdp_settings_set_uint32(settings, FreeRDP_GfxCapsFilter,
2340 (UINT32)v))
2341 rc = COMMAND_LINE_ERROR;
2342 }
2343 }
2344 else if (option_starts_with("small-cache", val))
2345 {
2346 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2347 if (bval == PARSE_FAIL)
2348 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2349 else if (!freerdp_settings_set_bool(settings, FreeRDP_GfxSmallCache,
2350 bval != PARSE_OFF))
2351 rc = COMMAND_LINE_ERROR;
2352 }
2353 else if (option_starts_with("thin-client", val))
2354 {
2355 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2356 if (bval == PARSE_FAIL)
2357 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2358 else if (!freerdp_settings_set_bool(settings, FreeRDP_GfxThinClient,
2359 bval != PARSE_OFF))
2360 rc = COMMAND_LINE_ERROR;
2361 if ((rc == CHANNEL_RC_OK) && (bval > 0))
2362 {
2363 if (!freerdp_settings_set_bool(settings, FreeRDP_GfxSmallCache,
2364 bval != PARSE_OFF))
2365 rc = COMMAND_LINE_ERROR;
2366 }
2367 }
2368 else if (option_starts_with("frame-ack", val))
2369 {
2370 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2371 if (bval == PARSE_FAIL)
2372 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2373 else if (!freerdp_settings_set_bool(settings, FreeRDP_GfxSuspendFrameAck,
2374 bval == PARSE_OFF))
2375 rc = COMMAND_LINE_ERROR;
2376 }
2377 else
2378 rc = COMMAND_LINE_ERROR;
2379 }
2380
2381 if ((rc == CHANNEL_RC_OK) && codecSelected)
2382 {
2383 if (!freerdp_settings_set_bool(settings, FreeRDP_GfxAVC444, GfxAVC444))
2384 rc = COMMAND_LINE_ERROR;
2385 if (!freerdp_settings_set_bool(settings, FreeRDP_GfxAVC444v2, GfxAVC444))
2386 rc = COMMAND_LINE_ERROR;
2387 if (!freerdp_settings_set_bool(settings, FreeRDP_GfxH264, GfxH264))
2388 rc = COMMAND_LINE_ERROR;
2389 if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteFxCodec, RemoteFxCodec))
2390 rc = COMMAND_LINE_ERROR;
2391 if (!freerdp_settings_set_bool(settings, FreeRDP_GfxProgressive, GfxProgressive))
2392 rc = COMMAND_LINE_ERROR;
2393 }
2394 }
2395 CommandLineParserFree(ptr);
2396 if (rc != CHANNEL_RC_OK)
2397 return rc;
2398 }
2399 return CHANNEL_RC_OK;
2400}
2401
2402static int parse_kbd_layout(rdpSettings* settings, const char* value)
2403{
2404 WINPR_ASSERT(settings);
2405 WINPR_ASSERT(value);
2406
2407 int rc = 0;
2408 LONGLONG ival = 0;
2409 const BOOL isInt = value_to_int(value, &ival, 1, UINT32_MAX);
2410 if (!isInt)
2411 {
2412 ival = freerdp_map_keyboard_layout_name_to_id(value);
2413
2414 if (ival == 0)
2415 {
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;
2419 }
2420 }
2421
2422 if (rc == 0)
2423 {
2424 if (!freerdp_settings_set_uint32(settings, FreeRDP_KeyboardLayout, (UINT32)ival))
2425 rc = COMMAND_LINE_ERROR;
2426 }
2427 return rc;
2428}
2429
2430#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
2431static int parse_codec_cache_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2432{
2433 WINPR_ASSERT(settings);
2434 WINPR_ASSERT(arg);
2435
2436 if (!arg->Value)
2437 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2438 if (!freerdp_settings_set_bool(settings, FreeRDP_BitmapCacheV3Enabled, TRUE))
2439 return COMMAND_LINE_ERROR;
2440
2441 if (option_equals(arg->Value, "rfx"))
2442 {
2443 if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteFxCodec, TRUE))
2444 return COMMAND_LINE_ERROR;
2445 }
2446 else if (option_equals(arg->Value, "nsc"))
2447 {
2448 if (!freerdp_settings_set_bool(settings, FreeRDP_NSCodec, TRUE))
2449 return COMMAND_LINE_ERROR;
2450 }
2451
2452#if defined(WITH_JPEG)
2453 else if (option_equals(arg->Value, "jpeg"))
2454 {
2455 if (!freerdp_settings_set_bool(settings, FreeRDP_JpegCodec, TRUE))
2456 return COMMAND_LINE_ERROR;
2457
2458 if (freerdp_settings_get_uint32(settings, FreeRDP_JpegQuality) == 0)
2459 {
2460 if (!freerdp_settings_set_uint32(settings, FreeRDP_JpegQuality, 75))
2461 return COMMAND_LINE_ERROR;
2462 }
2463 }
2464
2465#endif
2466 return 0;
2467}
2468#endif
2469
2470static BOOL check_kbd_remap_valid(const char* token)
2471{
2472 UINT32 key = 0;
2473 UINT32 value = 0;
2474
2475 WINPR_ASSERT(token);
2476 /* The remapping is only allowed for scancodes, so maximum is 999=999 */
2477 if (strlen(token) > 10)
2478 return FALSE;
2479
2480 if (!freerdp_extract_key_value(token, &key, &value))
2481 {
2482 WLog_WARN(TAG, "/kbd:remap invalid entry '%s'", token);
2483 return FALSE;
2484 }
2485 return TRUE;
2486}
2487
2488static int parse_host_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2489{
2490 WINPR_ASSERT(settings);
2491 WINPR_ASSERT(arg);
2492
2493 if (!arg->Value)
2494 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2495 if (!freerdp_settings_set_string(settings, FreeRDP_ServerHostname, NULL))
2496 return COMMAND_LINE_ERROR_MEMORY;
2497 char* p = strchr(arg->Value, '[');
2498
2499 /* ipv4 */
2500 if (!p)
2501 {
2502 const char scheme[] = "://";
2503 const char* val = strstr(arg->Value, scheme);
2504 if (val)
2505 val += strnlen(scheme, sizeof(scheme));
2506 else
2507 val = arg->Value;
2508 p = strchr(val, ':');
2509
2510 if (p)
2511 {
2512 LONGLONG lval = 0;
2513 size_t length = 0;
2514
2515 if (!value_to_int(&p[1], &lval, 1, UINT16_MAX))
2516 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2517
2518 length = (size_t)(p - arg->Value);
2519 if (!freerdp_settings_set_uint32(settings, FreeRDP_ServerPort, (UINT16)lval))
2520 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2521 if (!freerdp_settings_set_string_len(settings, FreeRDP_ServerHostname, arg->Value,
2522 length))
2523 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2524 }
2525 else
2526 {
2527 if (!freerdp_settings_set_string(settings, FreeRDP_ServerHostname, arg->Value))
2528 return COMMAND_LINE_ERROR_MEMORY;
2529 }
2530 }
2531 else /* ipv6 */
2532 {
2533 size_t length = 0;
2534 char* p2 = strchr(arg->Value, ']');
2535
2536 /* not a valid [] ipv6 addr found */
2537 if (!p2)
2538 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2539
2540 length = (size_t)(p2 - p);
2541 if (!freerdp_settings_set_string_len(settings, FreeRDP_ServerHostname, p + 1, length - 1))
2542 return COMMAND_LINE_ERROR_MEMORY;
2543
2544 if (*(p2 + 1) == ':')
2545 {
2546 LONGLONG val = 0;
2547
2548 if (!value_to_int(&p2[2], &val, 0, UINT16_MAX))
2549 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2550
2551 if (!freerdp_settings_set_uint32(settings, FreeRDP_ServerPort, (UINT16)val))
2552 return COMMAND_LINE_ERROR;
2553 }
2554
2555 printf("hostname %s port %" PRIu32 "\n",
2556 freerdp_settings_get_string(settings, FreeRDP_ServerHostname),
2557 freerdp_settings_get_uint32(settings, FreeRDP_ServerPort));
2558 }
2559 return 0;
2560}
2561
2562static int parse_redirect_prefer_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2563{
2564 WINPR_ASSERT(settings);
2565 WINPR_ASSERT(arg);
2566
2567 size_t count = 0;
2568 char* cur = arg->Value;
2569 if (!arg->Value)
2570 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2571 if (!freerdp_settings_set_uint32(settings, FreeRDP_RedirectionPreferType, 0))
2572 return COMMAND_LINE_ERROR;
2573
2574 UINT32 value = 0;
2575 do
2576 {
2577 UINT32 mask = 0;
2578 char* next = strchr(cur, ',');
2579
2580 if (next)
2581 {
2582 *next = '\0';
2583 next++;
2584 }
2585
2586 if (option_equals("fqdn", cur))
2587 mask = 0x06U;
2588 else if (option_equals("ip", cur))
2589 mask = 0x05U;
2590 else if (option_equals("netbios", cur))
2591 mask = 0x03U;
2592 else
2593 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2594
2595 cur = next;
2596 mask = (mask & 0x07);
2597 value |= mask << (count * 3);
2598 count++;
2599 } while (cur != NULL);
2600
2601 if (!freerdp_settings_set_uint32(settings, FreeRDP_RedirectionPreferType, value))
2602 return COMMAND_LINE_ERROR;
2603
2604 if (count > 3)
2605 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2606 return 0;
2607}
2608
2609static int parse_prevent_session_lock_options(rdpSettings* settings,
2610 const COMMAND_LINE_ARGUMENT_A* arg)
2611{
2612 WINPR_ASSERT(settings);
2613 WINPR_ASSERT(arg);
2614
2615 if (!freerdp_settings_set_uint32(settings, FreeRDP_FakeMouseMotionInterval, 180))
2616 return COMMAND_LINE_ERROR_MEMORY;
2617
2618 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2619 {
2620 LONGLONG val = 0;
2621
2622 if (!value_to_int(arg->Value, &val, 1, UINT32_MAX))
2623 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2624
2625 if (!freerdp_settings_set_uint32(settings, FreeRDP_FakeMouseMotionInterval, (UINT32)val))
2626 return COMMAND_LINE_ERROR_MEMORY;
2627 }
2628
2629 return 0;
2630}
2631
2632static int parse_vmconnect_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2633{
2634 WINPR_ASSERT(settings);
2635 WINPR_ASSERT(arg);
2636
2637 if (!freerdp_settings_set_bool(settings, FreeRDP_VmConnectMode, TRUE))
2638 return COMMAND_LINE_ERROR;
2639
2640 UINT32 port = freerdp_settings_get_uint32(settings, FreeRDP_ServerPort);
2641 if (port == 3389)
2642 port = 2179;
2643
2644 if (!freerdp_settings_set_uint32(settings, FreeRDP_ServerPort, port))
2645 return COMMAND_LINE_ERROR;
2646 if (!freerdp_settings_set_bool(settings, FreeRDP_NegotiateSecurityLayer, FALSE))
2647 return COMMAND_LINE_ERROR;
2648
2649 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2650 {
2651 if (!freerdp_settings_set_bool(settings, FreeRDP_SendPreconnectionPdu, TRUE))
2652 return COMMAND_LINE_ERROR;
2653
2654 if (!freerdp_settings_set_string(settings, FreeRDP_PreconnectionBlob, arg->Value))
2655 return COMMAND_LINE_ERROR_MEMORY;
2656 }
2657 return 0;
2658}
2659
2660static int parse_size_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2661{
2662 int status = 0;
2663 WINPR_ASSERT(settings);
2664 WINPR_ASSERT(arg);
2665
2666 if (!arg->Value)
2667 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2668 char* p = strchr(arg->Value, 'x');
2669
2670 if (p)
2671 {
2672 unsigned long w = 0;
2673 unsigned long h = 0;
2674
2675 if (!parseSizeValue(arg->Value, &w, &h) || (w > UINT16_MAX) || (h > UINT16_MAX))
2676 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2677
2678 if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopWidth, (UINT32)w))
2679 return COMMAND_LINE_ERROR;
2680 if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopHeight, (UINT32)h))
2681 return COMMAND_LINE_ERROR;
2682 }
2683 else
2684 {
2685 char* str = _strdup(arg->Value);
2686 if (!str)
2687 return COMMAND_LINE_ERROR_MEMORY;
2688
2689 p = strchr(str, '%');
2690
2691 if (p)
2692 {
2693 BOOL partial = FALSE;
2694
2695 status = COMMAND_LINE_ERROR;
2696 if (strchr(p, 'w'))
2697 {
2698 if (!freerdp_settings_set_bool(settings, FreeRDP_PercentScreenUseWidth, TRUE))
2699 goto fail;
2700 partial = TRUE;
2701 }
2702
2703 if (strchr(p, 'h'))
2704 {
2705 if (!freerdp_settings_set_bool(settings, FreeRDP_PercentScreenUseHeight, TRUE))
2706 goto fail;
2707 partial = TRUE;
2708 }
2709
2710 if (!partial)
2711 {
2712 if (!freerdp_settings_set_bool(settings, FreeRDP_PercentScreenUseWidth, TRUE))
2713 goto fail;
2714 if (!freerdp_settings_set_bool(settings, FreeRDP_PercentScreenUseHeight, TRUE))
2715 goto fail;
2716 }
2717
2718 *p = '\0';
2719 {
2720 LONGLONG val = 0;
2721
2722 if (!value_to_int(str, &val, 0, 100))
2723 {
2724 status = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2725 goto fail;
2726 }
2727
2728 if (!freerdp_settings_set_uint32(settings, FreeRDP_PercentScreen, (UINT32)val))
2729 goto fail;
2730 }
2731
2732 status = 0;
2733 }
2734
2735 fail:
2736 free(str);
2737 }
2738
2739 return status;
2740}
2741
2742static int parse_monitors_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2743{
2744 WINPR_ASSERT(settings);
2745 WINPR_ASSERT(arg);
2746
2747 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2748 {
2749 size_t count = 0;
2750 UINT32* MonitorIds = NULL;
2751 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2752
2753 if (!ptr)
2754 return COMMAND_LINE_ERROR_MEMORY;
2755
2756 if (count > 16)
2757 count = 16;
2758
2759 if (!freerdp_settings_set_pointer_len(settings, FreeRDP_MonitorIds, NULL, count))
2760 {
2761 CommandLineParserFree(ptr);
2762 return FALSE;
2763 }
2764
2765 MonitorIds = freerdp_settings_get_pointer_array_writable(settings, FreeRDP_MonitorIds, 0);
2766 for (UINT32 i = 0; i < count; i++)
2767 {
2768 LONGLONG val = 0;
2769
2770 if (!value_to_int(ptr[i], &val, 0, UINT16_MAX))
2771 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2772
2773 MonitorIds[i] = (UINT32)val;
2774 }
2775
2776 CommandLineParserFree(ptr);
2777 }
2778
2779 return 0;
2780}
2781
2782static int parse_dynamic_resolution_options(rdpSettings* settings,
2783 const COMMAND_LINE_ARGUMENT_A* arg)
2784{
2785 WINPR_ASSERT(settings);
2786 WINPR_ASSERT(arg);
2787
2788 const BOOL val = arg->Value != 0;
2789
2790 if (val && freerdp_settings_get_bool(settings, FreeRDP_SmartSizing))
2791 {
2792 WLog_ERR(TAG, "Smart sizing and dynamic resolution are mutually exclusive options");
2793 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2794 }
2795
2796 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportDisplayControl, val))
2797 return COMMAND_LINE_ERROR;
2798 if (!freerdp_settings_set_bool(settings, FreeRDP_DynamicResolutionUpdate, val))
2799 return COMMAND_LINE_ERROR;
2800
2801 return 0;
2802}
2803
2804static int parse_smart_sizing_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2805{
2806 WINPR_ASSERT(settings);
2807 WINPR_ASSERT(arg);
2808
2809 if (freerdp_settings_get_bool(settings, FreeRDP_DynamicResolutionUpdate))
2810 {
2811 WLog_ERR(TAG, "Smart sizing and dynamic resolution are mutually exclusive options");
2812 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2813 }
2814
2815 if (!freerdp_settings_set_bool(settings, FreeRDP_SmartSizing, TRUE))
2816 return COMMAND_LINE_ERROR;
2817
2818 if (arg->Value)
2819 {
2820 unsigned long w = 0;
2821 unsigned long h = 0;
2822
2823 if (!parseSizeValue(arg->Value, &w, &h) || (w > UINT16_MAX) || (h > UINT16_MAX))
2824 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2825
2826 if (!freerdp_settings_set_uint32(settings, FreeRDP_SmartSizingWidth, (UINT32)w))
2827 return COMMAND_LINE_ERROR;
2828 if (!freerdp_settings_set_uint32(settings, FreeRDP_SmartSizingHeight, (UINT32)h))
2829 return COMMAND_LINE_ERROR;
2830 }
2831 return 0;
2832}
2833
2834static int parse_bpp_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2835{
2836 WINPR_ASSERT(settings);
2837 WINPR_ASSERT(arg);
2838
2839 LONGLONG val = 0;
2840
2841 if (!value_to_int(arg->Value, &val, 0, UINT32_MAX))
2842 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2843
2844 switch (val)
2845 {
2846 case 32:
2847 case 24:
2848 case 16:
2849 case 15:
2850 case 8:
2851 if (!freerdp_settings_set_uint32(settings, FreeRDP_ColorDepth, (UINT32)val))
2852 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2853 break;
2854
2855 default:
2856 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2857 }
2858 return 0;
2859}
2860
2861static int parse_kbd_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2862{
2863 WINPR_ASSERT(settings);
2864 WINPR_ASSERT(arg);
2865
2866 int rc = CHANNEL_RC_OK;
2867 size_t count = 0;
2868 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
2869 if (!ptr || (count == 0))
2870 rc = COMMAND_LINE_ERROR;
2871 else
2872 {
2873 for (size_t x = 0; x < count; x++)
2874 {
2875 const char* val = ptr[x];
2876
2877 if (option_starts_with("remap:", val))
2878 {
2879 /* Append this new occurrence to the already existing list */
2880 char* now = _strdup(&val[6]);
2881 const char* old =
2882 freerdp_settings_get_string(settings, FreeRDP_KeyboardRemappingList);
2883
2884 /* Basic sanity test. Entries must be like <key>=<value>, e.g. 1=2 */
2885 if (!check_kbd_remap_valid(now))
2886 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2887 else if (old)
2888 {
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));
2893 if (!tmp)
2894 rc = COMMAND_LINE_ERROR_MEMORY;
2895 else
2896 (void)_snprintf(tmp, tlen, "%s,%s", old, now);
2897 free(now);
2898 now = tmp;
2899 }
2900
2901 if (rc == 0)
2902 {
2903 if (!freerdp_settings_set_string(settings, FreeRDP_KeyboardRemappingList, now))
2904 rc = COMMAND_LINE_ERROR;
2905 }
2906 free(now);
2907 }
2908 else if (option_starts_with("layout:", val))
2909 {
2910 rc = parse_kbd_layout(settings, &val[7]);
2911 }
2912 else if (option_starts_with("lang:", val))
2913 {
2914 LONGLONG ival = 0;
2915 const BOOL isInt = value_to_int(&val[5], &ival, 1, UINT32_MAX);
2916 if (!isInt)
2917 ival = freerdp_get_locale_id_from_string(&val[5]);
2918
2919 if (ival <= 0)
2920 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2921 else if (!freerdp_settings_set_uint32(settings, FreeRDP_KeyboardCodePage,
2922 (UINT32)ival))
2923 rc = COMMAND_LINE_ERROR;
2924 }
2925 else if (option_starts_with("type:", val))
2926 {
2927 LONGLONG ival = 0;
2928 const BOOL isInt = value_to_int(&val[5], &ival, 1, UINT32_MAX);
2929 if (!isInt)
2930 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2931 else if (!freerdp_settings_set_uint32(settings, FreeRDP_KeyboardType, (UINT32)ival))
2932 rc = COMMAND_LINE_ERROR;
2933 }
2934 else if (option_starts_with("subtype:", val))
2935 {
2936 LONGLONG ival = 0;
2937 const BOOL isInt = value_to_int(&val[8], &ival, 1, UINT32_MAX);
2938 if (!isInt)
2939 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2940 else if (!freerdp_settings_set_uint32(settings, FreeRDP_KeyboardSubType,
2941 (UINT32)ival))
2942 rc = COMMAND_LINE_ERROR;
2943 }
2944 else if (option_starts_with("fn-key:", val))
2945 {
2946 LONGLONG ival = 0;
2947 const BOOL isInt = value_to_int(&val[7], &ival, 1, UINT32_MAX);
2948 if (!isInt)
2949 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2950 else if (!freerdp_settings_set_uint32(settings, FreeRDP_KeyboardFunctionKey,
2951 (UINT32)ival))
2952 rc = COMMAND_LINE_ERROR;
2953 }
2954 else if (option_starts_with("unicode", val))
2955 {
2956 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
2957 if (bval == PARSE_FAIL)
2958 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2959 else if (!freerdp_settings_set_bool(settings, FreeRDP_UnicodeInput,
2960 bval != PARSE_OFF))
2961 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2962 }
2963 else if (option_starts_with("pipe:", val))
2964 {
2965 if (!freerdp_settings_set_bool(settings, FreeRDP_UnicodeInput, TRUE))
2966 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2967 else if (!freerdp_settings_set_string(settings, FreeRDP_KeyboardPipeName, &val[5]))
2968 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2969 }
2970#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
2971 else if (count == 1)
2972 {
2973 /* Legacy, allow /kbd:<value> for setting keyboard layout */
2974 rc = parse_kbd_layout(settings, val);
2975 }
2976#endif
2977 else
2978 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
2979
2980 if (rc != 0)
2981 break;
2982 }
2983 }
2984 CommandLineParserFree(ptr);
2985 return rc;
2986}
2987
2988static int parse_proxy_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
2989{
2990 WINPR_ASSERT(settings);
2991 WINPR_ASSERT(arg);
2992
2993 /* initial value */
2994 if (!freerdp_settings_set_uint32(settings, FreeRDP_ProxyType, PROXY_TYPE_HTTP))
2995 return COMMAND_LINE_ERROR_MEMORY;
2996
2997 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
2998 {
2999 const char* cur = arg->Value;
3000
3001 if (!cur)
3002 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3003 /* value is [scheme://][user:password@]hostname:port */
3004 if (!proxy_parse_uri(settings, cur))
3005 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3006 }
3007 else
3008 {
3009 WLog_ERR(TAG, "Option http-proxy needs argument.");
3010 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3011 }
3012 return 0;
3013}
3014
3015static int parse_dump_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3016{
3017 WINPR_ASSERT(settings);
3018 WINPR_ASSERT(arg);
3019
3020 BOOL failed = FALSE;
3021 size_t count = 0;
3022 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3023 if (!ptr)
3024 failed = TRUE;
3025 else
3026 {
3027 BOOL modernsyntax = FALSE;
3028 BOOL oldsyntax = FALSE;
3029 for (size_t x = 0; (x < count) && !failed; x++)
3030 {
3031 const char* carg = ptr[x];
3032 if (option_starts_with("file:", carg))
3033 {
3034 const char* val = &carg[5];
3035 if (oldsyntax)
3036 failed = TRUE;
3037 else if (!freerdp_settings_set_string(settings, FreeRDP_TransportDumpFile, val))
3038 failed = TRUE;
3039 modernsyntax = TRUE;
3040 }
3041 else if (option_equals("replay", carg))
3042 {
3043 if (!freerdp_settings_set_bool(settings, FreeRDP_TransportDump, FALSE))
3044 failed = TRUE;
3045 else if (!freerdp_settings_set_bool(settings, FreeRDP_TransportDumpReplay, TRUE))
3046 failed = TRUE;
3047 }
3048 else if (option_equals("record", carg))
3049 {
3050 if (!freerdp_settings_set_bool(settings, FreeRDP_TransportDump, TRUE))
3051 failed = TRUE;
3052 else if (!freerdp_settings_set_bool(settings, FreeRDP_TransportDumpReplay, FALSE))
3053 failed = TRUE;
3054 }
3055 else if (option_equals("nodelay", carg))
3056 {
3057 if (oldsyntax)
3058 failed = TRUE;
3059 else if (!freerdp_settings_set_bool(settings, FreeRDP_TransportDumpReplayNodelay,
3060 TRUE))
3061 failed = TRUE;
3062 modernsyntax = TRUE;
3063 }
3064 else
3065 {
3066 /* compat:
3067 * support syntax record,<filename> and replay,<filename>
3068 */
3069 if (modernsyntax)
3070 failed = TRUE;
3071 else if (!freerdp_settings_set_string(settings, FreeRDP_TransportDumpFile, carg))
3072 failed = TRUE;
3073 oldsyntax = TRUE;
3074 }
3075 }
3076
3077 if (oldsyntax && (count != 2))
3078 failed = TRUE;
3079 }
3080 CommandLineParserFree(ptr);
3081 if (failed)
3082 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3083 return 0;
3084}
3085
3086static int parse_clipboard_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3087{
3088 WINPR_ASSERT(settings);
3089 WINPR_ASSERT(arg);
3090
3091 if (arg->Value == BoolValueTrue || arg->Value == BoolValueFalse)
3092 {
3093 if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectClipboard,
3094 (arg->Value == BoolValueTrue)))
3095 return COMMAND_LINE_ERROR;
3096 }
3097 else
3098 {
3099 int rc = 0;
3100 size_t count = 0;
3101 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3102 for (size_t x = 0; (x < count) && (rc == 0); x++)
3103 {
3104 const char* usesel = "use-selection:";
3105
3106 const char* cur = ptr[x];
3107 if (option_starts_with(usesel, cur))
3108 {
3109 const char* val = &cur[strlen(usesel)];
3110 if (!freerdp_settings_set_string(settings, FreeRDP_ClipboardUseSelection, val))
3111 rc = COMMAND_LINE_ERROR_MEMORY;
3112 if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectClipboard, TRUE))
3113 return COMMAND_LINE_ERROR;
3114 }
3115 else if (option_starts_with("direction-to", cur))
3116 {
3117 const UINT32 mask =
3118 freerdp_settings_get_uint32(settings, FreeRDP_ClipboardFeatureMask) &
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);
3121 UINT32 bflags = 0;
3122 switch (bval)
3123 {
3124 case CLIP_DIR_PARSE_ALL:
3125 bflags |= CLIPRDR_FLAG_LOCAL_TO_REMOTE | CLIPRDR_FLAG_REMOTE_TO_LOCAL;
3126 break;
3127 case CLIP_DIR_PARSE_LOCAL:
3128 bflags |= CLIPRDR_FLAG_REMOTE_TO_LOCAL;
3129 break;
3130 case CLIP_DIR_PARSE_REMOTE:
3131 bflags |= CLIPRDR_FLAG_LOCAL_TO_REMOTE;
3132 break;
3133 case CLIP_DIR_PARSE_OFF:
3134 break;
3135 case CLIP_DIR_PARSE_FAIL:
3136 default:
3137 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3138 break;
3139 }
3140
3141 if (!freerdp_settings_set_uint32(settings, FreeRDP_ClipboardFeatureMask,
3142 mask | bflags))
3143 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3144 }
3145 else if (option_starts_with("files-to", cur))
3146 {
3147 const UINT32 mask =
3148 freerdp_settings_get_uint32(settings, FreeRDP_ClipboardFeatureMask) &
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);
3152 UINT32 bflags = 0;
3153 switch (bval)
3154 {
3155 case CLIP_DIR_PARSE_ALL:
3156 bflags |=
3157 CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES | CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES;
3158 break;
3159 case CLIP_DIR_PARSE_LOCAL:
3160 bflags |= CLIPRDR_FLAG_REMOTE_TO_LOCAL_FILES;
3161 break;
3162 case CLIP_DIR_PARSE_REMOTE:
3163 bflags |= CLIPRDR_FLAG_LOCAL_TO_REMOTE_FILES;
3164 break;
3165 case CLIP_DIR_PARSE_OFF:
3166 break;
3167 case CLIP_DIR_PARSE_FAIL:
3168 default:
3169 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3170 break;
3171 }
3172
3173 if (!freerdp_settings_set_uint32(settings, FreeRDP_ClipboardFeatureMask,
3174 mask | bflags))
3175 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3176 }
3177 else
3178 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3179 }
3180 CommandLineParserFree(ptr);
3181
3182 if (rc)
3183 return rc;
3184 }
3185 return 0;
3186}
3187
3188static int parse_audio_mode_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3189{
3190 WINPR_ASSERT(settings);
3191 WINPR_ASSERT(arg);
3192
3193 LONGLONG val = 0;
3194
3195 if (!value_to_int(arg->Value, &val, 0, UINT32_MAX))
3196 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3197
3198 switch (val)
3199 {
3200 case AUDIO_MODE_REDIRECT:
3201 if (!freerdp_settings_set_bool(settings, FreeRDP_AudioPlayback, TRUE))
3202 return COMMAND_LINE_ERROR;
3203 break;
3204
3205 case AUDIO_MODE_PLAY_ON_SERVER:
3206 if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteConsoleAudio, TRUE))
3207 return COMMAND_LINE_ERROR;
3208 break;
3209
3210 case AUDIO_MODE_NONE:
3211 if (!freerdp_settings_set_bool(settings, FreeRDP_AudioPlayback, FALSE))
3212 return COMMAND_LINE_ERROR;
3213 if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteConsoleAudio, FALSE))
3214 return COMMAND_LINE_ERROR;
3215 break;
3216
3217 default:
3218 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3219 }
3220 return 0;
3221}
3222
3223static int parse_network_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3224{
3225 WINPR_ASSERT(settings);
3226 WINPR_ASSERT(arg);
3227
3228 UINT32 type = CONNECTION_TYPE_INVALID;
3229
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")))
3246 {
3247 type = CONNECTION_TYPE_AUTODETECT;
3248 }
3249 else
3250 {
3251 LONGLONG val = 0;
3252
3253 if (!value_to_int(arg->Value, &val, 0, 7))
3254 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3255
3256 type = (UINT32)val;
3257 }
3258
3259 if (!freerdp_set_connection_type(settings, type))
3260 return COMMAND_LINE_ERROR;
3261 return 0;
3262}
3263
3264static int parse_sec_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3265{
3266 WINPR_ASSERT(settings);
3267 WINPR_ASSERT(arg);
3268
3269 size_t count = 0;
3270 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3271 if (count == 0)
3272 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3273
3274 FreeRDP_Settings_Keys_Bool singleOptionWithoutOnOff = FreeRDP_BOOL_UNUSED;
3275 for (size_t x = 0; x < count; x++)
3276 {
3277 const char* cur = ptr[x];
3278 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(cur);
3279 if (bval == PARSE_FAIL)
3280 {
3281 CommandLineParserFree(ptr);
3282 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3283 }
3284
3285 const BOOL val = bval != PARSE_OFF;
3286 FreeRDP_Settings_Keys_Bool id = FreeRDP_BOOL_UNUSED;
3287 if (option_starts_with("rdp", cur)) /* Standard RDP */
3288 {
3289 id = FreeRDP_RdpSecurity;
3290 if (!freerdp_settings_set_bool(settings, FreeRDP_UseRdpSecurityLayer, val))
3291 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3292 }
3293 else if (option_starts_with("tls", cur)) /* TLS */
3294 id = FreeRDP_TlsSecurity;
3295 else if (option_starts_with("nla", cur)) /* NLA */
3296 id = FreeRDP_NlaSecurity;
3297 else if (option_starts_with("ext", cur)) /* NLA Extended */
3298 id = FreeRDP_ExtSecurity;
3299 else if (option_equals("aad", cur)) /* RDSAAD */
3300 id = FreeRDP_AadSecurity;
3301 else
3302 {
3303 WLog_ERR(TAG, "unknown protocol security: %s", arg->Value);
3304 CommandLineParserFree(ptr);
3305 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3306 }
3307
3308 if ((bval == PARSE_NONE) && (count == 1))
3309 singleOptionWithoutOnOff = id;
3310 if (!freerdp_settings_set_bool(settings, id, val))
3311 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3312 }
3313
3314 if (singleOptionWithoutOnOff != FreeRDP_BOOL_UNUSED)
3315 {
3316 const FreeRDP_Settings_Keys_Bool options[] = { FreeRDP_AadSecurity,
3317 FreeRDP_UseRdpSecurityLayer,
3318 FreeRDP_RdpSecurity, FreeRDP_NlaSecurity,
3319 FreeRDP_TlsSecurity };
3320
3321 for (size_t i = 0; i < ARRAYSIZE(options); i++)
3322 {
3323 if (!freerdp_settings_set_bool(settings, options[i], FALSE))
3324 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3325 }
3326
3327 if (!freerdp_settings_set_bool(settings, singleOptionWithoutOnOff, TRUE))
3328 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3329 if (singleOptionWithoutOnOff == FreeRDP_RdpSecurity)
3330 {
3331 if (!freerdp_settings_set_bool(settings, FreeRDP_UseRdpSecurityLayer, TRUE))
3332 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3333 }
3334 }
3335 CommandLineParserFree(ptr);
3336 return 0;
3337}
3338
3339static int parse_encryption_methods_options(rdpSettings* settings,
3340 const COMMAND_LINE_ARGUMENT_A* arg)
3341{
3342 WINPR_ASSERT(settings);
3343 WINPR_ASSERT(arg);
3344
3345 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
3346 {
3347 size_t count = 0;
3348 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3349
3350 UINT32 EncryptionMethods = 0;
3351 for (UINT32 i = 0; i < count; i++)
3352 {
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;
3361 else
3362 WLog_ERR(TAG, "unknown encryption method '%s'", ptr[i]);
3363 }
3364
3365 if (!freerdp_settings_set_uint32(settings, FreeRDP_EncryptionMethods, EncryptionMethods))
3366 return COMMAND_LINE_ERROR;
3367 CommandLineParserFree(ptr);
3368 }
3369 return 0;
3370}
3371
3372static int parse_cert_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3373{
3374 WINPR_ASSERT(settings);
3375 WINPR_ASSERT(arg);
3376
3377 int rc = 0;
3378 size_t count = 0;
3379 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3380 for (size_t x = 0; (x < count) && (rc == 0); x++)
3381 {
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:";
3387
3388 const char* cur = ptr[x];
3389 if (option_equals(deny, cur))
3390 {
3391 if (!freerdp_settings_set_bool(settings, FreeRDP_AutoDenyCertificate, TRUE))
3392 return COMMAND_LINE_ERROR;
3393 }
3394 else if (option_equals(ignore, cur))
3395 {
3396 if (!freerdp_settings_set_bool(settings, FreeRDP_IgnoreCertificate, TRUE))
3397 return COMMAND_LINE_ERROR;
3398 }
3399 else if (option_equals(tofu, cur))
3400 {
3401 if (!freerdp_settings_set_bool(settings, FreeRDP_AutoAcceptCertificate, TRUE))
3402 return COMMAND_LINE_ERROR;
3403 }
3404 else if (option_starts_with(name, cur))
3405 {
3406 const char* val = &cur[strnlen(name, sizeof(name))];
3407 if (!freerdp_settings_set_string(settings, FreeRDP_CertificateName, val))
3408 rc = COMMAND_LINE_ERROR_MEMORY;
3409 }
3410 else if (option_starts_with(fingerprints, cur))
3411 {
3412 const char* val = &cur[strnlen(fingerprints, sizeof(fingerprints))];
3413 if (!freerdp_settings_append_string(settings, FreeRDP_CertificateAcceptedFingerprints,
3414 ",", val))
3415 rc = COMMAND_LINE_ERROR_MEMORY;
3416 }
3417 else
3418 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3419 }
3420 CommandLineParserFree(ptr);
3421
3422 return rc;
3423}
3424
3425static int parse_mouse_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3426{
3427 WINPR_ASSERT(settings);
3428 WINPR_ASSERT(arg);
3429
3430 size_t count = 0;
3431 char** ptr = CommandLineParseCommaSeparatedValuesEx("mouse", arg->Value, &count);
3432 int rc = 0;
3433 if (ptr)
3434 {
3435 for (size_t x = 1; x < count; x++)
3436 {
3437 const char* cur = ptr[x];
3438
3439 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(cur);
3440 if (bval == PARSE_FAIL)
3441 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3442 else
3443 {
3444 const BOOL val = bval != PARSE_OFF;
3445
3446 if (option_starts_with("relative", cur))
3447 {
3448 if (!freerdp_settings_set_bool(settings, FreeRDP_MouseUseRelativeMove, val))
3449 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3450 }
3451 else if (option_starts_with("grab", cur))
3452 {
3453 if (!freerdp_settings_set_bool(settings, FreeRDP_GrabMouse, val))
3454 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3455 }
3456 }
3457
3458 if (rc != 0)
3459 break;
3460 }
3461 }
3462 CommandLineParserFree(ptr);
3463
3464 return rc;
3465}
3466
3467static int parse_floatbar_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3468{
3469 WINPR_ASSERT(settings);
3470 WINPR_ASSERT(arg);
3471
3472 /* Defaults are enabled, visible, sticky, fullscreen */
3473 UINT32 Floatbar = 0x0017;
3474
3475 if (arg->Value)
3476 {
3477 char* start = arg->Value;
3478
3479 do
3480 {
3481 char* cur = start;
3482 start = strchr(start, ',');
3483
3484 if (start)
3485 {
3486 *start = '\0';
3487 start = start + 1;
3488 }
3489
3490 /* sticky:[on|off] */
3491 if (option_starts_with("sticky:", cur))
3492 {
3493 Floatbar &= ~0x02u;
3494
3495 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(cur);
3496 switch (bval)
3497 {
3498 case PARSE_ON:
3499 case PARSE_NONE:
3500 Floatbar |= 0x02u;
3501 break;
3502 case PARSE_OFF:
3503 Floatbar &= ~0x02u;
3504 break;
3505 case PARSE_FAIL:
3506 default:
3507 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3508 }
3509 }
3510 /* default:[visible|hidden] */
3511 else if (option_starts_with("default:", cur))
3512 {
3513 const char* val = cur + 8;
3514 Floatbar &= ~0x04u;
3515
3516 if (option_equals("visible", val))
3517 Floatbar |= 0x04u;
3518 else if (option_equals("hidden", val))
3519 Floatbar &= ~0x04u;
3520 else
3521 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3522 }
3523 /* show:[always|fullscreen|window] */
3524 else if (option_starts_with("show:", cur))
3525 {
3526 const char* val = cur + 5;
3527 Floatbar &= ~0x30u;
3528
3529 if (option_equals("always", val))
3530 Floatbar |= 0x30u;
3531 else if (option_equals("fullscreen", val))
3532 Floatbar |= 0x10u;
3533 else if (option_equals("window", val))
3534 Floatbar |= 0x20u;
3535 else
3536 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3537 }
3538 else
3539 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3540 } while (start);
3541 }
3542 if (!freerdp_settings_set_uint32(settings, FreeRDP_Floatbar, Floatbar))
3543 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3544 return 0;
3545}
3546
3547static int parse_reconnect_cookie_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3548{
3549 WINPR_ASSERT(settings);
3550 WINPR_ASSERT(arg);
3551
3552 BYTE* base64 = NULL;
3553 size_t length = 0;
3554 if (!arg->Value)
3555 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3556
3557 crypto_base64_decode((const char*)(arg->Value), strlen(arg->Value), &base64, &length);
3558
3559 if ((base64 != NULL) && (length == sizeof(ARC_SC_PRIVATE_PACKET)))
3560 {
3561 if (!freerdp_settings_set_pointer_len(settings, FreeRDP_ServerAutoReconnectCookie, base64,
3562 1))
3563 return COMMAND_LINE_ERROR;
3564 }
3565 else
3566 {
3567 WLog_ERR(TAG, "reconnect-cookie: invalid base64 '%s'", arg->Value);
3568 }
3569
3570 free(base64);
3571 return 0;
3572}
3573
3574static BOOL set_monitor_override(rdpSettings* settings, uint64_t flag)
3575{
3576 const FreeRDP_Settings_Keys_UInt64 key = FreeRDP_MonitorOverrideFlags;
3577 uint64_t mask = freerdp_settings_get_uint64(settings, key);
3578 mask |= flag;
3579 return freerdp_settings_set_uint64(settings, key, mask);
3580}
3581
3582static int parse_scale_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3583{
3584 WINPR_ASSERT(settings);
3585 WINPR_ASSERT(arg);
3586
3587 LONGLONG val = 0;
3588
3589 if (!value_to_int(arg->Value, &val, 100, 180))
3590 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3591
3592 switch (val)
3593 {
3594 case 100:
3595 case 140:
3596 case 180:
3597 if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopScaleFactor, (UINT32)val))
3598 return COMMAND_LINE_ERROR;
3599 if (!freerdp_settings_set_uint32(settings, FreeRDP_DeviceScaleFactor, (UINT32)val))
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);
3604 break;
3605
3606 default:
3607 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3608 }
3609 return 0;
3610}
3611
3612static int parse_scale_device_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3613{
3614 WINPR_ASSERT(settings);
3615 WINPR_ASSERT(arg);
3616
3617 LONGLONG val = 0;
3618
3619 if (!value_to_int(arg->Value, &val, 100, 180))
3620 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3621
3622 switch (val)
3623 {
3624 case 100:
3625 case 140:
3626 case 180:
3627 if (!freerdp_settings_set_uint32(settings, FreeRDP_DeviceScaleFactor, (UINT32)val))
3628 return COMMAND_LINE_ERROR;
3629 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_DEVICE_SCALE))
3630 return fail_at(arg, COMMAND_LINE_ERROR);
3631 break;
3632
3633 default:
3634 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3635 }
3636 return 0;
3637}
3638
3639static int parse_smartcard_logon_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3640{
3641 WINPR_ASSERT(settings);
3642 WINPR_ASSERT(arg);
3643
3644 size_t count = 0;
3645
3646 if (!freerdp_settings_set_bool(settings, FreeRDP_SmartcardLogon, TRUE))
3647 return COMMAND_LINE_ERROR;
3648
3649 char** ptr = CommandLineParseCommaSeparatedValuesEx("smartcard-logon", arg->Value, &count);
3650 if (ptr)
3651 {
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 }
3661 };
3662
3663 for (size_t x = 1; x < count; x++)
3664 {
3665 const char* cur = ptr[x];
3666 if (!parseSubOptions(settings, opts, ARRAYSIZE(opts), cur))
3667 {
3668 CommandLineParserFree(ptr);
3669 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3670 }
3671 }
3672 }
3673 CommandLineParserFree(ptr);
3674 return 0;
3675}
3676
3677static int parse_tune_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3678{
3679 WINPR_ASSERT(settings);
3680 WINPR_ASSERT(arg);
3681
3682 size_t count = 0;
3683 char** ptr = CommandLineParseCommaSeparatedValuesEx("tune", arg->Value, &count);
3684 if (!ptr)
3685 return COMMAND_LINE_ERROR;
3686 for (size_t x = 1; x < count; x++)
3687 {
3688 const char* cur = ptr[x];
3689 char* sep = strchr(cur, ':');
3690 if (!sep)
3691 {
3692 CommandLineParserFree(ptr);
3693 return COMMAND_LINE_ERROR;
3694 }
3695 *sep++ = '\0';
3696 if (!freerdp_settings_set_value_for_name(settings, cur, sep))
3697 {
3698 CommandLineParserFree(ptr);
3699 return COMMAND_LINE_ERROR;
3700 }
3701 }
3702
3703 CommandLineParserFree(ptr);
3704 return 0;
3705}
3706
3707static int parse_app_option_program(rdpSettings* settings, const char* cmd)
3708{
3709 const FreeRDP_Settings_Keys_Bool ids[] = { FreeRDP_RemoteApplicationMode,
3710 FreeRDP_RemoteAppLanguageBarSupported,
3711 FreeRDP_Workarea, FreeRDP_DisableWallpaper,
3712 FreeRDP_DisableFullWindowDrag };
3713
3714 if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationProgram, cmd))
3715 return COMMAND_LINE_ERROR_MEMORY;
3716
3717 for (size_t y = 0; y < ARRAYSIZE(ids); y++)
3718 {
3719 if (!freerdp_settings_set_bool(settings, ids[y], TRUE))
3720 return COMMAND_LINE_ERROR;
3721 }
3722 return CHANNEL_RC_OK;
3723}
3724
3725static int parse_aad_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3726{
3727 WINPR_ASSERT(settings);
3728 WINPR_ASSERT(arg);
3729
3730 int rc = CHANNEL_RC_OK;
3731 size_t count = 0;
3732 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3733 if (!ptr || (count == 0))
3734 rc = COMMAND_LINE_ERROR;
3735 else
3736 {
3737 struct app_map
3738 {
3739 const char* name;
3740 SSIZE_T id;
3741 int (*fkt)(rdpSettings* settings, const char* value);
3742 };
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 }
3748
3749 };
3750 for (size_t x = 0; x < count; x++)
3751 {
3752 BOOL handled = FALSE;
3753 const char* val = ptr[x];
3754
3755 if (option_starts_with("use-tenantid", val))
3756 {
3757 PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
3758 if (bval == PARSE_FAIL)
3759 {
3760 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3761 break;
3762 }
3763 else
3764 {
3765 if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayAvdUseTenantid,
3766 bval != PARSE_OFF))
3767 {
3768 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3769 break;
3770 }
3771 }
3772 continue;
3773 }
3774 for (size_t y = 0; y < ARRAYSIZE(amap); y++)
3775 {
3776 const struct app_map* cur = &amap[y];
3777 if (option_starts_with(cur->name, val))
3778 {
3779 const char* xval = &val[strlen(cur->name)];
3780 if (cur->fkt)
3781 rc = cur->fkt(settings, xval);
3782 else
3783 {
3784 const char* name = freerdp_settings_get_name_for_key(cur->id);
3785 if (!freerdp_settings_set_value_for_name(settings, name, xval))
3786 rc = COMMAND_LINE_ERROR_MEMORY;
3787 }
3788
3789 handled = TRUE;
3790 break;
3791 }
3792 }
3793
3794 if (!handled)
3795 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3796
3797 if (rc != 0)
3798 break;
3799 }
3800 }
3801
3802 CommandLineParserFree(ptr);
3803 return rc;
3804}
3805
3806static int parse_app_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3807{
3808 WINPR_ASSERT(settings);
3809 WINPR_ASSERT(arg);
3810
3811 int rc = CHANNEL_RC_OK;
3812 size_t count = 0;
3813 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3814 if (!ptr || (count == 0))
3815 rc = COMMAND_LINE_ERROR;
3816 else
3817 {
3818 struct app_map
3819 {
3820 const char* name;
3821 SSIZE_T id;
3822 int (*fkt)(rdpSettings* settings, const char* value);
3823 };
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++)
3834 {
3835 BOOL handled = FALSE;
3836 const char* val = ptr[x];
3837
3838 for (size_t y = 0; y < ARRAYSIZE(amap); y++)
3839 {
3840 const struct app_map* cur = &amap[y];
3841 if (option_starts_with(cur->name, val))
3842 {
3843 const char* xval = &val[strlen(cur->name)];
3844 if (cur->fkt)
3845 rc = cur->fkt(settings, xval);
3846 else
3847 {
3848 const char* name = freerdp_settings_get_name_for_key(cur->id);
3849 if (!freerdp_settings_set_value_for_name(settings, name, xval))
3850 rc = COMMAND_LINE_ERROR_MEMORY;
3851 }
3852
3853 handled = TRUE;
3854 break;
3855 }
3856 }
3857
3858#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
3859 if (!handled && (count == 1))
3860 {
3861 /* Legacy path, allow /app:command and /app:||command syntax */
3862 rc = parse_app_option_program(settings, val);
3863 }
3864 else
3865#endif
3866 if (!handled)
3867 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3868
3869 if (rc != 0)
3870 break;
3871 }
3872 }
3873
3874 CommandLineParserFree(ptr);
3875 return rc;
3876}
3877
3878static int parse_cache_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
3879{
3880 WINPR_ASSERT(settings);
3881 WINPR_ASSERT(arg);
3882
3883 int rc = CHANNEL_RC_OK;
3884 size_t count = 0;
3885 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
3886 if (!ptr || (count == 0))
3887 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3888
3889 for (size_t x = 0; x < count; x++)
3890 {
3891 const char* val = ptr[x];
3892
3893 if (option_starts_with("codec:", val))
3894 {
3895 if (!freerdp_settings_set_bool(settings, FreeRDP_BitmapCacheV3Enabled, TRUE))
3896 rc = COMMAND_LINE_ERROR;
3897 else if (option_equals(arg->Value, "rfx"))
3898 {
3899 if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteFxCodec, TRUE))
3900 rc = COMMAND_LINE_ERROR;
3901 }
3902 else if (option_equals(arg->Value, "nsc"))
3903 {
3904 if (!freerdp_settings_set_bool(settings, FreeRDP_NSCodec, TRUE))
3905 rc = COMMAND_LINE_ERROR;
3906 }
3907
3908#if defined(WITH_JPEG)
3909 else if (option_equals(arg->Value, "jpeg"))
3910 {
3911 if (!freerdp_settings_set_bool(settings, FreeRDP_JpegCodec, TRUE))
3912 rc = COMMAND_LINE_ERROR;
3913
3914 if (freerdp_settings_get_uint32(settings, FreeRDP_JpegQuality) == 0)
3915 {
3916 if (!freerdp_settings_set_uint32(settings, FreeRDP_JpegQuality, 75))
3917 return COMMAND_LINE_ERROR;
3918 }
3919 }
3920
3921#endif
3922 }
3923 else if (option_starts_with("persist-file:", val))
3924 {
3925
3926 if (!freerdp_settings_set_string(settings, FreeRDP_BitmapCachePersistFile, &val[13]))
3927 rc = COMMAND_LINE_ERROR_MEMORY;
3928 else if (!freerdp_settings_set_bool(settings, FreeRDP_BitmapCachePersistEnabled, TRUE))
3929 rc = COMMAND_LINE_ERROR;
3930 }
3931 else
3932 {
3933 const PARSE_ON_OFF_RESULT bval = parse_on_off_option(val);
3934 if (bval == PARSE_FAIL)
3935 rc = COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
3936 else
3937 {
3938 if (option_starts_with("bitmap", val))
3939 {
3940 if (!freerdp_settings_set_bool(settings, FreeRDP_BitmapCacheEnabled,
3941 bval != PARSE_OFF))
3942 rc = COMMAND_LINE_ERROR;
3943 }
3944 else if (option_starts_with("glyph", val))
3945 {
3946 if (!freerdp_settings_set_uint32(settings, FreeRDP_GlyphSupportLevel,
3947 bval != PARSE_OFF ? GLYPH_SUPPORT_FULL
3948 : GLYPH_SUPPORT_NONE))
3949 rc = COMMAND_LINE_ERROR;
3950 }
3951 else if (option_starts_with("persist", val))
3952 {
3953 if (!freerdp_settings_set_bool(settings, FreeRDP_BitmapCachePersistEnabled,
3954 bval != PARSE_OFF))
3955 rc = COMMAND_LINE_ERROR;
3956 }
3957 else if (option_starts_with("offscreen", val))
3958 {
3959 if (!freerdp_settings_set_uint32(settings, FreeRDP_OffscreenSupportLevel,
3960 bval != PARSE_OFF))
3961 rc = COMMAND_LINE_ERROR;
3962 }
3963 }
3964 }
3965 }
3966
3967 CommandLineParserFree(ptr);
3968 return rc;
3969}
3970
3971static BOOL parse_gateway_host_option(rdpSettings* settings, const char* host)
3972{
3973 WINPR_ASSERT(settings);
3974 WINPR_ASSERT(host);
3975
3976 char* name = NULL;
3977 int port = -1;
3978 if (!freerdp_parse_hostname(host, &name, &port))
3979 return FALSE;
3980 const BOOL rc = freerdp_settings_set_string(settings, FreeRDP_GatewayHostname, name);
3981 free(name);
3982 if (!rc)
3983 return FALSE;
3984 if (port != -1)
3985 {
3986 if (!freerdp_settings_set_uint32(settings, FreeRDP_GatewayPort, (UINT32)port))
3987 return FALSE;
3988 }
3989 if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayUseSameCredentials, TRUE))
3990 return FALSE;
3991 if (!freerdp_set_gateway_usage_method(settings, TSC_PROXY_MODE_DIRECT))
3992 return FALSE;
3993
3994 return TRUE;
3995}
3996
3997static BOOL parse_gateway_cred_option(rdpSettings* settings, const char* value,
3998 FreeRDP_Settings_Keys_String what)
3999{
4000 WINPR_ASSERT(settings);
4001 WINPR_ASSERT(value);
4002
4003 switch (what)
4004 {
4005 case FreeRDP_GatewayUsername:
4006 if (!freerdp_parse_username_settings(value, settings, FreeRDP_GatewayUsername,
4007 FreeRDP_GatewayDomain))
4008 return FALSE;
4009 break;
4010 default:
4011 if (!freerdp_settings_set_string(settings, what, value))
4012 return FALSE;
4013 break;
4014 }
4015
4016 return freerdp_settings_set_bool(settings, FreeRDP_GatewayUseSameCredentials, FALSE);
4017}
4018
4019static BOOL parse_gateway_type_option(rdpSettings* settings, const char* value)
4020{
4021 BOOL rc = FALSE;
4022
4023 WINPR_ASSERT(settings);
4024 WINPR_ASSERT(value);
4025
4026 if (option_equals(value, "rpc"))
4027 {
4028 if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayRpcTransport, TRUE) ||
4029 !freerdp_settings_set_bool(settings, FreeRDP_GatewayHttpTransport, FALSE) ||
4030 !freerdp_settings_set_bool(settings, FreeRDP_GatewayHttpUseWebsockets, FALSE) ||
4031 !freerdp_settings_set_bool(settings, FreeRDP_GatewayArmTransport, FALSE))
4032 return FALSE;
4033 rc = TRUE;
4034 }
4035 else
4036 {
4037 if (option_equals(value, "http"))
4038 {
4039 if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayRpcTransport, FALSE) ||
4040 !freerdp_settings_set_bool(settings, FreeRDP_GatewayHttpTransport, TRUE) ||
4041 !freerdp_settings_set_bool(settings, FreeRDP_GatewayArmTransport, FALSE))
4042 return FALSE;
4043 rc = TRUE;
4044 }
4045 else if (option_equals(value, "auto"))
4046 {
4047 if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayRpcTransport, TRUE) ||
4048 !freerdp_settings_set_bool(settings, FreeRDP_GatewayHttpTransport, TRUE) ||
4049 !freerdp_settings_set_bool(settings, FreeRDP_GatewayArmTransport, FALSE))
4050 return FALSE;
4051 rc = TRUE;
4052 }
4053 else if (option_equals(value, "arm"))
4054 {
4055 if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayRpcTransport, FALSE) ||
4056 !freerdp_settings_set_bool(settings, FreeRDP_GatewayHttpTransport, FALSE) ||
4057 !freerdp_settings_set_bool(settings, FreeRDP_GatewayHttpUseWebsockets, FALSE) ||
4058 !freerdp_settings_set_bool(settings, FreeRDP_GatewayArmTransport, TRUE))
4059 return FALSE;
4060 rc = TRUE;
4061 }
4062 }
4063 return rc;
4064}
4065
4066static BOOL parse_gateway_usage_option(rdpSettings* settings, const char* value)
4067{
4068 UINT32 type = 0;
4069
4070 WINPR_ASSERT(settings);
4071 WINPR_ASSERT(value);
4072
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;
4081 else
4082 {
4083 LONGLONG val = 0;
4084
4085 if (!value_to_int(value, &val, TSC_PROXY_MODE_NONE_DIRECT, TSC_PROXY_MODE_NONE_DETECT))
4086 return FALSE;
4087 }
4088
4089 return freerdp_set_gateway_usage_method(settings, type);
4090}
4091
4092static char* unescape(const char* str)
4093{
4094 char* copy = _strdup(str);
4095 if (!copy)
4096 return NULL;
4097
4098 bool escaped = false;
4099 char* dst = copy;
4100 while (*str != '\0')
4101 {
4102 char cur = *str++;
4103
4104 switch (cur)
4105 {
4106 case '\\':
4107 if (!escaped)
4108 {
4109 escaped = true;
4110 continue;
4111 }
4112 // fallthrough
4113 WINPR_FALLTHROUGH
4114 default:
4115 *dst++ = cur;
4116 escaped = false;
4117 break;
4118 }
4119 }
4120
4121 *dst = '\0';
4122
4123 return copy;
4124}
4125
4126static BOOL parse_gateway_options(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
4127{
4128 char* argval = NULL;
4129 BOOL rc = FALSE;
4130
4131 WINPR_ASSERT(settings);
4132 WINPR_ASSERT(arg);
4133
4134 size_t count = 0;
4135 char** ptr = CommandLineParseCommaSeparatedValues(arg->Value, &count);
4136 if (count == 0)
4137 return TRUE;
4138 WINPR_ASSERT(ptr);
4139
4140 if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayEnabled, TRUE))
4141 goto fail;
4142
4143 {
4144 BOOL allowHttpOpts = FALSE;
4145 for (size_t x = 0; x < count; x++)
4146 {
4147 BOOL validOption = FALSE;
4148 free(argval);
4149 argval = unescape(ptr[x]);
4150 if (!argval)
4151 goto fail;
4152
4153 const char* gw = option_starts_with("g:", argval);
4154 if (gw)
4155 {
4156 if (!parse_gateway_host_option(settings, gw))
4157 goto fail;
4158 validOption = TRUE;
4159 allowHttpOpts = FALSE;
4160 }
4161
4162 const char* gu = option_starts_with("u:", argval);
4163 if (gu)
4164 {
4165 if (!parse_gateway_cred_option(settings, gu, FreeRDP_GatewayUsername))
4166 goto fail;
4167 validOption = TRUE;
4168 allowHttpOpts = FALSE;
4169 }
4170
4171 const char* gd = option_starts_with("d:", argval);
4172 if (gd)
4173 {
4174 if (!parse_gateway_cred_option(settings, gd, FreeRDP_GatewayDomain))
4175 goto fail;
4176 validOption = TRUE;
4177 allowHttpOpts = FALSE;
4178 }
4179
4180 const char* gp = option_starts_with("p:", argval);
4181 if (gp)
4182 {
4183 if (!parse_gateway_cred_option(settings, gp, FreeRDP_GatewayPassword))
4184 goto fail;
4185 validOption = TRUE;
4186 allowHttpOpts = FALSE;
4187 }
4188
4189 const char* gt = option_starts_with("type:", argval);
4190 if (gt)
4191 {
4192 if (!parse_gateway_type_option(settings, gt))
4193 goto fail;
4194 validOption = TRUE;
4195 allowHttpOpts = freerdp_settings_get_bool(settings, FreeRDP_GatewayHttpTransport);
4196 }
4197
4198 const char* gat = option_starts_with("access-token:", argval);
4199 if (gat)
4200 {
4201 if (!freerdp_settings_set_string(settings, FreeRDP_GatewayAccessToken, gat))
4202 goto fail;
4203 validOption = TRUE;
4204 allowHttpOpts = FALSE;
4205 }
4206
4207 const char* bearer = option_starts_with("bearer:", argval);
4208 if (bearer)
4209 {
4210 if (!freerdp_settings_set_string(settings, FreeRDP_GatewayHttpExtAuthBearer,
4211 bearer))
4212 goto fail;
4213 validOption = TRUE;
4214 allowHttpOpts = FALSE;
4215 }
4216
4217 const char* gwurl = option_starts_with("url:", argval);
4218 if (gwurl)
4219 {
4220 if (!freerdp_settings_set_string(settings, FreeRDP_GatewayUrl, gwurl))
4221 goto fail;
4222 if (!freerdp_set_gateway_usage_method(settings, TSC_PROXY_MODE_DIRECT))
4223 goto fail;
4224 validOption = TRUE;
4225 allowHttpOpts = FALSE;
4226 }
4227
4228 const char* um = option_starts_with("usage-method:", argval);
4229 if (um)
4230 {
4231 if (!parse_gateway_usage_option(settings, um))
4232 goto fail;
4233 validOption = TRUE;
4234 allowHttpOpts = FALSE;
4235 }
4236
4237 if (allowHttpOpts)
4238 {
4239 if (option_equals(argval, "no-websockets"))
4240 {
4241 if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayHttpUseWebsockets,
4242 FALSE))
4243 goto fail;
4244 validOption = TRUE;
4245 }
4246 else if (option_equals(argval, "extauth-sspi-ntlm"))
4247 {
4248 if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayHttpExtAuthSspiNtlm,
4249 TRUE))
4250 goto fail;
4251 validOption = TRUE;
4252 }
4253 }
4254
4255 if (!validOption)
4256 goto fail;
4257 }
4258 }
4259
4260 rc = TRUE;
4261fail:
4262 free(argval);
4263 CommandLineParserFree(ptr);
4264 return rc;
4265}
4266
4267static void fill_credential_string(COMMAND_LINE_ARGUMENT_A* args, const char* value)
4268{
4269 WINPR_ASSERT(args);
4270 WINPR_ASSERT(value);
4271
4272 const COMMAND_LINE_ARGUMENT_A* arg = CommandLineFindArgumentA(args, value);
4273 if (!arg)
4274 return;
4275
4276 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
4277 FillMemory(arg->Value, strlen(arg->Value), '*');
4278}
4279
4280static void fill_credential_strings(COMMAND_LINE_ARGUMENT_A* args)
4281{
4282 for (size_t x = 0; x < ARRAYSIZE(credential_args); x++)
4283 {
4284 const char* cred = credential_args[x];
4285 fill_credential_string(args, cred);
4286 }
4287
4288 const COMMAND_LINE_ARGUMENT_A* arg = CommandLineFindArgumentA(args, "gateway");
4289 if (arg && ((arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT) != 0))
4290 {
4291 const char* gwcreds[] = { "p:", "access-token:" };
4292 char* saveptr = NULL;
4293 char* tok = strtok_s(arg->Value, ",", &saveptr);
4294 while (tok)
4295 {
4296 for (size_t x = 0; x < ARRAYSIZE(gwcreds); x++)
4297 {
4298 const char* opt = gwcreds[x];
4299 if (option_starts_with(opt, tok))
4300 {
4301 char* val = &tok[strlen(opt)];
4302 FillMemory(val, strlen(val), '*');
4303 }
4304 }
4305 tok = strtok_s(NULL, ",", &saveptr);
4306 }
4307 }
4308}
4309
4310static int parse_command_line_option_uint32(rdpSettings* settings,
4311 const COMMAND_LINE_ARGUMENT_A* arg,
4312 FreeRDP_Settings_Keys_UInt32 key, LONGLONG min,
4313 LONGLONG max)
4314{
4315 LONGLONG val = 0;
4316
4317 if (!value_to_int(arg->Value, &val, min, max))
4318 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4319
4320 if (!freerdp_settings_set_uint32(settings, key, (UINT32)val))
4321 return fail_at(arg, COMMAND_LINE_ERROR);
4322 return 0;
4323}
4324
4325#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
4326static int parse_deprecated_command_line(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg)
4327{
4328 int status = 0;
4329
4330 WINPR_ASSERT(settings);
4331 WINPR_ASSERT(arg);
4332
4333 BOOL enable = arg->Value ? TRUE : FALSE;
4334 CommandLineSwitchStart(arg) CommandLineSwitchCase(arg, "gfx-thin-client")
4335 {
4336 WLog_WARN(TAG, "/gfx-thin-client is deprecated, use /gfx:thin-client[:on|off] instead");
4337 if (!freerdp_settings_set_bool(settings, FreeRDP_GfxThinClient, enable))
4338 return fail_at(arg, COMMAND_LINE_ERROR);
4339
4340 if (freerdp_settings_get_bool(settings, FreeRDP_GfxThinClient))
4341 {
4342 if (!freerdp_settings_set_bool(settings, FreeRDP_GfxSmallCache, TRUE))
4343 return fail_at(arg, COMMAND_LINE_ERROR);
4344 }
4345
4346 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportGraphicsPipeline, TRUE))
4347 return fail_at(arg, COMMAND_LINE_ERROR);
4348 }
4349 CommandLineSwitchCase(arg, "gfx-small-cache")
4350 {
4351 WLog_WARN(TAG, "/gfx-small-cache is deprecated, use /gfx:small-cache[:on|off] instead");
4352 if (!freerdp_settings_set_bool(settings, FreeRDP_GfxSmallCache, enable))
4353 return fail_at(arg, COMMAND_LINE_ERROR);
4354
4355 if (enable)
4356 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportGraphicsPipeline, TRUE))
4357 return fail_at(arg, COMMAND_LINE_ERROR);
4358 }
4359 CommandLineSwitchCase(arg, "gfx-progressive")
4360 {
4361 WLog_WARN(TAG, "/gfx-progressive is deprecated, use /gfx:progressive[:on|off] instead");
4362 if (!freerdp_settings_set_bool(settings, FreeRDP_GfxProgressive, enable))
4363 return fail_at(arg, COMMAND_LINE_ERROR);
4364 if (!freerdp_settings_set_bool(settings, FreeRDP_GfxThinClient, !enable))
4365 return fail_at(arg, COMMAND_LINE_ERROR);
4366
4367 if (enable)
4368 {
4369 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportGraphicsPipeline, TRUE))
4370 return fail_at(arg, COMMAND_LINE_ERROR);
4371 }
4372 }
4373#ifdef WITH_GFX_H264
4374 CommandLineSwitchCase(arg, "gfx-h264")
4375 {
4376 WLog_WARN(TAG, "/gfx-h264 is deprecated, use /gfx:avc420 instead");
4377 int rc = parse_gfx_options(settings, arg);
4378 if (rc != 0)
4379 return fail_at(arg, rc);
4380 }
4381#endif
4382 CommandLineSwitchCase(arg, "app-workdir")
4383 {
4384 WLog_WARN(TAG,
4385 "/app-workdir:<directory> is deprecated, use /app:workdir:<directory> instead");
4386 if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationWorkingDir, arg->Value))
4387 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4388 }
4389 CommandLineSwitchCase(arg, "app-name")
4390 {
4391 WLog_WARN(TAG, "/app-name:<directory> is deprecated, use /app:name:<name> instead");
4392 if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationName, arg->Value))
4393 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4394 }
4395 CommandLineSwitchCase(arg, "app-icon")
4396 {
4397 WLog_WARN(TAG, "/app-icon:<filename> is deprecated, use /app:icon:<filename> instead");
4398 if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationIcon, arg->Value))
4399 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4400 }
4401 CommandLineSwitchCase(arg, "app-cmd")
4402 {
4403 WLog_WARN(TAG, "/app-cmd:<command> is deprecated, use /app:cmd:<command> instead");
4404 if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationCmdLine, arg->Value))
4405 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4406 }
4407 CommandLineSwitchCase(arg, "app-file")
4408 {
4409 WLog_WARN(TAG, "/app-file:<filename> is deprecated, use /app:file:<filename> instead");
4410 if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationFile, arg->Value))
4411 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4412 }
4413 CommandLineSwitchCase(arg, "app-guid")
4414 {
4415 WLog_WARN(TAG, "/app-guid:<guid> is deprecated, use /app:guid:<guid> instead");
4416 if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationGuid, arg->Value))
4417 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4418 }
4419 CommandLineSwitchCase(arg, "g")
4420 {
4421 if (!parse_gateway_host_option(settings, arg->Value))
4422 return fail_at(arg, COMMAND_LINE_ERROR);
4423 }
4424 CommandLineSwitchCase(arg, "gu")
4425 {
4426 if (!parse_gateway_cred_option(settings, arg->Value, FreeRDP_GatewayUsername))
4427 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4428 }
4429 CommandLineSwitchCase(arg, "gd")
4430 {
4431 if (!parse_gateway_cred_option(settings, arg->Value, FreeRDP_GatewayDomain))
4432 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4433 }
4434 CommandLineSwitchCase(arg, "gp")
4435 {
4436 if (!parse_gateway_cred_option(settings, arg->Value, FreeRDP_GatewayPassword))
4437 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4438 }
4439 CommandLineSwitchCase(arg, "gt")
4440 {
4441 if (!parse_gateway_type_option(settings, arg->Value))
4442 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4443 }
4444 CommandLineSwitchCase(arg, "gat")
4445 {
4446 if (!freerdp_settings_set_string(settings, FreeRDP_GatewayAccessToken, arg->Value))
4447 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4448 }
4449 CommandLineSwitchCase(arg, "gateway-usage-method")
4450 {
4451 if (!parse_gateway_usage_option(settings, arg->Value))
4452 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4453 }
4454 CommandLineSwitchCase(arg, "kbd-remap")
4455 {
4456 WLog_WARN(TAG, "/kbd-remap:<key>=<value>,<key2>=<value2> is deprecated, use "
4457 "/kbd:remap:<key>=<value>,remap:<key2>=<value2>,... instead");
4458 if (!freerdp_settings_set_string(settings, FreeRDP_KeyboardRemappingList, arg->Value))
4459 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4460 }
4461 CommandLineSwitchCase(arg, "kbd-lang")
4462 {
4463 LONGLONG val = 0;
4464
4465 WLog_WARN(TAG, "/kbd-lang:<value> is deprecated, use /kbd:lang:<value> instead");
4466 if (!value_to_int(arg->Value, &val, 1, UINT32_MAX))
4467 {
4468 WLog_ERR(TAG, "Could not identify keyboard active language %s", arg->Value);
4469 WLog_ERR(TAG, "Use /list:kbd-lang to list available layouts");
4470 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4471 }
4472
4473 if (!freerdp_settings_set_uint32(settings, FreeRDP_KeyboardCodePage, (UINT32)val))
4474 return fail_at(arg, COMMAND_LINE_ERROR);
4475 }
4476 CommandLineSwitchCase(arg, "kbd-type")
4477 {
4478 WLog_WARN(TAG, "/kbd-type:<value> is deprecated, use /kbd:type:<value> instead");
4479 const int rc =
4480 parse_command_line_option_uint32(settings, arg, FreeRDP_KeyboardType, 0, UINT32_MAX);
4481 if (rc != 0)
4482 return fail_at(arg, rc);
4483 }
4484 CommandLineSwitchCase(arg, "kbd-unicode")
4485 {
4486 WLog_WARN(TAG, "/kbd-unicode is deprecated, use /kbd:unicode[:on|off] instead");
4487 if (!freerdp_settings_set_bool(settings, FreeRDP_UnicodeInput, enable))
4488 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4489 }
4490 CommandLineSwitchCase(arg, "kbd-subtype")
4491 {
4492 WLog_WARN(TAG, "/kbd-subtype:<value> is deprecated, use /kbd:subtype:<value> instead");
4493 const int rc =
4494 parse_command_line_option_uint32(settings, arg, FreeRDP_KeyboardSubType, 0, UINT32_MAX);
4495 if (rc != 0)
4496 return fail_at(arg, rc);
4497 }
4498 CommandLineSwitchCase(arg, "kbd-fn-key")
4499 {
4500 WLog_WARN(TAG, "/kbd-fn-key:<value> is deprecated, use /kbd:fn-key:<value> instead");
4501 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_KeyboardFunctionKey,
4502 0, UINT32_MAX);
4503 if (rc != 0)
4504 return fail_at(arg, rc);
4505 }
4506 CommandLineSwitchCase(arg, "bitmap-cache")
4507 {
4508 WLog_WARN(TAG, "/bitmap-cache is deprecated, use /cache:bitmap[:on|off] instead");
4509 if (!freerdp_settings_set_bool(settings, FreeRDP_BitmapCacheEnabled, enable))
4510 return fail_at(arg, COMMAND_LINE_ERROR);
4511 }
4512 CommandLineSwitchCase(arg, "persist-cache")
4513 {
4514 WLog_WARN(TAG, "/persist-cache is deprecated, use /cache:persist[:on|off] instead");
4515 if (!freerdp_settings_set_bool(settings, FreeRDP_BitmapCachePersistEnabled, enable))
4516 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4517 }
4518 CommandLineSwitchCase(arg, "persist-cache-file")
4519 {
4520 WLog_WARN(TAG, "/persist-cache-file:<filename> is deprecated, use "
4521 "/cache:persist-file:<filename> instead");
4522 if (!freerdp_settings_set_string(settings, FreeRDP_BitmapCachePersistFile, arg->Value))
4523 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4524
4525 if (!freerdp_settings_set_bool(settings, FreeRDP_BitmapCachePersistEnabled, TRUE))
4526 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4527 }
4528 CommandLineSwitchCase(arg, "offscreen-cache")
4529 {
4530 WLog_WARN(TAG, "/bitmap-cache is deprecated, use /cache:bitmap[:on|off] instead");
4531 if (!freerdp_settings_set_uint32(settings, FreeRDP_OffscreenSupportLevel, (UINT32)enable))
4532 return fail_at(arg, COMMAND_LINE_ERROR);
4533 }
4534 CommandLineSwitchCase(arg, "glyph-cache")
4535 {
4536 WLog_WARN(TAG, "/glyph-cache is deprecated, use /cache:glyph[:on|off] instead");
4537 if (!freerdp_settings_set_uint32(settings, FreeRDP_GlyphSupportLevel,
4538 arg->Value ? GLYPH_SUPPORT_FULL : GLYPH_SUPPORT_NONE))
4539 return fail_at(arg, COMMAND_LINE_ERROR);
4540 }
4541 CommandLineSwitchCase(arg, "codec-cache")
4542 {
4543 WLog_WARN(TAG, "/codec-cache:<option> is deprecated, use /cache:codec:<option> instead");
4544 const int rc = parse_codec_cache_options(settings, arg);
4545 if (rc != 0)
4546 return fail_at(arg, rc);
4547 }
4548 CommandLineSwitchCase(arg, "sec-rdp")
4549 {
4550 WLog_WARN(TAG, "/sec-rdp is deprecated, use /sec:rdp[:on|off] instead");
4551 if (!freerdp_settings_set_bool(settings, FreeRDP_RdpSecurity, enable))
4552 return fail_at(arg, COMMAND_LINE_ERROR);
4553 }
4554 CommandLineSwitchCase(arg, "sec-tls")
4555 {
4556 WLog_WARN(TAG, "/sec-tls is deprecated, use /sec:tls[:on|off] instead");
4557 if (!freerdp_settings_set_bool(settings, FreeRDP_TlsSecurity, enable))
4558 return fail_at(arg, COMMAND_LINE_ERROR);
4559 }
4560 CommandLineSwitchCase(arg, "sec-nla")
4561 {
4562 WLog_WARN(TAG, "/sec-nla is deprecated, use /sec:nla[:on|off] instead");
4563 if (!freerdp_settings_set_bool(settings, FreeRDP_NlaSecurity, enable))
4564 return fail_at(arg, COMMAND_LINE_ERROR);
4565 }
4566 CommandLineSwitchCase(arg, "sec-ext")
4567 {
4568 WLog_WARN(TAG, "/sec-ext is deprecated, use /sec:ext[:on|off] instead");
4569 if (!freerdp_settings_set_bool(settings, FreeRDP_ExtSecurity, enable))
4570 return fail_at(arg, COMMAND_LINE_ERROR);
4571 }
4572 CommandLineSwitchCase(arg, "tls-ciphers")
4573 {
4574 WLog_WARN(TAG, "/tls-ciphers:<cipher list> is deprecated, use "
4575 "/tls:ciphers:<cipher list> instead");
4576 int rc = parse_tls_cipher_options(settings, arg);
4577 if (rc != 0)
4578 return fail_at(arg, rc);
4579 }
4580 CommandLineSwitchCase(arg, "tls-seclevel")
4581 {
4582 WLog_WARN(TAG, "/tls-seclevel:<level> is deprecated, use /tls:sec-level:<level> instead");
4583 int rc = parse_tls_cipher_options(settings, arg);
4584 if (rc != 0)
4585 return fail_at(arg, rc);
4586 }
4587 CommandLineSwitchCase(arg, "tls-secrets-file")
4588 {
4589 WLog_WARN(TAG, "/tls-secrets-file:<filename> is deprecated, use "
4590 "/tls:secrets-file:<filename> instead");
4591 int rc = parse_tls_cipher_options(settings, arg);
4592 if (rc != 0)
4593 return fail_at(arg, rc);
4594 }
4595 CommandLineSwitchCase(arg, "enforce-tlsv1_2")
4596 {
4597 WLog_WARN(TAG, "/enforce-tlsv1_2 is deprecated, use /tls:enforce:1.2 instead");
4598 int rc = parse_tls_cipher_options(settings, arg);
4599 if (rc != 0)
4600 return fail_at(arg, rc);
4601 }
4602 CommandLineSwitchCase(arg, "cert-name")
4603 {
4604 WLog_WARN(TAG, "/cert-name is deprecated, use /cert:name instead");
4605 if (!freerdp_settings_set_string(settings, FreeRDP_CertificateName, arg->Value))
4606 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4607 }
4608 CommandLineSwitchCase(arg, "cert-ignore")
4609 {
4610 WLog_WARN(TAG, "/cert-ignore is deprecated, use /cert:ignore instead");
4611 if (!freerdp_settings_set_bool(settings, FreeRDP_IgnoreCertificate, enable))
4612 return fail_at(arg, COMMAND_LINE_ERROR);
4613 }
4614 CommandLineSwitchCase(arg, "cert-tofu")
4615 {
4616 WLog_WARN(TAG, "/cert-tofu is deprecated, use /cert:tofu instead");
4617 if (!freerdp_settings_set_bool(settings, FreeRDP_AutoAcceptCertificate, enable))
4618 return fail_at(arg, COMMAND_LINE_ERROR);
4619 }
4620 CommandLineSwitchCase(arg, "cert-deny")
4621 {
4622 WLog_WARN(TAG, "/cert-deny is deprecated, use /cert:deny instead");
4623 if (!freerdp_settings_set_bool(settings, FreeRDP_AutoDenyCertificate, enable))
4624 return fail_at(arg, COMMAND_LINE_ERROR);
4625 }
4626 CommandLineSwitchDefault(arg)
4627 {
4628 status = -1;
4629 }
4630 CommandLineSwitchEnd(arg);
4631 return status;
4632}
4633#endif
4634
4635static int parse_command_line_option_timezone(rdpSettings* settings,
4636 const COMMAND_LINE_ARGUMENT_A* arg)
4637{
4638 BOOL found = FALSE;
4639 DWORD index = 0;
4640 DYNAMIC_TIME_ZONE_INFORMATION info = { 0 };
4641 char TimeZoneKeyName[ARRAYSIZE(info.TimeZoneKeyName) + 1] = { 0 };
4642 while (EnumDynamicTimeZoneInformation(index++, &info) != ERROR_NO_MORE_ITEMS)
4643 {
4644 (void)ConvertWCharNToUtf8(info.TimeZoneKeyName, ARRAYSIZE(info.TimeZoneKeyName),
4645 TimeZoneKeyName, ARRAYSIZE(TimeZoneKeyName));
4646
4647 WINPR_ASSERT(arg->Value);
4648 if (strncmp(TimeZoneKeyName, arg->Value, ARRAYSIZE(TimeZoneKeyName)) == 0)
4649 {
4650 found = TRUE;
4651 break;
4652 }
4653 }
4654 if (!found)
4655 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
4656
4657 if (!freerdp_settings_set_string(settings, FreeRDP_DynamicDSTTimeZoneKeyName, TimeZoneKeyName))
4658 return fail_at(arg, COMMAND_LINE_ERROR);
4659
4661 freerdp_settings_get_pointer_writable(settings, FreeRDP_ClientTimeZone);
4662 if (!tz)
4663 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4664
4665 tz->Bias = info.Bias;
4666 tz->DaylightBias = info.DaylightBias;
4667 tz->DaylightDate = info.DaylightDate;
4668 memcpy(tz->DaylightName, info.DaylightName, sizeof(tz->DaylightName));
4669 tz->StandardBias = info.StandardBias;
4670 tz->StandardDate = info.StandardDate;
4671 memcpy(tz->StandardName, info.StandardName, sizeof(tz->StandardName));
4672
4673 return 0;
4674}
4675
4676static int parse_command_line_option_window_pos(rdpSettings* settings,
4677 const COMMAND_LINE_ARGUMENT_A* arg)
4678{
4679 WINPR_ASSERT(settings);
4680 WINPR_ASSERT(arg);
4681
4682 unsigned long x = 0;
4683 unsigned long y = 0;
4684
4685 if (!arg->Value)
4686 return fail_at(arg, COMMAND_LINE_ERROR_MISSING_ARGUMENT);
4687
4688 if (!parseSizeValue(arg->Value, &x, &y) || x > UINT16_MAX || y > UINT16_MAX)
4689 {
4690 WLog_ERR(TAG, "invalid window-position argument");
4691 return fail_at(arg, COMMAND_LINE_ERROR_MISSING_ARGUMENT);
4692 }
4693
4694 if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopPosX, (UINT32)x))
4695 return fail_at(arg, COMMAND_LINE_ERROR);
4696 if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopPosY, (UINT32)y))
4697 return fail_at(arg, COMMAND_LINE_ERROR);
4698 return 0;
4699}
4700
4701static int parse_command_line(rdpSettings* settings, const COMMAND_LINE_ARGUMENT_A* arg,
4702 freerdp_command_line_handle_option_t handle_option,
4703 void* handle_userdata, BOOL* promptForPassword, char** user)
4704{
4705 WINPR_ASSERT(promptForPassword);
4706 WINPR_ASSERT(user);
4707
4708 do
4709 {
4710 BOOL enable = arg->Value ? TRUE : FALSE;
4711
4712 if (!(arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
4713 continue;
4714
4715 CommandLineSwitchStart(arg)
4716
4717 CommandLineSwitchCase(arg, "v")
4718 {
4719 const int rc = parse_host_options(settings, arg);
4720 if (rc != 0)
4721 return fail_at(arg, rc);
4722 }
4723 CommandLineSwitchCase(arg, "spn-class")
4724 {
4725 if (!freerdp_settings_set_string(settings, FreeRDP_AuthenticationServiceClass,
4726 arg->Value))
4727 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4728 }
4729 CommandLineSwitchCase(arg, "sspi-module")
4730 {
4731 if (!freerdp_settings_set_string(settings, FreeRDP_SspiModule, arg->Value))
4732 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4733 }
4734 CommandLineSwitchCase(arg, "winscard-module")
4735 {
4736 if (!freerdp_settings_set_string(settings, FreeRDP_WinSCardModule, arg->Value))
4737 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4738 }
4739 CommandLineSwitchCase(arg, "redirect-prefer")
4740 {
4741 const int rc = parse_redirect_prefer_options(settings, arg);
4742 if (rc != 0)
4743 return fail_at(arg, rc);
4744 }
4745 CommandLineSwitchCase(arg, "credentials-delegation")
4746 {
4747 if (!freerdp_settings_set_bool(settings, FreeRDP_DisableCredentialsDelegation, !enable))
4748 return fail_at(arg, COMMAND_LINE_ERROR);
4749 }
4750 CommandLineSwitchCase(arg, "prevent-session-lock")
4751 {
4752 const int rc = parse_prevent_session_lock_options(settings, arg);
4753 if (rc != 0)
4754 return fail_at(arg, rc);
4755 }
4756 CommandLineSwitchCase(arg, "vmconnect")
4757 {
4758 const int rc = parse_vmconnect_options(settings, arg);
4759 if (rc != 0)
4760 return fail_at(arg, rc);
4761 }
4762 CommandLineSwitchCase(arg, "w")
4763 {
4764 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_DesktopWidth, -1,
4765 UINT32_MAX);
4766 if (rc != 0)
4767 return fail_at(arg, rc);
4768 }
4769 CommandLineSwitchCase(arg, "h")
4770 {
4771 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_DesktopHeight,
4772 -1, UINT32_MAX);
4773 if (rc != 0)
4774 return fail_at(arg, rc);
4775 }
4776 CommandLineSwitchCase(arg, "size")
4777 {
4778 const int rc = parse_size_options(settings, arg);
4779 if (rc != 0)
4780 return fail_at(arg, rc);
4781 }
4782 CommandLineSwitchCase(arg, "f")
4783 {
4784 if (!freerdp_settings_set_bool(settings, FreeRDP_Fullscreen, enable))
4785 return fail_at(arg, COMMAND_LINE_ERROR);
4786 }
4787 CommandLineSwitchCase(arg, "suppress-output")
4788 {
4789 if (!freerdp_settings_set_bool(settings, FreeRDP_SuppressOutput, enable))
4790 return fail_at(arg, COMMAND_LINE_ERROR);
4791 }
4792 CommandLineSwitchCase(arg, "multimon")
4793 {
4794 if (!freerdp_settings_set_bool(settings, FreeRDP_UseMultimon, TRUE))
4795 return fail_at(arg, COMMAND_LINE_ERROR);
4796
4797 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
4798 {
4799 if (option_equals(arg->Value, str_force))
4800 {
4801 if (!freerdp_settings_set_bool(settings, FreeRDP_ForceMultimon, TRUE))
4802 return fail_at(arg, COMMAND_LINE_ERROR);
4803 }
4804 }
4805 }
4806 CommandLineSwitchCase(arg, "span")
4807 {
4808 if (!freerdp_settings_set_bool(settings, FreeRDP_SpanMonitors, enable))
4809 return fail_at(arg, COMMAND_LINE_ERROR);
4810 }
4811 CommandLineSwitchCase(arg, "workarea")
4812 {
4813 if (!freerdp_settings_set_bool(settings, FreeRDP_Workarea, enable))
4814 return fail_at(arg, COMMAND_LINE_ERROR);
4815 }
4816 CommandLineSwitchCase(arg, "monitors")
4817 {
4818 const int rc = parse_monitors_options(settings, arg);
4819 if (rc != 0)
4820 return fail_at(arg, rc);
4821 }
4822 CommandLineSwitchCase(arg, "t")
4823 {
4824 if (!freerdp_settings_set_string(settings, FreeRDP_WindowTitle, arg->Value))
4825 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4826 }
4827 CommandLineSwitchCase(arg, "decorations")
4828 {
4829 if (!freerdp_settings_set_bool(settings, FreeRDP_Decorations, enable))
4830 return fail_at(arg, COMMAND_LINE_ERROR);
4831 }
4832 CommandLineSwitchCase(arg, "dynamic-resolution")
4833 {
4834 const int rc = parse_dynamic_resolution_options(settings, arg);
4835 if (rc != 0)
4836 return fail_at(arg, rc);
4837 }
4838 CommandLineSwitchCase(arg, "smart-sizing")
4839 {
4840 const int rc = parse_smart_sizing_options(settings, arg);
4841 if (rc != 0)
4842 return fail_at(arg, rc);
4843 }
4844 CommandLineSwitchCase(arg, "bpp")
4845 {
4846 const int rc = parse_bpp_options(settings, arg);
4847 if (rc != 0)
4848 return fail_at(arg, rc);
4849 }
4850 CommandLineSwitchCase(arg, "admin")
4851 {
4852 if (!freerdp_settings_set_bool(settings, FreeRDP_ConsoleSession, enable))
4853 return fail_at(arg, COMMAND_LINE_ERROR);
4854 }
4855 CommandLineSwitchCase(arg, "relax-order-checks")
4856 {
4857 if (!freerdp_settings_set_bool(settings, FreeRDP_AllowUnanouncedOrdersFromServer,
4858 enable))
4859 return fail_at(arg, COMMAND_LINE_ERROR);
4860 }
4861 CommandLineSwitchCase(arg, "restricted-admin")
4862 {
4863 if (!freerdp_settings_set_bool(settings, FreeRDP_ConsoleSession, enable))
4864 return fail_at(arg, COMMAND_LINE_ERROR);
4865 if (!freerdp_settings_set_bool(settings, FreeRDP_RestrictedAdminModeRequired, enable))
4866 return fail_at(arg, COMMAND_LINE_ERROR);
4867 }
4868#ifndef CHANNEL_RDPEAR_CLIENT
4869 CommandLineSwitchCase(arg, "remoteGuard")
4870 {
4871 if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteCredentialGuard, TRUE))
4872 return fail_at(arg, COMMAND_LINE_ERROR);
4873 if (!freerdp_settings_set_bool(settings, FreeRDP_ExtSecurity, TRUE))
4874 return fail_at(arg, COMMAND_LINE_ERROR);
4875 }
4876#endif
4877 CommandLineSwitchCase(arg, "pth")
4878 {
4879 if (!freerdp_settings_set_bool(settings, FreeRDP_ConsoleSession, TRUE))
4880 return fail_at(arg, COMMAND_LINE_ERROR);
4881 if (!freerdp_settings_set_bool(settings, FreeRDP_RestrictedAdminModeRequired, TRUE))
4882 return fail_at(arg, COMMAND_LINE_ERROR);
4883
4884 if (!freerdp_settings_set_string(settings, FreeRDP_PasswordHash, arg->Value))
4885 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4886 }
4887 CommandLineSwitchCase(arg, "client-hostname")
4888 {
4889 if (!freerdp_settings_set_string(settings, FreeRDP_ClientHostname, arg->Value))
4890 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4891 }
4892 CommandLineSwitchCase(arg, "kbd")
4893 {
4894 int rc = parse_kbd_options(settings, arg);
4895 if (rc != 0)
4896 return fail_at(arg, rc);
4897 }
4898
4899 CommandLineSwitchCase(arg, "u")
4900 {
4901 WINPR_ASSERT(arg->Value);
4902 *user = arg->Value;
4903 }
4904 CommandLineSwitchCase(arg, "d")
4905 {
4906 if (!freerdp_settings_set_string(settings, FreeRDP_Domain, arg->Value))
4907 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4908 }
4909 CommandLineSwitchCase(arg, "p")
4910 {
4911 if (!freerdp_settings_set_string(settings, FreeRDP_Password, arg->Value))
4912 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
4913 }
4914 CommandLineSwitchCase(arg, "gateway")
4915 {
4916 if (!parse_gateway_options(settings, arg))
4917 return fail_at(arg, COMMAND_LINE_ERROR);
4918 }
4919 CommandLineSwitchCase(arg, "proxy")
4920 {
4921 const int rc = parse_proxy_options(settings, arg);
4922 if (rc != 0)
4923 return fail_at(arg, rc);
4924 }
4925
4926 CommandLineSwitchCase(arg, "azure")
4927 {
4928 int rc = parse_aad_options(settings, arg);
4929 if (rc != 0)
4930 return fail_at(arg, rc);
4931 }
4932 CommandLineSwitchCase(arg, "app")
4933 {
4934 int rc = parse_app_options(settings, arg);
4935 if (rc != 0)
4936 return fail_at(arg, rc);
4937 }
4938 CommandLineSwitchCase(arg, "load-balance-info")
4939 {
4940 WINPR_ASSERT(arg->Value);
4941 if (!freerdp_settings_set_pointer_len(settings, FreeRDP_LoadBalanceInfo, arg->Value,
4942 strlen(arg->Value)))
4943 return fail_at(arg, COMMAND_LINE_ERROR);
4944 }
4945
4946 CommandLineSwitchCase(arg, "compression")
4947 {
4948 if (!freerdp_settings_set_bool(settings, FreeRDP_CompressionEnabled, enable))
4949 return fail_at(arg, COMMAND_LINE_ERROR);
4950 }
4951 CommandLineSwitchCase(arg, "compression-level")
4952 {
4953 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_CompressionLevel,
4954 0, UINT32_MAX);
4955 if (rc != 0)
4956 return fail_at(arg, rc);
4957 }
4958 CommandLineSwitchCase(arg, "drives")
4959 {
4960 if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectDrives, enable))
4961 return fail_at(arg, COMMAND_LINE_ERROR);
4962 }
4963 CommandLineSwitchCase(arg, "dump")
4964 {
4965 const int rc = parse_dump_options(settings, arg);
4966 if (rc != 0)
4967 return fail_at(arg, rc);
4968 }
4969 CommandLineSwitchCase(arg, "disable-output")
4970 {
4971 if (!freerdp_settings_set_bool(settings, FreeRDP_DeactivateClientDecoding, enable))
4972 return fail_at(arg, COMMAND_LINE_ERROR);
4973 }
4974 CommandLineSwitchCase(arg, "home-drive")
4975 {
4976 if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectHomeDrive, enable))
4977 return fail_at(arg, COMMAND_LINE_ERROR);
4978 }
4979 CommandLineSwitchCase(arg, "ipv4")
4980 {
4981 if (arg->Value != NULL && strncmp(arg->Value, str_force, ARRAYSIZE(str_force)) == 0)
4982 {
4983 if (!freerdp_settings_set_uint32(settings, FreeRDP_ForceIPvX, 4))
4984 return fail_at(arg, COMMAND_LINE_ERROR);
4985 }
4986 else
4987 {
4988 if (!freerdp_settings_set_bool(settings, FreeRDP_PreferIPv6OverIPv4, FALSE))
4989 return fail_at(arg, COMMAND_LINE_ERROR);
4990 }
4991 }
4992 CommandLineSwitchCase(arg, "ipv6")
4993 {
4994 if (arg->Value != NULL && strncmp(arg->Value, str_force, ARRAYSIZE(str_force)) == 0)
4995 {
4996 if (!freerdp_settings_set_uint32(settings, FreeRDP_ForceIPvX, 6))
4997 return fail_at(arg, COMMAND_LINE_ERROR);
4998 }
4999 else
5000 {
5001 if (!freerdp_settings_set_bool(settings, FreeRDP_PreferIPv6OverIPv4, TRUE))
5002 return fail_at(arg, COMMAND_LINE_ERROR);
5003 }
5004 }
5005 CommandLineSwitchCase(arg, "clipboard")
5006 {
5007 const int rc = parse_clipboard_options(settings, arg);
5008 if (rc != 0)
5009 return fail_at(arg, rc);
5010 }
5011 CommandLineSwitchCase(arg, "server-name")
5012 {
5013 if (!freerdp_settings_set_string(settings, FreeRDP_UserSpecifiedServerName, arg->Value))
5014 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5015 }
5016 CommandLineSwitchCase(arg, "shell")
5017 {
5018 if (!freerdp_settings_set_string(settings, FreeRDP_AlternateShell, arg->Value))
5019 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5020 }
5021 CommandLineSwitchCase(arg, "shell-dir")
5022 {
5023 if (!freerdp_settings_set_string(settings, FreeRDP_ShellWorkingDirectory, arg->Value))
5024 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5025 }
5026 CommandLineSwitchCase(arg, "audio-mode")
5027 {
5028 const int rc = parse_audio_mode_options(settings, arg);
5029 if (rc != 0)
5030 return fail_at(arg, rc);
5031 }
5032 CommandLineSwitchCase(arg, "network")
5033 {
5034 const int rc = parse_network_options(settings, arg);
5035 if (rc != 0)
5036 return fail_at(arg, rc);
5037 }
5038 CommandLineSwitchCase(arg, "fonts")
5039 {
5040 if (!freerdp_settings_set_bool(settings, FreeRDP_AllowFontSmoothing, enable))
5041 return fail_at(arg, COMMAND_LINE_ERROR);
5042 }
5043 CommandLineSwitchCase(arg, "wallpaper")
5044 {
5045 if (!freerdp_settings_set_bool(settings, FreeRDP_DisableWallpaper, !enable))
5046 return fail_at(arg, COMMAND_LINE_ERROR);
5047 }
5048 CommandLineSwitchCase(arg, "window-drag")
5049 {
5050 if (!freerdp_settings_set_bool(settings, FreeRDP_DisableFullWindowDrag, !enable))
5051 return fail_at(arg, COMMAND_LINE_ERROR);
5052 }
5053 CommandLineSwitchCase(arg, "window-position")
5054 {
5055 const int rc = parse_command_line_option_window_pos(settings, arg);
5056 if (rc != 0)
5057 return fail_at(arg, rc);
5058 }
5059 CommandLineSwitchCase(arg, "menu-anims")
5060 {
5061 if (!freerdp_settings_set_bool(settings, FreeRDP_DisableMenuAnims, !enable))
5062 return fail_at(arg, COMMAND_LINE_ERROR);
5063 }
5064 CommandLineSwitchCase(arg, "themes")
5065 {
5066 if (!freerdp_settings_set_bool(settings, FreeRDP_DisableThemes, !enable))
5067 return fail_at(arg, COMMAND_LINE_ERROR);
5068 }
5069 CommandLineSwitchCase(arg, "timeout")
5070 {
5071 const int rc =
5072 parse_command_line_option_uint32(settings, arg, FreeRDP_TcpAckTimeout, 0, 600000);
5073 if (rc != 0)
5074 return fail_at(arg, rc);
5075 }
5076 CommandLineSwitchCase(arg, "timezone")
5077 {
5078 const int rc = parse_command_line_option_timezone(settings, arg);
5079 if (rc != 0)
5080 return fail_at(arg, rc);
5081 }
5082 CommandLineSwitchCase(arg, "aero")
5083 {
5084 if (!freerdp_settings_set_bool(settings, FreeRDP_AllowDesktopComposition, enable))
5085 return fail_at(arg, COMMAND_LINE_ERROR);
5086 }
5087 CommandLineSwitchCase(arg, "gdi")
5088 {
5089 if (option_equals(arg->Value, "sw"))
5090 {
5091 if (!freerdp_settings_set_bool(settings, FreeRDP_SoftwareGdi, TRUE))
5092 return fail_at(arg, COMMAND_LINE_ERROR);
5093 }
5094 else if (option_equals(arg->Value, "hw"))
5095 {
5096 if (!freerdp_settings_set_bool(settings, FreeRDP_SoftwareGdi, FALSE))
5097 return fail_at(arg, COMMAND_LINE_ERROR);
5098 }
5099 else
5100 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5101 }
5102 CommandLineSwitchCase(arg, "gfx")
5103 {
5104 int rc = parse_gfx_options(settings, arg);
5105 if (rc != 0)
5106 return fail_at(arg, rc);
5107 }
5108
5109 CommandLineSwitchCase(arg, "rfx")
5110 {
5111 if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteFxCodec, enable))
5112 return fail_at(arg, COMMAND_LINE_ERROR);
5113 }
5114 CommandLineSwitchCase(arg, "rfx-mode")
5115 {
5116 if (!arg->Value)
5117 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5118
5119 if (option_equals(arg->Value, "video"))
5120 {
5121 if (!freerdp_settings_set_uint32(settings, FreeRDP_RemoteFxCodecMode, 0x00))
5122 return fail_at(arg, COMMAND_LINE_ERROR);
5123 }
5124 else if (option_equals(arg->Value, "image"))
5125 {
5126 if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteFxImageCodec, TRUE))
5127 return fail_at(arg, COMMAND_LINE_ERROR);
5128 if (!freerdp_settings_set_uint32(settings, FreeRDP_RemoteFxCodecMode, 0x02))
5129 return fail_at(arg, COMMAND_LINE_ERROR);
5130 }
5131 }
5132 CommandLineSwitchCase(arg, "frame-ack")
5133 {
5134 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_FrameAcknowledge,
5135 0, UINT32_MAX);
5136 if (rc != 0)
5137 return fail_at(arg, rc);
5138 }
5139 CommandLineSwitchCase(arg, "nsc")
5140 {
5141 if (!freerdp_settings_set_bool(settings, FreeRDP_NSCodec, enable))
5142 return fail_at(arg, COMMAND_LINE_ERROR);
5143 }
5144#if defined(WITH_JPEG)
5145 CommandLineSwitchCase(arg, "jpeg")
5146 {
5147 if (!freerdp_settings_set_bool(settings, FreeRDP_JpegCodec, enable))
5148 return fail_at(arg, COMMAND_LINE_ERROR);
5149 if (!freerdp_settings_set_uint32(settings, FreeRDP_JpegQuality, 75))
5150 return fail_at(arg, COMMAND_LINE_ERROR);
5151 }
5152 CommandLineSwitchCase(arg, "jpeg-quality")
5153 {
5154 LONGLONG val = 0;
5155
5156 if (!value_to_int(arg->Value, &val, 0, 100))
5157 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5158
5159 if (!freerdp_settings_set_uint32(settings, FreeRDP_JpegQuality, (UINT32)val))
5160 return fail_at(arg, COMMAND_LINE_ERROR);
5161 }
5162#endif
5163 CommandLineSwitchCase(arg, "nego")
5164 {
5165 if (!freerdp_settings_set_bool(settings, FreeRDP_NegotiateSecurityLayer, enable))
5166 return fail_at(arg, COMMAND_LINE_ERROR);
5167 }
5168 CommandLineSwitchCase(arg, "pcb")
5169 {
5170 if (!freerdp_settings_set_bool(settings, FreeRDP_SendPreconnectionPdu, TRUE))
5171 return fail_at(arg, COMMAND_LINE_ERROR);
5172
5173 if (!freerdp_settings_set_string(settings, FreeRDP_PreconnectionBlob, arg->Value))
5174 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5175 }
5176 CommandLineSwitchCase(arg, "pcid")
5177 {
5178 const int rc = parse_command_line_option_uint32(settings, arg, FreeRDP_PreconnectionId,
5179 0, UINT32_MAX);
5180 if (rc != 0)
5181 return fail_at(arg, rc);
5182 if (!freerdp_settings_set_bool(settings, FreeRDP_SendPreconnectionPdu, TRUE))
5183 return fail_at(arg, COMMAND_LINE_ERROR);
5184 }
5185#ifdef _WIN32
5186 CommandLineSwitchCase(arg, "connect-child-session")
5187 {
5188 if (!freerdp_settings_set_string(settings, FreeRDP_AuthenticationServiceClass,
5189 "vs-debug") ||
5190 !freerdp_settings_set_string(settings, FreeRDP_ServerHostname, "localhost") ||
5191 !freerdp_settings_set_string(settings, FreeRDP_AuthenticationPackageList, "ntlm") ||
5192 !freerdp_settings_set_string(settings, FreeRDP_ClientAddress, "0.0.0.0") ||
5193 !freerdp_settings_set_bool(settings, FreeRDP_NegotiateSecurityLayer, FALSE) ||
5194 !freerdp_settings_set_bool(settings, FreeRDP_VmConnectMode, TRUE) ||
5195 !freerdp_settings_set_bool(settings, FreeRDP_ConnectChildSession, TRUE) ||
5196 !freerdp_settings_set_bool(settings, FreeRDP_NlaSecurity, TRUE) ||
5197 !freerdp_settings_set_uint32(settings, FreeRDP_AuthenticationLevel, 0) ||
5198 !freerdp_settings_set_bool(settings, FreeRDP_NetworkAutoDetect, TRUE) ||
5199 !freerdp_settings_set_uint32(settings, FreeRDP_ConnectionType, CONNECTION_TYPE_LAN))
5200 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5201 }
5202#endif
5203 CommandLineSwitchCase(arg, "sec")
5204 {
5205 const int rc = parse_sec_options(settings, arg);
5206 if (rc != 0)
5207 return fail_at(arg, rc);
5208 }
5209 CommandLineSwitchCase(arg, "encryption-methods")
5210 {
5211 const int rc = parse_encryption_methods_options(settings, arg);
5212 if (rc != 0)
5213 return fail_at(arg, rc);
5214 }
5215 CommandLineSwitchCase(arg, "args-from")
5216 {
5217 WLog_ERR(TAG, "/args-from:%s can not be used in combination with other arguments!",
5218 arg->Value);
5219 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5220 }
5221 CommandLineSwitchCase(arg, "from-stdin")
5222 {
5223 if (!freerdp_settings_set_bool(settings, FreeRDP_CredentialsFromStdin, TRUE))
5224 return fail_at(arg, COMMAND_LINE_ERROR);
5225
5226 if (arg->Flags & COMMAND_LINE_VALUE_PRESENT)
5227 {
5228 if (!arg->Value)
5229 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5230 *promptForPassword = (option_equals(arg->Value, str_force));
5231
5232 if (!*promptForPassword)
5233 return fail_at(arg, COMMAND_LINE_ERROR);
5234 }
5235 }
5236 CommandLineSwitchCase(arg, "log-level")
5237 {
5238 wLog* root = WLog_GetRoot();
5239
5240 if (!WLog_SetStringLogLevel(root, arg->Value))
5241 return fail_at(arg, COMMAND_LINE_ERROR);
5242 }
5243 CommandLineSwitchCase(arg, "log-filters")
5244 {
5245 if (!WLog_AddStringLogFilters(arg->Value))
5246 return fail_at(arg, COMMAND_LINE_ERROR);
5247 }
5248 CommandLineSwitchCase(arg, "tls")
5249 {
5250 int rc = parse_tls_options(settings, arg);
5251 if (rc != 0)
5252 return fail_at(arg, rc);
5253 }
5254 CommandLineSwitchCase(arg, "cert")
5255 {
5256 const int rc = parse_cert_options(settings, arg);
5257 if (rc != 0)
5258 return fail_at(arg, rc);
5259 }
5260 CommandLineSwitchCase(arg, "authentication")
5261 {
5262 if (!freerdp_settings_set_bool(settings, FreeRDP_Authentication, enable))
5263 return fail_at(arg, COMMAND_LINE_ERROR);
5264 }
5265 CommandLineSwitchCase(arg, "encryption")
5266 {
5267 if (!freerdp_settings_set_bool(settings, FreeRDP_UseRdpSecurityLayer, !enable))
5268 return fail_at(arg, COMMAND_LINE_ERROR);
5269 }
5270 CommandLineSwitchCase(arg, "grab-keyboard")
5271 {
5272 if (!freerdp_settings_set_bool(settings, FreeRDP_GrabKeyboard, enable))
5273 return fail_at(arg, COMMAND_LINE_ERROR);
5274 }
5275 CommandLineSwitchCase(arg, "grab-mouse")
5276 {
5277 if (!freerdp_settings_set_bool(settings, FreeRDP_GrabMouse, enable))
5278 return fail_at(arg, COMMAND_LINE_ERROR);
5279 }
5280 CommandLineSwitchCase(arg, "mouse-relative")
5281 {
5282 if (!freerdp_settings_set_bool(settings, FreeRDP_MouseUseRelativeMove, enable))
5283 return fail_at(arg, COMMAND_LINE_ERROR);
5284 }
5285 CommandLineSwitchCase(arg, "mouse")
5286 {
5287 const int rc = parse_mouse_options(settings, arg);
5288 if (rc != 0)
5289 return fail_at(arg, rc);
5290 }
5291 CommandLineSwitchCase(arg, "unmap-buttons")
5292 {
5293 if (!freerdp_settings_set_bool(settings, FreeRDP_UnmapButtons, enable))
5294 return fail_at(arg, COMMAND_LINE_ERROR);
5295 }
5296 CommandLineSwitchCase(arg, "toggle-fullscreen")
5297 {
5298 if (!freerdp_settings_set_bool(settings, FreeRDP_ToggleFullscreen, enable))
5299 return fail_at(arg, COMMAND_LINE_ERROR);
5300 }
5301 CommandLineSwitchCase(arg, "force-console-callbacks")
5302 {
5303 if (!freerdp_settings_set_bool(settings, FreeRDP_UseCommonStdioCallbacks, enable))
5304 return fail_at(arg, COMMAND_LINE_ERROR);
5305 }
5306 CommandLineSwitchCase(arg, "floatbar")
5307 {
5308 const int rc = parse_floatbar_options(settings, arg);
5309 if (rc != 0)
5310 return fail_at(arg, rc);
5311 }
5312 CommandLineSwitchCase(arg, "mouse-motion")
5313 {
5314 if (!freerdp_settings_set_bool(settings, FreeRDP_MouseMotion, enable))
5315 return fail_at(arg, COMMAND_LINE_ERROR);
5316 }
5317 CommandLineSwitchCase(arg, "parent-window")
5318 {
5319 ULONGLONG val = 0;
5320
5321 if (!value_to_uint(arg->Value, &val, 0, UINT64_MAX))
5322 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5323
5324 if (!freerdp_settings_set_uint64(settings, FreeRDP_ParentWindowId, (UINT64)val))
5325 return fail_at(arg, COMMAND_LINE_ERROR);
5326 }
5327 CommandLineSwitchCase(arg, "client-build-number")
5328 {
5329 const int rc =
5330 parse_command_line_option_uint32(settings, arg, FreeRDP_ClientBuild, 0, UINT32_MAX);
5331 if (rc != 0)
5332 return fail_at(arg, rc);
5333 }
5334 CommandLineSwitchCase(arg, "cache")
5335 {
5336 int rc = parse_cache_options(settings, arg);
5337 if (rc != 0)
5338 return fail_at(arg, rc);
5339 }
5340
5341 CommandLineSwitchCase(arg, "max-fast-path-size")
5342 {
5343 const int rc = parse_command_line_option_uint32(
5344 settings, arg, FreeRDP_MultifragMaxRequestSize, 0, UINT32_MAX);
5345 if (rc != 0)
5346 return fail_at(arg, rc);
5347 }
5348 CommandLineSwitchCase(arg, "auto-request-control")
5349 {
5350 if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteAssistanceRequestControl,
5351 enable))
5352 return fail_at(arg, COMMAND_LINE_ERROR);
5353 }
5354 CommandLineSwitchCase(arg, "async-update")
5355 {
5356 if (!freerdp_settings_set_bool(settings, FreeRDP_AsyncUpdate, enable))
5357 return fail_at(arg, COMMAND_LINE_ERROR);
5358 }
5359 CommandLineSwitchCase(arg, "async-channels")
5360 {
5361 if (!freerdp_settings_set_bool(settings, FreeRDP_AsyncChannels, enable))
5362 return fail_at(arg, COMMAND_LINE_ERROR);
5363 }
5364 CommandLineSwitchCase(arg, "wm-class")
5365 {
5366 if (!freerdp_settings_set_string(settings, FreeRDP_WmClass, arg->Value))
5367 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5368 }
5369 CommandLineSwitchCase(arg, "play-rfx")
5370 {
5371 if (!freerdp_settings_set_string(settings, FreeRDP_PlayRemoteFxFile, arg->Value))
5372 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5373
5374 if (!freerdp_settings_set_bool(settings, FreeRDP_PlayRemoteFx, TRUE))
5375 return fail_at(arg, COMMAND_LINE_ERROR);
5376 }
5377 CommandLineSwitchCase(arg, "auth-only")
5378 {
5379 if (!freerdp_settings_set_bool(settings, FreeRDP_AuthenticationOnly, enable))
5380 return fail_at(arg, COMMAND_LINE_ERROR);
5381 }
5382 CommandLineSwitchCase(arg, "auth-pkg-list")
5383 {
5384 if (!freerdp_settings_set_string(settings, FreeRDP_AuthenticationPackageList,
5385 arg->Value))
5386 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5387 }
5388 CommandLineSwitchCase(arg, "auto-reconnect")
5389 {
5390 if (!freerdp_settings_set_bool(settings, FreeRDP_AutoReconnectionEnabled, enable))
5391 return fail_at(arg, COMMAND_LINE_ERROR);
5392 }
5393 CommandLineSwitchCase(arg, "auto-reconnect-max-retries")
5394 {
5395 const int rc = parse_command_line_option_uint32(
5396 settings, arg, FreeRDP_AutoReconnectMaxRetries, 0, 1000);
5397 if (rc != 0)
5398 return fail_at(arg, rc);
5399 }
5400 CommandLineSwitchCase(arg, "reconnect-cookie")
5401 {
5402 const int rc = parse_reconnect_cookie_options(settings, arg);
5403 if (rc != 0)
5404 return fail_at(arg, rc);
5405 }
5406 CommandLineSwitchCase(arg, "print-reconnect-cookie")
5407 {
5408 if (!freerdp_settings_set_bool(settings, FreeRDP_PrintReconnectCookie, enable))
5409 return fail_at(arg, COMMAND_LINE_ERROR);
5410 }
5411 CommandLineSwitchCase(arg, "pwidth")
5412 {
5413 const int rc = parse_command_line_option_uint32(
5414 settings, arg, FreeRDP_DesktopPhysicalWidth, 0, UINT32_MAX);
5415 if (rc != 0)
5416 return fail_at(arg, rc);
5417 }
5418 CommandLineSwitchCase(arg, "pheight")
5419 {
5420 const int rc = parse_command_line_option_uint32(
5421 settings, arg, FreeRDP_DesktopPhysicalHeight, 0, UINT32_MAX);
5422 if (rc != 0)
5423 return fail_at(arg, rc);
5424 }
5425 CommandLineSwitchCase(arg, "orientation")
5426 {
5427 LONGLONG val = 0;
5428
5429 if (!value_to_int(arg->Value, &val, 0, UINT16_MAX))
5430 return fail_at(arg, COMMAND_LINE_ERROR_UNEXPECTED_VALUE);
5431
5432 if (!freerdp_settings_set_uint16(settings, FreeRDP_DesktopOrientation, (UINT16)val))
5433 return fail_at(arg, COMMAND_LINE_ERROR);
5434 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_ORIENTATION))
5435 return fail_at(arg, COMMAND_LINE_ERROR);
5436 }
5437 CommandLineSwitchCase(arg, "old-license")
5438 {
5439 if (!freerdp_settings_set_bool(settings, FreeRDP_OldLicenseBehaviour, TRUE))
5440 return fail_at(arg, COMMAND_LINE_ERROR);
5441 }
5442 CommandLineSwitchCase(arg, "scale")
5443 {
5444 const int rc = parse_scale_options(settings, arg);
5445 if (rc != 0)
5446 return fail_at(arg, rc);
5447 }
5448 CommandLineSwitchCase(arg, "scale-desktop")
5449 {
5450 const int rc = parse_command_line_option_uint32(settings, arg,
5451 FreeRDP_DesktopScaleFactor, 100, 500);
5452 if (rc != 0)
5453 return fail_at(arg, rc);
5454 if (!set_monitor_override(settings, FREERDP_MONITOR_OVERRIDE_DESKTOP_SCALE))
5455 return fail_at(arg, COMMAND_LINE_ERROR);
5456 }
5457 CommandLineSwitchCase(arg, "scale-device")
5458 {
5459 const int rc = parse_scale_device_options(settings, arg);
5460 if (rc != 0)
5461 return fail_at(arg, rc);
5462 }
5463 CommandLineSwitchCase(arg, "action-script")
5464 {
5465 if (!freerdp_settings_set_string(settings, FreeRDP_ActionScript, arg->Value))
5466 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5467 }
5468 CommandLineSwitchCase(arg, RDP2TCP_DVC_CHANNEL_NAME)
5469 {
5470 if (!freerdp_settings_set_string(settings, FreeRDP_RDP2TCPArgs, arg->Value))
5471 return fail_at(arg, COMMAND_LINE_ERROR_MEMORY);
5472 }
5473 CommandLineSwitchCase(arg, "fipsmode")
5474 {
5475 if (!freerdp_settings_set_bool(settings, FreeRDP_FIPSMode, enable))
5476 return fail_at(arg, COMMAND_LINE_ERROR);
5477 }
5478 CommandLineSwitchCase(arg, "smartcard-logon")
5479 {
5480 const int rc = parse_smartcard_logon_options(settings, arg);
5481 if (rc != 0)
5482 return fail_at(arg, rc);
5483 }
5484 CommandLineSwitchCase(arg, "tune")
5485 {
5486 const int rc = parse_tune_options(settings, arg);
5487 if (rc != 0)
5488 return fail_at(arg, rc);
5489 }
5490 CommandLineSwitchDefault(arg)
5491 {
5492#if defined(WITH_FREERDP_DEPRECATED_COMMANDLINE)
5493 const int status = parse_deprecated_command_line(settings, arg);
5494 /* option handled, continue with next */
5495 if (status != -1)
5496 continue;
5497#endif
5498 if (handle_option)
5499 {
5500 const int rc = handle_option(arg, handle_userdata);
5501 if (rc != 0)
5502 return fail_at(arg, rc);
5503 }
5504 }
5505 CommandLineSwitchEnd(arg)
5506 } while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
5507 return 0;
5508}
5509
5510static void warn_credential_args(const COMMAND_LINE_ARGUMENT_A* args)
5511{
5512 WINPR_ASSERT(args);
5513 bool insecureArgFound = false;
5514 for (size_t x = 0; x < ARRAYSIZE(credential_args); x++)
5515 {
5516 const char* cred = credential_args[x];
5517 const COMMAND_LINE_ARGUMENT_A* arg = CommandLineFindArgumentA(args, cred);
5518 if (!arg)
5519 continue;
5520 if ((arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT) == 0)
5521 continue;
5522
5523 WLog_WARN(TAG, "Using /%s is insecure", arg->Name);
5524 insecureArgFound = true;
5525 }
5526
5527 if (insecureArgFound)
5528 {
5529 WLog_WARN(TAG, "Passing credentials or secrets via command line might expose these in the "
5530 "process list");
5531 WLog_WARN(TAG, "Consider using one of the following (more secure) alternatives:");
5532 WLog_WARN(TAG, " - /args-from: pipe in arguments from stdin, file or file descriptor");
5533 WLog_WARN(TAG, " - /from-stdin pass the credential via stdin");
5534 WLog_WARN(TAG, " - set environment variable FREERDP_ASKPASS to have a gui tool query for "
5535 "credentials");
5536 }
5537}
5538
5539static int freerdp_client_settings_parse_command_line_arguments_int(
5540 rdpSettings* settings, int argc, char* argv[], BOOL allowUnknown,
5541 COMMAND_LINE_ARGUMENT_A* largs, WINPR_ATTR_UNUSED size_t count,
5542 freerdp_command_line_handle_option_t handle_option, void* handle_userdata, bool isArgsFrom)
5543{
5544 char* user = NULL;
5545 int status = 0;
5546 BOOL ext = FALSE;
5547 BOOL assist = FALSE;
5548 DWORD flags = 0;
5549 BOOL promptForPassword = FALSE;
5550 BOOL compatibility = FALSE;
5551 const COMMAND_LINE_ARGUMENT_A* arg = NULL;
5552
5553 /* Command line detection fails if only a .rdp or .msrcIncident file
5554 * is supplied. Check this case first, only then try to detect
5555 * legacy command line syntax. */
5556 if (argc > 1)
5557 {
5558 ext = option_is_rdp_file(argv[1]);
5559 assist = option_is_incident_file(argv[1]);
5560 }
5561
5562 if (!ext && !assist)
5563 compatibility = freerdp_client_detect_command_line(argc, argv, &flags);
5564 else
5565 compatibility = freerdp_client_detect_command_line(argc - 1, &argv[1], &flags);
5566
5567 if (!freerdp_settings_set_string(settings, FreeRDP_ProxyHostname, NULL))
5568 return -1;
5569 if (!freerdp_settings_set_string(settings, FreeRDP_ProxyUsername, NULL))
5570 return -1;
5571 if (!freerdp_settings_set_string(settings, FreeRDP_ProxyPassword, NULL))
5572 return -1;
5573
5574 if (compatibility)
5575 {
5576 WLog_WARN(TAG, "Unsupported command line syntax!");
5577 WLog_WARN(TAG, "FreeRDP 1.0 style syntax was dropped with version 3!");
5578 return -1;
5579 }
5580
5581 if (allowUnknown)
5582 flags |= COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
5583
5584 if (ext)
5585 {
5586 if (freerdp_client_settings_parse_connection_file(settings, argv[1]))
5587 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
5588 }
5589
5590 if (assist)
5591 {
5592 if (freerdp_client_settings_parse_assistance_file(settings, argc, argv) < 0)
5593 return COMMAND_LINE_ERROR_UNEXPECTED_VALUE;
5594 }
5595
5596 CommandLineClearArgumentsA(largs);
5597 status = CommandLineParseArgumentsA(argc, argv, largs, flags, settings,
5598 freerdp_client_command_line_pre_filter,
5599 freerdp_client_command_line_post_filter);
5600
5601 if (status < 0)
5602 return status;
5603
5604 prepare_default_settings(settings, largs, ext);
5605 if (!isArgsFrom)
5606 warn_credential_args(largs);
5607
5608 CommandLineFindArgumentA(largs, "v");
5609 arg = largs;
5610 errno = 0;
5611
5612 /* Disable unicode input unless requested. */
5613 if (!freerdp_settings_set_bool(settings, FreeRDP_UnicodeInput, FALSE))
5614 return COMMAND_LINE_ERROR_MEMORY;
5615
5616 status = parse_command_line(settings, arg, handle_option, handle_userdata, &promptForPassword,
5617 &user);
5618
5619 if (user)
5620 {
5621 if (!freerdp_settings_get_string(settings, FreeRDP_Domain) && user)
5622 {
5623 if (!freerdp_settings_set_string(settings, FreeRDP_Username, NULL))
5624 return COMMAND_LINE_ERROR;
5625
5626 if (!freerdp_settings_set_string(settings, FreeRDP_Domain, NULL))
5627 return COMMAND_LINE_ERROR;
5628
5629 if (!freerdp_parse_username_settings(user, settings, FreeRDP_Username, FreeRDP_Domain))
5630 return COMMAND_LINE_ERROR;
5631 }
5632 else
5633 {
5634 if (!freerdp_settings_set_string(settings, FreeRDP_Username, user))
5635 return COMMAND_LINE_ERROR;
5636 }
5637 }
5638
5639 if (promptForPassword)
5640 {
5641 freerdp* instance = freerdp_settings_get_pointer_writable(settings, FreeRDP_instance);
5642 if (!freerdp_settings_get_string(settings, FreeRDP_Password))
5643 {
5644 char buffer[512 + 1] = { 0 };
5645
5646 if (!freerdp_passphrase_read(instance->context, "Password: ", buffer,
5647 ARRAYSIZE(buffer) - 1, 1))
5648 return COMMAND_LINE_ERROR;
5649 if (!freerdp_settings_set_string(settings, FreeRDP_Password, buffer))
5650 return COMMAND_LINE_ERROR;
5651 }
5652
5653 if (freerdp_settings_get_bool(settings, FreeRDP_GatewayEnabled) &&
5654 !freerdp_settings_get_bool(settings, FreeRDP_GatewayUseSameCredentials))
5655 {
5656 if (!freerdp_settings_get_string(settings, FreeRDP_GatewayPassword))
5657 {
5658 char buffer[512 + 1] = { 0 };
5659
5660 if (!freerdp_passphrase_read(instance->context, "Gateway Password: ", buffer,
5661 ARRAYSIZE(buffer) - 1, 1))
5662 return COMMAND_LINE_ERROR;
5663 if (!freerdp_settings_set_string(settings, FreeRDP_GatewayPassword, buffer))
5664 return COMMAND_LINE_ERROR;
5665 }
5666 }
5667 }
5668
5669 freerdp_performance_flags_make(settings);
5670
5671 if (freerdp_settings_get_bool(settings, FreeRDP_RemoteFxCodec) ||
5672 freerdp_settings_get_bool(settings, FreeRDP_NSCodec) ||
5673 freerdp_settings_get_bool(settings, FreeRDP_SupportGraphicsPipeline))
5674 {
5675 if (!freerdp_settings_set_bool(settings, FreeRDP_FastPathOutput, TRUE))
5676 return COMMAND_LINE_ERROR;
5677 if (!freerdp_settings_set_bool(settings, FreeRDP_FrameMarkerCommandEnabled, TRUE))
5678 return COMMAND_LINE_ERROR;
5679 }
5680
5681 arg = CommandLineFindArgumentA(largs, "port");
5682 if (arg->Flags & COMMAND_LINE_ARGUMENT_PRESENT)
5683 {
5684 const int rc =
5685 parse_command_line_option_uint32(settings, arg, FreeRDP_ServerPort, 0, UINT16_MAX);
5686 if (rc != 0)
5687 return fail_at(arg, rc);
5688 }
5689
5690 if (freerdp_settings_get_bool(settings, FreeRDP_VmConnectMode))
5691 {
5692 const COMMAND_LINE_ARGUMENT_A* nego = CommandLineFindArgumentA(largs, "nego");
5693 if (nego && (nego->Flags & COMMAND_LINE_ARGUMENT_PRESENT))
5694 return fail_at(arg, COMMAND_LINE_ERROR);
5695
5696 const UINT32 port = freerdp_settings_get_uint32(settings, FreeRDP_ServerPort);
5697 WLog_INFO(TAG, "/vmconnect uses custom port %" PRIu32, port);
5698 }
5699
5700 fill_credential_strings(largs);
5701
5702 return status;
5703}
5704
5705static void argv_free(int* pargc, char** pargv[])
5706{
5707 WINPR_ASSERT(pargc);
5708 WINPR_ASSERT(pargv);
5709 const int argc = *pargc;
5710 char** argv = *pargv;
5711 *pargc = 0;
5712 *pargv = NULL;
5713
5714 if (!argv)
5715 return;
5716 for (int x = 0; x < argc; x++)
5717 free(argv[x]);
5718 free((void*)argv);
5719}
5720
5721static BOOL argv_append(int* pargc, char** pargv[], char* what)
5722{
5723 WINPR_ASSERT(pargc);
5724 WINPR_ASSERT(pargv);
5725
5726 if (*pargc < 0)
5727 return FALSE;
5728
5729 if (!what)
5730 return FALSE;
5731
5732 int nargc = *pargc + 1;
5733 char** tmp = (char**)realloc((void*)*pargv, (size_t)nargc * sizeof(char*));
5734 if (!tmp)
5735 return FALSE;
5736
5737 tmp[*pargc] = what;
5738 *pargv = tmp;
5739 *pargc = nargc;
5740 return TRUE;
5741}
5742
5743static BOOL argv_append_dup(int* pargc, char** pargv[], const char* what)
5744{
5745 char* copy = NULL;
5746 if (what)
5747 copy = _strdup(what);
5748
5749 const BOOL rc = argv_append(pargc, pargv, copy);
5750 if (!rc)
5751 free(copy);
5752 return rc;
5753}
5754
5755static BOOL args_from_fp(FILE* fp, int* aargc, char** aargv[], const char* file, const char* cmd)
5756{
5757 BOOL success = FALSE;
5758
5759 WINPR_ASSERT(aargc);
5760 WINPR_ASSERT(aargv);
5761 WINPR_ASSERT(cmd);
5762
5763 if (!fp)
5764 {
5765 WLog_ERR(TAG, "Failed to read command line options from file '%s'", file);
5766 return FALSE;
5767 }
5768 if (!argv_append_dup(aargc, aargv, cmd))
5769 goto fail;
5770 while (!feof(fp))
5771 {
5772 char* line = NULL;
5773 size_t size = 0;
5774 INT64 rc = GetLine(&line, &size, fp);
5775 if ((rc < 0) || !line)
5776 {
5777 /* abort if GetLine failed due to reaching EOF */
5778 if (feof(fp))
5779 break;
5780 goto fail;
5781 }
5782
5783 while (rc > 0)
5784 {
5785 const char cur = (line[rc - 1]);
5786 if ((cur == '\n') || (cur == '\r'))
5787 {
5788 line[rc - 1] = '\0';
5789 rc--;
5790 }
5791 else
5792 break;
5793 }
5794 /* abort on empty lines */
5795 if (rc == 0)
5796 {
5797 free(line);
5798 break;
5799 }
5800 if (!argv_append(aargc, aargv, line))
5801 {
5802 free(line);
5803 goto fail;
5804 }
5805 }
5806
5807 success = TRUE;
5808fail:
5809 fclose(fp);
5810 if (!success)
5811 argv_free(aargc, aargv);
5812 return success;
5813}
5814
5815static BOOL args_from_env(const char* name, int* aargc, char** aargv[], const char* arg,
5816 const char* cmd)
5817{
5818 BOOL success = FALSE;
5819 char* env = NULL;
5820
5821 WINPR_ASSERT(aargc);
5822 WINPR_ASSERT(aargv);
5823 WINPR_ASSERT(cmd);
5824
5825 if (!name)
5826 {
5827 WLog_ERR(TAG, "%s - environment variable name empty", arg);
5828 goto cleanup;
5829 }
5830
5831 {
5832 const DWORD size = GetEnvironmentVariableX(name, env, 0);
5833 if (size == 0)
5834 {
5835 WLog_ERR(TAG, "%s - no environment variable '%s'", arg, name);
5836 goto cleanup;
5837 }
5838 env = calloc(size + 1, sizeof(char));
5839 if (!env)
5840 goto cleanup;
5841
5842 {
5843 const DWORD rc = GetEnvironmentVariableX(name, env, size);
5844 if (rc != size - 1)
5845 goto cleanup;
5846 if (rc == 0)
5847 {
5848 WLog_ERR(TAG, "%s - environment variable '%s' is empty", arg);
5849 goto cleanup;
5850 }
5851 }
5852 }
5853
5854 if (!argv_append_dup(aargc, aargv, cmd))
5855 goto cleanup;
5856
5857 {
5858 char* context = NULL;
5859 char* tok = strtok_s(env, "\n", &context);
5860 while (tok)
5861 {
5862 if (!argv_append_dup(aargc, aargv, tok))
5863 goto cleanup;
5864 tok = strtok_s(NULL, "\n", &context);
5865 }
5866 }
5867
5868 success = TRUE;
5869cleanup:
5870 free(env);
5871 if (!success)
5872 argv_free(aargc, aargv);
5873 return success;
5874}
5875
5876int freerdp_client_settings_parse_command_line_arguments(rdpSettings* settings, int oargc,
5877 char* oargv[], BOOL allowUnknown)
5878{
5879 return freerdp_client_settings_parse_command_line_arguments_ex(
5880 settings, oargc, oargv, allowUnknown, NULL, 0, NULL, NULL);
5881}
5882
5883int freerdp_client_settings_parse_command_line_arguments_ex(
5884 rdpSettings* settings, int oargc, char** oargv, BOOL allowUnknown,
5885 COMMAND_LINE_ARGUMENT_A* args, size_t count, freerdp_command_line_handle_option_t handle_option,
5886 void* handle_userdata)
5887{
5888 int argc = oargc;
5889 char** argv = oargv;
5890 int res = -1;
5891 int aargc = 0;
5892 char** aargv = NULL;
5893
5894 bool isArgsFrom = false;
5895 if ((argc == 2) && option_starts_with("/args-from:", argv[1]))
5896 {
5897 isArgsFrom = true;
5898 BOOL success = FALSE;
5899 const char* file = strchr(argv[1], ':') + 1;
5900 FILE* fp = stdin;
5901
5902 if (option_starts_with("fd:", file))
5903 {
5904 ULONGLONG result = 0;
5905 const char* val = strchr(file, ':') + 1;
5906 if (!value_to_uint(val, &result, 0, INT_MAX))
5907 return -1;
5908 fp = fdopen((int)result, "r");
5909 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
5910 }
5911 else if (strncmp(file, "env:", 4) == 0)
5912 {
5913 const char* name = strchr(file, ':') + 1;
5914 success = args_from_env(name, &aargc, &aargv, oargv[1], oargv[0]);
5915 }
5916 else if (strcmp(file, "stdin") != 0)
5917 {
5918 fp = winpr_fopen(file, "r");
5919 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
5920 }
5921 else
5922 success = args_from_fp(fp, &aargc, &aargv, file, oargv[0]);
5923
5924 if (!success)
5925 return -1;
5926 argc = aargc;
5927 argv = aargv;
5928 }
5929
5930 WINPR_ASSERT(count <= SSIZE_MAX);
5931 size_t lcount = 0;
5932 COMMAND_LINE_ARGUMENT_A* largs = create_merged_args(args, (SSIZE_T)count, &lcount);
5933 if (!largs)
5934 goto fail;
5935
5936 res = freerdp_client_settings_parse_command_line_arguments_int(
5937 settings, argc, argv, allowUnknown, largs, lcount, handle_option, handle_userdata,
5938 isArgsFrom);
5939fail:
5940 free(largs);
5941 argv_free(&aargc, &aargv);
5942 return res;
5943}
5944
5945static BOOL freerdp_client_load_static_channel_addin(rdpChannels* channels, rdpSettings* settings,
5946 const char* name, void* data)
5947{
5948 PVIRTUALCHANNELENTRY entry = NULL;
5949 PVIRTUALCHANNELENTRY pvce = freerdp_load_channel_addin_entry(
5950 name, NULL, NULL, FREERDP_ADDIN_CHANNEL_STATIC | FREERDP_ADDIN_CHANNEL_ENTRYEX);
5951 PVIRTUALCHANNELENTRYEX pvceex = WINPR_FUNC_PTR_CAST(pvce, PVIRTUALCHANNELENTRYEX);
5952
5953 if (!pvceex)
5954 entry = freerdp_load_channel_addin_entry(name, NULL, NULL, FREERDP_ADDIN_CHANNEL_STATIC);
5955
5956 if (pvceex)
5957 {
5958 if (freerdp_channels_client_load_ex(channels, settings, pvceex, data) == 0)
5959 {
5960 WLog_DBG(TAG, "loading channelEx %s", name);
5961 return TRUE;
5962 }
5963 }
5964 else if (entry)
5965 {
5966 if (freerdp_channels_client_load(channels, settings, entry, data) == 0)
5967 {
5968 WLog_DBG(TAG, "loading channel %s", name);
5969 return TRUE;
5970 }
5971 }
5972
5973 return FALSE;
5974}
5975
5976typedef struct
5977{
5978 FreeRDP_Settings_Keys_Bool settingId;
5979 const char* channelName;
5980 void* args;
5981} ChannelToLoad;
5982
5983BOOL freerdp_client_load_addins(rdpChannels* channels, rdpSettings* settings)
5984{
5985 ChannelToLoad dynChannels[] = {
5986#if defined(CHANNEL_AINPUT_CLIENT)
5987 { FreeRDP_BOOL_UNUSED, AINPUT_CHANNEL_NAME, NULL }, /* always loaded */
5988#endif
5989#ifdef CHANNEL_AUDIN_CLIENT
5990 { FreeRDP_AudioCapture, AUDIN_CHANNEL_NAME, NULL },
5991#endif
5992#ifdef CHANNEL_RDPSND_CLIENT
5993 { FreeRDP_AudioPlayback, RDPSND_CHANNEL_NAME, NULL },
5994#endif
5995#ifdef CHANNEL_RDPEI_CLIENT
5996 { FreeRDP_MultiTouchInput, RDPEI_CHANNEL_NAME, NULL },
5997#endif
5998#ifdef CHANNEL_RDPGFX_CLIENT
5999 { FreeRDP_SupportGraphicsPipeline, RDPGFX_CHANNEL_NAME, NULL },
6000#endif
6001#ifdef CHANNEL_ECHO_CLIENT
6002 { FreeRDP_SupportEchoChannel, ECHO_CHANNEL_NAME, NULL },
6003#endif
6004#ifdef CHANNEL_SSHAGENT_CLIENT
6005 { FreeRDP_SupportSSHAgentChannel, "sshagent", NULL },
6006#endif
6007#ifdef CHANNEL_DISP_CLIENT
6008 { FreeRDP_SupportDisplayControl, DISP_CHANNEL_NAME, NULL },
6009#endif
6010#ifdef CHANNEL_GEOMETRY_CLIENT
6011 { FreeRDP_SupportGeometryTracking, GEOMETRY_CHANNEL_NAME, NULL },
6012#endif
6013#ifdef CHANNEL_VIDEO_CLIENT
6014 { FreeRDP_SupportVideoOptimized, VIDEO_CHANNEL_NAME, NULL },
6015#endif
6016#ifdef CHANNEL_RDPEAR_CLIENT
6017 { FreeRDP_RemoteCredentialGuard, RDPEAR_CHANNEL_NAME, NULL },
6018#endif
6019 };
6020
6021 ChannelToLoad staticChannels[] = {
6022#if defined(CHANNEL_RDPSND_CLIENT)
6023 { FreeRDP_AudioPlayback, RDPSND_CHANNEL_NAME, NULL },
6024#endif
6025#if defined(CHANNEL_CLIPRDR_CLIENT)
6026 { FreeRDP_RedirectClipboard, CLIPRDR_SVC_CHANNEL_NAME, NULL },
6027#endif
6028#if defined(CHANNEL_ENCOMSP_CLIENT)
6029 { FreeRDP_EncomspVirtualChannel, ENCOMSP_SVC_CHANNEL_NAME, settings },
6030#endif
6031#if defined(CHANNEL_REMDESK_CLIENT)
6032 { FreeRDP_RemdeskVirtualChannel, REMDESK_SVC_CHANNEL_NAME, settings },
6033#endif
6034#if defined(CHANNEL_RAIL_CLIENT)
6035 { FreeRDP_RemoteApplicationMode, RAIL_SVC_CHANNEL_NAME, settings }
6036#endif
6037 };
6038
6042 for (size_t i = 0; i < ARRAYSIZE(dynChannels); i++)
6043 {
6044 if ((dynChannels[i].settingId == FreeRDP_BOOL_UNUSED) ||
6045 freerdp_settings_get_bool(settings, dynChannels[i].settingId))
6046 {
6047 const char* const p[] = { dynChannels[i].channelName };
6048
6049 if (!freerdp_client_add_dynamic_channel(settings, ARRAYSIZE(p), p))
6050 return FALSE;
6051 }
6052 }
6053
6057 if ((freerdp_static_channel_collection_find(settings, RDPSND_CHANNEL_NAME)) ||
6058 (freerdp_dynamic_channel_collection_find(settings, RDPSND_CHANNEL_NAME))
6059#if defined(CHANNEL_TSMF_CLIENT)
6060 || (freerdp_dynamic_channel_collection_find(settings, "tsmf"))
6061#endif
6062 )
6063 {
6064 if (!freerdp_settings_set_bool(settings, FreeRDP_DeviceRedirection, TRUE))
6065 return FALSE; /* rdpsnd requires rdpdr to be registered */
6066 if (!freerdp_settings_set_bool(settings, FreeRDP_AudioPlayback, TRUE))
6067 return FALSE; /* Both rdpsnd and tsmf require this flag to be set */
6068 }
6069
6070 if (freerdp_dynamic_channel_collection_find(settings, AUDIN_CHANNEL_NAME))
6071 {
6072 if (!freerdp_settings_set_bool(settings, FreeRDP_AudioCapture, TRUE))
6073 return FALSE;
6074 }
6075
6076 if (freerdp_settings_get_bool(settings, FreeRDP_NetworkAutoDetect) ||
6077 freerdp_settings_get_bool(settings, FreeRDP_SupportHeartbeatPdu) ||
6078 freerdp_settings_get_bool(settings, FreeRDP_SupportMultitransport))
6079 {
6080 if (!freerdp_settings_set_bool(settings, FreeRDP_DeviceRedirection, TRUE))
6081 return FALSE; /* these RDP8 features require rdpdr to be registered */
6082 }
6083
6084 const char* DrivesToRedirect = freerdp_settings_get_string(settings, FreeRDP_DrivesToRedirect);
6085
6086 if (DrivesToRedirect && (strlen(DrivesToRedirect) != 0))
6087 {
6088 /*
6089 * Drives to redirect:
6090 *
6091 * Very similar to DevicesToRedirect, but can contain a
6092 * comma-separated list of drive letters to redirect.
6093 */
6094 char* value = NULL;
6095 char* tok = NULL;
6096 char* context = NULL;
6097
6098 value = _strdup(DrivesToRedirect);
6099 if (!value)
6100 return FALSE;
6101
6102 tok = strtok_s(value, ";", &context);
6103 if (!tok)
6104 {
6105 WLog_ERR(TAG, "DrivesToRedirect contains invalid data: '%s'", DrivesToRedirect);
6106 free(value);
6107 return FALSE;
6108 }
6109
6110 while (tok)
6111 {
6112 /* Syntax: Comma separated list of the following entries:
6113 * '*' ... Redirect all drives, including hotplug
6114 * 'DynamicDrives' ... hotplug
6115 * '%' ... user home directory
6116 * <label>(<path>) ... One or more paths to redirect.
6117 * <path>(<label>) ... One or more paths to redirect.
6118 * <path> ... One or more paths to redirect.
6119 */
6120 /* TODO: Need to properly escape labels and paths */
6121 BOOL success = 0;
6122 const char* name = NULL;
6123 const char* drive = tok;
6124 char* subcontext = NULL;
6125 char* start = strtok_s(tok, "(", &subcontext);
6126 char* end = strtok_s(NULL, ")", &subcontext);
6127 if (start && end)
6128 name = end;
6129
6130 if (freerdp_path_valid(name, NULL) && freerdp_path_valid(drive, NULL))
6131 {
6132 success = freerdp_client_add_drive(settings, name, NULL);
6133 if (success)
6134 success = freerdp_client_add_drive(settings, drive, NULL);
6135 }
6136 else
6137 success = freerdp_client_add_drive(settings, drive, name);
6138
6139 if (!success)
6140 {
6141 free(value);
6142 return FALSE;
6143 }
6144
6145 tok = strtok_s(NULL, ";", &context);
6146 }
6147 free(value);
6148
6149 if (!freerdp_settings_set_bool(settings, FreeRDP_DeviceRedirection, TRUE))
6150 return FALSE;
6151 }
6152 else if (freerdp_settings_get_bool(settings, FreeRDP_RedirectDrives))
6153 {
6154 if (!freerdp_device_collection_find(settings, "drive"))
6155 {
6156 const char* const params[] = { "drive", "media", "*" };
6157
6158 if (!freerdp_client_add_device_channel(settings, ARRAYSIZE(params), params))
6159 return FALSE;
6160 }
6161 }
6162
6163 if (freerdp_settings_get_bool(settings, FreeRDP_RedirectDrives) ||
6164 freerdp_settings_get_bool(settings, FreeRDP_RedirectHomeDrive) ||
6165 freerdp_settings_get_bool(settings, FreeRDP_RedirectSerialPorts) ||
6166 freerdp_settings_get_bool(settings, FreeRDP_RedirectSmartCards) ||
6167 freerdp_settings_get_bool(settings, FreeRDP_RedirectPrinters))
6168 {
6169 if (!freerdp_settings_set_bool(settings, FreeRDP_DeviceRedirection, TRUE))
6170 return FALSE; /* All of these features require rdpdr */
6171 }
6172
6173 if (freerdp_settings_get_bool(settings, FreeRDP_RedirectHomeDrive))
6174 {
6175 if (!freerdp_device_collection_find(settings, "drive"))
6176 {
6177 const char* params[] = { "drive", "home", "%" };
6178
6179 if (!freerdp_client_add_device_channel(settings, ARRAYSIZE(params), params))
6180 return FALSE;
6181 }
6182 }
6183
6184 if (freerdp_settings_get_bool(settings, FreeRDP_DeviceRedirection))
6185 {
6186 if (!freerdp_client_load_static_channel_addin(channels, settings, RDPDR_SVC_CHANNEL_NAME,
6187 settings))
6188 return FALSE;
6189
6190 if (!freerdp_static_channel_collection_find(settings, RDPSND_CHANNEL_NAME) &&
6191 !freerdp_dynamic_channel_collection_find(settings, RDPSND_CHANNEL_NAME))
6192 {
6193 const char* const params[] = { RDPSND_CHANNEL_NAME, "sys:fake" };
6194
6195 if (!freerdp_client_add_static_channel(settings, ARRAYSIZE(params), params))
6196 return FALSE;
6197
6198 if (!freerdp_client_add_dynamic_channel(settings, ARRAYSIZE(params), params))
6199 return FALSE;
6200 }
6201 }
6202
6203 if (freerdp_settings_get_bool(settings, FreeRDP_RedirectSmartCards))
6204 {
6205 if (!freerdp_device_collection_find_type(settings, RDPDR_DTYP_SMARTCARD))
6206 {
6207 RDPDR_DEVICE* smartcard = freerdp_device_new(RDPDR_DTYP_SMARTCARD, 0, NULL);
6208
6209 if (!smartcard)
6210 return FALSE;
6211
6212 if (!freerdp_device_collection_add(settings, smartcard))
6213 {
6214 freerdp_device_free(smartcard);
6215 return FALSE;
6216 }
6217 }
6218 }
6219
6220 if (freerdp_settings_get_bool(settings, FreeRDP_RedirectPrinters))
6221 {
6222 if (!freerdp_device_collection_find_type(settings, RDPDR_DTYP_PRINT))
6223 {
6224 RDPDR_DEVICE* printer = freerdp_device_new(RDPDR_DTYP_PRINT, 0, NULL);
6225
6226 if (!printer)
6227 return FALSE;
6228
6229 if (!freerdp_device_collection_add(settings, printer))
6230 {
6231 freerdp_device_free(printer);
6232 return FALSE;
6233 }
6234 }
6235 }
6236
6237 if (freerdp_settings_get_bool(settings, FreeRDP_LyncRdpMode))
6238 {
6239 if (!freerdp_settings_set_bool(settings, FreeRDP_EncomspVirtualChannel, TRUE))
6240 return FALSE;
6241 if (!freerdp_settings_set_bool(settings, FreeRDP_RemdeskVirtualChannel, TRUE))
6242 return FALSE;
6243 if (!freerdp_settings_set_bool(settings, FreeRDP_CompressionEnabled, FALSE))
6244 return FALSE;
6245 }
6246
6247 if (freerdp_settings_get_bool(settings, FreeRDP_RemoteAssistanceMode))
6248 {
6249 if (!freerdp_settings_set_bool(settings, FreeRDP_EncomspVirtualChannel, TRUE))
6250 return FALSE;
6251 if (!freerdp_settings_set_bool(settings, FreeRDP_RemdeskVirtualChannel, TRUE))
6252 return FALSE;
6253 if (!freerdp_settings_set_bool(settings, FreeRDP_NlaSecurity, FALSE))
6254 return FALSE;
6255 }
6256
6257 /* step 3: schedule some static channels to load depending on the settings */
6258 for (size_t i = 0; i < ARRAYSIZE(staticChannels); i++)
6259 {
6260 if ((staticChannels[i].settingId == 0) ||
6261 freerdp_settings_get_bool(settings, staticChannels[i].settingId))
6262 {
6263 if (staticChannels[i].args)
6264 {
6265 if (!freerdp_client_load_static_channel_addin(
6266 channels, settings, staticChannels[i].channelName, staticChannels[i].args))
6267 return FALSE;
6268 }
6269 else
6270 {
6271 const char* const p[] = { staticChannels[i].channelName };
6272 if (!freerdp_client_add_static_channel(settings, ARRAYSIZE(p), p))
6273 return FALSE;
6274 }
6275 }
6276 }
6277
6278 {
6279 char* RDP2TCPArgs = freerdp_settings_get_string_writable(settings, FreeRDP_RDP2TCPArgs);
6280 if (RDP2TCPArgs)
6281 {
6282 const char* const p[] = { RDP2TCP_DVC_CHANNEL_NAME, RDP2TCPArgs };
6283 if (!freerdp_client_add_static_channel(settings, ARRAYSIZE(p), p))
6284 return FALSE;
6285 }
6286 }
6287
6288 /* step 4: do the static channels loading and init */
6289 for (UINT32 i = 0; i < freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelCount); i++)
6290 {
6291 ADDIN_ARGV* _args =
6292 freerdp_settings_get_pointer_array_writable(settings, FreeRDP_StaticChannelArray, i);
6293
6294 if (!freerdp_client_load_static_channel_addin(channels, settings, _args->argv[0], _args))
6295 return FALSE;
6296 }
6297
6298 if (freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelCount) > 0)
6299 {
6300 if (!freerdp_settings_set_bool(settings, FreeRDP_SupportDynamicChannels, TRUE))
6301 return FALSE;
6302 }
6303
6304 if (freerdp_settings_get_bool(settings, FreeRDP_SupportDynamicChannels))
6305 {
6306 if (!freerdp_client_load_static_channel_addin(channels, settings, DRDYNVC_SVC_CHANNEL_NAME,
6307 settings))
6308 return FALSE;
6309 }
6310
6311 return TRUE;
6312}
6313
6314void freerdp_client_warn_unmaintained(int argc, char* argv[])
6315{
6316 const char* app = (argc > 0) ? argv[0] : "INVALID_ARGV";
6317 const DWORD log_level = WLOG_WARN;
6318 wLog* log = WLog_Get(TAG);
6319 WINPR_ASSERT(log);
6320
6321 if (!WLog_IsLevelActive(log, log_level))
6322 return;
6323
6324 WLog_Print_unchecked(log, log_level, "[unmaintained] %s client is currently unmaintained!",
6325 app);
6326 WLog_Print_unchecked(
6327 log, log_level,
6328 " If problems occur please check https://github.com/FreeRDP/FreeRDP/issues for "
6329 "known issues!");
6330 WLog_Print_unchecked(
6331 log, log_level,
6332 "Be prepared to fix issues yourself though as nobody is actively working on this.");
6333 WLog_Print_unchecked(
6334 log, log_level,
6335 " Developers hang out in https://matrix.to/#/#FreeRDP:matrix.org?via=matrix.org "
6336 "- don't hesitate to ask some questions. (replies might take some time depending "
6337 "on your timezone) - if you intend using this component write us a message");
6338}
6339
6340void freerdp_client_warn_experimental(int argc, char* argv[])
6341{
6342 const char* app = (argc > 0) ? argv[0] : "INVALID_ARGV";
6343 const DWORD log_level = WLOG_WARN;
6344 wLog* log = WLog_Get(TAG);
6345 WINPR_ASSERT(log);
6346
6347 if (!WLog_IsLevelActive(log, log_level))
6348 return;
6349
6350 WLog_Print_unchecked(log, log_level, "[experimental] %s client is currently experimental!",
6351 app);
6352 WLog_Print_unchecked(
6353 log, log_level,
6354 " If problems occur please check https://github.com/FreeRDP/FreeRDP/issues for "
6355 "known issues or create a new one!");
6356 WLog_Print_unchecked(
6357 log, log_level,
6358 " Developers hang out in https://matrix.to/#/#FreeRDP:matrix.org?via=matrix.org "
6359 "- don't hesitate to ask some questions. (replies might take some time depending "
6360 "on your timezone)");
6361}
6362
6363void freerdp_client_warn_deprecated(int argc, char* argv[])
6364{
6365 const char* app = (argc > 0) ? argv[0] : "INVALID_ARGV";
6366 const DWORD log_level = WLOG_WARN;
6367 wLog* log = WLog_Get(TAG);
6368 WINPR_ASSERT(log);
6369
6370 if (!WLog_IsLevelActive(log, log_level))
6371 return;
6372
6373 WLog_Print_unchecked(log, log_level, "[deprecated] %s client has been deprecated", app);
6374 WLog_Print_unchecked(log, log_level, "As replacement there is a SDL3 based client available.");
6375 WLog_Print_unchecked(
6376 log, log_level,
6377 "If you are interested in keeping %s alive get in touch with the developers", app);
6378 WLog_Print_unchecked(
6379 log, log_level,
6380 "The project is hosted at https://github.com/freerdp/freerdp and "
6381 " developers hang out in https://matrix.to/#/#FreeRDP:matrix.org?via=matrix.org "
6382 "- don't hesitate to ask some questions. (replies might take some time depending "
6383 "on your timezone)");
6384}
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.