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