Coverage Report

Created: 2026-05-14 07:08

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