/** * FreeRDP: A Remote Desktop Protocol Implementation * Settings Management * * Copyright 2012 Marc-Andre Moreau * Copyright 2016 Armin Novak * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include #include "../core/settings.h" #include "../core/certificate.h" #include #include #include #define TAG FREERDP_TAG("common") BOOL freerdp_addin_argv_add_argument_ex(ADDIN_ARGV* args, const char* argument, size_t len) { char* str; char** new_argv; if (!args || !argument) return FALSE; if (len == 0) len = strlen(argument); new_argv = (char**)realloc(args->argv, sizeof(char*) * (args->argc + 1)); if (!new_argv) return FALSE; args->argv = new_argv; str = calloc(len + 1, sizeof(char)); if (!str) return FALSE; memcpy(str, argument, len); args->argv[args->argc++] = str; return TRUE; } BOOL freerdp_addin_argv_add_argument(ADDIN_ARGV* args, const char* argument) { return freerdp_addin_argv_add_argument_ex(args, argument, 0); } BOOL freerdp_addin_argv_del_argument(ADDIN_ARGV* args, const char* argument) { int x; if (!args || !argument) return FALSE; for (x = 0; x < args->argc; x++) { char* arg = args->argv[x]; if (strcmp(argument, arg) == 0) { free(arg); memmove_s(&args->argv[x], (args->argc - x) * sizeof(char*), &args->argv[x + 1], (args->argc - x - 1) * sizeof(char*)); args->argv[args->argc - 1] = NULL; args->argc--; return TRUE; } } return FALSE; } int freerdp_addin_set_argument(ADDIN_ARGV* args, const char* argument) { int i; if (!args || !argument) return -2; for (i = 0; i < args->argc; i++) { if (strcmp(args->argv[i], argument) == 0) { return 1; } } if (!freerdp_addin_argv_add_argument(args, argument)) return -1; return 0; } int freerdp_addin_replace_argument(ADDIN_ARGV* args, const char* previous, const char* argument) { int i; if (!args || !previous || !argument) return -2; for (i = 0; i < args->argc; i++) { if (strcmp(args->argv[i], previous) == 0) { free(args->argv[i]); if (!(args->argv[i] = _strdup(argument))) return -1; return 1; } } if (!freerdp_addin_argv_add_argument(args, argument)) return -1; return 0; } int freerdp_addin_set_argument_value(ADDIN_ARGV* args, const char* option, const char* value) { BOOL rc; int i; char* p; char* str; size_t length; if (!args || !option || !value) return -2; length = strlen(option) + strlen(value) + 1; str = (char*)calloc(length + 1, sizeof(char)); if (!str) return -1; sprintf_s(str, length + 1, "%s:%s", option, value); for (i = 0; i < args->argc; i++) { p = strchr(args->argv[i], ':'); if (p) { if (strncmp(args->argv[i], option, p - args->argv[i]) == 0) { free(args->argv[i]); args->argv[i] = str; return 1; } } } rc = freerdp_addin_argv_add_argument(args, str); free(str); if (!rc) return -1; return 0; } int freerdp_addin_replace_argument_value(ADDIN_ARGV* args, const char* previous, const char* option, const char* value) { int i; BOOL rc; char* str; size_t length; if (!args || !previous || !option || !value) return -2; length = strlen(option) + strlen(value) + 1; str = (char*)calloc(length + 1, sizeof(char)); if (!str) return -1; sprintf_s(str, length + 1, "%s:%s", option, value); for (i = 0; i < args->argc; i++) { if (strcmp(args->argv[i], previous) == 0) { free(args->argv[i]); args->argv[i] = str; return 1; } } rc = freerdp_addin_argv_add_argument(args, str); free(str); if (!rc) return -1; return 0; } BOOL freerdp_device_collection_add(rdpSettings* settings, RDPDR_DEVICE* device) { UINT32 count; WINPR_ASSERT(settings); WINPR_ASSERT(device); count = settings->DeviceCount + 1; if (freerdp_settings_get_uint32(settings, FreeRDP_DeviceArraySize) < count) { UINT32 new_size; RDPDR_DEVICE** new_array; new_size = freerdp_settings_get_uint32(settings, FreeRDP_DeviceArraySize) * 2; if (new_size == 0) new_size = count * 2; new_array = (RDPDR_DEVICE**)realloc(settings->DeviceArray, new_size * sizeof(RDPDR_DEVICE*)); if (!new_array) return FALSE; settings->DeviceArray = new_array; if (!freerdp_settings_set_uint32(settings, FreeRDP_DeviceArraySize, new_size)) return FALSE; } settings->DeviceArray[settings->DeviceCount++] = device; return TRUE; } RDPDR_DEVICE* freerdp_device_collection_find(rdpSettings* settings, const char* name) { UINT32 index; RDPDR_DEVICE* device; for (index = 0; index < settings->DeviceCount; index++) { device = (RDPDR_DEVICE*)settings->DeviceArray[index]; if (!device->Name) continue; if (strcmp(device->Name, name) == 0) return device; } return NULL; } RDPDR_DEVICE* freerdp_device_collection_find_type(rdpSettings* settings, UINT32 type) { UINT32 index; RDPDR_DEVICE* device; for (index = 0; index < settings->DeviceCount; index++) { device = (RDPDR_DEVICE*)settings->DeviceArray[index]; if (device->Type == type) return device; } return NULL; } RDPDR_DEVICE* freerdp_device_new(UINT32 Type, size_t count, const char* args[]) { size_t size; union { RDPDR_DEVICE* base; RDPDR_DRIVE* drive; RDPDR_SERIAL* serial; RDPDR_PRINTER* printer; RDPDR_PARALLEL* parallel; RDPDR_SMARTCARD* smartcard; } device; device.base = NULL; WINPR_ASSERT(args || (count == 0)); switch (Type) { case RDPDR_DTYP_PRINT: size = sizeof(RDPDR_PRINTER); break; case RDPDR_DTYP_SERIAL: size = sizeof(RDPDR_SERIAL); break; case RDPDR_DTYP_PARALLEL: size = sizeof(RDPDR_PARALLEL); break; case RDPDR_DTYP_SMARTCARD: size = sizeof(RDPDR_SMARTCARD); break; case RDPDR_DTYP_FILESYSTEM: size = sizeof(RDPDR_DRIVE); break; default: goto fail; } device.base = calloc(1, size); if (!device.base) goto fail; device.base->Id = 0; device.base->Type = Type; if (count > 0) { device.base->Name = _strdup(args[0]); if (!device.base->Name) goto fail; switch (Type) { case RDPDR_DTYP_PRINT: if (count > 1) { device.printer->DriverName = _strdup(args[1]); if (!device.printer->DriverName) goto fail; } break; case RDPDR_DTYP_SERIAL: if (count > 1) { device.serial->Path = _strdup(args[1]); if (!device.serial->Path) goto fail; } if (count > 2) { device.serial->Driver = _strdup(args[2]); if (!device.serial->Driver) goto fail; } if (count > 3) { device.serial->Permissive = _strdup(args[1]); if (!device.serial->Permissive) goto fail; } size = sizeof(RDPDR_SERIAL); break; case RDPDR_DTYP_PARALLEL: if (count > 1) { device.parallel->Path = _strdup(args[1]); if (!device.serial->Path) goto fail; } break; case RDPDR_DTYP_SMARTCARD: break; case RDPDR_DTYP_FILESYSTEM: if (count > 1) { device.drive->Path = _strdup(args[1]); if (!device.drive->Path) goto fail; } if (count > 2) device.drive->automount = (args[2] == NULL) ? FALSE : TRUE; break; default: goto fail; } } return device.base; fail: freerdp_device_free(device.base); return NULL; } void freerdp_device_free(RDPDR_DEVICE* device) { union { RDPDR_DEVICE* dev; RDPDR_DRIVE* drive; RDPDR_SERIAL* serial; RDPDR_PRINTER* printer; RDPDR_PARALLEL* parallel; RDPDR_SMARTCARD* smartcard; } cnv; cnv.dev = device; if (!cnv.dev) return; switch (device->Type) { case RDPDR_DTYP_PRINT: free(cnv.printer->DriverName); break; case RDPDR_DTYP_SERIAL: free(cnv.serial->Path); free(cnv.serial->Driver); free(cnv.serial->Permissive); break; case RDPDR_DTYP_PARALLEL: free(cnv.parallel->Path); break; case RDPDR_DTYP_SMARTCARD: break; case RDPDR_DTYP_FILESYSTEM: free(cnv.drive->Path); break; default: break; } free(cnv.dev->Name); free(cnv.dev); } RDPDR_DEVICE* freerdp_device_clone(const RDPDR_DEVICE* device) { union { const RDPDR_DEVICE* dev; const RDPDR_DRIVE* drive; const RDPDR_SERIAL* serial; const RDPDR_PRINTER* printer; const RDPDR_PARALLEL* parallel; const RDPDR_SMARTCARD* smartcard; } src; union { RDPDR_DEVICE* dev; RDPDR_DRIVE* drive; RDPDR_SERIAL* serial; RDPDR_PRINTER* printer; RDPDR_PARALLEL* parallel; RDPDR_SMARTCARD* smartcard; } copy; const char* args[] = { device->Name }; WINPR_ASSERT(device); src.dev = device; copy.dev = freerdp_device_new(device->Type, ARRAYSIZE(args), args); if (!copy.dev) return NULL; copy.dev->Id = device->Id; switch (device->Type) { case RDPDR_DTYP_FILESYSTEM: { if (src.drive->Path) copy.drive->Path = _strdup(src.drive->Path); if (!copy.drive->Path) goto fail; } break; case RDPDR_DTYP_PRINT: { if (copy.printer->DriverName) { copy.printer->DriverName = _strdup(src.printer->DriverName); if (!copy.printer->DriverName) goto fail; } } break; case RDPDR_DTYP_SMARTCARD: break; case RDPDR_DTYP_SERIAL: { if (copy.serial->Path) { copy.serial->Path = _strdup(src.serial->Path); if (!copy.serial->Path) goto fail; } if (copy.serial->Driver) { copy.serial->Driver = _strdup(src.serial->Driver); if (!copy.serial->Driver) goto fail; } if (copy.serial->Permissive) { copy.serial->Permissive = _strdup(src.serial->Permissive); if (!copy.serial->Permissive) goto fail; } } break; case RDPDR_DTYP_PARALLEL: { if (src.parallel->Path) copy.parallel->Path = _strdup(src.parallel->Path); if (!copy.parallel->Path) goto fail; } break; default: WLog_ERR(TAG, "unknown device type %" PRIu32 "", device->Type); break; } return copy.dev; fail: freerdp_device_free(copy.dev); return NULL; } void freerdp_device_collection_free(rdpSettings* settings) { UINT32 index; WINPR_ASSERT(settings); for (index = 0; index < settings->DeviceCount; index++) { RDPDR_DEVICE* device = (RDPDR_DEVICE*)settings->DeviceArray[index]; freerdp_device_free(device); } free(settings->DeviceArray); freerdp_settings_set_uint32(settings, FreeRDP_DeviceArraySize, 0); settings->DeviceArray = NULL; settings->DeviceCount = 0; } BOOL freerdp_static_channel_collection_del(rdpSettings* settings, const char* name) { UINT32 x; const UINT32 count = freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelCount); if (!settings || !settings->StaticChannelArray) return FALSE; for (x = 0; x < count; x++) { ADDIN_ARGV* cur = settings->StaticChannelArray[x]; if (cur && (cur->argc > 0)) { if (strcmp(name, cur->argv[0]) == 0) { memmove_s(&settings->StaticChannelArray[x], (count - x) * sizeof(ADDIN_ARGV*), &settings->StaticChannelArray[x + 1], (count - x - 1) * sizeof(ADDIN_ARGV*)); return freerdp_settings_set_uint32(settings, FreeRDP_StaticChannelCount, count - 1); } } } return FALSE; } BOOL freerdp_static_channel_collection_add(rdpSettings* settings, ADDIN_ARGV* channel) { UINT32 count; WINPR_ASSERT(settings); WINPR_ASSERT(channel); count = freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelCount) + 1; if (freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelArraySize) < count) { UINT32 new_size; ADDIN_ARGV** new_array; new_size = freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelArraySize) * 2; if (new_size == 0) new_size = count * 2; new_array = (ADDIN_ARGV**)realloc(settings->StaticChannelArray, new_size * sizeof(ADDIN_ARGV*)); if (!new_array) return FALSE; settings->StaticChannelArray = new_array; if (!freerdp_settings_set_uint32(settings, FreeRDP_StaticChannelArraySize, new_size)) return FALSE; } count = freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelCount); settings->StaticChannelArray[count++] = channel; return freerdp_settings_set_uint32(settings, FreeRDP_StaticChannelCount, count); } ADDIN_ARGV* freerdp_static_channel_collection_find(rdpSettings* settings, const char* name) { UINT32 index; ADDIN_ARGV* channel; WINPR_ASSERT(settings); WINPR_ASSERT(name); for (index = 0; index < freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelCount); index++) { channel = settings->StaticChannelArray[index]; if (strcmp(channel->argv[0], name) == 0) return channel; } return NULL; } void freerdp_static_channel_collection_free(rdpSettings* settings) { UINT32 i; if (!settings) return; for (i = 0; i < freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelCount); i++) { freerdp_addin_argv_free(settings->StaticChannelArray[i]); } free(settings->StaticChannelArray); freerdp_settings_set_uint32(settings, FreeRDP_StaticChannelArraySize, 0); settings->StaticChannelArray = NULL; freerdp_settings_set_uint32(settings, FreeRDP_StaticChannelCount, 0); } BOOL freerdp_dynamic_channel_collection_del(rdpSettings* settings, const char* name) { UINT32 x; const UINT32 count = freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelCount); if (!settings || !settings->DynamicChannelArray) return FALSE; for (x = 0; x < count; x++) { ADDIN_ARGV* cur = settings->DynamicChannelArray[x]; if (cur && (cur->argc > 0)) { if (strcmp(name, cur->argv[0])) { memmove_s(&settings->DynamicChannelArray[x], (count - x) * sizeof(ADDIN_ARGV*), &settings->DynamicChannelArray[x + 1], (count - x - 1) * sizeof(ADDIN_ARGV*)); return freerdp_settings_set_uint32(settings, FreeRDP_DynamicChannelCount, count - 1); } } } return FALSE; } BOOL freerdp_dynamic_channel_collection_add(rdpSettings* settings, ADDIN_ARGV* channel) { UINT32 count; WINPR_ASSERT(settings); WINPR_ASSERT(channel); count = freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelCount) + 1; if (freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelArraySize) < count) { ADDIN_ARGV** new_array; size_t size = freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelArraySize) * 2; if (size == 0) size = count * 2; new_array = realloc(settings->DynamicChannelArray, sizeof(ADDIN_ARGV*) * size); if (!new_array) return FALSE; settings->DynamicChannelArray = new_array; if (!freerdp_settings_set_uint32(settings, FreeRDP_DynamicChannelArraySize, size)) return FALSE; } count = freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelCount); settings->DynamicChannelArray[count++] = channel; return freerdp_settings_set_uint32(settings, FreeRDP_DynamicChannelCount, count); } ADDIN_ARGV* freerdp_dynamic_channel_collection_find(const rdpSettings* settings, const char* name) { UINT32 index; ADDIN_ARGV* channel; WINPR_ASSERT(settings); WINPR_ASSERT(name); for (index = 0; index < freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelCount); index++) { channel = settings->DynamicChannelArray[index]; if (strcmp(channel->argv[0], name) == 0) return channel; } return NULL; } void freerdp_addin_argv_free(ADDIN_ARGV* args) { int index; if (!args) return; if (args->argv) { for (index = 0; index < args->argc; index++) free(args->argv[index]); free(args->argv); } free(args); } ADDIN_ARGV* freerdp_addin_argv_new(size_t argc, const char* argv[]) { ADDIN_ARGV* args = calloc(1, sizeof(ADDIN_ARGV)); if (!args) return NULL; if (argc == 0) return args; args->argc = argc; args->argv = calloc(argc, sizeof(char*)); if (!args->argv) goto fail; if (argv) { size_t x; for (x = 0; x < argc; x++) { args->argv[x] = _strdup(argv[x]); if (!args->argv[x]) goto fail; } } return args; fail: freerdp_addin_argv_free(args); return NULL; } ADDIN_ARGV* freerdp_addin_argv_clone(const ADDIN_ARGV* args) { union { char** c; const char** cc; } cnv; if (!args) return NULL; cnv.c = args->argv; return freerdp_addin_argv_new(args->argc, cnv.cc); } void freerdp_dynamic_channel_collection_free(rdpSettings* settings) { UINT32 i; for (i = 0; i < freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelCount); i++) { freerdp_addin_argv_free(settings->DynamicChannelArray[i]); } free(settings->DynamicChannelArray); freerdp_settings_set_uint32(settings, FreeRDP_DynamicChannelArraySize, 0); settings->DynamicChannelArray = NULL; freerdp_settings_set_uint32(settings, FreeRDP_DynamicChannelCount, 0); } void freerdp_target_net_addresses_free(rdpSettings* settings) { UINT32 index; for (index = 0; index < settings->TargetNetAddressCount; index++) free(settings->TargetNetAddresses[index]); free(settings->TargetNetAddresses); free(settings->TargetNetPorts); settings->TargetNetAddressCount = 0; settings->TargetNetAddresses = NULL; settings->TargetNetPorts = NULL; } void freerdp_performance_flags_make(rdpSettings* settings) { UINT32 PerformanceFlags = PERF_FLAG_NONE; if (freerdp_settings_get_bool(settings, FreeRDP_AllowFontSmoothing)) PerformanceFlags |= PERF_ENABLE_FONT_SMOOTHING; if (freerdp_settings_get_bool(settings, FreeRDP_AllowDesktopComposition)) PerformanceFlags |= PERF_ENABLE_DESKTOP_COMPOSITION; if (freerdp_settings_get_bool(settings, FreeRDP_DisableWallpaper)) PerformanceFlags |= PERF_DISABLE_WALLPAPER; if (freerdp_settings_get_bool(settings, FreeRDP_DisableFullWindowDrag)) PerformanceFlags |= PERF_DISABLE_FULLWINDOWDRAG; if (freerdp_settings_get_bool(settings, FreeRDP_DisableMenuAnims)) PerformanceFlags |= PERF_DISABLE_MENUANIMATIONS; if (freerdp_settings_get_bool(settings, FreeRDP_DisableThemes)) PerformanceFlags |= PERF_DISABLE_THEMING; freerdp_settings_set_uint32(settings, FreeRDP_PerformanceFlags, PerformanceFlags); } void freerdp_performance_flags_split(rdpSettings* settings) { freerdp_settings_set_bool(settings, FreeRDP_AllowFontSmoothing, (freerdp_settings_get_uint32(settings, FreeRDP_PerformanceFlags) & PERF_ENABLE_FONT_SMOOTHING) ? TRUE : FALSE); freerdp_settings_set_bool(settings, FreeRDP_AllowDesktopComposition, (freerdp_settings_get_uint32(settings, FreeRDP_PerformanceFlags) & PERF_ENABLE_DESKTOP_COMPOSITION) ? TRUE : FALSE); freerdp_settings_set_bool( settings, FreeRDP_DisableWallpaper, (freerdp_settings_get_uint32(settings, FreeRDP_PerformanceFlags) & PERF_DISABLE_WALLPAPER) ? TRUE : FALSE); freerdp_settings_set_bool(settings, FreeRDP_DisableFullWindowDrag, (freerdp_settings_get_uint32(settings, FreeRDP_PerformanceFlags) & PERF_DISABLE_FULLWINDOWDRAG) ? TRUE : FALSE); freerdp_settings_set_bool(settings, FreeRDP_DisableMenuAnims, (freerdp_settings_get_uint32(settings, FreeRDP_PerformanceFlags) & PERF_DISABLE_MENUANIMATIONS) ? TRUE : FALSE); freerdp_settings_set_bool( settings, FreeRDP_DisableThemes, (freerdp_settings_get_uint32(settings, FreeRDP_PerformanceFlags) & PERF_DISABLE_THEMING) ? TRUE : FALSE); } BOOL freerdp_set_gateway_usage_method(rdpSettings* settings, UINT32 GatewayUsageMethod) { if (!freerdp_settings_set_uint32(settings, FreeRDP_GatewayUsageMethod, GatewayUsageMethod)) return FALSE; if (GatewayUsageMethod == TSC_PROXY_MODE_NONE_DIRECT) { if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayEnabled, FALSE) || !freerdp_settings_set_bool(settings, FreeRDP_GatewayBypassLocal, FALSE)) return FALSE; } else if (GatewayUsageMethod == TSC_PROXY_MODE_DIRECT) { if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayEnabled, TRUE) || !freerdp_settings_set_bool(settings, FreeRDP_GatewayBypassLocal, FALSE)) return FALSE; } else if (GatewayUsageMethod == TSC_PROXY_MODE_DETECT) { if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayEnabled, TRUE) || !freerdp_settings_set_bool(settings, FreeRDP_GatewayBypassLocal, TRUE)) return FALSE; } else if (GatewayUsageMethod == TSC_PROXY_MODE_DEFAULT) { /** * This corresponds to "Automatically detect RD Gateway server settings", * which means the client attempts to use gateway group policy settings * http://technet.microsoft.com/en-us/library/cc770601.aspx */ if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayEnabled, FALSE) || !freerdp_settings_set_bool(settings, FreeRDP_GatewayBypassLocal, FALSE)) return FALSE; } else if (GatewayUsageMethod == TSC_PROXY_MODE_NONE_DETECT) { if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayEnabled, FALSE) || !freerdp_settings_set_bool(settings, FreeRDP_GatewayBypassLocal, FALSE)) return FALSE; } return TRUE; } void freerdp_update_gateway_usage_method(rdpSettings* settings, UINT32 GatewayEnabled, UINT32 GatewayBypassLocal) { UINT32 GatewayUsageMethod = 0; if (!GatewayEnabled && !GatewayBypassLocal) GatewayUsageMethod = TSC_PROXY_MODE_NONE_DIRECT; else if (GatewayEnabled && !GatewayBypassLocal) GatewayUsageMethod = TSC_PROXY_MODE_DIRECT; else if (GatewayEnabled && GatewayBypassLocal) GatewayUsageMethod = TSC_PROXY_MODE_DETECT; freerdp_set_gateway_usage_method(settings, GatewayUsageMethod); } #if defined(WITH_FREERDP_DEPRECATED) BOOL freerdp_get_param_bool(const rdpSettings* settings, int id) { return freerdp_settings_get_bool(settings, (size_t)id); } int freerdp_set_param_bool(rdpSettings* settings, int id, BOOL param) { return freerdp_settings_set_bool(settings, (size_t)id, param) ? 0 : -1; } int freerdp_get_param_int(const rdpSettings* settings, int id) { return freerdp_settings_get_int32(settings, (size_t)id); } int freerdp_set_param_int(rdpSettings* settings, int id, int param) { return freerdp_settings_set_int32(settings, (size_t)id, param) ? 0 : -1; } UINT32 freerdp_get_param_uint32(const rdpSettings* settings, int id) { return freerdp_settings_get_uint32(settings, (size_t)id); } int freerdp_set_param_uint32(rdpSettings* settings, int id, UINT32 param) { return freerdp_settings_set_uint32(settings, (size_t)id, param) ? 0 : -1; } UINT64 freerdp_get_param_uint64(const rdpSettings* settings, int id) { return freerdp_settings_get_uint64(settings, (size_t)id); } int freerdp_set_param_uint64(rdpSettings* settings, int id, UINT64 param) { return freerdp_settings_set_uint64(settings, (size_t)id, param) ? 0 : -1; } char* freerdp_get_param_string(const rdpSettings* settings, int id) { return (char*)freerdp_settings_get_string(settings, (size_t)id); } int freerdp_set_param_string(rdpSettings* settings, int id, const char* param) { return freerdp_settings_set_string(settings, (size_t)id, param) ? 0 : -1; } #endif static BOOL value_to_uint(const char* value, ULONGLONG* result, ULONGLONG min, ULONGLONG max) { unsigned long long rc; if (!value || !result) return FALSE; errno = 0; rc = _strtoui64(value, NULL, 0); if (errno != 0) return FALSE; if ((rc < min) || (rc > max)) return FALSE; *result = rc; return TRUE; } static BOOL value_to_int(const char* value, LONGLONG* result, LONGLONG min, LONGLONG max) { long long rc; if (!value || !result) return FALSE; errno = 0; rc = _strtoi64(value, NULL, 0); if (errno != 0) return FALSE; if ((rc < min) || (rc > max)) return FALSE; *result = rc; return TRUE; } static BOOL parsing_fail(const char* key, const char* type, const char* value) { WLog_ERR(TAG, "Failed to parse key [%s] of type [%s]: value [%s]", key, type, value); return FALSE; } BOOL freerdp_settings_set_value_for_name(rdpSettings* settings, const char* name, const char* value) { ULONGLONG uval; LONGLONG ival; SSIZE_T index, type; if (!settings || !name) return FALSE; index = freerdp_settings_get_key_for_name(name); if (index < 0) { WLog_ERR(TAG, "Invalid settings key [%s]", name); return FALSE; } type = freerdp_settings_get_type_for_key((size_t)index); switch (type) { case RDP_SETTINGS_TYPE_BOOL: { BOOL val = _strnicmp(value, "TRUE", 5) == 0; if (!val && _strnicmp(value, "FALSE", 5) != 0) return parsing_fail(name, "BOOL", value); return freerdp_settings_set_bool(settings, index, val); } case RDP_SETTINGS_TYPE_UINT16: if (!value_to_uint(value, &uval, 0, UINT16_MAX)) return parsing_fail(name, "UINT16", value); if (!freerdp_settings_set_uint16(settings, index, uval)) return parsing_fail(name, "UINT16", value); return TRUE; case RDP_SETTINGS_TYPE_INT16: if (!value_to_int(value, &ival, INT16_MIN, INT16_MAX)) return parsing_fail(name, "INT16", value); if (!freerdp_settings_set_int16(settings, index, ival)) return parsing_fail(name, "INT16", value); return TRUE; case RDP_SETTINGS_TYPE_UINT32: if (!value_to_uint(value, &uval, 0, UINT32_MAX)) return parsing_fail(name, "UINT32", value); if (!freerdp_settings_set_uint32(settings, index, uval)) return parsing_fail(name, "UINT32", value); return TRUE; case RDP_SETTINGS_TYPE_INT32: if (!value_to_int(value, &ival, INT32_MIN, INT32_MAX)) return parsing_fail(name, "INT32", value); if (!freerdp_settings_set_int32(settings, index, ival)) return parsing_fail(name, "INT32", value); return TRUE; case RDP_SETTINGS_TYPE_UINT64: if (!value_to_uint(value, &uval, 0, UINT64_MAX)) return parsing_fail(name, "UINT64", value); if (!freerdp_settings_set_uint64(settings, index, uval)) return parsing_fail(name, "UINT64", value); return TRUE; case RDP_SETTINGS_TYPE_INT64: if (!value_to_int(value, &ival, INT64_MIN, INT64_MAX)) return parsing_fail(name, "INT64", value); if (!freerdp_settings_set_int64(settings, index, ival)) return parsing_fail(name, "INT64", value); return TRUE; case RDP_SETTINGS_TYPE_STRING: return freerdp_settings_set_string(settings, index, value); case RDP_SETTINGS_TYPE_POINTER: return parsing_fail(name, "POINTER", value); default: return FALSE; } return FALSE; } static BOOL freerdp_settings_set_pointer_len_(rdpSettings* settings, size_t id, SSIZE_T lenId, const void* data, size_t len, size_t size) { BOOL rc; void* copy; void* old = freerdp_settings_get_pointer_writable(settings, id); free(old); if (!freerdp_settings_set_pointer(settings, id, NULL)) return FALSE; if (lenId >= 0) { if (!freerdp_settings_set_uint32(settings, lenId, 0)) return FALSE; } if (len == 0) return TRUE; copy = calloc(len, size); if (!copy) return FALSE; if (data) memcpy(copy, data, len * size); rc = freerdp_settings_set_pointer(settings, id, copy); if (!rc) { free(copy); return FALSE; } if (lenId < 0) return TRUE; return freerdp_settings_set_uint32(settings, lenId, len); } const void* freerdp_settings_get_pointer(const rdpSettings* settings, size_t id) { return freerdp_settings_get_pointer_writable(settings, id); } BOOL freerdp_settings_set_pointer_len(rdpSettings* settings, size_t id, const void* data, size_t len) { union { const void* cv; void* v; } cnv; cnv.cv = data; if (!settings) return FALSE; switch (id) { case FreeRDP_RdpServerCertificate: certificate_free(settings->RdpServerCertificate); settings->RdpServerCertificate = (rdpCertificate*)cnv.v; return TRUE; case FreeRDP_RdpServerRsaKey: key_free(settings->RdpServerRsaKey); settings->RdpServerRsaKey = (rdpRsaKey*)cnv.v; return TRUE; case FreeRDP_RedirectionPassword: return freerdp_settings_set_pointer_len_( settings, id, FreeRDP_RedirectionPasswordLength, data, len, sizeof(char)); case FreeRDP_RedirectionTsvUrl: return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_RedirectionTsvUrlLength, data, len, sizeof(char)); case FreeRDP_LoadBalanceInfo: return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_LoadBalanceInfoLength, data, len, sizeof(char)); case FreeRDP_ServerRandom: return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_ServerRandomLength, data, len, sizeof(char)); case FreeRDP_ClientRandom: return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_ClientRandomLength, data, len, sizeof(char)); case FreeRDP_ServerCertificate: return freerdp_settings_set_pointer_len_(settings, id, FreeRDP_ServerCertificateLength, data, len, sizeof(char)); case FreeRDP_TargetNetAddresses: if (data == NULL) { freerdp_target_net_addresses_free(settings); if (!freerdp_settings_set_uint32(settings, FreeRDP_TargetNetAddressCount, len)) return FALSE; } return freerdp_settings_set_pointer_len_(settings, FreeRDP_TargetNetAddresses, FreeRDP_TargetNetAddressCount, data, len, sizeof(char)); case FreeRDP_TargetNetPorts: if (data == NULL) { freerdp_target_net_addresses_free(settings); if (!freerdp_settings_set_uint32(settings, FreeRDP_TargetNetAddressCount, len)) return FALSE; } return freerdp_settings_set_pointer_len_(settings, FreeRDP_TargetNetPorts, FreeRDP_TargetNetAddressCount, data, len, sizeof(char)); case FreeRDP_ChannelDefArray: if (!freerdp_settings_set_pointer_len_(settings, FreeRDP_ChannelDefArray, FreeRDP_ChannelDefArraySize, data, len, sizeof(CHANNEL_DEF))) return FALSE; return freerdp_settings_set_uint32(settings, FreeRDP_ChannelCount, len); case FreeRDP_ClientAutoReconnectCookie: case FreeRDP_ServerAutoReconnectCookie: case FreeRDP_MonitorDefArray: case FreeRDP_ReceivedCapabilities: case FreeRDP_OrderSupport: case FreeRDP_ClientTimeZone: case FreeRDP_BitmapCacheV2CellInfo: case FreeRDP_GlyphCache: case FreeRDP_FragCache: return freerdp_settings_set_pointer_len_(settings, id, -1, data, len, sizeof(char)); case FreeRDP_MonitorIds: return freerdp_settings_set_pointer_len_( settings, FreeRDP_MonitorIds, FreeRDP_NumMonitorIds, data, len, sizeof(UINT32)); default: if ((data == NULL) && (len == 0)) { freerdp_settings_set_pointer(settings, id, NULL); } else WLog_WARN(TAG, "Invalid id %" PRIuz " for %s", id, __FUNCTION__); return FALSE; } } void* freerdp_settings_get_pointer_array_writable(const rdpSettings* settings, size_t id, size_t offset) { if (!settings) return NULL; switch (id) { case FreeRDP_OrderSupport: if (offset >= 32) return FALSE; return &settings->OrderSupport[offset]; case FreeRDP_MonitorIds: if (offset > freerdp_settings_get_uint32(settings, FreeRDP_NumMonitorIds)) return NULL; return &settings->MonitorIds[offset]; case FreeRDP_MonitorDefArray: if (offset > freerdp_settings_get_uint32(settings, FreeRDP_MonitorDefArraySize)) return NULL; return &settings->MonitorDefArray[offset]; case FreeRDP_ChannelDefArray: if (offset > freerdp_settings_get_uint32(settings, FreeRDP_ChannelDefArraySize)) return NULL; return &settings->ChannelDefArray[offset]; case FreeRDP_DeviceArray: if (offset > freerdp_settings_get_uint32(settings, FreeRDP_DeviceArraySize)) return NULL; return &settings->DeviceArray[offset]; case FreeRDP_StaticChannelArray: if (offset > freerdp_settings_get_uint32(settings, FreeRDP_StaticChannelArraySize)) return NULL; return settings->StaticChannelArray[offset]; case FreeRDP_DynamicChannelArray: if (offset > freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelArraySize)) return NULL; return settings->DynamicChannelArray[offset]; case FreeRDP_FragCache: if (offset >= 1) return NULL; return &settings->FragCache[offset]; case FreeRDP_GlyphCache: if (offset >= 10) return NULL; return &settings->GlyphCache[offset]; case FreeRDP_BitmapCacheV2CellInfo: /* TODO: BitmapCacheV2NumCells should be limited to 4 if (offset > freerdp_settings_get_uint32(settings, FreeRDP_BitmapCacheV2NumCells)) return NULL; */ return &settings->BitmapCacheV2CellInfo[offset]; case FreeRDP_ReceivedCapabilities: if (offset > settings->ReceivedCapabilitiesSize) return 0; return &settings->ReceivedCapabilities[offset]; default: WLog_WARN(TAG, "Invalid id %" PRIuz " for %s", id, __FUNCTION__); return NULL; } } BOOL freerdp_settings_set_pointer_array(rdpSettings* settings, size_t id, size_t offset, const void* data) { if (!settings) return FALSE; switch (id) { case FreeRDP_TargetNetAddresses: if ((offset >= settings->TargetNetAddressCount) || !data) return FALSE; free(settings->TargetNetAddresses[offset]); settings->TargetNetAddresses[offset] = _strdup((const char*)data); return settings->TargetNetAddresses[offset] != NULL; case FreeRDP_TargetNetPorts: if ((offset >= settings->TargetNetAddressCount) || !data) return FALSE; settings->TargetNetPorts[offset] = *((const UINT32*)data); return TRUE; case FreeRDP_BitmapCacheV2CellInfo: if ((offset > 5) || !data) return FALSE; settings->BitmapCacheV2CellInfo[offset] = *(const BITMAP_CACHE_V2_CELL_INFO*)data; return TRUE; case FreeRDP_OrderSupport: if ((offset >= 32) || !data) return FALSE; settings->OrderSupport[offset] = *(const BOOL*)data; return TRUE; case FreeRDP_GlyphCache: if ((offset >= 10) || !data) return FALSE; settings->GlyphCache[offset] = *(const GLYPH_CACHE_DEFINITION*)data; return TRUE; case FreeRDP_FragCache: if ((offset >= 1) || !data) return FALSE; settings->FragCache[offset] = *(const GLYPH_CACHE_DEFINITION*)data; return TRUE; case FreeRDP_MonitorIds: if ((offset > freerdp_settings_get_uint32(settings, FreeRDP_NumMonitorIds)) || !data) return FALSE; settings->MonitorIds[offset] = *(const UINT32*)data; return TRUE; case FreeRDP_ChannelDefArray: if (offset > freerdp_settings_get_uint32(settings, FreeRDP_ChannelDefArraySize)) return FALSE; settings->ChannelDefArray[offset] = *(const CHANNEL_DEF*)data; return TRUE; default: WLog_WARN(TAG, "Invalid id %" PRIuz " for %s", id, __FUNCTION__); return FALSE; } } const void* freerdp_settings_get_pointer_array(const rdpSettings* settings, size_t id, size_t offset) { return freerdp_settings_get_pointer_array_writable(settings, id, offset); } UINT32 freerdp_settings_get_codecs_flags(const rdpSettings* settings) { UINT32 flags = FREERDP_CODEC_ALL; if (settings->RemoteFxCodec == FALSE) { flags &= ~FREERDP_CODEC_REMOTEFX; } if (settings->NSCodec == FALSE) { flags &= ~FREERDP_CODEC_NSCODEC; } /*TODO: check other codecs flags */ return flags; } #if defined(WITH_FREERDP_DEPRECATED) ADDIN_ARGV* freerdp_static_channel_clone(ADDIN_ARGV* channel) { return freerdp_addin_argv_clone(channel); } ADDIN_ARGV* freerdp_dynamic_channel_clone(ADDIN_ARGV* channel) { return freerdp_addin_argv_clone(channel); } #endif BOOL freerdp_target_net_addresses_copy(rdpSettings* settings, char** addresses, UINT32 count) { UINT32 i; WINPR_ASSERT(settings); WINPR_ASSERT(addresses); freerdp_target_net_addresses_free(settings); settings->TargetNetAddressCount = count; settings->TargetNetAddresses = (char**)calloc(settings->TargetNetAddressCount, sizeof(char*)); if (!settings->TargetNetAddresses) { freerdp_target_net_addresses_free(settings); return FALSE; } for (i = 0; i < settings->TargetNetAddressCount; i++) { settings->TargetNetAddresses[i] = _strdup(addresses[i]); if (!settings->TargetNetAddresses[i]) { freerdp_target_net_addresses_free(settings); return FALSE; } } return TRUE; }