2013-07-29 11:00:15 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
2014-05-02 13:34:02 +00:00
|
|
|
/*
|
2013-07-29 11:00:15 +00:00
|
|
|
* Copyright (C) 2001, 2002 Havoc Pennington
|
|
|
|
* Copyright (C) 2002, 2003 Red Hat Inc.
|
|
|
|
* Some ICCCM manager selection code derived from fvwm2,
|
|
|
|
* Copyright (C) 2001 Dominik Vogt, Matthias Clasen, and fvwm2 team
|
|
|
|
* Copyright (C) 2003 Rob Adams
|
|
|
|
* Copyright (C) 2004-2006 Elijah Newren
|
|
|
|
* Copyright (C) 2013 Red Hat Inc.
|
2014-05-02 13:34:02 +00:00
|
|
|
*
|
2013-07-29 11:00:15 +00:00
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
2014-05-02 13:34:02 +00:00
|
|
|
*
|
2013-07-29 11:00:15 +00:00
|
|
|
* You should have received a copy of the GNU General Public License
|
2014-01-12 01:42:06 +00:00
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
2013-07-29 11:00:15 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2014-03-31 00:25:31 +00:00
|
|
|
#include "meta-monitor-manager-xrandr.h"
|
|
|
|
|
2013-07-29 11:00:15 +00:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <clutter/clutter.h>
|
|
|
|
|
2017-04-18 09:06:18 +00:00
|
|
|
#include <X11/Xlibint.h>
|
2013-07-29 11:00:15 +00:00
|
|
|
#include <X11/extensions/dpms.h>
|
2014-08-19 15:02:33 +00:00
|
|
|
#include <X11/Xlib-xcb.h>
|
|
|
|
#include <xcb/randr.h>
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2014-04-23 13:57:16 +00:00
|
|
|
#include "meta-backend-x11.h"
|
2013-07-29 11:00:15 +00:00
|
|
|
#include <meta/main.h>
|
|
|
|
#include <meta/errors.h>
|
2017-03-28 04:35:19 +00:00
|
|
|
#include "backends/meta-crtc.h"
|
2017-01-09 06:31:18 +00:00
|
|
|
#include "backends/meta-monitor-config-manager.h"
|
2016-12-13 02:37:11 +00:00
|
|
|
#include "backends/meta-logical-monitor.h"
|
2017-03-24 09:35:51 +00:00
|
|
|
#include "backends/meta-output.h"
|
2017-07-04 10:42:16 +00:00
|
|
|
#include "backends/x11/meta-output-xrandr.h"
|
2013-08-16 15:06:46 +00:00
|
|
|
|
2014-07-10 21:39:47 +00:00
|
|
|
#define ALL_TRANSFORMS ((1 << (META_MONITOR_TRANSFORM_FLIPPED_270 + 1)) - 1)
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2013-08-16 15:39:07 +00:00
|
|
|
/* Look for DPI_FALLBACK in:
|
|
|
|
* http://git.gnome.org/browse/gnome-settings-daemon/tree/plugins/xsettings/gsd-xsettings-manager.c
|
|
|
|
* for the reasoning */
|
|
|
|
#define DPI_FALLBACK 96.0
|
|
|
|
|
2013-07-29 11:00:15 +00:00
|
|
|
struct _MetaMonitorManagerXrandr
|
|
|
|
{
|
|
|
|
MetaMonitorManager parent_instance;
|
|
|
|
|
|
|
|
Display *xdisplay;
|
|
|
|
XRRScreenResources *resources;
|
|
|
|
int rr_event_base;
|
|
|
|
int rr_error_base;
|
2015-03-31 00:38:44 +00:00
|
|
|
gboolean has_randr15;
|
2016-12-13 10:58:22 +00:00
|
|
|
|
2017-04-21 10:01:15 +00:00
|
|
|
xcb_timestamp_t last_xrandr_set_timestamp;
|
|
|
|
|
2016-12-13 10:58:22 +00:00
|
|
|
#ifdef HAVE_XRANDR15
|
|
|
|
GHashTable *tiled_monitor_atoms;
|
|
|
|
#endif /* HAVE_XRANDR15 */
|
2017-02-15 08:06:46 +00:00
|
|
|
|
|
|
|
int max_screen_width;
|
|
|
|
int max_screen_height;
|
2013-07-29 11:00:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct _MetaMonitorManagerXrandrClass
|
|
|
|
{
|
|
|
|
MetaMonitorManagerClass parent_class;
|
|
|
|
};
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (MetaMonitorManagerXrandr, meta_monitor_manager_xrandr, META_TYPE_MONITOR_MANAGER);
|
|
|
|
|
2016-12-13 10:58:22 +00:00
|
|
|
#ifdef HAVE_XRANDR15
|
|
|
|
typedef struct _MetaMonitorXrandrData
|
|
|
|
{
|
|
|
|
Atom xrandr_name;
|
|
|
|
} MetaMonitorXrandrData;
|
|
|
|
|
|
|
|
GQuark quark_meta_monitor_xrandr_data;
|
|
|
|
#endif /* HAVE_RANDR15 */
|
|
|
|
|
2017-07-04 10:42:16 +00:00
|
|
|
Display *
|
|
|
|
meta_monitor_manager_xrandr_get_xdisplay (MetaMonitorManagerXrandr *manager_xrandr)
|
|
|
|
{
|
|
|
|
return manager_xrandr->xdisplay;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_monitor_manager_xrandr_has_randr15 (MetaMonitorManagerXrandr *manager_xrandr)
|
|
|
|
{
|
|
|
|
return manager_xrandr->has_randr15;
|
|
|
|
}
|
|
|
|
|
2014-07-10 21:39:47 +00:00
|
|
|
static MetaMonitorTransform
|
|
|
|
meta_monitor_transform_from_xrandr (Rotation rotation)
|
2013-07-29 11:00:15 +00:00
|
|
|
{
|
2014-07-10 21:39:47 +00:00
|
|
|
static const MetaMonitorTransform y_reflected_map[4] = {
|
|
|
|
META_MONITOR_TRANSFORM_FLIPPED_180,
|
|
|
|
META_MONITOR_TRANSFORM_FLIPPED_90,
|
|
|
|
META_MONITOR_TRANSFORM_FLIPPED,
|
|
|
|
META_MONITOR_TRANSFORM_FLIPPED_270
|
2013-07-29 11:00:15 +00:00
|
|
|
};
|
2014-07-10 21:39:47 +00:00
|
|
|
MetaMonitorTransform ret;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
|
|
|
switch (rotation & 0x7F)
|
|
|
|
{
|
|
|
|
default:
|
|
|
|
case RR_Rotate_0:
|
2014-07-10 21:39:47 +00:00
|
|
|
ret = META_MONITOR_TRANSFORM_NORMAL;
|
2013-07-29 11:00:15 +00:00
|
|
|
break;
|
|
|
|
case RR_Rotate_90:
|
2014-07-10 21:39:47 +00:00
|
|
|
ret = META_MONITOR_TRANSFORM_90;
|
2013-07-29 11:00:15 +00:00
|
|
|
break;
|
|
|
|
case RR_Rotate_180:
|
2014-07-10 21:39:47 +00:00
|
|
|
ret = META_MONITOR_TRANSFORM_180;
|
2013-07-29 11:00:15 +00:00
|
|
|
break;
|
|
|
|
case RR_Rotate_270:
|
2014-07-10 21:39:47 +00:00
|
|
|
ret = META_MONITOR_TRANSFORM_270;
|
2013-07-29 11:00:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rotation & RR_Reflect_X)
|
|
|
|
return ret + 4;
|
|
|
|
else if (rotation & RR_Reflect_Y)
|
|
|
|
return y_reflected_map[ret];
|
|
|
|
else
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define ALL_ROTATIONS (RR_Rotate_0 | RR_Rotate_90 | RR_Rotate_180 | RR_Rotate_270)
|
|
|
|
|
2014-07-10 21:39:47 +00:00
|
|
|
static MetaMonitorTransform
|
|
|
|
meta_monitor_transform_from_xrandr_all (Rotation rotation)
|
2013-07-29 11:00:15 +00:00
|
|
|
{
|
|
|
|
unsigned ret;
|
|
|
|
|
|
|
|
/* Handle the common cases first (none or all) */
|
|
|
|
if (rotation == 0 || rotation == RR_Rotate_0)
|
2014-07-10 21:39:47 +00:00
|
|
|
return (1 << META_MONITOR_TRANSFORM_NORMAL);
|
2013-07-29 11:00:15 +00:00
|
|
|
|
|
|
|
/* All rotations and one reflection -> all of them by composition */
|
|
|
|
if ((rotation & ALL_ROTATIONS) &&
|
|
|
|
((rotation & RR_Reflect_X) || (rotation & RR_Reflect_Y)))
|
2014-07-10 21:39:47 +00:00
|
|
|
return ALL_TRANSFORMS;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2014-07-10 21:39:47 +00:00
|
|
|
ret = 1 << META_MONITOR_TRANSFORM_NORMAL;
|
2013-07-29 11:00:15 +00:00
|
|
|
if (rotation & RR_Rotate_90)
|
2014-07-10 21:39:47 +00:00
|
|
|
ret |= 1 << META_MONITOR_TRANSFORM_90;
|
2013-07-29 11:00:15 +00:00
|
|
|
if (rotation & RR_Rotate_180)
|
2014-07-10 21:39:47 +00:00
|
|
|
ret |= 1 << META_MONITOR_TRANSFORM_180;
|
2013-07-29 11:00:15 +00:00
|
|
|
if (rotation & RR_Rotate_270)
|
2014-07-10 21:39:47 +00:00
|
|
|
ret |= 1 << META_MONITOR_TRANSFORM_270;
|
2013-07-29 11:00:15 +00:00
|
|
|
if (rotation & (RR_Rotate_0 | RR_Reflect_X))
|
2014-07-10 21:39:47 +00:00
|
|
|
ret |= 1 << META_MONITOR_TRANSFORM_FLIPPED;
|
2013-07-29 11:00:15 +00:00
|
|
|
if (rotation & (RR_Rotate_90 | RR_Reflect_X))
|
2014-07-10 21:39:47 +00:00
|
|
|
ret |= 1 << META_MONITOR_TRANSFORM_FLIPPED_90;
|
2013-07-29 11:00:15 +00:00
|
|
|
if (rotation & (RR_Rotate_180 | RR_Reflect_X))
|
2014-07-10 21:39:47 +00:00
|
|
|
ret |= 1 << META_MONITOR_TRANSFORM_FLIPPED_180;
|
2013-07-29 11:00:15 +00:00
|
|
|
if (rotation & (RR_Rotate_270 | RR_Reflect_X))
|
2014-07-10 21:39:47 +00:00
|
|
|
ret |= 1 << META_MONITOR_TRANSFORM_FLIPPED_270;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
compare_outputs (const void *one,
|
|
|
|
const void *two)
|
|
|
|
{
|
|
|
|
const MetaOutput *o_one = one, *o_two = two;
|
|
|
|
|
|
|
|
return strcmp (o_one->name, o_two->name);
|
|
|
|
}
|
|
|
|
|
2014-08-23 12:48:42 +00:00
|
|
|
static char *
|
|
|
|
get_xmode_name (XRRModeInfo *xmode)
|
|
|
|
{
|
|
|
|
int width = xmode->width;
|
|
|
|
int height = xmode->height;
|
|
|
|
|
|
|
|
return g_strdup_printf ("%dx%d", width, height);
|
|
|
|
}
|
|
|
|
|
2013-07-29 11:00:15 +00:00
|
|
|
static void
|
|
|
|
meta_monitor_manager_xrandr_read_current (MetaMonitorManager *manager)
|
|
|
|
{
|
|
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
|
|
|
XRRScreenResources *resources;
|
|
|
|
RROutput primary_output;
|
2017-03-24 09:35:51 +00:00
|
|
|
unsigned int i, j;
|
|
|
|
GList *l;
|
2013-07-29 11:00:15 +00:00
|
|
|
int min_width, min_height;
|
|
|
|
Screen *screen;
|
|
|
|
BOOL dpms_capable, dpms_enabled;
|
|
|
|
CARD16 dpms_state;
|
|
|
|
|
|
|
|
if (manager_xrandr->resources)
|
|
|
|
XRRFreeScreenResources (manager_xrandr->resources);
|
|
|
|
manager_xrandr->resources = NULL;
|
|
|
|
|
|
|
|
dpms_capable = DPMSCapable (manager_xrandr->xdisplay);
|
|
|
|
|
|
|
|
if (dpms_capable &&
|
|
|
|
DPMSInfo (manager_xrandr->xdisplay, &dpms_state, &dpms_enabled) &&
|
|
|
|
dpms_enabled)
|
|
|
|
{
|
|
|
|
switch (dpms_state)
|
2013-08-21 17:07:01 +00:00
|
|
|
{
|
|
|
|
case DPMSModeOn:
|
|
|
|
manager->power_save_mode = META_POWER_SAVE_ON;
|
|
|
|
break;
|
|
|
|
case DPMSModeStandby:
|
|
|
|
manager->power_save_mode = META_POWER_SAVE_STANDBY;
|
|
|
|
break;
|
|
|
|
case DPMSModeSuspend:
|
|
|
|
manager->power_save_mode = META_POWER_SAVE_SUSPEND;
|
|
|
|
break;
|
|
|
|
case DPMSModeOff:
|
|
|
|
manager->power_save_mode = META_POWER_SAVE_OFF;
|
|
|
|
break;
|
|
|
|
default:
|
2013-09-09 11:12:20 +00:00
|
|
|
manager->power_save_mode = META_POWER_SAVE_UNSUPPORTED;
|
2013-08-21 17:07:01 +00:00
|
|
|
break;
|
|
|
|
}
|
2013-07-29 11:00:15 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-09-09 11:12:20 +00:00
|
|
|
manager->power_save_mode = META_POWER_SAVE_UNSUPPORTED;
|
2013-07-29 11:00:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
XRRGetScreenSizeRange (manager_xrandr->xdisplay, DefaultRootWindow (manager_xrandr->xdisplay),
|
|
|
|
&min_width,
|
|
|
|
&min_height,
|
2017-02-15 08:06:46 +00:00
|
|
|
&manager_xrandr->max_screen_width,
|
|
|
|
&manager_xrandr->max_screen_height);
|
2013-07-29 11:00:15 +00:00
|
|
|
|
|
|
|
screen = ScreenOfDisplay (manager_xrandr->xdisplay,
|
|
|
|
DefaultScreen (manager_xrandr->xdisplay));
|
|
|
|
/* This is updated because we called RRUpdateConfiguration below */
|
|
|
|
manager->screen_width = WidthOfScreen (screen);
|
|
|
|
manager->screen_height = HeightOfScreen (screen);
|
|
|
|
|
|
|
|
resources = XRRGetScreenResourcesCurrent (manager_xrandr->xdisplay,
|
|
|
|
DefaultRootWindow (manager_xrandr->xdisplay));
|
|
|
|
if (!resources)
|
|
|
|
return;
|
|
|
|
|
|
|
|
manager_xrandr->resources = resources;
|
2017-03-24 09:35:51 +00:00
|
|
|
manager->outputs = NULL;
|
2017-03-28 08:52:15 +00:00
|
|
|
manager->modes = NULL;
|
2017-03-28 04:35:19 +00:00
|
|
|
manager->crtcs = NULL;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
|
|
|
for (i = 0; i < (unsigned)resources->nmode; i++)
|
|
|
|
{
|
|
|
|
XRRModeInfo *xmode = &resources->modes[i];
|
2016-12-09 07:49:54 +00:00
|
|
|
MetaCrtcMode *mode;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2017-03-28 08:52:15 +00:00
|
|
|
mode = g_object_new (META_TYPE_CRTC_MODE, NULL);
|
2013-07-29 11:00:15 +00:00
|
|
|
|
|
|
|
mode->mode_id = xmode->id;
|
|
|
|
mode->width = xmode->width;
|
|
|
|
mode->height = xmode->height;
|
|
|
|
mode->refresh_rate = (xmode->dotClock /
|
|
|
|
((float)xmode->hTotal * xmode->vTotal));
|
2015-04-21 15:49:55 +00:00
|
|
|
mode->flags = xmode->modeFlags;
|
2014-08-23 12:48:42 +00:00
|
|
|
mode->name = get_xmode_name (xmode);
|
2017-03-28 08:52:15 +00:00
|
|
|
|
|
|
|
manager->modes = g_list_append (manager->modes, mode);
|
2013-07-29 11:00:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < (unsigned)resources->ncrtc; i++)
|
|
|
|
{
|
2017-03-24 10:36:28 +00:00
|
|
|
XRRCrtcInfo *xrandr_crtc;
|
|
|
|
MetaCrtc *crtc;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2017-03-24 10:36:28 +00:00
|
|
|
xrandr_crtc = XRRGetCrtcInfo (manager_xrandr->xdisplay, resources,
|
|
|
|
resources->crtcs[i]);
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2017-03-28 04:35:19 +00:00
|
|
|
crtc = g_object_new (META_TYPE_CRTC, NULL);
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2017-07-04 07:58:44 +00:00
|
|
|
crtc->monitor_manager = manager;
|
2017-03-24 10:36:28 +00:00
|
|
|
crtc->crtc_id = resources->crtcs[i];
|
|
|
|
crtc->rect.x = xrandr_crtc->x;
|
|
|
|
crtc->rect.y = xrandr_crtc->y;
|
|
|
|
crtc->rect.width = xrandr_crtc->width;
|
|
|
|
crtc->rect.height = xrandr_crtc->height;
|
|
|
|
crtc->is_dirty = FALSE;
|
|
|
|
crtc->transform =
|
|
|
|
meta_monitor_transform_from_xrandr (xrandr_crtc->rotation);
|
|
|
|
crtc->all_transforms =
|
|
|
|
meta_monitor_transform_from_xrandr_all (xrandr_crtc->rotations);
|
2013-07-29 11:00:15 +00:00
|
|
|
|
|
|
|
for (j = 0; j < (unsigned)resources->nmode; j++)
|
|
|
|
{
|
2017-03-24 10:36:28 +00:00
|
|
|
if (resources->modes[j].id == xrandr_crtc->mode)
|
2013-07-29 11:00:15 +00:00
|
|
|
{
|
2017-03-28 08:52:15 +00:00
|
|
|
crtc->current_mode = g_list_nth_data (manager->modes, j);
|
2013-07-29 11:00:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-24 10:36:28 +00:00
|
|
|
XRRFreeCrtcInfo (xrandr_crtc);
|
2017-03-28 04:35:19 +00:00
|
|
|
|
|
|
|
manager->crtcs = g_list_append (manager->crtcs, crtc);
|
2013-07-29 11:00:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
primary_output = XRRGetOutputPrimary (manager_xrandr->xdisplay,
|
|
|
|
DefaultRootWindow (manager_xrandr->xdisplay));
|
|
|
|
|
|
|
|
for (i = 0; i < (unsigned)resources->noutput; i++)
|
|
|
|
{
|
2017-07-04 10:42:16 +00:00
|
|
|
RROutput output_id;
|
2017-03-24 05:59:15 +00:00
|
|
|
XRROutputInfo *xrandr_output;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2017-07-04 10:42:16 +00:00
|
|
|
output_id = resources->outputs[i];
|
2017-03-24 05:59:15 +00:00
|
|
|
xrandr_output = XRRGetOutputInfo (manager_xrandr->xdisplay,
|
2017-07-04 10:42:16 +00:00
|
|
|
resources, output_id);
|
2017-03-24 05:59:15 +00:00
|
|
|
if (!xrandr_output)
|
2015-10-15 17:34:40 +00:00
|
|
|
continue;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2017-03-24 05:59:15 +00:00
|
|
|
if (xrandr_output->connection != RR_Disconnected)
|
2013-07-29 11:00:15 +00:00
|
|
|
{
|
2017-07-04 10:42:16 +00:00
|
|
|
MetaOutput *output;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2017-07-04 10:42:16 +00:00
|
|
|
output = meta_create_xrandr_output (manager,
|
|
|
|
xrandr_output,
|
|
|
|
output_id,
|
|
|
|
primary_output);
|
|
|
|
if (output)
|
2017-03-24 09:35:51 +00:00
|
|
|
manager->outputs = g_list_prepend (manager->outputs, output);
|
2013-07-29 11:00:15 +00:00
|
|
|
}
|
|
|
|
|
2017-03-24 05:59:15 +00:00
|
|
|
XRRFreeOutputInfo (xrandr_output);
|
2013-07-29 11:00:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Sort the outputs for easier handling in MetaMonitorConfig */
|
2017-03-24 09:35:51 +00:00
|
|
|
manager->outputs = g_list_sort (manager->outputs, compare_outputs);
|
2013-07-29 11:00:15 +00:00
|
|
|
|
|
|
|
/* Now fix the clones */
|
2017-03-24 09:35:51 +00:00
|
|
|
for (l = manager->outputs; l; l = l->next)
|
2013-07-29 11:00:15 +00:00
|
|
|
{
|
2017-03-24 09:35:51 +00:00
|
|
|
MetaOutput *output = l->data;
|
|
|
|
GList *k;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2017-03-24 05:59:15 +00:00
|
|
|
for (j = 0; j < output->n_possible_clones; j++)
|
2013-07-29 11:00:15 +00:00
|
|
|
{
|
2017-03-24 05:59:15 +00:00
|
|
|
RROutput clone = GPOINTER_TO_INT (output->possible_clones[j]);
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2017-03-24 09:35:51 +00:00
|
|
|
for (k = manager->outputs; k; k = k->next)
|
2013-07-29 11:00:15 +00:00
|
|
|
{
|
2017-03-24 09:35:51 +00:00
|
|
|
MetaOutput *possible_clone = k->data;
|
|
|
|
|
|
|
|
if (clone == (XID) possible_clone->winsys_id)
|
2013-07-29 11:00:15 +00:00
|
|
|
{
|
2017-03-24 09:35:51 +00:00
|
|
|
output->possible_clones[j] = possible_clone;
|
2013-07-29 11:00:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-08 11:32:05 +00:00
|
|
|
static GBytes *
|
|
|
|
meta_monitor_manager_xrandr_read_edid (MetaMonitorManager *manager,
|
|
|
|
MetaOutput *output)
|
|
|
|
{
|
2017-07-04 10:42:16 +00:00
|
|
|
return meta_output_xrandr_read_edid (output);
|
2013-08-08 11:32:05 +00:00
|
|
|
}
|
|
|
|
|
2013-07-29 11:00:15 +00:00
|
|
|
static void
|
|
|
|
meta_monitor_manager_xrandr_set_power_save_mode (MetaMonitorManager *manager,
|
|
|
|
MetaPowerSave mode)
|
|
|
|
{
|
|
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
|
|
|
CARD16 state;
|
|
|
|
|
|
|
|
switch (mode) {
|
|
|
|
case META_POWER_SAVE_ON:
|
|
|
|
state = DPMSModeOn;
|
|
|
|
break;
|
|
|
|
case META_POWER_SAVE_STANDBY:
|
|
|
|
state = DPMSModeStandby;
|
|
|
|
break;
|
|
|
|
case META_POWER_SAVE_SUSPEND:
|
|
|
|
state = DPMSModeSuspend;
|
|
|
|
break;
|
|
|
|
case META_POWER_SAVE_OFF:
|
|
|
|
state = DPMSModeOff;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
DPMSForceLevel (manager_xrandr->xdisplay, state);
|
|
|
|
DPMSSetTimeouts (manager_xrandr->xdisplay, 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
2017-04-18 09:06:18 +00:00
|
|
|
static xcb_randr_rotation_t
|
2014-07-10 21:39:47 +00:00
|
|
|
meta_monitor_transform_to_xrandr (MetaMonitorTransform transform)
|
2013-07-29 11:00:15 +00:00
|
|
|
{
|
|
|
|
switch (transform)
|
|
|
|
{
|
2014-07-10 21:39:47 +00:00
|
|
|
case META_MONITOR_TRANSFORM_NORMAL:
|
2017-04-18 09:06:18 +00:00
|
|
|
return XCB_RANDR_ROTATION_ROTATE_0;
|
2014-07-10 21:39:47 +00:00
|
|
|
case META_MONITOR_TRANSFORM_90:
|
2017-04-18 09:06:18 +00:00
|
|
|
return XCB_RANDR_ROTATION_ROTATE_90;
|
2014-07-10 21:39:47 +00:00
|
|
|
case META_MONITOR_TRANSFORM_180:
|
2017-04-18 09:06:18 +00:00
|
|
|
return XCB_RANDR_ROTATION_ROTATE_180;
|
2014-07-10 21:39:47 +00:00
|
|
|
case META_MONITOR_TRANSFORM_270:
|
2017-04-18 09:06:18 +00:00
|
|
|
return XCB_RANDR_ROTATION_ROTATE_270;
|
2014-07-10 21:39:47 +00:00
|
|
|
case META_MONITOR_TRANSFORM_FLIPPED:
|
2017-04-18 09:06:18 +00:00
|
|
|
return XCB_RANDR_ROTATION_REFLECT_X | XCB_RANDR_ROTATION_ROTATE_0;
|
2014-07-10 21:39:47 +00:00
|
|
|
case META_MONITOR_TRANSFORM_FLIPPED_90:
|
2017-04-18 09:06:18 +00:00
|
|
|
return XCB_RANDR_ROTATION_REFLECT_X | XCB_RANDR_ROTATION_ROTATE_90;
|
2014-07-10 21:39:47 +00:00
|
|
|
case META_MONITOR_TRANSFORM_FLIPPED_180:
|
2017-04-18 09:06:18 +00:00
|
|
|
return XCB_RANDR_ROTATION_REFLECT_X | XCB_RANDR_ROTATION_ROTATE_180;
|
2014-07-10 21:39:47 +00:00
|
|
|
case META_MONITOR_TRANSFORM_FLIPPED_270:
|
2017-04-18 09:06:18 +00:00
|
|
|
return XCB_RANDR_ROTATION_REFLECT_X | XCB_RANDR_ROTATION_ROTATE_270;
|
2013-07-29 11:00:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
g_assert_not_reached ();
|
|
|
|
}
|
|
|
|
|
2017-04-18 09:06:18 +00:00
|
|
|
static gboolean
|
|
|
|
xrandr_set_crtc_config (MetaMonitorManagerXrandr *manager_xrandr,
|
2017-04-21 10:01:15 +00:00
|
|
|
gboolean save_timestamp,
|
2017-04-18 09:06:18 +00:00
|
|
|
xcb_randr_crtc_t crtc,
|
|
|
|
xcb_timestamp_t timestamp,
|
|
|
|
int x,
|
|
|
|
int y,
|
|
|
|
xcb_randr_mode_t mode,
|
|
|
|
xcb_randr_rotation_t rotation,
|
|
|
|
xcb_randr_output_t *outputs,
|
|
|
|
int n_outputs)
|
|
|
|
{
|
|
|
|
xcb_connection_t *xcb_conn;
|
|
|
|
xcb_timestamp_t config_timestamp;
|
|
|
|
xcb_randr_set_crtc_config_cookie_t cookie;
|
|
|
|
xcb_randr_set_crtc_config_reply_t *reply;
|
|
|
|
xcb_generic_error_t *xcb_error = NULL;
|
|
|
|
|
|
|
|
xcb_conn = XGetXCBConnection (manager_xrandr->xdisplay);
|
|
|
|
config_timestamp = manager_xrandr->resources->configTimestamp;
|
|
|
|
cookie = xcb_randr_set_crtc_config (xcb_conn,
|
|
|
|
crtc,
|
|
|
|
timestamp,
|
|
|
|
config_timestamp,
|
|
|
|
x, y,
|
|
|
|
mode,
|
|
|
|
rotation,
|
|
|
|
n_outputs,
|
|
|
|
outputs);
|
|
|
|
reply = xcb_randr_set_crtc_config_reply (xcb_conn,
|
|
|
|
cookie,
|
|
|
|
&xcb_error);
|
|
|
|
if (xcb_error || !reply)
|
|
|
|
{
|
|
|
|
free (xcb_error);
|
|
|
|
free (reply);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2017-04-21 10:01:15 +00:00
|
|
|
if (save_timestamp)
|
|
|
|
manager_xrandr->last_xrandr_set_timestamp = reply->timestamp;
|
|
|
|
|
2017-04-18 09:06:18 +00:00
|
|
|
free (reply);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2017-04-21 10:01:15 +00:00
|
|
|
static gboolean
|
|
|
|
is_crtc_assignment_changed (MetaCrtc *crtc,
|
|
|
|
MetaCrtcInfo **crtc_infos,
|
|
|
|
unsigned int n_crtc_infos)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i < n_crtc_infos; i++)
|
|
|
|
{
|
|
|
|
MetaCrtcInfo *crtc_info = crtc_infos[i];
|
|
|
|
unsigned int j;
|
|
|
|
|
|
|
|
if (crtc_info->crtc != crtc)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (crtc->current_mode != crtc_info->mode)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if (crtc->rect.x != crtc_info->x)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if (crtc->rect.y != crtc_info->y)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if (crtc->transform != crtc_info->transform)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
for (j = 0; j < crtc_info->outputs->len; j++)
|
|
|
|
{
|
|
|
|
MetaOutput *output = ((MetaOutput**) crtc_info->outputs->pdata)[j];
|
|
|
|
|
|
|
|
if (output->crtc != crtc)
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return crtc->current_mode != NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
is_output_assignment_changed (MetaOutput *output,
|
|
|
|
MetaCrtcInfo **crtc_infos,
|
|
|
|
unsigned int n_crtc_infos,
|
|
|
|
MetaOutputInfo **output_infos,
|
|
|
|
unsigned int n_output_infos)
|
|
|
|
{
|
|
|
|
gboolean output_is_found = FALSE;
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i < n_output_infos; i++)
|
|
|
|
{
|
|
|
|
MetaOutputInfo *output_info = output_infos[i];
|
|
|
|
|
|
|
|
if (output_info->output != output)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (output->is_primary != output_info->is_primary)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if (output->is_presentation != output_info->is_presentation)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if (output->is_underscanning != output_info->is_underscanning)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
output_is_found = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!output_is_found)
|
|
|
|
return output->crtc != NULL;
|
|
|
|
|
|
|
|
for (i = 0; i < n_crtc_infos; i++)
|
|
|
|
{
|
|
|
|
MetaCrtcInfo *crtc_info = crtc_infos[i];
|
|
|
|
unsigned int j;
|
|
|
|
|
|
|
|
for (j = 0; j < crtc_info->outputs->len; j++)
|
|
|
|
{
|
|
|
|
MetaOutput *crtc_info_output =
|
|
|
|
((MetaOutput**) crtc_info->outputs->pdata)[j];
|
|
|
|
|
|
|
|
if (crtc_info_output == output &&
|
|
|
|
crtc_info->crtc == output->crtc)
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
is_assignments_changed (MetaMonitorManager *manager,
|
|
|
|
MetaCrtcInfo **crtc_infos,
|
|
|
|
unsigned int n_crtc_infos,
|
|
|
|
MetaOutputInfo **output_infos,
|
|
|
|
unsigned int n_output_infos)
|
|
|
|
{
|
2017-03-24 09:35:51 +00:00
|
|
|
GList *l;
|
2017-04-21 10:01:15 +00:00
|
|
|
|
2017-03-28 04:35:19 +00:00
|
|
|
for (l = manager->crtcs; l; l = l->next)
|
2017-04-21 10:01:15 +00:00
|
|
|
{
|
2017-03-28 04:35:19 +00:00
|
|
|
MetaCrtc *crtc = l->data;
|
2017-04-21 10:01:15 +00:00
|
|
|
|
|
|
|
if (is_crtc_assignment_changed (crtc, crtc_infos, n_crtc_infos))
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2017-03-24 09:35:51 +00:00
|
|
|
for (l = manager->outputs; l; l = l->next)
|
2017-04-21 10:01:15 +00:00
|
|
|
{
|
2017-03-24 09:35:51 +00:00
|
|
|
MetaOutput *output = l->data;
|
2017-04-21 10:01:15 +00:00
|
|
|
|
|
|
|
if (is_output_assignment_changed (output,
|
|
|
|
crtc_infos,
|
|
|
|
n_crtc_infos,
|
|
|
|
output_infos,
|
|
|
|
n_output_infos))
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2013-07-29 11:00:15 +00:00
|
|
|
static void
|
2017-01-09 06:31:18 +00:00
|
|
|
apply_crtc_assignments (MetaMonitorManager *manager,
|
2017-04-21 10:01:15 +00:00
|
|
|
gboolean save_timestamp,
|
2017-01-09 06:31:18 +00:00
|
|
|
MetaCrtcInfo **crtcs,
|
|
|
|
unsigned int n_crtcs,
|
|
|
|
MetaOutputInfo **outputs,
|
|
|
|
unsigned int n_outputs)
|
2013-07-29 11:00:15 +00:00
|
|
|
{
|
|
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
|
|
|
unsigned i;
|
2017-03-24 09:35:51 +00:00
|
|
|
GList *l;
|
2013-08-16 15:39:07 +00:00
|
|
|
int width, height, width_mm, height_mm;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2014-04-23 13:57:16 +00:00
|
|
|
XGrabServer (manager_xrandr->xdisplay);
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2013-08-16 17:02:57 +00:00
|
|
|
/* First compute the new size of the screen (framebuffer) */
|
2013-08-16 15:39:07 +00:00
|
|
|
width = 0; height = 0;
|
|
|
|
for (i = 0; i < n_crtcs; i++)
|
|
|
|
{
|
2016-12-08 04:15:28 +00:00
|
|
|
MetaCrtcInfo *crtc_info = crtcs[i];
|
|
|
|
MetaCrtc *crtc = crtc_info->crtc;
|
2013-08-16 17:02:57 +00:00
|
|
|
crtc->is_dirty = TRUE;
|
2013-08-16 15:39:07 +00:00
|
|
|
|
|
|
|
if (crtc_info->mode == NULL)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (meta_monitor_transform_is_rotated (crtc_info->transform))
|
|
|
|
{
|
|
|
|
width = MAX (width, crtc_info->x + crtc_info->mode->height);
|
|
|
|
height = MAX (height, crtc_info->y + crtc_info->mode->width);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
width = MAX (width, crtc_info->x + crtc_info->mode->width);
|
|
|
|
height = MAX (height, crtc_info->y + crtc_info->mode->height);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-16 17:02:57 +00:00
|
|
|
/* Second disable all newly disabled CRTCs, or CRTCs that in the previous
|
|
|
|
configuration would be outside the new framebuffer (otherwise X complains
|
|
|
|
loudly when resizing)
|
|
|
|
CRTC will be enabled again after resizing the FB
|
|
|
|
*/
|
|
|
|
for (i = 0; i < n_crtcs; i++)
|
|
|
|
{
|
2016-12-08 04:15:28 +00:00
|
|
|
MetaCrtcInfo *crtc_info = crtcs[i];
|
|
|
|
MetaCrtc *crtc = crtc_info->crtc;
|
2013-08-16 17:02:57 +00:00
|
|
|
|
|
|
|
if (crtc_info->mode == NULL ||
|
|
|
|
crtc->rect.x + crtc->rect.width > width ||
|
|
|
|
crtc->rect.y + crtc->rect.height > height)
|
|
|
|
{
|
2017-04-18 09:06:18 +00:00
|
|
|
xrandr_set_crtc_config (manager_xrandr,
|
2017-04-21 10:01:15 +00:00
|
|
|
save_timestamp,
|
2017-04-18 09:06:18 +00:00
|
|
|
(xcb_randr_crtc_t) crtc->crtc_id,
|
|
|
|
XCB_CURRENT_TIME,
|
|
|
|
0, 0, XCB_NONE,
|
|
|
|
XCB_RANDR_ROTATION_ROTATE_0,
|
|
|
|
NULL, 0);
|
2013-08-17 17:15:40 +00:00
|
|
|
|
|
|
|
crtc->rect.x = 0;
|
|
|
|
crtc->rect.y = 0;
|
|
|
|
crtc->rect.width = 0;
|
|
|
|
crtc->rect.height = 0;
|
|
|
|
crtc->current_mode = NULL;
|
2013-08-16 17:02:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Disable CRTCs not mentioned in the list */
|
2017-03-28 04:35:19 +00:00
|
|
|
for (l = manager->crtcs; l; l = l->next)
|
2013-08-16 17:02:57 +00:00
|
|
|
{
|
2017-03-28 04:35:19 +00:00
|
|
|
MetaCrtc *crtc = l->data;
|
2013-08-16 17:02:57 +00:00
|
|
|
|
|
|
|
if (crtc->is_dirty)
|
|
|
|
{
|
|
|
|
crtc->is_dirty = FALSE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (crtc->current_mode == NULL)
|
|
|
|
continue;
|
|
|
|
|
2017-04-18 09:06:18 +00:00
|
|
|
xrandr_set_crtc_config (manager_xrandr,
|
2017-04-21 10:01:15 +00:00
|
|
|
save_timestamp,
|
2017-04-18 09:06:18 +00:00
|
|
|
(xcb_randr_crtc_t) crtc->crtc_id,
|
|
|
|
XCB_CURRENT_TIME,
|
|
|
|
0, 0, XCB_NONE,
|
|
|
|
XCB_RANDR_ROTATION_ROTATE_0,
|
|
|
|
NULL, 0);
|
2013-08-17 17:15:40 +00:00
|
|
|
|
|
|
|
crtc->rect.x = 0;
|
|
|
|
crtc->rect.y = 0;
|
|
|
|
crtc->rect.width = 0;
|
|
|
|
crtc->rect.height = 0;
|
|
|
|
crtc->current_mode = NULL;
|
2013-08-16 17:02:57 +00:00
|
|
|
}
|
|
|
|
|
2013-08-16 15:39:07 +00:00
|
|
|
g_assert (width > 0 && height > 0);
|
|
|
|
/* The 'physical size' of an X screen is meaningless if that screen
|
|
|
|
* can consist of many monitors. So just pick a size that make the
|
|
|
|
* dpi 96.
|
|
|
|
*
|
|
|
|
* Firefox and Evince apparently believe what X tells them.
|
|
|
|
*/
|
|
|
|
width_mm = (width / DPI_FALLBACK) * 25.4 + 0.5;
|
|
|
|
height_mm = (height / DPI_FALLBACK) * 25.4 + 0.5;
|
|
|
|
XRRSetScreenSize (manager_xrandr->xdisplay, DefaultRootWindow (manager_xrandr->xdisplay),
|
|
|
|
width, height, width_mm, height_mm);
|
|
|
|
|
2013-07-29 11:00:15 +00:00
|
|
|
for (i = 0; i < n_crtcs; i++)
|
|
|
|
{
|
2016-12-08 04:15:28 +00:00
|
|
|
MetaCrtcInfo *crtc_info = crtcs[i];
|
|
|
|
MetaCrtc *crtc = crtc_info->crtc;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2013-08-16 17:02:57 +00:00
|
|
|
if (crtc_info->mode != NULL)
|
2013-07-29 11:00:15 +00:00
|
|
|
{
|
2016-12-09 07:49:54 +00:00
|
|
|
MetaCrtcMode *mode;
|
2017-04-18 09:06:18 +00:00
|
|
|
g_autofree xcb_randr_output_t *output_ids = NULL;
|
2015-09-23 23:24:33 +00:00
|
|
|
unsigned int j, n_output_ids;
|
2017-04-18 09:06:18 +00:00
|
|
|
xcb_randr_rotation_t rotation;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
|
|
|
mode = crtc_info->mode;
|
|
|
|
|
2015-09-23 23:24:33 +00:00
|
|
|
n_output_ids = crtc_info->outputs->len;
|
2017-04-18 09:06:18 +00:00
|
|
|
output_ids = g_new (xcb_randr_output_t, n_output_ids);
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2015-09-23 23:24:33 +00:00
|
|
|
for (j = 0; j < n_output_ids; j++)
|
2013-08-23 15:25:44 +00:00
|
|
|
{
|
|
|
|
MetaOutput *output;
|
|
|
|
|
|
|
|
output = ((MetaOutput**)crtc_info->outputs->pdata)[j];
|
|
|
|
|
|
|
|
output->is_dirty = TRUE;
|
|
|
|
output->crtc = crtc;
|
|
|
|
|
2015-09-23 23:24:33 +00:00
|
|
|
output_ids[j] = output->winsys_id;
|
2013-08-23 15:25:44 +00:00
|
|
|
}
|
|
|
|
|
2017-04-18 09:06:18 +00:00
|
|
|
rotation = meta_monitor_transform_to_xrandr (crtc_info->transform);
|
|
|
|
if (!xrandr_set_crtc_config (manager_xrandr,
|
2017-04-21 10:01:15 +00:00
|
|
|
save_timestamp,
|
2017-04-18 09:06:18 +00:00
|
|
|
(xcb_randr_crtc_t) crtc->crtc_id,
|
|
|
|
XCB_CURRENT_TIME,
|
|
|
|
crtc_info->x, crtc_info->y,
|
|
|
|
(xcb_randr_mode_t) mode->mode_id,
|
|
|
|
rotation,
|
|
|
|
output_ids, n_output_ids))
|
2013-08-17 17:15:40 +00:00
|
|
|
{
|
2015-01-05 23:23:17 +00:00
|
|
|
meta_warning ("Configuring CRTC %d with mode %d (%d x %d @ %f) at position %d, %d and transform %u failed\n",
|
2013-08-17 17:15:40 +00:00
|
|
|
(unsigned)(crtc->crtc_id), (unsigned)(mode->mode_id),
|
|
|
|
mode->width, mode->height, (float)mode->refresh_rate,
|
|
|
|
crtc_info->x, crtc_info->y, crtc_info->transform);
|
2015-05-30 00:17:34 +00:00
|
|
|
continue;
|
2013-08-17 17:15:40 +00:00
|
|
|
}
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2013-08-17 17:15:40 +00:00
|
|
|
if (meta_monitor_transform_is_rotated (crtc_info->transform))
|
|
|
|
{
|
|
|
|
width = mode->height;
|
|
|
|
height = mode->width;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
width = mode->width;
|
|
|
|
height = mode->height;
|
|
|
|
}
|
|
|
|
|
|
|
|
crtc->rect.x = crtc_info->x;
|
|
|
|
crtc->rect.y = crtc_info->y;
|
|
|
|
crtc->rect.width = width;
|
|
|
|
crtc->rect.height = height;
|
|
|
|
crtc->current_mode = mode;
|
|
|
|
crtc->transform = crtc_info->transform;
|
2013-07-29 11:00:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < n_outputs; i++)
|
|
|
|
{
|
|
|
|
MetaOutputInfo *output_info = outputs[i];
|
2013-08-17 17:15:40 +00:00
|
|
|
MetaOutput *output = output_info->output;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2013-08-17 17:15:40 +00:00
|
|
|
output->is_primary = output_info->is_primary;
|
|
|
|
output->is_presentation = output_info->is_presentation;
|
2014-04-16 08:20:55 +00:00
|
|
|
output->is_underscanning = output_info->is_underscanning;
|
2017-07-04 10:42:16 +00:00
|
|
|
|
|
|
|
meta_output_xrandr_apply_mode (output);
|
2013-08-17 17:15:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Disable outputs not mentioned in the list */
|
2017-03-24 09:35:51 +00:00
|
|
|
for (l = manager->outputs; l; l = l->next)
|
2013-08-17 17:15:40 +00:00
|
|
|
{
|
2017-03-24 09:35:51 +00:00
|
|
|
MetaOutput *output = l->data;
|
2013-08-17 17:15:40 +00:00
|
|
|
|
|
|
|
if (output->is_dirty)
|
|
|
|
{
|
|
|
|
output->is_dirty = FALSE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
output->crtc = NULL;
|
|
|
|
output->is_primary = FALSE;
|
2013-07-29 11:00:15 +00:00
|
|
|
}
|
|
|
|
|
2014-04-23 13:57:16 +00:00
|
|
|
XUngrabServer (manager_xrandr->xdisplay);
|
2014-05-22 21:38:49 +00:00
|
|
|
XFlush (manager_xrandr->xdisplay);
|
2013-07-29 11:00:15 +00:00
|
|
|
}
|
|
|
|
|
2017-01-06 05:27:21 +00:00
|
|
|
static void
|
|
|
|
meta_monitor_manager_xrandr_ensure_initial_config (MetaMonitorManager *manager)
|
|
|
|
{
|
2017-09-11 06:10:26 +00:00
|
|
|
MetaMonitorConfigManager *config_manager =
|
|
|
|
meta_monitor_manager_get_config_manager (manager);
|
|
|
|
MetaMonitorsConfig *config;
|
2017-04-21 10:01:15 +00:00
|
|
|
|
2017-01-06 05:27:21 +00:00
|
|
|
meta_monitor_manager_ensure_configured (manager);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Normally we don't rebuild our data structures until we see the
|
|
|
|
* RRScreenNotify event, but at least at startup we want to have the right
|
|
|
|
* configuration immediately.
|
|
|
|
*/
|
|
|
|
meta_monitor_manager_read_current_state (manager);
|
|
|
|
|
2017-09-11 06:10:26 +00:00
|
|
|
config = meta_monitor_config_manager_get_current (config_manager);
|
|
|
|
meta_monitor_manager_update_logical_state_derived (manager, config);
|
2017-01-06 05:27:21 +00:00
|
|
|
}
|
|
|
|
|
2017-01-09 06:31:18 +00:00
|
|
|
static gboolean
|
2017-02-14 11:54:04 +00:00
|
|
|
meta_monitor_manager_xrandr_apply_monitors_config (MetaMonitorManager *manager,
|
|
|
|
MetaMonitorsConfig *config,
|
|
|
|
MetaMonitorsConfigMethod method,
|
|
|
|
GError **error)
|
2017-01-09 06:31:18 +00:00
|
|
|
{
|
|
|
|
GPtrArray *crtc_infos;
|
|
|
|
GPtrArray *output_infos;
|
|
|
|
|
2017-01-20 06:50:11 +00:00
|
|
|
if (!config)
|
|
|
|
{
|
2017-09-11 06:10:26 +00:00
|
|
|
meta_monitor_manager_rebuild_derived (manager, NULL);
|
2017-01-20 06:50:11 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2017-01-09 06:31:18 +00:00
|
|
|
if (!meta_monitor_config_manager_assign (manager, config,
|
|
|
|
&crtc_infos, &output_infos,
|
|
|
|
error))
|
|
|
|
return FALSE;
|
|
|
|
|
2017-02-14 11:54:04 +00:00
|
|
|
if (method != META_MONITORS_CONFIG_METHOD_VERIFY)
|
|
|
|
{
|
2017-04-21 10:01:15 +00:00
|
|
|
/*
|
|
|
|
* If the assignment has not changed, we won't get any notification about
|
|
|
|
* any new configuration from the X server; but we still need to update
|
|
|
|
* our own configuration, as something not applicable in Xrandr might
|
|
|
|
* have changed locally, such as the logical monitors scale. This means we
|
|
|
|
* must check that our new assignment actually changes anything, otherwise
|
|
|
|
* just update the logical state.
|
|
|
|
*/
|
|
|
|
if (is_assignments_changed (manager,
|
|
|
|
(MetaCrtcInfo **) crtc_infos->pdata,
|
|
|
|
crtc_infos->len,
|
|
|
|
(MetaOutputInfo **) output_infos->pdata,
|
|
|
|
output_infos->len))
|
|
|
|
{
|
|
|
|
apply_crtc_assignments (manager,
|
|
|
|
TRUE,
|
|
|
|
(MetaCrtcInfo **) crtc_infos->pdata,
|
|
|
|
crtc_infos->len,
|
|
|
|
(MetaOutputInfo **) output_infos->pdata,
|
|
|
|
output_infos->len);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-09-11 06:10:26 +00:00
|
|
|
meta_monitor_manager_rebuild_derived (manager, config);
|
2017-04-21 10:01:15 +00:00
|
|
|
}
|
2017-02-14 11:54:04 +00:00
|
|
|
}
|
2017-01-09 06:31:18 +00:00
|
|
|
|
|
|
|
g_ptr_array_free (crtc_infos, TRUE);
|
|
|
|
g_ptr_array_free (output_infos, TRUE);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2013-07-29 11:00:15 +00:00
|
|
|
static void
|
|
|
|
meta_monitor_manager_xrandr_change_backlight (MetaMonitorManager *manager,
|
|
|
|
MetaOutput *output,
|
|
|
|
gint value)
|
|
|
|
{
|
2017-07-04 10:42:16 +00:00
|
|
|
meta_output_xrandr_change_backlight (output, value);
|
2013-07-29 11:00:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_monitor_manager_xrandr_get_crtc_gamma (MetaMonitorManager *manager,
|
2016-12-08 04:15:28 +00:00
|
|
|
MetaCrtc *crtc,
|
2013-07-29 11:00:15 +00:00
|
|
|
gsize *size,
|
|
|
|
unsigned short **red,
|
|
|
|
unsigned short **green,
|
|
|
|
unsigned short **blue)
|
|
|
|
{
|
|
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
|
|
|
XRRCrtcGamma *gamma;
|
|
|
|
|
|
|
|
gamma = XRRGetCrtcGamma (manager_xrandr->xdisplay, (XID)crtc->crtc_id);
|
|
|
|
|
|
|
|
*size = gamma->size;
|
|
|
|
*red = g_memdup (gamma->red, sizeof (unsigned short) * gamma->size);
|
|
|
|
*green = g_memdup (gamma->green, sizeof (unsigned short) * gamma->size);
|
|
|
|
*blue = g_memdup (gamma->blue, sizeof (unsigned short) * gamma->size);
|
|
|
|
|
|
|
|
XRRFreeGamma (gamma);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_monitor_manager_xrandr_set_crtc_gamma (MetaMonitorManager *manager,
|
2016-12-08 04:15:28 +00:00
|
|
|
MetaCrtc *crtc,
|
2013-07-29 11:00:15 +00:00
|
|
|
gsize size,
|
|
|
|
unsigned short *red,
|
|
|
|
unsigned short *green,
|
|
|
|
unsigned short *blue)
|
|
|
|
{
|
|
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
2013-08-18 09:44:45 +00:00
|
|
|
XRRCrtcGamma *gamma;
|
|
|
|
|
|
|
|
gamma = XRRAllocGamma (size);
|
|
|
|
memcpy (gamma->red, red, sizeof (unsigned short) * size);
|
|
|
|
memcpy (gamma->green, green, sizeof (unsigned short) * size);
|
|
|
|
memcpy (gamma->blue, blue, sizeof (unsigned short) * size);
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2013-08-18 09:44:45 +00:00
|
|
|
XRRSetCrtcGamma (manager_xrandr->xdisplay, (XID)crtc->crtc_id, gamma);
|
2013-07-29 11:00:15 +00:00
|
|
|
|
2013-08-18 09:44:45 +00:00
|
|
|
XRRFreeGamma (gamma);
|
2013-07-29 11:00:15 +00:00
|
|
|
}
|
|
|
|
|
2015-03-31 01:11:12 +00:00
|
|
|
#ifdef HAVE_XRANDR15
|
2016-12-13 10:58:22 +00:00
|
|
|
static MetaMonitorXrandrData *
|
|
|
|
meta_monitor_xrandr_data_from_monitor (MetaMonitor *monitor)
|
|
|
|
{
|
|
|
|
MetaMonitorXrandrData *monitor_xrandr_data;
|
|
|
|
|
|
|
|
monitor_xrandr_data = g_object_get_qdata (G_OBJECT (monitor),
|
|
|
|
quark_meta_monitor_xrandr_data);
|
|
|
|
if (monitor_xrandr_data)
|
|
|
|
return monitor_xrandr_data;
|
|
|
|
|
|
|
|
monitor_xrandr_data = g_new0 (MetaMonitorXrandrData, 1);
|
|
|
|
g_object_set_qdata_full (G_OBJECT (monitor),
|
|
|
|
quark_meta_monitor_xrandr_data,
|
|
|
|
monitor_xrandr_data,
|
|
|
|
g_free);
|
|
|
|
|
|
|
|
return monitor_xrandr_data;
|
|
|
|
}
|
|
|
|
|
2015-03-31 01:11:12 +00:00
|
|
|
static void
|
2016-12-13 10:58:22 +00:00
|
|
|
meta_monitor_manager_xrandr_increase_monitor_count (MetaMonitorManagerXrandr *manager_xrandr,
|
|
|
|
Atom name_atom)
|
|
|
|
{
|
|
|
|
int count;
|
|
|
|
|
|
|
|
count =
|
|
|
|
GPOINTER_TO_INT (g_hash_table_lookup (manager_xrandr->tiled_monitor_atoms,
|
|
|
|
GSIZE_TO_POINTER (name_atom)));
|
|
|
|
|
|
|
|
count++;
|
|
|
|
g_hash_table_insert (manager_xrandr->tiled_monitor_atoms,
|
|
|
|
GSIZE_TO_POINTER (name_atom),
|
|
|
|
GINT_TO_POINTER (count));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
meta_monitor_manager_xrandr_decrease_monitor_count (MetaMonitorManagerXrandr *manager_xrandr,
|
|
|
|
Atom name_atom)
|
|
|
|
{
|
|
|
|
int count;
|
|
|
|
|
|
|
|
count =
|
|
|
|
GPOINTER_TO_SIZE (g_hash_table_lookup (manager_xrandr->tiled_monitor_atoms,
|
|
|
|
GSIZE_TO_POINTER (name_atom)));
|
|
|
|
g_assert (count > 0);
|
|
|
|
|
|
|
|
count--;
|
|
|
|
g_hash_table_insert (manager_xrandr->tiled_monitor_atoms,
|
|
|
|
GSIZE_TO_POINTER (name_atom),
|
|
|
|
GINT_TO_POINTER (count));
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_monitor_manager_xrandr_tiled_monitor_added (MetaMonitorManager *manager,
|
|
|
|
MetaMonitor *monitor)
|
2015-03-31 01:11:12 +00:00
|
|
|
{
|
|
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
2016-12-13 10:58:22 +00:00
|
|
|
MetaMonitorTiled *monitor_tiled = META_MONITOR_TILED (monitor);
|
|
|
|
const char *product;
|
|
|
|
char *name;
|
|
|
|
uint32_t tile_group_id;
|
|
|
|
MetaMonitorXrandrData *monitor_xrandr_data;
|
|
|
|
Atom name_atom;
|
|
|
|
XRRMonitorInfo *xrandr_monitor_info;
|
|
|
|
GList *outputs;
|
|
|
|
GList *l;
|
|
|
|
int i;
|
2015-03-31 01:11:12 +00:00
|
|
|
|
|
|
|
if (manager_xrandr->has_randr15 == FALSE)
|
|
|
|
return;
|
|
|
|
|
2016-12-13 10:58:22 +00:00
|
|
|
product = meta_monitor_get_product (monitor);
|
|
|
|
tile_group_id = meta_monitor_tiled_get_tile_group_id (monitor_tiled);
|
2015-03-31 01:11:12 +00:00
|
|
|
|
2016-12-13 10:58:22 +00:00
|
|
|
if (product)
|
|
|
|
name = g_strdup_printf ("%s-%d", product, tile_group_id);
|
2015-03-31 01:11:12 +00:00
|
|
|
else
|
2016-12-13 10:58:22 +00:00
|
|
|
name = g_strdup_printf ("Tiled-%d", tile_group_id);
|
|
|
|
|
|
|
|
name_atom = XInternAtom (manager_xrandr->xdisplay, name, False);
|
|
|
|
g_free (name);
|
|
|
|
|
|
|
|
monitor_xrandr_data = meta_monitor_xrandr_data_from_monitor (monitor);
|
|
|
|
monitor_xrandr_data->xrandr_name = name_atom;
|
|
|
|
|
|
|
|
meta_monitor_manager_xrandr_increase_monitor_count (manager_xrandr,
|
|
|
|
name_atom);
|
|
|
|
|
|
|
|
outputs = meta_monitor_get_outputs (monitor);
|
|
|
|
xrandr_monitor_info = XRRAllocateMonitor (manager_xrandr->xdisplay,
|
|
|
|
g_list_length (outputs));
|
|
|
|
xrandr_monitor_info->name = name_atom;
|
|
|
|
xrandr_monitor_info->primary = meta_monitor_is_primary (monitor);
|
|
|
|
xrandr_monitor_info->automatic = True;
|
|
|
|
for (l = outputs, i = 0; l; l = l->next, i++)
|
|
|
|
{
|
|
|
|
MetaOutput *output = l->data;
|
|
|
|
|
|
|
|
xrandr_monitor_info->outputs[i] = output->winsys_id;
|
|
|
|
}
|
2015-03-31 01:11:12 +00:00
|
|
|
|
|
|
|
XRRSetMonitor (manager_xrandr->xdisplay,
|
|
|
|
DefaultRootWindow (manager_xrandr->xdisplay),
|
2016-12-13 10:58:22 +00:00
|
|
|
xrandr_monitor_info);
|
|
|
|
XRRFreeMonitors (xrandr_monitor_info);
|
2015-03-31 01:11:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2016-12-13 10:58:22 +00:00
|
|
|
meta_monitor_manager_xrandr_tiled_monitor_removed (MetaMonitorManager *manager,
|
|
|
|
MetaMonitor *monitor)
|
2015-03-31 01:11:12 +00:00
|
|
|
{
|
|
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
2016-12-13 10:58:22 +00:00
|
|
|
MetaMonitorXrandrData *monitor_xrandr_data;
|
|
|
|
Atom monitor_name;
|
|
|
|
|
|
|
|
int monitor_count;
|
2015-03-31 01:11:12 +00:00
|
|
|
|
|
|
|
if (manager_xrandr->has_randr15 == FALSE)
|
|
|
|
return;
|
2016-12-13 10:58:22 +00:00
|
|
|
|
|
|
|
monitor_xrandr_data = meta_monitor_xrandr_data_from_monitor (monitor);
|
|
|
|
monitor_name = monitor_xrandr_data->xrandr_name;
|
|
|
|
monitor_count =
|
|
|
|
meta_monitor_manager_xrandr_decrease_monitor_count (manager_xrandr,
|
|
|
|
monitor_name);
|
|
|
|
|
|
|
|
if (monitor_count == 0)
|
|
|
|
XRRDeleteMonitor (manager_xrandr->xdisplay,
|
|
|
|
DefaultRootWindow (manager_xrandr->xdisplay),
|
|
|
|
monitor_name);
|
2015-03-31 01:11:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2016-11-25 06:31:38 +00:00
|
|
|
meta_monitor_manager_xrandr_init_monitors (MetaMonitorManagerXrandr *manager_xrandr)
|
2015-03-31 01:11:12 +00:00
|
|
|
{
|
|
|
|
XRRMonitorInfo *m;
|
|
|
|
int n, i;
|
|
|
|
|
|
|
|
if (manager_xrandr->has_randr15 == FALSE)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* delete any tiled monitors setup, as mutter will want to recreate
|
|
|
|
things in its image */
|
|
|
|
m = XRRGetMonitors (manager_xrandr->xdisplay,
|
|
|
|
DefaultRootWindow (manager_xrandr->xdisplay),
|
|
|
|
FALSE, &n);
|
|
|
|
if (n == -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
{
|
|
|
|
if (m[i].noutput > 1)
|
|
|
|
XRRDeleteMonitor (manager_xrandr->xdisplay,
|
|
|
|
DefaultRootWindow (manager_xrandr->xdisplay),
|
|
|
|
m[i].name);
|
|
|
|
}
|
|
|
|
XRRFreeMonitors (m);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-03-08 08:05:00 +00:00
|
|
|
static gboolean
|
|
|
|
meta_monitor_manager_xrandr_is_transform_handled (MetaMonitorManager *manager,
|
|
|
|
MetaCrtc *crtc,
|
|
|
|
MetaMonitorTransform transform)
|
|
|
|
{
|
2017-03-24 08:12:51 +00:00
|
|
|
g_warn_if_fail ((crtc->all_transforms & transform) == transform);
|
2017-03-08 08:05:00 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2017-05-25 08:12:51 +00:00
|
|
|
static float
|
2017-01-20 07:07:12 +00:00
|
|
|
meta_monitor_manager_xrandr_calculate_monitor_mode_scale (MetaMonitorManager *manager,
|
|
|
|
MetaMonitor *monitor,
|
|
|
|
MetaMonitorMode *monitor_mode)
|
|
|
|
{
|
2017-04-21 10:01:15 +00:00
|
|
|
return meta_monitor_calculate_mode_scale (monitor, monitor_mode);
|
2017-01-20 07:07:12 +00:00
|
|
|
}
|
|
|
|
|
2017-06-05 07:59:47 +00:00
|
|
|
static float *
|
|
|
|
meta_monitor_manager_xrandr_calculate_supported_scales (MetaMonitorManager *manager,
|
|
|
|
MetaLogicalMonitorLayoutMode layout_mode,
|
|
|
|
MetaMonitor *monitor,
|
|
|
|
MetaMonitorMode *monitor_mode,
|
|
|
|
int *n_supported_scales)
|
2017-01-26 09:14:32 +00:00
|
|
|
{
|
2017-06-05 07:59:47 +00:00
|
|
|
MetaMonitorScalesConstraint constraints;
|
|
|
|
|
|
|
|
constraints = META_MONITOR_SCALES_CONSTRAINT_NO_FRAC;
|
|
|
|
return meta_monitor_calculate_supported_scales (monitor, monitor_mode,
|
|
|
|
constraints,
|
|
|
|
n_supported_scales);
|
2017-01-26 09:14:32 +00:00
|
|
|
}
|
|
|
|
|
2017-02-07 08:01:40 +00:00
|
|
|
static MetaMonitorManagerCapability
|
|
|
|
meta_monitor_manager_xrandr_get_capabilities (MetaMonitorManager *manager)
|
|
|
|
{
|
2017-04-21 10:01:15 +00:00
|
|
|
return (META_MONITOR_MANAGER_CAPABILITY_MIRRORING |
|
|
|
|
META_MONITOR_MANAGER_CAPABILITY_GLOBAL_SCALE_REQUIRED);
|
2017-02-07 08:01:40 +00:00
|
|
|
}
|
|
|
|
|
2017-02-15 08:06:46 +00:00
|
|
|
static gboolean
|
|
|
|
meta_monitor_manager_xrandr_get_max_screen_size (MetaMonitorManager *manager,
|
|
|
|
int *max_width,
|
|
|
|
int *max_height)
|
|
|
|
{
|
|
|
|
MetaMonitorManagerXrandr *manager_xrandr =
|
|
|
|
META_MONITOR_MANAGER_XRANDR (manager);
|
|
|
|
|
|
|
|
*max_width = manager_xrandr->max_screen_width;
|
|
|
|
*max_height = manager_xrandr->max_screen_height;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2017-02-24 10:10:52 +00:00
|
|
|
static MetaLogicalMonitorLayoutMode
|
|
|
|
meta_monitor_manager_xrandr_get_default_layout_mode (MetaMonitorManager *manager)
|
|
|
|
{
|
2017-04-21 10:01:15 +00:00
|
|
|
return META_LOGICAL_MONITOR_LAYOUT_MODE_PHYSICAL;
|
2017-02-24 10:10:52 +00:00
|
|
|
}
|
|
|
|
|
2014-04-23 16:07:38 +00:00
|
|
|
static void
|
|
|
|
meta_monitor_manager_xrandr_init (MetaMonitorManagerXrandr *manager_xrandr)
|
2013-07-29 11:00:15 +00:00
|
|
|
{
|
2014-04-23 16:07:38 +00:00
|
|
|
MetaBackendX11 *backend = META_BACKEND_X11 (meta_get_backend ());
|
|
|
|
|
|
|
|
manager_xrandr->xdisplay = meta_backend_x11_get_xdisplay (backend);
|
|
|
|
|
|
|
|
if (!XRRQueryExtension (manager_xrandr->xdisplay,
|
|
|
|
&manager_xrandr->rr_event_base,
|
|
|
|
&manager_xrandr->rr_error_base))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-03-31 00:38:44 +00:00
|
|
|
int major_version, minor_version;
|
2014-04-23 16:07:38 +00:00
|
|
|
/* We only use ScreenChangeNotify, but GDK uses the others,
|
|
|
|
and we don't want to step on its toes */
|
|
|
|
XRRSelectInput (manager_xrandr->xdisplay,
|
|
|
|
DefaultRootWindow (manager_xrandr->xdisplay),
|
|
|
|
RRScreenChangeNotifyMask
|
|
|
|
| RRCrtcChangeNotifyMask
|
|
|
|
| RROutputPropertyNotifyMask);
|
2015-03-31 00:38:44 +00:00
|
|
|
|
|
|
|
manager_xrandr->has_randr15 = FALSE;
|
|
|
|
XRRQueryVersion (manager_xrandr->xdisplay, &major_version,
|
|
|
|
&minor_version);
|
|
|
|
#ifdef HAVE_XRANDR15
|
|
|
|
if (major_version > 1 ||
|
|
|
|
(major_version == 1 &&
|
|
|
|
minor_version >= 5))
|
2016-12-13 10:58:22 +00:00
|
|
|
{
|
|
|
|
manager_xrandr->has_randr15 = TRUE;
|
|
|
|
manager_xrandr->tiled_monitor_atoms = g_hash_table_new (NULL, NULL);
|
|
|
|
}
|
2015-03-31 01:11:12 +00:00
|
|
|
meta_monitor_manager_xrandr_init_monitors (manager_xrandr);
|
2015-03-31 00:38:44 +00:00
|
|
|
#endif
|
2014-04-23 16:07:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_monitor_manager_xrandr_finalize (GObject *object)
|
|
|
|
{
|
|
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (object);
|
|
|
|
|
|
|
|
if (manager_xrandr->resources)
|
|
|
|
XRRFreeScreenResources (manager_xrandr->resources);
|
|
|
|
manager_xrandr->resources = NULL;
|
|
|
|
|
2016-12-13 10:58:22 +00:00
|
|
|
g_hash_table_destroy (manager_xrandr->tiled_monitor_atoms);
|
|
|
|
|
2014-04-23 16:07:38 +00:00
|
|
|
G_OBJECT_CLASS (meta_monitor_manager_xrandr_parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_monitor_manager_xrandr_class_init (MetaMonitorManagerXrandrClass *klass)
|
|
|
|
{
|
|
|
|
MetaMonitorManagerClass *manager_class = META_MONITOR_MANAGER_CLASS (klass);
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
object_class->finalize = meta_monitor_manager_xrandr_finalize;
|
|
|
|
|
|
|
|
manager_class->read_current = meta_monitor_manager_xrandr_read_current;
|
|
|
|
manager_class->read_edid = meta_monitor_manager_xrandr_read_edid;
|
2017-01-06 05:27:21 +00:00
|
|
|
manager_class->ensure_initial_config = meta_monitor_manager_xrandr_ensure_initial_config;
|
2017-01-09 06:31:18 +00:00
|
|
|
manager_class->apply_monitors_config = meta_monitor_manager_xrandr_apply_monitors_config;
|
2014-04-23 16:07:38 +00:00
|
|
|
manager_class->set_power_save_mode = meta_monitor_manager_xrandr_set_power_save_mode;
|
|
|
|
manager_class->change_backlight = meta_monitor_manager_xrandr_change_backlight;
|
|
|
|
manager_class->get_crtc_gamma = meta_monitor_manager_xrandr_get_crtc_gamma;
|
|
|
|
manager_class->set_crtc_gamma = meta_monitor_manager_xrandr_set_crtc_gamma;
|
2015-03-31 01:11:12 +00:00
|
|
|
#ifdef HAVE_XRANDR15
|
2016-12-13 10:58:22 +00:00
|
|
|
manager_class->tiled_monitor_added = meta_monitor_manager_xrandr_tiled_monitor_added;
|
|
|
|
manager_class->tiled_monitor_removed = meta_monitor_manager_xrandr_tiled_monitor_removed;
|
2015-03-31 01:11:12 +00:00
|
|
|
#endif
|
2017-03-08 08:05:00 +00:00
|
|
|
manager_class->is_transform_handled = meta_monitor_manager_xrandr_is_transform_handled;
|
2017-01-20 07:07:12 +00:00
|
|
|
manager_class->calculate_monitor_mode_scale = meta_monitor_manager_xrandr_calculate_monitor_mode_scale;
|
2017-06-05 07:59:47 +00:00
|
|
|
manager_class->calculate_supported_scales = meta_monitor_manager_xrandr_calculate_supported_scales;
|
2017-02-07 08:01:40 +00:00
|
|
|
manager_class->get_capabilities = meta_monitor_manager_xrandr_get_capabilities;
|
2017-02-15 08:06:46 +00:00
|
|
|
manager_class->get_max_screen_size = meta_monitor_manager_xrandr_get_max_screen_size;
|
2017-02-24 10:10:52 +00:00
|
|
|
manager_class->get_default_layout_mode = meta_monitor_manager_xrandr_get_default_layout_mode;
|
2016-12-13 10:58:22 +00:00
|
|
|
|
|
|
|
quark_meta_monitor_xrandr_data =
|
|
|
|
g_quark_from_static_string ("-meta-monitor-xrandr-data");
|
2014-04-23 16:07:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_monitor_manager_xrandr_handle_xevent (MetaMonitorManagerXrandr *manager_xrandr,
|
|
|
|
XEvent *event)
|
|
|
|
{
|
|
|
|
MetaMonitorManager *manager = META_MONITOR_MANAGER (manager_xrandr);
|
2017-04-21 10:01:15 +00:00
|
|
|
gboolean is_hotplug;
|
|
|
|
gboolean is_our_configuration;
|
2013-07-29 11:00:15 +00:00
|
|
|
|
|
|
|
if ((event->type - manager_xrandr->rr_event_base) != RRScreenChangeNotify)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
XRRUpdateConfiguration (event);
|
2013-09-02 13:04:05 +00:00
|
|
|
|
2016-12-19 07:46:31 +00:00
|
|
|
meta_monitor_manager_read_current_state (manager);
|
2013-09-02 13:04:05 +00:00
|
|
|
|
2017-04-21 10:01:15 +00:00
|
|
|
|
|
|
|
is_hotplug = (manager_xrandr->resources->timestamp <
|
|
|
|
manager_xrandr->resources->configTimestamp);
|
|
|
|
is_our_configuration = (manager_xrandr->resources->timestamp ==
|
|
|
|
manager_xrandr->last_xrandr_set_timestamp);
|
|
|
|
if (is_hotplug)
|
2014-10-16 18:31:03 +00:00
|
|
|
{
|
|
|
|
meta_monitor_manager_on_hotplug (manager);
|
|
|
|
}
|
|
|
|
else
|
2013-10-31 13:37:44 +00:00
|
|
|
{
|
2017-09-11 06:10:26 +00:00
|
|
|
MetaMonitorsConfig *config;
|
2017-04-21 10:01:15 +00:00
|
|
|
|
|
|
|
if (is_our_configuration)
|
2017-09-11 06:10:26 +00:00
|
|
|
{
|
|
|
|
MetaMonitorConfigManager *config_manager =
|
|
|
|
meta_monitor_manager_get_config_manager (manager);
|
|
|
|
|
|
|
|
config = meta_monitor_config_manager_get_current (config_manager);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
config = NULL;
|
|
|
|
}
|
2017-04-21 10:01:15 +00:00
|
|
|
|
2017-09-11 06:10:26 +00:00
|
|
|
meta_monitor_manager_rebuild_derived (manager, config);
|
2013-09-02 13:04:05 +00:00
|
|
|
}
|
2014-10-03 21:10:53 +00:00
|
|
|
|
2013-07-29 11:00:15 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|