fad0efa687
The color space and HDR metadata are eventually sent as metadata to the display. The color space informs the display of the colorimetry of the frames we produce, the HDR metadata informs the display of the transfer function and additional mastering display colorimetry and luminance to guide tone and gamut mapping. The only color spaces we support right now are the default color space and Rec bt.2020 which is typically used for HDR content. Other supported color spaces can be added when needed. The default color space corresponds to whatever colorimetry the display has when no further changes are made to the calibration of the display. The colorimetry is communicated to sources via EDID/DisplayID. Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2879>
713 lines
18 KiB
C
713 lines
18 KiB
C
/*
|
|
* Copyright (C) 2017 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.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "backends/edid.h"
|
|
#include "backends/meta-output.h"
|
|
|
|
#include "backends/meta-crtc.h"
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
|
|
PROP_ID,
|
|
PROP_GPU,
|
|
PROP_INFO,
|
|
PROP_IS_PRIVACY_SCREEN_ENABLED,
|
|
|
|
N_PROPS
|
|
};
|
|
|
|
static GParamSpec *obj_props[N_PROPS];
|
|
|
|
enum
|
|
{
|
|
COLOR_SPACE_CHANGED,
|
|
HDR_METADATA_CHANGED,
|
|
|
|
N_SIGNALS
|
|
};
|
|
|
|
static guint signals[N_SIGNALS];
|
|
|
|
typedef struct _MetaOutputPrivate
|
|
{
|
|
uint64_t id;
|
|
|
|
MetaGpu *gpu;
|
|
|
|
MetaOutputInfo *info;
|
|
|
|
MetaMonitor *monitor;
|
|
|
|
/* The CRTC driving this output, NULL if the output is not enabled */
|
|
MetaCrtc *crtc;
|
|
|
|
gboolean is_primary;
|
|
gboolean is_presentation;
|
|
|
|
gboolean is_underscanning;
|
|
|
|
gboolean has_max_bpc;
|
|
unsigned int max_bpc;
|
|
|
|
int backlight;
|
|
|
|
MetaPrivacyScreenState privacy_screen_state;
|
|
gboolean is_privacy_screen_enabled;
|
|
|
|
MetaOutputHdrMetadata hdr_metadata;
|
|
MetaOutputColorspace color_space;
|
|
} MetaOutputPrivate;
|
|
|
|
G_DEFINE_ABSTRACT_TYPE_WITH_PRIVATE (MetaOutput, meta_output, G_TYPE_OBJECT)
|
|
|
|
G_DEFINE_BOXED_TYPE (MetaOutputInfo, meta_output_info,
|
|
meta_output_info_ref,
|
|
meta_output_info_unref)
|
|
|
|
MetaOutputInfo *
|
|
meta_output_info_new (void)
|
|
{
|
|
MetaOutputInfo *output_info;
|
|
|
|
output_info = g_new0 (MetaOutputInfo, 1);
|
|
g_ref_count_init (&output_info->ref_count);
|
|
|
|
return output_info;
|
|
}
|
|
|
|
MetaOutputInfo *
|
|
meta_output_info_ref (MetaOutputInfo *output_info)
|
|
{
|
|
g_ref_count_inc (&output_info->ref_count);
|
|
return output_info;
|
|
}
|
|
|
|
void
|
|
meta_output_info_unref (MetaOutputInfo *output_info)
|
|
{
|
|
if (g_ref_count_dec (&output_info->ref_count))
|
|
{
|
|
g_free (output_info->name);
|
|
g_free (output_info->vendor);
|
|
g_free (output_info->product);
|
|
g_free (output_info->serial);
|
|
g_free (output_info->edid_checksum_md5);
|
|
g_free (output_info->edid_info);
|
|
g_free (output_info->modes);
|
|
g_free (output_info->possible_crtcs);
|
|
g_free (output_info->possible_clones);
|
|
g_free (output_info);
|
|
}
|
|
}
|
|
|
|
uint64_t
|
|
meta_output_get_id (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
return priv->id;
|
|
}
|
|
|
|
MetaGpu *
|
|
meta_output_get_gpu (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
return priv->gpu;
|
|
}
|
|
|
|
MetaMonitor *
|
|
meta_output_get_monitor (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
g_warn_if_fail (priv->monitor);
|
|
|
|
return priv->monitor;
|
|
}
|
|
|
|
void
|
|
meta_output_set_monitor (MetaOutput *output,
|
|
MetaMonitor *monitor)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
g_warn_if_fail (!priv->monitor);
|
|
|
|
priv->monitor = monitor;
|
|
}
|
|
|
|
void
|
|
meta_output_unset_monitor (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
g_warn_if_fail (priv->monitor);
|
|
|
|
priv->monitor = NULL;
|
|
}
|
|
|
|
const char *
|
|
meta_output_get_name (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
return priv->info->name;
|
|
}
|
|
|
|
gboolean
|
|
meta_output_is_primary (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
return priv->is_primary;
|
|
}
|
|
|
|
gboolean
|
|
meta_output_is_presentation (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
return priv->is_presentation;
|
|
}
|
|
|
|
gboolean
|
|
meta_output_is_underscanning (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
return priv->is_underscanning;
|
|
}
|
|
|
|
gboolean
|
|
meta_output_get_max_bpc (MetaOutput *output,
|
|
unsigned int *max_bpc)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
if (priv->has_max_bpc && max_bpc)
|
|
*max_bpc = priv->max_bpc;
|
|
|
|
return priv->has_max_bpc;
|
|
}
|
|
|
|
void
|
|
meta_output_set_backlight (MetaOutput *output,
|
|
int backlight)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
priv->backlight = backlight;
|
|
}
|
|
|
|
int
|
|
meta_output_get_backlight (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
return priv->backlight;
|
|
}
|
|
|
|
void
|
|
meta_output_add_possible_clone (MetaOutput *output,
|
|
MetaOutput *possible_clone)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
MetaOutputInfo *output_info = priv->info;
|
|
|
|
output_info->n_possible_clones++;
|
|
output_info->possible_clones = g_renew (MetaOutput *,
|
|
output_info->possible_clones,
|
|
output_info->n_possible_clones);
|
|
output_info->possible_clones[output_info->n_possible_clones - 1] =
|
|
possible_clone;
|
|
}
|
|
|
|
const MetaOutputInfo *
|
|
meta_output_get_info (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
return priv->info;
|
|
}
|
|
|
|
void
|
|
meta_output_assign_crtc (MetaOutput *output,
|
|
MetaCrtc *crtc,
|
|
const MetaOutputAssignment *output_assignment)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
g_assert (crtc);
|
|
|
|
meta_output_unassign_crtc (output);
|
|
|
|
g_set_object (&priv->crtc, crtc);
|
|
|
|
meta_crtc_assign_output (crtc, output);
|
|
|
|
priv->is_primary = output_assignment->is_primary;
|
|
priv->is_presentation = output_assignment->is_presentation;
|
|
priv->is_underscanning = output_assignment->is_underscanning;
|
|
|
|
priv->has_max_bpc = output_assignment->has_max_bpc;
|
|
if (priv->has_max_bpc)
|
|
priv->max_bpc = output_assignment->max_bpc;
|
|
}
|
|
|
|
void
|
|
meta_output_unassign_crtc (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
if (priv->crtc)
|
|
{
|
|
meta_crtc_unassign_output (priv->crtc, output);
|
|
g_clear_object (&priv->crtc);
|
|
}
|
|
|
|
priv->is_primary = FALSE;
|
|
priv->is_presentation = FALSE;
|
|
}
|
|
|
|
MetaCrtc *
|
|
meta_output_get_assigned_crtc (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
return priv->crtc;
|
|
}
|
|
|
|
MetaMonitorTransform
|
|
meta_output_logical_to_crtc_transform (MetaOutput *output,
|
|
MetaMonitorTransform transform)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
MetaMonitorTransform panel_orientation_transform;
|
|
|
|
panel_orientation_transform = priv->info->panel_orientation_transform;
|
|
return meta_monitor_transform_transform (transform,
|
|
panel_orientation_transform);
|
|
}
|
|
|
|
MetaMonitorTransform
|
|
meta_output_crtc_to_logical_transform (MetaOutput *output,
|
|
MetaMonitorTransform transform)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
MetaMonitorTransform inverted_panel_orientation_transform;
|
|
|
|
inverted_panel_orientation_transform =
|
|
meta_monitor_transform_invert (priv->info->panel_orientation_transform);
|
|
return meta_monitor_transform_transform (transform,
|
|
inverted_panel_orientation_transform);
|
|
}
|
|
|
|
static void
|
|
set_output_details_from_edid (MetaOutputInfo *output_info,
|
|
MetaEdidInfo *edid_info)
|
|
{
|
|
output_info->vendor = g_strndup (edid_info->manufacturer_code, 4);
|
|
if (!g_utf8_validate (output_info->vendor, -1, NULL))
|
|
g_clear_pointer (&output_info->vendor, g_free);
|
|
|
|
output_info->product = g_strndup (edid_info->dsc_product_name, 14);
|
|
if (!g_utf8_validate (output_info->product, -1, NULL) ||
|
|
output_info->product[0] == '\0')
|
|
{
|
|
g_clear_pointer (&output_info->product, g_free);
|
|
output_info->product =
|
|
g_strdup_printf ("0x%04x", (unsigned) edid_info->product_code);
|
|
}
|
|
|
|
output_info->serial = g_strndup (edid_info->dsc_serial_number, 14);
|
|
if (!g_utf8_validate (output_info->serial, -1, NULL) ||
|
|
output_info->serial[0] == '\0')
|
|
{
|
|
g_clear_pointer (&output_info->serial, g_free);
|
|
output_info->serial =
|
|
g_strdup_printf ("0x%08x", edid_info->serial_number);
|
|
}
|
|
}
|
|
|
|
void
|
|
meta_output_info_parse_edid (MetaOutputInfo *output_info,
|
|
GBytes *edid)
|
|
{
|
|
MetaEdidInfo *edid_info;
|
|
size_t len;
|
|
gconstpointer data;
|
|
|
|
g_return_if_fail (!output_info->edid_info);
|
|
g_return_if_fail (edid);
|
|
|
|
data = g_bytes_get_data (edid, &len);
|
|
edid_info = meta_edid_info_new_parse (data);
|
|
|
|
output_info->edid_checksum_md5 = g_compute_checksum_for_data (G_CHECKSUM_MD5,
|
|
data, len);
|
|
|
|
if (edid_info)
|
|
{
|
|
output_info->edid_info = edid_info;
|
|
set_output_details_from_edid (output_info, edid_info);
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
meta_output_is_laptop (MetaOutput *output)
|
|
{
|
|
const MetaOutputInfo *output_info = meta_output_get_info (output);
|
|
|
|
switch (output_info->connector_type)
|
|
{
|
|
case META_CONNECTOR_TYPE_eDP:
|
|
case META_CONNECTOR_TYPE_LVDS:
|
|
case META_CONNECTOR_TYPE_DSI:
|
|
return TRUE;
|
|
default:
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
static void
|
|
meta_output_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
MetaOutput *output = META_OUTPUT (object);
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_ID:
|
|
priv->id = g_value_get_uint64 (value);
|
|
break;
|
|
case PROP_GPU:
|
|
priv->gpu = g_value_get_object (value);
|
|
break;
|
|
case PROP_INFO:
|
|
priv->info = meta_output_info_ref (g_value_get_boxed (value));
|
|
break;
|
|
case PROP_IS_PRIVACY_SCREEN_ENABLED:
|
|
priv->is_privacy_screen_enabled = g_value_get_boolean (value);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
}
|
|
}
|
|
|
|
static void
|
|
meta_output_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
MetaOutput *output = META_OUTPUT (object);
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_ID:
|
|
g_value_set_uint64 (value, priv->id);
|
|
break;
|
|
case PROP_GPU:
|
|
g_value_set_object (value, priv->gpu);
|
|
break;
|
|
case PROP_INFO:
|
|
g_value_set_boxed (value, priv->info);
|
|
break;
|
|
case PROP_IS_PRIVACY_SCREEN_ENABLED:
|
|
g_value_set_boolean (value, priv->is_privacy_screen_enabled);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
}
|
|
}
|
|
|
|
static void
|
|
meta_output_dispose (GObject *object)
|
|
{
|
|
MetaOutput *output = META_OUTPUT (object);
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
g_clear_object (&priv->crtc);
|
|
|
|
G_OBJECT_CLASS (meta_output_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
meta_output_finalize (GObject *object)
|
|
{
|
|
MetaOutput *output = META_OUTPUT (object);
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
g_clear_pointer (&priv->info, meta_output_info_unref);
|
|
|
|
G_OBJECT_CLASS (meta_output_parent_class)->finalize (object);
|
|
}
|
|
|
|
MetaPrivacyScreenState
|
|
meta_output_get_privacy_screen_state (MetaOutput *output)
|
|
{
|
|
MetaOutputClass *output_class = META_OUTPUT_GET_CLASS (output);
|
|
|
|
if (!output_class->get_privacy_screen_state)
|
|
return META_PRIVACY_SCREEN_UNAVAILABLE;
|
|
|
|
return output_class->get_privacy_screen_state (output);
|
|
}
|
|
|
|
gboolean
|
|
meta_output_is_privacy_screen_supported (MetaOutput *output)
|
|
{
|
|
return !(meta_output_get_privacy_screen_state (output) ==
|
|
META_PRIVACY_SCREEN_UNAVAILABLE);
|
|
}
|
|
|
|
gboolean
|
|
meta_output_is_privacy_screen_enabled (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
return priv->privacy_screen_state;
|
|
}
|
|
|
|
gboolean
|
|
meta_output_set_privacy_screen_enabled (MetaOutput *output,
|
|
gboolean enabled,
|
|
GError **error)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
MetaPrivacyScreenState state;
|
|
|
|
state = priv->privacy_screen_state;
|
|
|
|
if (state == META_PRIVACY_SCREEN_UNAVAILABLE)
|
|
{
|
|
g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED,
|
|
"The privacy screen is not supported by this output");
|
|
return FALSE;
|
|
}
|
|
|
|
if (state & META_PRIVACY_SCREEN_LOCKED)
|
|
{
|
|
g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PERMISSION_DENIED,
|
|
"The privacy screen is locked at hardware level, "
|
|
"impossible to set it");
|
|
return FALSE;
|
|
}
|
|
|
|
if (priv->is_privacy_screen_enabled == enabled)
|
|
return TRUE;
|
|
|
|
priv->is_privacy_screen_enabled = enabled;
|
|
g_object_notify_by_pspec (G_OBJECT (output),
|
|
obj_props[PROP_IS_PRIVACY_SCREEN_ENABLED]);
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
meta_output_is_color_space_supported (MetaOutput *output,
|
|
MetaOutputColorspace color_space)
|
|
{
|
|
MetaOutputClass *output_class = META_OUTPUT_GET_CLASS (output);
|
|
uint64_t supported = 0;
|
|
|
|
if (output_class->get_supported_color_spaces)
|
|
supported = output_class->get_supported_color_spaces (output);
|
|
|
|
return supported & (1 << color_space);
|
|
}
|
|
|
|
void
|
|
meta_output_set_color_space (MetaOutput *output,
|
|
MetaOutputColorspace color_space)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
priv->color_space = color_space;
|
|
|
|
g_signal_emit (output, signals[COLOR_SPACE_CHANGED], 0);
|
|
}
|
|
|
|
MetaOutputColorspace
|
|
meta_output_peek_color_space (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
return priv->color_space;
|
|
}
|
|
|
|
gboolean
|
|
meta_output_is_hdr_metadata_supported (MetaOutput *output)
|
|
{
|
|
MetaOutputClass *output_class = META_OUTPUT_GET_CLASS (output);
|
|
|
|
if (!output_class->is_hdr_metadata_supported)
|
|
return FALSE;
|
|
|
|
return output_class->is_hdr_metadata_supported (output);
|
|
}
|
|
|
|
void
|
|
meta_output_set_hdr_metadata (MetaOutput *output,
|
|
MetaOutputHdrMetadata *metadata)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
priv->hdr_metadata = *metadata;
|
|
|
|
g_signal_emit (output, signals[HDR_METADATA_CHANGED], 0);
|
|
}
|
|
|
|
MetaOutputHdrMetadata *
|
|
meta_output_peek_hdr_metadata (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
return &priv->hdr_metadata;
|
|
}
|
|
|
|
static void
|
|
meta_output_init (MetaOutput *output)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
|
|
priv->backlight = -1;
|
|
priv->color_space = META_OUTPUT_COLORSPACE_UNKNOWN;
|
|
priv->hdr_metadata.active = FALSE;
|
|
}
|
|
|
|
static void
|
|
meta_output_class_init (MetaOutputClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->set_property = meta_output_set_property;
|
|
object_class->get_property = meta_output_get_property;
|
|
object_class->dispose = meta_output_dispose;
|
|
object_class->finalize = meta_output_finalize;
|
|
|
|
obj_props[PROP_ID] =
|
|
g_param_spec_uint64 ("id",
|
|
"id",
|
|
"CRTC id",
|
|
0, UINT64_MAX, 0,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY |
|
|
G_PARAM_STATIC_STRINGS);
|
|
obj_props[PROP_GPU] =
|
|
g_param_spec_object ("gpu",
|
|
"gpu",
|
|
"MetaGpu",
|
|
META_TYPE_GPU,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY |
|
|
G_PARAM_STATIC_STRINGS);
|
|
obj_props[PROP_INFO] =
|
|
g_param_spec_boxed ("info",
|
|
"info",
|
|
"MetaOutputInfo",
|
|
META_TYPE_OUTPUT_INFO,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY |
|
|
G_PARAM_STATIC_STRINGS);
|
|
obj_props[PROP_IS_PRIVACY_SCREEN_ENABLED] =
|
|
g_param_spec_boolean ("is-privacy-screen-enabled",
|
|
"is-privacy-screen-enabled",
|
|
"Is privacy screen enabled",
|
|
FALSE,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_STATIC_STRINGS);
|
|
g_object_class_install_properties (object_class, N_PROPS, obj_props);
|
|
|
|
signals[COLOR_SPACE_CHANGED] =
|
|
g_signal_new ("color-space-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
0,
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
signals[HDR_METADATA_CHANGED] =
|
|
g_signal_new ("hdr-metadata-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
0,
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
}
|
|
|
|
gboolean
|
|
meta_tile_info_equal (MetaTileInfo *a,
|
|
MetaTileInfo *b)
|
|
{
|
|
if (a == b)
|
|
return TRUE;
|
|
|
|
if (a->group_id != b->group_id)
|
|
return FALSE;
|
|
|
|
if (a->flags != b->flags)
|
|
return FALSE;
|
|
|
|
if (a->max_h_tiles != b->max_h_tiles)
|
|
return FALSE;
|
|
|
|
if (a->max_v_tiles != b->max_v_tiles)
|
|
return FALSE;
|
|
|
|
if (a->loc_h_tile != b->loc_h_tile)
|
|
return FALSE;
|
|
|
|
if (a->loc_v_tile != b->loc_v_tile)
|
|
return FALSE;
|
|
|
|
if (a->tile_w != b->tile_w)
|
|
return FALSE;
|
|
|
|
if (a->tile_h != b->tile_h)
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
meta_output_update_modes (MetaOutput *output,
|
|
MetaCrtcMode *preferred_mode,
|
|
MetaCrtcMode **modes,
|
|
int n_modes)
|
|
{
|
|
MetaOutputPrivate *priv = meta_output_get_instance_private (output);
|
|
int i;
|
|
|
|
for (i = 0; i < priv->info->n_modes; i++)
|
|
g_object_unref (priv->info->modes[i]);
|
|
g_free (priv->info->modes);
|
|
|
|
priv->info->preferred_mode = preferred_mode;
|
|
priv->info->modes = modes;
|
|
priv->info->n_modes = n_modes;
|
|
}
|