2014-09-12 19:38:12 +04:00
|
|
|
/**
|
|
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
|
|
|
* FreeRDP Windows Server
|
|
|
|
*
|
|
|
|
* Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
|
|
* Copyright 2012-2013 Corey Clayton <can.of.tuna@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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <winpr/tchar.h>
|
|
|
|
#include <winpr/windows.h>
|
|
|
|
|
|
|
|
#include "wf_mirage.h"
|
|
|
|
|
2018-11-14 17:13:11 +03:00
|
|
|
#include <freerdp/log.h>
|
|
|
|
#define TAG SERVER_TAG("Windows.mirror")
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
#define DEVICE_KEY_PREFIX _T("\\Registry\\Machine\\")
|
2014-09-12 19:38:12 +04:00
|
|
|
/*
|
|
|
|
This function will iterate over the loaded display devices until it finds
|
|
|
|
the mirror device we want to load. If found, it will then copy the registry
|
|
|
|
key corresponding to the device to the wfi and returns TRUE. Otherwise
|
|
|
|
the function returns FALSE.
|
|
|
|
*/
|
|
|
|
BOOL wf_mirror_driver_find_display_device(wfInfo* wfi)
|
|
|
|
{
|
|
|
|
BOOL result;
|
|
|
|
BOOL devFound;
|
|
|
|
DWORD deviceNumber;
|
|
|
|
DISPLAY_DEVICE deviceInfo;
|
|
|
|
devFound = FALSE;
|
|
|
|
deviceNumber = 0;
|
|
|
|
deviceInfo.cb = sizeof(deviceInfo);
|
|
|
|
|
|
|
|
while (result = EnumDisplayDevices(NULL, deviceNumber, &deviceInfo, 0))
|
|
|
|
{
|
|
|
|
if (_tcscmp(deviceInfo.DeviceString, _T("Mirage Driver")) == 0)
|
|
|
|
{
|
|
|
|
int deviceKeyLength;
|
|
|
|
int deviceKeyPrefixLength;
|
|
|
|
deviceKeyPrefixLength = _tcslen(DEVICE_KEY_PREFIX);
|
|
|
|
|
|
|
|
if (_tcsnicmp(deviceInfo.DeviceKey, DEVICE_KEY_PREFIX, deviceKeyPrefixLength) == 0)
|
|
|
|
{
|
|
|
|
deviceKeyLength = _tcslen(deviceInfo.DeviceKey) - deviceKeyPrefixLength;
|
2019-11-06 17:24:51 +03:00
|
|
|
wfi->deviceKey = (LPTSTR)malloc((deviceKeyLength + 1) * sizeof(TCHAR));
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2015-06-16 16:42:07 +03:00
|
|
|
if (!wfi->deviceKey)
|
|
|
|
return FALSE;
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
_tcsncpy_s(wfi->deviceKey, deviceKeyLength + 1,
|
2018-11-14 17:13:11 +03:00
|
|
|
&deviceInfo.DeviceKey[deviceKeyPrefixLength], deviceKeyLength);
|
2014-09-12 19:38:12 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
_tcsncpy_s(wfi->deviceName, 32, deviceInfo.DeviceName, _tcslen(deviceInfo.DeviceName));
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
deviceNumber++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This function will attempt to access the the windows registry using the device
|
|
|
|
* key stored in the current wfi. It will attempt to read the value of the
|
|
|
|
* "Attach.ToDesktop" subkey and will return TRUE if the value is already set to
|
2018-11-14 17:13:11 +03:00
|
|
|
* val. If unable to read the subkey, this function will return FALSE. If the
|
|
|
|
* subkey is not set to val it will then attempt to set it to val and return TRUE. If
|
|
|
|
* unsuccessful or an unexpected value is encountered, the function returns
|
2014-09-12 19:38:12 +04:00
|
|
|
* FALSE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
BOOL wf_mirror_driver_display_device_attach(wfInfo* wfi, DWORD mode)
|
|
|
|
{
|
|
|
|
HKEY hKey;
|
|
|
|
LONG status;
|
|
|
|
DWORD dwType;
|
|
|
|
DWORD dwSize;
|
|
|
|
DWORD dwValue;
|
2019-11-06 17:24:51 +03:00
|
|
|
status = RegOpenKeyEx(HKEY_LOCAL_MACHINE, wfi->deviceKey, 0, KEY_ALL_ACCESS | KEY_WOW64_64KEY,
|
|
|
|
&hKey);
|
2014-09-12 19:38:12 +04:00
|
|
|
|
|
|
|
if (status != ERROR_SUCCESS)
|
|
|
|
{
|
2016-12-14 00:47:08 +03:00
|
|
|
WLog_DBG(TAG, "Error opening RegKey: status=0x%08lX", status);
|
2014-09-12 19:38:12 +04:00
|
|
|
|
|
|
|
if (status == ERROR_ACCESS_DENIED)
|
|
|
|
WLog_DBG(TAG, "access denied. Do you have admin privleges?");
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
dwSize = sizeof(DWORD);
|
2019-11-06 17:24:51 +03:00
|
|
|
status = RegQueryValueEx(hKey, _T("Attach.ToDesktop"), NULL, &dwType, (BYTE*)&dwValue, &dwSize);
|
2014-09-12 19:38:12 +04:00
|
|
|
|
|
|
|
if (status != ERROR_SUCCESS)
|
|
|
|
{
|
2016-12-14 00:47:08 +03:00
|
|
|
WLog_DBG(TAG, "Error querying RegKey: status=0x%08lX", status);
|
2014-09-12 19:38:12 +04:00
|
|
|
|
|
|
|
if (status == ERROR_ACCESS_DENIED)
|
|
|
|
WLog_DBG(TAG, "access denied. Do you have admin privleges?");
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
if (dwValue ^ mode) // only if we want to change modes
|
2014-09-12 19:38:12 +04:00
|
|
|
{
|
|
|
|
dwValue = mode;
|
|
|
|
dwSize = sizeof(DWORD);
|
2019-11-06 17:24:51 +03:00
|
|
|
status = RegSetValueEx(hKey, _T("Attach.ToDesktop"), 0, REG_DWORD, (BYTE*)&dwValue, dwSize);
|
2014-09-12 19:38:12 +04:00
|
|
|
|
|
|
|
if (status != ERROR_SUCCESS)
|
|
|
|
{
|
2016-12-14 00:47:08 +03:00
|
|
|
WLog_DBG(TAG, "Error writing registry key: %ld", status);
|
2014-09-12 19:38:12 +04:00
|
|
|
|
|
|
|
if (status == ERROR_ACCESS_DENIED)
|
|
|
|
WLog_DBG(TAG, "access denied. Do you have admin privleges?");
|
|
|
|
|
|
|
|
WLog_DBG(TAG, "");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void wf_mirror_driver_print_display_change_status(LONG status)
|
|
|
|
{
|
|
|
|
TCHAR disp_change[64];
|
|
|
|
|
|
|
|
switch (status)
|
|
|
|
{
|
|
|
|
case DISP_CHANGE_SUCCESSFUL:
|
|
|
|
_tcscpy(disp_change, _T("DISP_CHANGE_SUCCESSFUL"));
|
|
|
|
break;
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
case DISP_CHANGE_BADDUALVIEW:
|
|
|
|
_tcscpy(disp_change, _T("DISP_CHANGE_BADDUALVIEW"));
|
|
|
|
break;
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
case DISP_CHANGE_BADFLAGS:
|
|
|
|
_tcscpy(disp_change, _T("DISP_CHANGE_BADFLAGS"));
|
|
|
|
break;
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
case DISP_CHANGE_BADMODE:
|
|
|
|
_tcscpy(disp_change, _T("DISP_CHANGE_BADMODE"));
|
|
|
|
break;
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
case DISP_CHANGE_BADPARAM:
|
|
|
|
_tcscpy(disp_change, _T("DISP_CHANGE_BADPARAM"));
|
|
|
|
break;
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
case DISP_CHANGE_FAILED:
|
|
|
|
_tcscpy(disp_change, _T("DISP_CHANGE_FAILED"));
|
|
|
|
break;
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
case DISP_CHANGE_NOTUPDATED:
|
|
|
|
_tcscpy(disp_change, _T("DISP_CHANGE_NOTUPDATED"));
|
|
|
|
break;
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
case DISP_CHANGE_RESTART:
|
|
|
|
_tcscpy(disp_change, _T("DISP_CHANGE_RESTART"));
|
|
|
|
break;
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
default:
|
|
|
|
_tcscpy(disp_change, _T("DISP_CHANGE_UNKNOWN"));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status != DISP_CHANGE_SUCCESSFUL)
|
2016-12-14 00:47:08 +03:00
|
|
|
WLog_ERR(TAG, "ChangeDisplaySettingsEx() failed with %s (%ld)", disp_change, status);
|
2014-09-12 19:38:12 +04:00
|
|
|
else
|
2016-12-14 00:47:08 +03:00
|
|
|
WLog_INFO(TAG, "ChangeDisplaySettingsEx() succeeded with %s (%ld)", disp_change, status);
|
2014-09-12 19:38:12 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-11-14 17:13:11 +03:00
|
|
|
* This function will attempt to apply the currently configured display settings
|
|
|
|
* in the registry to the display driver. It will return TRUE if successful
|
2014-09-12 19:38:12 +04:00
|
|
|
* otherwise it returns FALSE.
|
|
|
|
* If mode is MIRROR_UNLOAD then the the driver will be asked to remove itself.
|
|
|
|
*/
|
|
|
|
|
|
|
|
BOOL wf_mirror_driver_update(wfInfo* wfi, int mode)
|
|
|
|
{
|
|
|
|
BOOL status;
|
|
|
|
DWORD* extHdr;
|
|
|
|
WORD drvExtraSaved;
|
|
|
|
DEVMODE* deviceMode;
|
|
|
|
LONG disp_change_status;
|
|
|
|
DWORD dmf_devmodewext_magic_sig = 0xDF20C0DE;
|
|
|
|
|
2018-11-14 17:13:11 +03:00
|
|
|
if ((mode != MIRROR_LOAD) && (mode != MIRROR_UNLOAD))
|
2014-09-12 19:38:12 +04:00
|
|
|
{
|
|
|
|
WLog_DBG(TAG, "Invalid mirror mode!");
|
|
|
|
return FALSE;
|
|
|
|
}
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
deviceMode = (DEVMODE*)malloc(sizeof(DEVMODE) + EXT_DEVMODE_SIZE_MAX);
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2015-06-16 16:42:07 +03:00
|
|
|
if (!deviceMode)
|
|
|
|
return FALSE;
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
deviceMode->dmDriverExtra = 2 * sizeof(DWORD);
|
2019-11-06 17:24:51 +03:00
|
|
|
extHdr = (DWORD*)((BYTE*)&deviceMode + sizeof(DEVMODE));
|
2014-09-12 19:38:12 +04:00
|
|
|
extHdr[0] = dmf_devmodewext_magic_sig;
|
|
|
|
extHdr[1] = 0;
|
|
|
|
drvExtraSaved = deviceMode->dmDriverExtra;
|
|
|
|
memset(deviceMode, 0, sizeof(DEVMODE) + EXT_DEVMODE_SIZE_MAX);
|
|
|
|
deviceMode->dmSize = sizeof(DEVMODE);
|
|
|
|
deviceMode->dmDriverExtra = drvExtraSaved;
|
|
|
|
|
|
|
|
if (mode == MIRROR_LOAD)
|
|
|
|
{
|
|
|
|
wfi->virtscreen_width = GetSystemMetrics(SM_CXVIRTUALSCREEN);
|
|
|
|
wfi->virtscreen_height = GetSystemMetrics(SM_CYVIRTUALSCREEN);
|
|
|
|
deviceMode->dmPelsWidth = wfi->virtscreen_width;
|
|
|
|
deviceMode->dmPelsHeight = wfi->virtscreen_height;
|
|
|
|
deviceMode->dmBitsPerPel = wfi->bitsPerPixel;
|
|
|
|
deviceMode->dmPosition.x = wfi->servscreen_xoffset;
|
|
|
|
deviceMode->dmPosition.y = wfi->servscreen_yoffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
deviceMode->dmFields = DM_BITSPERPEL | DM_PELSWIDTH | DM_PELSHEIGHT | DM_POSITION;
|
|
|
|
_tcsncpy_s(deviceMode->dmDeviceName, 32, wfi->deviceName, _tcslen(wfi->deviceName));
|
2019-11-06 17:24:51 +03:00
|
|
|
disp_change_status =
|
|
|
|
ChangeDisplaySettingsEx(wfi->deviceName, deviceMode, NULL, CDS_UPDATEREGISTRY, NULL);
|
2014-09-12 19:38:12 +04:00
|
|
|
status = (disp_change_status == DISP_CHANGE_SUCCESSFUL) ? TRUE : FALSE;
|
|
|
|
|
|
|
|
if (!status)
|
|
|
|
wf_mirror_driver_print_display_change_status(disp_change_status);
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL wf_mirror_driver_map_memory(wfInfo* wfi)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
wfi->driverDC = CreateDC(wfi->deviceName, NULL, NULL, NULL);
|
|
|
|
|
|
|
|
if (wfi->driverDC == NULL)
|
|
|
|
{
|
|
|
|
WLog_ERR(TAG, "Could not create device driver context!");
|
|
|
|
{
|
|
|
|
LPVOID lpMsgBuf;
|
2018-11-14 17:13:11 +03:00
|
|
|
DWORD dw = GetLastError();
|
2019-11-06 17:24:51 +03:00
|
|
|
FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM |
|
|
|
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
|
|
|
NULL, dw, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR)&lpMsgBuf, 0,
|
|
|
|
NULL);
|
2014-09-12 19:38:12 +04:00
|
|
|
// Display the error message and exit the process
|
2019-11-06 17:24:51 +03:00
|
|
|
WLog_ERR(TAG, "CreateDC failed on device [%s] with error %lu: %s", wfi->deviceName, dw,
|
|
|
|
lpMsgBuf);
|
2014-09-12 19:38:12 +04:00
|
|
|
LocalFree(lpMsgBuf);
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2015-06-16 16:42:07 +03:00
|
|
|
wfi->changeBuffer = calloc(1, sizeof(GETCHANGESBUF));
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2015-06-16 16:42:07 +03:00
|
|
|
if (!wfi->changeBuffer)
|
|
|
|
return FALSE;
|
2018-11-14 17:13:11 +03:00
|
|
|
|
|
|
|
status = ExtEscape(wfi->driverDC, dmf_esc_usm_pipe_map, 0, 0, sizeof(GETCHANGESBUF),
|
2019-11-06 17:24:51 +03:00
|
|
|
(LPSTR)wfi->changeBuffer);
|
2014-09-12 19:38:12 +04:00
|
|
|
|
|
|
|
if (status <= 0)
|
|
|
|
{
|
|
|
|
WLog_ERR(TAG, "Failed to map shared memory from the driver! code %d", status);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Unmap the shared memory and release the DC */
|
|
|
|
|
|
|
|
BOOL wf_mirror_driver_cleanup(wfInfo* wfi)
|
|
|
|
{
|
|
|
|
int status;
|
2018-11-14 17:13:11 +03:00
|
|
|
status = ExtEscape(wfi->driverDC, dmf_esc_usm_pipe_unmap, sizeof(GETCHANGESBUF),
|
2019-11-06 17:24:51 +03:00
|
|
|
(LPSTR)wfi->changeBuffer, 0, 0);
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
if (status <= 0)
|
|
|
|
{
|
|
|
|
WLog_ERR(TAG, "Failed to unmap shared memory from the driver! code %d", status);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wfi->driverDC != NULL)
|
|
|
|
{
|
|
|
|
status = DeleteDC(wfi->driverDC);
|
|
|
|
|
|
|
|
if (status == 0)
|
|
|
|
{
|
2017-02-17 12:54:31 +03:00
|
|
|
WLog_ERR(TAG, "Failed to release DC!");
|
2014-09-12 19:38:12 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
free(wfi->changeBuffer);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL wf_mirror_driver_activate(wfInfo* wfi)
|
|
|
|
{
|
|
|
|
if (!wfi->mirrorDriverActive)
|
|
|
|
{
|
|
|
|
WLog_DBG(TAG, "Activating Mirror Driver");
|
|
|
|
|
|
|
|
if (wf_mirror_driver_find_display_device(wfi) == FALSE)
|
|
|
|
{
|
|
|
|
WLog_DBG(TAG, "Could not find dfmirage mirror driver! Is it installed?");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wf_mirror_driver_display_device_attach(wfi, 1) == FALSE)
|
|
|
|
{
|
|
|
|
WLog_DBG(TAG, "Could not attach display device!");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wf_mirror_driver_update(wfi, MIRROR_LOAD) == FALSE)
|
|
|
|
{
|
|
|
|
WLog_DBG(TAG, "could not update system with new display settings!");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wf_mirror_driver_map_memory(wfi) == FALSE)
|
|
|
|
{
|
|
|
|
WLog_DBG(TAG, "Unable to map memory for mirror driver!");
|
|
|
|
return FALSE;
|
|
|
|
}
|
2018-11-14 17:13:11 +03:00
|
|
|
|
2014-09-12 19:38:12 +04:00
|
|
|
wfi->mirrorDriverActive = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void wf_mirror_driver_deactivate(wfInfo* wfi)
|
|
|
|
{
|
|
|
|
if (wfi->mirrorDriverActive)
|
|
|
|
{
|
|
|
|
WLog_DBG(TAG, "Deactivating Mirror Driver");
|
|
|
|
wf_mirror_driver_cleanup(wfi);
|
|
|
|
wf_mirror_driver_display_device_attach(wfi, 0);
|
|
|
|
wf_mirror_driver_update(wfi, MIRROR_UNLOAD);
|
|
|
|
wfi->mirrorDriverActive = FALSE;
|
|
|
|
}
|
2015-06-16 16:42:07 +03:00
|
|
|
}
|