FreeRDP/client/common/file.c

2724 lines
90 KiB
C
Raw Normal View History

/**
* FreeRDP: A Remote Desktop Protocol Implementation
* .rdp file
*
* Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
*
* 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.
*/
2022-02-16 13:20:38 +03:00
#include <freerdp/config.h>
2017-11-14 18:10:52 +03:00
#include <errno.h>
#include <ctype.h>
2021-04-12 12:06:45 +03:00
#include <stdlib.h>
#include <winpr/string.h>
#include <winpr/file.h>
2017-11-14 18:10:52 +03:00
2023-03-14 12:39:18 +03:00
#include <freerdp/client.h>
#include <freerdp/client/file.h>
#include <freerdp/client/cmdline.h>
2021-04-12 12:06:45 +03:00
#include <freerdp/channels/urbdrc.h>
#include <freerdp/channels/rdpecam.h>
#include <freerdp/channels/location.h>
2021-04-12 12:06:45 +03:00
/**
* Remote Desktop Plus - Overview of .rdp file settings:
* http://www.donkz.nl/files/rdpsettings.html
*
* RDP Settings for Remote Desktop Services in Windows Server 2008 R2:
* http://technet.microsoft.com/en-us/library/ff393699/
2021-04-12 12:06:45 +03:00
*
* https://docs.microsoft.com/en-us/windows-server/remote/remote-desktop-services/clients/rdp-files
*/
#include <stdio.h>
#include <string.h>
#include <winpr/wtypes.h>
#include <winpr/crt.h>
#include <winpr/path.h>
2014-09-12 19:13:01 +04:00
#include <freerdp/log.h>
#define TAG CLIENT_TAG("common")
/*#define DEBUG_CLIENT_FILE 1*/
2012-10-28 20:12:36 +04:00
static const BYTE BOM_UTF16_LE[2] = { 0xFF, 0xFE };
2019-07-17 16:48:26 +03:00
#define INVALID_INTEGER_VALUE 0xFFFFFFFF
2019-07-17 16:48:26 +03:00
#define RDP_FILE_LINE_FLAG_FORMATTED 0x00000001
#define RDP_FILE_LINE_FLAG_STANDARD 0x00000002
#define RDP_FILE_LINE_FLAG_TYPE_STRING 0x00000010
#define RDP_FILE_LINE_FLAG_TYPE_INTEGER 0x00000020
#define RDP_FILE_LINE_FLAG_TYPE_BINARY 0x00000040
2019-07-15 11:08:39 +03:00
struct rdp_file_line
{
char* name;
LPSTR sValue;
PBYTE bValue;
size_t index;
long iValue;
DWORD flags;
2019-07-15 11:08:39 +03:00
int valueLength;
};
typedef struct rdp_file_line rdpFileLine;
struct rdp_file
{
2019-07-17 16:48:26 +03:00
DWORD UseMultiMon; /* use multimon */
2021-04-12 12:06:45 +03:00
LPSTR SelectedMonitors; /* selectedmonitors */
DWORD MaximizeToCurrentDisplays; /* maximizetocurrentdisplays */
DWORD SingleMonInWindowedMode; /* singlemoninwindowedmode */
2019-07-17 16:48:26 +03:00
DWORD ScreenModeId; /* screen mode id */
DWORD SpanMonitors; /* span monitors */
DWORD SmartSizing; /* smartsizing */
2021-04-12 12:06:45 +03:00
DWORD DynamicResolution; /* dynamic resolution */
2019-07-17 16:48:26 +03:00
DWORD EnableSuperSpan; /* enablesuperpan */
2019-07-15 11:08:39 +03:00
DWORD SuperSpanAccelerationFactor; /* superpanaccelerationfactor */
DWORD DesktopWidth; /* desktopwidth */
DWORD DesktopHeight; /* desktopheight */
DWORD DesktopSizeId; /* desktop size id */
DWORD SessionBpp; /* session bpp */
2021-04-12 12:06:45 +03:00
DWORD DesktopScaleFactor; /* desktopscalefactor */
2019-07-15 11:08:39 +03:00
2019-07-17 16:48:26 +03:00
DWORD Compression; /* compression */
DWORD KeyboardHook; /* keyboardhook */
2019-07-15 11:08:39 +03:00
DWORD DisableCtrlAltDel; /* disable ctrl+alt+del */
DWORD AudioMode; /* audiomode */
DWORD AudioQualityMode; /* audioqualitymode */
DWORD AudioCaptureMode; /* audiocapturemode */
2021-04-12 12:06:45 +03:00
DWORD EncodeRedirectedVideoCapture; /* encode redirected video capture */
DWORD RedirectedVideoCaptureEncodingQuality; /* redirected video capture encoding quality */
DWORD VideoPlaybackMode; /* videoplaybackmode */
2019-07-15 11:08:39 +03:00
DWORD ConnectionType; /* connection type */
2019-07-17 16:48:26 +03:00
DWORD NetworkAutoDetect; /* networkautodetect */
2019-07-15 11:08:39 +03:00
DWORD BandwidthAutoDetect; /* bandwidthautodetect */
2019-07-17 16:48:26 +03:00
DWORD PinConnectionBar; /* pinconnectionbar */
2019-07-15 11:08:39 +03:00
DWORD DisplayConnectionBar; /* displayconnectionbar */
2019-07-17 16:48:26 +03:00
DWORD WorkspaceId; /* workspaceid */
2019-07-15 11:08:39 +03:00
DWORD EnableWorkspaceReconnect; /* enableworkspacereconnect */
2019-07-17 16:48:26 +03:00
DWORD DisableWallpaper; /* disable wallpaper */
DWORD AllowFontSmoothing; /* allow font smoothing */
2019-07-15 11:08:39 +03:00
DWORD AllowDesktopComposition; /* allow desktop composition */
2019-07-17 16:48:26 +03:00
DWORD DisableFullWindowDrag; /* disable full window drag */
DWORD DisableMenuAnims; /* disable menu anims */
DWORD DisableThemes; /* disable themes */
DWORD DisableCursorSetting; /* disable cursor setting */
2019-07-15 11:08:39 +03:00
2019-07-17 16:48:26 +03:00
DWORD BitmapCacheSize; /* bitmapcachesize */
2019-07-15 11:08:39 +03:00
DWORD BitmapCachePersistEnable; /* bitmapcachepersistenable */
2019-11-06 17:24:51 +03:00
DWORD ServerPort; /* server port */
2019-07-17 16:48:26 +03:00
LPSTR Username; /* username */
LPSTR Domain; /* domain */
LPSTR Password; /*password*/
2019-07-15 11:08:39 +03:00
PBYTE Password51; /* password 51 */
2019-07-17 16:48:26 +03:00
LPSTR FullAddress; /* full address */
2019-07-15 11:08:39 +03:00
LPSTR AlternateFullAddress; /* alternate full address */
2019-07-17 16:48:26 +03:00
LPSTR UsbDevicesToRedirect; /* usbdevicestoredirect */
2019-07-17 16:48:26 +03:00
DWORD RedirectDrives; /* redirectdrives */
DWORD RedirectPrinters; /* redirectprinters */
DWORD RedirectComPorts; /* redirectcomports */
DWORD RedirectLocation; /* redirectlocation */
2019-07-17 16:48:26 +03:00
DWORD RedirectSmartCards; /* redirectsmartcards */
DWORD RedirectWebauthN; /* redirectwebauthn */
LPSTR RedirectCameras; /* camerastoredirect */
2019-07-17 16:48:26 +03:00
DWORD RedirectClipboard; /* redirectclipboard */
DWORD RedirectPosDevices; /* redirectposdevices */
DWORD RedirectDirectX; /* redirectdirectx */
DWORD DisablePrinterRedirection; /* disableprinterredirection */
2019-07-15 11:08:39 +03:00
DWORD DisableClipboardRedirection; /* disableclipboardredirection */
2019-07-17 16:48:26 +03:00
DWORD ConnectToConsole; /* connect to console */
DWORD AdministrativeSession; /* administrative session */
2019-07-15 11:08:39 +03:00
DWORD AutoReconnectionEnabled; /* autoreconnection enabled */
DWORD AutoReconnectMaxRetries; /* autoreconnect max retries */
2019-07-17 16:48:26 +03:00
DWORD PublicMode; /* public mode */
DWORD AuthenticationLevel; /* authentication level */
DWORD PromptCredentialOnce; /* promptcredentialonce */
DWORD PromptForCredentials; /* prompt for credentials */
2019-07-15 11:08:39 +03:00
DWORD NegotiateSecurityLayer; /* negotiate security layer */
2019-07-17 16:48:26 +03:00
DWORD EnableCredSSPSupport; /* enablecredsspsupport */
DWORD EnableRdsAadAuth; /* enablerdsaadauth */
2019-07-17 16:48:26 +03:00
2019-11-06 17:24:51 +03:00
DWORD RemoteApplicationMode; /* remoteapplicationmode */
LPSTR LoadBalanceInfo; /* loadbalanceinfo */
2019-07-17 16:48:26 +03:00
LPSTR RemoteApplicationName; /* remoteapplicationname */
LPSTR RemoteApplicationIcon; /* remoteapplicationicon */
LPSTR RemoteApplicationProgram; /* remoteapplicationprogram */
LPSTR RemoteApplicationFile; /* remoteapplicationfile */
LPSTR RemoteApplicationGuid; /* remoteapplicationguid */
LPSTR RemoteApplicationCmdLine; /* remoteapplicationcmdline */
DWORD RemoteApplicationExpandCmdLine; /* remoteapplicationexpandcmdline */
2019-07-15 11:08:39 +03:00
DWORD RemoteApplicationExpandWorkingDir; /* remoteapplicationexpandworkingdir */
2019-07-17 16:48:26 +03:00
DWORD DisableConnectionSharing; /* disableconnectionsharing */
DWORD DisableRemoteAppCapsCheck; /* disableremoteappcapscheck */
2019-07-15 11:08:39 +03:00
2019-07-17 16:48:26 +03:00
LPSTR AlternateShell; /* alternate shell */
2019-07-15 11:08:39 +03:00
LPSTR ShellWorkingDirectory; /* shell working directory */
2019-07-17 16:48:26 +03:00
LPSTR GatewayHostname; /* gatewayhostname */
DWORD GatewayUsageMethod; /* gatewayusagemethod */
2019-07-15 11:08:39 +03:00
DWORD GatewayProfileUsageMethod; /* gatewayprofileusagemethod */
2019-07-17 16:48:26 +03:00
DWORD GatewayCredentialsSource; /* gatewaycredentialssource */
2019-07-15 11:08:39 +03:00
2023-07-26 19:36:34 +03:00
LPSTR ResourceProvider; /* resourceprovider */
LPSTR WvdEndpointPool; /* wvd endpoint pool */
LPSTR geo; /* geo */
LPSTR armpath; /* armpath */
LPSTR aadtenantid; /* aadtenantid" */
LPSTR diagnosticserviceurl; /* diagnosticserviceurl */
LPSTR hubdiscoverygeourl; /* hubdiscoverygeourl" */
LPSTR activityhint; /* activityhint */
2019-07-15 11:08:39 +03:00
DWORD UseRedirectionServerName; /* use redirection server name */
2019-11-06 17:24:51 +03:00
LPSTR GatewayAccessToken; /* gatewayaccesstoken */
2019-07-15 11:08:39 +03:00
2019-07-17 16:48:26 +03:00
LPSTR DrivesToRedirect; /* drivestoredirect */
2019-07-15 11:08:39 +03:00
LPSTR DevicesToRedirect; /* devicestoredirect */
2019-07-17 16:48:26 +03:00
LPSTR WinPosStr; /* winposstr */
2019-07-15 11:08:39 +03:00
LPSTR PreconnectionBlob; /* pcb */
LPSTR KdcProxyName; /* kdcproxyname */
DWORD RdgIsKdcProxy; /* rdgiskdcproxy */
DWORD align1;
size_t lineCount;
size_t lineSize;
2019-07-15 11:08:39 +03:00
rdpFileLine* lines;
ADDIN_ARGV* args;
void* context;
DWORD flags;
2019-07-15 11:08:39 +03:00
};
static const char key_str_username[] = "username";
static const char key_str_domain[] = "domain";
static const char key_str_password[] = "password";
static const char key_str_full_address[] = "full address";
static const char key_str_alternate_full_address[] = "alternate full address";
static const char key_str_usbdevicestoredirect[] = "usbdevicestoredirect";
static const char key_str_camerastoredirect[] = "camerastoredirect";
static const char key_str_loadbalanceinfo[] = "loadbalanceinfo";
static const char key_str_remoteapplicationname[] = "remoteapplicationname";
static const char key_str_remoteapplicationicon[] = "remoteapplicationicon";
static const char key_str_remoteapplicationprogram[] = "remoteapplicationprogram";
static const char key_str_remoteapplicationfile[] = "remoteapplicationfile";
static const char key_str_remoteapplicationguid[] = "remoteapplicationguid";
static const char key_str_remoteapplicationcmdline[] = "remoteapplicationcmdline";
static const char key_str_alternate_shell[] = "alternate shell";
static const char key_str_shell_working_directory[] = "shell working directory";
static const char key_str_gatewayhostname[] = "gatewayhostname";
static const char key_str_gatewayaccesstoken[] = "gatewayaccesstoken";
static const char key_str_resourceprovider[] = "resourceprovider";
static const char str_resourceprovider_arm[] = "arm";
static const char key_str_kdcproxyname[] = "kdcproxyname";
static const char key_str_drivestoredirect[] = "drivestoredirect";
static const char key_str_devicestoredirect[] = "devicestoredirect";
static const char key_str_winposstr[] = "winposstr";
static const char key_str_pcb[] = "pcb";
static const char key_str_selectedmonitors[] = "selectedmonitors";
static const char key_str_wvd[] = "wvd endpoint pool";
static const char key_str_geo[] = "geo";
static const char key_str_armpath[] = "armpath";
static const char key_str_aadtenantid[] = "aadtenantid";
static const char key_str_diagnosticserviceurl[] = "diagnosticserviceurl";
static const char key_str_hubdiscoverygeourl[] = "hubdiscoverygeourl";
static const char key_str_activityhint[] = "activityhint";
2023-07-05 11:18:30 +03:00
static const char key_int_rdgiskdcproxy[] = "rdgiskdcproxy";
static const char key_int_use_redirection_server_name[] = "use redirection server name";
static const char key_int_gatewaycredentialssource[] = "gatewaycredentialssource";
static const char key_int_gatewayprofileusagemethod[] = "gatewayprofileusagemethod";
static const char key_int_gatewayusagemethod[] = "gatewayusagemethod";
static const char key_int_disableremoteappcapscheck[] = "disableremoteappcapscheck";
static const char key_int_disableconnectionsharing[] = "disableconnectionsharing";
static const char key_int_remoteapplicationexpandworkingdir[] = "remoteapplicationexpandworkingdir";
static const char key_int_remoteapplicationexpandcmdline[] = "remoteapplicationexpandcmdline";
static const char key_int_remoteapplicationmode[] = "remoteapplicationmode";
static const char key_int_enablecredsspsupport[] = "enablecredsspsupport";
static const char key_int_enablerdsaadauth[] = "enablerdsaadauth";
2023-07-05 11:18:30 +03:00
static const char key_int_negotiate_security_layer[] = "negotiate security layer";
static const char key_int_prompt_for_credentials[] = "prompt for credentials";
static const char key_int_promptcredentialonce[] = "promptcredentialonce";
static const char key_int_authentication_level[] = "authentication level";
static const char key_int_public_mode[] = "public mode";
static const char key_int_autoreconnect_max_retries[] = "autoreconnect max retries";
static const char key_int_autoreconnection_enabled[] = "autoreconnection enabled";
static const char key_int_administrative_session[] = "administrative session";
static const char key_int_connect_to_console[] = "connect to console";
static const char key_int_disableclipboardredirection[] = "disableclipboardredirection";
static const char key_int_disableprinterredirection[] = "disableprinterredirection";
static const char key_int_redirectdirectx[] = "redirectdirectx";
static const char key_int_redirectposdevices[] = "redirectposdevices";
static const char key_int_redirectclipboard[] = "redirectclipboard";
static const char key_int_redirectsmartcards[] = "redirectsmartcards";
static const char key_int_redirectcomports[] = "redirectcomports";
static const char key_int_redirectlocation[] = "redirectlocation";
2023-07-05 11:18:30 +03:00
static const char key_int_redirectprinters[] = "redirectprinters";
static const char key_int_redirectdrives[] = "redirectdrives";
static const char key_int_server_port[] = "server port";
static const char key_int_bitmapcachepersistenable[] = "bitmapcachepersistenable";
static const char key_int_bitmapcachesize[] = "bitmapcachesize";
static const char key_int_disable_cursor_setting[] = "disable cursor setting";
static const char key_int_disable_themes[] = "disable themes";
static const char key_int_disable_menu_anims[] = "disable menu anims";
2023-07-05 11:18:30 +03:00
static const char key_int_disable_full_window_drag[] = "disable full window drag";
static const char key_int_allow_desktop_composition[] = "allow desktop composition";
static const char key_int_allow_font_smoothing[] = "allow font smoothing";
static const char key_int_disable_wallpaper[] = "disable wallpaper";
static const char key_int_enableworkspacereconnect[] = "enableworkspacereconnect";
static const char key_int_workspaceid[] = "workspaceid";
static const char key_int_displayconnectionbar[] = "displayconnectionbar";
static const char key_int_pinconnectionbar[] = "pinconnectionbar";
static const char key_int_bandwidthautodetect[] = "bandwidthautodetect";
static const char key_int_networkautodetect[] = "networkautodetect";
static const char key_int_connection_type[] = "connection type";
static const char key_int_videoplaybackmode[] = "videoplaybackmode";
static const char key_int_redirected_video_capture_encoding_quality[] =
"redirected video capture encoding quality";
static const char key_int_encode_redirected_video_capture[] = "encode redirected video capture";
static const char key_int_audiocapturemode[] = "audiocapturemode";
static const char key_int_audioqualitymode[] = "audioqualitymode";
static const char key_int_audiomode[] = "audiomode";
static const char key_int_disable_ctrl_alt_del[] = "disable ctrl+alt+del";
static const char key_int_keyboardhook[] = "keyboardhook";
static const char key_int_compression[] = "compression";
static const char key_int_desktopscalefactor[] = "desktopscalefactor";
static const char key_int_session_bpp[] = "session bpp";
static const char key_int_desktop_size_id[] = "desktop size id";
static const char key_int_desktopheight[] = "desktopheight";
static const char key_int_desktopwidth[] = "desktopwidth";
static const char key_int_superpanaccelerationfactor[] = "superpanaccelerationfactor";
static const char key_int_enablesuperpan[] = "enablesuperpan";
static const char key_int_dynamic_resolution[] = "dynamic resolution";
static const char key_int_smart_sizing[] = "smart sizing";
static const char key_int_span_monitors[] = "span monitors";
static const char key_int_screen_mode_id[] = "screen mode id";
static const char key_int_singlemoninwindowedmode[] = "singlemoninwindowedmode";
static const char key_int_maximizetocurrentdisplays[] = "maximizetocurrentdisplays";
static const char key_int_use_multimon[] = "use multimon";
static const char key_int_redirectwebauthn[] = "redirectwebauthn";
static BOOL utils_str_is_empty(const char* str)
{
if (!str)
return TRUE;
if (strlen(str) == 0)
return TRUE;
return FALSE;
}
static SSIZE_T freerdp_client_rdp_file_add_line(rdpFile* file);
static rdpFileLine* freerdp_client_rdp_file_find_line_by_name(const rdpFile* file,
const char* name);
static void freerdp_client_file_string_check_free(LPSTR str);
static BOOL freerdp_client_rdp_file_find_integer_entry(rdpFile* file, const char* name,
DWORD** outValue, rdpFileLine** outLine)
{
WINPR_ASSERT(file);
WINPR_ASSERT(name);
WINPR_ASSERT(outValue);
WINPR_ASSERT(outLine);
2012-10-28 20:12:36 +04:00
*outValue = NULL;
*outLine = NULL;
2019-07-18 14:53:30 +03:00
if (_stricmp(name, key_int_use_multimon) == 0)
*outValue = &file->UseMultiMon;
else if (_stricmp(name, key_int_maximizetocurrentdisplays) == 0)
*outValue = &file->MaximizeToCurrentDisplays;
else if (_stricmp(name, key_int_singlemoninwindowedmode) == 0)
*outValue = &file->SingleMonInWindowedMode;
else if (_stricmp(name, key_int_screen_mode_id) == 0)
*outValue = &file->ScreenModeId;
else if (_stricmp(name, key_int_span_monitors) == 0)
*outValue = &file->SpanMonitors;
else if (_stricmp(name, key_int_smart_sizing) == 0)
*outValue = &file->SmartSizing;
else if (_stricmp(name, key_int_dynamic_resolution) == 0)
*outValue = &file->DynamicResolution;
else if (_stricmp(name, key_int_enablesuperpan) == 0)
*outValue = &file->EnableSuperSpan;
else if (_stricmp(name, key_int_superpanaccelerationfactor) == 0)
*outValue = &file->SuperSpanAccelerationFactor;
else if (_stricmp(name, key_int_desktopwidth) == 0)
*outValue = &file->DesktopWidth;
else if (_stricmp(name, key_int_desktopheight) == 0)
*outValue = &file->DesktopHeight;
else if (_stricmp(name, key_int_desktop_size_id) == 0)
*outValue = &file->DesktopSizeId;
else if (_stricmp(name, key_int_session_bpp) == 0)
*outValue = &file->SessionBpp;
else if (_stricmp(name, key_int_desktopscalefactor) == 0)
*outValue = &file->DesktopScaleFactor;
else if (_stricmp(name, key_int_compression) == 0)
*outValue = &file->Compression;
else if (_stricmp(name, key_int_keyboardhook) == 0)
*outValue = &file->KeyboardHook;
else if (_stricmp(name, key_int_disable_ctrl_alt_del) == 0)
*outValue = &file->DisableCtrlAltDel;
else if (_stricmp(name, key_int_audiomode) == 0)
*outValue = &file->AudioMode;
else if (_stricmp(name, key_int_audioqualitymode) == 0)
*outValue = &file->AudioQualityMode;
else if (_stricmp(name, key_int_audiocapturemode) == 0)
*outValue = &file->AudioCaptureMode;
else if (_stricmp(name, key_int_encode_redirected_video_capture) == 0)
*outValue = &file->EncodeRedirectedVideoCapture;
else if (_stricmp(name, key_int_redirected_video_capture_encoding_quality) == 0)
*outValue = &file->RedirectedVideoCaptureEncodingQuality;
else if (_stricmp(name, key_int_videoplaybackmode) == 0)
*outValue = &file->VideoPlaybackMode;
else if (_stricmp(name, key_int_connection_type) == 0)
*outValue = &file->ConnectionType;
else if (_stricmp(name, key_int_networkautodetect) == 0)
*outValue = &file->NetworkAutoDetect;
else if (_stricmp(name, key_int_bandwidthautodetect) == 0)
*outValue = &file->BandwidthAutoDetect;
else if (_stricmp(name, key_int_pinconnectionbar) == 0)
*outValue = &file->PinConnectionBar;
else if (_stricmp(name, key_int_displayconnectionbar) == 0)
*outValue = &file->DisplayConnectionBar;
else if (_stricmp(name, key_int_workspaceid) == 0)
*outValue = &file->WorkspaceId;
else if (_stricmp(name, key_int_enableworkspacereconnect) == 0)
*outValue = &file->EnableWorkspaceReconnect;
else if (_stricmp(name, key_int_disable_wallpaper) == 0)
*outValue = &file->DisableWallpaper;
else if (_stricmp(name, key_int_allow_font_smoothing) == 0)
*outValue = &file->AllowFontSmoothing;
else if (_stricmp(name, key_int_allow_desktop_composition) == 0)
*outValue = &file->AllowDesktopComposition;
else if (_stricmp(name, key_int_disable_full_window_drag) == 0)
*outValue = &file->DisableFullWindowDrag;
else if (_stricmp(name, key_int_disable_menu_anims) == 0)
*outValue = &file->DisableMenuAnims;
else if (_stricmp(name, key_int_disable_themes) == 0)
*outValue = &file->DisableThemes;
else if (_stricmp(name, key_int_disable_cursor_setting) == 0)
*outValue = &file->DisableCursorSetting;
else if (_stricmp(name, key_int_bitmapcachesize) == 0)
*outValue = &file->BitmapCacheSize;
else if (_stricmp(name, key_int_bitmapcachepersistenable) == 0)
*outValue = &file->BitmapCachePersistEnable;
else if (_stricmp(name, key_int_server_port) == 0)
*outValue = &file->ServerPort;
else if (_stricmp(name, key_int_redirectdrives) == 0)
*outValue = &file->RedirectDrives;
else if (_stricmp(name, key_int_redirectprinters) == 0)
*outValue = &file->RedirectPrinters;
else if (_stricmp(name, key_int_redirectcomports) == 0)
*outValue = &file->RedirectComPorts;
else if (_stricmp(name, key_int_redirectlocation) == 0)
*outValue = &file->RedirectLocation;
else if (_stricmp(name, key_int_redirectsmartcards) == 0)
*outValue = &file->RedirectSmartCards;
else if (_stricmp(name, key_int_redirectclipboard) == 0)
*outValue = &file->RedirectClipboard;
else if (_stricmp(name, key_int_redirectposdevices) == 0)
*outValue = &file->RedirectPosDevices;
else if (_stricmp(name, key_int_redirectdirectx) == 0)
*outValue = &file->RedirectDirectX;
else if (_stricmp(name, key_int_disableprinterredirection) == 0)
*outValue = &file->DisablePrinterRedirection;
else if (_stricmp(name, key_int_disableclipboardredirection) == 0)
*outValue = &file->DisableClipboardRedirection;
else if (_stricmp(name, key_int_connect_to_console) == 0)
*outValue = &file->ConnectToConsole;
else if (_stricmp(name, key_int_administrative_session) == 0)
*outValue = &file->AdministrativeSession;
else if (_stricmp(name, key_int_autoreconnection_enabled) == 0)
*outValue = &file->AutoReconnectionEnabled;
else if (_stricmp(name, key_int_autoreconnect_max_retries) == 0)
*outValue = &file->AutoReconnectMaxRetries;
else if (_stricmp(name, key_int_public_mode) == 0)
*outValue = &file->PublicMode;
else if (_stricmp(name, key_int_authentication_level) == 0)
*outValue = &file->AuthenticationLevel;
else if (_stricmp(name, key_int_promptcredentialonce) == 0)
*outValue = &file->PromptCredentialOnce;
else if ((_stricmp(name, key_int_prompt_for_credentials) == 0))
*outValue = &file->PromptForCredentials;
else if (_stricmp(name, key_int_negotiate_security_layer) == 0)
*outValue = &file->NegotiateSecurityLayer;
else if (_stricmp(name, key_int_enablecredsspsupport) == 0)
*outValue = &file->EnableCredSSPSupport;
else if (_stricmp(name, key_int_enablerdsaadauth) == 0)
*outValue = &file->EnableRdsAadAuth;
else if (_stricmp(name, key_int_remoteapplicationmode) == 0)
*outValue = &file->RemoteApplicationMode;
else if (_stricmp(name, key_int_remoteapplicationexpandcmdline) == 0)
*outValue = &file->RemoteApplicationExpandCmdLine;
else if (_stricmp(name, key_int_remoteapplicationexpandworkingdir) == 0)
*outValue = &file->RemoteApplicationExpandWorkingDir;
else if (_stricmp(name, key_int_disableconnectionsharing) == 0)
*outValue = &file->DisableConnectionSharing;
else if (_stricmp(name, key_int_disableremoteappcapscheck) == 0)
*outValue = &file->DisableRemoteAppCapsCheck;
else if (_stricmp(name, key_int_gatewayusagemethod) == 0)
*outValue = &file->GatewayUsageMethod;
else if (_stricmp(name, key_int_gatewayprofileusagemethod) == 0)
*outValue = &file->GatewayProfileUsageMethod;
else if (_stricmp(name, key_int_gatewaycredentialssource) == 0)
*outValue = &file->GatewayCredentialsSource;
else if (_stricmp(name, key_int_use_redirection_server_name) == 0)
*outValue = &file->UseRedirectionServerName;
else if (_stricmp(name, key_int_rdgiskdcproxy) == 0)
*outValue = &file->RdgIsKdcProxy;
else if (_stricmp(name, key_int_redirectwebauthn) == 0)
*outValue = &file->RedirectWebauthN;
else
{
rdpFileLine* line = freerdp_client_rdp_file_find_line_by_name(file, name);
if (!line)
return FALSE;
if (!(line->flags & RDP_FILE_LINE_FLAG_TYPE_INTEGER))
return FALSE;
*outLine = line;
}
return TRUE;
}
static BOOL freerdp_client_rdp_file_find_string_entry(rdpFile* file, const char* name,
LPSTR** outValue, rdpFileLine** outLine)
{
WINPR_ASSERT(file);
WINPR_ASSERT(name);
WINPR_ASSERT(outValue);
WINPR_ASSERT(outLine);
*outValue = NULL;
*outLine = NULL;
if (_stricmp(name, key_str_username) == 0)
*outValue = &file->Username;
else if (_stricmp(name, key_str_domain) == 0)
*outValue = &file->Domain;
else if (_stricmp(name, key_str_password) == 0)
*outValue = &file->Password;
else if (_stricmp(name, key_str_full_address) == 0)
*outValue = &file->FullAddress;
else if (_stricmp(name, key_str_alternate_full_address) == 0)
*outValue = &file->AlternateFullAddress;
else if (_stricmp(name, key_str_usbdevicestoredirect) == 0)
*outValue = &file->UsbDevicesToRedirect;
else if (_stricmp(name, key_str_camerastoredirect) == 0)
*outValue = &file->RedirectCameras;
else if (_stricmp(name, key_str_loadbalanceinfo) == 0)
*outValue = &file->LoadBalanceInfo;
else if (_stricmp(name, key_str_remoteapplicationname) == 0)
*outValue = &file->RemoteApplicationName;
else if (_stricmp(name, key_str_remoteapplicationicon) == 0)
*outValue = &file->RemoteApplicationIcon;
else if (_stricmp(name, key_str_remoteapplicationprogram) == 0)
*outValue = &file->RemoteApplicationProgram;
else if (_stricmp(name, key_str_remoteapplicationfile) == 0)
*outValue = &file->RemoteApplicationFile;
else if (_stricmp(name, key_str_remoteapplicationguid) == 0)
*outValue = &file->RemoteApplicationGuid;
else if (_stricmp(name, key_str_remoteapplicationcmdline) == 0)
*outValue = &file->RemoteApplicationCmdLine;
else if (_stricmp(name, key_str_alternate_shell) == 0)
*outValue = &file->AlternateShell;
else if (_stricmp(name, key_str_shell_working_directory) == 0)
*outValue = &file->ShellWorkingDirectory;
else if (_stricmp(name, key_str_gatewayhostname) == 0)
*outValue = &file->GatewayHostname;
else if (_stricmp(name, key_str_resourceprovider) == 0)
*outValue = &file->ResourceProvider;
else if (_stricmp(name, key_str_wvd) == 0)
*outValue = &file->WvdEndpointPool;
else if (_stricmp(name, key_str_geo) == 0)
*outValue = &file->geo;
else if (_stricmp(name, key_str_armpath) == 0)
*outValue = &file->armpath;
else if (_stricmp(name, key_str_aadtenantid) == 0)
*outValue = &file->aadtenantid;
else if (_stricmp(name, key_str_diagnosticserviceurl) == 0)
*outValue = &file->diagnosticserviceurl;
else if (_stricmp(name, key_str_hubdiscoverygeourl) == 0)
*outValue = &file->hubdiscoverygeourl;
else if (_stricmp(name, key_str_activityhint) == 0)
*outValue = &file->activityhint;
else if (_stricmp(name, key_str_gatewayaccesstoken) == 0)
*outValue = &file->GatewayAccessToken;
else if (_stricmp(name, key_str_kdcproxyname) == 0)
*outValue = &file->KdcProxyName;
else if (_stricmp(name, key_str_drivestoredirect) == 0)
*outValue = &file->DrivesToRedirect;
else if (_stricmp(name, key_str_devicestoredirect) == 0)
*outValue = &file->DevicesToRedirect;
else if (_stricmp(name, key_str_winposstr) == 0)
*outValue = &file->WinPosStr;
else if (_stricmp(name, key_str_pcb) == 0)
*outValue = &file->PreconnectionBlob;
else if (_stricmp(name, key_str_selectedmonitors) == 0)
*outValue = &file->SelectedMonitors;
else
{
rdpFileLine* line = freerdp_client_rdp_file_find_line_by_name(file, name);
if (!line)
return FALSE;
if (!(line->flags & RDP_FILE_LINE_FLAG_TYPE_STRING))
return FALSE;
*outLine = line;
}
return TRUE;
}
/*
* Set an integer in a rdpFile
*
* @return FALSE if a standard name was set, TRUE for a non-standard name, FALSE on error
*
*/
static BOOL freerdp_client_rdp_file_set_integer(rdpFile* file, const char* name, long value)
{
DWORD* targetValue = NULL;
rdpFileLine* line = NULL;
#ifdef DEBUG_CLIENT_FILE
WLog_DBG(TAG, "%s:i:%ld", name, value);
#endif
if (value < 0)
return FALSE;
if (!freerdp_client_rdp_file_find_integer_entry(file, name, &targetValue, &line))
{
SSIZE_T index = freerdp_client_rdp_file_add_line(file);
if (index == -1)
2019-07-18 14:53:30 +03:00
return FALSE;
line = &file->lines[index];
}
if (targetValue)
{
*targetValue = (DWORD)value;
return TRUE;
}
if (line)
{
free(line->name);
line->name = _strdup(name);
if (!line->name)
{
free(line->name);
line->name = NULL;
return FALSE;
}
line->iValue = value;
line->flags = RDP_FILE_LINE_FLAG_FORMATTED;
line->flags |= RDP_FILE_LINE_FLAG_TYPE_INTEGER;
line->valueLength = 0;
return TRUE;
}
return FALSE;
}
static BOOL freerdp_client_parse_rdp_file_integer(rdpFile* file, const char* name,
const char* value)
{
char* endptr = NULL;
long ivalue = 0;
2017-11-14 18:10:52 +03:00
errno = 0;
ivalue = strtol(value, &endptr, 0);
2017-11-14 18:10:52 +03:00
2019-11-06 17:24:51 +03:00
if ((endptr == NULL) || (errno != 0) || (endptr == value) || (ivalue > INT32_MAX) ||
(ivalue < INT32_MIN))
{
if (file->flags & RDP_FILE_FLAG_PARSE_INT_RELAXED)
{
WLog_WARN(TAG, "Integer option %s has invalid value %s, using default", name, value);
return TRUE;
}
else
{
WLog_ERR(TAG, "Failed to convert RDP file integer option %s [value=%s]", name, value);
return FALSE;
}
}
return freerdp_client_rdp_file_set_integer(file, name, ivalue);
}
/** set a string value in the provided rdp file context
*
* @param file rdpFile
* @param name name of the string
* @param value value of the string to set
* @return 0 on success, 1 if the key wasn't found (not a standard key), -1 on error
*/
static BOOL freerdp_client_rdp_file_set_string(rdpFile* file, const char* name, const char* value)
{
LPSTR* targetValue = NULL;
rdpFileLine* line = NULL;
2012-10-28 20:12:36 +04:00
#ifdef DEBUG_CLIENT_FILE
2019-07-17 16:48:26 +03:00
WLog_DBG(TAG, "%s:s:%s", name, value);
2012-10-28 20:12:36 +04:00
#endif
if (!name || !value)
return FALSE;
if (!freerdp_client_rdp_file_find_string_entry(file, name, &targetValue, &line))
{
SSIZE_T index = freerdp_client_rdp_file_add_line(file);
if (index == -1)
return FALSE;
line = &file->lines[index];
}
2015-06-23 13:17:37 +03:00
if (targetValue)
{
*targetValue = _strdup(value);
if (!(*targetValue))
return FALSE;
return TRUE;
}
if (line)
{
free(line->name);
free(line->sValue);
line->name = _strdup(name);
line->sValue = _strdup(value);
if (!line->name || !line->sValue)
{
free(line->name);
free(line->sValue);
line->name = NULL;
line->sValue = NULL;
return FALSE;
}
line->flags = RDP_FILE_LINE_FLAG_FORMATTED;
line->flags |= RDP_FILE_LINE_FLAG_TYPE_STRING;
line->valueLength = 0;
return TRUE;
}
return FALSE;
}
static BOOL freerdp_client_add_option(rdpFile* file, const char* option)
{
return freerdp_addin_argv_add_argument(file->args, option);
}
static SSIZE_T freerdp_client_rdp_file_add_line(rdpFile* file)
{
SSIZE_T index = (SSIZE_T)file->lineCount;
while ((file->lineCount + 1) > file->lineSize)
{
size_t new_size = 0;
rdpFileLine* new_line = NULL;
new_size = file->lineSize * 2;
2019-07-17 16:48:26 +03:00
new_line = (rdpFileLine*)realloc(file->lines, new_size * sizeof(rdpFileLine));
if (!new_line)
return -1;
file->lines = new_line;
file->lineSize = new_size;
}
ZeroMemory(&(file->lines[file->lineCount]), sizeof(rdpFileLine));
file->lines[file->lineCount].index = (size_t)index;
(file->lineCount)++;
return index;
}
static BOOL freerdp_client_parse_rdp_file_string(rdpFile* file, char* name, char* value)
{
return freerdp_client_rdp_file_set_string(file, name, value);
}
static BOOL freerdp_client_parse_rdp_file_option(rdpFile* file, const char* option)
{
return freerdp_client_add_option(file, option);
}
2019-07-17 16:48:26 +03:00
BOOL freerdp_client_parse_rdp_file_buffer(rdpFile* file, const BYTE* buffer, size_t size)
{
return freerdp_client_parse_rdp_file_buffer_ex(file, buffer, size, NULL);
}
static BOOL trim(char** strptr)
{
char* start = NULL;
char* str = NULL;
char* end = NULL;
start = str = *strptr;
if (!str)
return TRUE;
if (!(~((size_t)str)))
return TRUE;
end = str + strlen(str) - 1;
2019-11-06 17:24:51 +03:00
while (isspace(*str))
str++;
2019-11-06 17:24:51 +03:00
while ((end > str) && isspace(*end))
end--;
end[1] = '\0';
if (start == str)
*strptr = str;
else
{
*strptr = _strdup(str);
free(start);
return *strptr != NULL;
}
return TRUE;
}
static BOOL trim_strings(rdpFile* file)
{
if (!trim(&file->Username))
return FALSE;
if (!trim(&file->Domain))
return FALSE;
if (!trim(&file->AlternateFullAddress))
return FALSE;
if (!trim(&file->FullAddress))
return FALSE;
if (!trim(&file->UsbDevicesToRedirect))
return FALSE;
if (!trim(&file->RedirectCameras))
return FALSE;
if (!trim(&file->LoadBalanceInfo))
return FALSE;
if (!trim(&file->GatewayHostname))
return FALSE;
if (!trim(&file->GatewayAccessToken))
return FALSE;
if (!trim(&file->RemoteApplicationName))
return FALSE;
if (!trim(&file->RemoteApplicationIcon))
return FALSE;
if (!trim(&file->RemoteApplicationProgram))
return FALSE;
if (!trim(&file->RemoteApplicationFile))
return FALSE;
if (!trim(&file->RemoteApplicationGuid))
return FALSE;
if (!trim(&file->RemoteApplicationCmdLine))
return FALSE;
if (!trim(&file->AlternateShell))
return FALSE;
if (!trim(&file->ShellWorkingDirectory))
return FALSE;
if (!trim(&file->DrivesToRedirect))
return FALSE;
if (!trim(&file->DevicesToRedirect))
return FALSE;
if (!trim(&file->DevicesToRedirect))
return FALSE;
if (!trim(&file->WinPosStr))
return FALSE;
if (!trim(&file->PreconnectionBlob))
return FALSE;
if (!trim(&file->KdcProxyName))
return FALSE;
2021-04-12 12:06:45 +03:00
if (!trim(&file->SelectedMonitors))
return FALSE;
for (size_t i = 0; i < file->lineCount; ++i)
{
rdpFileLine* curLine = &file->lines[i];
if (curLine->flags & RDP_FILE_LINE_FLAG_TYPE_STRING)
{
if (!trim(&curLine->sValue))
return FALSE;
}
}
return TRUE;
}
2019-07-17 16:48:26 +03:00
BOOL freerdp_client_parse_rdp_file_buffer_ex(rdpFile* file, const BYTE* buffer, size_t size,
rdp_file_fkt_parse parse)
{
2017-02-15 16:22:15 +03:00
BOOL rc = FALSE;
size_t length = 0;
char* line = NULL;
char* type = NULL;
char* context = NULL;
char* d1 = NULL;
char* d2 = NULL;
char* beg = NULL;
char* name = NULL;
char* value = NULL;
char* copy = NULL;
if (!file)
return FALSE;
if (size < 2)
return FALSE;
if ((buffer[0] == BOM_UTF16_LE[0]) && (buffer[1] == BOM_UTF16_LE[1]))
{
2019-07-18 14:53:30 +03:00
LPCWSTR uc = (LPCWSTR)(&buffer[2]);
size = size / sizeof(WCHAR) - 1;
copy = ConvertWCharNToUtf8Alloc(uc, size, NULL);
if (!copy)
{
WLog_ERR(TAG, "Failed to convert RDP file from UCS2 to UTF8");
return FALSE;
}
}
else
{
copy = calloc(1, size + sizeof(BYTE));
if (!copy)
return FALSE;
memcpy(copy, buffer, size);
}
line = strtok_s(copy, "\r\n", &context);
while (line)
{
length = strnlen(line, size);
if (length > 1)
{
beg = line;
if (beg[0] == '/')
{
if (!freerdp_client_parse_rdp_file_option(file, line))
goto fail;
goto next_line; /* FreeRDP option */
}
d1 = strchr(line, ':');
if (!d1)
goto next_line; /* not first delimiter */
type = &d1[1];
d2 = strchr(type, ':');
if (!d2)
goto next_line; /* no second delimiter */
if ((d2 - d1) != 2)
goto next_line; /* improper type length */
*d1 = 0;
*d2 = 0;
name = beg;
value = &d2[1];
if (parse && parse(file->context, name, *type, value))
{
}
else if (*type == 'i')
{
/* integer type */
if (!freerdp_client_parse_rdp_file_integer(file, name, value))
2017-02-15 16:22:15 +03:00
goto fail;
}
else if (*type == 's')
{
/* string type */
if (!freerdp_client_parse_rdp_file_string(file, name, value))
2017-02-15 16:22:15 +03:00
goto fail;
}
else if (*type == 'b')
{
/* binary type */
WLog_ERR(TAG, "Unsupported RDP file binary option %s [value=%s]", name, value);
}
}
next_line:
line = strtok_s(NULL, "\r\n", &context);
}
rc = trim_strings(file);
2017-02-15 16:22:15 +03:00
fail:
free(copy);
2017-02-15 16:22:15 +03:00
return rc;
}
BOOL freerdp_client_parse_rdp_file(rdpFile* file, const char* name)
{
2019-07-17 16:48:26 +03:00
return freerdp_client_parse_rdp_file_ex(file, name, NULL);
}
2019-07-17 16:48:26 +03:00
BOOL freerdp_client_parse_rdp_file_ex(rdpFile* file, const char* name, rdp_file_fkt_parse parse)
{
BOOL status = 0;
BYTE* buffer = NULL;
FILE* fp = NULL;
size_t read_size = 0;
INT64 file_size = 0;
const char* fname = name;
if (!file || !name)
return FALSE;
if (_strnicmp(fname, "file://", 7) == 0)
fname = &name[7];
fp = winpr_fopen(fname, "r");
if (!fp)
{
WLog_ERR(TAG, "Failed to open RDP file %s", name);
return FALSE;
}
(void)_fseeki64(fp, 0, SEEK_END);
file_size = _ftelli64(fp);
(void)_fseeki64(fp, 0, SEEK_SET);
if (file_size < 1)
2013-08-28 18:13:09 +04:00
{
WLog_ERR(TAG, "RDP file %s is empty", name);
(void)fclose(fp);
return FALSE;
2013-08-28 18:13:09 +04:00
}
2019-07-18 14:53:30 +03:00
buffer = (BYTE*)malloc((size_t)file_size + 2);
if (!buffer)
{
(void)fclose(fp);
return FALSE;
}
2019-07-18 14:53:30 +03:00
read_size = fread(buffer, (size_t)file_size, 1, fp);
if (!read_size)
{
if (!ferror(fp))
2019-07-18 14:53:30 +03:00
read_size = (size_t)file_size;
}
(void)fclose(fp);
if (read_size < 1)
{
WLog_ERR(TAG, "Could not read from RDP file %s", name);
free(buffer);
return FALSE;
}
buffer[file_size] = '\0';
buffer[file_size + 1] = '\0';
2019-07-18 14:53:30 +03:00
status = freerdp_client_parse_rdp_file_buffer_ex(file, buffer, (size_t)file_size, parse);
free(buffer);
return status;
}
2021-04-12 12:06:45 +03:00
static INLINE BOOL FILE_POPULATE_STRING(char** _target, const rdpSettings* _settings,
2023-10-16 20:37:36 +03:00
FreeRDP_Settings_Keys_String _option)
2021-04-12 12:06:45 +03:00
{
WINPR_ASSERT(_target);
WINPR_ASSERT(_settings);
2021-04-12 12:06:45 +03:00
const char* str = freerdp_settings_get_string(_settings, _option);
2021-04-12 12:06:45 +03:00
freerdp_client_file_string_check_free(*_target);
*_target = (void*)~((size_t)NULL);
2021-04-12 12:06:45 +03:00
if (str)
{
*_target = _strdup(str);
if (!_target)
return FALSE;
}
return TRUE;
}
static char* freerdp_client_channel_args_to_string(const rdpSettings* settings, const char* channel,
const char* option)
{
ADDIN_ARGV* args = freerdp_dynamic_channel_collection_find(settings, channel);
2021-04-12 12:06:45 +03:00
const char* filters[] = { option };
if (!args || (args->argc < 2))
return NULL;
2021-04-12 12:06:45 +03:00
return CommandLineToCommaSeparatedValuesEx(args->argc - 1, args->argv + 1, filters,
ARRAYSIZE(filters));
}
2023-10-16 20:37:36 +03:00
static BOOL rdp_opt_duplicate(const rdpSettings* _settings, FreeRDP_Settings_Keys_String _id,
char** _key)
{
WINPR_ASSERT(_settings);
WINPR_ASSERT(_key);
const char* tmp = freerdp_settings_get_string(_settings, _id);
if (tmp)
{
*_key = _strdup(tmp);
if (!*_key)
return FALSE;
}
return TRUE;
}
BOOL freerdp_client_populate_rdp_file_from_settings(rdpFile* file, const rdpSettings* settings)
{
FreeRDP_Settings_Keys_String index = FreeRDP_STRING_UNUSED;
UINT32 LoadBalanceInfoLength = 0;
2021-04-12 12:06:45 +03:00
const char* GatewayHostname = NULL;
char* redirectCameras = NULL;
2021-04-12 12:06:45 +03:00
if (!file || !settings)
return FALSE;
2021-04-12 12:06:45 +03:00
if (!FILE_POPULATE_STRING(&file->Domain, settings, FreeRDP_Domain) ||
!FILE_POPULATE_STRING(&file->Username, settings, FreeRDP_Username) ||
!FILE_POPULATE_STRING(&file->Password, settings, FreeRDP_Password) ||
!FILE_POPULATE_STRING(&file->FullAddress, settings, FreeRDP_ServerHostname) ||
!FILE_POPULATE_STRING(&file->AlternateFullAddress, settings, FreeRDP_ServerHostname) ||
!FILE_POPULATE_STRING(&file->AlternateShell, settings, FreeRDP_AlternateShell) ||
!FILE_POPULATE_STRING(&file->DrivesToRedirect, settings, FreeRDP_DrivesToRedirect))
return FALSE;
file->ServerPort = freerdp_settings_get_uint32(settings, FreeRDP_ServerPort);
file->DesktopWidth = freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth);
file->DesktopHeight = freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight);
file->SessionBpp = freerdp_settings_get_uint32(settings, FreeRDP_ColorDepth);
file->DesktopScaleFactor = freerdp_settings_get_uint32(settings, FreeRDP_DesktopScaleFactor);
file->DynamicResolution = freerdp_settings_get_bool(settings, FreeRDP_DynamicResolutionUpdate);
file->VideoPlaybackMode = freerdp_settings_get_bool(settings, FreeRDP_SupportVideoOptimized);
// TODO file->MaximizeToCurrentDisplays;
// TODO file->SingleMonInWindowedMode;
// TODO file->EncodeRedirectedVideoCapture;
// TODO file->RedirectedVideoCaptureEncodingQuality;
file->ConnectToConsole = freerdp_settings_get_bool(settings, FreeRDP_ConsoleSession);
file->NegotiateSecurityLayer =
freerdp_settings_get_bool(settings, FreeRDP_NegotiateSecurityLayer);
file->EnableCredSSPSupport = freerdp_settings_get_bool(settings, FreeRDP_NlaSecurity);
file->EnableRdsAadAuth = freerdp_settings_get_bool(settings, FreeRDP_AadSecurity);
2021-04-12 12:06:45 +03:00
if (freerdp_settings_get_bool(settings, FreeRDP_RemoteApplicationMode))
index = FreeRDP_RemoteApplicationWorkingDir;
else
2021-04-12 12:06:45 +03:00
index = FreeRDP_ShellWorkingDirectory;
if (!FILE_POPULATE_STRING(&file->ShellWorkingDirectory, settings, index))
return FALSE;
file->ConnectionType = freerdp_settings_get_uint32(settings, FreeRDP_ConnectionType);
file->ScreenModeId = freerdp_settings_get_bool(settings, FreeRDP_Fullscreen) ? 2 : 1;
2021-04-12 12:06:45 +03:00
LoadBalanceInfoLength = freerdp_settings_get_uint32(settings, FreeRDP_LoadBalanceInfoLength);
if (LoadBalanceInfoLength > 0)
{
2021-04-12 12:06:45 +03:00
const BYTE* LoadBalanceInfo =
freerdp_settings_get_pointer(settings, FreeRDP_LoadBalanceInfo);
file->LoadBalanceInfo = calloc(LoadBalanceInfoLength + 1, 1);
if (!file->LoadBalanceInfo)
return FALSE;
2021-04-12 12:06:45 +03:00
memcpy(file->LoadBalanceInfo, LoadBalanceInfo, LoadBalanceInfoLength);
}
2021-04-12 12:06:45 +03:00
if (freerdp_settings_get_bool(settings, FreeRDP_AudioPlayback))
file->AudioMode = AUDIO_MODE_REDIRECT;
2021-04-12 12:06:45 +03:00
else if (freerdp_settings_get_bool(settings, FreeRDP_RemoteConsoleAudio))
file->AudioMode = AUDIO_MODE_PLAY_ON_SERVER;
else
file->AudioMode = AUDIO_MODE_NONE;
/* The gateway hostname should also contain a port specifier unless it is the default port 443
*/
2021-04-12 12:06:45 +03:00
GatewayHostname = freerdp_settings_get_string(settings, FreeRDP_GatewayHostname);
if (GatewayHostname)
{
2021-04-12 12:06:45 +03:00
const UINT32 GatewayPort = freerdp_settings_get_uint32(settings, FreeRDP_GatewayPort);
freerdp_client_file_string_check_free(file->GatewayHostname);
2021-04-12 12:06:45 +03:00
if (GatewayPort == 443)
file->GatewayHostname = _strdup(GatewayHostname);
else
{
2021-04-12 12:06:45 +03:00
int length = _scprintf("%s:%" PRIu32, GatewayHostname, GatewayPort);
if (length < 0)
return FALSE;
file->GatewayHostname = (char*)malloc((size_t)length + 1);
if (!file->GatewayHostname)
return FALSE;
2021-04-12 12:06:45 +03:00
if (sprintf_s(file->GatewayHostname, (size_t)length + 1, "%s:%" PRIu32, GatewayHostname,
GatewayPort) < 0)
return FALSE;
}
if (!file->GatewayHostname)
return FALSE;
}
if (freerdp_settings_get_bool(settings, FreeRDP_GatewayArmTransport))
file->ResourceProvider = _strdup(str_resourceprovider_arm);
if (!rdp_opt_duplicate(settings, FreeRDP_GatewayAvdWvdEndpointPool, &file->WvdEndpointPool))
return FALSE;
if (!rdp_opt_duplicate(settings, FreeRDP_GatewayAvdGeo, &file->geo))
return FALSE;
if (!rdp_opt_duplicate(settings, FreeRDP_GatewayAvdArmpath, &file->armpath))
return FALSE;
if (!rdp_opt_duplicate(settings, FreeRDP_GatewayAvdAadtenantid, &file->aadtenantid))
return FALSE;
if (!rdp_opt_duplicate(settings, FreeRDP_GatewayAvdDiagnosticserviceurl,
&file->diagnosticserviceurl))
return FALSE;
if (!rdp_opt_duplicate(settings, FreeRDP_GatewayAvdHubdiscoverygeourl,
&file->hubdiscoverygeourl))
return FALSE;
if (!rdp_opt_duplicate(settings, FreeRDP_GatewayAvdActivityhint, &file->activityhint))
return FALSE;
2021-04-12 12:06:45 +03:00
file->AudioCaptureMode = freerdp_settings_get_bool(settings, FreeRDP_AudioCapture);
file->BitmapCachePersistEnable =
freerdp_settings_get_bool(settings, FreeRDP_BitmapCachePersistEnabled);
file->Compression = freerdp_settings_get_bool(settings, FreeRDP_CompressionEnabled);
file->AuthenticationLevel = freerdp_settings_get_uint32(settings, FreeRDP_AuthenticationLevel);
file->GatewayUsageMethod = freerdp_settings_get_uint32(settings, FreeRDP_GatewayUsageMethod);
file->GatewayCredentialsSource =
freerdp_settings_get_uint32(settings, FreeRDP_GatewayCredentialsSource);
2021-04-12 12:06:45 +03:00
file->PromptCredentialOnce =
freerdp_settings_get_bool(settings, FreeRDP_GatewayUseSameCredentials);
file->PromptForCredentials = freerdp_settings_get_bool(settings, FreeRDP_PromptForCredentials);
file->RemoteApplicationMode =
freerdp_settings_get_bool(settings, FreeRDP_RemoteApplicationMode);
if (!FILE_POPULATE_STRING(&file->GatewayAccessToken, settings, FreeRDP_GatewayAccessToken) ||
!FILE_POPULATE_STRING(&file->RemoteApplicationProgram, settings,
FreeRDP_RemoteApplicationProgram) ||
!FILE_POPULATE_STRING(&file->RemoteApplicationName, settings,
FreeRDP_RemoteApplicationName) ||
!FILE_POPULATE_STRING(&file->RemoteApplicationIcon, settings,
FreeRDP_RemoteApplicationIcon) ||
!FILE_POPULATE_STRING(&file->RemoteApplicationFile, settings,
FreeRDP_RemoteApplicationFile) ||
!FILE_POPULATE_STRING(&file->RemoteApplicationGuid, settings,
FreeRDP_RemoteApplicationGuid) ||
!FILE_POPULATE_STRING(&file->RemoteApplicationCmdLine, settings,
FreeRDP_RemoteApplicationCmdLine))
return FALSE;
file->SpanMonitors = freerdp_settings_get_bool(settings, FreeRDP_SpanMonitors);
file->UseMultiMon = freerdp_settings_get_bool(settings, FreeRDP_UseMultimon);
file->AllowDesktopComposition =
freerdp_settings_get_bool(settings, FreeRDP_AllowDesktopComposition);
file->AllowFontSmoothing = freerdp_settings_get_bool(settings, FreeRDP_AllowFontSmoothing);
file->DisableWallpaper = freerdp_settings_get_bool(settings, FreeRDP_DisableWallpaper);
file->DisableFullWindowDrag =
freerdp_settings_get_bool(settings, FreeRDP_DisableFullWindowDrag);
file->DisableMenuAnims = freerdp_settings_get_bool(settings, FreeRDP_DisableMenuAnims);
file->DisableThemes = freerdp_settings_get_bool(settings, FreeRDP_DisableThemes);
file->BandwidthAutoDetect = (freerdp_settings_get_uint32(settings, FreeRDP_ConnectionType) >=
CONNECTION_TYPE_AUTODETECT)
? TRUE
: FALSE;
2021-04-12 12:06:45 +03:00
file->NetworkAutoDetect =
freerdp_settings_get_bool(settings, FreeRDP_NetworkAutoDetect) ? 1 : 0;
2021-04-12 12:06:45 +03:00
file->AutoReconnectionEnabled =
freerdp_settings_get_bool(settings, FreeRDP_AutoReconnectionEnabled);
file->RedirectSmartCards = freerdp_settings_get_bool(settings, FreeRDP_RedirectSmartCards);
file->RedirectWebauthN = freerdp_settings_get_bool(settings, FreeRDP_RedirectWebAuthN);
redirectCameras =
freerdp_client_channel_args_to_string(settings, RDPECAM_DVC_CHANNEL_NAME, "device:");
if (redirectCameras)
2021-04-12 12:06:45 +03:00
{
char* str =
freerdp_client_channel_args_to_string(settings, RDPECAM_DVC_CHANNEL_NAME, "encode:");
2021-04-12 12:06:45 +03:00
file->EncodeRedirectedVideoCapture = 0;
if (str)
{
unsigned long val = 0;
2021-04-12 12:06:45 +03:00
errno = 0;
val = strtoul(str, NULL, 0);
if ((val < UINT32_MAX) && (errno == 0))
file->EncodeRedirectedVideoCapture = val;
}
free(str);
str = freerdp_client_channel_args_to_string(settings, RDPECAM_DVC_CHANNEL_NAME, "quality:");
2021-04-12 12:06:45 +03:00
file->RedirectedVideoCaptureEncodingQuality = 0;
if (str)
{
unsigned long val = 0;
2021-04-12 12:06:45 +03:00
errno = 0;
val = strtoul(str, NULL, 0);
if ((val <= 2) && (errno == 0))
{
file->RedirectedVideoCaptureEncodingQuality = val;
}
}
free(str);
file->RedirectCameras = redirectCameras;
2021-04-12 12:06:45 +03:00
}
#ifdef CHANNEL_URBDRC_CLIENT
2024-09-15 11:39:55 +03:00
char* redirectUsb =
freerdp_client_channel_args_to_string(settings, URBDRC_CHANNEL_NAME, "device:");
if (redirectUsb)
file->UsbDevicesToRedirect = redirectUsb;
2021-04-12 12:06:45 +03:00
#endif
file->RedirectClipboard =
freerdp_settings_get_bool(settings, FreeRDP_RedirectClipboard) ? 1 : 0;
file->RedirectPrinters = freerdp_settings_get_bool(settings, FreeRDP_RedirectPrinters) ? 1 : 0;
file->RedirectDrives = freerdp_settings_get_bool(settings, FreeRDP_RedirectDrives) ? 1 : 0;
file->RdgIsKdcProxy = freerdp_settings_get_bool(settings, FreeRDP_KerberosRdgIsProxy) ? 1 : 0;
2021-04-12 12:06:45 +03:00
file->RedirectComPorts = (freerdp_settings_get_bool(settings, FreeRDP_RedirectSerialPorts) ||
freerdp_settings_get_bool(settings, FreeRDP_RedirectParallelPorts));
file->RedirectLocation =
freerdp_dynamic_channel_collection_find(settings, LOCATION_CHANNEL_NAME) ? TRUE : FALSE;
2021-04-12 12:06:45 +03:00
if (!FILE_POPULATE_STRING(&file->DrivesToRedirect, settings, FreeRDP_DrivesToRedirect) ||
!FILE_POPULATE_STRING(&file->PreconnectionBlob, settings, FreeRDP_PreconnectionBlob) ||
!FILE_POPULATE_STRING(&file->KdcProxyName, settings, FreeRDP_KerberosKdcUrl))
2021-04-12 12:06:45 +03:00
return FALSE;
{
size_t offset = 0;
UINT32 count = freerdp_settings_get_uint32(settings, FreeRDP_NumMonitorIds);
2021-04-12 12:06:45 +03:00
const UINT32* MonitorIds = freerdp_settings_get_pointer(settings, FreeRDP_MonitorIds);
/* String size: 10 char UINT32 max string length, 1 char separator, one element NULL */
size_t size = count * (10 + 1) + 1;
char* str = calloc(size, sizeof(char));
for (UINT32 x = 0; x < count; x++)
2021-04-12 12:06:45 +03:00
{
int rc = _snprintf(&str[offset], size - offset, "%" PRIu32 ",", MonitorIds[x]);
if (rc <= 0)
{
free(str);
return FALSE;
}
offset += (size_t)rc;
}
if (offset > 0)
str[offset - 1] = '\0';
freerdp_client_file_string_check_free(file->SelectedMonitors);
file->SelectedMonitors = str;
}
file->KeyboardHook = freerdp_settings_get_uint32(settings, FreeRDP_KeyboardHook);
return TRUE;
}
BOOL freerdp_client_write_rdp_file(const rdpFile* file, const char* name, BOOL unicode)
{
int status = 0;
WCHAR* unicodestr = NULL;
if (!file || !name)
return FALSE;
2024-09-03 11:38:12 +03:00
const size_t size = freerdp_client_write_rdp_file_buffer(file, NULL, 0);
2019-07-18 14:53:30 +03:00
if (size == 0)
return FALSE;
2024-09-03 11:38:12 +03:00
char* buffer = calloc(size + 1ULL, sizeof(char));
2024-08-29 12:11:11 +03:00
if (freerdp_client_write_rdp_file_buffer(file, buffer, size + 1) != size)
{
2019-07-17 16:48:26 +03:00
WLog_ERR(TAG, "freerdp_client_write_rdp_file: error writing to output buffer");
free(buffer);
return FALSE;
}
2024-09-03 11:38:12 +03:00
FILE* fp = winpr_fopen(name, "w+b");
if (fp)
{
if (unicode)
{
size_t len = 0;
unicodestr = ConvertUtf8NToWCharAlloc(buffer, size, &len);
if (!unicodestr)
{
free(buffer);
(void)fclose(fp);
return FALSE;
}
/* Write multi-byte header */
if ((fwrite(BOM_UTF16_LE, sizeof(BYTE), 2, fp) != 2) ||
(fwrite(unicodestr, sizeof(WCHAR), len, fp) != len))
{
free(buffer);
free(unicodestr);
(void)fclose(fp);
return FALSE;
}
free(unicodestr);
}
else
{
2024-08-29 12:11:11 +03:00
if (fwrite(buffer, 1, size, fp) != size)
{
free(buffer);
(void)fclose(fp);
return FALSE;
}
}
(void)fflush(fp);
status = fclose(fp);
}
2015-05-11 10:07:39 +03:00
free(buffer);
return (status == 0) ? TRUE : FALSE;
}
2023-10-11 18:03:39 +03:00
WINPR_ATTR_FORMAT_ARG(3, 4)
2019-07-18 14:53:30 +03:00
static SSIZE_T freerdp_client_write_setting_to_buffer(char** buffer, size_t* bufferSize,
2023-10-11 18:03:39 +03:00
WINPR_FORMAT_ARG const char* fmt, ...)
{
va_list ap;
SSIZE_T len = 0;
char* buf = NULL;
size_t bufSize = 0;
if (!buffer || !bufferSize || !fmt)
return -1;
buf = *buffer;
bufSize = *bufferSize;
va_start(ap, fmt);
len = vsnprintf(buf, bufSize, fmt, ap);
va_end(ap);
if (len < 0)
return -1;
/* _snprintf doesn't add the ending \0 to its return value */
++len;
/* we just want to know the size - return it */
if (!buf && !bufSize)
return len;
if (!buf)
return -1;
/* update buffer size and buffer position and replace \0 with \n */
2019-07-18 14:53:30 +03:00
if (bufSize >= (size_t)len)
{
2019-07-18 14:53:30 +03:00
*bufferSize -= (size_t)len;
buf[len - 1] = '\n';
*buffer = buf + len;
}
else
return -1;
return len;
}
size_t freerdp_client_write_rdp_file_buffer(const rdpFile* file, char* buffer, size_t size)
{
size_t totalSize = 0;
if (!file)
return 0;
/* either buffer and size are null or non-null */
if ((!buffer || !size) && (buffer || size))
2019-07-18 14:53:30 +03:00
return 0;
#define WRITE_SETTING_(fmt_, ...) \
{ \
SSIZE_T res = freerdp_client_write_setting_to_buffer(&buffer, &size, fmt_, __VA_ARGS__); \
if (res < 0) \
return 0; \
totalSize += (size_t)res; \
}
#define WRITE_SETTING_INT(key_, param_) \
do \
{ \
if (~(param_)) \
WRITE_SETTING_("%s:i:%" PRIu32, key_, param_) \
2021-06-18 11:00:21 +03:00
} while (0)
#define WRITE_SETTING_STR(key_, param_) \
do \
{ \
if (~(size_t)(param_)) \
WRITE_SETTING_("%s:s:%s", key_, param_) \
2021-06-18 11:00:21 +03:00
} while (0)
2019-07-18 14:53:30 +03:00
/* integer parameters */
WRITE_SETTING_INT(key_int_use_multimon, file->UseMultiMon);
WRITE_SETTING_INT(key_int_maximizetocurrentdisplays, file->MaximizeToCurrentDisplays);
WRITE_SETTING_INT(key_int_singlemoninwindowedmode, file->SingleMonInWindowedMode);
WRITE_SETTING_INT(key_int_screen_mode_id, file->ScreenModeId);
WRITE_SETTING_INT(key_int_span_monitors, file->SpanMonitors);
WRITE_SETTING_INT(key_int_smart_sizing, file->SmartSizing);
WRITE_SETTING_INT(key_int_dynamic_resolution, file->DynamicResolution);
WRITE_SETTING_INT(key_int_enablesuperpan, file->EnableSuperSpan);
WRITE_SETTING_INT(key_int_superpanaccelerationfactor, file->SuperSpanAccelerationFactor);
WRITE_SETTING_INT(key_int_desktopwidth, file->DesktopWidth);
WRITE_SETTING_INT(key_int_desktopheight, file->DesktopHeight);
WRITE_SETTING_INT(key_int_desktop_size_id, file->DesktopSizeId);
WRITE_SETTING_INT(key_int_session_bpp, file->SessionBpp);
WRITE_SETTING_INT(key_int_desktopscalefactor, file->DesktopScaleFactor);
WRITE_SETTING_INT(key_int_compression, file->Compression);
WRITE_SETTING_INT(key_int_keyboardhook, file->KeyboardHook);
WRITE_SETTING_INT(key_int_disable_ctrl_alt_del, file->DisableCtrlAltDel);
WRITE_SETTING_INT(key_int_audiomode, file->AudioMode);
WRITE_SETTING_INT(key_int_audioqualitymode, file->AudioQualityMode);
WRITE_SETTING_INT(key_int_audiocapturemode, file->AudioCaptureMode);
WRITE_SETTING_INT(key_int_encode_redirected_video_capture, file->EncodeRedirectedVideoCapture);
WRITE_SETTING_INT(key_int_redirected_video_capture_encoding_quality,
2021-04-12 12:06:45 +03:00
file->RedirectedVideoCaptureEncodingQuality);
WRITE_SETTING_INT(key_int_videoplaybackmode, file->VideoPlaybackMode);
WRITE_SETTING_INT(key_int_connection_type, file->ConnectionType);
WRITE_SETTING_INT(key_int_networkautodetect, file->NetworkAutoDetect);
WRITE_SETTING_INT(key_int_bandwidthautodetect, file->BandwidthAutoDetect);
WRITE_SETTING_INT(key_int_pinconnectionbar, file->PinConnectionBar);
WRITE_SETTING_INT(key_int_displayconnectionbar, file->DisplayConnectionBar);
WRITE_SETTING_INT(key_int_workspaceid, file->WorkspaceId);
WRITE_SETTING_INT(key_int_enableworkspacereconnect, file->EnableWorkspaceReconnect);
WRITE_SETTING_INT(key_int_disable_wallpaper, file->DisableWallpaper);
WRITE_SETTING_INT(key_int_allow_font_smoothing, file->AllowFontSmoothing);
2023-07-05 11:18:30 +03:00
WRITE_SETTING_INT(key_int_allow_desktop_composition, file->AllowDesktopComposition);
WRITE_SETTING_INT(key_int_disable_full_window_drag, file->DisableFullWindowDrag);
WRITE_SETTING_INT(key_int_disable_menu_anims, file->DisableMenuAnims);
WRITE_SETTING_INT(key_int_disable_themes, file->DisableThemes);
WRITE_SETTING_INT(key_int_disable_cursor_setting, file->DisableCursorSetting);
WRITE_SETTING_INT(key_int_bitmapcachesize, file->BitmapCacheSize);
WRITE_SETTING_INT(key_int_bitmapcachepersistenable, file->BitmapCachePersistEnable);
WRITE_SETTING_INT(key_int_server_port, file->ServerPort);
WRITE_SETTING_INT(key_int_redirectdrives, file->RedirectDrives);
WRITE_SETTING_INT(key_int_redirectprinters, file->RedirectPrinters);
WRITE_SETTING_INT(key_int_redirectcomports, file->RedirectComPorts);
WRITE_SETTING_INT(key_int_redirectlocation, file->RedirectLocation);
WRITE_SETTING_INT(key_int_redirectsmartcards, file->RedirectSmartCards);
WRITE_SETTING_INT(key_int_redirectclipboard, file->RedirectClipboard);
WRITE_SETTING_INT(key_int_redirectposdevices, file->RedirectPosDevices);
WRITE_SETTING_INT(key_int_redirectdirectx, file->RedirectDirectX);
WRITE_SETTING_INT(key_int_disableprinterredirection, file->DisablePrinterRedirection);
WRITE_SETTING_INT(key_int_disableclipboardredirection, file->DisableClipboardRedirection);
WRITE_SETTING_INT(key_int_connect_to_console, file->ConnectToConsole);
WRITE_SETTING_INT(key_int_administrative_session, file->AdministrativeSession);
WRITE_SETTING_INT(key_int_autoreconnection_enabled, file->AutoReconnectionEnabled);
WRITE_SETTING_INT(key_int_autoreconnect_max_retries, file->AutoReconnectMaxRetries);
WRITE_SETTING_INT(key_int_public_mode, file->PublicMode);
WRITE_SETTING_INT(key_int_authentication_level, file->AuthenticationLevel);
WRITE_SETTING_INT(key_int_promptcredentialonce, file->PromptCredentialOnce);
WRITE_SETTING_INT(key_int_prompt_for_credentials, file->PromptForCredentials);
WRITE_SETTING_INT(key_int_negotiate_security_layer, file->NegotiateSecurityLayer);
WRITE_SETTING_INT(key_int_enablecredsspsupport, file->EnableCredSSPSupport);
WRITE_SETTING_INT(key_int_enablerdsaadauth, file->EnableRdsAadAuth);
WRITE_SETTING_INT(key_int_remoteapplicationmode, file->RemoteApplicationMode);
WRITE_SETTING_INT(key_int_remoteapplicationexpandcmdline, file->RemoteApplicationExpandCmdLine);
WRITE_SETTING_INT(key_int_remoteapplicationexpandworkingdir,
2019-11-06 17:24:51 +03:00
file->RemoteApplicationExpandWorkingDir);
WRITE_SETTING_INT(key_int_disableconnectionsharing, file->DisableConnectionSharing);
WRITE_SETTING_INT(key_int_disableremoteappcapscheck, file->DisableRemoteAppCapsCheck);
WRITE_SETTING_INT(key_int_gatewayusagemethod, file->GatewayUsageMethod);
WRITE_SETTING_INT(key_int_gatewayprofileusagemethod, file->GatewayProfileUsageMethod);
WRITE_SETTING_INT(key_int_gatewaycredentialssource, file->GatewayCredentialsSource);
WRITE_SETTING_INT(key_int_use_redirection_server_name, file->UseRedirectionServerName);
WRITE_SETTING_INT(key_int_rdgiskdcproxy, file->RdgIsKdcProxy);
WRITE_SETTING_INT(key_int_redirectwebauthn, file->RedirectWebauthN);
/* string parameters */
WRITE_SETTING_STR(key_str_username, file->Username);
WRITE_SETTING_STR(key_str_domain, file->Domain);
WRITE_SETTING_STR(key_str_password, file->Password);
WRITE_SETTING_STR(key_str_full_address, file->FullAddress);
WRITE_SETTING_STR(key_str_alternate_full_address, file->AlternateFullAddress);
WRITE_SETTING_STR(key_str_usbdevicestoredirect, file->UsbDevicesToRedirect);
WRITE_SETTING_STR(key_str_camerastoredirect, file->RedirectCameras);
WRITE_SETTING_STR(key_str_loadbalanceinfo, file->LoadBalanceInfo);
WRITE_SETTING_STR(key_str_remoteapplicationname, file->RemoteApplicationName);
WRITE_SETTING_STR(key_str_remoteapplicationicon, file->RemoteApplicationIcon);
WRITE_SETTING_STR(key_str_remoteapplicationprogram, file->RemoteApplicationProgram);
WRITE_SETTING_STR(key_str_remoteapplicationfile, file->RemoteApplicationFile);
WRITE_SETTING_STR(key_str_remoteapplicationguid, file->RemoteApplicationGuid);
WRITE_SETTING_STR(key_str_remoteapplicationcmdline, file->RemoteApplicationCmdLine);
WRITE_SETTING_STR(key_str_alternate_shell, file->AlternateShell);
WRITE_SETTING_STR(key_str_shell_working_directory, file->ShellWorkingDirectory);
WRITE_SETTING_STR(key_str_gatewayhostname, file->GatewayHostname);
WRITE_SETTING_STR(key_str_resourceprovider, file->ResourceProvider);
WRITE_SETTING_STR(key_str_wvd, file->WvdEndpointPool);
WRITE_SETTING_STR(key_str_geo, file->geo);
WRITE_SETTING_STR(key_str_armpath, file->armpath);
WRITE_SETTING_STR(key_str_aadtenantid, file->aadtenantid);
WRITE_SETTING_STR(key_str_diagnosticserviceurl, file->diagnosticserviceurl);
WRITE_SETTING_STR(key_str_hubdiscoverygeourl, file->hubdiscoverygeourl);
WRITE_SETTING_STR(key_str_activityhint, file->activityhint);
WRITE_SETTING_STR(key_str_gatewayaccesstoken, file->GatewayAccessToken);
WRITE_SETTING_STR(key_str_kdcproxyname, file->KdcProxyName);
WRITE_SETTING_STR(key_str_drivestoredirect, file->DrivesToRedirect);
WRITE_SETTING_STR(key_str_devicestoredirect, file->DevicesToRedirect);
WRITE_SETTING_STR(key_str_winposstr, file->WinPosStr);
WRITE_SETTING_STR(key_str_pcb, file->PreconnectionBlob);
WRITE_SETTING_STR(key_str_selectedmonitors, file->SelectedMonitors);
/* custom parameters */
for (size_t i = 0; i < file->lineCount; ++i)
{
2023-03-13 14:53:40 +03:00
SSIZE_T res = -1;
const rdpFileLine* curLine = &file->lines[i];
if (curLine->flags & RDP_FILE_LINE_FLAG_TYPE_INTEGER)
res = freerdp_client_write_setting_to_buffer(&buffer, &size, "%s:i:%" PRIu32,
curLine->name, (UINT32)curLine->iValue);
else if (curLine->flags & RDP_FILE_LINE_FLAG_TYPE_STRING)
res = freerdp_client_write_setting_to_buffer(&buffer, &size, "%s:s:%s", curLine->name,
curLine->sValue);
if (res < 0)
return 0;
totalSize += (size_t)res;
}
return totalSize;
}
2021-04-12 12:06:45 +03:00
static ADDIN_ARGV* rdp_file_to_args(const char* channel, const char* values)
{
size_t count = 0;
2021-04-12 12:06:45 +03:00
char** p = NULL;
ADDIN_ARGV* args = freerdp_addin_argv_new(0, NULL);
if (!args)
return NULL;
if (!freerdp_addin_argv_add_argument(args, channel))
goto fail;
p = CommandLineParseCommaSeparatedValues(values, &count);
for (size_t x = 0; x < count; x++)
2021-04-12 12:06:45 +03:00
{
BOOL rc = 0;
2021-04-12 12:06:45 +03:00
const char* val = p[x];
const size_t len = strlen(val) + 8;
char* str = calloc(len, sizeof(char));
if (!str)
goto fail;
(void)_snprintf(str, len, "device:%s", val);
2021-04-12 12:06:45 +03:00
rc = freerdp_addin_argv_add_argument(args, str);
free(str);
if (!rc)
goto fail;
}
free(p);
return args;
fail:
free(p);
freerdp_addin_argv_free(args);
return NULL;
}
2023-10-13 10:48:44 +03:00
BOOL freerdp_client_populate_settings_from_rdp_file(const rdpFile* file, rdpSettings* settings)
{
BOOL setDefaultConnectionType = TRUE;
if (!file || !settings)
return FALSE;
2019-07-17 16:48:26 +03:00
if (~((size_t)file->Domain))
{
if (!freerdp_settings_set_string(settings, FreeRDP_Domain, file->Domain))
return FALSE;
}
2019-07-17 16:48:26 +03:00
if (~((size_t)file->Username))
{
2013-08-28 18:16:03 +04:00
char* user = NULL;
char* domain = NULL;
if (!freerdp_parse_username(file->Username, &user, &domain))
return FALSE;
if (!freerdp_settings_set_string(settings, FreeRDP_Username, user))
return FALSE;
if (!(~((size_t)file->Domain)) && domain)
{
if (!freerdp_settings_set_string(settings, FreeRDP_Domain, domain))
return FALSE;
}
2013-08-28 18:16:03 +04:00
2015-05-11 10:07:39 +03:00
free(user);
free(domain);
}
if (~((size_t)file->Password))
{
if (!freerdp_settings_set_string(settings, FreeRDP_Password, file->Password))
return FALSE;
}
{
const char* address = NULL;
/* With MSTSC alternate full address always wins,
* so mimic this. */
if (~((size_t)file->AlternateFullAddress))
address = file->AlternateFullAddress;
else if (~((size_t)file->FullAddress))
address = file->FullAddress;
if (address)
{
int port = -1;
char* host = NULL;
if (!freerdp_parse_hostname(address, &host, &port))
return FALSE;
2019-05-06 10:51:42 +03:00
const BOOL rc = freerdp_settings_set_string(settings, FreeRDP_ServerHostname, host);
free(host);
if (!rc)
return FALSE;
if (port > 0)
{
if (!freerdp_settings_set_uint32(settings, FreeRDP_ServerPort, (UINT32)port))
return FALSE;
}
2019-05-06 10:51:42 +03:00
}
}
if (~file->ServerPort)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_uint32(settings, FreeRDP_ServerPort, file->ServerPort))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->DesktopSizeId)
{
switch (file->DesktopSizeId)
{
case 0:
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopWidth, 640))
return FALSE;
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopHeight, 480))
return FALSE;
break;
case 1:
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopWidth, 800))
return FALSE;
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopHeight, 600))
return FALSE;
break;
case 2:
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopWidth, 1024))
return FALSE;
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopHeight, 768))
return FALSE;
break;
case 3:
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopWidth, 1280))
return FALSE;
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopHeight, 1024))
return FALSE;
break;
case 4:
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopWidth, 1600))
return FALSE;
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopHeight, 1200))
return FALSE;
break;
default:
WLog_WARN(TAG, "Unsupported 'desktop size id' value %" PRIu32, file->DesktopSizeId);
break;
}
}
if (~file->DesktopWidth)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopWidth, file->DesktopWidth))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->DesktopHeight)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopHeight, file->DesktopHeight))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->SessionBpp)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_uint32(settings, FreeRDP_ColorDepth, file->SessionBpp))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->ConnectToConsole)
2019-05-06 10:51:42 +03:00
{
2019-11-06 17:24:51 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_ConsoleSession,
file->ConnectToConsole != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->AdministrativeSession)
2019-05-06 10:51:42 +03:00
{
2019-07-17 16:48:26 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_ConsoleSession,
2019-11-06 17:24:51 +03:00
file->AdministrativeSession != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->NegotiateSecurityLayer)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_bool(settings, FreeRDP_NegotiateSecurityLayer,
2019-11-06 17:24:51 +03:00
file->NegotiateSecurityLayer != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->EnableCredSSPSupport)
2019-05-06 10:51:42 +03:00
{
2019-11-06 17:24:51 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_NlaSecurity,
file->EnableCredSSPSupport != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->EnableRdsAadAuth)
{
if (!freerdp_settings_set_bool(settings, FreeRDP_AadSecurity, file->EnableRdsAadAuth != 0))
return FALSE;
}
2019-07-17 16:48:26 +03:00
if (~((size_t)file->AlternateShell))
{
if (!freerdp_settings_set_string(settings, FreeRDP_AlternateShell, file->AlternateShell))
return FALSE;
}
2019-07-17 16:48:26 +03:00
if (~((size_t)file->ShellWorkingDirectory))
{
/* ShellWorkingDir is used for either, shell working dir or remote app working dir */
2023-10-16 20:37:36 +03:00
FreeRDP_Settings_Keys_String targetId =
(~file->RemoteApplicationMode && file->RemoteApplicationMode != 0)
? FreeRDP_RemoteApplicationWorkingDir
: FreeRDP_ShellWorkingDirectory;
if (!freerdp_settings_set_string(settings, targetId, file->ShellWorkingDirectory))
return FALSE;
}
if (~file->ScreenModeId)
{
/**
* Screen Mode Id:
* http://technet.microsoft.com/en-us/library/ff393692/
*
* This setting corresponds to the selection in the Display
* configuration slider on the Display tab under Options in RDC.
*
* Values:
*
* 1: The remote session will appear in a window.
* 2: The remote session will appear full screen.
*/
if (!freerdp_settings_set_bool(settings, FreeRDP_Fullscreen,
(file->ScreenModeId == 2) ? TRUE : FALSE))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~(file->SmartSizing))
2014-04-30 01:33:59 +04:00
{
if (!freerdp_settings_set_bool(settings, FreeRDP_SmartSizing,
(file->SmartSizing == 1) ? TRUE : FALSE))
2019-05-06 10:51:42 +03:00
return FALSE;
/**
* SmartSizingWidth and SmartSizingHeight:
*
* Adding this option to use the DesktopHeight and DesktopWidth as
* parameters for the SmartSizingWidth and SmartSizingHeight, as there
* are no options for that in standard RDP files.
*
* Equivalent of doing /smart-sizing:WxH
*/
if (((~(file->DesktopWidth) && ~(file->DesktopHeight)) || ~(file->DesktopSizeId)) &&
(file->SmartSizing == 1))
{
if (!freerdp_settings_set_uint32(settings, FreeRDP_SmartSizingWidth,
file->DesktopWidth))
return FALSE;
if (!freerdp_settings_set_uint32(settings, FreeRDP_SmartSizingHeight,
file->DesktopHeight))
return FALSE;
}
2014-04-30 01:33:59 +04:00
}
2019-07-17 16:48:26 +03:00
if (~((size_t)file->LoadBalanceInfo))
{
2021-04-12 12:06:45 +03:00
const size_t len = strlen(file->LoadBalanceInfo);
if (!freerdp_settings_set_pointer_len(settings, FreeRDP_LoadBalanceInfo,
file->LoadBalanceInfo, len))
return FALSE;
}
if (~file->AuthenticationLevel)
{
/**
* Authentication Level:
* http://technet.microsoft.com/en-us/library/ff393709/
*
* This setting corresponds to the selection in the If server authentication
* fails drop-down list on the Advanced tab under Options in RDC.
*
* Values:
*
2019-07-17 16:48:26 +03:00
* 0: If server authentication fails, connect to the computer without warning (Connect and
* dont warn me). 1: If server authentication fails, do not establish a connection (Do not
* connect). 2: If server authentication fails, show a warning and allow me to connect or
* refuse the connection (Warn me). 3: No authentication requirement is specified.
*/
2021-04-12 12:06:45 +03:00
if (!freerdp_settings_set_uint32(settings, FreeRDP_AuthenticationLevel,
file->AuthenticationLevel))
return FALSE;
}
if (~file->ConnectionType)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_set_connection_type(settings, file->ConnectionType))
2019-05-06 10:51:42 +03:00
return FALSE;
setDefaultConnectionType = FALSE;
2019-05-06 10:51:42 +03:00
}
if (~file->AudioMode)
{
2019-11-06 17:24:51 +03:00
switch (file->AudioMode)
{
case AUDIO_MODE_REDIRECT:
if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteConsoleAudio, FALSE))
return FALSE;
if (!freerdp_settings_set_bool(settings, FreeRDP_AudioPlayback, TRUE))
return FALSE;
break;
case AUDIO_MODE_PLAY_ON_SERVER:
if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteConsoleAudio, TRUE))
return FALSE;
if (!freerdp_settings_set_bool(settings, FreeRDP_AudioPlayback, FALSE))
return FALSE;
break;
case AUDIO_MODE_NONE:
default:
if (!freerdp_settings_set_bool(settings, FreeRDP_AudioPlayback, FALSE))
return FALSE;
if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteConsoleAudio, FALSE))
return FALSE;
break;
}
}
if (~file->AudioCaptureMode)
{
if (!freerdp_settings_set_bool(settings, FreeRDP_AudioCapture, file->AudioCaptureMode != 0))
return FALSE;
}
if (~file->Compression)
2019-05-06 10:51:42 +03:00
{
2019-11-06 17:24:51 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_CompressionEnabled,
file->Compression != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
2019-07-17 16:48:26 +03:00
if (~((size_t)file->GatewayHostname))
{
int port = -1;
char* host = NULL;
if (!freerdp_parse_hostname(file->GatewayHostname, &host, &port))
return FALSE;
const BOOL rc = freerdp_settings_set_string(settings, FreeRDP_GatewayHostname, host);
free(host);
if (!rc)
return FALSE;
2019-05-06 10:51:42 +03:00
if (port > 0)
{
2019-07-17 16:48:26 +03:00
if (!freerdp_settings_set_uint32(settings, FreeRDP_GatewayPort, (UINT32)port))
2019-05-06 10:51:42 +03:00
return FALSE;
}
}
if (~((size_t)file->ResourceProvider))
{
if (_stricmp(file->ResourceProvider, str_resourceprovider_arm) == 0)
{
if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayArmTransport, TRUE))
return FALSE;
}
}
if (~((size_t)file->WvdEndpointPool))
{
if (!freerdp_settings_set_string(settings, FreeRDP_GatewayAvdWvdEndpointPool,
file->WvdEndpointPool))
return FALSE;
}
if (~((size_t)file->geo))
{
if (!freerdp_settings_set_string(settings, FreeRDP_GatewayAvdGeo, file->geo))
return FALSE;
}
if (~((size_t)file->armpath))
{
if (!freerdp_settings_set_string(settings, FreeRDP_GatewayAvdArmpath, file->armpath))
return FALSE;
}
if (~((size_t)file->aadtenantid))
{
if (!freerdp_settings_set_string(settings, FreeRDP_GatewayAvdAadtenantid,
file->aadtenantid))
return FALSE;
}
if (~((size_t)file->diagnosticserviceurl))
{
if (!freerdp_settings_set_string(settings, FreeRDP_GatewayAvdDiagnosticserviceurl,
file->diagnosticserviceurl))
return FALSE;
}
if (~((size_t)file->hubdiscoverygeourl))
{
if (!freerdp_settings_set_string(settings, FreeRDP_GatewayAvdHubdiscoverygeourl,
file->hubdiscoverygeourl))
return FALSE;
}
if (~((size_t)file->activityhint))
{
if (!freerdp_settings_set_string(settings, FreeRDP_GatewayAvdActivityhint,
file->activityhint))
return FALSE;
}
2019-07-17 16:48:26 +03:00
if (~((size_t)file->GatewayAccessToken))
{
2019-07-17 16:48:26 +03:00
if (!freerdp_settings_set_string(settings, FreeRDP_GatewayAccessToken,
file->GatewayAccessToken))
return FALSE;
}
2012-10-28 20:12:36 +04:00
if (~file->GatewayUsageMethod)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_set_gateway_usage_method(settings, file->GatewayUsageMethod))
return FALSE;
}
2012-10-28 20:12:36 +04:00
if (~file->PromptCredentialOnce)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_bool(settings, FreeRDP_GatewayUseSameCredentials,
2019-11-06 17:24:51 +03:00
file->PromptCredentialOnce != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
2019-07-15 11:08:39 +03:00
if (~file->PromptForCredentials)
{
if (!freerdp_settings_set_bool(settings, FreeRDP_PromptForCredentials,
2019-11-06 17:24:51 +03:00
file->PromptForCredentials != 0))
2019-07-15 11:08:39 +03:00
return FALSE;
}
if (~file->RemoteApplicationMode)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteApplicationMode,
2019-11-06 17:24:51 +03:00
file->RemoteApplicationMode != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
2019-07-17 16:48:26 +03:00
if (~((size_t)file->RemoteApplicationProgram))
{
if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationProgram,
file->RemoteApplicationProgram))
return FALSE;
}
2019-07-17 16:48:26 +03:00
if (~((size_t)file->RemoteApplicationName))
{
if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationName,
file->RemoteApplicationName))
return FALSE;
}
2019-07-17 16:48:26 +03:00
if (~((size_t)file->RemoteApplicationIcon))
{
if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationIcon,
file->RemoteApplicationIcon))
return FALSE;
}
2019-07-17 16:48:26 +03:00
if (~((size_t)file->RemoteApplicationFile))
{
if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationFile,
file->RemoteApplicationFile))
return FALSE;
}
if (~((size_t)file->RemoteApplicationGuid))
{
if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationGuid,
2019-07-17 16:48:26 +03:00
file->RemoteApplicationGuid))
return FALSE;
}
2019-07-17 16:48:26 +03:00
if (~((size_t)file->RemoteApplicationCmdLine))
{
if (!freerdp_settings_set_string(settings, FreeRDP_RemoteApplicationCmdLine,
file->RemoteApplicationCmdLine))
return FALSE;
}
if (~file->SpanMonitors)
2019-05-06 10:51:42 +03:00
{
2019-07-18 14:53:30 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_SpanMonitors, file->SpanMonitors != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->UseMultiMon)
2019-05-06 10:51:42 +03:00
{
2019-07-18 14:53:30 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_UseMultimon, file->UseMultiMon != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->AllowFontSmoothing)
2019-05-06 10:51:42 +03:00
{
2019-07-17 16:48:26 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_AllowFontSmoothing,
2019-11-06 17:24:51 +03:00
file->AllowFontSmoothing != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->DisableWallpaper)
2019-05-06 10:51:42 +03:00
{
2019-11-06 17:24:51 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_DisableWallpaper,
file->DisableWallpaper != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->DisableFullWindowDrag)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_bool(settings, FreeRDP_DisableFullWindowDrag,
2019-11-06 17:24:51 +03:00
file->DisableFullWindowDrag != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->DisableMenuAnims)
2019-05-06 10:51:42 +03:00
{
2019-11-06 17:24:51 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_DisableMenuAnims,
file->DisableMenuAnims != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->DisableThemes)
2019-05-06 10:51:42 +03:00
{
2019-07-18 14:53:30 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_DisableThemes, file->DisableThemes != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->AllowDesktopComposition)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_bool(settings, FreeRDP_AllowDesktopComposition,
2019-11-06 17:24:51 +03:00
file->AllowDesktopComposition != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->BitmapCachePersistEnable)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_bool(settings, FreeRDP_BitmapCachePersistEnabled,
2019-11-06 17:24:51 +03:00
file->BitmapCachePersistEnable != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->DisableRemoteAppCapsCheck)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_bool(settings, FreeRDP_DisableRemoteAppCapsCheck,
2019-08-22 17:00:46 +03:00
file->DisableRemoteAppCapsCheck != 0))
return FALSE;
}
if (~file->BandwidthAutoDetect)
{
if (file->BandwidthAutoDetect != 0)
{
if ((~file->NetworkAutoDetect) && (file->NetworkAutoDetect == 0))
{
WLog_WARN(TAG,
"Got networkautodetect:i:%" PRIu32 " and bandwidthautodetect:i:%" PRIu32
". Correcting to networkautodetect:i:1",
file->NetworkAutoDetect, file->BandwidthAutoDetect);
WLog_WARN(TAG,
"Add networkautodetect:i:1 to your RDP file to eliminate this warning.");
}
if (!freerdp_set_connection_type(settings, CONNECTION_TYPE_AUTODETECT))
return FALSE;
setDefaultConnectionType = FALSE;
}
2019-11-06 17:24:51 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_NetworkAutoDetect,
(file->BandwidthAutoDetect != 0) ||
(file->NetworkAutoDetect != 0)))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->NetworkAutoDetect)
{
if (file->NetworkAutoDetect != 0)
2019-08-22 17:00:46 +03:00
{
if ((~file->BandwidthAutoDetect) && (file->BandwidthAutoDetect == 0))
{
2019-11-06 17:24:51 +03:00
WLog_WARN(TAG,
"Got networkautodetect:i:%" PRIu32 " and bandwidthautodetect:i:%" PRIu32
". Correcting to bandwidthautodetect:i:1",
file->NetworkAutoDetect, file->BandwidthAutoDetect);
WLog_WARN(
TAG, "Add bandwidthautodetect:i:1 to your RDP file to eliminate this warning.");
}
2019-08-22 17:00:46 +03:00
if (!freerdp_set_connection_type(settings, CONNECTION_TYPE_AUTODETECT))
return FALSE;
setDefaultConnectionType = FALSE;
}
2019-11-06 17:24:51 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_NetworkAutoDetect,
(file->BandwidthAutoDetect != 0) ||
(file->NetworkAutoDetect != 0)))
return FALSE;
}
if (~file->AutoReconnectionEnabled)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_bool(settings, FreeRDP_AutoReconnectionEnabled,
2019-11-06 17:24:51 +03:00
file->AutoReconnectionEnabled != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->AutoReconnectMaxRetries)
2019-05-06 10:51:42 +03:00
{
if (!freerdp_settings_set_uint32(settings, FreeRDP_AutoReconnectMaxRetries,
file->AutoReconnectMaxRetries))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->RedirectSmartCards)
2019-05-06 10:51:42 +03:00
{
2019-07-17 16:48:26 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectSmartCards,
2019-11-06 17:24:51 +03:00
file->RedirectSmartCards != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->RedirectWebauthN)
{
if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectWebAuthN,
file->RedirectWebauthN != 0))
return FALSE;
}
if (~file->RedirectClipboard)
2019-05-06 10:51:42 +03:00
{
2019-07-17 16:48:26 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectClipboard,
2019-11-06 17:24:51 +03:00
file->RedirectClipboard != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->RedirectPrinters)
2019-05-06 10:51:42 +03:00
{
2019-11-06 17:24:51 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectPrinters,
file->RedirectPrinters != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->RedirectDrives)
2019-05-06 10:51:42 +03:00
{
2019-07-18 14:53:30 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectDrives, file->RedirectDrives != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->RedirectPosDevices)
{
2019-07-17 16:48:26 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectSerialPorts,
2019-11-06 17:24:51 +03:00
file->RedirectComPorts != 0) ||
2019-07-17 16:48:26 +03:00
!freerdp_settings_set_bool(settings, FreeRDP_RedirectParallelPorts,
2019-11-06 17:24:51 +03:00
file->RedirectComPorts != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->RedirectComPorts)
{
2019-07-17 16:48:26 +03:00
if (!freerdp_settings_set_bool(settings, FreeRDP_RedirectSerialPorts,
2019-11-06 17:24:51 +03:00
file->RedirectComPorts != 0) ||
2019-07-17 16:48:26 +03:00
!freerdp_settings_set_bool(settings, FreeRDP_RedirectParallelPorts,
2019-11-06 17:24:51 +03:00
file->RedirectComPorts != 0))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~file->RedirectLocation && (file->RedirectLocation != 0))
{
size_t count = 0;
union
{
void* pv;
char** str;
const char** cstr;
} cnv;
cnv.str = CommandLineParseCommaSeparatedValuesEx(LOCATION_CHANNEL_NAME, NULL, &count);
const BOOL rc = freerdp_client_add_dynamic_channel(settings, count, cnv.cstr);
free(cnv.pv);
if (!rc)
return FALSE;
}
if (~file->RedirectDirectX)
{
/* What is this?! */
}
if ((~((size_t)file->DevicesToRedirect)) && !utils_str_is_empty(file->DevicesToRedirect))
{
/**
* Devices to redirect:
* http://technet.microsoft.com/en-us/library/ff393728/
*
* This setting corresponds to the selections for Other supported Plug and Play
* (PnP) devices under More on the Local Resources tab under Options in RDC.
*
* Values:
*
* '*':
* Redirect all supported Plug and Play devices.
*
* 'DynamicDevices':
* Redirect any supported Plug and Play devices that are connected later.
*
* The hardware ID for the supported Plug and Play device:
* Redirect the specified supported Plug and Play device.
*
* Examples:
* devicestoredirect:s:*
* devicestoredirect:s:DynamicDevices
* devicestoredirect:s:USB\VID_04A9&PID_30C1\6&4BD985D&0&2;,DynamicDevices
*
*/
if (!freerdp_settings_set_bool(settings, FreeRDP_DeviceRedirection, TRUE))
return FALSE;
}
if ((~((size_t)file->DrivesToRedirect)) && !utils_str_is_empty(file->DrivesToRedirect))
{
2019-11-06 17:24:51 +03:00
if (!freerdp_settings_set_string(settings, FreeRDP_DrivesToRedirect,
file->DrivesToRedirect))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if ((~((size_t)file->RedirectCameras)) && !utils_str_is_empty(file->RedirectCameras))
{
#if defined(CHANNEL_RDPECAM_CLIENT)
2021-10-07 10:22:27 +03:00
union
{
char** c;
const char** cc;
} cnv;
ADDIN_ARGV* args = rdp_file_to_args(RDPECAM_DVC_CHANNEL_NAME, file->RedirectCameras);
2021-04-12 12:06:45 +03:00
if (!args)
return FALSE;
if (~file->EncodeRedirectedVideoCapture)
{
char encode[64];
_snprintf(encode, sizeof(encode), "encode:%" PRIu32,
file->EncodeRedirectedVideoCapture);
freerdp_addin_argv_add_argument(args, encode);
}
if (~file->RedirectedVideoCaptureEncodingQuality)
{
char quality[64];
_snprintf(quality, sizeof(quality), "quality:%" PRIu32,
file->RedirectedVideoCaptureEncodingQuality);
freerdp_addin_argv_add_argument(args, quality);
}
2021-10-07 10:22:27 +03:00
cnv.c = args->argv;
const BOOL status = freerdp_client_add_dynamic_channel(settings, args->argc, cnv.cc);
2021-04-12 12:06:45 +03:00
freerdp_addin_argv_free(args);
if (!status)
return FALSE;
#else
WLog_WARN(
TAG,
"This build does not support [MS-RDPECAM] camera redirection channel. Ignoring '%s'",
key_str_camerastoredirect);
#endif
2021-04-12 12:06:45 +03:00
}
if ((~((size_t)file->UsbDevicesToRedirect)) && !utils_str_is_empty(file->UsbDevicesToRedirect))
2021-04-12 12:06:45 +03:00
{
#ifdef CHANNEL_URBDRC_CLIENT
2021-10-07 10:22:27 +03:00
union
{
char** c;
const char** cc;
} cnv;
2021-04-12 12:06:45 +03:00
ADDIN_ARGV* args = rdp_file_to_args(URBDRC_CHANNEL_NAME, file->UsbDevicesToRedirect);
if (!args)
return FALSE;
2021-10-07 10:22:27 +03:00
cnv.c = args->argv;
const BOOL status = freerdp_client_add_dynamic_channel(settings, args->argc, cnv.cc);
2021-04-12 12:06:45 +03:00
freerdp_addin_argv_free(args);
if (!status)
return FALSE;
#else
WLog_WARN(TAG,
"This build does not support [MS-RDPEUSB] usb redirection channel. Ignoring '%s'",
key_str_usbdevicestoredirect);
2021-04-12 12:06:45 +03:00
#endif
}
2014-07-09 00:32:28 +04:00
if (~file->KeyboardHook)
{
if (!freerdp_settings_set_uint32(settings, FreeRDP_KeyboardHook, file->KeyboardHook))
2019-05-06 10:51:42 +03:00
return FALSE;
2014-07-09 00:32:28 +04:00
}
2021-04-12 12:06:45 +03:00
if (~(size_t)file->SelectedMonitors)
{
size_t count = 0;
2021-04-12 12:06:45 +03:00
char** args = CommandLineParseCommaSeparatedValues(file->SelectedMonitors, &count);
UINT32* list = NULL;
2021-04-12 12:06:45 +03:00
if (!freerdp_settings_set_pointer_len(settings, FreeRDP_MonitorIds, NULL, count))
{
free(args);
return FALSE;
}
list = freerdp_settings_get_pointer_writable(settings, FreeRDP_MonitorIds);
2021-04-12 12:06:45 +03:00
if (!list && (count > 0))
{
free(args);
return FALSE;
}
for (size_t x = 0; x < count; x++)
2021-04-12 12:06:45 +03:00
{
unsigned long val = 0;
2021-04-12 12:06:45 +03:00
errno = 0;
val = strtoul(args[x], NULL, 0);
2021-05-11 09:06:00 +03:00
if ((val >= UINT32_MAX) && (errno != 0))
2021-04-12 12:06:45 +03:00
{
free(args);
free(list);
return FALSE;
}
list[x] = val;
}
free(args);
}
if (~file->DynamicResolution)
{
const BOOL val = file->DynamicResolution != 0;
if (val)
{
if (!freerdp_settings_set_bool(settings, FreeRDP_SupportDisplayControl, TRUE))
return FALSE;
}
if (!freerdp_settings_set_bool(settings, FreeRDP_DynamicResolutionUpdate, val))
return FALSE;
}
if (~file->DesktopScaleFactor)
{
if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopScaleFactor,
file->DesktopScaleFactor))
return FALSE;
}
if (~file->VideoPlaybackMode)
{
if (file->VideoPlaybackMode != 0)
{
if (!freerdp_settings_set_bool(settings, FreeRDP_SupportGeometryTracking, TRUE) ||
!freerdp_settings_set_bool(settings, FreeRDP_SupportVideoOptimized, TRUE))
return FALSE;
}
else
{
if (!freerdp_settings_set_bool(settings, FreeRDP_SupportVideoOptimized, FALSE))
return FALSE;
}
}
// TODO file->MaximizeToCurrentDisplays;
// TODO file->SingleMonInWindowedMode;
// TODO file->EncodeRedirectedVideoCapture;
// TODO file->RedirectedVideoCaptureEncodingQuality;
2019-07-17 16:48:26 +03:00
if (~((size_t)file->PreconnectionBlob))
{
2019-07-17 16:48:26 +03:00
if (!freerdp_settings_set_string(settings, FreeRDP_PreconnectionBlob,
file->PreconnectionBlob) ||
!freerdp_settings_set_bool(settings, FreeRDP_SendPreconnectionPdu, TRUE))
2019-05-06 10:51:42 +03:00
return FALSE;
}
if (~((size_t)file->KdcProxyName))
{
if (!freerdp_settings_set_string(settings, FreeRDP_KerberosKdcUrl, file->KdcProxyName))
return FALSE;
}
if (~((size_t)file->RdgIsKdcProxy))
{
if (!freerdp_settings_set_bool(settings, FreeRDP_KerberosRdgIsProxy,
file->RdgIsKdcProxy != 0))
return FALSE;
}
if (file->args->argc > 1)
{
2023-10-13 10:48:44 +03:00
WCHAR* ConnectionFile =
freerdp_settings_get_string_as_utf16(settings, FreeRDP_ConnectionFile, NULL);
2021-04-12 12:06:45 +03:00
if (freerdp_client_settings_parse_command_line(settings, file->args->argc, file->args->argv,
FALSE) < 0)
2023-10-13 10:48:44 +03:00
{
free(ConnectionFile);
return FALSE;
2023-10-13 10:48:44 +03:00
}
2023-10-13 10:48:44 +03:00
BOOL rc = freerdp_settings_set_string_from_utf16(settings, FreeRDP_ConnectionFile,
ConnectionFile);
free(ConnectionFile);
if (!rc)
2021-04-12 12:06:45 +03:00
return FALSE;
}
if (setDefaultConnectionType)
{
if (!freerdp_set_connection_type(settings, CONNECTION_TYPE_AUTODETECT))
return FALSE;
}
return TRUE;
}
static rdpFileLine* freerdp_client_rdp_file_find_line_by_name(const rdpFile* file, const char* name)
{
BOOL bFound = FALSE;
rdpFileLine* line = NULL;
for (size_t index = 0; index < file->lineCount; index++)
{
line = &(file->lines[index]);
if (line->flags & RDP_FILE_LINE_FLAG_FORMATTED)
{
if (_stricmp(name, line->name) == 0)
{
bFound = TRUE;
break;
}
}
}
return (bFound) ? line : NULL;
}
/**
* Set a string option to a rdpFile
* @param file rdpFile
* @param name name of the option
* @param value value of the option
* @return 0 on success
*/
int freerdp_client_rdp_file_set_string_option(rdpFile* file, const char* name, const char* value)
{
return freerdp_client_rdp_file_set_string(file, name, value);
}
2019-07-18 14:53:30 +03:00
const char* freerdp_client_rdp_file_get_string_option(const rdpFile* file, const char* name)
{
LPSTR* value = NULL;
rdpFileLine* line = NULL;
rdpFile* wfile = WINPR_CAST_CONST_PTR_AWAY(file, rdpFile*);
if (freerdp_client_rdp_file_find_string_entry(wfile, name, &value, &line))
{
if (value && ~(size_t)(*value))
return *value;
if (line)
return line->sValue;
}
return NULL;
}
2019-07-18 14:53:30 +03:00
int freerdp_client_rdp_file_set_integer_option(rdpFile* file, const char* name, int value)
{
return freerdp_client_rdp_file_set_integer(file, name, value);
}
2019-07-18 14:53:30 +03:00
int freerdp_client_rdp_file_get_integer_option(const rdpFile* file, const char* name)
{
DWORD* value = NULL;
rdpFileLine* line = NULL;
rdpFile* wfile = WINPR_CAST_CONST_PTR_AWAY(file, rdpFile*);
if (freerdp_client_rdp_file_find_integer_entry(wfile, name, &value, &line))
{
if (value && ~(*value))
return *value;
if (line)
return (int)line->iValue;
}
return -1;
}
2019-07-18 14:53:30 +03:00
static void freerdp_client_file_string_check_free(LPSTR str)
{
2019-07-17 16:48:26 +03:00
if (~((size_t)str))
free(str);
}
rdpFile* freerdp_client_rdp_file_new(void)
{
return freerdp_client_rdp_file_new_ex(0);
}
rdpFile* freerdp_client_rdp_file_new_ex(DWORD flags)
{
rdpFile* file = (rdpFile*)calloc(1, sizeof(rdpFile));
if (!file)
return NULL;
file->flags = flags;
FillMemory(file, sizeof(rdpFile), 0xFF);
file->lines = NULL;
file->lineCount = 0;
file->lineSize = 32;
file->GatewayProfileUsageMethod = 1;
file->lines = (rdpFileLine*)calloc(file->lineSize, sizeof(rdpFileLine));
file->args = freerdp_addin_argv_new(0, NULL);
if (!file->lines || !file->args)
goto fail;
if (!freerdp_client_add_option(file, "freerdp"))
goto fail;
return file;
fail:
WINPR_PRAGMA_DIAG_PUSH
WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
2020-03-23 17:21:47 +03:00
freerdp_client_rdp_file_free(file);
WINPR_PRAGMA_DIAG_POP
return NULL;
}
void freerdp_client_rdp_file_free(rdpFile* file)
{
if (file)
{
if (file->lineCount)
{
for (size_t i = 0; i < file->lineCount; i++)
{
free(file->lines[i].name);
free(file->lines[i].sValue);
}
}
2020-03-23 17:21:47 +03:00
free(file->lines);
freerdp_addin_argv_free(file->args);
freerdp_client_file_string_check_free(file->Username);
freerdp_client_file_string_check_free(file->Domain);
freerdp_client_file_string_check_free(file->Password);
freerdp_client_file_string_check_free(file->FullAddress);
freerdp_client_file_string_check_free(file->AlternateFullAddress);
freerdp_client_file_string_check_free(file->UsbDevicesToRedirect);
freerdp_client_file_string_check_free(file->RedirectCameras);
2021-04-12 12:06:45 +03:00
freerdp_client_file_string_check_free(file->SelectedMonitors);
freerdp_client_file_string_check_free(file->LoadBalanceInfo);
freerdp_client_file_string_check_free(file->RemoteApplicationName);
freerdp_client_file_string_check_free(file->RemoteApplicationIcon);
freerdp_client_file_string_check_free(file->RemoteApplicationProgram);
freerdp_client_file_string_check_free(file->RemoteApplicationFile);
freerdp_client_file_string_check_free(file->RemoteApplicationGuid);
freerdp_client_file_string_check_free(file->RemoteApplicationCmdLine);
freerdp_client_file_string_check_free(file->AlternateShell);
freerdp_client_file_string_check_free(file->ShellWorkingDirectory);
freerdp_client_file_string_check_free(file->GatewayHostname);
freerdp_client_file_string_check_free(file->GatewayAccessToken);
freerdp_client_file_string_check_free(file->KdcProxyName);
freerdp_client_file_string_check_free(file->DrivesToRedirect);
freerdp_client_file_string_check_free(file->DevicesToRedirect);
freerdp_client_file_string_check_free(file->WinPosStr);
2023-07-26 19:36:34 +03:00
freerdp_client_file_string_check_free(file->ResourceProvider);
freerdp_client_file_string_check_free(file->WvdEndpointPool);
freerdp_client_file_string_check_free(file->geo);
freerdp_client_file_string_check_free(file->armpath);
freerdp_client_file_string_check_free(file->aadtenantid);
freerdp_client_file_string_check_free(file->diagnosticserviceurl);
freerdp_client_file_string_check_free(file->hubdiscoverygeourl);
freerdp_client_file_string_check_free(file->activityhint);
free(file);
}
}
void freerdp_client_rdp_file_set_callback_context(rdpFile* file, void* context)
{
file->context = context;
}