265 lines
7.4 KiB
C
265 lines
7.4 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-2017 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 "backends/x11/meta-gpu-xrandr.h"
|
||
|
|
||
|
#include <string.h>
|
||
|
#include <X11/extensions/dpms.h>
|
||
|
#include <X11/Xlibint.h>
|
||
|
|
||
|
#include "backends/meta-output.h"
|
||
|
#include "backends/x11/meta-crtc-xrandr.h"
|
||
|
#include "backends/x11/meta-monitor-manager-xrandr.h"
|
||
|
#include "backends/x11/meta-output-xrandr.h"
|
||
|
|
||
|
struct _MetaGpuXrandr
|
||
|
{
|
||
|
MetaGpu parent;
|
||
|
|
||
|
XRRScreenResources *resources;
|
||
|
|
||
|
int max_screen_width;
|
||
|
int max_screen_height;
|
||
|
};
|
||
|
|
||
|
G_DEFINE_TYPE (MetaGpuXrandr, meta_gpu_xrandr, META_TYPE_GPU)
|
||
|
|
||
|
XRRScreenResources *
|
||
|
meta_gpu_xrandr_get_resources (MetaGpuXrandr *gpu_xrandr)
|
||
|
{
|
||
|
return gpu_xrandr->resources;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
meta_gpu_xrandr_get_max_screen_size (MetaGpuXrandr *gpu_xrandr,
|
||
|
int *max_width,
|
||
|
int *max_height)
|
||
|
{
|
||
|
*max_width = gpu_xrandr->max_screen_width;
|
||
|
*max_height = gpu_xrandr->max_screen_height;
|
||
|
}
|
||
|
|
||
|
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 char *
|
||
|
get_xmode_name (XRRModeInfo *xmode)
|
||
|
{
|
||
|
int width = xmode->width;
|
||
|
int height = xmode->height;
|
||
|
|
||
|
return g_strdup_printf ("%dx%d", width, height);
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
meta_gpu_xrandr_read_current (MetaGpu *gpu,
|
||
|
GError **error)
|
||
|
{
|
||
|
MetaGpuXrandr *gpu_xrandr = META_GPU_XRANDR (gpu);
|
||
|
MetaMonitorManager *monitor_manager = meta_gpu_get_monitor_manager (gpu);
|
||
|
MetaMonitorManagerXrandr *monitor_manager_xrandr =
|
||
|
META_MONITOR_MANAGER_XRANDR (monitor_manager);
|
||
|
Display *xdisplay =
|
||
|
meta_monitor_manager_xrandr_get_xdisplay (monitor_manager_xrandr);
|
||
|
XRRScreenResources *resources;
|
||
|
RROutput primary_output;
|
||
|
unsigned int i, j;
|
||
|
GList *l;
|
||
|
int min_width, min_height;
|
||
|
Screen *screen;
|
||
|
GList *outputs = NULL;
|
||
|
GList *modes = NULL;
|
||
|
GList *crtcs = NULL;
|
||
|
|
||
|
if (gpu_xrandr->resources)
|
||
|
XRRFreeScreenResources (gpu_xrandr->resources);
|
||
|
gpu_xrandr->resources = NULL;
|
||
|
|
||
|
XRRGetScreenSizeRange (xdisplay, DefaultRootWindow (xdisplay),
|
||
|
&min_width,
|
||
|
&min_height,
|
||
|
&gpu_xrandr->max_screen_width,
|
||
|
&gpu_xrandr->max_screen_height);
|
||
|
|
||
|
screen = ScreenOfDisplay (xdisplay, DefaultScreen (xdisplay));
|
||
|
/* This is updated because we called XRRUpdateConfiguration. */
|
||
|
monitor_manager->screen_width = WidthOfScreen (screen);
|
||
|
monitor_manager->screen_height = HeightOfScreen (screen);
|
||
|
|
||
|
resources = XRRGetScreenResourcesCurrent (xdisplay,
|
||
|
DefaultRootWindow (xdisplay));
|
||
|
if (!resources)
|
||
|
{
|
||
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
|
||
|
"Failed to retrieve Xrandr screen resources");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
gpu_xrandr->resources = resources;
|
||
|
|
||
|
outputs = NULL;
|
||
|
modes = NULL;
|
||
|
crtcs = NULL;
|
||
|
|
||
|
for (i = 0; i < (unsigned)resources->nmode; i++)
|
||
|
{
|
||
|
XRRModeInfo *xmode = &resources->modes[i];
|
||
|
MetaCrtcMode *mode;
|
||
|
|
||
|
mode = g_object_new (META_TYPE_CRTC_MODE, NULL);
|
||
|
|
||
|
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);
|
||
|
|
||
|
modes = g_list_append (modes, mode);
|
||
|
}
|
||
|
meta_gpu_take_modes (gpu, modes);
|
||
|
|
||
|
for (i = 0; i < (unsigned)resources->ncrtc; i++)
|
||
|
{
|
||
|
XRRCrtcInfo *xrandr_crtc;
|
||
|
RRCrtc crtc_id;
|
||
|
MetaCrtc *crtc;
|
||
|
|
||
|
crtc_id = resources->crtcs[i];
|
||
|
xrandr_crtc = XRRGetCrtcInfo (xdisplay,
|
||
|
resources, crtc_id);
|
||
|
crtc = meta_create_xrandr_crtc (gpu_xrandr,
|
||
|
xrandr_crtc, crtc_id, resources);
|
||
|
XRRFreeCrtcInfo (xrandr_crtc);
|
||
|
|
||
|
crtcs = g_list_append (crtcs, crtc);
|
||
|
}
|
||
|
|
||
|
meta_gpu_take_crtcs (gpu, crtcs);
|
||
|
|
||
|
primary_output = XRRGetOutputPrimary (xdisplay,
|
||
|
DefaultRootWindow (xdisplay));
|
||
|
|
||
|
for (i = 0; i < (unsigned)resources->noutput; i++)
|
||
|
{
|
||
|
RROutput output_id;
|
||
|
XRROutputInfo *xrandr_output;
|
||
|
|
||
|
output_id = resources->outputs[i];
|
||
|
xrandr_output = XRRGetOutputInfo (xdisplay,
|
||
|
resources, output_id);
|
||
|
if (!xrandr_output)
|
||
|
continue;
|
||
|
|
||
|
if (xrandr_output->connection != RR_Disconnected)
|
||
|
{
|
||
|
MetaOutput *output;
|
||
|
|
||
|
output = meta_create_xrandr_output (gpu_xrandr,
|
||
|
xrandr_output,
|
||
|
output_id,
|
||
|
primary_output);
|
||
|
if (output)
|
||
|
outputs = g_list_prepend (outputs, output);
|
||
|
}
|
||
|
|
||
|
XRRFreeOutputInfo (xrandr_output);
|
||
|
}
|
||
|
|
||
|
/* Sort the outputs for easier handling in MetaMonitorConfig */
|
||
|
outputs = g_list_sort (outputs, compare_outputs);
|
||
|
|
||
|
meta_gpu_take_outputs (gpu, outputs);
|
||
|
|
||
|
/* Now fix the clones */
|
||
|
for (l = outputs; l; l = l->next)
|
||
|
{
|
||
|
MetaOutput *output = l->data;
|
||
|
GList *k;
|
||
|
|
||
|
for (j = 0; j < output->n_possible_clones; j++)
|
||
|
{
|
||
|
RROutput clone = GPOINTER_TO_INT (output->possible_clones[j]);
|
||
|
|
||
|
for (k = outputs; k; k = k->next)
|
||
|
{
|
||
|
MetaOutput *possible_clone = k->data;
|
||
|
|
||
|
if (clone == (XID) possible_clone->winsys_id)
|
||
|
{
|
||
|
output->possible_clones[j] = possible_clone;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
MetaGpuXrandr *
|
||
|
meta_gpu_xrandr_new (MetaMonitorManagerXrandr *monitor_manager_xrandr)
|
||
|
{
|
||
|
return g_object_new (META_TYPE_GPU_XRANDR,
|
||
|
"monitor-manager", monitor_manager_xrandr,
|
||
|
NULL);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
meta_gpu_xrandr_finalize (GObject *object)
|
||
|
{
|
||
|
MetaGpuXrandr *gpu_xrandr = META_GPU_XRANDR (object);
|
||
|
|
||
|
g_clear_pointer (&gpu_xrandr->resources,
|
||
|
XRRFreeScreenResources);
|
||
|
|
||
|
G_OBJECT_CLASS (meta_gpu_xrandr_parent_class)->finalize (object);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
meta_gpu_xrandr_init (MetaGpuXrandr *gpu_xrandr)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
meta_gpu_xrandr_class_init (MetaGpuXrandrClass *klass)
|
||
|
{
|
||
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
||
|
MetaGpuClass *gpu_class = META_GPU_CLASS (klass);
|
||
|
|
||
|
object_class->finalize = meta_gpu_xrandr_finalize;
|
||
|
|
||
|
gpu_class->read_current = meta_gpu_xrandr_read_current;
|
||
|
}
|