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