1fb0e8e9bb
Instead of storing the logical monitors in an array and having users either look up them in the array given an index or iterate using indices, put it in a GList, and use GList iterators when iterating and alternative API where array indices were previously used. This allows for more liberty regarding the type of the logical monitor. https://bugzilla.gnome.org/show_bug.cgi?id=777732
311 lines
10 KiB
C
311 lines
10 KiB
C
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
/*
|
|
* Copyright (C) 2014 Red Hat
|
|
*
|
|
* 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, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
* 02111-1307, USA.
|
|
*
|
|
* Written by:
|
|
* Jasper St. Pierre <jstpierre@mecheye.net>
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "meta-wayland-outputs.h"
|
|
|
|
#include "meta-wayland-private.h"
|
|
#include "meta-monitor-manager-private.h"
|
|
|
|
#include <string.h>
|
|
|
|
enum {
|
|
OUTPUT_DESTROYED,
|
|
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static guint signals[LAST_SIGNAL];
|
|
|
|
G_DEFINE_TYPE (MetaWaylandOutput, meta_wayland_output, G_TYPE_OBJECT)
|
|
|
|
static void
|
|
output_resource_destroy (struct wl_resource *res)
|
|
{
|
|
MetaWaylandOutput *wayland_output;
|
|
|
|
wayland_output = wl_resource_get_user_data (res);
|
|
if (!wayland_output)
|
|
return;
|
|
|
|
wayland_output->resources = g_list_remove (wayland_output->resources, res);
|
|
}
|
|
|
|
static void
|
|
send_output_events (struct wl_resource *resource,
|
|
MetaWaylandOutput *wayland_output,
|
|
MetaLogicalMonitor *logical_monitor,
|
|
gboolean need_all_events)
|
|
{
|
|
int version = wl_resource_get_version (resource);
|
|
MetaOutput *output = logical_monitor->outputs[0];
|
|
guint mode_flags = WL_OUTPUT_MODE_CURRENT;
|
|
MetaLogicalMonitor *old_logical_monitor = wayland_output->logical_monitor;
|
|
guint old_mode_flags = wayland_output->mode_flags;
|
|
gint old_scale = wayland_output->scale;
|
|
|
|
gboolean need_done = FALSE;
|
|
|
|
if (need_all_events ||
|
|
old_logical_monitor->rect.x != logical_monitor->rect.x ||
|
|
old_logical_monitor->rect.y != logical_monitor->rect.y)
|
|
{
|
|
/*
|
|
* TODO: When we support wl_surface.set_buffer_transform, pass along
|
|
* the correct transform here instead of always pretending its 'normal'.
|
|
* The reason for this is to try stopping clients from setting any buffer
|
|
* transform other than 'normal'.
|
|
*/
|
|
wl_output_send_geometry (resource,
|
|
logical_monitor->rect.x,
|
|
logical_monitor->rect.y,
|
|
logical_monitor->width_mm,
|
|
logical_monitor->height_mm,
|
|
output->subpixel_order,
|
|
output->vendor,
|
|
output->product,
|
|
WL_OUTPUT_TRANSFORM_NORMAL);
|
|
need_done = TRUE;
|
|
}
|
|
|
|
if (output->crtc->current_mode == output->preferred_mode)
|
|
mode_flags |= WL_OUTPUT_MODE_PREFERRED;
|
|
|
|
if (need_all_events ||
|
|
old_logical_monitor->rect.width != logical_monitor->rect.width ||
|
|
old_logical_monitor->rect.height != logical_monitor->rect.height ||
|
|
old_logical_monitor->refresh_rate != logical_monitor->refresh_rate ||
|
|
old_mode_flags != mode_flags)
|
|
{
|
|
wl_output_send_mode (resource,
|
|
mode_flags,
|
|
logical_monitor->rect.width,
|
|
logical_monitor->rect.height,
|
|
(int32_t) (logical_monitor->refresh_rate * 1000));
|
|
need_done = TRUE;
|
|
}
|
|
|
|
if (version >= WL_OUTPUT_SCALE_SINCE_VERSION)
|
|
{
|
|
if (need_all_events ||
|
|
old_scale != output->scale)
|
|
{
|
|
wl_output_send_scale (resource, output->scale);
|
|
need_done = TRUE;
|
|
}
|
|
|
|
if (need_done)
|
|
wl_output_send_done (resource);
|
|
}
|
|
}
|
|
|
|
static void
|
|
bind_output (struct wl_client *client,
|
|
void *data,
|
|
guint32 version,
|
|
guint32 id)
|
|
{
|
|
MetaWaylandOutput *wayland_output = data;
|
|
MetaLogicalMonitor *logical_monitor = wayland_output->logical_monitor;
|
|
struct wl_resource *resource;
|
|
MetaOutput *output = logical_monitor->outputs[0];
|
|
|
|
resource = wl_resource_create (client, &wl_output_interface, version, id);
|
|
wayland_output->resources = g_list_prepend (wayland_output->resources, resource);
|
|
|
|
wl_resource_set_user_data (resource, wayland_output);
|
|
wl_resource_set_destructor (resource, output_resource_destroy);
|
|
|
|
meta_verbose ("Binding monitor %p/%s (%u, %u, %u, %u) x %f\n",
|
|
logical_monitor, output->name,
|
|
logical_monitor->rect.x, logical_monitor->rect.y,
|
|
logical_monitor->rect.width, logical_monitor->rect.height,
|
|
logical_monitor->refresh_rate);
|
|
|
|
send_output_events (resource, wayland_output, logical_monitor, TRUE);
|
|
}
|
|
|
|
static void
|
|
wayland_output_destroy_notify (gpointer data)
|
|
{
|
|
MetaWaylandOutput *wayland_output = data;
|
|
|
|
g_signal_emit (wayland_output, signals[OUTPUT_DESTROYED], 0);
|
|
g_object_unref (wayland_output);
|
|
}
|
|
|
|
static void
|
|
meta_wayland_output_set_logical_monitor (MetaWaylandOutput *wayland_output,
|
|
MetaLogicalMonitor *logical_monitor)
|
|
{
|
|
MetaOutput *output = logical_monitor->outputs[0];
|
|
|
|
wayland_output->logical_monitor = logical_monitor;
|
|
wayland_output->mode_flags = WL_OUTPUT_MODE_CURRENT;
|
|
if (output->crtc->current_mode == output->preferred_mode)
|
|
wayland_output->mode_flags |= WL_OUTPUT_MODE_PREFERRED;
|
|
wayland_output->scale = output->scale;
|
|
}
|
|
|
|
static void
|
|
wayland_output_update_for_output (MetaWaylandOutput *wayland_output,
|
|
MetaLogicalMonitor *logical_monitor)
|
|
{
|
|
GList *iter;
|
|
|
|
for (iter = wayland_output->resources; iter; iter = iter->next)
|
|
{
|
|
struct wl_resource *resource = iter->data;
|
|
send_output_events (resource, wayland_output, logical_monitor, FALSE);
|
|
}
|
|
|
|
/* It's very important that we change the output pointer here, as
|
|
the old structure is about to be freed by MetaMonitorManager */
|
|
meta_wayland_output_set_logical_monitor (wayland_output, logical_monitor);
|
|
}
|
|
|
|
static MetaWaylandOutput *
|
|
meta_wayland_output_new (MetaWaylandCompositor *compositor,
|
|
MetaLogicalMonitor *logical_monitor)
|
|
{
|
|
MetaWaylandOutput *wayland_output;
|
|
|
|
wayland_output = g_object_new (META_TYPE_WAYLAND_OUTPUT, NULL);
|
|
wayland_output->global = wl_global_create (compositor->wayland_display,
|
|
&wl_output_interface,
|
|
META_WL_OUTPUT_VERSION,
|
|
wayland_output, bind_output);
|
|
meta_wayland_output_set_logical_monitor (wayland_output, logical_monitor);
|
|
|
|
return wayland_output;
|
|
}
|
|
|
|
static GHashTable *
|
|
meta_wayland_compositor_update_outputs (MetaWaylandCompositor *compositor,
|
|
MetaMonitorManager *monitor_manager)
|
|
{
|
|
GHashTable *new_table;
|
|
GList *logical_monitors, *l;
|
|
|
|
logical_monitors =
|
|
meta_monitor_manager_get_logical_monitors (monitor_manager);
|
|
new_table = g_hash_table_new_full (NULL, NULL, NULL,
|
|
wayland_output_destroy_notify);
|
|
|
|
for (l = logical_monitors; l; l = l->next)
|
|
{
|
|
MetaLogicalMonitor *logical_monitor = l->data;
|
|
MetaWaylandOutput *wayland_output;
|
|
|
|
if (logical_monitor->winsys_id == 0)
|
|
continue;
|
|
|
|
wayland_output =
|
|
g_hash_table_lookup (compositor->outputs,
|
|
GSIZE_TO_POINTER (logical_monitor->winsys_id));
|
|
|
|
if (wayland_output)
|
|
{
|
|
g_hash_table_steal (compositor->outputs,
|
|
GSIZE_TO_POINTER (logical_monitor->winsys_id));
|
|
}
|
|
else
|
|
{
|
|
wayland_output = meta_wayland_output_new (compositor, logical_monitor);
|
|
}
|
|
|
|
wayland_output_update_for_output (wayland_output, logical_monitor);
|
|
g_hash_table_insert (new_table,
|
|
GSIZE_TO_POINTER (logical_monitor->winsys_id),
|
|
wayland_output);
|
|
}
|
|
|
|
g_hash_table_destroy (compositor->outputs);
|
|
return new_table;
|
|
}
|
|
|
|
static void
|
|
on_monitors_changed (MetaMonitorManager *monitors,
|
|
MetaWaylandCompositor *compositor)
|
|
{
|
|
compositor->outputs = meta_wayland_compositor_update_outputs (compositor, monitors);
|
|
}
|
|
|
|
static void
|
|
meta_wayland_output_init (MetaWaylandOutput *wayland_output)
|
|
{
|
|
}
|
|
|
|
static void
|
|
meta_wayland_output_finalize (GObject *object)
|
|
{
|
|
MetaWaylandOutput *wayland_output = META_WAYLAND_OUTPUT (object);
|
|
GList *l;
|
|
|
|
wl_global_destroy (wayland_output->global);
|
|
|
|
/* Make sure the wl_output destructor doesn't try to access MetaWaylandOutput
|
|
* after we have freed it.
|
|
*/
|
|
for (l = wayland_output->resources; l; l = l->next)
|
|
{
|
|
struct wl_resource *output_resource = l->data;
|
|
|
|
wl_resource_set_user_data (output_resource, NULL);
|
|
}
|
|
|
|
g_list_free (wayland_output->resources);
|
|
|
|
G_OBJECT_CLASS (meta_wayland_output_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
meta_wayland_output_class_init (MetaWaylandOutputClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->finalize = meta_wayland_output_finalize;
|
|
|
|
signals[OUTPUT_DESTROYED] = g_signal_new ("output-destroyed",
|
|
G_TYPE_FROM_CLASS (object_class),
|
|
G_SIGNAL_RUN_LAST,
|
|
0,
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
}
|
|
|
|
void
|
|
meta_wayland_outputs_init (MetaWaylandCompositor *compositor)
|
|
{
|
|
MetaMonitorManager *monitors;
|
|
|
|
monitors = meta_monitor_manager_get ();
|
|
g_signal_connect (monitors, "monitors-changed",
|
|
G_CALLBACK (on_monitors_changed), compositor);
|
|
|
|
compositor->outputs = g_hash_table_new_full (NULL, NULL, NULL, wayland_output_destroy_notify);
|
|
compositor->outputs = meta_wayland_compositor_update_outputs (compositor, monitors);
|
|
}
|