Coverage Report

Created: 2025-07-11 06:46

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