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