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