Blame client/X11/xf_disp.c

Packit Service fa4841
/**
Packit Service fa4841
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit Service fa4841
 * X11 Display Control channel
Packit Service fa4841
 *
Packit Service fa4841
 * Copyright 2017 David Fort <contact@hardening-consulting.com>
Packit Service fa4841
 *
Packit Service fa4841
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit Service fa4841
 * you may not use this file except in compliance with the License.
Packit Service fa4841
 * You may obtain a copy of the License at
Packit Service fa4841
 *
Packit Service fa4841
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit Service fa4841
 *
Packit Service fa4841
 * Unless required by applicable law or agreed to in writing, software
Packit Service fa4841
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit Service fa4841
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit Service fa4841
 * See the License for the specific language governing permissions and
Packit Service fa4841
 * limitations under the License.
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
#include <winpr/sysinfo.h>
Packit Service fa4841
#include <X11/Xutil.h>
Packit Service fa4841
Packit Service fa4841
#ifdef WITH_XRANDR
Packit Service fa4841
#include <X11/extensions/Xrandr.h>
Packit Service fa4841
#include <X11/extensions/randr.h>
Packit Service fa4841
Packit Service fa4841
#if (RANDR_MAJOR * 100 + RANDR_MINOR) >= 105
Packit Service b1ea74
#define USABLE_XRANDR
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#include "xf_disp.h"
Packit Service fa4841
#include "xf_monitor.h"
Packit Service fa4841
Packit Service fa4841
#define TAG CLIENT_TAG("x11disp")
Packit Service fa4841
#define RESIZE_MIN_DELAY 200 /* minimum delay in ms between two resizes */
Packit Service fa4841
Packit Service fa4841
struct _xfDispContext
Packit Service fa4841
{
Packit Service fa4841
	xfContext* xfc;
Packit Service fa4841
	DispClientContext* disp;
Packit Service fa4841
	BOOL haveXRandr;
Packit Service fa4841
	int eventBase, errorBase;
Packit Service fa4841
	int lastSentWidth, lastSentHeight;
Packit Service fa4841
	UINT64 lastSentDate;
Packit Service fa4841
	int targetWidth, targetHeight;
Packit Service fa4841
	BOOL activated;
Packit Service fa4841
	BOOL waitingResize;
Packit Service fa4841
	BOOL fullscreen;
Packit Service fa4841
	UINT16 lastSentDesktopOrientation;
Packit Service fa4841
	UINT32 lastSentDesktopScaleFactor;
Packit Service fa4841
	UINT32 lastSentDeviceScaleFactor;
Packit Service fa4841
};
Packit Service fa4841
Packit Service fa4841
static UINT xf_disp_sendLayout(DispClientContext* disp, rdpMonitor* monitors, int nmonitors);
Packit Service fa4841
Packit Service fa4841
static BOOL xf_disp_settings_changed(xfDispContext* xfDisp)
Packit Service fa4841
{
Packit Service fa4841
	rdpSettings* settings = xfDisp->xfc->context.settings;
Packit Service fa4841
Packit Service fa4841
	if (xfDisp->lastSentWidth != xfDisp->targetWidth)
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
	if (xfDisp->lastSentHeight != xfDisp->targetHeight)
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
	if (xfDisp->lastSentDesktopOrientation != settings->DesktopOrientation)
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
	if (xfDisp->lastSentDesktopScaleFactor != settings->DesktopScaleFactor)
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
	if (xfDisp->lastSentDeviceScaleFactor != settings->DeviceScaleFactor)
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
	if (xfDisp->fullscreen != xfDisp->xfc->fullscreen)
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
	return FALSE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL xf_update_last_sent(xfDispContext* xfDisp)
Packit Service fa4841
{
Packit Service fa4841
	rdpSettings* settings = xfDisp->xfc->context.settings;
Packit Service fa4841
	xfDisp->lastSentWidth = xfDisp->targetWidth;
Packit Service fa4841
	xfDisp->lastSentHeight = xfDisp->targetHeight;
Packit Service fa4841
	xfDisp->lastSentDesktopOrientation = settings->DesktopOrientation;
Packit Service fa4841
	xfDisp->lastSentDesktopScaleFactor = settings->DesktopScaleFactor;
Packit Service fa4841
	xfDisp->lastSentDeviceScaleFactor = settings->DeviceScaleFactor;
Packit Service fa4841
	xfDisp->fullscreen = xfDisp->xfc->fullscreen;
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL xf_disp_sendResize(xfDispContext* xfDisp)
Packit Service fa4841
{
Packit Service fa4841
	DISPLAY_CONTROL_MONITOR_LAYOUT layout;
Packit Service fa4841
	xfContext* xfc;
Packit Service fa4841
	rdpSettings* settings;
Packit Service fa4841
Packit Service fa4841
	if (!xfDisp || !xfDisp->xfc)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	xfc = xfDisp->xfc;
Packit Service fa4841
	settings = xfc->context.settings;
Packit Service fa4841
Packit Service fa4841
	if (!settings)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	if (!xfDisp->activated || !xfDisp->disp)
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
	if (GetTickCount64() - xfDisp->lastSentDate < RESIZE_MIN_DELAY)
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
	xfDisp->lastSentDate = GetTickCount64();
Packit Service fa4841
Packit Service fa4841
	if (!xf_disp_settings_changed(xfDisp))
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
	if (xfc->fullscreen && (settings->MonitorCount > 0))
Packit Service fa4841
	{
Packit Service b1ea74
		if (xf_disp_sendLayout(xfDisp->disp, settings->MonitorDefArray, settings->MonitorCount) !=
Packit Service b1ea74
		    CHANNEL_RC_OK)
Packit Service fa4841
			return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		xfDisp->waitingResize = TRUE;
Packit Service fa4841
		layout.Flags = DISPLAY_CONTROL_MONITOR_PRIMARY;
Packit Service fa4841
		layout.Top = layout.Left = 0;
Packit Service fa4841
		layout.Width = xfDisp->targetWidth;
Packit Service fa4841
		layout.Height = xfDisp->targetHeight;
Packit Service fa4841
		layout.Orientation = settings->DesktopOrientation;
Packit Service fa4841
		layout.DesktopScaleFactor = settings->DesktopScaleFactor;
Packit Service fa4841
		layout.DeviceScaleFactor = settings->DeviceScaleFactor;
Packit Service fa4841
		layout.PhysicalWidth = xfDisp->targetWidth;
Packit Service fa4841
		layout.PhysicalHeight = xfDisp->targetHeight;
Packit Service fa4841
Packit Service fa4841
		if (IFCALLRESULT(CHANNEL_RC_OK, xfDisp->disp->SendMonitorLayout, xfDisp->disp, 1,
Packit Service fa4841
		                 &layout) != CHANNEL_RC_OK)
Packit Service fa4841
			return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return xf_update_last_sent(xfDisp);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL xf_disp_set_window_resizable(xfDispContext* xfDisp)
Packit Service fa4841
{
Packit Service fa4841
	XSizeHints* size_hints;
Packit Service fa4841
Packit Service fa4841
	if (!(size_hints = XAllocSizeHints()))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	size_hints->flags = PMinSize | PMaxSize | PWinGravity;
Packit Service fa4841
	size_hints->win_gravity = NorthWestGravity;
Packit Service fa4841
	size_hints->min_width = size_hints->min_height = 320;
Packit Service fa4841
	size_hints->max_width = size_hints->max_height = 8192;
Packit Service fa4841
Packit Service fa4841
	if (xfDisp->xfc->window)
Packit Service fa4841
		XSetWMNormalHints(xfDisp->xfc->display, xfDisp->xfc->window->handle, size_hints);
Packit Service fa4841
Packit Service fa4841
	XFree(size_hints);
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL xf_disp_check_context(void* context, xfContext** ppXfc, xfDispContext** ppXfDisp,
Packit Service fa4841
                                  rdpSettings** ppSettings)
Packit Service fa4841
{
Packit Service fa4841
	xfContext* xfc;
Packit Service fa4841
Packit Service fa4841
	if (!context)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	xfc = (xfContext*)context;
Packit Service fa4841
Packit Service fa4841
	if (!(xfc->xfDisp))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	if (!xfc->context.settings)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	*ppXfc = xfc;
Packit Service fa4841
	*ppXfDisp = xfc->xfDisp;
Packit Service fa4841
	*ppSettings = xfc->context.settings;
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_disp_OnActivated(void* context, ActivatedEventArgs* e)
Packit Service fa4841
{
Packit Service fa4841
	xfContext* xfc;
Packit Service fa4841
	xfDispContext* xfDisp;
Packit Service fa4841
	rdpSettings* settings;
Packit Service fa4841
Packit Service fa4841
	if (!xf_disp_check_context(context, &xfc, &xfDisp, &settings))
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	xfDisp->waitingResize = FALSE;
Packit Service fa4841
Packit Service fa4841
	if (xfDisp->activated && !settings->Fullscreen)
Packit Service fa4841
	{
Packit Service fa4841
		xf_disp_set_window_resizable(xfDisp);
Packit Service fa4841
Packit Service fa4841
		if (e->firstActivation)
Packit Service fa4841
			return;
Packit Service fa4841
Packit Service fa4841
		xf_disp_sendResize(xfDisp);
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_disp_OnGraphicsReset(void* context, GraphicsResetEventArgs* e)
Packit Service fa4841
{
Packit Service fa4841
	xfContext* xfc;
Packit Service fa4841
	xfDispContext* xfDisp;
Packit Service fa4841
	rdpSettings* settings;
Packit Service fa4841
Packit Service b1ea74
	WINPR_UNUSED(e);
Packit Service b1ea74
Packit Service fa4841
	if (!xf_disp_check_context(context, &xfc, &xfDisp, &settings))
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	xfDisp->waitingResize = FALSE;
Packit Service fa4841
Packit Service fa4841
	if (xfDisp->activated && !settings->Fullscreen)
Packit Service fa4841
	{
Packit Service fa4841
		xf_disp_set_window_resizable(xfDisp);
Packit Service fa4841
		xf_disp_sendResize(xfDisp);
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_disp_OnTimer(void* context, TimerEventArgs* e)
Packit Service fa4841
{
Packit Service fa4841
	xfContext* xfc;
Packit Service fa4841
	xfDispContext* xfDisp;
Packit Service fa4841
	rdpSettings* settings;
Packit Service fa4841
Packit Service b1ea74
	WINPR_UNUSED(e);
Packit Service b1ea74
Packit Service fa4841
	if (!xf_disp_check_context(context, &xfc, &xfDisp, &settings))
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	if (!xfDisp->activated || settings->Fullscreen)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	xf_disp_sendResize(xfDisp);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
xfDispContext* xf_disp_new(xfContext* xfc)
Packit Service fa4841
{
Packit Service fa4841
	xfDispContext* ret;
Packit Service fa4841
Packit Service fa4841
	if (!xfc || !xfc->context.settings || !xfc->context.pubSub)
Packit Service fa4841
		return NULL;
Packit Service fa4841
Packit Service fa4841
	ret = calloc(1, sizeof(xfDispContext));
Packit Service fa4841
Packit Service fa4841
	if (!ret)
Packit Service fa4841
		return NULL;
Packit Service fa4841
Packit Service fa4841
	ret->xfc = xfc;
Packit Service fa4841
#ifdef USABLE_XRANDR
Packit Service fa4841
Packit Service fa4841
	if (XRRQueryExtension(xfc->display, &ret->eventBase, &ret->errorBase))
Packit Service fa4841
	{
Packit Service fa4841
		ret->haveXRandr = TRUE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
#endif
Packit Service fa4841
	ret->lastSentWidth = ret->targetWidth = xfc->context.settings->DesktopWidth;
Packit Service fa4841
	ret->lastSentHeight = ret->targetHeight = xfc->context.settings->DesktopHeight;
Packit Service fa4841
	PubSub_SubscribeActivated(xfc->context.pubSub, xf_disp_OnActivated);
Packit Service fa4841
	PubSub_SubscribeGraphicsReset(xfc->context.pubSub, xf_disp_OnGraphicsReset);
Packit Service fa4841
	PubSub_SubscribeTimer(xfc->context.pubSub, xf_disp_OnTimer);
Packit Service fa4841
	return ret;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_disp_free(xfDispContext* disp)
Packit Service fa4841
{
Packit Service fa4841
	if (!disp)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	if (disp->xfc)
Packit Service fa4841
	{
Packit Service fa4841
		PubSub_UnsubscribeActivated(disp->xfc->context.pubSub, xf_disp_OnActivated);
Packit Service fa4841
		PubSub_UnsubscribeGraphicsReset(disp->xfc->context.pubSub, xf_disp_OnGraphicsReset);
Packit Service fa4841
		PubSub_UnsubscribeTimer(disp->xfc->context.pubSub, xf_disp_OnTimer);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	free(disp);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
UINT xf_disp_sendLayout(DispClientContext* disp, rdpMonitor* monitors, int nmonitors)
Packit Service fa4841
{
Packit Service fa4841
	UINT ret = CHANNEL_RC_OK;
Packit Service fa4841
	DISPLAY_CONTROL_MONITOR_LAYOUT* layouts;
Packit Service fa4841
	int i;
Packit Service fa4841
	xfDispContext* xfDisp = (xfDispContext*)disp->custom;
Packit Service fa4841
	rdpSettings* settings = xfDisp->xfc->context.settings;
Packit Service fa4841
	layouts = calloc(nmonitors, sizeof(DISPLAY_CONTROL_MONITOR_LAYOUT));
Packit Service fa4841
Packit Service fa4841
	if (!layouts)
Packit Service fa4841
		return CHANNEL_RC_NO_MEMORY;
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < nmonitors; i++)
Packit Service fa4841
	{
Packit Service fa4841
		layouts[i].Flags = (monitors[i].is_primary ? DISPLAY_CONTROL_MONITOR_PRIMARY : 0);
Packit Service fa4841
		layouts[i].Left = monitors[i].x;
Packit Service fa4841
		layouts[i].Top = monitors[i].y;
Packit Service fa4841
		layouts[i].Width = monitors[i].width;
Packit Service fa4841
		layouts[i].Height = monitors[i].height;
Packit Service fa4841
		layouts[i].Orientation = ORIENTATION_LANDSCAPE;
Packit Service fa4841
		layouts[i].PhysicalWidth = monitors[i].attributes.physicalWidth;
Packit Service fa4841
		layouts[i].PhysicalHeight = monitors[i].attributes.physicalHeight;
Packit Service fa4841
Packit Service fa4841
		switch (monitors[i].attributes.orientation)
Packit Service fa4841
		{
Packit Service fa4841
			case 90:
Packit Service fa4841
				layouts[i].Orientation = ORIENTATION_PORTRAIT;
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case 180:
Packit Service fa4841
				layouts[i].Orientation = ORIENTATION_LANDSCAPE_FLIPPED;
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case 270:
Packit Service fa4841
				layouts[i].Orientation = ORIENTATION_PORTRAIT_FLIPPED;
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case 0:
Packit Service fa4841
			default:
Packit Service fa4841
				/* MS-RDPEDISP - 2.2.2.2.1:
Packit Service fa4841
				 * Orientation (4 bytes): A 32-bit unsigned integer that specifies the
Packit Service fa4841
				 * orientation of the monitor in degrees. Valid values are 0, 90, 180
Packit Service fa4841
				 * or 270
Packit Service fa4841
				 *
Packit Service fa4841
				 * So we default to ORIENTATION_LANDSCAPE
Packit Service fa4841
				 */
Packit Service fa4841
				layouts[i].Orientation = ORIENTATION_LANDSCAPE;
Packit Service fa4841
				break;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		layouts[i].DesktopScaleFactor = settings->DesktopScaleFactor;
Packit Service fa4841
		layouts[i].DeviceScaleFactor = settings->DeviceScaleFactor;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	ret = IFCALLRESULT(CHANNEL_RC_OK, disp->SendMonitorLayout, disp, nmonitors, layouts);
Packit Service fa4841
	free(layouts);
Packit Service fa4841
	return ret;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
BOOL xf_disp_handle_xevent(xfContext* xfc, const XEvent* event)
Packit Service fa4841
{
Packit Service fa4841
	xfDispContext* xfDisp;
Packit Service fa4841
	rdpSettings* settings;
Packit Service fa4841
	UINT32 maxWidth, maxHeight;
Packit Service fa4841
Packit Service fa4841
	if (!xfc || !event)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	xfDisp = xfc->xfDisp;
Packit Service fa4841
Packit Service fa4841
	if (!xfDisp)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	settings = xfc->context.settings;
Packit Service fa4841
Packit Service fa4841
	if (!settings)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	if (!xfDisp->haveXRandr || !xfDisp->disp)
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
#ifdef USABLE_XRANDR
Packit Service fa4841
Packit Service fa4841
	if (event->type != xfDisp->eventBase + RRScreenChangeNotify)
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
#endif
Packit Service fa4841
	xf_detect_monitors(xfc, &maxWidth, &maxHeight);
Packit Service b1ea74
	return xf_disp_sendLayout(xfDisp->disp, settings->MonitorDefArray, settings->MonitorCount) ==
Packit Service b1ea74
	       CHANNEL_RC_OK;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
BOOL xf_disp_handle_configureNotify(xfContext* xfc, int width, int height)
Packit Service fa4841
{
Packit Service fa4841
	xfDispContext* xfDisp;
Packit Service fa4841
Packit Service fa4841
	if (!xfc)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	xfDisp = xfc->xfDisp;
Packit Service fa4841
Packit Service fa4841
	if (!xfDisp)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	xfDisp->targetWidth = width;
Packit Service fa4841
	xfDisp->targetHeight = height;
Packit Service fa4841
	return xf_disp_sendResize(xfDisp);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static UINT xf_DisplayControlCaps(DispClientContext* disp, UINT32 maxNumMonitors,
Packit Service fa4841
                                  UINT32 maxMonitorAreaFactorA, UINT32 maxMonitorAreaFactorB)
Packit Service fa4841
{
Packit Service fa4841
	/* we're called only if dynamic resolution update is activated */
Packit Service fa4841
	xfDispContext* xfDisp = (xfDispContext*)disp->custom;
Packit Service fa4841
	rdpSettings* settings = xfDisp->xfc->context.settings;
Packit Service fa4841
	WLog_DBG(TAG,
Packit Service b1ea74
	         "DisplayControlCapsPdu: MaxNumMonitors: %" PRIu32 " MaxMonitorAreaFactorA: %" PRIu32
Packit Service b1ea74
	         " MaxMonitorAreaFactorB: %" PRIu32 "",
Packit Service fa4841
	         maxNumMonitors, maxMonitorAreaFactorA, maxMonitorAreaFactorB);
Packit Service fa4841
	xfDisp->activated = TRUE;
Packit Service fa4841
Packit Service fa4841
	if (settings->Fullscreen)
Packit Service fa4841
		return CHANNEL_RC_OK;
Packit Service fa4841
Packit Service b1ea74
	WLog_DBG(TAG, "DisplayControlCapsPdu: setting the window as resizable");
Packit Service fa4841
	return xf_disp_set_window_resizable(xfDisp) ? CHANNEL_RC_OK : CHANNEL_RC_NO_MEMORY;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
BOOL xf_disp_init(xfDispContext* xfDisp, DispClientContext* disp)
Packit Service fa4841
{
Packit Service fa4841
	rdpSettings* settings;
Packit Service fa4841
Packit Service fa4841
	if (!xfDisp || !xfDisp->xfc || !disp)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	settings = xfDisp->xfc->context.settings;
Packit Service fa4841
Packit Service fa4841
	if (!settings)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	xfDisp->disp = disp;
Packit Service b1ea74
	disp->custom = (void*)xfDisp;
Packit Service fa4841
Packit Service fa4841
	if (settings->DynamicResolutionUpdate)
Packit Service fa4841
	{
Packit Service fa4841
		disp->DisplayControlCaps = xf_DisplayControlCaps;
Packit Service fa4841
#ifdef USABLE_XRANDR
Packit Service fa4841
Packit Service fa4841
		if (settings->Fullscreen)
Packit Service fa4841
		{
Packit Service fa4841
			/* ask X11 to notify us of screen changes */
Packit Service fa4841
			XRRSelectInput(xfDisp->xfc->display, DefaultRootWindow(xfDisp->xfc->display),
Packit Service fa4841
			               RRScreenChangeNotifyMask);
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
#endif
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
BOOL xf_disp_uninit(xfDispContext* xfDisp, DispClientContext* disp)
Packit Service fa4841
{
Packit Service fa4841
	if (!xfDisp || !disp)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	xfDisp->disp = NULL;
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}