Coverage Report

Created: 2026-04-12 07:03

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/FreeRDP/libfreerdp/common/settings.c
Line
Count
Source
1
/**
2
 * FreeRDP: A Remote Desktop Protocol Implementation
3
 * Settings Management
4
 *
5
 * Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
6
 * Copyright 2016 Armin Novak <armin.novak@gmail.com>
7
 * Copyright 2023 Armin Novak <anovak@thincast.com>
8
 * Copyright 2023 Thincast Technologies GmbH
9
 *
10
 * Licensed under the Apache License, Version 2.0 (the "License");
11
 * you may not use this file except in compliance with the License.
12
 * You may obtain a copy of the License at
13
 *
14
 *     http://www.apache.org/licenses/LICENSE-2.0
15
 *
16
 * Unless required by applicable law or agreed to in writing, software
17
 * distributed under the License is distributed on an "AS IS" BASIS,
18
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
19
 * See the License for the specific language governing permissions and
20
 * limitations under the License.
21
 */
22
23
#include <freerdp/config.h>
24
25
#include <stdio.h>
26
#include <stdlib.h>
27
#include <string.h>
28
#include <errno.h>
29
#include <math.h>
30
31
#include <winpr/crt.h>
32
#include <winpr/assert.h>
33
#include <winpr/cast.h>
34
35
#include "../core/settings.h"
36
#include "../core/capabilities.h"
37
38
#include <freerdp/crypto/certificate.h>
39
#include <freerdp/settings.h>
40
#include <freerdp/freerdp.h>
41
#include <freerdp/log.h>
42
43
0
#define TAG FREERDP_TAG("common")
44
45
BOOL freerdp_addin_argv_add_argument_ex(ADDIN_ARGV* args, const char* argument, size_t len)
46
0
{
47
0
  if (!args || !argument)
48
0
    return FALSE;
49
50
0
  if (len == 0)
51
0
    len = strlen(argument);
52
53
0
  char** new_argv = (char**)realloc(
54
0
      (void*)args->argv, sizeof(char*) * (WINPR_ASSERTING_INT_CAST(uint32_t, args->argc) + 1));
55
56
0
  if (!new_argv)
57
0
    return FALSE;
58
59
0
  args->argv = new_argv;
60
61
0
  char* str = calloc(len + 1, sizeof(char));
62
0
  if (!str)
63
0
    return FALSE;
64
0
  memcpy(str, argument, len);
65
0
  args->argv[args->argc++] = str;
66
0
  return TRUE;
67
0
}
68
69
BOOL freerdp_addin_argv_add_argument(ADDIN_ARGV* args, const char* argument)
70
0
{
71
0
  return freerdp_addin_argv_add_argument_ex(args, argument, 0);
72
0
}
73
74
BOOL freerdp_addin_argv_del_argument(ADDIN_ARGV* args, const char* argument)
75
0
{
76
0
  if (!args || !argument)
77
0
    return FALSE;
78
0
  for (int x = 0; x < args->argc; x++)
79
0
  {
80
0
    char* arg = args->argv[x];
81
0
    if (strcmp(argument, arg) == 0)
82
0
    {
83
0
      free(arg);
84
0
      const BOOL res =
85
0
          memmove_s((void*)&args->argv[x],
86
0
                    (WINPR_ASSERTING_INT_CAST(uint32_t, args->argc - x)) * sizeof(char*),
87
0
                    (void*)&args->argv[x + 1],
88
0
                    (WINPR_ASSERTING_INT_CAST(uint32_t, args->argc - x - 1)) *
89
0
                        sizeof(char*)) >= 0;
90
0
      args->argv[args->argc - 1] = nullptr;
91
0
      args->argc--;
92
0
      return res;
93
0
    }
94
0
  }
95
0
  return FALSE;
96
0
}
97
98
int freerdp_addin_set_argument(ADDIN_ARGV* args, const char* argument)
99
0
{
100
0
  if (!args || !argument)
101
0
    return -2;
102
103
0
  for (int i = 0; i < args->argc; i++)
104
0
  {
105
0
    if (strcmp(args->argv[i], argument) == 0)
106
0
    {
107
0
      return 1;
108
0
    }
109
0
  }
110
111
0
  if (!freerdp_addin_argv_add_argument(args, argument))
112
0
    return -1;
113
0
  return 0;
114
0
}
115
116
int freerdp_addin_replace_argument(ADDIN_ARGV* args, const char* previous, const char* argument)
117
0
{
118
0
  if (!args || !previous || !argument)
119
0
    return -2;
120
121
0
  for (int i = 0; i < args->argc; i++)
122
0
  {
123
0
    if (strcmp(args->argv[i], previous) == 0)
124
0
    {
125
0
      free(args->argv[i]);
126
127
0
      if (!(args->argv[i] = _strdup(argument)))
128
0
        return -1;
129
130
0
      return 1;
131
0
    }
132
0
  }
133
134
0
  if (!freerdp_addin_argv_add_argument(args, argument))
135
0
    return -1;
136
0
  return 0;
137
0
}
138
139
int freerdp_addin_set_argument_value(ADDIN_ARGV* args, const char* option, const char* value)
140
0
{
141
0
  BOOL rc = 0;
142
0
  char* p = nullptr;
143
0
  char* str = nullptr;
144
0
  size_t length = 0;
145
0
  if (!args || !option || !value)
146
0
    return -2;
147
0
  length = strlen(option) + strlen(value) + 1;
148
0
  str = (char*)calloc(length + 1, sizeof(char));
149
150
0
  if (!str)
151
0
    return -1;
152
153
0
  (void)sprintf_s(str, length + 1, "%s:%s", option, value);
154
155
0
  for (int i = 0; i < args->argc; i++)
156
0
  {
157
0
    p = strchr(args->argv[i], ':');
158
159
0
    if (p)
160
0
    {
161
0
      if (strncmp(args->argv[i], option,
162
0
                  WINPR_ASSERTING_INT_CAST(size_t, p - args->argv[i])) == 0)
163
0
      {
164
0
        free(args->argv[i]);
165
0
        args->argv[i] = str;
166
0
        return 1;
167
0
      }
168
0
    }
169
0
  }
170
171
0
  rc = freerdp_addin_argv_add_argument(args, str);
172
0
  free(str);
173
0
  if (!rc)
174
0
    return -1;
175
0
  return 0;
176
0
}
177
178
int freerdp_addin_replace_argument_value(ADDIN_ARGV* args, const char* previous, const char* option,
179
                                         const char* value)
180
0
{
181
0
  BOOL rc = 0;
182
0
  char* str = nullptr;
183
0
  size_t length = 0;
184
0
  if (!args || !previous || !option || !value)
185
0
    return -2;
186
0
  length = strlen(option) + strlen(value) + 1;
187
0
  str = (char*)calloc(length + 1, sizeof(char));
188
189
0
  if (!str)
190
0
    return -1;
191
192
0
  (void)sprintf_s(str, length + 1, "%s:%s", option, value);
193
194
0
  for (int i = 0; i < args->argc; i++)
195
0
  {
196
0
    if (strcmp(args->argv[i], previous) == 0)
197
0
    {
198
0
      free(args->argv[i]);
199
0
      args->argv[i] = str;
200
0
      return 1;
201
0
    }
202
0
  }
203
204
0
  rc = freerdp_addin_argv_add_argument(args, str);
205
0
  free(str);
206
0
  if (!rc)
207
0
    return -1;
208
0
  return 0;
209
0
}
210
211
BOOL freerdp_device_collection_add(rdpSettings* settings, RDPDR_DEVICE* device)
212
0
{
213
0
  UINT32 count = 0;
214
0
  UINT32 old = 0;
215
0
  WINPR_ASSERT(settings);
216
0
  WINPR_ASSERT(device);
217
218
0
  count = freerdp_settings_get_uint32(settings, FreeRDP_DeviceCount) + 1;
219
0
  old = freerdp_settings_get_uint32(settings, FreeRDP_DeviceArraySize);
220
0
  if (old < count)
221
0
  {
222
0
    const size_t new_size = (old + 32);
223
0
    RDPDR_DEVICE** new_array =
224
0
        (RDPDR_DEVICE**)realloc((void*)settings->DeviceArray, new_size * sizeof(RDPDR_DEVICE*));
225
226
0
    if (!new_array)
227
0
      return FALSE;
228
229
0
    settings->DeviceArray = new_array;
230
0
    for (size_t x = old; x < new_size; x++)
231
0
      settings->DeviceArray[x] = nullptr;
232
233
0
    if (!freerdp_settings_set_uint32(settings, FreeRDP_DeviceArraySize,
234
0
                                     WINPR_ASSERTING_INT_CAST(uint32_t, new_size)))
235
0
      return FALSE;
236
0
  }
237
238
0
  settings->DeviceArray[settings->DeviceCount++] = device;
239
0
  return TRUE;
240
0
}
241
242
BOOL freerdp_device_collection_del(rdpSettings* settings, const RDPDR_DEVICE* device)
243
0
{
244
0
  WINPR_ASSERT(settings);
245
246
0
  if (!device)
247
0
    return FALSE;
248
249
0
  const UINT32 count = settings->DeviceCount;
250
0
  for (size_t x = 0; x < count; x++)
251
0
  {
252
0
    const RDPDR_DEVICE* cur = settings->DeviceArray[x];
253
0
    if (cur == device)
254
0
    {
255
0
      for (size_t y = x + 1; y < count; y++)
256
0
      {
257
0
        RDPDR_DEVICE* next = settings->DeviceArray[y];
258
0
        settings->DeviceArray[y - 1] = next;
259
0
      }
260
0
      settings->DeviceArray[count - 1] = nullptr;
261
0
      settings->DeviceCount--;
262
0
      return TRUE;
263
0
    }
264
0
  }
265
266
0
  return FALSE;
267
0
}
268
269
RDPDR_DEVICE* freerdp_device_collection_find(rdpSettings* settings, const char* name)
270
0
{
271
0
  RDPDR_DEVICE* device = nullptr;
272
273
0
  WINPR_ASSERT(settings);
274
0
  WINPR_ASSERT(name);
275
0
  for (UINT32 index = 0; index < settings->DeviceCount; index++)
276
0
  {
277
0
    device = settings->DeviceArray[index];
278
279
0
    if (!device->Name)
280
0
      continue;
281
282
0
    if (strcmp(device->Name, name) == 0)
283
0
      return device;
284
0
  }
285
286
0
  return nullptr;
287
0
}
288
289
RDPDR_DEVICE* freerdp_device_collection_find_type(rdpSettings* settings, UINT32 type)
290
0
{
291
0
  RDPDR_DEVICE* device = nullptr;
292
0
  WINPR_ASSERT(settings);
293
294
0
  for (UINT32 index = 0; index < settings->DeviceCount; index++)
295
0
  {
296
0
    device = settings->DeviceArray[index];
297
298
0
    if (device->Type == type)
299
0
      return device;
300
0
  }
301
302
0
  return nullptr;
303
0
}
304
305
RDPDR_DEVICE* freerdp_device_new(UINT32 Type, size_t count, const char* const args[])
306
0
{
307
0
  size_t size = 0;
308
0
  union
309
0
  {
310
0
    RDPDR_DEVICE* base;
311
0
    RDPDR_DRIVE* drive;
312
0
    RDPDR_SERIAL* serial;
313
0
    RDPDR_PRINTER* printer;
314
0
    RDPDR_PARALLEL* parallel;
315
0
    RDPDR_SMARTCARD* smartcard;
316
0
  } device;
317
318
0
  device.base = nullptr;
319
0
  WINPR_ASSERT(args || (count == 0));
320
321
0
  switch (Type)
322
0
  {
323
0
    case RDPDR_DTYP_PRINT:
324
0
      size = sizeof(RDPDR_PRINTER);
325
0
      break;
326
0
    case RDPDR_DTYP_SERIAL:
327
0
      size = sizeof(RDPDR_SERIAL);
328
0
      break;
329
0
    case RDPDR_DTYP_PARALLEL:
330
0
      size = sizeof(RDPDR_PARALLEL);
331
0
      break;
332
0
    case RDPDR_DTYP_SMARTCARD:
333
0
      size = sizeof(RDPDR_SMARTCARD);
334
0
      break;
335
0
    case RDPDR_DTYP_FILESYSTEM:
336
0
      size = sizeof(RDPDR_DRIVE);
337
0
      break;
338
0
    default:
339
0
      goto fail;
340
0
  }
341
342
0
  device.base = calloc(1, size);
343
0
  if (!device.base)
344
0
    goto fail;
345
0
  device.base->Id = 0;
346
0
  device.base->Type = Type;
347
348
0
  if (count > 0)
349
0
  {
350
0
    device.base->Name = _strdup(args[0]);
351
0
    if (!device.base->Name)
352
0
      goto fail;
353
354
0
    switch (Type)
355
0
    {
356
0
      case RDPDR_DTYP_PRINT:
357
0
        if (count > 1)
358
0
        {
359
0
          device.printer->DriverName = _strdup(args[1]);
360
0
          if (!device.printer->DriverName)
361
0
            goto fail;
362
0
        }
363
364
0
        if (count > 2)
365
0
        {
366
0
          device.printer->IsDefault = _stricmp(args[2], "default") == 0;
367
0
        }
368
0
        break;
369
0
      case RDPDR_DTYP_SERIAL:
370
0
        if (count > 1)
371
0
        {
372
0
          device.serial->Path = _strdup(args[1]);
373
0
          if (!device.serial->Path)
374
0
            goto fail;
375
0
        }
376
377
0
        if (count > 2)
378
0
        {
379
0
          device.serial->Driver = _strdup(args[2]);
380
0
          if (!device.serial->Driver)
381
0
            goto fail;
382
0
        }
383
384
0
        if (count > 3)
385
0
        {
386
0
          device.serial->Permissive = _strdup(args[3]);
387
0
          if (!device.serial->Permissive)
388
0
            goto fail;
389
0
        }
390
0
        break;
391
0
      case RDPDR_DTYP_PARALLEL:
392
0
        if (count > 1)
393
0
        {
394
0
          device.parallel->Path = _strdup(args[1]);
395
0
          if (!device.serial->Path)
396
0
            goto fail;
397
0
        }
398
0
        break;
399
0
      case RDPDR_DTYP_SMARTCARD:
400
0
        break;
401
0
      case RDPDR_DTYP_FILESYSTEM:
402
0
        if (count > 1)
403
0
        {
404
0
          device.drive->Path = _strdup(args[1]);
405
0
          if (!device.drive->Path)
406
0
            goto fail;
407
0
        }
408
0
        if (count > 2)
409
0
          device.drive->automount = (args[2] == nullptr);
410
0
        break;
411
0
      default:
412
0
        goto fail;
413
0
    }
414
0
  }
415
0
  return device.base;
416
417
0
fail:
418
0
  freerdp_device_free(device.base);
419
0
  return nullptr;
420
0
}
421
422
void freerdp_device_free(RDPDR_DEVICE* device)
423
0
{
424
0
  if (!device)
425
0
    return;
426
427
0
  union
428
0
  {
429
0
    RDPDR_DEVICE* dev;
430
0
    RDPDR_DRIVE* drive;
431
0
    RDPDR_SERIAL* serial;
432
0
    RDPDR_PRINTER* printer;
433
0
    RDPDR_PARALLEL* parallel;
434
0
    RDPDR_SMARTCARD* smartcard;
435
0
  } cnv;
436
437
0
  cnv.dev = device;
438
439
0
  switch (device->Type)
440
0
  {
441
0
    case RDPDR_DTYP_PRINT:
442
0
      free(cnv.printer->DriverName);
443
0
      break;
444
0
    case RDPDR_DTYP_SERIAL:
445
0
      free(cnv.serial->Path);
446
0
      free(cnv.serial->Driver);
447
0
      free(cnv.serial->Permissive);
448
0
      break;
449
0
    case RDPDR_DTYP_PARALLEL:
450
0
      free(cnv.parallel->Path);
451
0
      break;
452
0
    case RDPDR_DTYP_SMARTCARD:
453
0
      break;
454
0
    case RDPDR_DTYP_FILESYSTEM:
455
0
      free(cnv.drive->Path);
456
0
      break;
457
0
    default:
458
0
      break;
459
0
  }
460
0
  free(cnv.dev->Name);
461
0
  free(cnv.dev);
462
0
}
463
464
RDPDR_DEVICE* freerdp_device_clone(const RDPDR_DEVICE* device)
465
0
{
466
0
  union
467
0
  {
468
0
    const RDPDR_DEVICE* dev;
469
0
    const RDPDR_DRIVE* drive;
470
0
    const RDPDR_SERIAL* serial;
471
0
    const RDPDR_PRINTER* printer;
472
0
    const RDPDR_PARALLEL* parallel;
473
0
    const RDPDR_SMARTCARD* smartcard;
474
0
  } src;
475
476
0
  union
477
0
  {
478
0
    RDPDR_DEVICE* dev;
479
0
    RDPDR_DRIVE* drive;
480
0
    RDPDR_SERIAL* serial;
481
0
    RDPDR_PRINTER* printer;
482
0
    RDPDR_PARALLEL* parallel;
483
0
    RDPDR_SMARTCARD* smartcard;
484
0
  } copy;
485
0
  size_t count = 0;
486
0
  const char* args[4] = WINPR_C_ARRAY_INIT;
487
488
0
  copy.dev = nullptr;
489
0
  src.dev = device;
490
491
0
  if (!device)
492
0
    return nullptr;
493
494
0
  if (device->Name)
495
0
  {
496
0
    args[count++] = device->Name;
497
0
  }
498
499
0
  switch (device->Type)
500
0
  {
501
0
    case RDPDR_DTYP_FILESYSTEM:
502
0
      if (src.drive->Path)
503
0
      {
504
0
        args[count++] = src.drive->Path;
505
0
        args[count++] = src.drive->automount ? nullptr : src.drive->Path;
506
0
      }
507
0
      break;
508
509
0
    case RDPDR_DTYP_PRINT:
510
0
      if (src.printer->DriverName)
511
0
        args[count++] = src.printer->DriverName;
512
0
      break;
513
514
0
    case RDPDR_DTYP_SMARTCARD:
515
0
      break;
516
517
0
    case RDPDR_DTYP_SERIAL:
518
0
      if (src.serial->Path)
519
0
        args[count++] = src.serial->Path;
520
521
0
      if (src.serial->Driver)
522
0
        args[count++] = src.serial->Driver;
523
524
0
      if (src.serial->Permissive)
525
0
        args[count++] = src.serial->Permissive;
526
0
      break;
527
528
0
    case RDPDR_DTYP_PARALLEL:
529
0
      if (src.parallel->Path)
530
0
        args[count++] = src.parallel->Path;
531
0
      break;
532
0
    default:
533
0
      WLog_ERR(TAG, "unknown device type %" PRIu32 "", device->Type);
534
0
      break;
535
0
  }
536
537
0
  copy.dev = freerdp_device_new(device->Type, count, args);
538
0
  if (!copy.dev)
539
0
    return nullptr;
540
541
0
  copy.dev->Id = device->Id;
542
543
0
  return copy.dev;
544
0
}
545
546
void freerdp_device_collection_free(rdpSettings* settings)
547
230k
{
548
230k
  WINPR_ASSERT(settings);
549
550
230k
  if (settings->DeviceArray)
551
0
  {
552
0
    for (UINT32 index = 0; index < settings->DeviceArraySize; index++)
553
0
    {
554
0
      const BOOL rc =
555
0
          freerdp_settings_set_pointer_array(settings, FreeRDP_DeviceArray, index, nullptr);
556
0
      if (!rc)
557
0
        WLog_WARN(TAG,
558
0
                  "freerdp_settings_set_pointer_array(settings, FreeRDP_DeviceArray, "
559
0
                  "index=%" PRIu32 ", nullptr) failed",
560
0
                  index);
561
0
    }
562
0
  }
563
564
230k
  free((void*)settings->DeviceArray);
565
566
230k
  if (!freerdp_settings_set_pointer(settings, FreeRDP_DeviceArray, nullptr))
567
0
    WLog_WARN(TAG,
568
230k
              "freerdp_settings_set_pointer(settings, FreeRDP_DeviceArray, nullptr) failed");
569
230k
  if (!freerdp_settings_set_uint32(settings, FreeRDP_DeviceArraySize, 0))
570
0
    WLog_WARN(TAG, "freerdp_settings_set_uint32(settings, FreeRDP_DeviceArraySize, 0) failed");
571
230k
  if (!freerdp_settings_set_uint32(settings, FreeRDP_DeviceCount, 0))
572
0
    WLog_WARN(TAG, "freerdp_settings_set_uint32(settings, FreeRDP_DeviceCount, 0) failed");
573
230k
}
574
575
BOOL freerdp_static_channel_collection_del(rdpSettings* settings, const char* name)
576
0
{
577
0
  const UINT32 count = freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelCount);
578
0
  if (!settings || !settings->StaticChannelArray)
579
0
    return FALSE;
580
581
0
  for (UINT32 x = 0; x < count; x++)
582
0
  {
583
0
    ADDIN_ARGV* cur = settings->StaticChannelArray[x];
584
0
    if (cur && (cur->argc > 0))
585
0
    {
586
0
      if (strcmp(name, cur->argv[0]) == 0)
587
0
      {
588
0
        const BOOL success = memmove_s((void*)&settings->StaticChannelArray[x],
589
0
                                       (count - x) * sizeof(ADDIN_ARGV*),
590
0
                                       (void*)&settings->StaticChannelArray[x + 1],
591
0
                                       (count - x - 1) * sizeof(ADDIN_ARGV*)) >= 0;
592
0
        for (size_t y = count - 1; y < settings->StaticChannelArraySize; y++)
593
0
          settings->StaticChannelArray[y] = nullptr;
594
595
0
        freerdp_addin_argv_free(cur);
596
0
        if (!freerdp_settings_set_uint32(settings, FreeRDP_StaticChannelCount, count - 1))
597
0
          return FALSE;
598
0
        return success;
599
0
      }
600
0
    }
601
0
  }
602
0
  {
603
0
    for (size_t x = count; x < settings->StaticChannelArraySize; x++)
604
0
      settings->StaticChannelArray[x] = nullptr;
605
0
  }
606
0
  return FALSE;
607
0
}
608
609
BOOL freerdp_static_channel_collection_add(rdpSettings* settings, ADDIN_ARGV* channel)
610
0
{
611
0
  UINT32 count = 0;
612
613
0
  WINPR_ASSERT(settings);
614
0
  WINPR_ASSERT(channel);
615
616
0
  count = freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelCount) + 1;
617
0
  if (freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelArraySize) < count)
618
0
  {
619
0
    const UINT32 oldSize =
620
0
        freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelArraySize);
621
0
    const size_t new_size = oldSize + 32ul;
622
0
    ADDIN_ARGV** new_array = (ADDIN_ARGV**)realloc((void*)settings->StaticChannelArray,
623
0
                                                   new_size * sizeof(ADDIN_ARGV*));
624
625
0
    if (!new_array)
626
0
      return FALSE;
627
628
0
    settings->StaticChannelArray = new_array;
629
0
    {
630
0
      for (size_t x = oldSize; x < new_size; x++)
631
0
        settings->StaticChannelArray[x] = nullptr;
632
0
    }
633
0
    if (!freerdp_settings_set_uint32(settings, FreeRDP_StaticChannelArraySize,
634
0
                                     WINPR_ASSERTING_INT_CAST(uint32_t, new_size)))
635
0
      return FALSE;
636
0
  }
637
638
0
  count = freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelCount);
639
640
0
  ADDIN_ARGV** cur = &settings->StaticChannelArray[count++];
641
0
  freerdp_addin_argv_free(*cur);
642
0
  *cur = channel;
643
0
  return freerdp_settings_set_uint32(settings, FreeRDP_StaticChannelCount, count);
644
0
}
645
646
ADDIN_ARGV* freerdp_static_channel_collection_find(rdpSettings* settings, const char* name)
647
0
{
648
0
  ADDIN_ARGV* channel = nullptr;
649
650
0
  WINPR_ASSERT(settings);
651
0
  WINPR_ASSERT(name);
652
653
0
  for (UINT32 index = 0;
654
0
       index < freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelCount); index++)
655
0
  {
656
0
    channel = settings->StaticChannelArray[index];
657
658
0
    if (strcmp(channel->argv[0], name) == 0)
659
0
      return channel;
660
0
  }
661
662
0
  return nullptr;
663
0
}
664
665
void freerdp_static_channel_collection_free(rdpSettings* settings)
666
230k
{
667
230k
  if (!settings)
668
0
    return;
669
670
230k
  if (settings->StaticChannelArray)
671
0
  {
672
0
    for (UINT32 i = 0;
673
0
         i < freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelArraySize); i++)
674
0
      freerdp_addin_argv_free(settings->StaticChannelArray[i]);
675
0
  }
676
677
230k
  free((void*)settings->StaticChannelArray);
678
230k
  if (!freerdp_settings_set_uint32(settings, FreeRDP_StaticChannelArraySize, 0))
679
0
    WLog_WARN(
680
230k
        TAG, "freerdp_settings_set_uint32(settings, FreeRDP_StaticChannelArraySize, 0) failed");
681
230k
  settings->StaticChannelArray = nullptr;
682
230k
  if (!freerdp_settings_set_uint32(settings, FreeRDP_StaticChannelCount, 0))
683
0
    WLog_WARN(TAG,
684
230k
              "freerdp_settings_set_uint32(settings, FreeRDP_StaticChannelCount, 0) failed");
685
230k
}
686
687
BOOL freerdp_dynamic_channel_collection_del(rdpSettings* settings, const char* name)
688
0
{
689
0
  const UINT32 count = freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelCount);
690
0
  if (!settings || !settings->DynamicChannelArray)
691
0
    return FALSE;
692
693
0
  for (UINT32 x = 0; x < count; x++)
694
0
  {
695
0
    ADDIN_ARGV* cur = settings->DynamicChannelArray[x];
696
0
    if (cur && (cur->argc > 0))
697
0
    {
698
0
      if (strcmp(name, cur->argv[0]) == 0)
699
0
      {
700
0
        const BOOL success = memmove_s((void*)&settings->DynamicChannelArray[x],
701
0
                                       (count - x) * sizeof(ADDIN_ARGV*),
702
0
                                       (void*)&settings->DynamicChannelArray[x + 1],
703
0
                                       (count - x - 1) * sizeof(ADDIN_ARGV*)) >= 0;
704
0
        for (size_t y = count - 1; y < settings->DynamicChannelArraySize; y++)
705
0
          settings->DynamicChannelArray[y] = nullptr;
706
707
0
        freerdp_addin_argv_free(cur);
708
0
        if (!freerdp_settings_set_uint32(settings, FreeRDP_DynamicChannelCount, count - 1))
709
0
          return FALSE;
710
0
        return success;
711
0
      }
712
0
    }
713
0
  }
714
715
0
  return FALSE;
716
0
}
717
718
BOOL freerdp_dynamic_channel_collection_add(rdpSettings* settings, ADDIN_ARGV* channel)
719
0
{
720
0
  UINT32 count = 0;
721
0
  UINT32 oldSize = 0;
722
723
0
  WINPR_ASSERT(settings);
724
0
  WINPR_ASSERT(channel);
725
726
0
  count = freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelCount) + 1;
727
0
  oldSize = freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelArraySize);
728
0
  if (oldSize < count)
729
0
  {
730
731
0
    const size_t size = oldSize + 32;
732
0
    ADDIN_ARGV** new_array =
733
0
        (ADDIN_ARGV**)realloc((void*)settings->DynamicChannelArray, sizeof(ADDIN_ARGV*) * size);
734
735
0
    if (!new_array)
736
0
      return FALSE;
737
738
0
    settings->DynamicChannelArray = new_array;
739
0
    {
740
0
      for (size_t x = oldSize; x < size; x++)
741
0
        settings->DynamicChannelArray[x] = nullptr;
742
0
    }
743
0
    if (!freerdp_settings_set_uint32(settings, FreeRDP_DynamicChannelArraySize,
744
0
                                     WINPR_ASSERTING_INT_CAST(uint32_t, size)))
745
0
      return FALSE;
746
0
  }
747
748
0
  count = freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelCount);
749
0
  settings->DynamicChannelArray[count++] = channel;
750
0
  return freerdp_settings_set_uint32(settings, FreeRDP_DynamicChannelCount, count);
751
0
}
752
753
ADDIN_ARGV* freerdp_dynamic_channel_collection_find(const rdpSettings* settings, const char* name)
754
0
{
755
0
  WINPR_ASSERT(settings);
756
0
  WINPR_ASSERT(name);
757
758
0
  for (UINT32 index = 0;
759
0
       index < freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelCount); index++)
760
0
  {
761
0
    ADDIN_ARGV* channel = settings->DynamicChannelArray[index];
762
763
0
    if (strcmp(channel->argv[0], name) == 0)
764
0
      return channel;
765
0
  }
766
767
0
  return nullptr;
768
0
}
769
770
void freerdp_addin_argv_free(ADDIN_ARGV* args)
771
0
{
772
0
  if (!args)
773
0
    return;
774
775
0
  if (args->argv)
776
0
  {
777
0
    for (int index = 0; index < args->argc; index++)
778
0
      free(args->argv[index]);
779
0
    free((void*)args->argv);
780
0
  }
781
782
0
  free(args);
783
0
}
784
785
ADDIN_ARGV* freerdp_addin_argv_new(size_t argc, const char* const argv[])
786
0
{
787
0
  if (argc > INT32_MAX)
788
0
    return nullptr;
789
790
0
  ADDIN_ARGV* args = calloc(1, sizeof(ADDIN_ARGV));
791
0
  if (!args)
792
0
    return nullptr;
793
0
  if (argc == 0)
794
0
    return args;
795
796
0
  args->argc = (int)argc;
797
0
  args->argv = (char**)calloc(argc, sizeof(char*));
798
0
  if (!args->argv)
799
0
    goto fail;
800
801
0
  if (argv)
802
0
  {
803
0
    for (size_t x = 0; x < argc; x++)
804
0
    {
805
0
      args->argv[x] = _strdup(argv[x]);
806
0
      if (!args->argv[x])
807
0
        goto fail;
808
0
    }
809
0
  }
810
0
  return args;
811
812
0
fail:
813
0
  WINPR_PRAGMA_DIAG_PUSH
814
0
  WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
815
0
  freerdp_addin_argv_free(args);
816
0
  WINPR_PRAGMA_DIAG_POP
817
0
  return nullptr;
818
0
}
819
820
ADDIN_ARGV* freerdp_addin_argv_clone(const ADDIN_ARGV* args)
821
0
{
822
0
  union
823
0
  {
824
0
    char** c;
825
0
    const char** cc;
826
0
  } cnv;
827
0
  if (!args)
828
0
    return nullptr;
829
0
  cnv.c = args->argv;
830
0
  return freerdp_addin_argv_new(WINPR_ASSERTING_INT_CAST(uint32_t, args->argc), cnv.cc);
831
0
}
832
833
void freerdp_dynamic_channel_collection_free(rdpSettings* settings)
834
230k
{
835
230k
  WINPR_ASSERT(settings);
836
837
230k
  if (settings->DynamicChannelArray)
838
0
  {
839
0
    for (UINT32 i = 0;
840
0
         i < freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelArraySize); i++)
841
0
      freerdp_addin_argv_free(settings->DynamicChannelArray[i]);
842
0
  }
843
844
230k
  free((void*)settings->DynamicChannelArray);
845
230k
  if (!freerdp_settings_set_uint32(settings, FreeRDP_DynamicChannelArraySize, 0))
846
0
    WLog_WARN(
847
230k
        TAG,
848
230k
        "freerdp_settings_set_uint32(settings, FreeRDP_DynamicChannelArraySize, 0) failed");
849
230k
  settings->DynamicChannelArray = nullptr;
850
230k
  if (!freerdp_settings_set_uint32(settings, FreeRDP_DynamicChannelCount, 0))
851
0
    WLog_WARN(TAG,
852
230k
              "freerdp_settings_set_uint32(settings, FreeRDP_DynamicChannelCount, 0) failed");
853
230k
}
854
855
static void freerdp_capability_data_free(rdpSettings* settings, size_t offset, BOOL full)
856
304k
{
857
304k
  WINPR_ASSERT(settings);
858
859
304k
  if (settings->ReceivedCapabilityData)
860
91.6k
  {
861
2.93M
    for (size_t x = offset; x < settings->ReceivedCapabilitiesSize; x++)
862
2.84M
    {
863
2.84M
      free(settings->ReceivedCapabilityData[x]);
864
2.84M
      settings->ReceivedCapabilityData[x] = nullptr;
865
2.84M
    }
866
91.6k
    if (full)
867
88.7k
    {
868
88.7k
      free((void*)settings->ReceivedCapabilityData);
869
88.7k
      settings->ReceivedCapabilityData = nullptr;
870
88.7k
    }
871
91.6k
  }
872
304k
}
873
874
void freerdp_capability_buffer_free(rdpSettings* settings)
875
106k
{
876
106k
  WINPR_ASSERT(settings);
877
878
106k
  freerdp_capability_data_free(settings, 0, TRUE);
879
880
106k
  free(settings->ReceivedCapabilityDataSizes);
881
106k
  settings->ReceivedCapabilityDataSizes = nullptr;
882
883
106k
  free(settings->ReceivedCapabilities);
884
106k
  settings->ReceivedCapabilities = nullptr;
885
886
106k
  settings->ReceivedCapabilitiesSize = 0;
887
106k
}
888
889
static BOOL resize_setting(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer id, size_t oldsize,
890
                           size_t size, size_t base)
891
183k
{
892
183k
  void* old = freerdp_settings_get_pointer_writable(settings, id);
893
183k
  uint8_t* ptr = realloc(old, size * base);
894
183k
  if (!ptr)
895
0
    return FALSE;
896
897
183k
  if (size > oldsize)
898
177k
  {
899
177k
    const size_t diff = size - oldsize;
900
177k
    memset(&ptr[oldsize * base], 0, diff * base);
901
177k
  }
902
903
  // NOLINTNEXTLINE(clang-analyzer-unix.Malloc
904
183k
  return freerdp_settings_set_pointer(settings, id, ptr);
905
183k
}
906
907
static BOOL resize_setting_ptr(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer id,
908
                               size_t oldsize, size_t size, size_t base)
909
91.6k
{
910
91.6k
  WINPR_ASSERT(base == sizeof(void*));
911
912
91.6k
  uint8_t* old = freerdp_settings_get_pointer_writable(settings, id);
913
91.6k
  if (size < oldsize)
914
0
  {
915
0
    uint8_t** optr = WINPR_REINTERPRET_CAST(old, uint8_t*, uint8_t**);
916
0
    for (size_t x = size; x < oldsize; x++)
917
0
    {
918
0
      uint8_t* ptr = optr[x];
919
0
      free(ptr);
920
0
    }
921
0
  }
922
91.6k
  uint8_t* ptr = realloc(old, size * base);
923
91.6k
  if (!ptr)
924
0
    return FALSE;
925
926
91.6k
  uint8_t** optr = WINPR_REINTERPRET_CAST(ptr, uint8_t*, uint8_t**);
927
2.93M
  for (size_t x = oldsize; x < size; x++)
928
2.84M
  {
929
2.84M
    optr[x] = nullptr;
930
2.84M
  }
931
932
  // NOLINTNEXTLINE(clang-analyzer-unix.Malloc
933
91.6k
  return freerdp_settings_set_pointer(settings, id, ptr);
934
91.6k
}
935
936
BOOL freerdp_capability_buffer_resize(rdpSettings* settings, size_t count, BOOL force)
937
411k
{
938
411k
  WINPR_ASSERT(settings);
939
940
411k
  const uint32_t len = settings->ReceivedCapabilitiesSize;
941
411k
  if (!force)
942
390k
  {
943
390k
    if (len == count)
944
319k
      return TRUE;
945
390k
  }
946
947
91.6k
  freerdp_capability_data_free(settings, count, FALSE);
948
949
91.6k
  if (count == 0)
950
0
  {
951
0
    freerdp_capability_buffer_free(settings);
952
0
    return TRUE;
953
0
  }
954
955
91.6k
  const size_t oldsize = settings->ReceivedCapabilitiesSize;
956
91.6k
  if (!resize_setting(settings, FreeRDP_ReceivedCapabilityDataSizes, oldsize, count,
957
91.6k
                      sizeof(uint32_t)))
958
0
    return FALSE;
959
91.6k
  if (!resize_setting_ptr(settings, FreeRDP_ReceivedCapabilityData, oldsize, count,
960
91.6k
                          sizeof(uint8_t*)))
961
0
    return FALSE;
962
91.6k
  if (!resize_setting(settings, FreeRDP_ReceivedCapabilities, oldsize, count, sizeof(uint32_t)))
963
0
    return FALSE;
964
965
91.6k
  settings->ReceivedCapabilitiesSize = WINPR_ASSERTING_INT_CAST(uint32_t, count);
966
91.6k
  return TRUE;
967
91.6k
}
968
969
BOOL freerdp_capability_buffer_copy(rdpSettings* settings, const rdpSettings* src)
970
20.6k
{
971
20.6k
  WINPR_ASSERT(settings);
972
20.6k
  WINPR_ASSERT(src);
973
974
20.6k
  if (src->ReceivedCapabilitiesSize == 0)
975
0
    return TRUE;
976
977
20.6k
  if (!freerdp_capability_buffer_resize(settings, src->ReceivedCapabilitiesSize, TRUE))
978
0
    return FALSE;
979
980
680k
  for (UINT32 x = 0; x < src->ReceivedCapabilitiesSize; x++)
981
660k
  {
982
660k
    WINPR_ASSERT(settings->ReceivedCapabilities);
983
660k
    settings->ReceivedCapabilities[x] = src->ReceivedCapabilities[x];
984
985
660k
    WINPR_ASSERT(settings->ReceivedCapabilityDataSizes);
986
660k
    settings->ReceivedCapabilityDataSizes[x] = src->ReceivedCapabilityDataSizes[x];
987
988
660k
    WINPR_ASSERT(settings->ReceivedCapabilityData);
989
660k
    if (src->ReceivedCapabilityDataSizes[x] > 0)
990
637
    {
991
637
      void* tmp = realloc(settings->ReceivedCapabilityData[x],
992
637
                          settings->ReceivedCapabilityDataSizes[x]);
993
637
      if (!tmp)
994
0
        return FALSE;
995
637
      memcpy(tmp, src->ReceivedCapabilityData[x], src->ReceivedCapabilityDataSizes[x]);
996
637
      settings->ReceivedCapabilityData[x] = tmp;
997
637
    }
998
659k
    else
999
659k
    {
1000
659k
      free(settings->ReceivedCapabilityData[x]);
1001
659k
      settings->ReceivedCapabilityData[x] = nullptr;
1002
659k
    }
1003
660k
  }
1004
20.6k
  return TRUE;
1005
20.6k
}
1006
1007
static void target_net_addresses_free(rdpSettings* settings, size_t offset)
1008
461k
{
1009
461k
  WINPR_ASSERT(settings);
1010
1011
461k
  if (settings->TargetNetAddresses)
1012
0
  {
1013
0
    for (size_t index = offset; index < settings->TargetNetAddressCount; index++)
1014
0
    {
1015
0
      free(settings->TargetNetAddresses[index]);
1016
0
      settings->TargetNetAddresses[index] = nullptr;
1017
0
    }
1018
0
  }
1019
461k
}
1020
1021
void freerdp_target_net_addresses_free(rdpSettings* settings)
1022
355k
{
1023
355k
  WINPR_ASSERT(settings);
1024
1025
355k
  target_net_addresses_free(settings, 0);
1026
1027
355k
  free((void*)settings->TargetNetAddresses);
1028
355k
  settings->TargetNetAddresses = nullptr;
1029
1030
355k
  free(settings->TargetNetPorts);
1031
355k
  settings->TargetNetPorts = nullptr;
1032
1033
355k
  settings->TargetNetAddressCount = 0;
1034
355k
}
1035
1036
BOOL freerdp_target_net_addresses_resize(rdpSettings* settings, size_t count)
1037
230k
{
1038
230k
  WINPR_ASSERT(settings);
1039
1040
230k
  if (count == 0)
1041
230k
  {
1042
230k
    freerdp_target_net_addresses_free(settings);
1043
230k
    return TRUE;
1044
230k
  }
1045
1046
0
  const uint32_t len = settings->TargetNetAddressCount;
1047
0
  if (!resize_setting_ptr(settings, FreeRDP_TargetNetAddresses, len, count, sizeof(char*)))
1048
0
    return FALSE;
1049
0
  if (!resize_setting(settings, FreeRDP_TargetNetPorts, len, count, sizeof(uint32_t)))
1050
0
    return FALSE;
1051
1052
0
  settings->TargetNetAddressCount = WINPR_ASSERTING_INT_CAST(uint32_t, count);
1053
0
  return TRUE;
1054
0
}
1055
1056
void freerdp_server_license_issuers_free(rdpSettings* settings)
1057
301k
{
1058
301k
  WINPR_ASSERT(settings);
1059
1060
301k
  if (settings->ServerLicenseProductIssuers)
1061
88.7k
  {
1062
621k
    for (UINT32 x = 0; x < settings->ServerLicenseProductIssuersCount; x++)
1063
532k
      free(settings->ServerLicenseProductIssuers[x]);
1064
88.7k
  }
1065
301k
  free((void*)settings->ServerLicenseProductIssuers);
1066
301k
  settings->ServerLicenseProductIssuers = nullptr;
1067
301k
  settings->ServerLicenseProductIssuersCount = 0;
1068
301k
}
1069
1070
BOOL freerdp_server_license_issuers_copy(rdpSettings* settings, char** issuers, UINT32 count)
1071
88.7k
{
1072
88.7k
  WINPR_ASSERT(settings);
1073
88.7k
  WINPR_ASSERT(issuers || (count == 0));
1074
1075
88.7k
  if (!freerdp_settings_set_pointer_len(settings, FreeRDP_ServerLicenseProductIssuers, nullptr,
1076
88.7k
                                        count))
1077
0
    return FALSE;
1078
1079
621k
  for (UINT32 x = 0; x < count; x++)
1080
532k
  {
1081
532k
    char* issuer = _strdup(issuers[x]);
1082
532k
    if (!issuer)
1083
0
      return FALSE;
1084
532k
    settings->ServerLicenseProductIssuers[x] = issuer;
1085
532k
  }
1086
1087
88.7k
  return TRUE;
1088
88.7k
}
1089
1090
void freerdp_performance_flags_make(rdpSettings* settings)
1091
0
{
1092
0
  UINT32 PerformanceFlags = PERF_FLAG_NONE;
1093
1094
0
  if (freerdp_settings_get_bool(settings, FreeRDP_AllowFontSmoothing))
1095
0
    PerformanceFlags |= PERF_ENABLE_FONT_SMOOTHING;
1096
1097
0
  if (freerdp_settings_get_bool(settings, FreeRDP_AllowDesktopComposition))
1098
0
    PerformanceFlags |= PERF_ENABLE_DESKTOP_COMPOSITION;
1099
1100
0
  if (freerdp_settings_get_bool(settings, FreeRDP_DisableWallpaper))
1101
0
    PerformanceFlags |= PERF_DISABLE_WALLPAPER;
1102
1103
0
  if (freerdp_settings_get_bool(settings, FreeRDP_DisableFullWindowDrag))
1104
0
    PerformanceFlags |= PERF_DISABLE_FULLWINDOWDRAG;
1105
1106
0
  if (freerdp_settings_get_bool(settings, FreeRDP_DisableMenuAnims))
1107
0
    PerformanceFlags |= PERF_DISABLE_MENUANIMATIONS;
1108
1109
0
  if (freerdp_settings_get_bool(settings, FreeRDP_DisableThemes))
1110
0
    PerformanceFlags |= PERF_DISABLE_THEMING;
1111
0
  if (!freerdp_settings_set_uint32(settings, FreeRDP_PerformanceFlags, PerformanceFlags))
1112
0
    WLog_WARN(TAG,
1113
0
              "freerdp_settings_set_uint32(settings, FreeRDP_PerformanceFlags, 0x%08" PRIx32
1114
0
              ") failed",
1115
0
              PerformanceFlags);
1116
0
}
1117
1118
void freerdp_performance_flags_split(rdpSettings* settings)
1119
619
{
1120
619
  BOOL res =
1121
619
      freerdp_settings_set_bool(settings, FreeRDP_AllowFontSmoothing,
1122
619
                                (freerdp_settings_get_uint32(settings, FreeRDP_PerformanceFlags) &
1123
619
                                 PERF_ENABLE_FONT_SMOOTHING) != 0);
1124
619
  if (!freerdp_settings_set_bool(
1125
619
          settings, FreeRDP_AllowDesktopComposition,
1126
619
          (freerdp_settings_get_uint32(settings, FreeRDP_PerformanceFlags) &
1127
619
           PERF_ENABLE_DESKTOP_COMPOSITION) != 0))
1128
0
    res = FALSE;
1129
619
  if (!freerdp_settings_set_bool(
1130
619
          settings, FreeRDP_DisableWallpaper,
1131
619
          (freerdp_settings_get_uint32(settings, FreeRDP_PerformanceFlags) &
1132
619
           PERF_DISABLE_WALLPAPER) != 0))
1133
0
    res = FALSE;
1134
619
  if (!freerdp_settings_set_bool(
1135
619
          settings, FreeRDP_DisableFullWindowDrag,
1136
619
          (freerdp_settings_get_uint32(settings, FreeRDP_PerformanceFlags) &
1137
619
           PERF_DISABLE_FULLWINDOWDRAG) != 0))
1138
0
    res = FALSE;
1139
619
  if (!freerdp_settings_set_bool(
1140
619
          settings, FreeRDP_DisableMenuAnims,
1141
619
          (freerdp_settings_get_uint32(settings, FreeRDP_PerformanceFlags) &
1142
619
           PERF_DISABLE_MENUANIMATIONS) != 0))
1143
0
    res = FALSE;
1144
619
  if (!freerdp_settings_set_bool(
1145
619
          settings, FreeRDP_DisableThemes,
1146
619
          (freerdp_settings_get_uint32(settings, FreeRDP_PerformanceFlags) &
1147
619
           PERF_DISABLE_THEMING) != 0))
1148
0
    res = FALSE;
1149
619
  if (!res)
1150
0
  {
1151
0
    WLog_WARN(TAG, "freerdp_performance_flags_split() failed");
1152
0
  }
1153
619
}
1154
1155
BOOL freerdp_set_gateway_usage_method(rdpSettings* settings, UINT32 GatewayUsageMethod)
1156
0
{
1157
0
  if (!freerdp_settings_set_uint32(settings, FreeRDP_GatewayUsageMethod, GatewayUsageMethod))
1158
0
    return FALSE;
1159
1160
0
  if (GatewayUsageMethod == TSC_PROXY_MODE_NONE_DIRECT)
1161
0
  {
1162
0
    if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayEnabled, FALSE) ||
1163
0
        !freerdp_settings_set_bool(settings, FreeRDP_GatewayBypassLocal, FALSE))
1164
0
      return FALSE;
1165
0
  }
1166
0
  else if (GatewayUsageMethod == TSC_PROXY_MODE_DIRECT)
1167
0
  {
1168
0
    if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayEnabled, TRUE) ||
1169
0
        !freerdp_settings_set_bool(settings, FreeRDP_GatewayBypassLocal, FALSE))
1170
0
      return FALSE;
1171
0
  }
1172
0
  else if (GatewayUsageMethod == TSC_PROXY_MODE_DETECT)
1173
0
  {
1174
0
    if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayEnabled, TRUE) ||
1175
0
        !freerdp_settings_set_bool(settings, FreeRDP_GatewayBypassLocal, TRUE))
1176
0
      return FALSE;
1177
0
  }
1178
0
  else if (GatewayUsageMethod == TSC_PROXY_MODE_DEFAULT)
1179
0
  {
1180
    /**
1181
     * This corresponds to "Automatically detect RD Gateway server settings",
1182
     * which means the client attempts to use gateway group policy settings
1183
     * http://technet.microsoft.com/en-us/library/cc770601.aspx
1184
     */
1185
0
    if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayEnabled, FALSE) ||
1186
0
        !freerdp_settings_set_bool(settings, FreeRDP_GatewayBypassLocal, FALSE))
1187
0
      return FALSE;
1188
0
  }
1189
0
  else if (GatewayUsageMethod == TSC_PROXY_MODE_NONE_DETECT)
1190
0
  {
1191
0
    if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayEnabled, FALSE) ||
1192
0
        !freerdp_settings_set_bool(settings, FreeRDP_GatewayBypassLocal, FALSE))
1193
0
      return FALSE;
1194
0
  }
1195
1196
0
  return TRUE;
1197
0
}
1198
1199
UINT32 freerdp_get_gateway_usage_method(const rdpSettings* settings)
1200
0
{
1201
0
  if (freerdp_settings_get_bool(settings, FreeRDP_GatewayEnabled))
1202
0
  {
1203
0
    if (freerdp_settings_get_bool(settings, FreeRDP_GatewayBypassLocal))
1204
0
      return TSC_PROXY_MODE_DETECT;
1205
0
    return TSC_PROXY_MODE_DIRECT;
1206
0
  }
1207
1208
0
  return freerdp_settings_get_uint32(settings, FreeRDP_GatewayUsageMethod);
1209
0
}
1210
1211
void freerdp_update_gateway_usage_method(rdpSettings* settings, UINT32 GatewayEnabled,
1212
                                         UINT32 GatewayBypassLocal)
1213
0
{
1214
0
  UINT32 GatewayUsageMethod = 0;
1215
1216
0
  if (!GatewayEnabled && !GatewayBypassLocal)
1217
0
    GatewayUsageMethod = TSC_PROXY_MODE_NONE_DIRECT;
1218
0
  else if (GatewayEnabled && !GatewayBypassLocal)
1219
0
    GatewayUsageMethod = TSC_PROXY_MODE_DIRECT;
1220
0
  else if (GatewayEnabled && GatewayBypassLocal)
1221
0
    GatewayUsageMethod = TSC_PROXY_MODE_DETECT;
1222
1223
0
  if (!freerdp_set_gateway_usage_method(settings, GatewayUsageMethod))
1224
0
    WLog_WARN(TAG,
1225
0
              "freerdp_set_gateway_usage_method(settings, GatewayUsageMethod=%" PRIu32
1226
0
              ") failed)",
1227
0
              GatewayUsageMethod);
1228
0
}
1229
1230
#if defined(WITH_FREERDP_DEPRECATED)
1231
BOOL freerdp_get_param_bool(const rdpSettings* settings, int id)
1232
{
1233
  return freerdp_settings_get_bool(settings, (FreeRDP_Settings_Keys_Bool)id);
1234
}
1235
1236
int freerdp_set_param_bool(rdpSettings* settings, int id, BOOL param)
1237
{
1238
  return freerdp_settings_set_bool(settings, (FreeRDP_Settings_Keys_Bool)id, param) ? 0 : -1;
1239
}
1240
1241
int freerdp_get_param_int(const rdpSettings* settings, int id)
1242
{
1243
  return freerdp_settings_get_int32(settings, (FreeRDP_Settings_Keys_Int32)id);
1244
}
1245
1246
int freerdp_set_param_int(rdpSettings* settings, int id, int param)
1247
{
1248
  return freerdp_settings_set_int32(settings, (FreeRDP_Settings_Keys_Int32)id, param) ? 0 : -1;
1249
}
1250
1251
UINT32 freerdp_get_param_uint32(const rdpSettings* settings, int id)
1252
{
1253
  return freerdp_settings_get_uint32(settings, (FreeRDP_Settings_Keys_UInt32)id);
1254
}
1255
1256
int freerdp_set_param_uint32(rdpSettings* settings, int id, UINT32 param)
1257
{
1258
  return freerdp_settings_set_uint32(settings, (FreeRDP_Settings_Keys_UInt32)id, param) ? 0 : -1;
1259
}
1260
1261
UINT64 freerdp_get_param_uint64(const rdpSettings* settings, int id)
1262
{
1263
  return freerdp_settings_get_uint64(settings, (FreeRDP_Settings_Keys_UInt64)id);
1264
}
1265
1266
int freerdp_set_param_uint64(rdpSettings* settings, int id, UINT64 param)
1267
{
1268
  return freerdp_settings_set_uint64(settings, (FreeRDP_Settings_Keys_UInt64)id, param) ? 0 : -1;
1269
}
1270
1271
char* freerdp_get_param_string(const rdpSettings* settings, int id)
1272
{
1273
  const char* str = freerdp_settings_get_string(settings, (FreeRDP_Settings_Keys_String)id);
1274
  return WINPR_CAST_CONST_PTR_AWAY(str, char*);
1275
}
1276
1277
int freerdp_set_param_string(rdpSettings* settings, int id, const char* param)
1278
{
1279
  return freerdp_settings_set_string(settings, (FreeRDP_Settings_Keys_String)id, param) ? 0 : -1;
1280
}
1281
#endif
1282
1283
static BOOL value_to_uint(const char* value, ULONGLONG* result, ULONGLONG min, ULONGLONG max)
1284
0
{
1285
0
  char* endptr = nullptr;
1286
0
  unsigned long long rc = 0;
1287
1288
0
  if (!value || !result)
1289
0
    return FALSE;
1290
1291
0
  errno = 0;
1292
0
  rc = _strtoui64(value, &endptr, 0);
1293
1294
0
  if (errno != 0)
1295
0
    return FALSE;
1296
1297
0
  if (endptr == value)
1298
0
    return FALSE;
1299
1300
0
  if ((rc < min) || (rc > max))
1301
0
    return FALSE;
1302
1303
0
  *result = rc;
1304
0
  return TRUE;
1305
0
}
1306
1307
static BOOL value_to_int(const char* value, LONGLONG* result, LONGLONG min, LONGLONG max)
1308
0
{
1309
0
  char* endptr = nullptr;
1310
0
  long long rc = 0;
1311
1312
0
  if (!value || !result)
1313
0
    return FALSE;
1314
1315
0
  errno = 0;
1316
0
  rc = _strtoi64(value, &endptr, 0);
1317
1318
0
  if (errno != 0)
1319
0
    return FALSE;
1320
1321
0
  if (endptr == value)
1322
0
    return FALSE;
1323
1324
0
  if ((rc < min) || (rc > max))
1325
0
    return FALSE;
1326
1327
0
  *result = rc;
1328
0
  return TRUE;
1329
0
}
1330
1331
static BOOL parsing_fail(const char* key, const char* type, const char* value)
1332
0
{
1333
0
  WLog_ERR(TAG, "Failed to parse key [%s] of type [%s]: value [%s]", key, type, value);
1334
0
  return FALSE;
1335
0
}
1336
1337
BOOL freerdp_settings_set_value_for_name(rdpSettings* settings, const char* name, const char* value)
1338
0
{
1339
0
  ULONGLONG uval = 0;
1340
0
  LONGLONG ival = 0;
1341
0
  SSIZE_T type = 0;
1342
1343
0
  if (!settings || !name)
1344
0
    return FALSE;
1345
1346
0
  const SSIZE_T i = freerdp_settings_get_key_for_name(name);
1347
0
  if (i < 0)
1348
0
  {
1349
0
    WLog_ERR(TAG, "Invalid settings key [%s]", name);
1350
0
    return FALSE;
1351
0
  }
1352
1353
0
  const SSIZE_T index = i;
1354
1355
0
  type = freerdp_settings_get_type_for_key(index);
1356
0
  switch (type)
1357
0
  {
1358
1359
0
    case RDP_SETTINGS_TYPE_BOOL:
1360
0
    {
1361
0
      const BOOL val = (_strnicmp(value, "TRUE", 5) == 0) || (_strnicmp(value, "ON", 5) == 0);
1362
0
      const BOOL nval =
1363
0
          (_strnicmp(value, "FALSE", 6) == 0) || (_strnicmp(value, "OFF", 6) == 0);
1364
0
      if (!val && !nval)
1365
0
        return parsing_fail(name, "BOOL", value);
1366
1367
      // NOLINTNEXTLINE(clang-analyzer-optin.core.EnumCastOutOfRange)
1368
0
      return freerdp_settings_set_bool(settings, (FreeRDP_Settings_Keys_Bool)index, val);
1369
0
    }
1370
0
    case RDP_SETTINGS_TYPE_UINT16:
1371
0
      if (!value_to_uint(value, &uval, 0, UINT16_MAX))
1372
0
        return parsing_fail(name, "UINT16", value);
1373
1374
      // NOLINTNEXTLINE(clang-analyzer-optin.core.EnumCastOutOfRange)
1375
0
      if (!freerdp_settings_set_uint16(settings, (FreeRDP_Settings_Keys_UInt16)index,
1376
0
                                       (UINT16)uval))
1377
0
        return parsing_fail(name, "UINT16", value);
1378
0
      return TRUE;
1379
1380
0
    case RDP_SETTINGS_TYPE_INT16:
1381
0
      if (!value_to_int(value, &ival, INT16_MIN, INT16_MAX))
1382
0
        return parsing_fail(name, "INT16", value);
1383
1384
      // NOLINTNEXTLINE(clang-analyzer-optin.core.EnumCastOutOfRange)
1385
0
      if (!freerdp_settings_set_int16(settings, (FreeRDP_Settings_Keys_Int16)index,
1386
0
                                      (INT16)ival))
1387
0
        return parsing_fail(name, "INT16", value);
1388
0
      return TRUE;
1389
0
    case RDP_SETTINGS_TYPE_UINT32:
1390
0
      if (!value_to_uint(value, &uval, 0, UINT32_MAX))
1391
0
        return parsing_fail(name, "UINT32", value);
1392
1393
      // NOLINTNEXTLINE(clang-analyzer-optin.core.EnumCastOutOfRange)
1394
0
      if (!freerdp_settings_set_uint32(settings, (FreeRDP_Settings_Keys_UInt32)index,
1395
0
                                       (UINT32)uval))
1396
0
        return parsing_fail(name, "UINT32", value);
1397
0
      return TRUE;
1398
0
    case RDP_SETTINGS_TYPE_INT32:
1399
0
      if (!value_to_int(value, &ival, INT32_MIN, INT32_MAX))
1400
0
        return parsing_fail(name, "INT32", value);
1401
1402
      // NOLINTNEXTLINE(clang-analyzer-optin.core.EnumCastOutOfRange)
1403
0
      if (!freerdp_settings_set_int32(settings, (FreeRDP_Settings_Keys_Int32)index,
1404
0
                                      (INT32)ival))
1405
0
        return parsing_fail(name, "INT32", value);
1406
0
      return TRUE;
1407
0
    case RDP_SETTINGS_TYPE_UINT64:
1408
0
      if (!value_to_uint(value, &uval, 0, UINT64_MAX))
1409
0
        return parsing_fail(name, "UINT64", value);
1410
1411
      // NOLINTNEXTLINE(clang-analyzer-optin.core.EnumCastOutOfRange)
1412
0
      if (!freerdp_settings_set_uint64(settings, (FreeRDP_Settings_Keys_UInt64)index, uval))
1413
0
        return parsing_fail(name, "UINT64", value);
1414
0
      return TRUE;
1415
0
    case RDP_SETTINGS_TYPE_INT64:
1416
0
      if (!value_to_int(value, &ival, INT64_MIN, INT64_MAX))
1417
0
        return parsing_fail(name, "INT64", value);
1418
1419
      // NOLINTNEXTLINE(clang-analyzer-optin.core.EnumCastOutOfRange)
1420
0
      if (!freerdp_settings_set_int64(settings, (FreeRDP_Settings_Keys_Int64)index, ival))
1421
0
        return parsing_fail(name, "INT64", value);
1422
0
      return TRUE;
1423
1424
0
    case RDP_SETTINGS_TYPE_STRING:
1425
      // NOLINTNEXTLINE(clang-analyzer-optin.core.EnumCastOutOfRange)
1426
0
      return freerdp_settings_set_string(settings, (FreeRDP_Settings_Keys_String)index,
1427
0
                                         value);
1428
0
    case RDP_SETTINGS_TYPE_POINTER:
1429
0
      return parsing_fail(name, "POINTER", value);
1430
0
    default:
1431
0
      return FALSE;
1432
0
  }
1433
0
  return FALSE;
1434
0
}
1435
1436
BOOL freerdp_settings_set_pointer_len_(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer id,
1437
                                       FreeRDP_Settings_Keys_UInt32 lenId, const void* data,
1438
                                       size_t len, size_t size)
1439
3.13M
{
1440
3.13M
  BOOL rc = FALSE;
1441
3.13M
  void* copy = nullptr;
1442
3.13M
  void* old = freerdp_settings_get_pointer_writable(settings, id);
1443
3.13M
  free(old);
1444
3.13M
  if (!freerdp_settings_set_pointer(settings, id, nullptr))
1445
0
    return FALSE;
1446
3.13M
  if (lenId != FreeRDP_UINT32_UNUSED)
1447
2.26M
  {
1448
2.26M
    if (!freerdp_settings_set_uint32(settings, lenId, 0))
1449
0
      return FALSE;
1450
2.26M
  }
1451
1452
3.13M
  if (len > UINT32_MAX)
1453
0
    return FALSE;
1454
3.13M
  if (len == 0)
1455
2.61M
    return TRUE;
1456
521k
  copy = calloc(len, size);
1457
521k
  if (!copy)
1458
0
    return FALSE;
1459
521k
  if (data)
1460
284k
    memcpy(copy, data, len * size);
1461
521k
  rc = freerdp_settings_set_pointer(settings, id, copy);
1462
521k
  if (!rc)
1463
0
  {
1464
0
    free(copy);
1465
0
    return FALSE;
1466
0
  }
1467
1468
  // freerdp_settings_set_pointer takes ownership of copy
1469
  //  NOLINTNEXTLINE(clang-analyzer-unix.Malloc)
1470
521k
  if (lenId == FreeRDP_UINT32_UNUSED)
1471
230k
    return TRUE;
1472
290k
  return freerdp_settings_set_uint32(settings, lenId, (UINT32)len);
1473
521k
}
1474
1475
const void* freerdp_settings_get_pointer(const rdpSettings* settings,
1476
                                         FreeRDP_Settings_Keys_Pointer id)
1477
301k
{
1478
301k
  union
1479
301k
  {
1480
301k
    const rdpSettings* pc;
1481
301k
    rdpSettings* p;
1482
301k
  } cnv;
1483
301k
  cnv.pc = settings;
1484
301k
  return freerdp_settings_get_pointer_writable(cnv.p, id);
1485
301k
}
1486
1487
BOOL freerdp_settings_set_pointer_len(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer id,
1488
                                      const void* data, size_t len)
1489
4.20M
{
1490
4.20M
  union
1491
4.20M
  {
1492
4.20M
    const void* cv;
1493
4.20M
    void* v;
1494
4.20M
  } cnv;
1495
1496
4.20M
  cnv.cv = data;
1497
4.20M
  if (!settings)
1498
0
    return FALSE;
1499
1500
4.20M
  switch (id)
1501
4.20M
  {
1502
106k
    case FreeRDP_instance:
1503
106k
      if ((len != 0) && (len != sizeof(void*)))
1504
0
      {
1505
0
        WLog_ERR(TAG, "FreeRDP_instance::len must be 0 or %" PRIuz, sizeof(void*));
1506
0
        return FALSE;
1507
0
      }
1508
106k
      settings->instance = cnv.v;
1509
106k
      return TRUE;
1510
195k
    case FreeRDP_RdpServerCertificate:
1511
195k
      freerdp_certificate_free(settings->RdpServerCertificate);
1512
1513
195k
      if (len > 1)
1514
0
      {
1515
0
        WLog_ERR(TAG, "FreeRDP_RdpServerCertificate::len must be 0 or 1");
1516
0
        return FALSE;
1517
0
      }
1518
195k
      settings->RdpServerCertificate = cnv.v;
1519
195k
      if (!settings->RdpServerCertificate && (len > 0))
1520
71.0k
      {
1521
71.0k
        settings->RdpServerCertificate = freerdp_certificate_new();
1522
71.0k
        if (!settings->RdpServerCertificate)
1523
0
          return FALSE;
1524
71.0k
      }
1525
195k
      return TRUE;
1526
124k
    case FreeRDP_RdpServerRsaKey:
1527
124k
      freerdp_key_free(settings->RdpServerRsaKey);
1528
124k
      if (len > 1)
1529
0
      {
1530
0
        WLog_ERR(TAG, "FreeRDP_RdpServerRsaKey::len must be 0 or 1");
1531
0
        return FALSE;
1532
0
      }
1533
124k
      settings->RdpServerRsaKey = (rdpPrivateKey*)cnv.v;
1534
124k
      if (!settings->RdpServerRsaKey && (len > 0))
1535
0
      {
1536
0
        settings->RdpServerRsaKey = freerdp_key_new();
1537
0
        if (!settings->RdpServerRsaKey)
1538
0
          return FALSE;
1539
0
      }
1540
124k
      return TRUE;
1541
124k
    case FreeRDP_RedirectionPassword:
1542
124k
      return freerdp_settings_set_pointer_len_(
1543
124k
          settings, id, FreeRDP_RedirectionPasswordLength, data, len, sizeof(char));
1544
124k
    case FreeRDP_RedirectionTsvUrl:
1545
124k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_RedirectionTsvUrlLength,
1546
124k
                                               data, len, sizeof(char));
1547
106k
    case FreeRDP_RedirectionTargetCertificate:
1548
106k
      freerdp_certificate_free(settings->RedirectionTargetCertificate);
1549
1550
106k
      if (len > 1)
1551
0
      {
1552
0
        WLog_ERR(TAG, "FreeRDP_RedirectionTargetCertificate::len must be 0 or 1");
1553
0
        return FALSE;
1554
0
      }
1555
106k
      settings->RedirectionTargetCertificate = cnv.v;
1556
106k
      if (!settings->RedirectionTargetCertificate && (len > 0))
1557
0
      {
1558
0
        settings->RedirectionTargetCertificate = freerdp_certificate_new();
1559
0
        if (!settings->RedirectionTargetCertificate)
1560
0
          return FALSE;
1561
0
      }
1562
106k
      return TRUE;
1563
124k
    case FreeRDP_RedirectionGuid:
1564
124k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_RedirectionGuidLength,
1565
124k
                                               data, len, sizeof(BYTE));
1566
124k
    case FreeRDP_LoadBalanceInfo:
1567
124k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_LoadBalanceInfoLength,
1568
124k
                                               data, len, sizeof(char));
1569
124k
    case FreeRDP_ServerRandom:
1570
124k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_ServerRandomLength, data,
1571
124k
                                               len, sizeof(char));
1572
124k
    case FreeRDP_ClientRandom:
1573
124k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_ClientRandomLength, data,
1574
124k
                                               len, sizeof(char));
1575
124k
    case FreeRDP_ServerCertificate:
1576
124k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_ServerCertificateLength,
1577
124k
                                               data, len, sizeof(char));
1578
106k
    case FreeRDP_TargetNetAddresses:
1579
106k
      if (!freerdp_target_net_addresses_resize(settings, len))
1580
0
        return FALSE;
1581
106k
      if (data == nullptr)
1582
106k
        target_net_addresses_free(settings, 0);
1583
106k
      return TRUE;
1584
195k
    case FreeRDP_ServerLicenseProductIssuers:
1585
195k
      if (data == nullptr)
1586
195k
        freerdp_server_license_issuers_free(settings);
1587
195k
      return freerdp_settings_set_pointer_len_(
1588
195k
          settings, id, FreeRDP_ServerLicenseProductIssuersCount, data, len, sizeof(char*));
1589
106k
    case FreeRDP_TargetNetPorts:
1590
106k
      if (!freerdp_target_net_addresses_resize(settings, len))
1591
0
        return FALSE;
1592
106k
      if (data == nullptr)
1593
106k
      {
1594
106k
        for (size_t x = 0; x < len; x++)
1595
0
          settings->TargetNetPorts[x] = 0;
1596
106k
      }
1597
106k
      return TRUE;
1598
124k
    case FreeRDP_DeviceArray:
1599
124k
      if (data == nullptr)
1600
124k
        freerdp_device_collection_free(settings);
1601
124k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_DeviceArraySize, data,
1602
124k
                                               len, sizeof(RDPDR_DEVICE*));
1603
195k
    case FreeRDP_ChannelDefArray:
1604
195k
      if ((len > 0) && (len < CHANNEL_MAX_COUNT))
1605
0
        WLog_WARN(TAG,
1606
195k
                  "FreeRDP_ChannelDefArray::len expected to be >= %d, but have %" PRIuz,
1607
195k
                  CHANNEL_MAX_COUNT, len);
1608
195k
      return freerdp_settings_set_pointer_len_(settings, FreeRDP_ChannelDefArray,
1609
195k
                                               FreeRDP_ChannelDefArraySize, data, len,
1610
195k
                                               sizeof(CHANNEL_DEF));
1611
195k
    case FreeRDP_MonitorDefArray:
1612
195k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_MonitorDefArraySize,
1613
195k
                                               data, len, sizeof(rdpMonitor));
1614
195k
    case FreeRDP_ClientAutoReconnectCookie:
1615
195k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_UINT32_UNUSED, data, len,
1616
195k
                                               sizeof(ARC_CS_PRIVATE_PACKET));
1617
195k
    case FreeRDP_ServerAutoReconnectCookie:
1618
195k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_UINT32_UNUSED, data, len,
1619
195k
                                               sizeof(ARC_SC_PRIVATE_PACKET));
1620
124k
    case FreeRDP_ClientTimeZone:
1621
124k
      if (len > 1)
1622
0
      {
1623
0
        WLog_ERR(TAG, "FreeRDP_ClientTimeZone::len must be 0 or 1");
1624
0
        return FALSE;
1625
0
      }
1626
124k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_UINT32_UNUSED, data, len,
1627
124k
                                               sizeof(TIME_ZONE_INFORMATION));
1628
133k
    case FreeRDP_BitmapCacheV2CellInfo:
1629
133k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_BitmapCacheV2NumCells,
1630
133k
                                               data, len, sizeof(BITMAP_CACHE_V2_CELL_INFO));
1631
124k
    case FreeRDP_GlyphCache:
1632
124k
      if ((len != 0) && (len != 10))
1633
0
      {
1634
0
        WLog_ERR(TAG, "FreeRDP_GlyphCache::len must be 0 or 10");
1635
0
        return FALSE;
1636
0
      }
1637
124k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_UINT32_UNUSED, data, len,
1638
124k
                                               sizeof(GLYPH_CACHE_DEFINITION));
1639
124k
    case FreeRDP_FragCache:
1640
124k
      if (len > 1)
1641
0
      {
1642
0
        WLog_ERR(TAG, "FreeRDP_FragCache::len must be 0 or 1");
1643
0
        return FALSE;
1644
0
      }
1645
124k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_UINT32_UNUSED, data, len,
1646
124k
                                               sizeof(GLYPH_CACHE_DEFINITION));
1647
124k
    case FreeRDP_StaticChannelArray:
1648
124k
      if (data == nullptr)
1649
124k
        freerdp_static_channel_collection_free(settings);
1650
124k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_StaticChannelArraySize,
1651
124k
                                               data, len, sizeof(ADDIN_ARGV*));
1652
124k
    case FreeRDP_DynamicChannelArray:
1653
124k
      if (data == nullptr)
1654
124k
        freerdp_dynamic_channel_collection_free(settings);
1655
124k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_DynamicChannelArraySize,
1656
124k
                                               data, len, sizeof(ADDIN_ARGV*));
1657
106k
    case FreeRDP_ReceivedCapabilityData:
1658
106k
      if (!freerdp_capability_buffer_resize(settings, len, FALSE))
1659
0
        return FALSE;
1660
106k
      if (data == nullptr)
1661
106k
      {
1662
106k
        freerdp_capability_data_free(settings, 0, FALSE);
1663
106k
      }
1664
106k
      return TRUE;
1665
106k
    case FreeRDP_ReceivedCapabilities:
1666
106k
      if (!freerdp_capability_buffer_resize(settings, len, FALSE))
1667
0
        return FALSE;
1668
106k
      if (data == nullptr)
1669
106k
      {
1670
106k
        for (size_t x = 0; x < settings->ReceivedCapabilitiesSize; x++)
1671
0
        {
1672
0
          settings->ReceivedCapabilities[x] = 0;
1673
0
        }
1674
106k
      }
1675
106k
      return TRUE;
1676
106k
    case FreeRDP_OrderSupport:
1677
106k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_UINT32_UNUSED, data, len,
1678
106k
                                               sizeof(char));
1679
1680
195k
    case FreeRDP_MonitorIds:
1681
195k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_NumMonitorIds, data, len,
1682
195k
                                               sizeof(UINT32));
1683
1684
106k
    case FreeRDP_ReceivedCapabilityDataSizes:
1685
106k
      if (!freerdp_capability_buffer_resize(settings, len, FALSE))
1686
0
        return FALSE;
1687
106k
      if (data == nullptr)
1688
106k
      {
1689
106k
        for (size_t x = 0; x < settings->ReceivedCapabilitiesSize; x++)
1690
0
          settings->ReceivedCapabilityDataSizes[x] = 0;
1691
106k
      }
1692
106k
      return TRUE;
1693
1694
106k
    case FreeRDP_Password51:
1695
106k
      return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_Password51Length, data,
1696
106k
                                               len, sizeof(char));
1697
0
    default:
1698
0
      if ((data == nullptr) && (len == 0))
1699
0
      {
1700
0
        if (!freerdp_settings_set_pointer(settings, id, nullptr))
1701
0
          return FALSE;
1702
0
      }
1703
0
      else
1704
0
        WLog_WARN(TAG, "Invalid id %d", id);
1705
0
      return FALSE;
1706
4.20M
  }
1707
4.20M
}
1708
1709
void* freerdp_settings_get_pointer_array_writable(const rdpSettings* settings,
1710
                                                  FreeRDP_Settings_Keys_Pointer id, size_t offset)
1711
28.2k
{
1712
28.2k
  size_t max = 0;
1713
28.2k
  if (!settings)
1714
0
    return nullptr;
1715
28.2k
  switch (id)
1716
28.2k
  {
1717
0
    case FreeRDP_ClientAutoReconnectCookie:
1718
0
      max = 1;
1719
0
      if ((offset >= max) || !settings->ClientAutoReconnectCookie)
1720
0
        goto fail;
1721
0
      return &settings->ClientAutoReconnectCookie[offset];
1722
0
    case FreeRDP_ServerAutoReconnectCookie:
1723
0
      max = 1;
1724
0
      if ((offset >= max) || !settings->ServerAutoReconnectCookie)
1725
0
        goto fail;
1726
0
      return &settings->ServerAutoReconnectCookie[offset];
1727
0
    case FreeRDP_ServerCertificate:
1728
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_ServerCertificateLength);
1729
0
      if (offset >= max)
1730
0
        goto fail;
1731
0
      return &settings->ServerCertificate[offset];
1732
0
    case FreeRDP_ServerRandom:
1733
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_ServerRandomLength);
1734
0
      if (offset >= max)
1735
0
        goto fail;
1736
0
      return &settings->ServerRandom[offset];
1737
0
    case FreeRDP_ClientRandom:
1738
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_ClientRandomLength);
1739
0
      if (offset >= max)
1740
0
        goto fail;
1741
0
      return &settings->ClientRandom[offset];
1742
0
    case FreeRDP_LoadBalanceInfo:
1743
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_LoadBalanceInfoLength);
1744
0
      if (offset >= max)
1745
0
        goto fail;
1746
0
      return &settings->LoadBalanceInfo[offset];
1747
1748
0
    case FreeRDP_RedirectionTsvUrl:
1749
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_RedirectionTsvUrlLength);
1750
0
      if (offset >= max)
1751
0
        goto fail;
1752
0
      return &settings->RedirectionTsvUrl[offset];
1753
1754
0
    case FreeRDP_RedirectionPassword:
1755
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_RedirectionPasswordLength);
1756
0
      if (offset >= max)
1757
0
        goto fail;
1758
0
      return &settings->RedirectionPassword[offset];
1759
1760
0
    case FreeRDP_OrderSupport:
1761
0
      max = 32;
1762
0
      if (offset >= max)
1763
0
        goto fail;
1764
0
      return &settings->OrderSupport[offset];
1765
0
    case FreeRDP_MonitorIds:
1766
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_NumMonitorIds);
1767
0
      if (offset >= max)
1768
0
        goto fail;
1769
0
      return &settings->MonitorIds[offset];
1770
0
    case FreeRDP_MonitorDefArray:
1771
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_MonitorDefArraySize);
1772
0
      if (offset >= max)
1773
0
        goto fail;
1774
0
      return &settings->MonitorDefArray[offset];
1775
0
    case FreeRDP_ChannelDefArray:
1776
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_ChannelDefArraySize);
1777
0
      if (offset >= max)
1778
0
        goto fail;
1779
0
      return &settings->ChannelDefArray[offset];
1780
0
    case FreeRDP_DeviceArray:
1781
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_DeviceArraySize);
1782
0
      if (offset >= max)
1783
0
        goto fail;
1784
0
      return settings->DeviceArray[offset];
1785
0
    case FreeRDP_StaticChannelArray:
1786
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelArraySize);
1787
0
      if (offset >= max)
1788
0
        goto fail;
1789
0
      return settings->StaticChannelArray[offset];
1790
0
    case FreeRDP_DynamicChannelArray:
1791
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelArraySize);
1792
0
      if (offset >= max)
1793
0
        goto fail;
1794
0
      return settings->DynamicChannelArray[offset];
1795
0
    case FreeRDP_FragCache:
1796
0
      max = 1;
1797
0
      if (offset >= max)
1798
0
        goto fail;
1799
0
      return &settings->FragCache[offset];
1800
0
    case FreeRDP_GlyphCache:
1801
0
      max = 10;
1802
0
      if (offset >= max)
1803
0
        goto fail;
1804
0
      return &settings->GlyphCache[offset];
1805
27.5k
    case FreeRDP_BitmapCacheV2CellInfo:
1806
27.5k
      max = freerdp_settings_get_uint32(settings, FreeRDP_BitmapCacheV2NumCells);
1807
27.5k
      if (offset >= max)
1808
0
        goto fail;
1809
27.5k
      return &settings->BitmapCacheV2CellInfo[offset];
1810
0
    case FreeRDP_ReceivedCapabilities:
1811
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_ReceivedCapabilitiesSize);
1812
0
      if (offset >= max)
1813
0
        goto fail;
1814
0
      return &settings->ReceivedCapabilities[offset];
1815
0
    case FreeRDP_TargetNetAddresses:
1816
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_TargetNetAddressCount);
1817
0
      if (offset >= max)
1818
0
        goto fail;
1819
0
      WINPR_ASSERT(settings->TargetNetAddresses);
1820
0
      return settings->TargetNetAddresses[offset];
1821
0
    case FreeRDP_TargetNetPorts:
1822
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_TargetNetAddressCount);
1823
0
      if (offset >= max)
1824
0
        goto fail;
1825
0
      WINPR_ASSERT(settings->TargetNetPorts);
1826
0
      return &settings->TargetNetPorts[offset];
1827
633
    case FreeRDP_ClientTimeZone:
1828
633
      max = 1;
1829
633
      if (offset >= max)
1830
0
        goto fail;
1831
633
      return settings->ClientTimeZone;
1832
0
    case FreeRDP_RdpServerCertificate:
1833
0
      max = 1;
1834
0
      if (offset >= max)
1835
0
        goto fail;
1836
0
      return settings->RdpServerCertificate;
1837
0
    case FreeRDP_RdpServerRsaKey:
1838
0
      max = 1;
1839
0
      if (offset >= max)
1840
0
        goto fail;
1841
0
      return settings->RdpServerRsaKey;
1842
0
    case FreeRDP_ServerLicenseProductIssuers:
1843
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_ServerLicenseProductIssuersCount);
1844
0
      if (offset >= max)
1845
0
        goto fail;
1846
0
      return settings->ServerLicenseProductIssuers[offset];
1847
0
    case FreeRDP_ReceivedCapabilityData:
1848
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_ReceivedCapabilitiesSize);
1849
0
      if (offset >= max)
1850
0
        goto fail;
1851
0
      WINPR_ASSERT(settings->ReceivedCapabilityData);
1852
0
      return settings->ReceivedCapabilityData[offset];
1853
1854
0
    case FreeRDP_ReceivedCapabilityDataSizes:
1855
0
      max = freerdp_settings_get_uint32(settings, FreeRDP_ReceivedCapabilitiesSize);
1856
0
      if (offset >= max)
1857
0
        goto fail;
1858
0
      WINPR_ASSERT(settings->ReceivedCapabilityDataSizes);
1859
0
      return &settings->ReceivedCapabilityDataSizes[offset];
1860
0
    default:
1861
0
      WLog_WARN(TAG, "Invalid id %s [%d]", freerdp_settings_get_name_for_key(id), id);
1862
0
      return nullptr;
1863
28.2k
  }
1864
1865
0
fail:
1866
0
  WLog_WARN(TAG, "Invalid offset for %s [%d]: size=%" PRIuz ", offset=%" PRIuz,
1867
0
            freerdp_settings_get_name_for_key(id), id, max, offset);
1868
0
  return nullptr;
1869
28.2k
}
1870
1871
BOOL freerdp_settings_set_pointer_array(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer id,
1872
                                        size_t offset, const void* data)
1873
1.14M
{
1874
1.14M
  size_t maxOffset = 0;
1875
1.14M
  if (!settings)
1876
0
    return FALSE;
1877
1.14M
  switch (id)
1878
1.14M
  {
1879
0
    case FreeRDP_ClientAutoReconnectCookie:
1880
0
      maxOffset = 1;
1881
0
      if ((offset >= maxOffset) || !data || !settings->ClientAutoReconnectCookie)
1882
0
        goto fail;
1883
0
      settings->ClientAutoReconnectCookie[offset] = *(const ARC_CS_PRIVATE_PACKET*)data;
1884
0
      return TRUE;
1885
0
    case FreeRDP_ServerAutoReconnectCookie:
1886
0
      maxOffset = 1;
1887
0
      if ((offset >= maxOffset) || !data || !settings->ServerAutoReconnectCookie)
1888
0
        goto fail;
1889
0
      settings->ServerAutoReconnectCookie[offset] = *(const ARC_SC_PRIVATE_PACKET*)data;
1890
0
      return TRUE;
1891
0
    case FreeRDP_ServerCertificate:
1892
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_ServerCertificateLength);
1893
0
      if ((offset >= maxOffset) || !data)
1894
0
        goto fail;
1895
0
      settings->ServerCertificate[offset] = *(const BYTE*)data;
1896
0
      return TRUE;
1897
0
    case FreeRDP_DeviceArray:
1898
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_DeviceArraySize);
1899
0
      if (offset >= maxOffset)
1900
0
        goto fail;
1901
0
      freerdp_device_free(settings->DeviceArray[offset]);
1902
0
      settings->DeviceArray[offset] = freerdp_device_clone(data);
1903
0
      return TRUE;
1904
0
    case FreeRDP_TargetNetAddresses:
1905
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_TargetNetAddressCount);
1906
0
      if ((offset >= maxOffset) || !data)
1907
0
        goto fail;
1908
0
      WINPR_ASSERT(settings->TargetNetAddresses);
1909
0
      free(settings->TargetNetAddresses[offset]);
1910
0
      settings->TargetNetAddresses[offset] = _strdup((const char*)data);
1911
0
      return settings->TargetNetAddresses[offset] != nullptr;
1912
0
    case FreeRDP_TargetNetPorts:
1913
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_TargetNetAddressCount);
1914
0
      if ((offset >= maxOffset) || !data)
1915
0
        goto fail;
1916
0
      WINPR_ASSERT(settings->TargetNetPorts);
1917
0
      settings->TargetNetPorts[offset] = *((const UINT32*)data);
1918
0
      return TRUE;
1919
0
    case FreeRDP_StaticChannelArray:
1920
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelArraySize);
1921
0
      if ((offset >= maxOffset) || !data)
1922
0
        goto fail;
1923
0
      freerdp_addin_argv_free(settings->StaticChannelArray[offset]);
1924
0
      settings->StaticChannelArray[offset] = freerdp_addin_argv_clone(data);
1925
0
      return TRUE;
1926
0
    case FreeRDP_DynamicChannelArray:
1927
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelArraySize);
1928
0
      if ((offset >= maxOffset) || !data)
1929
0
        goto fail;
1930
0
      freerdp_addin_argv_free(settings->DynamicChannelArray[offset]);
1931
0
      settings->DynamicChannelArray[offset] = freerdp_addin_argv_clone(data);
1932
0
      return TRUE;
1933
368k
    case FreeRDP_BitmapCacheV2CellInfo:
1934
368k
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_BitmapCacheV2NumCells);
1935
368k
      if ((offset >= maxOffset) || !data)
1936
0
        goto fail;
1937
368k
      {
1938
368k
        const BITMAP_CACHE_V2_CELL_INFO* cdata = (const BITMAP_CACHE_V2_CELL_INFO*)data;
1939
368k
        settings->BitmapCacheV2CellInfo[offset] = *cdata;
1940
368k
      }
1941
368k
      return TRUE;
1942
0
    case FreeRDP_ServerRandom:
1943
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_ServerRandomLength);
1944
0
      if ((offset >= maxOffset) || !data)
1945
0
        goto fail;
1946
0
      settings->ServerRandom[offset] = *(const BYTE*)data;
1947
0
      return TRUE;
1948
0
    case FreeRDP_ClientRandom:
1949
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_ClientRandomLength);
1950
0
      if ((offset >= maxOffset) || !data)
1951
0
        goto fail;
1952
0
      settings->ClientRandom[offset] = *(const BYTE*)data;
1953
0
      return TRUE;
1954
0
    case FreeRDP_LoadBalanceInfo:
1955
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_LoadBalanceInfoLength);
1956
0
      if ((offset >= maxOffset) || !data)
1957
0
        goto fail;
1958
0
      settings->LoadBalanceInfo[offset] = *(const BYTE*)data;
1959
0
      return TRUE;
1960
0
    case FreeRDP_RedirectionTsvUrl:
1961
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_RedirectionTsvUrlLength);
1962
0
      if ((offset >= maxOffset) || !data)
1963
0
        goto fail;
1964
0
      settings->RedirectionTsvUrl[offset] = *(const BYTE*)data;
1965
0
      return TRUE;
1966
0
    case FreeRDP_RedirectionPassword:
1967
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_RedirectionPasswordLength);
1968
0
      if ((offset >= maxOffset) || !data)
1969
0
        goto fail;
1970
0
      settings->RedirectionPassword[offset] = *(const BYTE*)data;
1971
0
      return TRUE;
1972
0
    case FreeRDP_OrderSupport:
1973
0
      maxOffset = 32;
1974
0
      if (!settings->OrderSupport)
1975
0
        goto fail;
1976
0
      if ((offset >= maxOffset) || !data)
1977
0
        goto fail;
1978
0
      settings->OrderSupport[offset] = *(const BOOL*)data ? 1 : 0;
1979
0
      return TRUE;
1980
710k
    case FreeRDP_GlyphCache:
1981
710k
      maxOffset = 10;
1982
710k
      if (!settings->GlyphCache)
1983
0
        goto fail;
1984
710k
      if ((offset >= maxOffset) || !data)
1985
0
        goto fail;
1986
710k
      settings->GlyphCache[offset] = *(const GLYPH_CACHE_DEFINITION*)data;
1987
710k
      return TRUE;
1988
71.0k
    case FreeRDP_FragCache:
1989
71.0k
      maxOffset = 1;
1990
71.0k
      if (!settings->FragCache)
1991
0
        goto fail;
1992
71.0k
      if ((offset >= maxOffset) || !data)
1993
0
        goto fail;
1994
71.0k
      settings->FragCache[offset] = *(const GLYPH_CACHE_DEFINITION*)data;
1995
71.0k
      return TRUE;
1996
0
    case FreeRDP_MonitorIds:
1997
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_NumMonitorIds);
1998
0
      if ((offset >= maxOffset) || !data)
1999
0
        goto fail;
2000
0
      settings->MonitorIds[offset] = *(const UINT32*)data;
2001
0
      return TRUE;
2002
0
    case FreeRDP_ChannelDefArray:
2003
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_ChannelDefArraySize);
2004
0
      if ((offset >= maxOffset) || !data)
2005
0
        goto fail;
2006
0
      settings->ChannelDefArray[offset] = *(const CHANNEL_DEF*)data;
2007
0
      return TRUE;
2008
0
    case FreeRDP_MonitorDefArray:
2009
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_MonitorDefArraySize);
2010
0
      if ((offset >= maxOffset) || !data)
2011
0
        goto fail;
2012
0
      settings->MonitorDefArray[offset] = *(const rdpMonitor*)data;
2013
0
      return TRUE;
2014
2015
0
    case FreeRDP_ClientTimeZone:
2016
0
      maxOffset = 1;
2017
0
      if ((offset >= maxOffset) || !data || !settings->ClientTimeZone)
2018
0
        goto fail;
2019
0
      settings->ClientTimeZone[offset] = *(const TIME_ZONE_INFORMATION*)data;
2020
0
      return TRUE;
2021
2022
0
    case FreeRDP_ServerLicenseProductIssuers:
2023
0
      maxOffset =
2024
0
          freerdp_settings_get_uint32(settings, FreeRDP_ServerLicenseProductIssuersCount);
2025
0
      if ((offset >= maxOffset) || !settings->ServerLicenseProductIssuers)
2026
0
        goto fail;
2027
0
      free(settings->ServerLicenseProductIssuers[offset]);
2028
0
      settings->ServerLicenseProductIssuers[offset] = nullptr;
2029
0
      if (data)
2030
0
        settings->ServerLicenseProductIssuers[offset] = _strdup((const char*)data);
2031
0
      return TRUE;
2032
2033
0
    case FreeRDP_ReceivedCapabilityData:
2034
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_ReceivedCapabilitiesSize);
2035
0
      if (offset >= maxOffset)
2036
0
        goto fail;
2037
0
      WINPR_ASSERT(settings->ReceivedCapabilityData);
2038
0
      settings->ReceivedCapabilityData[offset] = WINPR_CAST_CONST_PTR_AWAY(data, BYTE*);
2039
0
      return TRUE;
2040
0
    case FreeRDP_ReceivedCapabilityDataSizes:
2041
0
      maxOffset = freerdp_settings_get_uint32(settings, FreeRDP_ReceivedCapabilitiesSize);
2042
0
      if (offset >= maxOffset)
2043
0
        goto fail;
2044
0
      WINPR_ASSERT(settings->ReceivedCapabilityDataSizes);
2045
0
      settings->ReceivedCapabilityDataSizes[offset] = *(const uint32_t*)data;
2046
0
      return TRUE;
2047
0
    default:
2048
0
      WLog_WARN(TAG, "Invalid id %s [%d]", freerdp_settings_get_name_for_key(id), id);
2049
0
      return FALSE;
2050
1.14M
  }
2051
2052
0
fail:
2053
0
  WLog_WARN(TAG, "[%s] Invalid offset=%" PRIuz " [%" PRIuz "] or nullptr data=%p",
2054
0
            freerdp_settings_get_name_for_key(id), offset, maxOffset, data);
2055
0
  return FALSE;
2056
1.14M
}
2057
2058
const void* freerdp_settings_get_pointer_array(const rdpSettings* settings,
2059
                                               FreeRDP_Settings_Keys_Pointer id, size_t offset)
2060
13.7k
{
2061
13.7k
  return freerdp_settings_get_pointer_array_writable(settings, id, offset);
2062
13.7k
}
2063
2064
UINT32 freerdp_settings_get_codecs_flags(const rdpSettings* settings)
2065
0
{
2066
0
  UINT32 flags = FREERDP_CODEC_ALL;
2067
0
  if (settings->RemoteFxCodec == FALSE)
2068
0
  {
2069
0
    flags &= (uint32_t)~FREERDP_CODEC_REMOTEFX;
2070
0
  }
2071
0
  if (settings->NSCodec == FALSE)
2072
0
  {
2073
0
    flags &= (uint32_t)~FREERDP_CODEC_NSCODEC;
2074
0
  }
2075
  /*TODO: check other codecs flags */
2076
0
  return flags;
2077
0
}
2078
2079
const char* freerdp_settings_get_server_name(const rdpSettings* settings)
2080
0
{
2081
0
  WINPR_ASSERT(settings);
2082
0
  const char* hostname = settings->ServerHostname;
2083
2084
0
  if (settings->UserSpecifiedServerName)
2085
0
    hostname = settings->UserSpecifiedServerName;
2086
2087
0
  return hostname;
2088
0
}
2089
2090
#if defined(WITH_FREERDP_DEPRECATED)
2091
ADDIN_ARGV* freerdp_static_channel_clone(ADDIN_ARGV* channel)
2092
{
2093
  return freerdp_addin_argv_clone(channel);
2094
}
2095
2096
ADDIN_ARGV* freerdp_dynamic_channel_clone(ADDIN_ARGV* channel)
2097
{
2098
  return freerdp_addin_argv_clone(channel);
2099
}
2100
#endif
2101
2102
BOOL freerdp_target_net_addresses_copy(rdpSettings* settings, char** addresses, UINT32 count)
2103
0
{
2104
0
  WINPR_ASSERT(settings);
2105
0
  WINPR_ASSERT(addresses);
2106
2107
0
  if (!freerdp_target_net_adresses_reset(settings, count))
2108
0
    return FALSE;
2109
2110
0
  for (UINT32 i = 0; i < settings->TargetNetAddressCount; i++)
2111
0
  {
2112
0
    const char* address = addresses[i];
2113
0
    if (!freerdp_settings_set_pointer_array(settings, FreeRDP_TargetNetAddresses, i, address))
2114
0
    {
2115
0
      freerdp_target_net_addresses_free(settings);
2116
0
      return FALSE;
2117
0
    }
2118
0
  }
2119
2120
0
  return TRUE;
2121
0
}
2122
2123
BOOL freerdp_device_equal(const RDPDR_DEVICE* what, const RDPDR_DEVICE* other)
2124
0
{
2125
0
  if (!what && !other)
2126
0
    return TRUE;
2127
0
  if (!what || !other)
2128
0
    return FALSE;
2129
2130
0
  if (what->Id != other->Id)
2131
0
    return FALSE;
2132
0
  if (what->Type != other->Type)
2133
0
    return FALSE;
2134
0
  if (what->Name && other->Name)
2135
0
  {
2136
0
    if (strcmp(what->Name, other->Name) != 0)
2137
0
      return FALSE;
2138
0
  }
2139
0
  else
2140
0
  {
2141
0
    if (what->Name != other->Name)
2142
0
      return FALSE;
2143
0
  }
2144
2145
0
  switch (what->Type)
2146
0
  {
2147
0
    case RDPDR_DTYP_PRINT:
2148
0
    {
2149
0
      const RDPDR_PRINTER* a = (const RDPDR_PRINTER*)what;
2150
0
      const RDPDR_PRINTER* b = (const RDPDR_PRINTER*)other;
2151
0
      if (a->DriverName && b->DriverName)
2152
0
        return strcmp(a->DriverName, b->DriverName) == 0;
2153
0
      return a->DriverName == b->DriverName;
2154
0
    }
2155
2156
0
    case RDPDR_DTYP_SERIAL:
2157
0
    {
2158
0
      const RDPDR_SERIAL* a = (const RDPDR_SERIAL*)what;
2159
0
      const RDPDR_SERIAL* b = (const RDPDR_SERIAL*)other;
2160
2161
0
      if (a->Path && b->Path)
2162
0
      {
2163
0
        if (strcmp(a->Path, b->Path) != 0)
2164
0
          return FALSE;
2165
0
      }
2166
0
      else if (a->Path != b->Path)
2167
0
        return FALSE;
2168
2169
0
      if (a->Driver && b->Driver)
2170
0
      {
2171
0
        if (strcmp(a->Driver, b->Driver) != 0)
2172
0
          return FALSE;
2173
0
      }
2174
0
      else if (a->Driver != b->Driver)
2175
0
        return FALSE;
2176
0
      if (a->Permissive && b->Permissive)
2177
0
        return strcmp(a->Permissive, b->Permissive) == 0;
2178
0
      return a->Permissive == b->Permissive;
2179
0
    }
2180
2181
0
    case RDPDR_DTYP_PARALLEL:
2182
0
    {
2183
0
      const RDPDR_PARALLEL* a = (const RDPDR_PARALLEL*)what;
2184
0
      const RDPDR_PARALLEL* b = (const RDPDR_PARALLEL*)other;
2185
0
      if (a->Path && b->Path)
2186
0
        return strcmp(a->Path, b->Path) == 0;
2187
0
      return a->Path == b->Path;
2188
0
    }
2189
2190
0
    case RDPDR_DTYP_SMARTCARD:
2191
0
      break;
2192
0
    case RDPDR_DTYP_FILESYSTEM:
2193
0
    {
2194
0
      const RDPDR_DRIVE* a = (const RDPDR_DRIVE*)what;
2195
0
      const RDPDR_DRIVE* b = (const RDPDR_DRIVE*)other;
2196
0
      if (a->automount != b->automount)
2197
0
        return FALSE;
2198
0
      if (a->Path && b->Path)
2199
0
        return strcmp(a->Path, b->Path) == 0;
2200
0
      return a->Path == b->Path;
2201
0
    }
2202
2203
0
    default:
2204
0
      return FALSE;
2205
0
  }
2206
2207
0
  return TRUE;
2208
0
}
2209
2210
const char* freerdp_rail_support_flags_to_string(UINT32 flags, char* buffer, size_t length)
2211
0
{
2212
0
  const UINT32 mask =
2213
0
      RAIL_LEVEL_SUPPORTED | RAIL_LEVEL_DOCKED_LANGBAR_SUPPORTED |
2214
0
      RAIL_LEVEL_SHELL_INTEGRATION_SUPPORTED | RAIL_LEVEL_LANGUAGE_IME_SYNC_SUPPORTED |
2215
0
      RAIL_LEVEL_SERVER_TO_CLIENT_IME_SYNC_SUPPORTED | RAIL_LEVEL_HIDE_MINIMIZED_APPS_SUPPORTED |
2216
0
      RAIL_LEVEL_WINDOW_CLOAKING_SUPPORTED | RAIL_LEVEL_HANDSHAKE_EX_SUPPORTED;
2217
2218
0
  if (flags & RAIL_LEVEL_SUPPORTED)
2219
0
    winpr_str_append("RAIL_LEVEL_SUPPORTED", buffer, length, "|");
2220
0
  if (flags & RAIL_LEVEL_DOCKED_LANGBAR_SUPPORTED)
2221
0
    winpr_str_append("RAIL_LEVEL_DOCKED_LANGBAR_SUPPORTED", buffer, length, "|");
2222
0
  if (flags & RAIL_LEVEL_SHELL_INTEGRATION_SUPPORTED)
2223
0
    winpr_str_append("RAIL_LEVEL_SHELL_INTEGRATION_SUPPORTED", buffer, length, "|");
2224
0
  if (flags & RAIL_LEVEL_LANGUAGE_IME_SYNC_SUPPORTED)
2225
0
    winpr_str_append("RAIL_LEVEL_LANGUAGE_IME_SYNC_SUPPORTED", buffer, length, "|");
2226
0
  if (flags & RAIL_LEVEL_SERVER_TO_CLIENT_IME_SYNC_SUPPORTED)
2227
0
    winpr_str_append("RAIL_LEVEL_SERVER_TO_CLIENT_IME_SYNC_SUPPORTED", buffer, length, "|");
2228
0
  if (flags & RAIL_LEVEL_HIDE_MINIMIZED_APPS_SUPPORTED)
2229
0
    winpr_str_append("RAIL_LEVEL_HIDE_MINIMIZED_APPS_SUPPORTED", buffer, length, "|");
2230
0
  if (flags & RAIL_LEVEL_WINDOW_CLOAKING_SUPPORTED)
2231
0
    winpr_str_append("RAIL_LEVEL_WINDOW_CLOAKING_SUPPORTED", buffer, length, "|");
2232
0
  if (flags & RAIL_LEVEL_HANDSHAKE_EX_SUPPORTED)
2233
0
    winpr_str_append("RAIL_LEVEL_HANDSHAKE_EX_SUPPORTED", buffer, length, "|");
2234
0
  if (flags & RAIL_LEVEL_LANGUAGE_IME_SYNC_SUPPORTED)
2235
0
    winpr_str_append("RAIL_LEVEL_LANGUAGE_IME_SYNC_SUPPORTED", buffer, length, "|");
2236
0
  if ((flags & ~mask) != 0)
2237
0
  {
2238
0
    char tbuffer[64] = WINPR_C_ARRAY_INIT;
2239
0
    (void)_snprintf(tbuffer, sizeof(tbuffer), "RAIL_FLAG_UNKNOWN 0x%08" PRIx32, flags & mask);
2240
0
    winpr_str_append(tbuffer, buffer, length, "|");
2241
0
  }
2242
0
  return buffer;
2243
0
}
2244
2245
BOOL freerdp_settings_update_from_caps(rdpSettings* settings, const BYTE* capsFlags,
2246
                                       const BYTE** capsData, const UINT32* capsSizes,
2247
                                       UINT32 capsCount, BOOL serverReceivedCaps)
2248
0
{
2249
0
  WINPR_ASSERT(settings);
2250
0
  WINPR_ASSERT(capsFlags || (capsCount == 0));
2251
0
  WINPR_ASSERT(capsData || (capsCount == 0));
2252
0
  WINPR_ASSERT(capsSizes || (capsCount == 0));
2253
0
  WINPR_ASSERT(capsCount <= UINT16_MAX);
2254
2255
0
  wLog* log = WLog_Get(TAG);
2256
2257
0
  for (UINT32 x = 0; x < capsCount; x++)
2258
0
  {
2259
0
    if (capsFlags[x])
2260
0
    {
2261
0
      wStream buffer = WINPR_C_ARRAY_INIT;
2262
0
      wStream* sub = Stream_StaticConstInit(&buffer, capsData[x], capsSizes[x]);
2263
2264
0
      if (!rdp_read_capability_set(log, sub, (UINT16)x, settings, serverReceivedCaps))
2265
0
        return FALSE;
2266
0
    }
2267
0
  }
2268
2269
0
  return TRUE;
2270
0
}
2271
2272
const char* freerdp_rdp_version_string(UINT32 version)
2273
0
{
2274
0
  switch (version)
2275
0
  {
2276
0
    case RDP_VERSION_4:
2277
0
      return "RDP_VERSION_4";
2278
0
    case RDP_VERSION_5_PLUS:
2279
0
      return "RDP_VERSION_5_PLUS";
2280
0
    case RDP_VERSION_10_0:
2281
0
      return "RDP_VERSION_10_0";
2282
0
    case RDP_VERSION_10_1:
2283
0
      return "RDP_VERSION_10_1";
2284
0
    case RDP_VERSION_10_2:
2285
0
      return "RDP_VERSION_10_2";
2286
0
    case RDP_VERSION_10_3:
2287
0
      return "RDP_VERSION_10_3";
2288
0
    case RDP_VERSION_10_4:
2289
0
      return "RDP_VERSION_10_4";
2290
0
    case RDP_VERSION_10_5:
2291
0
      return "RDP_VERSION_10_5";
2292
0
    case RDP_VERSION_10_6:
2293
0
      return "RDP_VERSION_10_6";
2294
0
    case RDP_VERSION_10_7:
2295
0
      return "RDP_VERSION_10_7";
2296
0
    case RDP_VERSION_10_8:
2297
0
      return "RDP_VERSION_10_8";
2298
0
    case RDP_VERSION_10_9:
2299
0
      return "RDP_VERSION_10_9";
2300
0
    case RDP_VERSION_10_10:
2301
0
      return "RDP_VERSION_10_10";
2302
0
    case RDP_VERSION_10_11:
2303
0
      return "RDP_VERSION_10_11";
2304
0
    case RDP_VERSION_10_12:
2305
0
      return "RDP_VERSION_10_12";
2306
0
    default:
2307
0
      return "RDP_VERSION_UNKNOWN";
2308
0
  }
2309
0
}
2310
2311
BOOL freerdp_settings_set_string_from_utf16(rdpSettings* settings, FreeRDP_Settings_Keys_String id,
2312
                                            const WCHAR* param)
2313
0
{
2314
0
  WINPR_ASSERT(settings);
2315
2316
0
  if (!param)
2317
0
    return freerdp_settings_set_string_copy_(settings, id, nullptr, 0, TRUE);
2318
2319
0
  size_t len = 0;
2320
2321
0
  char* str = ConvertWCharToUtf8Alloc(param, &len);
2322
0
  if (!str && (len != 0))
2323
0
    return FALSE;
2324
2325
0
  return freerdp_settings_set_string_(settings, id, str, len);
2326
0
}
2327
2328
BOOL freerdp_settings_set_string_from_utf16N(rdpSettings* settings, FreeRDP_Settings_Keys_String id,
2329
                                             const WCHAR* param, size_t length)
2330
53.5k
{
2331
53.5k
  size_t len = 0;
2332
2333
53.5k
  WINPR_ASSERT(settings);
2334
2335
53.5k
  if (!param)
2336
0
    return freerdp_settings_set_string_copy_(settings, id, nullptr, length, TRUE);
2337
2338
53.5k
  char* str = ConvertWCharNToUtf8Alloc(param, length, &len);
2339
53.5k
  if (!str && (length != 0))
2340
40
  {
2341
    /* If the input string is an empty string, but length > 0
2342
     * consider the conversion a success */
2343
40
    const size_t wlen = _wcsnlen(param, length);
2344
40
    if (wlen != 0)
2345
40
      return FALSE;
2346
40
  }
2347
2348
53.5k
  return freerdp_settings_set_string_(settings, id, str, len);
2349
53.5k
}
2350
2351
WCHAR* freerdp_settings_get_string_as_utf16(const rdpSettings* settings,
2352
                                            FreeRDP_Settings_Keys_String id, size_t* pCharLen)
2353
0
{
2354
0
  const char* str = freerdp_settings_get_string(settings, id);
2355
0
  if (pCharLen)
2356
0
    *pCharLen = 0;
2357
0
  if (!str)
2358
0
    return nullptr;
2359
0
  return ConvertUtf8ToWCharAlloc(str, pCharLen);
2360
0
}
2361
2362
const char* freerdp_rdpdr_dtyp_string(UINT32 type)
2363
0
{
2364
0
  switch (type)
2365
0
  {
2366
0
    case RDPDR_DTYP_FILESYSTEM:
2367
0
      return "RDPDR_DTYP_FILESYSTEM";
2368
0
    case RDPDR_DTYP_PARALLEL:
2369
0
      return "RDPDR_DTYP_PARALLEL";
2370
0
    case RDPDR_DTYP_PRINT:
2371
0
      return "RDPDR_DTYP_PRINT";
2372
0
    case RDPDR_DTYP_SERIAL:
2373
0
      return "RDPDR_DTYP_SERIAL";
2374
0
    case RDPDR_DTYP_SMARTCARD:
2375
0
      return "RDPDR_DTYP_SMARTCARD";
2376
0
    default:
2377
0
      return "RDPDR_DTYP_UNKNOWN";
2378
0
  }
2379
0
}
2380
2381
const char* freerdp_encryption_level_string(UINT32 EncryptionLevel)
2382
0
{
2383
0
  switch (EncryptionLevel)
2384
0
  {
2385
0
    case ENCRYPTION_LEVEL_NONE:
2386
0
      return "ENCRYPTION_LEVEL_NONE";
2387
0
    case ENCRYPTION_LEVEL_LOW:
2388
0
      return "ENCRYPTION_LEVEL_LOW";
2389
0
    case ENCRYPTION_LEVEL_CLIENT_COMPATIBLE:
2390
0
      return "ENCRYPTION_LEVEL_CLIENT_COMPATIBLE";
2391
0
    case ENCRYPTION_LEVEL_HIGH:
2392
0
      return "ENCRYPTION_LEVEL_HIGH";
2393
0
    case ENCRYPTION_LEVEL_FIPS:
2394
0
      return "ENCRYPTION_LEVEL_FIPS";
2395
0
    default:
2396
0
      return "ENCRYPTION_LEVEL_UNKNOWN";
2397
0
  }
2398
0
}
2399
2400
const char* freerdp_encryption_methods_string(UINT32 EncryptionMethods, char* buffer, size_t size)
2401
0
{
2402
0
  if (EncryptionMethods == ENCRYPTION_METHOD_NONE)
2403
0
  {
2404
0
    winpr_str_append("ENCRYPTION_METHOD_NONE", buffer, size, "|");
2405
0
    return buffer;
2406
0
  }
2407
2408
0
  if (EncryptionMethods & ENCRYPTION_METHOD_40BIT)
2409
0
  {
2410
0
    winpr_str_append("ENCRYPTION_METHOD_40BIT", buffer, size, "|");
2411
0
  }
2412
0
  if (EncryptionMethods & ENCRYPTION_METHOD_128BIT)
2413
0
  {
2414
0
    winpr_str_append("ENCRYPTION_METHOD_128BIT", buffer, size, "|");
2415
0
  }
2416
0
  if (EncryptionMethods & ENCRYPTION_METHOD_56BIT)
2417
0
  {
2418
0
    winpr_str_append("ENCRYPTION_METHOD_56BIT", buffer, size, "|");
2419
0
  }
2420
0
  if (EncryptionMethods & ENCRYPTION_METHOD_FIPS)
2421
0
  {
2422
0
    winpr_str_append("ENCRYPTION_METHOD_FIPS", buffer, size, "|");
2423
0
  }
2424
2425
0
  return buffer;
2426
0
}
2427
2428
const char* freerdp_supported_color_depths_string(UINT16 mask, char* buffer, size_t size)
2429
0
{
2430
0
  const UINT32 invalid = mask & ~(RNS_UD_32BPP_SUPPORT | RNS_UD_24BPP_SUPPORT |
2431
0
                                  RNS_UD_16BPP_SUPPORT | RNS_UD_15BPP_SUPPORT);
2432
2433
0
  if (mask & RNS_UD_32BPP_SUPPORT)
2434
0
    winpr_str_append("RNS_UD_32BPP_SUPPORT", buffer, size, "|");
2435
0
  if (mask & RNS_UD_24BPP_SUPPORT)
2436
0
    winpr_str_append("RNS_UD_24BPP_SUPPORT", buffer, size, "|");
2437
0
  if (mask & RNS_UD_16BPP_SUPPORT)
2438
0
    winpr_str_append("RNS_UD_16BPP_SUPPORT", buffer, size, "|");
2439
0
  if (mask & RNS_UD_15BPP_SUPPORT)
2440
0
    winpr_str_append("RNS_UD_15BPP_SUPPORT", buffer, size, "|");
2441
2442
0
  if (invalid != 0)
2443
0
  {
2444
0
    char str[32] = WINPR_C_ARRAY_INIT;
2445
0
    (void)_snprintf(str, sizeof(str), "RNS_UD_INVALID[0x%04" PRIx32 "]", invalid);
2446
0
    winpr_str_append(str, buffer, size, "|");
2447
0
  }
2448
0
  char hex[32] = WINPR_C_ARRAY_INIT;
2449
0
  (void)_snprintf(hex, sizeof(hex), "[0x%04" PRIx16 "]", mask);
2450
0
  return buffer;
2451
0
}
2452
2453
BOOL freerdp_settings_append_string(rdpSettings* settings, FreeRDP_Settings_Keys_String id,
2454
                                    const char* separator, const char* param)
2455
0
{
2456
0
  const char* old = freerdp_settings_get_string(settings, id);
2457
2458
0
  size_t len = 0;
2459
0
  char* str = nullptr;
2460
2461
0
  if (!old)
2462
0
    winpr_asprintf(&str, &len, "%s", param);
2463
0
  else if (!separator)
2464
0
    winpr_asprintf(&str, &len, "%s%s", old, param);
2465
0
  else
2466
0
    winpr_asprintf(&str, &len, "%s%s%s", old, separator, param);
2467
2468
0
  const BOOL rc = freerdp_settings_set_string_len(settings, id, str, len);
2469
0
  free(str);
2470
0
  return rc;
2471
0
}
2472
2473
BOOL freerdp_settings_are_valid(const rdpSettings* settings)
2474
0
{
2475
0
  return settings != nullptr;
2476
0
}
2477
2478
/* Function to sort rdpMonitor arrays:
2479
 * 1. first element is primary monitor
2480
 * 2. all others are sorted by coordinates of x/y
2481
 */
2482
static int sort_monitor_fn(const void* pva, const void* pvb)
2483
0
{
2484
0
  const rdpMonitor* a = pva;
2485
0
  const rdpMonitor* b = pvb;
2486
0
  WINPR_ASSERT(a);
2487
0
  WINPR_ASSERT(b);
2488
0
  if (a->is_primary && b->is_primary)
2489
0
    return 0;
2490
0
  if (a->is_primary)
2491
0
    return -1;
2492
0
  if (b->is_primary)
2493
0
    return 1;
2494
2495
0
  if (a->x != b->x)
2496
0
    return a->x - b->x;
2497
0
  if (a->y != b->y)
2498
0
    return a->y - b->y;
2499
0
  return 0;
2500
0
}
2501
2502
BOOL freerdp_settings_set_monitor_def_array_sorted(rdpSettings* settings,
2503
                                                   const rdpMonitor* monitors, size_t count)
2504
0
{
2505
0
  WINPR_ASSERT(monitors || (count == 0));
2506
0
  if (count == 0)
2507
0
  {
2508
0
    if (!freerdp_settings_set_int32(settings, FreeRDP_MonitorLocalShiftX, 0))
2509
0
      return FALSE;
2510
0
    if (!freerdp_settings_set_int32(settings, FreeRDP_MonitorLocalShiftY, 0))
2511
0
      return FALSE;
2512
0
    if (!freerdp_settings_set_pointer_len(settings, FreeRDP_MonitorDefArray, nullptr, 0))
2513
0
      return FALSE;
2514
0
    return freerdp_settings_set_uint32(settings, FreeRDP_MonitorCount, 0);
2515
0
  }
2516
2517
  // Find primary or alternatively the monitor at 0/0
2518
0
  const rdpMonitor* primary = nullptr;
2519
0
  for (size_t x = 0; x < count; x++)
2520
0
  {
2521
0
    const rdpMonitor* cur = &monitors[x];
2522
0
    if (cur->is_primary)
2523
0
    {
2524
0
      primary = cur;
2525
0
      break;
2526
0
    }
2527
0
  }
2528
0
  if (!primary)
2529
0
  {
2530
0
    for (size_t x = 0; x < count; x++)
2531
0
    {
2532
0
      const rdpMonitor* cur = &monitors[x];
2533
0
      if ((cur->x == 0) && (cur->y == 0))
2534
0
      {
2535
0
        primary = cur;
2536
0
        break;
2537
0
      }
2538
0
    }
2539
0
  }
2540
2541
0
  if (!primary)
2542
0
  {
2543
0
    WLog_ERR(TAG, "Could not find primary monitor, aborting");
2544
0
    return FALSE;
2545
0
  }
2546
2547
0
  if (!freerdp_settings_set_pointer_len(settings, FreeRDP_MonitorDefArray, nullptr, count))
2548
0
    return FALSE;
2549
0
  rdpMonitor* sorted = freerdp_settings_get_pointer_writable(settings, FreeRDP_MonitorDefArray);
2550
0
  WINPR_ASSERT(sorted);
2551
2552
0
  size_t sortpos = 0;
2553
2554
  /* Set primary. Ensure left/top is at 0/0 and flags contains MONITOR_PRIMARY */
2555
0
  sorted[sortpos] = *primary;
2556
0
  sorted[sortpos].x = 0;
2557
0
  sorted[sortpos].y = 0;
2558
0
  sorted[sortpos].is_primary = TRUE;
2559
0
  sortpos++;
2560
2561
  /* Set monitor shift to original layout */
2562
0
  const INT32 offsetX = primary->x;
2563
0
  const INT32 offsetY = primary->y;
2564
0
  if (!freerdp_settings_set_int32(settings, FreeRDP_MonitorLocalShiftX, offsetX))
2565
0
    return FALSE;
2566
0
  if (!freerdp_settings_set_int32(settings, FreeRDP_MonitorLocalShiftY, offsetY))
2567
0
    return FALSE;
2568
2569
0
  for (size_t x = 0; x < count; x++)
2570
0
  {
2571
0
    const rdpMonitor* cur = &monitors[x];
2572
0
    if (cur == primary)
2573
0
      continue;
2574
2575
0
    rdpMonitor m = monitors[x];
2576
0
    m.x -= offsetX;
2577
0
    m.y -= offsetY;
2578
0
    sorted[sortpos++] = m;
2579
0
  }
2580
2581
  // Sort remaining monitors by x/y ?
2582
0
  qsort(sorted, count, sizeof(rdpMonitor), sort_monitor_fn);
2583
2584
0
  return freerdp_settings_set_uint32(settings, FreeRDP_MonitorCount,
2585
0
                                     WINPR_ASSERTING_INT_CAST(uint32_t, count));
2586
0
}
2587
2588
static BOOL fill_array(WINPR_JSON* array, const void* data, size_t length)
2589
0
{
2590
0
  const BYTE* pdata = data;
2591
0
  for (size_t x = 0; x < length; x++)
2592
0
  {
2593
0
    BYTE val = pdata[x];
2594
0
    if (!WINPR_JSON_AddItemToArray(array, WINPR_JSON_CreateNumber(val)))
2595
0
      return FALSE;
2596
0
  }
2597
0
  return TRUE;
2598
0
}
2599
2600
static BOOL fill_uint32_array(WINPR_JSON* array, const uint32_t* data, size_t length)
2601
0
{
2602
0
  for (size_t x = 0; x < length; x++)
2603
0
  {
2604
0
    uint32_t val = data[x];
2605
0
    if (!WINPR_JSON_AddItemToArray(array, WINPR_JSON_CreateNumber(val)))
2606
0
      return FALSE;
2607
0
  }
2608
0
  return TRUE;
2609
0
}
2610
2611
static WINPR_JSON* json_from_addin_item(const ADDIN_ARGV* val)
2612
0
{
2613
0
  WINPR_JSON* obj = WINPR_JSON_CreateObject();
2614
0
  if (!obj)
2615
0
    goto fail;
2616
2617
0
  if (val)
2618
0
  {
2619
0
    if (!WINPR_JSON_AddNumberToObject(obj, "argc", val->argc))
2620
0
      goto fail;
2621
2622
0
    WINPR_JSON* array = WINPR_JSON_AddArrayToObject(obj, "argv");
2623
0
    if (!array)
2624
0
      goto fail;
2625
0
    for (int x = 0; x < val->argc; x++)
2626
0
    {
2627
0
      const char* str = val->argv[x];
2628
2629
0
      WINPR_JSON* item = nullptr;
2630
0
      if (!str)
2631
0
      {
2632
0
        item = WINPR_JSON_CreateNull();
2633
0
      }
2634
0
      else
2635
0
      {
2636
0
        item = WINPR_JSON_CreateString(str);
2637
0
      }
2638
2639
0
      if (!WINPR_JSON_AddItemToArray(array, item))
2640
0
        goto fail;
2641
0
    }
2642
0
  }
2643
0
  return obj;
2644
2645
0
fail:
2646
0
  WINPR_JSON_Delete(obj);
2647
0
  return nullptr;
2648
0
}
2649
2650
static BOOL json_from_addin_item_array(WINPR_JSON* json, const rdpSettings* settings,
2651
                                       FreeRDP_Settings_Keys_Pointer key, size_t count)
2652
0
{
2653
0
  if (!json)
2654
0
    return FALSE;
2655
2656
0
  for (uint32_t x = 0; x < count; x++)
2657
0
  {
2658
0
    const ADDIN_ARGV* cval = freerdp_settings_get_pointer_array(settings, key, x);
2659
0
    if (!WINPR_JSON_AddItemToArray(json, json_from_addin_item(cval)))
2660
0
      return FALSE;
2661
0
  }
2662
0
  return TRUE;
2663
0
}
2664
2665
static BOOL add_string_or_null(WINPR_JSON* json, const char* key, const char* value)
2666
0
{
2667
0
  if (value)
2668
0
    return WINPR_JSON_AddStringToObject(json, key, value) != nullptr;
2669
2670
0
  return WINPR_JSON_AddNullToObject(json, key) != nullptr;
2671
0
}
2672
2673
static WINPR_JSON* json_from_device_item(const RDPDR_DEVICE* val)
2674
0
{
2675
0
  WINPR_JSON* obj = WINPR_JSON_CreateObject();
2676
0
  if (!obj)
2677
0
    goto fail;
2678
2679
0
  if (val)
2680
0
  {
2681
0
    union
2682
0
    {
2683
0
      const RDPDR_DEVICE* base;
2684
0
      const RDPDR_PARALLEL* parallel;
2685
0
      const RDPDR_SERIAL* serial;
2686
0
      const RDPDR_SMARTCARD* smartcard;
2687
0
      const RDPDR_PRINTER* printer;
2688
0
      const RDPDR_DRIVE* drive;
2689
0
      const RDPDR_DEVICE* device;
2690
0
    } device;
2691
2692
0
    device.base = val;
2693
2694
0
    if (!WINPR_JSON_AddNumberToObject(obj, "Id", val->Id))
2695
0
      goto fail;
2696
0
    if (!WINPR_JSON_AddNumberToObject(obj, "Type", val->Type))
2697
0
      goto fail;
2698
0
    if (!add_string_or_null(obj, "Name", val->Name))
2699
0
      goto fail;
2700
0
    switch (val->Type)
2701
0
    {
2702
0
      case RDPDR_DTYP_SERIAL:
2703
0
        if (!add_string_or_null(obj, "Path", device.serial->Path))
2704
0
          goto fail;
2705
0
        if (!add_string_or_null(obj, "Driver", device.serial->Driver))
2706
0
          goto fail;
2707
0
        if (!add_string_or_null(obj, "Permissive", device.serial->Permissive))
2708
0
          goto fail;
2709
0
        break;
2710
0
      case RDPDR_DTYP_PARALLEL:
2711
0
        if (!add_string_or_null(obj, "Path", device.parallel->Path))
2712
0
          goto fail;
2713
0
        break;
2714
0
      case RDPDR_DTYP_PRINT:
2715
0
        if (!add_string_or_null(obj, "DriverName", device.printer->DriverName))
2716
0
          goto fail;
2717
0
        if (!WINPR_JSON_AddBoolToObject(obj, "IsDefault", device.printer->IsDefault))
2718
0
          goto fail;
2719
0
        break;
2720
0
      case RDPDR_DTYP_FILESYSTEM:
2721
0
        if (!add_string_or_null(obj, "Path", device.drive->Path))
2722
0
          goto fail;
2723
0
        if (!WINPR_JSON_AddBoolToObject(obj, "IsDefault", device.drive->automount))
2724
0
          goto fail;
2725
0
        break;
2726
0
      case RDPDR_DTYP_SMARTCARD:
2727
0
      default:
2728
0
        break;
2729
0
    }
2730
0
  }
2731
0
  return obj;
2732
2733
0
fail:
2734
0
  WINPR_JSON_Delete(obj);
2735
0
  return nullptr;
2736
0
}
2737
2738
static BOOL json_from_device_item_array(WINPR_JSON* json, const rdpSettings* settings,
2739
                                        FreeRDP_Settings_Keys_Pointer key, size_t count)
2740
0
{
2741
0
  if (!json)
2742
0
    return FALSE;
2743
2744
0
  for (uint32_t x = 0; x < count; x++)
2745
0
  {
2746
0
    const RDPDR_DEVICE* cval = freerdp_settings_get_pointer_array(settings, key, x);
2747
0
    if (!WINPR_JSON_AddItemToArray(json, json_from_device_item(cval)))
2748
0
      return FALSE;
2749
0
  }
2750
0
  return TRUE;
2751
0
}
2752
2753
static BOOL string_array_to_json(WINPR_JSON* json, const rdpSettings* settings, uint32_t argc,
2754
                                 FreeRDP_Settings_Keys_Pointer key)
2755
0
{
2756
0
  for (uint32_t x = 0; x < argc; x++)
2757
0
  {
2758
0
    const char* cval = freerdp_settings_get_pointer_array(settings, key, x);
2759
2760
0
    WINPR_JSON* item = nullptr;
2761
0
    if (!cval)
2762
0
      item = WINPR_JSON_CreateNull();
2763
0
    else
2764
0
      item = WINPR_JSON_CreateString(cval);
2765
0
    if (!WINPR_JSON_AddItemToArray(json, item))
2766
0
      return FALSE;
2767
0
  }
2768
0
  return TRUE;
2769
0
}
2770
2771
static BOOL wchar_to_json(WINPR_JSON* obj, const char* key, const WCHAR* wstr, size_t len)
2772
0
{
2773
0
  if (len == 0)
2774
0
    return WINPR_JSON_AddStringToObject(obj, key, "") != nullptr;
2775
2776
0
  const size_t slen = len * 6;
2777
0
  char* str = calloc(1, slen);
2778
0
  if (!str)
2779
0
    return FALSE;
2780
2781
0
  WINPR_JSON* jstr = nullptr;
2782
0
  SSIZE_T rc = ConvertWCharNToUtf8(wstr, len, str, slen);
2783
0
  if (rc >= 0)
2784
0
    jstr = WINPR_JSON_AddStringToObject(obj, key, str);
2785
0
  free(str);
2786
0
  return jstr != nullptr;
2787
0
}
2788
2789
static BOOL wchar_from_json(WCHAR* wstr, size_t len, const WINPR_JSON* obj, const char* key)
2790
0
{
2791
0
  if (!obj || !WINPR_JSON_IsObject(obj))
2792
0
    return FALSE;
2793
0
  WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(obj, key);
2794
0
  if (!item || !WINPR_JSON_IsString(item))
2795
0
    return FALSE;
2796
2797
0
  const char* str = WINPR_JSON_GetStringValue(item);
2798
0
  if (!str)
2799
0
  {
2800
0
    memset(wstr, 0, sizeof(WCHAR) * len);
2801
0
    return TRUE;
2802
0
  }
2803
2804
0
  SSIZE_T rc = ConvertUtf8ToWChar(str, wstr, len);
2805
0
  return rc >= 0;
2806
0
}
2807
2808
static int64_t int_from_json_item(const WINPR_JSON* item, int64_t min, int64_t max)
2809
0
{
2810
0
  if (!item || !WINPR_JSON_IsNumber(item))
2811
0
  {
2812
0
    errno = EINVAL;
2813
0
    return 0;
2814
0
  }
2815
2816
0
  const double val = WINPR_JSON_GetNumberValue(item);
2817
0
  if (isinf(val) || isnan(val))
2818
0
  {
2819
0
    errno = ERANGE;
2820
0
    return 0;
2821
0
  }
2822
2823
0
  const int64_t ival = (int64_t)val;
2824
0
  if ((ival < min) || (ival > max))
2825
0
  {
2826
0
    errno = ERANGE;
2827
0
    return 0;
2828
0
  }
2829
2830
0
  return ival;
2831
0
}
2832
2833
static int64_t int_from_json(const WINPR_JSON* obj, const char* key, int64_t min, int64_t max)
2834
0
{
2835
0
  if (!obj || !WINPR_JSON_IsObject(obj))
2836
0
  {
2837
0
    errno = EINVAL;
2838
0
    return 0;
2839
0
  }
2840
0
  WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(obj, key);
2841
0
  return int_from_json_item(item, min, max);
2842
0
}
2843
2844
static uint64_t uint_from_json_item(const WINPR_JSON* item, uint64_t max)
2845
0
{
2846
0
  if (!item || !WINPR_JSON_IsNumber(item))
2847
0
  {
2848
0
    errno = EINVAL;
2849
0
    return 0;
2850
0
  }
2851
2852
0
  const double val = WINPR_JSON_GetNumberValue(item);
2853
0
  if (isinf(val) || isnan(val) || (val < 0.0))
2854
0
  {
2855
0
    errno = ERANGE;
2856
0
    return 0;
2857
0
  }
2858
2859
0
  const uint64_t uval = (uint64_t)val;
2860
0
  if (uval > max)
2861
0
  {
2862
0
    errno = ERANGE;
2863
0
    return 0;
2864
0
  }
2865
0
  return uval;
2866
0
}
2867
2868
static uint64_t uint_from_json(const WINPR_JSON* obj, const char* key, uint64_t max)
2869
0
{
2870
0
  if (!obj || !WINPR_JSON_IsObject(obj))
2871
0
  {
2872
0
    errno = EINVAL;
2873
0
    return 0;
2874
0
  }
2875
2876
0
  WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(obj, key);
2877
0
  return uint_from_json_item(item, max);
2878
0
}
2879
2880
static WINPR_JSON* systemtime_to_json(WINPR_JSON* parent, const char* key, const SYSTEMTIME* st)
2881
0
{
2882
0
  WINPR_ASSERT(st);
2883
2884
0
  WINPR_JSON* obj = WINPR_JSON_AddObjectToObject(parent, key);
2885
0
  if (!obj)
2886
0
    return nullptr;
2887
2888
0
  if (!WINPR_JSON_AddNumberToObject(obj, "wYear", st->wYear))
2889
0
    goto fail;
2890
0
  if (!WINPR_JSON_AddNumberToObject(obj, "wMonth", st->wMonth))
2891
0
    goto fail;
2892
0
  if (!WINPR_JSON_AddNumberToObject(obj, "wDayOfWeek", st->wDayOfWeek))
2893
0
    goto fail;
2894
0
  if (!WINPR_JSON_AddNumberToObject(obj, "wDay", st->wDay))
2895
0
    goto fail;
2896
0
  if (!WINPR_JSON_AddNumberToObject(obj, "wHour", st->wHour))
2897
0
    goto fail;
2898
0
  if (!WINPR_JSON_AddNumberToObject(obj, "wMinute", st->wMinute))
2899
0
    goto fail;
2900
0
  if (!WINPR_JSON_AddNumberToObject(obj, "wSecond", st->wSecond))
2901
0
    goto fail;
2902
0
  if (!WINPR_JSON_AddNumberToObject(obj, "wMilliseconds", st->wMilliseconds))
2903
0
    goto fail;
2904
2905
0
  return obj;
2906
0
fail:
2907
0
  WINPR_JSON_Delete(obj);
2908
0
  return nullptr;
2909
0
}
2910
2911
static BOOL systemtime_from_json(const WINPR_JSON* pobj, const char* key, SYSTEMTIME* st)
2912
0
{
2913
0
  WINPR_ASSERT(st);
2914
2915
0
  if (!pobj || !WINPR_JSON_IsObject(pobj))
2916
0
    return FALSE;
2917
2918
0
  WINPR_JSON* obj = WINPR_JSON_GetObjectItemCaseSensitive(pobj, key);
2919
0
  if (!obj || !WINPR_JSON_IsObject(obj))
2920
0
    return FALSE;
2921
2922
0
  errno = 0;
2923
0
  st->wYear = (uint16_t)uint_from_json(obj, "wYear", UINT16_MAX);
2924
0
  st->wMonth = (uint16_t)uint_from_json(obj, "wMonth", UINT16_MAX);
2925
0
  st->wDayOfWeek = (uint16_t)uint_from_json(obj, "wDayOfWeek", UINT16_MAX);
2926
0
  st->wDay = (uint16_t)uint_from_json(obj, "wDay", UINT16_MAX);
2927
0
  st->wHour = (uint16_t)uint_from_json(obj, "wHour", UINT16_MAX);
2928
0
  st->wMinute = (uint16_t)uint_from_json(obj, "wMinute", UINT16_MAX);
2929
0
  st->wSecond = (uint16_t)uint_from_json(obj, "wSecond", UINT16_MAX);
2930
0
  st->wMilliseconds = (uint16_t)uint_from_json(obj, "wMilliseconds", UINT16_MAX);
2931
0
  return errno == 0;
2932
0
}
2933
2934
static BOOL ts_info_from_json(TIME_ZONE_INFORMATION* tz, const WINPR_JSON* json)
2935
0
{
2936
0
  WINPR_ASSERT(tz);
2937
2938
0
  if (!json || !WINPR_JSON_IsObject(json))
2939
0
    return FALSE;
2940
2941
0
  errno = 0;
2942
0
  tz->Bias = (int32_t)int_from_json(json, "Bias", INT32_MIN, INT32_MAX);
2943
0
  tz->StandardBias = (int32_t)int_from_json(json, "StandardBias", INT32_MIN, INT32_MAX);
2944
0
  tz->DaylightBias = (int32_t)int_from_json(json, "DaylightBias", INT32_MIN, INT32_MAX);
2945
0
  if (errno != 0)
2946
0
    return FALSE;
2947
2948
0
  if (!systemtime_from_json(json, "StandardDate", &tz->StandardDate))
2949
0
    return FALSE;
2950
0
  if (!systemtime_from_json(json, "DaylightDate", &tz->DaylightDate))
2951
0
    return FALSE;
2952
2953
0
  if (!wchar_from_json(tz->StandardName, ARRAYSIZE(tz->StandardName), json, "StandardName"))
2954
0
    return FALSE;
2955
0
  if (!wchar_from_json(tz->DaylightName, ARRAYSIZE(tz->DaylightName), json, "DaylightName"))
2956
0
    return FALSE;
2957
2958
0
  return TRUE;
2959
0
}
2960
2961
static BOOL ts_info_array_from_json(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer key,
2962
                                    const WINPR_JSON* json)
2963
0
{
2964
0
  if (!json || !WINPR_JSON_IsArray(json))
2965
0
    return FALSE;
2966
2967
0
  const size_t count = WINPR_JSON_GetArraySize(json);
2968
0
  if (!freerdp_settings_set_pointer_len(settings, key, nullptr, count))
2969
0
    return FALSE;
2970
2971
0
  for (size_t x = 0; x < count; x++)
2972
0
  {
2973
0
    TIME_ZONE_INFORMATION* tz = freerdp_settings_get_pointer_array_writable(settings, key, x);
2974
0
    WINPR_JSON* obj = WINPR_JSON_GetArrayItem(json, x);
2975
0
    if (!ts_info_from_json(tz, obj))
2976
0
      return FALSE;
2977
0
  }
2978
0
  return TRUE;
2979
0
}
2980
2981
static BOOL tz_info_to_json(WINPR_JSON* json, const TIME_ZONE_INFORMATION* ptz)
2982
0
{
2983
0
  WINPR_JSON* obj = WINPR_JSON_CreateObject();
2984
0
  if (!obj)
2985
0
    return FALSE;
2986
0
  if (!WINPR_JSON_AddItemToArray(json, obj))
2987
0
    return FALSE;
2988
2989
0
  TIME_ZONE_INFORMATION tz = WINPR_C_ARRAY_INIT;
2990
0
  if (ptz)
2991
0
    tz = *ptz;
2992
2993
0
  if (!WINPR_JSON_AddNumberToObject(obj, "Bias", tz.Bias))
2994
0
    return FALSE;
2995
2996
0
  if (!wchar_to_json(obj, "StandardName", tz.StandardName, ARRAYSIZE(tz.StandardName)))
2997
0
    return FALSE;
2998
2999
0
  if (!systemtime_to_json(obj, "StandardDate", &tz.StandardDate))
3000
0
    return FALSE;
3001
3002
0
  if (!WINPR_JSON_AddNumberToObject(obj, "StandardBias", tz.Bias))
3003
0
    return FALSE;
3004
3005
0
  if (!wchar_to_json(obj, "DaylightName", tz.DaylightName, ARRAYSIZE(tz.DaylightName)))
3006
0
    return FALSE;
3007
3008
0
  if (!systemtime_to_json(obj, "DaylightDate", &tz.DaylightDate))
3009
0
    return FALSE;
3010
3011
0
  if (!WINPR_JSON_AddNumberToObject(obj, "DaylightBias", tz.Bias))
3012
0
    return FALSE;
3013
0
  return TRUE;
3014
0
}
3015
3016
static BOOL glyph_cache_def_to_json(WINPR_JSON* json, const GLYPH_CACHE_DEFINITION* def)
3017
0
{
3018
0
  WINPR_ASSERT(def);
3019
0
  if (!json || !WINPR_JSON_IsArray(json))
3020
0
    return FALSE;
3021
3022
0
  WINPR_JSON* obj = WINPR_JSON_CreateObject();
3023
0
  if (!obj)
3024
0
    return FALSE;
3025
0
  if (!WINPR_JSON_AddItemToArray(json, obj))
3026
0
    return FALSE;
3027
3028
0
  if (!WINPR_JSON_AddNumberToObject(obj, "cacheEntries", def->cacheEntries))
3029
0
    return FALSE;
3030
0
  if (!WINPR_JSON_AddNumberToObject(obj, "cacheMaximumCellSize", def->cacheMaximumCellSize))
3031
0
    return FALSE;
3032
0
  return TRUE;
3033
0
}
3034
3035
static BOOL glyph_cache_def_array_to_json(WINPR_JSON* json, const GLYPH_CACHE_DEFINITION* def,
3036
                                          size_t count)
3037
0
{
3038
0
  if (!json || !WINPR_JSON_IsArray(json))
3039
0
    return FALSE;
3040
3041
0
  for (size_t x = 0; x < count; x++)
3042
0
  {
3043
0
    if (!glyph_cache_def_to_json(json, &def[x]))
3044
0
      return FALSE;
3045
0
  }
3046
0
  return TRUE;
3047
0
}
3048
3049
static BOOL glyph_cache_def_from_json(GLYPH_CACHE_DEFINITION* def, const WINPR_JSON* json)
3050
0
{
3051
0
  WINPR_ASSERT(def);
3052
0
  WINPR_ASSERT(json);
3053
3054
0
  if (!WINPR_JSON_IsObject(json))
3055
0
    return FALSE;
3056
3057
0
  errno = 0;
3058
0
  def->cacheEntries = (uint16_t)uint_from_json(json, "cacheEntries", UINT16_MAX);
3059
0
  def->cacheMaximumCellSize = (uint16_t)uint_from_json(json, "cacheMaximumCellSize", UINT16_MAX);
3060
0
  return errno == 0;
3061
0
}
3062
3063
static BOOL glyph_cache_def_array_from_json(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer id,
3064
                                            const WINPR_JSON* json)
3065
0
{
3066
0
  if (!json || !WINPR_JSON_IsArray(json))
3067
0
    return FALSE;
3068
3069
0
  const size_t count = WINPR_JSON_GetArraySize(json);
3070
0
  if (!freerdp_settings_set_pointer_len(settings, id, nullptr, count))
3071
0
    return FALSE;
3072
3073
0
  for (size_t x = 0; x < count; x++)
3074
0
  {
3075
0
    GLYPH_CACHE_DEFINITION* cache =
3076
0
        freerdp_settings_get_pointer_array_writable(settings, id, x);
3077
0
    WINPR_JSON* obj = WINPR_JSON_GetArrayItem(json, x);
3078
0
    if (!glyph_cache_def_from_json(cache, obj))
3079
0
      return FALSE;
3080
0
  }
3081
0
  return TRUE;
3082
0
}
3083
3084
static BOOL bitmap_cache_v2_from_json(BITMAP_CACHE_V2_CELL_INFO* info, const WINPR_JSON* json)
3085
0
{
3086
0
  WINPR_ASSERT(info);
3087
3088
0
  if (!json || !WINPR_JSON_IsObject(json))
3089
0
    return FALSE;
3090
3091
0
  errno = 0;
3092
0
  info->numEntries = (uint32_t)uint_from_json(json, "numEntries", UINT32_MAX);
3093
0
  if (errno != 0)
3094
0
    return FALSE;
3095
3096
0
  WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(json, "persistent");
3097
0
  if (!item || !WINPR_JSON_IsBool(item))
3098
0
    return FALSE;
3099
3100
0
  info->persistent = WINPR_JSON_IsTrue(item);
3101
0
  return TRUE;
3102
0
}
3103
3104
static BOOL bitmap_cache_v2_array_from_json(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer id,
3105
                                            const WINPR_JSON* json)
3106
0
{
3107
0
  if (!json || !WINPR_JSON_IsArray(json))
3108
0
    return FALSE;
3109
3110
0
  const size_t count = WINPR_JSON_GetArraySize(json);
3111
0
  if (!freerdp_settings_set_pointer_len(settings, id, nullptr, count))
3112
0
    return FALSE;
3113
3114
0
  for (size_t x = 0; x < count; x++)
3115
0
  {
3116
0
    BITMAP_CACHE_V2_CELL_INFO* cache =
3117
0
        freerdp_settings_get_pointer_array_writable(settings, id, x);
3118
0
    WINPR_JSON* obj = WINPR_JSON_GetArrayItem(json, x);
3119
0
    if (!bitmap_cache_v2_from_json(cache, obj))
3120
0
      return FALSE;
3121
0
  }
3122
0
  return TRUE;
3123
0
}
3124
3125
static BOOL client_cookie_from_json(ARC_CS_PRIVATE_PACKET* cookie, const WINPR_JSON* json)
3126
0
{
3127
0
  WINPR_ASSERT(cookie);
3128
0
  WINPR_ASSERT(json);
3129
3130
0
  if (!WINPR_JSON_IsObject(json))
3131
0
    return FALSE;
3132
3133
0
  errno = 0;
3134
0
  cookie->cbLen = (uint32_t)uint_from_json(json, "cbLen", UINT32_MAX);
3135
0
  cookie->version = (uint32_t)uint_from_json(json, "version", UINT32_MAX);
3136
0
  cookie->logonId = (uint32_t)uint_from_json(json, "logonId", UINT32_MAX);
3137
0
  if (errno != 0)
3138
0
    return FALSE;
3139
3140
0
  WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(json, "securityVerifier");
3141
0
  if (!item || !WINPR_JSON_IsArray(item))
3142
0
    return FALSE;
3143
3144
0
  const size_t len = WINPR_JSON_GetArraySize(item);
3145
0
  if (len != ARRAYSIZE(cookie->securityVerifier))
3146
0
    return FALSE;
3147
3148
0
  errno = 0;
3149
0
  for (size_t x = 0; x < len; x++)
3150
0
  {
3151
0
    WINPR_JSON* citem = WINPR_JSON_GetArrayItem(item, x);
3152
0
    const uint64_t val = uint_from_json_item(citem, UINT8_MAX);
3153
0
    cookie->securityVerifier[x] = (uint8_t)val;
3154
0
  }
3155
0
  return errno == 0;
3156
0
}
3157
3158
static BOOL client_cookie_array_from_json(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer id,
3159
                                          const WINPR_JSON* json)
3160
0
{
3161
0
  if (!json || !WINPR_JSON_IsArray(json))
3162
0
    return FALSE;
3163
3164
0
  const size_t count = WINPR_JSON_GetArraySize(json);
3165
0
  if (!freerdp_settings_set_pointer_len(settings, id, nullptr, count))
3166
0
    return FALSE;
3167
3168
0
  for (size_t x = 0; x < count; x++)
3169
0
  {
3170
0
    ARC_CS_PRIVATE_PACKET* cache = freerdp_settings_get_pointer_array_writable(settings, id, x);
3171
0
    WINPR_JSON* obj = WINPR_JSON_GetArrayItem(json, x);
3172
0
    if (!client_cookie_from_json(cache, obj))
3173
0
      return FALSE;
3174
0
  }
3175
0
  return TRUE;
3176
0
}
3177
3178
static BOOL server_cookie_from_json(ARC_SC_PRIVATE_PACKET* cookie, const WINPR_JSON* json)
3179
0
{
3180
0
  WINPR_ASSERT(cookie);
3181
3182
0
  if (!json || !WINPR_JSON_IsObject(json))
3183
0
    return FALSE;
3184
3185
0
  errno = 0;
3186
0
  cookie->cbLen = (uint32_t)uint_from_json(json, "cbLen", UINT32_MAX);
3187
0
  cookie->version = (uint32_t)uint_from_json(json, "version", UINT32_MAX);
3188
0
  cookie->logonId = (uint32_t)uint_from_json(json, "logonId", UINT32_MAX);
3189
0
  if (errno != 0)
3190
0
    return FALSE;
3191
3192
0
  WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(json, "arcRandomBits");
3193
0
  if (!item || !WINPR_JSON_IsArray(item))
3194
0
    return FALSE;
3195
3196
0
  const size_t len = WINPR_JSON_GetArraySize(item);
3197
0
  if (len != ARRAYSIZE(cookie->arcRandomBits))
3198
0
    return FALSE;
3199
3200
0
  errno = 0;
3201
0
  for (size_t x = 0; x < len; x++)
3202
0
  {
3203
0
    WINPR_JSON* citem = WINPR_JSON_GetArrayItem(item, x);
3204
0
    cookie->arcRandomBits[x] = (uint8_t)uint_from_json_item(citem, UINT8_MAX);
3205
0
  }
3206
0
  return errno == 0;
3207
0
}
3208
3209
static BOOL server_cookie_array_from_json(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer id,
3210
                                          const WINPR_JSON* json)
3211
0
{
3212
0
  if (!json || !WINPR_JSON_IsArray(json))
3213
0
    return FALSE;
3214
3215
0
  const size_t count = WINPR_JSON_GetArraySize(json);
3216
0
  if (!freerdp_settings_set_pointer_len(settings, id, nullptr, count))
3217
0
    return FALSE;
3218
3219
0
  for (size_t x = 0; x < count; x++)
3220
0
  {
3221
0
    ARC_SC_PRIVATE_PACKET* cache = freerdp_settings_get_pointer_array_writable(settings, id, x);
3222
0
    WINPR_JSON* obj = WINPR_JSON_GetArrayItem(json, x);
3223
0
    if (!server_cookie_from_json(cache, obj))
3224
0
      return FALSE;
3225
0
  }
3226
0
  return TRUE;
3227
0
}
3228
3229
static BOOL channel_def_from_json(CHANNEL_DEF* cookie, const WINPR_JSON* json)
3230
0
{
3231
0
  WINPR_ASSERT(cookie);
3232
0
  WINPR_ASSERT(json);
3233
3234
0
  if (!WINPR_JSON_IsObject(json))
3235
0
    return FALSE;
3236
3237
0
  errno = 0;
3238
0
  cookie->options = (uint32_t)uint_from_json(json, "options", UINT32_MAX);
3239
0
  if (errno != 0)
3240
0
    return FALSE;
3241
3242
0
  WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(json, "name");
3243
0
  if (!item || !WINPR_JSON_IsString(item))
3244
0
    return FALSE;
3245
3246
0
  const char* str = WINPR_JSON_GetStringValue(item);
3247
0
  if (!str)
3248
0
    memset(cookie->name, 0, sizeof(cookie->name));
3249
0
  else
3250
0
  {
3251
0
    strncpy(cookie->name, str, strnlen(str, ARRAYSIZE(cookie->name)));
3252
0
  }
3253
0
  return TRUE;
3254
0
}
3255
3256
static BOOL channel_def_array_from_json(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer id,
3257
                                        const WINPR_JSON* json)
3258
0
{
3259
0
  if (!json || !WINPR_JSON_IsArray(json))
3260
0
    return FALSE;
3261
3262
0
  const size_t count = WINPR_JSON_GetArraySize(json);
3263
0
  if (!freerdp_settings_set_pointer_len(settings, id, nullptr, count))
3264
0
    return FALSE;
3265
3266
0
  for (size_t x = 0; x < count; x++)
3267
0
  {
3268
0
    CHANNEL_DEF* cache = freerdp_settings_get_pointer_array_writable(settings, id, x);
3269
0
    WINPR_JSON* obj = WINPR_JSON_GetArrayItem(json, x);
3270
0
    if (!channel_def_from_json(cache, obj))
3271
0
      return FALSE;
3272
0
  }
3273
0
  return TRUE;
3274
0
}
3275
3276
static BOOL monitor_attributes_from_json(MONITOR_ATTRIBUTES* attributes, const WINPR_JSON* json)
3277
0
{
3278
0
  WINPR_ASSERT(attributes);
3279
0
  if (!json || !WINPR_JSON_IsObject(json))
3280
0
    return FALSE;
3281
3282
0
  WINPR_JSON* obj = WINPR_JSON_GetObjectItemCaseSensitive(json, "attributes");
3283
0
  if (!obj || !WINPR_JSON_IsObject(obj))
3284
0
    return FALSE;
3285
3286
0
  errno = 0;
3287
0
  attributes->physicalWidth = (uint32_t)uint_from_json(obj, "physicalWidth", UINT32_MAX);
3288
0
  attributes->physicalHeight = (uint32_t)uint_from_json(obj, "physicalHeight", UINT32_MAX);
3289
0
  attributes->orientation = (uint32_t)uint_from_json(obj, "orientation", UINT32_MAX);
3290
0
  attributes->desktopScaleFactor =
3291
0
      (uint32_t)uint_from_json(obj, "desktopScaleFactor", UINT32_MAX);
3292
0
  attributes->deviceScaleFactor = (uint32_t)uint_from_json(obj, "deviceScaleFactor", UINT32_MAX);
3293
0
  return errno == 0;
3294
0
}
3295
3296
static BOOL monitor_def_from_json(rdpMonitor* monitor, const WINPR_JSON* json)
3297
0
{
3298
0
  WINPR_ASSERT(monitor);
3299
3300
0
  if (!json || !WINPR_JSON_IsObject(json))
3301
0
    return FALSE;
3302
3303
0
  errno = 0;
3304
0
  monitor->x = (int32_t)int_from_json(json, "x", INT32_MIN, INT32_MAX);
3305
0
  monitor->y = (int32_t)int_from_json(json, "y", INT32_MIN, INT32_MAX);
3306
0
  monitor->width = (int32_t)int_from_json(json, "width", 0, INT32_MAX);
3307
0
  monitor->height = (int32_t)int_from_json(json, "height", 0, INT32_MAX);
3308
0
  monitor->orig_screen = (uint32_t)uint_from_json(json, "orig_screen", UINT32_MAX);
3309
0
  if (errno != 0)
3310
0
    return FALSE;
3311
3312
0
  WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(json, "is_primary");
3313
0
  if (!item)
3314
0
    return FALSE;
3315
0
  if (!WINPR_JSON_IsBool(item))
3316
0
    return FALSE;
3317
0
  monitor->is_primary = WINPR_JSON_IsTrue(item) ? 1 : 0;
3318
3319
0
  return monitor_attributes_from_json(&monitor->attributes, json);
3320
0
}
3321
3322
static BOOL monitor_def_array_from_json(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer id,
3323
                                        const WINPR_JSON* json)
3324
0
{
3325
0
  if (!json || !WINPR_JSON_IsArray(json))
3326
0
    return FALSE;
3327
3328
0
  const size_t count = WINPR_JSON_GetArraySize(json);
3329
0
  if (!freerdp_settings_set_pointer_len(settings, id, nullptr, count))
3330
0
    return FALSE;
3331
3332
0
  for (size_t x = 0; x < count; x++)
3333
0
  {
3334
0
    rdpMonitor* cache = freerdp_settings_get_pointer_array_writable(settings, id, x);
3335
0
    WINPR_JSON* obj = WINPR_JSON_GetArrayItem(json, x);
3336
0
    if (!monitor_def_from_json(cache, obj))
3337
0
      return FALSE;
3338
0
  }
3339
0
  return TRUE;
3340
0
}
3341
3342
static BOOL client_cookie_to_json(WINPR_JSON* json, const ARC_CS_PRIVATE_PACKET* cs)
3343
0
{
3344
0
  WINPR_JSON* obj = WINPR_JSON_CreateObject();
3345
0
  if (!obj)
3346
0
    return FALSE;
3347
0
  if (!WINPR_JSON_AddItemToArray(json, obj))
3348
0
    return FALSE;
3349
3350
0
  if (!WINPR_JSON_AddNumberToObject(obj, "cbLen", cs->cbLen))
3351
0
    return FALSE;
3352
0
  if (!WINPR_JSON_AddNumberToObject(obj, "version", cs->version))
3353
0
    return FALSE;
3354
0
  if (!WINPR_JSON_AddNumberToObject(obj, "logonId", cs->logonId))
3355
0
    return FALSE;
3356
0
  WINPR_JSON* array = WINPR_JSON_AddArrayToObject(obj, "securityVerifier");
3357
0
  if (!array)
3358
0
    return FALSE;
3359
3360
0
  for (size_t x = 0; x < ARRAYSIZE(cs->securityVerifier); x++)
3361
0
  {
3362
0
    WINPR_JSON* item = WINPR_JSON_CreateNumber(cs->securityVerifier[x]);
3363
0
    if (!item)
3364
0
      return FALSE;
3365
0
    if (!WINPR_JSON_AddItemToArray(array, item))
3366
0
      return FALSE;
3367
0
  }
3368
0
  return TRUE;
3369
0
}
3370
3371
static BOOL client_cookie_array_to_json(WINPR_JSON* json, const ARC_CS_PRIVATE_PACKET* cs,
3372
                                        size_t count)
3373
0
{
3374
0
  for (size_t x = 0; x < count; x++)
3375
0
  {
3376
0
    if (!client_cookie_to_json(json, &cs[x]))
3377
0
      return FALSE;
3378
0
  }
3379
0
  return TRUE;
3380
0
}
3381
3382
static BOOL server_cookie_to_json(WINPR_JSON* json, const ARC_SC_PRIVATE_PACKET* cs)
3383
0
{
3384
0
  WINPR_JSON* obj = WINPR_JSON_CreateObject();
3385
0
  if (!obj)
3386
0
    return FALSE;
3387
0
  if (!WINPR_JSON_AddItemToArray(json, obj))
3388
0
    return FALSE;
3389
3390
0
  if (!WINPR_JSON_AddNumberToObject(obj, "cbLen", cs->cbLen))
3391
0
    return FALSE;
3392
0
  if (!WINPR_JSON_AddNumberToObject(obj, "version", cs->version))
3393
0
    return FALSE;
3394
0
  if (!WINPR_JSON_AddNumberToObject(obj, "logonId", cs->logonId))
3395
0
    return FALSE;
3396
0
  WINPR_JSON* array = WINPR_JSON_AddArrayToObject(obj, "arcRandomBits");
3397
0
  if (!array)
3398
0
    return FALSE;
3399
3400
0
  for (size_t x = 0; x < ARRAYSIZE(cs->arcRandomBits); x++)
3401
0
  {
3402
0
    WINPR_JSON* item = WINPR_JSON_CreateNumber(cs->arcRandomBits[x]);
3403
0
    if (!item)
3404
0
      return FALSE;
3405
0
    if (!WINPR_JSON_AddItemToArray(array, item))
3406
0
      return FALSE;
3407
0
  }
3408
0
  return TRUE;
3409
0
}
3410
3411
static BOOL server_cookie_array_to_json(WINPR_JSON* json, const ARC_SC_PRIVATE_PACKET* cs,
3412
                                        size_t count)
3413
0
{
3414
0
  for (size_t x = 0; x < count; x++)
3415
0
  {
3416
0
    if (!server_cookie_to_json(json, &cs[x]))
3417
0
      return FALSE;
3418
0
  }
3419
0
  return TRUE;
3420
0
}
3421
3422
static BOOL bitmap_cache_v2_to_json(WINPR_JSON* json, const BITMAP_CACHE_V2_CELL_INFO* info)
3423
0
{
3424
0
  WINPR_JSON* obj = WINPR_JSON_CreateObject();
3425
0
  if (!obj)
3426
0
    return FALSE;
3427
0
  if (!WINPR_JSON_AddItemToArray(json, obj))
3428
0
    return FALSE;
3429
3430
0
  if (!WINPR_JSON_AddNumberToObject(obj, "numEntries", info->numEntries))
3431
0
    return FALSE;
3432
0
  if (!WINPR_JSON_AddBoolToObject(obj, "persistent", info->persistent))
3433
0
    return FALSE;
3434
0
  return TRUE;
3435
0
}
3436
3437
static BOOL bitmap_cache_v2_array_to_json(WINPR_JSON* json, const BITMAP_CACHE_V2_CELL_INFO* info,
3438
                                          size_t count)
3439
0
{
3440
0
  for (size_t x = 0; x < count; x++)
3441
0
  {
3442
0
    if (!bitmap_cache_v2_to_json(json, &info[x]))
3443
0
      return FALSE;
3444
0
  }
3445
0
  return TRUE;
3446
0
}
3447
3448
static BOOL monitor_attributes_to_json(WINPR_JSON* pobj, const MONITOR_ATTRIBUTES* attributes)
3449
0
{
3450
0
  WINPR_ASSERT(attributes);
3451
0
  WINPR_JSON* obj = WINPR_JSON_AddObjectToObject(pobj, "attributes");
3452
0
  if (!obj)
3453
0
    return FALSE;
3454
0
  if (!WINPR_JSON_AddNumberToObject(obj, "physicalWidth", attributes->physicalWidth))
3455
0
    return FALSE;
3456
0
  if (!WINPR_JSON_AddNumberToObject(obj, "physicalHeight", attributes->physicalHeight))
3457
0
    return FALSE;
3458
0
  if (!WINPR_JSON_AddNumberToObject(obj, "orientation", attributes->orientation))
3459
0
    return FALSE;
3460
0
  if (!WINPR_JSON_AddNumberToObject(obj, "desktopScaleFactor", attributes->desktopScaleFactor))
3461
0
    return FALSE;
3462
0
  if (!WINPR_JSON_AddNumberToObject(obj, "deviceScaleFactor", attributes->deviceScaleFactor))
3463
0
    return FALSE;
3464
0
  return TRUE;
3465
0
}
3466
3467
static BOOL monitor_def_to_json(WINPR_JSON* json, const rdpMonitor* monitor)
3468
0
{
3469
0
  WINPR_ASSERT(monitor);
3470
0
  WINPR_JSON* obj = WINPR_JSON_CreateObject();
3471
0
  if (!obj)
3472
0
    return FALSE;
3473
0
  if (!WINPR_JSON_AddItemToArray(json, obj))
3474
0
    return FALSE;
3475
3476
0
  if (!WINPR_JSON_AddNumberToObject(obj, "x", monitor->x))
3477
0
    return FALSE;
3478
0
  if (!WINPR_JSON_AddNumberToObject(obj, "y", monitor->y))
3479
0
    return FALSE;
3480
0
  if (!WINPR_JSON_AddNumberToObject(obj, "width", monitor->width))
3481
0
    return FALSE;
3482
0
  if (!WINPR_JSON_AddNumberToObject(obj, "height", monitor->height))
3483
0
    return FALSE;
3484
0
  if (!WINPR_JSON_AddBoolToObject(obj, "is_primary", monitor->is_primary != 0))
3485
0
    return FALSE;
3486
0
  if (!WINPR_JSON_AddNumberToObject(obj, "orig_screen", monitor->orig_screen))
3487
0
    return FALSE;
3488
0
  return monitor_attributes_to_json(obj, &monitor->attributes);
3489
0
}
3490
3491
static BOOL monitor_def_array_to_json(WINPR_JSON* json, const rdpMonitor* monitors, size_t count)
3492
0
{
3493
0
  for (size_t x = 0; x < count; x++)
3494
0
  {
3495
0
    if (!monitor_def_to_json(json, &monitors[x]))
3496
0
      return FALSE;
3497
0
  }
3498
0
  return TRUE;
3499
0
}
3500
3501
static BOOL channel_def_to_json(WINPR_JSON* json, const CHANNEL_DEF* channel)
3502
0
{
3503
0
  WINPR_ASSERT(channel);
3504
0
  WINPR_JSON* obj = WINPR_JSON_CreateObject();
3505
0
  if (!obj)
3506
0
    return FALSE;
3507
0
  if (!WINPR_JSON_AddItemToArray(json, obj))
3508
0
    return FALSE;
3509
3510
0
  if (!WINPR_JSON_AddStringToObject(obj, "name", channel->name))
3511
0
    return FALSE;
3512
0
  if (!WINPR_JSON_AddNumberToObject(obj, "options", channel->options))
3513
0
    return FALSE;
3514
0
  return TRUE;
3515
0
}
3516
3517
static BOOL channel_def_array_to_json(WINPR_JSON* json, const CHANNEL_DEF* channels, size_t count)
3518
0
{
3519
0
  for (size_t x = 0; x < count; x++)
3520
0
  {
3521
0
    if (!channel_def_to_json(json, &channels[x]))
3522
0
      return FALSE;
3523
0
  }
3524
0
  return TRUE;
3525
0
}
3526
3527
static BOOL serialize_pointer(const rdpSettings* settings, WINPR_JSON* json,
3528
                              FreeRDP_Settings_Keys_Pointer id)
3529
0
{
3530
0
  const char* name = freerdp_settings_get_name_for_key(id);
3531
0
  if (!name)
3532
0
    return FALSE;
3533
3534
0
  WINPR_JSON* jval = WINPR_JSON_AddArrayToObject(json, name);
3535
0
  if (!jval)
3536
0
    return FALSE;
3537
3538
0
  const void* val = freerdp_settings_get_pointer(settings, id);
3539
0
  if (!val)
3540
0
    return TRUE;
3541
3542
0
  switch (id)
3543
0
  {
3544
0
    case FreeRDP_instance:
3545
0
    {
3546
0
      union
3547
0
      {
3548
0
        const void* v;
3549
0
        uintptr_t u;
3550
0
      } ptr;
3551
3552
0
      ptr.v = val;
3553
0
      return fill_array(jval, &ptr.u, sizeof(ptr.u));
3554
0
    }
3555
0
    case FreeRDP_ServerRandom:
3556
0
    {
3557
0
      const uint32_t len = freerdp_settings_get_uint32(settings, FreeRDP_ServerRandomLength);
3558
0
      return fill_array(jval, val, len);
3559
0
    }
3560
0
    case FreeRDP_ServerCertificate:
3561
0
    {
3562
0
      const uint32_t len =
3563
0
          freerdp_settings_get_uint32(settings, FreeRDP_ServerCertificateLength);
3564
0
      return fill_array(jval, val, len);
3565
0
    }
3566
0
    case FreeRDP_ClientRandom:
3567
0
    {
3568
0
      const uint32_t len = freerdp_settings_get_uint32(settings, FreeRDP_ClientRandomLength);
3569
0
      return fill_array(jval, val, len);
3570
0
    }
3571
0
    case FreeRDP_ServerLicenseProductIssuers:
3572
0
    {
3573
0
      const uint32_t len =
3574
0
          freerdp_settings_get_uint32(settings, FreeRDP_ServerLicenseProductIssuersCount);
3575
0
      return string_array_to_json(jval, settings, len, FreeRDP_ServerLicenseProductIssuers);
3576
0
    }
3577
0
    case FreeRDP_RedirectionPassword:
3578
0
    {
3579
0
      const uint32_t len =
3580
0
          freerdp_settings_get_uint32(settings, FreeRDP_RedirectionPasswordLength);
3581
0
      return fill_array(jval, val, len);
3582
0
    }
3583
0
    case FreeRDP_RedirectionGuid:
3584
0
    {
3585
0
      const uint32_t len =
3586
0
          freerdp_settings_get_uint32(settings, FreeRDP_RedirectionGuidLength);
3587
0
      return fill_array(jval, val, len);
3588
0
    }
3589
0
    case FreeRDP_LoadBalanceInfo:
3590
0
    {
3591
0
      const uint32_t len =
3592
0
          freerdp_settings_get_uint32(settings, FreeRDP_LoadBalanceInfoLength);
3593
0
      return fill_array(jval, val, len);
3594
0
    }
3595
0
    case FreeRDP_ClientTimeZone:
3596
0
    {
3597
0
      return tz_info_to_json(jval, val);
3598
0
    }
3599
0
    case FreeRDP_RedirectionTsvUrl:
3600
0
    {
3601
0
      const uint32_t len =
3602
0
          freerdp_settings_get_uint32(settings, FreeRDP_RedirectionTsvUrlLength);
3603
0
      return fill_array(jval, val, len);
3604
0
    }
3605
0
    case FreeRDP_GlyphCache:
3606
0
    {
3607
0
      return glyph_cache_def_array_to_json(jval, val, val ? 10 : 0);
3608
0
    }
3609
0
    case FreeRDP_FragCache:
3610
0
    {
3611
0
      return glyph_cache_def_array_to_json(jval, val, val ? 1 : 0);
3612
0
    }
3613
0
    case FreeRDP_BitmapCacheV2CellInfo:
3614
0
    {
3615
0
      const uint32_t len =
3616
0
          freerdp_settings_get_uint32(settings, FreeRDP_BitmapCacheV2NumCells);
3617
0
      return bitmap_cache_v2_array_to_json(jval, val, len);
3618
0
    }
3619
0
    case FreeRDP_OrderSupport:
3620
0
    {
3621
0
      const uint32_t len = 32;
3622
0
      return fill_array(jval, val, len);
3623
0
    }
3624
0
    case FreeRDP_ClientAutoReconnectCookie:
3625
0
    {
3626
0
      return client_cookie_array_to_json(jval, val, 1);
3627
0
    }
3628
0
    case FreeRDP_ServerAutoReconnectCookie:
3629
0
    {
3630
0
      return server_cookie_array_to_json(jval, val, 1);
3631
0
    }
3632
0
    case FreeRDP_Password51:
3633
0
    {
3634
0
      const uint32_t len = freerdp_settings_get_uint32(settings, FreeRDP_Password51Length);
3635
0
      return fill_array(jval, val, len);
3636
0
    }
3637
0
    case FreeRDP_ReceivedCapabilities:
3638
0
    {
3639
0
      const uint32_t len =
3640
0
          freerdp_settings_get_uint32(settings, FreeRDP_ReceivedCapabilitiesSize);
3641
0
      return fill_array(jval, val, len);
3642
0
    }
3643
0
    case FreeRDP_MonitorIds:
3644
0
    {
3645
0
      const uint32_t len = freerdp_settings_get_uint32(settings, FreeRDP_NumMonitorIds);
3646
0
      return fill_uint32_array(jval, val, len);
3647
0
    }
3648
0
    case FreeRDP_TargetNetPorts:
3649
0
    {
3650
0
      const uint32_t len =
3651
0
          freerdp_settings_get_uint32(settings, FreeRDP_TargetNetAddressCount);
3652
0
      return fill_uint32_array(jval, val, len);
3653
0
    }
3654
0
    case FreeRDP_MonitorDefArray:
3655
0
    {
3656
0
      const uint32_t len = freerdp_settings_get_uint32(settings, FreeRDP_MonitorDefArraySize);
3657
0
      return monitor_def_array_to_json(jval, val, len);
3658
0
    }
3659
0
    case FreeRDP_ChannelDefArray:
3660
0
    {
3661
0
      const uint32_t len = freerdp_settings_get_uint32(settings, FreeRDP_ChannelDefArraySize);
3662
0
      return channel_def_array_to_json(jval, val, len);
3663
0
    }
3664
0
    case FreeRDP_ReceivedCapabilityDataSizes:
3665
0
    {
3666
0
      const uint32_t len =
3667
0
          freerdp_settings_get_uint32(settings, FreeRDP_ReceivedCapabilitiesSize);
3668
0
      return fill_uint32_array(jval, val, len);
3669
0
    }
3670
0
    case FreeRDP_ReceivedCapabilityData:
3671
0
    {
3672
0
      const uint32_t len =
3673
0
          freerdp_settings_get_uint32(settings, FreeRDP_ReceivedCapabilitiesSize);
3674
0
      const UINT32* pclen =
3675
0
          freerdp_settings_get_pointer(settings, FreeRDP_ReceivedCapabilityDataSizes);
3676
0
      if (!pclen)
3677
0
        return FALSE;
3678
3679
0
      for (uint32_t x = 0; x < len; x++)
3680
0
      {
3681
0
        const char* cval = freerdp_settings_get_pointer_array(settings, id, x);
3682
3683
0
        WINPR_JSON* item = WINPR_JSON_CreateArray();
3684
0
        if (!item)
3685
0
          return FALSE;
3686
0
        if (!WINPR_JSON_AddItemToArray(jval, item))
3687
0
          return FALSE;
3688
0
        if (!fill_array(item, cval, pclen[x]))
3689
0
          return FALSE;
3690
0
      }
3691
0
      return TRUE;
3692
0
    }
3693
0
    case FreeRDP_TargetNetAddresses:
3694
0
    {
3695
0
      const uint32_t len =
3696
0
          freerdp_settings_get_uint32(settings, FreeRDP_TargetNetAddressCount);
3697
0
      return string_array_to_json(jval, settings, len, id);
3698
0
    }
3699
0
    case FreeRDP_RedirectionTargetCertificate:
3700
0
    case FreeRDP_RdpServerCertificate:
3701
0
    {
3702
0
      WINPR_JSON* item = nullptr;
3703
0
      size_t len = 0;
3704
0
      char* pem = freerdp_certificate_get_pem(val, &len);
3705
0
      if (pem)
3706
0
        item = WINPR_JSON_CreateString(pem);
3707
0
      else if (val)
3708
0
        item = WINPR_JSON_CreateString("");
3709
0
      else
3710
0
        item = WINPR_JSON_CreateNull();
3711
0
      free(pem);
3712
0
      if (!item)
3713
0
        return TRUE;
3714
3715
0
      return WINPR_JSON_AddItemToArray(jval, item);
3716
0
    }
3717
0
    case FreeRDP_RdpServerRsaKey:
3718
0
    {
3719
0
      WINPR_JSON* item = nullptr;
3720
0
      size_t len = 0;
3721
0
      char* pem = freerdp_key_get_pem(val, &len, nullptr);
3722
0
      if (pem)
3723
0
        item = WINPR_JSON_CreateString(pem);
3724
0
      free(pem);
3725
0
      if (!item)
3726
0
        return TRUE;
3727
3728
0
      return WINPR_JSON_AddItemToArray(jval, item);
3729
0
    }
3730
0
    case FreeRDP_DeviceArray:
3731
0
    {
3732
0
      const uint32_t len = freerdp_settings_get_uint32(settings, FreeRDP_DeviceArraySize);
3733
0
      return json_from_device_item_array(jval, settings, id, len);
3734
0
    }
3735
0
    case FreeRDP_StaticChannelArray:
3736
0
    {
3737
0
      const uint32_t len =
3738
0
          freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelArraySize);
3739
0
      return json_from_addin_item_array(jval, settings, id, len);
3740
0
    }
3741
0
    case FreeRDP_DynamicChannelArray:
3742
0
    {
3743
0
      const uint32_t len =
3744
0
          freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelArraySize);
3745
0
      return json_from_addin_item_array(jval, settings, id, len);
3746
0
    }
3747
0
    case FreeRDP_POINTER_UNUSED:
3748
0
    default:
3749
0
      return FALSE;
3750
0
  }
3751
0
}
3752
3753
char* freerdp_settings_serialize(const rdpSettings* settings, BOOL pretty, size_t* plength)
3754
0
{
3755
0
  char* str = nullptr;
3756
3757
0
  if (plength)
3758
0
    *plength = 0;
3759
3760
0
  if (!settings)
3761
0
    return nullptr;
3762
3763
0
  WINPR_JSON* json = WINPR_JSON_CreateObject();
3764
0
  if (!json)
3765
0
    return nullptr;
3766
3767
0
  WINPR_JSON* jbool = WINPR_JSON_AddObjectToObject(
3768
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_BOOL));
3769
0
  WINPR_JSON* juint16 = WINPR_JSON_AddObjectToObject(
3770
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_UINT16));
3771
0
  WINPR_JSON* jint16 = WINPR_JSON_AddObjectToObject(
3772
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_INT16));
3773
0
  WINPR_JSON* juint32 = WINPR_JSON_AddObjectToObject(
3774
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_UINT32));
3775
0
  WINPR_JSON* jint32 = WINPR_JSON_AddObjectToObject(
3776
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_INT32));
3777
0
  WINPR_JSON* juint64 = WINPR_JSON_AddObjectToObject(
3778
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_UINT64));
3779
0
  WINPR_JSON* jint64 = WINPR_JSON_AddObjectToObject(
3780
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_INT64));
3781
0
  WINPR_JSON* jstring = WINPR_JSON_AddObjectToObject(
3782
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_STRING));
3783
0
  WINPR_JSON* jpointer = WINPR_JSON_AddObjectToObject(
3784
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_POINTER));
3785
0
  if (!jbool || !juint16 || !jint16 || !juint32 || !jint32 || !juint64 || !jint64 || !jstring ||
3786
0
      !jpointer)
3787
0
    goto fail;
3788
3789
0
  for (int x = 0; x < FreeRDP_Settings_StableAPI_MAX; x++)
3790
0
  {
3791
0
    union
3792
0
    {
3793
3794
0
      int s;
3795
0
      FreeRDP_Settings_Keys_Bool b;
3796
0
      FreeRDP_Settings_Keys_Int16 i16;
3797
0
      FreeRDP_Settings_Keys_UInt16 u16;
3798
0
      FreeRDP_Settings_Keys_Int32 i32;
3799
0
      FreeRDP_Settings_Keys_UInt32 u32;
3800
0
      FreeRDP_Settings_Keys_Int64 i64;
3801
0
      FreeRDP_Settings_Keys_UInt64 u64;
3802
0
      FreeRDP_Settings_Keys_String str;
3803
0
      FreeRDP_Settings_Keys_Pointer ptr;
3804
0
    } iter;
3805
0
    iter.s = x;
3806
3807
0
    const char* name = freerdp_settings_get_name_for_key(iter.s);
3808
0
    SSIZE_T type = freerdp_settings_get_type_for_key(iter.s);
3809
0
    switch (type)
3810
0
    {
3811
0
      case RDP_SETTINGS_TYPE_BOOL:
3812
0
      {
3813
0
        const BOOL val = freerdp_settings_get_bool(settings, iter.b);
3814
0
        if (!WINPR_JSON_AddBoolToObject(jbool, name, val))
3815
0
          goto fail;
3816
0
      }
3817
0
      break;
3818
0
      case RDP_SETTINGS_TYPE_UINT16:
3819
0
      {
3820
0
        const uint16_t val = freerdp_settings_get_uint16(settings, iter.u16);
3821
0
        if (!WINPR_JSON_AddNumberToObject(juint16, name, (double)val))
3822
0
          goto fail;
3823
0
      }
3824
0
      break;
3825
0
      case RDP_SETTINGS_TYPE_INT16:
3826
0
      {
3827
0
        const int16_t val = freerdp_settings_get_int16(settings, iter.i16);
3828
0
        if (!WINPR_JSON_AddNumberToObject(jint16, name, (double)val))
3829
0
          goto fail;
3830
0
      }
3831
0
      break;
3832
0
      case RDP_SETTINGS_TYPE_UINT32:
3833
0
      {
3834
0
        const uint32_t val = freerdp_settings_get_uint32(settings, iter.u32);
3835
0
        if (!WINPR_JSON_AddNumberToObject(juint32, name, (double)val))
3836
0
          goto fail;
3837
0
      }
3838
0
      break;
3839
0
      case RDP_SETTINGS_TYPE_INT32:
3840
0
      {
3841
0
        const int32_t val = freerdp_settings_get_int32(settings, iter.i32);
3842
0
        if (!WINPR_JSON_AddNumberToObject(jint32, name, (double)val))
3843
0
          goto fail;
3844
0
      }
3845
0
      break;
3846
0
      case RDP_SETTINGS_TYPE_UINT64:
3847
0
      {
3848
0
        const uint64_t val = freerdp_settings_get_uint64(settings, iter.u64);
3849
0
        if (!WINPR_JSON_AddNumberToObject(juint64, name, (double)val))
3850
0
          goto fail;
3851
0
      }
3852
0
      break;
3853
0
      case RDP_SETTINGS_TYPE_INT64:
3854
0
      {
3855
0
        const int64_t val = freerdp_settings_get_int64(settings, iter.i64);
3856
0
        if (!WINPR_JSON_AddNumberToObject(jint64, name, (double)val))
3857
0
          goto fail;
3858
0
      }
3859
0
      break;
3860
0
      case RDP_SETTINGS_TYPE_STRING:
3861
0
      {
3862
0
        const char* val = freerdp_settings_get_string(settings, iter.str);
3863
0
        if (val)
3864
0
        {
3865
0
          if (!WINPR_JSON_AddStringToObject(jstring, name, val))
3866
0
            goto fail;
3867
0
        }
3868
0
        else
3869
0
        {
3870
0
          if (!WINPR_JSON_AddNullToObject(jstring, name))
3871
0
            goto fail;
3872
0
        }
3873
0
      }
3874
0
      break;
3875
0
      case RDP_SETTINGS_TYPE_POINTER:
3876
0
        if (!serialize_pointer(settings, jpointer, iter.ptr))
3877
0
          goto fail;
3878
0
        break;
3879
0
      default:
3880
0
        break;
3881
0
    }
3882
0
  }
3883
3884
0
  if (pretty)
3885
0
    str = WINPR_JSON_Print(json);
3886
0
  else
3887
0
    str = WINPR_JSON_PrintUnformatted(json);
3888
3889
0
  if (!str)
3890
0
    goto fail;
3891
0
  if (plength)
3892
0
    *plength = strlen(str);
3893
3894
0
fail:
3895
0
  WINPR_JSON_Delete(json);
3896
0
  return str;
3897
0
}
3898
3899
static BOOL val_from_array(rdpSettings* settings, const WINPR_JSON* json,
3900
                           FreeRDP_Settings_Keys_Pointer key, size_t esize)
3901
0
{
3902
0
  if (WINPR_JSON_IsNull(json))
3903
0
    return freerdp_settings_set_pointer(settings, key, nullptr);
3904
0
  if (!WINPR_JSON_IsArray(json))
3905
0
    return FALSE;
3906
3907
0
  size_t len = WINPR_JSON_GetArraySize(json);
3908
0
  if (len == 0)
3909
0
    return freerdp_settings_set_pointer(settings, key, nullptr);
3910
3911
0
  size_t count = len / esize;
3912
0
  if (count * esize != len)
3913
0
    return FALSE;
3914
3915
0
  if (!freerdp_settings_set_pointer_len(settings, key, nullptr, count))
3916
0
    return FALSE;
3917
3918
0
  BYTE* data = freerdp_settings_get_pointer_writable(settings, key);
3919
0
  if (!data)
3920
0
    return FALSE;
3921
3922
0
  errno = 0;
3923
0
  for (size_t x = 0; x < len; x++)
3924
0
  {
3925
0
    WINPR_JSON* val = WINPR_JSON_GetArrayItem(json, x);
3926
0
    data[x] = (uint8_t)uint_from_json_item(val, UINT8_MAX);
3927
0
  }
3928
3929
0
  return errno == 0;
3930
0
}
3931
3932
static BOOL uintptr_from_array(rdpSettings* settings, const WINPR_JSON* json)
3933
0
{
3934
0
  FreeRDP_Settings_Keys_Pointer key = FreeRDP_instance;
3935
0
  if (!json || !WINPR_JSON_IsArray(json))
3936
0
    return FALSE;
3937
3938
0
  size_t len = WINPR_JSON_GetArraySize(json);
3939
0
  if (len > sizeof(void*))
3940
0
    return FALSE;
3941
3942
0
  if (len == 0)
3943
0
    return freerdp_settings_set_pointer(settings, key, nullptr);
3944
3945
0
  union
3946
0
  {
3947
0
    void* v;
3948
0
    uint8_t u[sizeof(void*)];
3949
0
  } ptr;
3950
3951
0
  errno = 0;
3952
0
  for (size_t x = 0; x < len; x++)
3953
0
  {
3954
0
    WINPR_JSON* val = WINPR_JSON_GetArrayItem(json, x);
3955
0
    ptr.u[x] = (uint8_t)uint_from_json_item(val, UINT8_MAX);
3956
0
  }
3957
0
  if (errno != 0)
3958
0
    return FALSE;
3959
0
  return freerdp_settings_set_pointer(settings, key, ptr.v);
3960
0
}
3961
3962
static BOOL val_from_uint32_array(rdpSettings* settings, const WINPR_JSON* json,
3963
                                  FreeRDP_Settings_Keys_Pointer key,
3964
                                  FreeRDP_Settings_Keys_UInt32 keyId)
3965
0
{
3966
0
  if (WINPR_JSON_IsNull(json))
3967
0
    return freerdp_settings_set_pointer(settings, key, nullptr);
3968
0
  if (!WINPR_JSON_IsArray(json))
3969
0
    return FALSE;
3970
3971
0
  const size_t len = WINPR_JSON_GetArraySize(json);
3972
0
  if ((FreeRDP_UINT32_UNUSED != keyId) && (freerdp_settings_get_uint32(settings, keyId) != len))
3973
0
  {
3974
0
    if (!freerdp_settings_set_pointer_len(settings, key, nullptr, len))
3975
0
      return FALSE;
3976
0
  }
3977
3978
0
  errno = 0;
3979
0
  for (size_t x = 0; x < len; x++)
3980
0
  {
3981
0
    UINT32* data = freerdp_settings_get_pointer_array_writable(settings, key, x);
3982
0
    if (!data)
3983
0
      return FALSE;
3984
3985
0
    WINPR_JSON* val = WINPR_JSON_GetArrayItem(json, x);
3986
0
    data[0] = (uint32_t)uint_from_json_item(val, UINT32_MAX);
3987
0
  }
3988
0
  return errno == 0;
3989
0
}
3990
3991
static BOOL caps_data_entry_from_json(rdpSettings* settings, size_t offset, const WINPR_JSON* json)
3992
0
{
3993
0
  if (!json || !WINPR_JSON_IsArray(json))
3994
0
    return FALSE;
3995
3996
0
  const size_t size = WINPR_JSON_GetArraySize(json);
3997
0
  if (size == 0)
3998
0
  {
3999
0
    return freerdp_settings_set_pointer_array(settings, FreeRDP_ReceivedCapabilityData, offset,
4000
0
                                              nullptr);
4001
0
  }
4002
4003
0
  uint8_t* data = calloc(size, sizeof(uint8_t));
4004
0
  if (!data)
4005
0
    return FALSE;
4006
4007
0
  if (!freerdp_settings_set_pointer_array(settings, FreeRDP_ReceivedCapabilityData, offset, data))
4008
0
  {
4009
0
    free(data);
4010
0
    return FALSE;
4011
0
  }
4012
4013
0
  errno = 0;
4014
0
  for (size_t x = 0; x < size; x++)
4015
0
  {
4016
0
    WINPR_JSON* item = WINPR_JSON_GetArrayItem(json, x);
4017
0
    data[x] = (uint8_t)uint_from_json_item(item, UINT8_MAX);
4018
0
  }
4019
4020
  // NOLINTNEXTLINE(clang-analyzer-unix.Malloc)
4021
0
  return errno == 0;
4022
0
}
4023
4024
static BOOL caps_data_array_from_json(rdpSettings* settings, const WINPR_JSON* json)
4025
0
{
4026
0
  if (!json || !WINPR_JSON_IsArray(json))
4027
0
    return FALSE;
4028
4029
0
  const size_t count = WINPR_JSON_GetArraySize(json);
4030
0
  if (!freerdp_settings_set_pointer_len(settings, FreeRDP_ReceivedCapabilityData, nullptr, count))
4031
0
    return FALSE;
4032
4033
0
  for (uint32_t x = 0; x < count; x++)
4034
0
  {
4035
0
    WINPR_JSON* array = WINPR_JSON_GetArrayItem(json, x);
4036
0
    if (!caps_data_entry_from_json(settings, x, array))
4037
0
      return FALSE;
4038
0
  }
4039
0
  return TRUE;
4040
0
}
4041
4042
static BOOL str_array_from_json(rdpSettings* settings, const WINPR_JSON* json,
4043
                                FreeRDP_Settings_Keys_Pointer key)
4044
0
{
4045
0
  if (WINPR_JSON_IsNull(json))
4046
0
    return freerdp_settings_set_pointer_len(settings, key, nullptr, 0);
4047
0
  if (!WINPR_JSON_IsArray(json))
4048
0
    return FALSE;
4049
4050
0
  size_t len = WINPR_JSON_GetArraySize(json);
4051
0
  if (!freerdp_settings_set_pointer_len(settings, key, nullptr, len))
4052
0
    return FALSE;
4053
4054
0
  for (size_t x = 0; x < len; x++)
4055
0
  {
4056
0
    WINPR_JSON* cval = WINPR_JSON_GetArrayItem(json, x);
4057
0
    if (!cval)
4058
0
      return FALSE;
4059
0
    if (!WINPR_JSON_IsString(cval))
4060
0
      return FALSE;
4061
4062
0
    const char* val = WINPR_JSON_GetStringValue(cval);
4063
0
    if (!freerdp_settings_set_pointer_array(settings, key, x, val))
4064
0
      return FALSE;
4065
0
  }
4066
0
  return TRUE;
4067
0
}
4068
4069
static BOOL addin_argv_from_json(rdpSettings* settings, const WINPR_JSON* json,
4070
                                 FreeRDP_Settings_Keys_Pointer key)
4071
0
{
4072
0
  if (WINPR_JSON_IsNull(json))
4073
0
    return freerdp_settings_set_pointer(settings, key, nullptr);
4074
4075
0
  if (!WINPR_JSON_IsArray(json))
4076
0
    return FALSE;
4077
4078
0
  size_t len = WINPR_JSON_GetArraySize(json);
4079
0
  if (!freerdp_settings_set_pointer_len(settings, key, nullptr, len))
4080
0
    return FALSE;
4081
4082
0
  for (size_t x = 0; x < len; x++)
4083
0
  {
4084
0
    WINPR_JSON* val = WINPR_JSON_GetArrayItem(json, x);
4085
0
    if (val && WINPR_JSON_IsObject(val))
4086
0
    {
4087
0
      WINPR_JSON* jargc = WINPR_JSON_GetObjectItemCaseSensitive(val, "argc");
4088
0
      WINPR_JSON* array = WINPR_JSON_GetObjectItemCaseSensitive(val, "argv");
4089
0
      if (!jargc || !array)
4090
0
        continue;
4091
0
      if (!WINPR_JSON_IsNumber(jargc) || !WINPR_JSON_IsArray(array))
4092
0
        continue;
4093
4094
0
      const int argc = (int)int_from_json_item(jargc, INT32_MIN, INT32_MAX);
4095
0
      if (errno != 0)
4096
0
        return FALSE;
4097
0
      const size_t jlen = WINPR_JSON_GetArraySize(array);
4098
0
      if (jlen != (size_t)argc)
4099
0
        return FALSE;
4100
0
      if (jlen == 0)
4101
0
        continue;
4102
4103
0
      const char** argv = (const char**)calloc(jlen, sizeof(char*));
4104
0
      if (!argv)
4105
0
        return FALSE;
4106
0
      for (size_t y = 0; y < jlen; y++)
4107
0
      {
4108
0
        WINPR_JSON* item = WINPR_JSON_GetArrayItem(array, y);
4109
0
        if (!item || !WINPR_JSON_IsString(item))
4110
0
        {
4111
0
          free((void*)argv);
4112
0
          return FALSE;
4113
0
        }
4114
0
        argv[y] = WINPR_JSON_GetStringValue(item);
4115
0
      }
4116
4117
0
      ADDIN_ARGV* cval = freerdp_addin_argv_new(jlen, argv);
4118
0
      free((void*)argv);
4119
0
      if (!cval)
4120
0
        return FALSE;
4121
0
      const BOOL rc = freerdp_settings_set_pointer_array(settings, key, x, cval);
4122
0
      freerdp_addin_argv_free(cval);
4123
0
      if (!rc)
4124
0
        return FALSE;
4125
0
    }
4126
0
  }
4127
0
  return TRUE;
4128
0
}
4129
4130
static char* get_string(const WINPR_JSON* json, const char* key)
4131
0
{
4132
0
  WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(json, key);
4133
0
  if (!item || !WINPR_JSON_IsString(item))
4134
0
    return nullptr;
4135
0
  const char* str = WINPR_JSON_GetStringValue(item);
4136
0
  return WINPR_CAST_CONST_PTR_AWAY(str, char*);
4137
0
}
4138
4139
static BOOL get_bool(const WINPR_JSON* json, const char* key)
4140
0
{
4141
0
  WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(json, key);
4142
0
  if (!item || !WINPR_JSON_IsBool(item))
4143
0
    return FALSE;
4144
0
  return WINPR_JSON_IsTrue(item);
4145
0
}
4146
4147
static BOOL device_from_json_item(rdpSettings* settings, FreeRDP_Settings_Keys_Pointer key,
4148
                                  size_t offset, const WINPR_JSON* val)
4149
0
{
4150
0
  if (!val || !WINPR_JSON_IsObject(val))
4151
0
    return FALSE;
4152
4153
0
  union
4154
0
  {
4155
0
    RDPDR_DEVICE base;
4156
0
    RDPDR_PARALLEL parallel;
4157
0
    RDPDR_SERIAL serial;
4158
0
    RDPDR_SMARTCARD smartcard;
4159
0
    RDPDR_PRINTER printer;
4160
0
    RDPDR_DRIVE drive;
4161
0
    RDPDR_DEVICE device;
4162
0
  } device;
4163
4164
0
  memset(&device, 0, sizeof(device));
4165
4166
0
  errno = 0;
4167
0
  device.base.Id = (uint32_t)uint_from_json(val, "Id", UINT32_MAX);
4168
0
  device.base.Type = (uint32_t)uint_from_json(val, "Type", UINT32_MAX);
4169
0
  if (errno != 0)
4170
0
    return FALSE;
4171
0
  device.base.Name = get_string(val, "Name");
4172
0
  if (!device.base.Name)
4173
0
    return FALSE;
4174
4175
0
  switch (device.base.Type)
4176
0
  {
4177
0
    case RDPDR_DTYP_SERIAL:
4178
0
      device.serial.Path = get_string(val, "Path");
4179
0
      device.serial.Driver = get_string(val, "Driver");
4180
0
      device.serial.Permissive = get_string(val, "Permissive");
4181
0
      break;
4182
0
    case RDPDR_DTYP_PARALLEL:
4183
0
      device.parallel.Path = get_string(val, "Path");
4184
0
      break;
4185
0
    case RDPDR_DTYP_PRINT:
4186
0
      device.printer.DriverName = get_string(val, "DriverName");
4187
0
      device.printer.IsDefault = get_bool(val, "IsDefault");
4188
0
      break;
4189
0
    case RDPDR_DTYP_FILESYSTEM:
4190
0
      device.drive.Path = get_string(val, "Path");
4191
0
      device.drive.automount = get_bool(val, "automount");
4192
0
      break;
4193
0
    case RDPDR_DTYP_SMARTCARD:
4194
0
    default:
4195
0
      break;
4196
0
  }
4197
0
  return freerdp_settings_set_pointer_array(settings, key, offset, &device);
4198
0
}
4199
4200
static BOOL device_array_from_json(rdpSettings* settings, const WINPR_JSON* json,
4201
                                   FreeRDP_Settings_Keys_Pointer key)
4202
0
{
4203
0
  if (WINPR_JSON_IsNull(json))
4204
0
    return freerdp_settings_set_pointer(settings, key, nullptr);
4205
4206
0
  if (!WINPR_JSON_IsArray(json))
4207
0
    return FALSE;
4208
4209
0
  size_t len = WINPR_JSON_GetArraySize(json);
4210
0
  if (!freerdp_settings_set_pointer_len(settings, key, nullptr, len))
4211
0
    return FALSE;
4212
4213
0
  for (size_t x = 0; x < len; x++)
4214
0
  {
4215
0
    WINPR_JSON* val = WINPR_JSON_GetArrayItem(json, x);
4216
0
    if (!device_from_json_item(settings, key, x, val))
4217
0
      return FALSE;
4218
0
  }
4219
0
  return TRUE;
4220
0
}
4221
4222
static const char* pem_from_json(const WINPR_JSON* jval, size_t* plen, BOOL* pvalid)
4223
0
{
4224
0
  WINPR_ASSERT(jval);
4225
0
  WINPR_ASSERT(plen);
4226
0
  WINPR_ASSERT(pvalid);
4227
4228
0
  *pvalid = FALSE;
4229
0
  *plen = 0;
4230
4231
0
  if (WINPR_JSON_IsNull(jval))
4232
0
    return nullptr;
4233
4234
0
  size_t len = WINPR_JSON_GetArraySize(jval);
4235
0
  if (len == 0)
4236
0
  {
4237
0
    *pvalid = TRUE;
4238
0
    return nullptr;
4239
0
  }
4240
4241
0
  WINPR_JSON* item = WINPR_JSON_GetArrayItem(jval, 0);
4242
0
  if (!item)
4243
0
    return nullptr;
4244
0
  if (!WINPR_JSON_IsString(item))
4245
0
    return nullptr;
4246
4247
0
  *plen = len;
4248
0
  *pvalid = TRUE;
4249
0
  return WINPR_JSON_GetStringValue(item);
4250
0
}
4251
4252
static BOOL deserialize_pointer(const WINPR_JSON* json, rdpSettings* settings,
4253
                                FreeRDP_Settings_Keys_Pointer id)
4254
0
{
4255
0
  const char* name = freerdp_settings_get_name_for_key(id);
4256
0
  if (!name)
4257
0
    return FALSE;
4258
4259
0
  if (!WINPR_JSON_HasObjectItem(json, name))
4260
0
    return FALSE;
4261
4262
0
  WINPR_JSON* jval = WINPR_JSON_GetObjectItemCaseSensitive(json, name);
4263
0
  if (!WINPR_JSON_IsNull(jval) && !WINPR_JSON_IsArray(jval))
4264
0
    return FALSE;
4265
4266
0
  switch (id)
4267
0
  {
4268
0
    case FreeRDP_instance:
4269
0
      return uintptr_from_array(settings, jval);
4270
0
    case FreeRDP_ServerRandom:
4271
0
    case FreeRDP_ServerCertificate:
4272
0
    case FreeRDP_ClientRandom:
4273
0
    case FreeRDP_RedirectionPassword:
4274
0
    case FreeRDP_RedirectionGuid:
4275
0
    case FreeRDP_LoadBalanceInfo:
4276
0
    case FreeRDP_RedirectionTsvUrl:
4277
0
    case FreeRDP_OrderSupport:
4278
0
    case FreeRDP_Password51:
4279
0
      return val_from_array(settings, jval, id, 1);
4280
0
    case FreeRDP_ReceivedCapabilities:
4281
0
      return val_from_array(settings, jval, id, 1);
4282
0
    case FreeRDP_ClientTimeZone:
4283
0
      return ts_info_array_from_json(settings, id, jval);
4284
0
    case FreeRDP_GlyphCache:
4285
0
      return glyph_cache_def_array_from_json(settings, id, jval);
4286
0
    case FreeRDP_FragCache:
4287
0
      return glyph_cache_def_array_from_json(settings, id, jval);
4288
0
    case FreeRDP_BitmapCacheV2CellInfo:
4289
0
      return bitmap_cache_v2_array_from_json(settings, id, jval);
4290
0
    case FreeRDP_ClientAutoReconnectCookie:
4291
0
      return client_cookie_array_from_json(settings, id, jval);
4292
0
    case FreeRDP_ServerAutoReconnectCookie:
4293
0
      return server_cookie_array_from_json(settings, id, jval);
4294
0
    case FreeRDP_MonitorDefArray:
4295
0
      return monitor_def_array_from_json(settings, id, jval);
4296
0
    case FreeRDP_ChannelDefArray:
4297
0
      return channel_def_array_from_json(settings, id, jval);
4298
0
    case FreeRDP_MonitorIds:
4299
0
      return val_from_uint32_array(settings, jval, id, FreeRDP_NumMonitorIds);
4300
0
    case FreeRDP_TargetNetPorts:
4301
0
      return val_from_uint32_array(settings, jval, id, FreeRDP_TargetNetAddressCount);
4302
0
    case FreeRDP_ServerLicenseProductIssuers:
4303
0
    case FreeRDP_TargetNetAddresses:
4304
0
      return str_array_from_json(settings, jval, id);
4305
0
    case FreeRDP_ReceivedCapabilityDataSizes:
4306
0
      return val_from_uint32_array(settings, jval, id, FreeRDP_ReceivedCapabilitiesSize);
4307
0
    case FreeRDP_ReceivedCapabilityData:
4308
0
      return caps_data_array_from_json(settings, jval);
4309
0
    case FreeRDP_RedirectionTargetCertificate:
4310
0
    case FreeRDP_RdpServerCertificate:
4311
0
    {
4312
0
      size_t len = 0;
4313
0
      BOOL valid = FALSE;
4314
0
      const char* pem = pem_from_json(jval, &len, &valid);
4315
0
      if (!valid)
4316
0
        return FALSE;
4317
0
      if (!freerdp_settings_set_pointer_len(settings, id, nullptr, len))
4318
0
        return FALSE;
4319
4320
0
      rdpCertificate* cert = nullptr;
4321
0
      if (!pem)
4322
0
        return TRUE;
4323
4324
0
      if (strnlen(pem, 2) == 0)
4325
0
        cert = freerdp_certificate_new();
4326
0
      else
4327
0
        cert = freerdp_certificate_new_from_pem(pem);
4328
0
      if (!cert)
4329
0
        return FALSE;
4330
0
      return freerdp_settings_set_pointer_len(settings, id, cert, 1);
4331
0
    }
4332
0
    case FreeRDP_RdpServerRsaKey:
4333
0
    {
4334
0
      size_t len = 0;
4335
0
      BOOL valid = FALSE;
4336
0
      const char* pem = pem_from_json(jval, &len, &valid);
4337
0
      if (!valid)
4338
0
        return FALSE;
4339
0
      if (!freerdp_settings_set_pointer_len(settings, id, nullptr, len))
4340
0
        return FALSE;
4341
0
      if (!pem)
4342
0
        return TRUE;
4343
4344
0
      rdpPrivateKey* key = freerdp_key_new_from_pem_enc(pem, nullptr);
4345
0
      if (!key)
4346
0
        return FALSE;
4347
0
      return freerdp_settings_set_pointer_len(settings, id, key, 1);
4348
0
    }
4349
0
    case FreeRDP_DeviceArray:
4350
0
      return device_array_from_json(settings, jval, id);
4351
0
    case FreeRDP_StaticChannelArray:
4352
0
    case FreeRDP_DynamicChannelArray:
4353
0
      return addin_argv_from_json(settings, jval, id);
4354
0
    case FreeRDP_POINTER_UNUSED:
4355
0
    default:
4356
0
      return TRUE;
4357
0
  }
4358
0
}
4359
4360
rdpSettings* freerdp_settings_deserialize(const char* jstr, size_t length)
4361
0
{
4362
0
  WINPR_JSON* json = WINPR_JSON_ParseWithLength(jstr, length);
4363
0
  if (!json)
4364
0
    return nullptr;
4365
4366
0
  WINPR_JSON* jbool = WINPR_JSON_GetObjectItemCaseSensitive(
4367
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_BOOL));
4368
0
  WINPR_JSON* juint16 = WINPR_JSON_GetObjectItemCaseSensitive(
4369
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_UINT16));
4370
0
  WINPR_JSON* jint16 = WINPR_JSON_GetObjectItemCaseSensitive(
4371
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_INT16));
4372
0
  WINPR_JSON* juint32 = WINPR_JSON_GetObjectItemCaseSensitive(
4373
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_UINT32));
4374
0
  WINPR_JSON* jint32 = WINPR_JSON_GetObjectItemCaseSensitive(
4375
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_INT32));
4376
0
  WINPR_JSON* juint64 = WINPR_JSON_GetObjectItemCaseSensitive(
4377
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_UINT64));
4378
0
  WINPR_JSON* jint64 = WINPR_JSON_GetObjectItemCaseSensitive(
4379
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_INT64));
4380
0
  WINPR_JSON* jstring = WINPR_JSON_GetObjectItemCaseSensitive(
4381
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_STRING));
4382
0
  WINPR_JSON* jpointer = WINPR_JSON_GetObjectItemCaseSensitive(
4383
0
      json, freerdp_settings_get_type_name_for_type(RDP_SETTINGS_TYPE_POINTER));
4384
4385
0
  rdpSettings* settings = freerdp_settings_new(0);
4386
0
  if (!settings)
4387
0
    goto fail;
4388
0
  if (!jbool || !juint16 || !jint16 || !juint32 || !jint32 || !juint64 || !jint64 || !jstring ||
4389
0
      !jpointer)
4390
0
    goto fail;
4391
4392
0
  for (int x = 0; x < FreeRDP_Settings_StableAPI_MAX; x++)
4393
0
  {
4394
0
    union
4395
0
    {
4396
4397
0
      int s;
4398
0
      FreeRDP_Settings_Keys_Bool b;
4399
0
      FreeRDP_Settings_Keys_Int16 i16;
4400
0
      FreeRDP_Settings_Keys_UInt16 u16;
4401
0
      FreeRDP_Settings_Keys_Int32 i32;
4402
0
      FreeRDP_Settings_Keys_UInt32 u32;
4403
0
      FreeRDP_Settings_Keys_Int64 i64;
4404
0
      FreeRDP_Settings_Keys_UInt64 u64;
4405
0
      FreeRDP_Settings_Keys_String str;
4406
0
      FreeRDP_Settings_Keys_Pointer ptr;
4407
0
    } iter;
4408
0
    iter.s = x;
4409
4410
0
    SSIZE_T type = freerdp_settings_get_type_for_key(iter.s);
4411
0
    switch (type)
4412
0
    {
4413
0
      case RDP_SETTINGS_TYPE_POINTER:
4414
0
        if (!deserialize_pointer(jpointer, settings, iter.ptr))
4415
0
          goto fail;
4416
0
        break;
4417
0
      default:
4418
0
        break;
4419
0
    }
4420
0
  }
4421
4422
0
  for (int x = 0; x < FreeRDP_Settings_StableAPI_MAX; x++)
4423
0
  {
4424
0
    union
4425
0
    {
4426
4427
0
      int s;
4428
0
      FreeRDP_Settings_Keys_Bool b;
4429
0
      FreeRDP_Settings_Keys_Int16 i16;
4430
0
      FreeRDP_Settings_Keys_UInt16 u16;
4431
0
      FreeRDP_Settings_Keys_Int32 i32;
4432
0
      FreeRDP_Settings_Keys_UInt32 u32;
4433
0
      FreeRDP_Settings_Keys_Int64 i64;
4434
0
      FreeRDP_Settings_Keys_UInt64 u64;
4435
0
      FreeRDP_Settings_Keys_String str;
4436
0
      FreeRDP_Settings_Keys_Pointer ptr;
4437
0
    } iter;
4438
0
    iter.s = x;
4439
4440
0
    const char* name = freerdp_settings_get_name_for_key(iter.s);
4441
0
    SSIZE_T type = freerdp_settings_get_type_for_key(iter.s);
4442
0
    switch (type)
4443
0
    {
4444
0
      case RDP_SETTINGS_TYPE_BOOL:
4445
0
      {
4446
0
        WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(jbool, name);
4447
0
        if (!item)
4448
0
          goto fail;
4449
0
        if (!WINPR_JSON_IsBool(item))
4450
0
          goto fail;
4451
0
        const BOOL val = WINPR_JSON_IsTrue(item);
4452
0
        if (!freerdp_settings_set_bool(settings, iter.b, val))
4453
0
          goto fail;
4454
0
      }
4455
0
      break;
4456
0
      case RDP_SETTINGS_TYPE_UINT16:
4457
0
      {
4458
0
        WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(juint16, name);
4459
0
        const uint16_t val = (uint16_t)uint_from_json_item(item, UINT16_MAX);
4460
0
        if (errno != 0)
4461
0
          goto fail;
4462
0
        if (!freerdp_settings_set_uint16(settings, iter.u16, val))
4463
0
          goto fail;
4464
0
      }
4465
0
      break;
4466
0
      case RDP_SETTINGS_TYPE_INT16:
4467
0
      {
4468
0
        WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(jint16, name);
4469
0
        const int16_t val = (int16_t)int_from_json_item(item, INT16_MIN, INT16_MAX);
4470
0
        if (errno != 0)
4471
0
          goto fail;
4472
0
        if (!freerdp_settings_set_int16(settings, iter.i16, val))
4473
0
          goto fail;
4474
0
      }
4475
0
      break;
4476
0
      case RDP_SETTINGS_TYPE_UINT32:
4477
0
      {
4478
0
        WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(juint32, name);
4479
0
        const uint32_t val = (uint32_t)uint_from_json_item(item, UINT32_MAX);
4480
0
        if (errno != 0)
4481
0
          goto fail;
4482
0
        if (!freerdp_settings_set_uint32(settings, iter.u32, val))
4483
0
          goto fail;
4484
0
      }
4485
0
      break;
4486
0
      case RDP_SETTINGS_TYPE_INT32:
4487
0
      {
4488
0
        const int64_t val = int_from_json(jint32, name, INT32_MIN, INT32_MAX);
4489
0
        if (errno != 0)
4490
0
          goto fail;
4491
0
        if (!freerdp_settings_set_int32(settings, iter.i32, (int32_t)val))
4492
0
          goto fail;
4493
0
      }
4494
0
      break;
4495
0
      case RDP_SETTINGS_TYPE_UINT64:
4496
0
      {
4497
0
        const uint64_t val = uint_from_json(juint64, name, UINT64_MAX);
4498
0
        if (errno != 0)
4499
0
          goto fail;
4500
0
        if (!freerdp_settings_set_uint64(settings, iter.u64, val))
4501
0
          goto fail;
4502
0
      }
4503
0
      break;
4504
0
      case RDP_SETTINGS_TYPE_INT64:
4505
0
      {
4506
0
        WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(jint64, name);
4507
0
        const int64_t val = int_from_json_item(item, INT64_MIN, INT64_MAX);
4508
0
        if (errno != 0)
4509
0
          goto fail;
4510
0
        if (!freerdp_settings_set_int64(settings, iter.i64, val))
4511
0
          goto fail;
4512
0
      }
4513
0
      break;
4514
0
      case RDP_SETTINGS_TYPE_STRING:
4515
0
      {
4516
0
        const char* val = nullptr;
4517
0
        WINPR_JSON* item = WINPR_JSON_GetObjectItemCaseSensitive(jstring, name);
4518
0
        if (item && !WINPR_JSON_IsNull(item))
4519
0
        {
4520
0
          if (!WINPR_JSON_IsString(item))
4521
0
            goto fail;
4522
0
          val = WINPR_JSON_GetStringValue(item);
4523
0
          if (!val)
4524
0
            goto fail;
4525
0
        }
4526
0
        if (!freerdp_settings_set_string(settings, iter.str, val))
4527
0
          goto fail;
4528
0
      }
4529
0
      break;
4530
0
      case RDP_SETTINGS_TYPE_POINTER:
4531
0
      default:
4532
0
        break;
4533
0
    }
4534
0
  }
4535
4536
0
  WINPR_JSON_Delete(json);
4537
0
  return settings;
4538
4539
0
fail:
4540
0
  freerdp_settings_free(settings);
4541
0
  WINPR_JSON_Delete(json);
4542
0
  return nullptr;
4543
0
}