842ff5d6aa
In preparation for further refactorizations, rename the MetaMonitorInfo struct to MetaLogicalMonitor. Eventually, part of MetaLogicalMonitor will be split into a MetaMonitor type. https://bugzilla.gnome.org/show_bug.cgi?id=777732
1517 lines
48 KiB
C
1517 lines
48 KiB
C
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
/*
|
|
* 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.
|
|
*
|
|
* 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.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "meta-monitor-manager-xrandr.h"
|
|
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <math.h>
|
|
#include <clutter/clutter.h>
|
|
|
|
#include <X11/Xatom.h>
|
|
#include <X11/extensions/Xrandr.h>
|
|
#include <X11/extensions/dpms.h>
|
|
#include <X11/Xlib-xcb.h>
|
|
#include <xcb/randr.h>
|
|
|
|
#include "meta-backend-x11.h"
|
|
#include <meta/main.h>
|
|
#include <meta/errors.h>
|
|
#include "meta-monitor-config.h"
|
|
|
|
#define ALL_TRANSFORMS ((1 << (META_MONITOR_TRANSFORM_FLIPPED_270 + 1)) - 1)
|
|
|
|
/* 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
|
|
|
|
struct _MetaMonitorManagerXrandr
|
|
{
|
|
MetaMonitorManager parent_instance;
|
|
|
|
Display *xdisplay;
|
|
XRRScreenResources *resources;
|
|
int rr_event_base;
|
|
int rr_error_base;
|
|
gboolean has_randr15;
|
|
};
|
|
|
|
struct _MetaMonitorManagerXrandrClass
|
|
{
|
|
MetaMonitorManagerClass parent_class;
|
|
};
|
|
|
|
G_DEFINE_TYPE (MetaMonitorManagerXrandr, meta_monitor_manager_xrandr, META_TYPE_MONITOR_MANAGER);
|
|
|
|
static MetaMonitorTransform
|
|
meta_monitor_transform_from_xrandr (Rotation rotation)
|
|
{
|
|
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
|
|
};
|
|
MetaMonitorTransform ret;
|
|
|
|
switch (rotation & 0x7F)
|
|
{
|
|
default:
|
|
case RR_Rotate_0:
|
|
ret = META_MONITOR_TRANSFORM_NORMAL;
|
|
break;
|
|
case RR_Rotate_90:
|
|
ret = META_MONITOR_TRANSFORM_90;
|
|
break;
|
|
case RR_Rotate_180:
|
|
ret = META_MONITOR_TRANSFORM_180;
|
|
break;
|
|
case RR_Rotate_270:
|
|
ret = META_MONITOR_TRANSFORM_270;
|
|
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)
|
|
|
|
static MetaMonitorTransform
|
|
meta_monitor_transform_from_xrandr_all (Rotation rotation)
|
|
{
|
|
unsigned ret;
|
|
|
|
/* Handle the common cases first (none or all) */
|
|
if (rotation == 0 || rotation == RR_Rotate_0)
|
|
return (1 << META_MONITOR_TRANSFORM_NORMAL);
|
|
|
|
/* All rotations and one reflection -> all of them by composition */
|
|
if ((rotation & ALL_ROTATIONS) &&
|
|
((rotation & RR_Reflect_X) || (rotation & RR_Reflect_Y)))
|
|
return ALL_TRANSFORMS;
|
|
|
|
ret = 1 << META_MONITOR_TRANSFORM_NORMAL;
|
|
if (rotation & RR_Rotate_90)
|
|
ret |= 1 << META_MONITOR_TRANSFORM_90;
|
|
if (rotation & RR_Rotate_180)
|
|
ret |= 1 << META_MONITOR_TRANSFORM_180;
|
|
if (rotation & RR_Rotate_270)
|
|
ret |= 1 << META_MONITOR_TRANSFORM_270;
|
|
if (rotation & (RR_Rotate_0 | RR_Reflect_X))
|
|
ret |= 1 << META_MONITOR_TRANSFORM_FLIPPED;
|
|
if (rotation & (RR_Rotate_90 | RR_Reflect_X))
|
|
ret |= 1 << META_MONITOR_TRANSFORM_FLIPPED_90;
|
|
if (rotation & (RR_Rotate_180 | RR_Reflect_X))
|
|
ret |= 1 << META_MONITOR_TRANSFORM_FLIPPED_180;
|
|
if (rotation & (RR_Rotate_270 | RR_Reflect_X))
|
|
ret |= 1 << META_MONITOR_TRANSFORM_FLIPPED_270;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static gboolean
|
|
output_get_integer_property (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output, const char *propname,
|
|
gint *value)
|
|
{
|
|
gboolean exists = FALSE;
|
|
Atom atom, actual_type;
|
|
int actual_format;
|
|
unsigned long nitems, bytes_after;
|
|
unsigned char *buffer;
|
|
|
|
atom = XInternAtom (manager_xrandr->xdisplay, propname, False);
|
|
XRRGetOutputProperty (manager_xrandr->xdisplay,
|
|
(XID)output->winsys_id,
|
|
atom,
|
|
0, G_MAXLONG, False, False, XA_INTEGER,
|
|
&actual_type, &actual_format,
|
|
&nitems, &bytes_after, &buffer);
|
|
|
|
exists = (actual_type == XA_INTEGER && actual_format == 32 && nitems == 1);
|
|
|
|
if (exists && value != NULL)
|
|
*value = ((int*)buffer)[0];
|
|
|
|
XFree (buffer);
|
|
return exists;
|
|
}
|
|
|
|
static gboolean
|
|
output_get_property_exists (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output, const char *propname)
|
|
{
|
|
gboolean exists = FALSE;
|
|
Atom atom, actual_type;
|
|
int actual_format;
|
|
unsigned long nitems, bytes_after;
|
|
unsigned char *buffer;
|
|
|
|
atom = XInternAtom (manager_xrandr->xdisplay, propname, False);
|
|
XRRGetOutputProperty (manager_xrandr->xdisplay,
|
|
(XID)output->winsys_id,
|
|
atom,
|
|
0, G_MAXLONG, False, False, AnyPropertyType,
|
|
&actual_type, &actual_format,
|
|
&nitems, &bytes_after, &buffer);
|
|
|
|
exists = (actual_type != None);
|
|
|
|
XFree (buffer);
|
|
return exists;
|
|
}
|
|
|
|
static gboolean
|
|
output_get_boolean_property (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output, const char *propname)
|
|
{
|
|
Atom atom, actual_type;
|
|
int actual_format;
|
|
unsigned long nitems, bytes_after;
|
|
g_autofree unsigned char *buffer = NULL;
|
|
|
|
atom = XInternAtom (manager_xrandr->xdisplay, propname, False);
|
|
XRRGetOutputProperty (manager_xrandr->xdisplay,
|
|
(XID)output->winsys_id,
|
|
atom,
|
|
0, G_MAXLONG, False, False, XA_CARDINAL,
|
|
&actual_type, &actual_format,
|
|
&nitems, &bytes_after, &buffer);
|
|
|
|
if (actual_type != XA_CARDINAL || actual_format != 32 || nitems < 1)
|
|
return FALSE;
|
|
|
|
return ((int*)buffer)[0];
|
|
}
|
|
|
|
static gboolean
|
|
output_get_presentation_xrandr (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output)
|
|
{
|
|
return output_get_boolean_property (manager_xrandr, output, "_MUTTER_PRESENTATION_OUTPUT");
|
|
}
|
|
|
|
static gboolean
|
|
output_get_underscanning_xrandr (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output)
|
|
{
|
|
Atom atom, actual_type;
|
|
int actual_format;
|
|
unsigned long nitems, bytes_after;
|
|
g_autofree unsigned char *buffer = NULL;
|
|
g_autofree char *str = NULL;
|
|
|
|
atom = XInternAtom (manager_xrandr->xdisplay, "underscan", False);
|
|
XRRGetOutputProperty (manager_xrandr->xdisplay,
|
|
(XID)output->winsys_id,
|
|
atom,
|
|
0, G_MAXLONG, False, False, XA_ATOM,
|
|
&actual_type, &actual_format,
|
|
&nitems, &bytes_after, &buffer);
|
|
|
|
if (actual_type != XA_ATOM || actual_format != 32 || nitems < 1)
|
|
return FALSE;
|
|
|
|
str = XGetAtomName (manager_xrandr->xdisplay, *(Atom *)buffer);
|
|
return (strcmp (str, "on") == 0);
|
|
}
|
|
|
|
static gboolean
|
|
output_get_supports_underscanning_xrandr (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output)
|
|
{
|
|
Atom atom, actual_type;
|
|
int actual_format, i;
|
|
unsigned long nitems, bytes_after;
|
|
g_autofree unsigned char *buffer = NULL;
|
|
XRRPropertyInfo *property_info;
|
|
Atom *values;
|
|
gboolean supports_underscanning = FALSE;
|
|
|
|
atom = XInternAtom (manager_xrandr->xdisplay, "underscan", False);
|
|
XRRGetOutputProperty (manager_xrandr->xdisplay,
|
|
(XID)output->winsys_id,
|
|
atom,
|
|
0, G_MAXLONG, False, False, XA_ATOM,
|
|
&actual_type, &actual_format,
|
|
&nitems, &bytes_after, &buffer);
|
|
|
|
if (actual_type != XA_ATOM || actual_format != 32 || nitems < 1)
|
|
return FALSE;
|
|
|
|
property_info = XRRQueryOutputProperty (manager_xrandr->xdisplay,
|
|
(XID) output->winsys_id,
|
|
atom);
|
|
values = (Atom *) property_info->values;
|
|
|
|
for (i = 0; i < property_info->num_values; i++)
|
|
{
|
|
/* The output supports underscanning if "on" is a valid value
|
|
* for the underscan property.
|
|
*/
|
|
char *name = XGetAtomName (manager_xrandr->xdisplay, values[i]);
|
|
if (strcmp (name, "on") == 0)
|
|
supports_underscanning = TRUE;
|
|
|
|
XFree (name);
|
|
}
|
|
|
|
XFree (property_info);
|
|
|
|
return supports_underscanning;
|
|
}
|
|
|
|
static int
|
|
normalize_backlight (MetaOutput *output,
|
|
int hw_value)
|
|
{
|
|
return round ((double)(hw_value - output->backlight_min) /
|
|
(output->backlight_max - output->backlight_min) * 100.0);
|
|
}
|
|
|
|
static int
|
|
output_get_backlight_xrandr (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output)
|
|
{
|
|
int value = -1;
|
|
Atom atom, actual_type;
|
|
int actual_format;
|
|
unsigned long nitems, bytes_after;
|
|
g_autofree unsigned char *buffer = NULL;
|
|
|
|
atom = XInternAtom (manager_xrandr->xdisplay, "Backlight", False);
|
|
XRRGetOutputProperty (manager_xrandr->xdisplay,
|
|
(XID)output->winsys_id,
|
|
atom,
|
|
0, G_MAXLONG, False, False, XA_INTEGER,
|
|
&actual_type, &actual_format,
|
|
&nitems, &bytes_after, &buffer);
|
|
|
|
if (actual_type != XA_INTEGER || actual_format != 32 || nitems < 1)
|
|
return FALSE;
|
|
|
|
value = ((int*)buffer)[0];
|
|
if (value > 0)
|
|
return normalize_backlight (output, value);
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
static void
|
|
output_get_backlight_limits_xrandr (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output)
|
|
{
|
|
Atom atom;
|
|
xcb_connection_t *xcb_conn;
|
|
g_autofree xcb_randr_query_output_property_reply_t *reply;
|
|
|
|
atom = XInternAtom (manager_xrandr->xdisplay, "Backlight", False);
|
|
|
|
xcb_conn = XGetXCBConnection (manager_xrandr->xdisplay);
|
|
reply = xcb_randr_query_output_property_reply (xcb_conn,
|
|
xcb_randr_query_output_property (xcb_conn,
|
|
(xcb_randr_output_t) output->winsys_id,
|
|
(xcb_atom_t) atom),
|
|
NULL);
|
|
|
|
/* This can happen on systems without backlights. */
|
|
if (reply == NULL)
|
|
return;
|
|
|
|
if (!reply->range || reply->length != 2)
|
|
{
|
|
meta_verbose ("backlight %s was not range\n", output->name);
|
|
return;
|
|
}
|
|
|
|
int32_t *values = xcb_randr_query_output_property_valid_values (reply);
|
|
output->backlight_min = values[0];
|
|
output->backlight_max = values[1];
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
static guint8 *
|
|
get_edid_property (Display *dpy,
|
|
RROutput output,
|
|
Atom atom,
|
|
gsize *len)
|
|
{
|
|
unsigned char *prop;
|
|
int actual_format;
|
|
unsigned long nitems, bytes_after;
|
|
Atom actual_type;
|
|
guint8 *result;
|
|
|
|
XRRGetOutputProperty (dpy, output, atom,
|
|
0, 100, False, False,
|
|
AnyPropertyType,
|
|
&actual_type, &actual_format,
|
|
&nitems, &bytes_after, &prop);
|
|
|
|
if (actual_type == XA_INTEGER && actual_format == 8)
|
|
{
|
|
result = g_memdup (prop, nitems);
|
|
if (len)
|
|
*len = nitems;
|
|
}
|
|
else
|
|
{
|
|
result = NULL;
|
|
}
|
|
|
|
XFree (prop);
|
|
|
|
return result;
|
|
}
|
|
|
|
static GBytes *
|
|
read_output_edid (MetaMonitorManagerXrandr *manager_xrandr,
|
|
XID winsys_id)
|
|
{
|
|
Atom edid_atom;
|
|
guint8 *result;
|
|
gsize len;
|
|
|
|
edid_atom = XInternAtom (manager_xrandr->xdisplay, "EDID", FALSE);
|
|
result = get_edid_property (manager_xrandr->xdisplay, winsys_id, edid_atom, &len);
|
|
|
|
if (!result)
|
|
{
|
|
edid_atom = XInternAtom (manager_xrandr->xdisplay, "EDID_DATA", FALSE);
|
|
result = get_edid_property (manager_xrandr->xdisplay, winsys_id, edid_atom, &len);
|
|
}
|
|
|
|
if (result)
|
|
{
|
|
if (len > 0 && len % 128 == 0)
|
|
return g_bytes_new_take (result, len);
|
|
else
|
|
g_free (result);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
output_get_tile_info (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output)
|
|
{
|
|
Atom tile_atom;
|
|
unsigned char *prop;
|
|
unsigned long nitems, bytes_after;
|
|
int actual_format;
|
|
Atom actual_type;
|
|
|
|
if (manager_xrandr->has_randr15 == FALSE)
|
|
return;
|
|
|
|
tile_atom = XInternAtom (manager_xrandr->xdisplay, "TILE", FALSE);
|
|
XRRGetOutputProperty (manager_xrandr->xdisplay,
|
|
output->winsys_id,
|
|
tile_atom, 0, 100, False,
|
|
False, AnyPropertyType,
|
|
&actual_type, &actual_format,
|
|
&nitems, &bytes_after, &prop);
|
|
|
|
if (actual_type == XA_INTEGER && actual_format == 32 && nitems == 8)
|
|
{
|
|
long *values = (long *)prop;
|
|
output->tile_info.group_id = values[0];
|
|
output->tile_info.flags = values[1];
|
|
output->tile_info.max_h_tiles = values[2];
|
|
output->tile_info.max_v_tiles = values[3];
|
|
output->tile_info.loc_h_tile = values[4];
|
|
output->tile_info.loc_v_tile = values[5];
|
|
output->tile_info.tile_w = values[6];
|
|
output->tile_info.tile_h = values[7];
|
|
}
|
|
XFree (prop);
|
|
}
|
|
|
|
static gboolean
|
|
output_get_hotplug_mode_update (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output)
|
|
{
|
|
return output_get_property_exists (manager_xrandr, output, "hotplug_mode_update");
|
|
}
|
|
|
|
static gint
|
|
output_get_suggested_x (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output)
|
|
{
|
|
gint val;
|
|
if (output_get_integer_property (manager_xrandr, output, "suggested X", &val))
|
|
return val;
|
|
|
|
return -1;
|
|
}
|
|
|
|
static gint
|
|
output_get_suggested_y (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output)
|
|
{
|
|
gint val;
|
|
if (output_get_integer_property (manager_xrandr, output, "suggested Y", &val))
|
|
return val;
|
|
|
|
return -1;
|
|
}
|
|
|
|
static MetaConnectorType
|
|
connector_type_from_atom (MetaMonitorManagerXrandr *manager_xrandr,
|
|
Atom atom)
|
|
{
|
|
Display *xdpy = manager_xrandr->xdisplay;
|
|
|
|
if (atom == XInternAtom (xdpy, "HDMI", True))
|
|
return META_CONNECTOR_TYPE_HDMIA;
|
|
if (atom == XInternAtom (xdpy, "VGA", True))
|
|
return META_CONNECTOR_TYPE_VGA;
|
|
/* Doesn't have a DRM equivalent, but means an internal panel.
|
|
* We could pick either LVDS or eDP here. */
|
|
if (atom == XInternAtom (xdpy, "Panel", True))
|
|
return META_CONNECTOR_TYPE_LVDS;
|
|
if (atom == XInternAtom (xdpy, "DVI", True) || atom == XInternAtom (xdpy, "DVI-I", True))
|
|
return META_CONNECTOR_TYPE_DVII;
|
|
if (atom == XInternAtom (xdpy, "DVI-A", True))
|
|
return META_CONNECTOR_TYPE_DVIA;
|
|
if (atom == XInternAtom (xdpy, "DVI-D", True))
|
|
return META_CONNECTOR_TYPE_DVID;
|
|
if (atom == XInternAtom (xdpy, "DisplayPort", True))
|
|
return META_CONNECTOR_TYPE_DisplayPort;
|
|
|
|
if (atom == XInternAtom (xdpy, "TV", True))
|
|
return META_CONNECTOR_TYPE_TV;
|
|
if (atom == XInternAtom (xdpy, "TV-Composite", True))
|
|
return META_CONNECTOR_TYPE_Composite;
|
|
if (atom == XInternAtom (xdpy, "TV-SVideo", True))
|
|
return META_CONNECTOR_TYPE_SVIDEO;
|
|
/* Another set of mismatches. */
|
|
if (atom == XInternAtom (xdpy, "TV-SCART", True))
|
|
return META_CONNECTOR_TYPE_TV;
|
|
if (atom == XInternAtom (xdpy, "TV-C4", True))
|
|
return META_CONNECTOR_TYPE_TV;
|
|
|
|
return META_CONNECTOR_TYPE_Unknown;
|
|
}
|
|
|
|
static MetaConnectorType
|
|
output_get_connector_type_from_prop (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output)
|
|
{
|
|
Atom atom, actual_type, connector_type_atom;
|
|
int actual_format;
|
|
unsigned long nitems, bytes_after;
|
|
g_autofree unsigned char *buffer = NULL;
|
|
|
|
atom = XInternAtom (manager_xrandr->xdisplay, "ConnectorType", False);
|
|
XRRGetOutputProperty (manager_xrandr->xdisplay,
|
|
(XID)output->winsys_id,
|
|
atom,
|
|
0, G_MAXLONG, False, False, XA_ATOM,
|
|
&actual_type, &actual_format,
|
|
&nitems, &bytes_after, &buffer);
|
|
|
|
if (actual_type != XA_ATOM || actual_format != 32 || nitems < 1)
|
|
return META_CONNECTOR_TYPE_Unknown;
|
|
|
|
connector_type_atom = ((Atom *) buffer)[0];
|
|
return connector_type_from_atom (manager_xrandr, connector_type_atom);
|
|
}
|
|
|
|
static MetaConnectorType
|
|
output_get_connector_type_from_name (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output)
|
|
{
|
|
const char *name = output->name;
|
|
|
|
/* drmmode_display.c, which was copy/pasted across all the FOSS
|
|
* xf86-video-* drivers, seems to name its outputs based on the
|
|
* connector type, so look for that....
|
|
*
|
|
* SNA has its own naming scheme, because what else did you expect
|
|
* from SNA, but it's not too different, so we can thankfully use
|
|
* that with minor changes.
|
|
*
|
|
* http://cgit.freedesktop.org/xorg/xserver/tree/hw/xfree86/drivers/modesetting/drmmode_display.c#n953
|
|
* http://cgit.freedesktop.org/xorg/driver/xf86-video-intel/tree/src/sna/sna_display.c#n3486
|
|
*/
|
|
|
|
if (g_str_has_prefix (name, "DVI"))
|
|
return META_CONNECTOR_TYPE_DVII;
|
|
if (g_str_has_prefix (name, "LVDS"))
|
|
return META_CONNECTOR_TYPE_LVDS;
|
|
if (g_str_has_prefix (name, "HDMI"))
|
|
return META_CONNECTOR_TYPE_HDMIA;
|
|
if (g_str_has_prefix (name, "VGA"))
|
|
return META_CONNECTOR_TYPE_VGA;
|
|
/* SNA uses DP, not DisplayPort. Test for both. */
|
|
if (g_str_has_prefix (name, "DP") || g_str_has_prefix (name, "DisplayPort"))
|
|
return META_CONNECTOR_TYPE_DisplayPort;
|
|
if (g_str_has_prefix (name, "eDP"))
|
|
return META_CONNECTOR_TYPE_eDP;
|
|
if (g_str_has_prefix (name, "Virtual"))
|
|
return META_CONNECTOR_TYPE_VIRTUAL;
|
|
if (g_str_has_prefix (name, "Composite"))
|
|
return META_CONNECTOR_TYPE_Composite;
|
|
if (g_str_has_prefix (name, "S-video"))
|
|
return META_CONNECTOR_TYPE_SVIDEO;
|
|
if (g_str_has_prefix (name, "TV"))
|
|
return META_CONNECTOR_TYPE_TV;
|
|
if (g_str_has_prefix (name, "CTV"))
|
|
return META_CONNECTOR_TYPE_Composite;
|
|
if (g_str_has_prefix (name, "DSI"))
|
|
return META_CONNECTOR_TYPE_DSI;
|
|
if (g_str_has_prefix (name, "DIN"))
|
|
return META_CONNECTOR_TYPE_9PinDIN;
|
|
|
|
return META_CONNECTOR_TYPE_Unknown;
|
|
}
|
|
|
|
static MetaConnectorType
|
|
output_get_connector_type (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output)
|
|
{
|
|
MetaConnectorType ret;
|
|
|
|
/* The "ConnectorType" property is considered mandatory since RandR 1.3,
|
|
* but none of the FOSS drivers support it, because we're a bunch of
|
|
* professional software developers.
|
|
*
|
|
* Try poking it first, without any expectations that it will work.
|
|
* If it's not there, we thankfully have other bonghits to try next.
|
|
*/
|
|
ret = output_get_connector_type_from_prop (manager_xrandr, output);
|
|
if (ret != META_CONNECTOR_TYPE_Unknown)
|
|
return ret;
|
|
|
|
/* Fall back to heuristics based on the output name. */
|
|
ret = output_get_connector_type_from_name (manager_xrandr, output);
|
|
if (ret != META_CONNECTOR_TYPE_Unknown)
|
|
return ret;
|
|
|
|
return META_CONNECTOR_TYPE_Unknown;
|
|
}
|
|
|
|
static void
|
|
output_get_modes (MetaMonitorManager *manager,
|
|
MetaOutput *meta_output,
|
|
XRROutputInfo *output)
|
|
{
|
|
guint j, k;
|
|
guint n_actual_modes;
|
|
|
|
meta_output->modes = g_new0 (MetaMonitorMode *, output->nmode);
|
|
|
|
n_actual_modes = 0;
|
|
for (j = 0; j < (guint)output->nmode; j++)
|
|
{
|
|
for (k = 0; k < manager->n_modes; k++)
|
|
{
|
|
if (output->modes[j] == (XID)manager->modes[k].mode_id)
|
|
{
|
|
meta_output->modes[n_actual_modes] = &manager->modes[k];
|
|
n_actual_modes += 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
meta_output->n_modes = n_actual_modes;
|
|
if (n_actual_modes > 0)
|
|
meta_output->preferred_mode = meta_output->modes[0];
|
|
}
|
|
|
|
static void
|
|
output_get_crtcs (MetaMonitorManager *manager,
|
|
MetaOutput *meta_output,
|
|
XRROutputInfo *output)
|
|
{
|
|
guint j, k;
|
|
guint n_actual_crtcs;
|
|
|
|
meta_output->possible_crtcs = g_new0 (MetaCRTC *, output->ncrtc);
|
|
|
|
n_actual_crtcs = 0;
|
|
for (j = 0; j < (unsigned)output->ncrtc; j++)
|
|
{
|
|
for (k = 0; k < manager->n_crtcs; k++)
|
|
{
|
|
if ((XID)manager->crtcs[k].crtc_id == output->crtcs[j])
|
|
{
|
|
meta_output->possible_crtcs[n_actual_crtcs] = &manager->crtcs[k];
|
|
n_actual_crtcs += 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
meta_output->n_possible_crtcs = n_actual_crtcs;
|
|
|
|
meta_output->crtc = NULL;
|
|
for (j = 0; j < manager->n_crtcs; j++)
|
|
{
|
|
if ((XID)manager->crtcs[j].crtc_id == output->crtc)
|
|
{
|
|
meta_output->crtc = &manager->crtcs[j];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static char *
|
|
get_xmode_name (XRRModeInfo *xmode)
|
|
{
|
|
int width = xmode->width;
|
|
int height = xmode->height;
|
|
|
|
return g_strdup_printf ("%dx%d", width, height);
|
|
}
|
|
|
|
static void
|
|
meta_monitor_manager_xrandr_read_current (MetaMonitorManager *manager)
|
|
{
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
|
XRRScreenResources *resources;
|
|
RROutput primary_output;
|
|
unsigned int i, j, k;
|
|
unsigned int n_actual_outputs;
|
|
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)
|
|
{
|
|
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:
|
|
manager->power_save_mode = META_POWER_SAVE_UNSUPPORTED;
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
manager->power_save_mode = META_POWER_SAVE_UNSUPPORTED;
|
|
}
|
|
|
|
XRRGetScreenSizeRange (manager_xrandr->xdisplay, DefaultRootWindow (manager_xrandr->xdisplay),
|
|
&min_width,
|
|
&min_height,
|
|
&manager->max_screen_width,
|
|
&manager->max_screen_height);
|
|
|
|
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;
|
|
manager->n_outputs = resources->noutput;
|
|
manager->n_crtcs = resources->ncrtc;
|
|
manager->n_modes = resources->nmode;
|
|
manager->outputs = g_new0 (MetaOutput, manager->n_outputs);
|
|
manager->modes = g_new0 (MetaMonitorMode, manager->n_modes);
|
|
manager->crtcs = g_new0 (MetaCRTC, manager->n_crtcs);
|
|
|
|
for (i = 0; i < (unsigned)resources->nmode; i++)
|
|
{
|
|
XRRModeInfo *xmode = &resources->modes[i];
|
|
MetaMonitorMode *mode;
|
|
|
|
mode = &manager->modes[i];
|
|
|
|
mode->mode_id = xmode->id;
|
|
mode->width = xmode->width;
|
|
mode->height = xmode->height;
|
|
mode->refresh_rate = (xmode->dotClock /
|
|
((float)xmode->hTotal * xmode->vTotal));
|
|
mode->flags = xmode->modeFlags;
|
|
mode->name = get_xmode_name (xmode);
|
|
}
|
|
|
|
for (i = 0; i < (unsigned)resources->ncrtc; i++)
|
|
{
|
|
XRRCrtcInfo *crtc;
|
|
MetaCRTC *meta_crtc;
|
|
|
|
crtc = XRRGetCrtcInfo (manager_xrandr->xdisplay, resources, resources->crtcs[i]);
|
|
|
|
meta_crtc = &manager->crtcs[i];
|
|
|
|
meta_crtc->crtc_id = resources->crtcs[i];
|
|
meta_crtc->rect.x = crtc->x;
|
|
meta_crtc->rect.y = crtc->y;
|
|
meta_crtc->rect.width = crtc->width;
|
|
meta_crtc->rect.height = crtc->height;
|
|
meta_crtc->is_dirty = FALSE;
|
|
meta_crtc->transform = meta_monitor_transform_from_xrandr (crtc->rotation);
|
|
meta_crtc->all_transforms = meta_monitor_transform_from_xrandr_all (crtc->rotations);
|
|
|
|
for (j = 0; j < (unsigned)resources->nmode; j++)
|
|
{
|
|
if (resources->modes[j].id == crtc->mode)
|
|
{
|
|
meta_crtc->current_mode = &manager->modes[j];
|
|
break;
|
|
}
|
|
}
|
|
|
|
XRRFreeCrtcInfo (crtc);
|
|
}
|
|
|
|
primary_output = XRRGetOutputPrimary (manager_xrandr->xdisplay,
|
|
DefaultRootWindow (manager_xrandr->xdisplay));
|
|
|
|
n_actual_outputs = 0;
|
|
for (i = 0; i < (unsigned)resources->noutput; i++)
|
|
{
|
|
XRROutputInfo *output;
|
|
MetaOutput *meta_output;
|
|
|
|
output = XRRGetOutputInfo (manager_xrandr->xdisplay, resources, resources->outputs[i]);
|
|
if (!output)
|
|
continue;
|
|
|
|
meta_output = &manager->outputs[n_actual_outputs];
|
|
|
|
if (output->connection != RR_Disconnected)
|
|
{
|
|
GBytes *edid;
|
|
|
|
meta_output->winsys_id = resources->outputs[i];
|
|
meta_output->name = g_strdup (output->name);
|
|
|
|
edid = read_output_edid (manager_xrandr, meta_output->winsys_id);
|
|
meta_output_parse_edid (meta_output, edid);
|
|
g_bytes_unref (edid);
|
|
|
|
meta_output->width_mm = output->mm_width;
|
|
meta_output->height_mm = output->mm_height;
|
|
meta_output->subpixel_order = COGL_SUBPIXEL_ORDER_UNKNOWN;
|
|
meta_output->hotplug_mode_update = output_get_hotplug_mode_update (manager_xrandr, meta_output);
|
|
meta_output->suggested_x = output_get_suggested_x (manager_xrandr, meta_output);
|
|
meta_output->suggested_y = output_get_suggested_y (manager_xrandr, meta_output);
|
|
meta_output->connector_type = output_get_connector_type (manager_xrandr, meta_output);
|
|
|
|
output_get_tile_info (manager_xrandr, meta_output);
|
|
output_get_modes (manager, meta_output, output);
|
|
output_get_crtcs (manager, meta_output, output);
|
|
|
|
meta_output->n_possible_clones = output->nclone;
|
|
meta_output->possible_clones = g_new0 (MetaOutput *, meta_output->n_possible_clones);
|
|
/* We can build the list of clones now, because we don't have the list of outputs
|
|
yet, so temporarily set the pointers to the bare XIDs, and then we'll fix them
|
|
in a second pass
|
|
*/
|
|
for (j = 0; j < (unsigned)output->nclone; j++)
|
|
{
|
|
meta_output->possible_clones[j] = GINT_TO_POINTER (output->clones[j]);
|
|
}
|
|
|
|
meta_output->is_primary = ((XID)meta_output->winsys_id == primary_output);
|
|
meta_output->is_presentation = output_get_presentation_xrandr (manager_xrandr, meta_output);
|
|
meta_output->is_underscanning = output_get_underscanning_xrandr (manager_xrandr, meta_output);
|
|
meta_output->supports_underscanning = output_get_supports_underscanning_xrandr (manager_xrandr, meta_output);
|
|
output_get_backlight_limits_xrandr (manager_xrandr, meta_output);
|
|
|
|
if (!(meta_output->backlight_min == 0 && meta_output->backlight_max == 0))
|
|
meta_output->backlight = output_get_backlight_xrandr (manager_xrandr, meta_output);
|
|
else
|
|
meta_output->backlight = -1;
|
|
|
|
if (meta_output->n_modes == 0 || meta_output->n_possible_crtcs == 0)
|
|
meta_monitor_manager_clear_output (meta_output);
|
|
else
|
|
n_actual_outputs++;
|
|
}
|
|
|
|
XRRFreeOutputInfo (output);
|
|
}
|
|
|
|
manager->n_outputs = n_actual_outputs;
|
|
|
|
/* Sort the outputs for easier handling in MetaMonitorConfig */
|
|
qsort (manager->outputs, manager->n_outputs, sizeof (MetaOutput), compare_outputs);
|
|
|
|
/* Now fix the clones */
|
|
for (i = 0; i < manager->n_outputs; i++)
|
|
{
|
|
MetaOutput *meta_output;
|
|
|
|
meta_output = &manager->outputs[i];
|
|
|
|
for (j = 0; j < meta_output->n_possible_clones; j++)
|
|
{
|
|
RROutput clone = GPOINTER_TO_INT (meta_output->possible_clones[j]);
|
|
|
|
for (k = 0; k < manager->n_outputs; k++)
|
|
{
|
|
if (clone == (XID)manager->outputs[k].winsys_id)
|
|
{
|
|
meta_output->possible_clones[j] = &manager->outputs[k];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static GBytes *
|
|
meta_monitor_manager_xrandr_read_edid (MetaMonitorManager *manager,
|
|
MetaOutput *output)
|
|
{
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
|
|
|
return read_output_edid (manager_xrandr, output->winsys_id);
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
static Rotation
|
|
meta_monitor_transform_to_xrandr (MetaMonitorTransform transform)
|
|
{
|
|
switch (transform)
|
|
{
|
|
case META_MONITOR_TRANSFORM_NORMAL:
|
|
return RR_Rotate_0;
|
|
case META_MONITOR_TRANSFORM_90:
|
|
return RR_Rotate_90;
|
|
case META_MONITOR_TRANSFORM_180:
|
|
return RR_Rotate_180;
|
|
case META_MONITOR_TRANSFORM_270:
|
|
return RR_Rotate_270;
|
|
case META_MONITOR_TRANSFORM_FLIPPED:
|
|
return RR_Reflect_X | RR_Rotate_0;
|
|
case META_MONITOR_TRANSFORM_FLIPPED_90:
|
|
return RR_Reflect_X | RR_Rotate_90;
|
|
case META_MONITOR_TRANSFORM_FLIPPED_180:
|
|
return RR_Reflect_X | RR_Rotate_180;
|
|
case META_MONITOR_TRANSFORM_FLIPPED_270:
|
|
return RR_Reflect_X | RR_Rotate_270;
|
|
}
|
|
|
|
g_assert_not_reached ();
|
|
}
|
|
|
|
static void
|
|
output_set_presentation_xrandr (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output,
|
|
gboolean presentation)
|
|
{
|
|
Atom atom;
|
|
int value = presentation;
|
|
|
|
atom = XInternAtom (manager_xrandr->xdisplay, "_MUTTER_PRESENTATION_OUTPUT", False);
|
|
|
|
xcb_randr_change_output_property (XGetXCBConnection (manager_xrandr->xdisplay),
|
|
(XID)output->winsys_id,
|
|
atom, XCB_ATOM_CARDINAL, 32,
|
|
XCB_PROP_MODE_REPLACE,
|
|
1, &value);
|
|
}
|
|
|
|
static void
|
|
output_set_underscanning_xrandr (MetaMonitorManagerXrandr *manager_xrandr,
|
|
MetaOutput *output,
|
|
gboolean underscanning)
|
|
{
|
|
Atom prop, valueatom;
|
|
const char *value;
|
|
|
|
prop = XInternAtom (manager_xrandr->xdisplay, "underscan", False);
|
|
|
|
value = underscanning ? "on" : "off";
|
|
valueatom = XInternAtom (manager_xrandr->xdisplay, value, False);
|
|
|
|
xcb_randr_change_output_property (XGetXCBConnection (manager_xrandr->xdisplay),
|
|
(XID)output->winsys_id,
|
|
prop, XCB_ATOM_ATOM, 32,
|
|
XCB_PROP_MODE_REPLACE,
|
|
1, &valueatom);
|
|
|
|
/* Configure the border at the same time. Currently, we use a
|
|
* 5% of the width/height of the mode. In the future, we should
|
|
* make the border configurable. */
|
|
if (underscanning)
|
|
{
|
|
uint32_t border_value;
|
|
|
|
prop = XInternAtom (manager_xrandr->xdisplay, "underscan hborder", False);
|
|
border_value = output->crtc->current_mode->width * 0.05;
|
|
|
|
xcb_randr_change_output_property (XGetXCBConnection (manager_xrandr->xdisplay),
|
|
(XID)output->winsys_id,
|
|
prop, XCB_ATOM_INTEGER, 32,
|
|
XCB_PROP_MODE_REPLACE,
|
|
1, &border_value);
|
|
|
|
prop = XInternAtom (manager_xrandr->xdisplay, "underscan vborder", False);
|
|
border_value = output->crtc->current_mode->height * 0.05;
|
|
|
|
xcb_randr_change_output_property (XGetXCBConnection (manager_xrandr->xdisplay),
|
|
(XID)output->winsys_id,
|
|
prop, XCB_ATOM_INTEGER, 32,
|
|
XCB_PROP_MODE_REPLACE,
|
|
1, &border_value);
|
|
}
|
|
}
|
|
|
|
static void
|
|
meta_monitor_manager_xrandr_apply_configuration (MetaMonitorManager *manager,
|
|
MetaCRTCInfo **crtcs,
|
|
unsigned int n_crtcs,
|
|
MetaOutputInfo **outputs,
|
|
unsigned int n_outputs)
|
|
{
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
|
unsigned i;
|
|
int width, height, width_mm, height_mm;
|
|
|
|
XGrabServer (manager_xrandr->xdisplay);
|
|
|
|
/* First compute the new size of the screen (framebuffer) */
|
|
width = 0; height = 0;
|
|
for (i = 0; i < n_crtcs; i++)
|
|
{
|
|
MetaCRTCInfo *crtc_info = crtcs[i];
|
|
MetaCRTC *crtc = crtc_info->crtc;
|
|
crtc->is_dirty = TRUE;
|
|
|
|
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);
|
|
}
|
|
}
|
|
|
|
/* 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++)
|
|
{
|
|
MetaCRTCInfo *crtc_info = crtcs[i];
|
|
MetaCRTC *crtc = crtc_info->crtc;
|
|
|
|
if (crtc_info->mode == NULL ||
|
|
crtc->rect.x + crtc->rect.width > width ||
|
|
crtc->rect.y + crtc->rect.height > height)
|
|
{
|
|
XRRSetCrtcConfig (manager_xrandr->xdisplay,
|
|
manager_xrandr->resources,
|
|
(XID)crtc->crtc_id,
|
|
CurrentTime,
|
|
0, 0,
|
|
None,
|
|
RR_Rotate_0,
|
|
NULL, 0);
|
|
|
|
crtc->rect.x = 0;
|
|
crtc->rect.y = 0;
|
|
crtc->rect.width = 0;
|
|
crtc->rect.height = 0;
|
|
crtc->current_mode = NULL;
|
|
}
|
|
}
|
|
|
|
/* Disable CRTCs not mentioned in the list */
|
|
for (i = 0; i < manager->n_crtcs; i++)
|
|
{
|
|
MetaCRTC *crtc = &manager->crtcs[i];
|
|
|
|
if (crtc->is_dirty)
|
|
{
|
|
crtc->is_dirty = FALSE;
|
|
continue;
|
|
}
|
|
if (crtc->current_mode == NULL)
|
|
continue;
|
|
|
|
XRRSetCrtcConfig (manager_xrandr->xdisplay,
|
|
manager_xrandr->resources,
|
|
(XID)crtc->crtc_id,
|
|
CurrentTime,
|
|
0, 0,
|
|
None,
|
|
RR_Rotate_0,
|
|
NULL, 0);
|
|
|
|
crtc->rect.x = 0;
|
|
crtc->rect.y = 0;
|
|
crtc->rect.width = 0;
|
|
crtc->rect.height = 0;
|
|
crtc->current_mode = NULL;
|
|
}
|
|
|
|
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);
|
|
|
|
for (i = 0; i < n_crtcs; i++)
|
|
{
|
|
MetaCRTCInfo *crtc_info = crtcs[i];
|
|
MetaCRTC *crtc = crtc_info->crtc;
|
|
|
|
if (crtc_info->mode != NULL)
|
|
{
|
|
MetaMonitorMode *mode;
|
|
g_autofree XID *output_ids = NULL;
|
|
unsigned int j, n_output_ids;
|
|
Status ok;
|
|
|
|
mode = crtc_info->mode;
|
|
|
|
n_output_ids = crtc_info->outputs->len;
|
|
output_ids = g_new (XID, n_output_ids);
|
|
|
|
for (j = 0; j < n_output_ids; j++)
|
|
{
|
|
MetaOutput *output;
|
|
|
|
output = ((MetaOutput**)crtc_info->outputs->pdata)[j];
|
|
|
|
output->is_dirty = TRUE;
|
|
output->crtc = crtc;
|
|
|
|
output_ids[j] = output->winsys_id;
|
|
}
|
|
|
|
ok = XRRSetCrtcConfig (manager_xrandr->xdisplay,
|
|
manager_xrandr->resources,
|
|
(XID)crtc->crtc_id,
|
|
CurrentTime,
|
|
crtc_info->x, crtc_info->y,
|
|
(XID)mode->mode_id,
|
|
meta_monitor_transform_to_xrandr (crtc_info->transform),
|
|
output_ids, n_output_ids);
|
|
|
|
if (ok != Success)
|
|
{
|
|
meta_warning ("Configuring CRTC %d with mode %d (%d x %d @ %f) at position %d, %d and transform %u failed\n",
|
|
(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);
|
|
continue;
|
|
}
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < n_outputs; i++)
|
|
{
|
|
MetaOutputInfo *output_info = outputs[i];
|
|
MetaOutput *output = output_info->output;
|
|
|
|
if (output_info->is_primary)
|
|
{
|
|
XRRSetOutputPrimary (manager_xrandr->xdisplay,
|
|
DefaultRootWindow (manager_xrandr->xdisplay),
|
|
(XID)output_info->output->winsys_id);
|
|
}
|
|
|
|
output_set_presentation_xrandr (manager_xrandr,
|
|
output_info->output,
|
|
output_info->is_presentation);
|
|
|
|
if (output_get_supports_underscanning_xrandr (manager_xrandr, output_info->output))
|
|
output_set_underscanning_xrandr (manager_xrandr,
|
|
output_info->output,
|
|
output_info->is_underscanning);
|
|
|
|
output->is_primary = output_info->is_primary;
|
|
output->is_presentation = output_info->is_presentation;
|
|
output->is_underscanning = output_info->is_underscanning;
|
|
}
|
|
|
|
/* Disable outputs not mentioned in the list */
|
|
for (i = 0; i < manager->n_outputs; i++)
|
|
{
|
|
MetaOutput *output = &manager->outputs[i];
|
|
|
|
if (output->is_dirty)
|
|
{
|
|
output->is_dirty = FALSE;
|
|
continue;
|
|
}
|
|
|
|
output->crtc = NULL;
|
|
output->is_primary = FALSE;
|
|
}
|
|
|
|
XUngrabServer (manager_xrandr->xdisplay);
|
|
XFlush (manager_xrandr->xdisplay);
|
|
}
|
|
|
|
static void
|
|
meta_monitor_manager_xrandr_change_backlight (MetaMonitorManager *manager,
|
|
MetaOutput *output,
|
|
gint value)
|
|
{
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
|
Atom atom;
|
|
int hw_value;
|
|
|
|
hw_value = round ((double)value / 100.0 * output->backlight_max + output->backlight_min);
|
|
|
|
atom = XInternAtom (manager_xrandr->xdisplay, "Backlight", False);
|
|
|
|
xcb_randr_change_output_property (XGetXCBConnection (manager_xrandr->xdisplay),
|
|
(XID)output->winsys_id,
|
|
atom, XCB_ATOM_INTEGER, 32,
|
|
XCB_PROP_MODE_REPLACE,
|
|
1, &hw_value);
|
|
|
|
/* We're not selecting for property notifies, so update the value immediately */
|
|
output->backlight = normalize_backlight (output, hw_value);
|
|
}
|
|
|
|
static void
|
|
meta_monitor_manager_xrandr_get_crtc_gamma (MetaMonitorManager *manager,
|
|
MetaCRTC *crtc,
|
|
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,
|
|
MetaCRTC *crtc,
|
|
gsize size,
|
|
unsigned short *red,
|
|
unsigned short *green,
|
|
unsigned short *blue)
|
|
{
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
|
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);
|
|
|
|
XRRSetCrtcGamma (manager_xrandr->xdisplay, (XID)crtc->crtc_id, gamma);
|
|
|
|
XRRFreeGamma (gamma);
|
|
}
|
|
|
|
#ifdef HAVE_XRANDR15
|
|
static void
|
|
meta_monitor_manager_xrandr_add_monitor (MetaMonitorManager *manager,
|
|
MetaLogicalMonitor *logical_monitor)
|
|
{
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
|
XRRMonitorInfo *m;
|
|
int o;
|
|
Atom name;
|
|
char name_buf[40];
|
|
|
|
if (manager_xrandr->has_randr15 == FALSE)
|
|
return;
|
|
|
|
if (logical_monitor->n_outputs <= 1)
|
|
return;
|
|
|
|
if (logical_monitor->outputs[0]->product)
|
|
snprintf (name_buf, 40, "%s-%d",
|
|
logical_monitor->outputs[0]->product,
|
|
logical_monitor->outputs[0]->tile_info.group_id);
|
|
else
|
|
snprintf (name_buf, 40, "Tiled-%d",
|
|
logical_monitor->outputs[0]->tile_info.group_id);
|
|
|
|
name = XInternAtom (manager_xrandr->xdisplay, name_buf, False);
|
|
logical_monitor->monitor_winsys_xid = name;
|
|
m = XRRAllocateMonitor (manager_xrandr->xdisplay,
|
|
logical_monitor->n_outputs);
|
|
if (!m)
|
|
return;
|
|
m->name = name;
|
|
m->primary = logical_monitor->is_primary;
|
|
m->automatic = True;
|
|
|
|
for (o = 0; o < logical_monitor->n_outputs; o++) {
|
|
MetaOutput *output = logical_monitor->outputs[o];
|
|
m->outputs[o] = output->winsys_id;
|
|
}
|
|
XRRSetMonitor (manager_xrandr->xdisplay,
|
|
DefaultRootWindow (manager_xrandr->xdisplay),
|
|
m);
|
|
XRRFreeMonitors (m);
|
|
}
|
|
|
|
static void
|
|
meta_monitor_manager_xrandr_delete_monitor (MetaMonitorManager *manager,
|
|
int monitor_winsys_xid)
|
|
{
|
|
MetaMonitorManagerXrandr *manager_xrandr = META_MONITOR_MANAGER_XRANDR (manager);
|
|
|
|
if (manager_xrandr->has_randr15 == FALSE)
|
|
return;
|
|
XRRDeleteMonitor (manager_xrandr->xdisplay,
|
|
DefaultRootWindow (manager_xrandr->xdisplay),
|
|
monitor_winsys_xid);
|
|
}
|
|
|
|
static void
|
|
meta_monitor_manager_xrandr_init_monitors (MetaMonitorManagerXrandr *manager_xrandr)
|
|
{
|
|
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
|
|
|
|
static void
|
|
meta_monitor_manager_xrandr_init (MetaMonitorManagerXrandr *manager_xrandr)
|
|
{
|
|
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
|
|
{
|
|
int major_version, minor_version;
|
|
/* 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);
|
|
|
|
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))
|
|
manager_xrandr->has_randr15 = TRUE;
|
|
meta_monitor_manager_xrandr_init_monitors (manager_xrandr);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
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;
|
|
|
|
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;
|
|
manager_class->apply_configuration = meta_monitor_manager_xrandr_apply_configuration;
|
|
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;
|
|
#ifdef HAVE_XRANDR15
|
|
manager_class->add_monitor = meta_monitor_manager_xrandr_add_monitor;
|
|
manager_class->delete_monitor = meta_monitor_manager_xrandr_delete_monitor;
|
|
#endif
|
|
}
|
|
|
|
gboolean
|
|
meta_monitor_manager_xrandr_handle_xevent (MetaMonitorManagerXrandr *manager_xrandr,
|
|
XEvent *event)
|
|
{
|
|
MetaMonitorManager *manager = META_MONITOR_MANAGER (manager_xrandr);
|
|
gboolean hotplug;
|
|
|
|
if ((event->type - manager_xrandr->rr_event_base) != RRScreenChangeNotify)
|
|
return FALSE;
|
|
|
|
XRRUpdateConfiguration (event);
|
|
|
|
meta_monitor_manager_read_current_config (manager);
|
|
|
|
hotplug = manager_xrandr->resources->timestamp < manager_xrandr->resources->configTimestamp;
|
|
if (hotplug)
|
|
{
|
|
/* This is a hotplug event, so go ahead and build a new configuration. */
|
|
meta_monitor_manager_on_hotplug (manager);
|
|
}
|
|
else
|
|
{
|
|
/* Something else changed -- tell the world about it. */
|
|
meta_monitor_manager_rebuild_derived (manager);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|