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