/* -*- 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 . */ /* * Portions of this file are derived from gnome-desktop/libgnome-desktop/gnome-rr-config.c * * Copyright 2007, 2008, Red Hat, Inc. * Copyright 2010 Giovanni Campagna * * Author: Soren Sandmann */ #include "config.h" #include "meta-monitor-config.h" #include #include #include #include #include /* These structures represent the intended/persistent configuration, as stored in the monitors.xml file. */ typedef struct { char *connector; char *vendor; char *product; char *serial; } MetaOutputKey; /* Keep this structure packed, so that we can use memcmp */ typedef struct { gboolean enabled; MetaRectangle rect; float refresh_rate; enum wl_output_transform transform; gboolean is_primary; gboolean is_presentation; } MetaOutputConfig; typedef struct { MetaOutputKey *keys; MetaOutputConfig *outputs; unsigned int n_outputs; } MetaConfiguration; struct _MetaMonitorConfig { GObject parent_instance; GHashTable *configs; MetaConfiguration *current; gboolean current_is_stored; MetaConfiguration *previous; GFile *file; GCancellable *save_cancellable; UpClient *up_client; gboolean lid_is_closed; }; struct _MetaMonitorConfigClass { GObjectClass parent; }; G_DEFINE_TYPE (MetaMonitorConfig, meta_monitor_config, G_TYPE_OBJECT); static gboolean meta_monitor_config_assign_crtcs (MetaConfiguration *config, MetaMonitorManager *manager, GPtrArray *crtcs, GPtrArray *outputs); static void power_client_changed_cb (UpClient *client, GParamSpec *pspec, gpointer user_data); static void free_output_key (MetaOutputKey *key) { g_free (key->connector); g_free (key->vendor); g_free (key->product); g_free (key->serial); } static void config_clear (MetaConfiguration *config) { unsigned int i; for (i = 0; i < config->n_outputs; i++) free_output_key (&config->keys[i]); g_free (config->keys); g_free (config->outputs); } static void config_free (gpointer config) { config_clear (config); g_slice_free (MetaConfiguration, config); } static unsigned long output_key_hash (const MetaOutputKey *key) { return g_str_hash (key->connector) ^ g_str_hash (key->vendor) ^ g_str_hash (key->product) ^ g_str_hash (key->serial); } static gboolean output_key_equal (const MetaOutputKey *one, const MetaOutputKey *two) { return strcmp (one->connector, two->connector) == 0 && strcmp (one->vendor, two->vendor) == 0 && strcmp (one->product, two->product) == 0 && strcmp (one->serial, two->serial) == 0; } static gboolean output_config_equal (const MetaOutputConfig *one, const MetaOutputConfig *two) { return memcmp (one, two, sizeof (MetaOutputConfig)) == 0; } static unsigned int config_hash (gconstpointer data) { const MetaConfiguration *config = data; unsigned int i, hash; hash = 0; for (i = 0; i < config->n_outputs; i++) hash ^= output_key_hash (&config->keys[i]); return hash; } static gboolean config_equal (gconstpointer one, gconstpointer two) { const MetaConfiguration *c_one = one; const MetaConfiguration *c_two = two; unsigned int i; gboolean ok; if (c_one->n_outputs != c_two->n_outputs) return FALSE; ok = TRUE; for (i = 0; i < c_one->n_outputs && ok; i++) ok = output_key_equal (&c_one->keys[i], &c_two->keys[i]); return ok; } static gboolean config_equal_full (gconstpointer one, gconstpointer two) { const MetaConfiguration *c_one = one; const MetaConfiguration *c_two = two; unsigned int i; gboolean ok; if (c_one->n_outputs != c_two->n_outputs) return FALSE; ok = TRUE; for (i = 0; i < c_one->n_outputs && ok; i++) { ok = output_key_equal (&c_one->keys[i], &c_two->keys[i]); ok = ok && output_config_equal (&c_one->outputs[i], &c_two->outputs[i]); } return ok; } static void meta_monitor_config_init (MetaMonitorConfig *self) { const char *filename; char *path; self->configs = g_hash_table_new_full (config_hash, config_equal, NULL, config_free); filename = g_getenv ("MUTTER_MONITOR_FILENAME"); if (filename == NULL) filename = "monitors.xml"; path = g_build_filename (g_get_user_config_dir (), filename, NULL); self->file = g_file_new_for_path (path); g_free (path); self->up_client = up_client_new (); self->lid_is_closed = up_client_get_lid_is_closed (self->up_client); g_signal_connect_object (self->up_client, "notify::lid-is-closed", G_CALLBACK (power_client_changed_cb), self, 0); } static void meta_monitor_config_finalize (GObject *object) { MetaMonitorConfig *self = META_MONITOR_CONFIG (object); g_hash_table_destroy (self->configs); } static void meta_monitor_config_class_init (MetaMonitorConfigClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->finalize = meta_monitor_config_finalize; } typedef enum { STATE_INITIAL, STATE_MONITORS, STATE_CONFIGURATION, STATE_OUTPUT, STATE_OUTPUT_FIELD, STATE_CLONE } ParserState; typedef struct { MetaMonitorConfig *config; ParserState state; int unknown_count; GArray *key_array; GArray *output_array; MetaOutputKey key; MetaOutputConfig output; char *output_field; } ConfigParser; static void handle_start_element (GMarkupParseContext *context, const char *element_name, const char **attribute_names, const char **attribute_values, gpointer user_data, GError **error) { ConfigParser *parser = user_data; switch (parser->state) { case STATE_INITIAL: { char *version; if (strcmp (element_name, "monitors") != 0) { g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_UNKNOWN_ELEMENT, "Invalid document element %s", element_name); return; } if (!g_markup_collect_attributes (element_name, attribute_names, attribute_values, error, G_MARKUP_COLLECT_STRING, "version", &version, G_MARKUP_COLLECT_INVALID)) return; if (strcmp (version, "1") != 0) { g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, "Invalid or unsupported version %s", version); return; } parser->state = STATE_MONITORS; return; } case STATE_MONITORS: { if (strcmp (element_name, "configuration") != 0) { g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_UNKNOWN_ELEMENT, "Invalid toplevel element %s", element_name); return; } parser->key_array = g_array_new (FALSE, FALSE, sizeof (MetaOutputKey)); parser->output_array = g_array_new (FALSE, FALSE, sizeof (MetaOutputConfig)); parser->state = STATE_CONFIGURATION; return; } case STATE_CONFIGURATION: { if (strcmp (element_name, "clone") == 0 && parser->unknown_count == 0) { parser->state = STATE_CLONE; } else if (strcmp (element_name, "output") == 0 && parser->unknown_count == 0) { char *name; if (!g_markup_collect_attributes (element_name, attribute_names, attribute_values, error, G_MARKUP_COLLECT_STRING, "name", &name, G_MARKUP_COLLECT_INVALID)) return; memset (&parser->key, 0, sizeof (MetaOutputKey)); memset (&parser->output, 0, sizeof (MetaOutputConfig)); parser->key.connector = g_strdup (name); parser->state = STATE_OUTPUT; } else { parser->unknown_count++; } return; } case STATE_OUTPUT: { if ((strcmp (element_name, "vendor") == 0 || strcmp (element_name, "product") == 0 || strcmp (element_name, "serial") == 0 || strcmp (element_name, "width") == 0 || strcmp (element_name, "height") == 0 || strcmp (element_name, "rate") == 0 || strcmp (element_name, "x") == 0 || strcmp (element_name, "y") == 0 || strcmp (element_name, "rotation") == 0 || strcmp (element_name, "reflect_x") == 0 || strcmp (element_name, "reflect_y") == 0 || strcmp (element_name, "primary") == 0 || strcmp (element_name, "presentation") == 0) && parser->unknown_count == 0) { parser->state = STATE_OUTPUT_FIELD; parser->output_field = g_strdup (element_name); } else { parser->unknown_count++; } return; } case STATE_CLONE: case STATE_OUTPUT_FIELD: { g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, "Unexpected element %s", element_name); return; } default: g_assert_not_reached (); } } static void handle_end_element (GMarkupParseContext *context, const char *element_name, gpointer user_data, GError **error) { ConfigParser *parser = user_data; switch (parser->state) { case STATE_MONITORS: { parser->state = STATE_INITIAL; return; } case STATE_CONFIGURATION: { if (strcmp (element_name, "configuration") == 0 && parser->unknown_count == 0) { MetaConfiguration *config = g_slice_new (MetaConfiguration); g_assert (parser->key_array->len == parser->output_array->len); config->n_outputs = parser->key_array->len; config->keys = (void*)g_array_free (parser->key_array, FALSE); config->outputs = (void*)g_array_free (parser->output_array, FALSE); g_hash_table_replace (parser->config->configs, config, config); parser->key_array = NULL; parser->output_array = NULL; parser->state = STATE_MONITORS; } else { parser->unknown_count--; g_assert (parser->unknown_count >= 0); } return; } case STATE_OUTPUT: { if (strcmp (element_name, "output") == 0 && parser->unknown_count == 0) { if (parser->key.vendor == NULL || parser->key.product == NULL || parser->key.serial == NULL) { /* Disconnected output, ignore */ free_output_key (&parser->key); } else { if (parser->output.rect.width == 0 && parser->output.rect.width == 0) parser->output.enabled = FALSE; else parser->output.enabled = TRUE; g_array_append_val (parser->key_array, parser->key); g_array_append_val (parser->output_array, parser->output); } memset (&parser->key, 0, sizeof (MetaOutputKey)); memset (&parser->output, 0, sizeof (MetaOutputConfig)); parser->state = STATE_CONFIGURATION; } else { parser->unknown_count--; g_assert (parser->unknown_count >= 0); } return; } case STATE_CLONE: { parser->state = STATE_CONFIGURATION; return; } case STATE_OUTPUT_FIELD: { g_free (parser->output_field); parser->output_field = NULL; parser->state = STATE_OUTPUT; return; } case STATE_INITIAL: default: g_assert_not_reached (); } } static void read_int (const char *text, gsize text_len, gint *field, GError **error) { char buf[64]; gint64 v; char *end; strncpy (buf, text, text_len); buf[MIN (63, text_len)] = 0; v = g_ascii_strtoll (buf, &end, 10); /* Limit reasonable values (actual limits are a lot smaller that these) */ if (*end || v < 0 || v > G_MAXINT16) g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, "Expected a number, got %s", buf); else *field = v; } static void read_float (const char *text, gsize text_len, gfloat *field, GError **error) { char buf[64]; gfloat v; char *end; strncpy (buf, text, text_len); buf[MIN (63, text_len)] = 0; v = g_ascii_strtod (buf, &end); /* Limit reasonable values (actual limits are a lot smaller that these) */ if (*end) g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, "Expected a number, got %s", buf); else *field = v; } static gboolean read_bool (const char *text, gsize text_len, GError **error) { if (strncmp (text, "no", text_len) == 0) return FALSE; else if (strncmp (text, "yes", text_len) == 0) return TRUE; else g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, "Invalid boolean value %.*s", (int)text_len, text); return FALSE; } static gboolean is_all_whitespace (const char *text, gsize text_len) { gsize i; for (i = 0; i < text_len; i++) if (!g_ascii_isspace (text[i])) return FALSE; return TRUE; } static void handle_text (GMarkupParseContext *context, const gchar *text, gsize text_len, gpointer user_data, GError **error) { ConfigParser *parser = user_data; switch (parser->state) { case STATE_MONITORS: { if (!is_all_whitespace (text, text_len)) g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, "Unexpected content at this point"); return; } case STATE_CONFIGURATION: { if (parser->unknown_count == 0) { if (!is_all_whitespace (text, text_len)) g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, "Unexpected content at this point"); } else { /* Handling unknown element, ignore */ } return; } case STATE_OUTPUT: { if (parser->unknown_count == 0) { if (!is_all_whitespace (text, text_len)) g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, "Unexpected content at this point"); } else { /* Handling unknown element, ignore */ } return; } case STATE_CLONE: { /* Ignore the clone flag */ return; } case STATE_OUTPUT_FIELD: { if (strcmp (parser->output_field, "vendor") == 0) parser->key.vendor = g_strndup (text, text_len); else if (strcmp (parser->output_field, "product") == 0) parser->key.product = g_strndup (text, text_len); else if (strcmp (parser->output_field, "serial") == 0) parser->key.serial = g_strndup (text, text_len); else if (strcmp (parser->output_field, "width") == 0) read_int (text, text_len, &parser->output.rect.width, error); else if (strcmp (parser->output_field, "height") == 0) read_int (text, text_len, &parser->output.rect.height, error); else if (strcmp (parser->output_field, "rate") == 0) read_float (text, text_len, &parser->output.refresh_rate, error); else if (strcmp (parser->output_field, "x") == 0) read_int (text, text_len, &parser->output.rect.x, error); else if (strcmp (parser->output_field, "y") == 0) read_int (text, text_len, &parser->output.rect.y, error); else if (strcmp (parser->output_field, "rotation") == 0) { if (strncmp (text, "normal", text_len) == 0) parser->output.transform = WL_OUTPUT_TRANSFORM_NORMAL; else if (strncmp (text, "left", text_len) == 0) parser->output.transform = WL_OUTPUT_TRANSFORM_90; else if (strncmp (text, "upside_down", text_len) == 0) parser->output.transform = WL_OUTPUT_TRANSFORM_180; else if (strncmp (text, "right", text_len) == 0) parser->output.transform = WL_OUTPUT_TRANSFORM_270; else g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, "Invalid rotation type %.*s", (int)text_len, text); } else if (strcmp (parser->output_field, "reflect_x") == 0) parser->output.transform += read_bool (text, text_len, error) ? WL_OUTPUT_TRANSFORM_FLIPPED : 0; else if (strcmp (parser->output_field, "reflect_y") == 0) { /* FIXME (look at the rotation map in monitor.c) */ if (read_bool (text, text_len, error)) g_set_error (error, G_MARKUP_ERROR, G_MARKUP_ERROR_INVALID_CONTENT, "Y reflection is not supported"); } else if (strcmp (parser->output_field, "primary") == 0) parser->output.is_primary = read_bool (text, text_len, error); else if (strcmp (parser->output_field, "presentation") == 0) parser->output.is_presentation = read_bool (text, text_len, error); else g_assert_not_reached (); return; } case STATE_INITIAL: default: g_assert_not_reached (); } } static const GMarkupParser config_parser = { .start_element = handle_start_element, .end_element = handle_end_element, .text = handle_text, }; static void meta_monitor_config_load (MetaMonitorConfig *self) { char *contents; gsize size; gboolean ok; GError *error; GMarkupParseContext *context; ConfigParser parser; /* Note: we're explicitly loading this file synchronously because we don't want to leave the default configuration on for even a frame, ie we want atomic modeset as much as possible. This function is called only at early initialization anyway, before we connect to X or create the wayland socket. */ error = NULL; ok = g_file_load_contents (self->file, NULL, &contents, &size, NULL, &error); if (!ok) { if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND)) meta_warning ("Failed to load stored monitor configuration: %s\n", error->message); g_error_free (error); return; } memset (&parser, 0, sizeof (ConfigParser)); parser.config = self; parser.state = STATE_INITIAL; context = g_markup_parse_context_new (&config_parser, G_MARKUP_TREAT_CDATA_AS_TEXT | G_MARKUP_PREFIX_ERROR_POSITION, &parser, NULL); ok = g_markup_parse_context_parse (context, contents, size, &error); if (!ok) { meta_warning ("Failed to parse stored monitor configuration: %s\n", error->message); g_error_free (error); if (parser.key_array) g_array_free (parser.key_array, TRUE); if (parser.output_array) g_array_free (parser.output_array, TRUE); free_output_key (&parser.key); } g_markup_parse_context_free (context); g_free (contents); } MetaMonitorConfig * meta_monitor_config_new (void) { MetaMonitorConfig *self; self = g_object_new (META_TYPE_MONITOR_CONFIG, NULL); meta_monitor_config_load (self); return self; } static void init_key_from_output (MetaOutputKey *key, MetaOutput *output) { key->connector = g_strdup (output->name); key->product = g_strdup (output->product); key->vendor = g_strdup (output->vendor); key->serial = g_strdup (output->serial); } static void make_config_key (MetaConfiguration *key, MetaOutput *outputs, unsigned n_outputs, unsigned skip) { unsigned int o, i; key->outputs = NULL; key->keys = g_new0 (MetaOutputKey, n_outputs); for (o = 0, i = 0; i < n_outputs; o++, i++) if (i == skip) o--; else init_key_from_output (&key->keys[o], &outputs[i]); key->n_outputs = o; } gboolean meta_monitor_config_match_current (MetaMonitorConfig *self, MetaMonitorManager *manager) { MetaOutput *outputs; unsigned n_outputs; MetaConfiguration key; gboolean ok; if (self->current == NULL) return FALSE; outputs = meta_monitor_manager_get_outputs (manager, &n_outputs); make_config_key (&key, outputs, n_outputs, -1); ok = config_equal (&key, self->current); config_clear (&key); return ok; } gboolean meta_monitor_manager_has_hotplug_mode_update (MetaMonitorManager *manager) { MetaOutput *outputs; unsigned n_outputs; unsigned int i; outputs = meta_monitor_manager_get_outputs (manager, &n_outputs); for (i = 0; i < n_outputs; i++) if (outputs[i].hotplug_mode_update) return TRUE; return FALSE; } static MetaConfiguration * meta_monitor_config_get_stored (MetaMonitorConfig *self, MetaOutput *outputs, unsigned n_outputs) { MetaConfiguration key; MetaConfiguration *stored; if (n_outputs == 0) return NULL; make_config_key (&key, outputs, n_outputs, -1); stored = g_hash_table_lookup (self->configs, &key); config_clear (&key); return stored; } static gboolean apply_configuration (MetaMonitorConfig *self, MetaConfiguration *config, MetaMonitorManager *manager, gboolean stored) { GPtrArray *crtcs, *outputs; crtcs = g_ptr_array_new_full (config->n_outputs, (GDestroyNotify)meta_crtc_info_free); outputs = g_ptr_array_new_full (config->n_outputs, (GDestroyNotify)meta_output_info_free); if (!meta_monitor_config_assign_crtcs (config, manager, crtcs, outputs)) { g_ptr_array_unref (crtcs); g_ptr_array_unref (outputs); if (!stored) config_free (config); return FALSE; } meta_monitor_manager_apply_configuration (manager, (MetaCRTCInfo**)crtcs->pdata, crtcs->len, (MetaOutputInfo**)outputs->pdata, outputs->len); /* Stored (persistent) configurations override the previous one always. Also, we clear the previous configuration if the current one (which is about to become previous) is stored, or if the current one has different outputs. */ if (stored || (self->current && self->current_is_stored)) { if (self->previous) config_free (self->previous); self->previous = NULL; } else { /* Despite the name, config_equal() only checks the set of outputs, not their modes */ if (self->current && config_equal (self->current, config)) { self->previous = self->current; } else { if (self->current) config_free (self->current); self->previous = NULL; } } self->current = config; self->current_is_stored = stored; if (self->current == self->previous) self->previous = NULL; g_ptr_array_unref (crtcs); g_ptr_array_unref (outputs); return TRUE; } static gboolean key_is_laptop (MetaOutputKey *key) { /* FIXME: extend with better heuristics */ return g_str_has_prefix (key->connector, "LVDS") || g_str_has_prefix (key->connector, "eDP"); } static gboolean laptop_display_is_on (MetaConfiguration *config) { unsigned int i; for (i = 0; i < config->n_outputs; i++) { MetaOutputKey *key = &config->keys[i]; MetaOutputConfig *output = &config->outputs[i]; if (key_is_laptop (key) && output->enabled) return TRUE; } return FALSE; } static MetaConfiguration * make_laptop_lid_config (MetaConfiguration *reference) { MetaConfiguration *new; unsigned int i; gboolean has_primary; int x_after, y_after; int x_offset, y_offset; g_assert (reference->n_outputs > 1); new = g_slice_new0 (MetaConfiguration); new->n_outputs = reference->n_outputs; new->keys = g_new0 (MetaOutputKey, reference->n_outputs); new->outputs = g_new0 (MetaOutputConfig, reference->n_outputs); x_after = G_MAXINT; y_after = G_MAXINT; x_offset = 0; y_offset = 0; for (i = 0; i < new->n_outputs; i++) { MetaOutputKey *current_key = &reference->keys[i]; MetaOutputConfig *current_output = &reference->outputs[i]; new->keys[i].connector = g_strdup (current_key->connector); new->keys[i].vendor = g_strdup (current_key->vendor); new->keys[i].product = g_strdup (current_key->product); new->keys[i].serial = g_strdup (current_key->serial); if (g_str_has_prefix (current_key->connector, "LVDS") || g_str_has_prefix (current_key->connector, "eDP")) { new->outputs[i].enabled = FALSE; x_after = current_output->rect.x; y_after = current_output->rect.y; x_offset = current_output->rect.width; y_offset = current_output->rect.height; } else new->outputs[i] = *current_output; } for (i = 0; i < new->n_outputs; i++) { if (new->outputs[i].enabled) { if (new->outputs[i].rect.x > x_after) new->outputs[i].rect.x -= x_offset; if (new->outputs[i].rect.y > y_after) new->outputs[i].rect.y -= y_offset; } } has_primary = FALSE; for (i = 0; i < new->n_outputs; i++) { if (new->outputs[i].is_primary) { has_primary = TRUE; break; } } if (!has_primary) new->outputs[0].is_primary = TRUE; return new; } gboolean meta_monitor_config_apply_stored (MetaMonitorConfig *self, MetaMonitorManager *manager) { MetaOutput *outputs; MetaConfiguration *stored; unsigned n_outputs; outputs = meta_monitor_manager_get_outputs (manager, &n_outputs); stored = meta_monitor_config_get_stored (self, outputs, n_outputs); if (stored) { if (self->lid_is_closed && stored->n_outputs > 1 && laptop_display_is_on (stored)) return apply_configuration (self, make_laptop_lid_config (stored), manager, FALSE); else return apply_configuration (self, stored, manager, TRUE); } else return FALSE; } /* * Tries to find the primary output according to the current layout, * or failing that, an output that is good to be a primary (LVDS or eDP, * which are internal monitors), or failing that, the one with the * best resolution */ static MetaOutput * find_primary_output (MetaOutput *outputs, unsigned n_outputs) { unsigned i; MetaOutput *best; int best_width, best_height; g_assert (n_outputs >= 1); for (i = 0; i < n_outputs; i++) { if (outputs[i].is_primary) return &outputs[i]; } for (i = 0; i < n_outputs; i++) { if (g_str_has_prefix (outputs[i].name, "LVDS") || g_str_has_prefix (outputs[i].name, "eDP")) return &outputs[i]; } best = NULL; best_width = 0; best_height = 0; for (i = 0; i < n_outputs; i++) { if (outputs[i].preferred_mode->width * outputs[i].preferred_mode->height > best_width * best_height) { best = &outputs[i]; best_width = outputs[i].preferred_mode->width; best_height = outputs[i].preferred_mode->height; } } return best; } static MetaConfiguration * make_default_config (MetaMonitorConfig *self, MetaOutput *outputs, unsigned n_outputs, int max_width, int max_height) { unsigned i, j; int x, y; MetaConfiguration *ret; MetaOutput *primary; ret = g_slice_new (MetaConfiguration); make_config_key (ret, outputs, n_outputs, -1); ret->outputs = g_new0 (MetaOutputConfig, n_outputs); /* Special case the simple case: one output, primary at preferred mode, nothing else to do */ if (n_outputs == 1) { ret->outputs[0].enabled = TRUE; ret->outputs[0].rect.x = 0; ret->outputs[0].rect.y = 0; ret->outputs[0].rect.width = outputs[0].preferred_mode->width; ret->outputs[0].rect.height = outputs[0].preferred_mode->height; ret->outputs[0].refresh_rate = outputs[0].preferred_mode->refresh_rate; ret->outputs[0].transform = WL_OUTPUT_TRANSFORM_NORMAL; ret->outputs[0].is_primary = TRUE; return ret; } /* If we reach this point, this is either the first time mutter runs on this system ever, or we just hotplugged a new screen. In the latter case, search for a configuration that includes one less screen, then add the new one as a presentation screen in preferred mode. XXX: but presentation mode is not implemented in the control-center or in mutter core, so let's do extended for now. */ x = 0; y = 0; for (i = 0; i < n_outputs; i++) { MetaConfiguration key; MetaConfiguration *ref; make_config_key (&key, outputs, n_outputs, i); ref = g_hash_table_lookup (self->configs, &key); config_clear (&key); if (ref) { for (j = 0; j < n_outputs; j++) { if (j < i) { g_assert (output_key_equal (&ret->keys[j], &ref->keys[j])); ret->outputs[j] = ref->outputs[j]; x = MAX (x, ref->outputs[j].rect.x + ref->outputs[j].rect.width); y = MAX (y, ref->outputs[j].rect.y + ref->outputs[j].rect.height); } else if (j > i) { g_assert (output_key_equal (&ret->keys[j], &ref->keys[j - 1])); ret->outputs[j] = ref->outputs[j - 1]; x = MAX (x, ref->outputs[j - 1].rect.x + ref->outputs[j - 1].rect.width); y = MAX (y, ref->outputs[j - 1].rect.y + ref->outputs[j - 1].rect.height); } else { ret->outputs[j].enabled = TRUE; ret->outputs[j].rect.x = 0; ret->outputs[j].rect.y = 0; ret->outputs[j].rect.width = outputs[0].preferred_mode->width; ret->outputs[j].rect.height = outputs[0].preferred_mode->height; ret->outputs[j].refresh_rate = outputs[0].preferred_mode->refresh_rate; ret->outputs[j].transform = WL_OUTPUT_TRANSFORM_NORMAL; ret->outputs[j].is_primary = FALSE; ret->outputs[j].is_presentation = FALSE; } } /* Place the new output at the right end of the screen, if it fits, otherwise below it, otherwise disable it (or apply_configuration will fail) */ if (x + ret->outputs[i].rect.width <= max_width) ret->outputs[i].rect.x = x; else if (y + ret->outputs[i].rect.height <= max_height) ret->outputs[i].rect.y = y; else ret->outputs[i].enabled = FALSE; return ret; } } /* No previous configuration found, try with a really default one, which is one primary that goes first and the rest to the right of it, extended. */ primary = find_primary_output (outputs, n_outputs); x = primary->preferred_mode->width; for (i = 0; i < n_outputs; i++) { MetaOutput *output = &outputs[i]; ret->outputs[i].enabled = TRUE; ret->outputs[i].rect.x = (output == primary) ? 0 : x; ret->outputs[i].rect.y = 0; ret->outputs[i].rect.width = output->preferred_mode->width; ret->outputs[i].rect.height = output->preferred_mode->height; ret->outputs[i].refresh_rate = output->preferred_mode->refresh_rate; ret->outputs[i].transform = WL_OUTPUT_TRANSFORM_NORMAL; ret->outputs[i].is_primary = (output == primary); /* Disable outputs that would go beyond framebuffer limits */ if (ret->outputs[i].rect.x + ret->outputs[i].rect.width > max_width) ret->outputs[i].enabled = FALSE; else if (output != primary) x += output->preferred_mode->width; } return ret; } static gboolean ensure_at_least_one_output (MetaMonitorConfig *self, MetaMonitorManager *manager, MetaOutput *outputs, unsigned n_outputs) { MetaConfiguration *ret; MetaOutput *primary; unsigned i; /* Check that we have at least one active output */ for (i = 0; i < n_outputs; i++) if (outputs[i].crtc != NULL) return TRUE; /* Oh no, we don't! Activate the primary one and disable everything else */ ret = g_slice_new (MetaConfiguration); make_config_key (ret, outputs, n_outputs, -1); ret->outputs = g_new0 (MetaOutputConfig, n_outputs); primary = find_primary_output (outputs, n_outputs); for (i = 0; i < n_outputs; i++) { MetaOutput *output = &outputs[i]; if (output == primary) { ret->outputs[i].enabled = TRUE; ret->outputs[i].rect.x = 0; ret->outputs[i].rect.y = 0; ret->outputs[i].rect.width = output->preferred_mode->width; ret->outputs[i].rect.height = output->preferred_mode->height; ret->outputs[i].refresh_rate = output->preferred_mode->refresh_rate; ret->outputs[i].transform = WL_OUTPUT_TRANSFORM_NORMAL; ret->outputs[i].is_primary = TRUE; } else { ret->outputs[i].enabled = FALSE; } } apply_configuration (self, ret, manager, FALSE); return FALSE; } void meta_monitor_config_make_default (MetaMonitorConfig *self, MetaMonitorManager *manager) { MetaOutput *outputs; MetaConfiguration *default_config; unsigned n_outputs; gboolean ok; int max_width, max_height; outputs = meta_monitor_manager_get_outputs (manager, &n_outputs); meta_monitor_manager_get_screen_limits (manager, &max_width, &max_height); if (n_outputs == 0) { meta_verbose ("No output connected, not applying configuration\n"); return; } default_config = make_default_config (self, outputs, n_outputs, max_width, max_height); if (default_config != NULL) { if (self->lid_is_closed && default_config->n_outputs > 1 && laptop_display_is_on (default_config)) { ok = apply_configuration (self, make_laptop_lid_config (default_config), manager, FALSE); config_free (default_config); } else ok = apply_configuration (self, default_config, manager, FALSE); } else ok = FALSE; if (!ok) { meta_warning ("Could not make default configuration for current output layout, leaving unconfigured\n"); if (ensure_at_least_one_output (self, manager, outputs, n_outputs)) meta_monitor_config_update_current (self, manager); } } static void init_config_from_output (MetaOutputConfig *config, MetaOutput *output) { config->enabled = (output->crtc != NULL); if (!config->enabled) return; config->rect = output->crtc->rect; config->refresh_rate = output->crtc->current_mode->refresh_rate; config->transform = output->crtc->transform; config->is_primary = output->is_primary; config->is_presentation = output->is_presentation; } void meta_monitor_config_update_current (MetaMonitorConfig *self, MetaMonitorManager *manager) { MetaOutput *outputs; unsigned n_outputs; MetaConfiguration *current; unsigned int i; outputs = meta_monitor_manager_get_outputs (manager, &n_outputs); current = g_slice_new (MetaConfiguration); current->n_outputs = n_outputs; current->outputs = g_new0 (MetaOutputConfig, n_outputs); current->keys = g_new0 (MetaOutputKey, n_outputs); for (i = 0; i < current->n_outputs; i++) { init_key_from_output (¤t->keys[i], &outputs[i]); init_config_from_output (¤t->outputs[i], &outputs[i]); } if (self->current && config_equal_full (current, self->current)) { config_free (current); return; } if (self->current && !self->current_is_stored) config_free (self->current); self->current = current; self->current_is_stored = FALSE; } void meta_monitor_config_restore_previous (MetaMonitorConfig *self, MetaMonitorManager *manager) { if (self->previous) apply_configuration (self, self->previous, manager, FALSE); else { if (!meta_monitor_config_apply_stored (self, manager)) meta_monitor_config_make_default (self, manager); } } static void turn_off_laptop_display (MetaMonitorConfig *self, MetaMonitorManager *manager) { MetaConfiguration *new; if (self->current->n_outputs == 1) return; new = make_laptop_lid_config (self->current); apply_configuration (self, new, manager, FALSE); } static void power_client_changed_cb (UpClient *client, GParamSpec *pspec, gpointer user_data) { MetaMonitorManager *manager = meta_monitor_manager_get (); MetaMonitorConfig *self = user_data; gboolean is_closed; is_closed = up_client_get_lid_is_closed (self->up_client); if (is_closed != self->lid_is_closed) { self->lid_is_closed = is_closed; if (is_closed) turn_off_laptop_display (self, manager); else meta_monitor_config_restore_previous (self, manager); } } typedef struct { MetaMonitorConfig *config; GString *buffer; } SaveClosure; static void saved_cb (GObject *object, GAsyncResult *result, gpointer user_data) { SaveClosure *closure = user_data; GError *error; gboolean ok; error = NULL; ok = g_file_replace_contents_finish (G_FILE (object), result, NULL, &error); if (!ok) { if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) meta_warning ("Saving monitor configuration failed: %s\n", error->message); g_error_free (error); } g_clear_object (&closure->config->save_cancellable); g_object_unref (closure->config); g_string_free (closure->buffer, TRUE); g_slice_free (SaveClosure, closure); } static void meta_monitor_config_save (MetaMonitorConfig *self) { static const char * const rotation_map[4] = { "normal", "left", "upside_down", "right" }; SaveClosure *closure; GString *buffer; GHashTableIter iter; MetaConfiguration *config; unsigned int i; if (self->save_cancellable) { g_cancellable_cancel (self->save_cancellable); g_object_unref (self->save_cancellable); self->save_cancellable = NULL; } self->save_cancellable = g_cancellable_new (); buffer = g_string_new ("\n"); g_hash_table_iter_init (&iter, self->configs); while (g_hash_table_iter_next (&iter, (gpointer*) &config, NULL)) { /* Note: we don't distinguish clone vs non-clone here, that's something for the UI (ie gnome-control-center) to handle, and our configurations are more complex anyway. */ g_string_append (buffer, " \n" " no\n"); for (i = 0; i < config->n_outputs; i++) { MetaOutputKey *key = &config->keys[i]; MetaOutputConfig *output = &config->outputs[i]; g_string_append_printf (buffer, " \n" " %s\n" " %s\n" " %s\n", key->connector, key->vendor, key->product, key->serial); if (output->enabled) { char refresh_rate[G_ASCII_DTOSTR_BUF_SIZE]; g_ascii_dtostr (refresh_rate, sizeof (refresh_rate), output->refresh_rate); g_string_append_printf (buffer, " %d\n" " %d\n" " %s\n" " %d\n" " %d\n" " %s\n" " %s\n" " no\n" " %s\n" " %s\n", output->rect.width, output->rect.height, refresh_rate, output->rect.x, output->rect.y, rotation_map[output->transform & 0x3], output->transform >= WL_OUTPUT_TRANSFORM_FLIPPED ? "yes" : "no", output->is_primary ? "yes" : "no", output->is_presentation ? "yes" : "no"); } g_string_append (buffer, " \n"); } g_string_append (buffer, " \n"); } g_string_append (buffer, "\n"); closure = g_slice_new (SaveClosure); closure->config = g_object_ref (self); closure->buffer = buffer; g_file_replace_contents_async (self->file, buffer->str, buffer->len, NULL, /* etag */ TRUE, G_FILE_CREATE_REPLACE_DESTINATION, self->save_cancellable, saved_cb, closure); } void meta_monitor_config_make_persistent (MetaMonitorConfig *self) { if (self->current_is_stored) return; self->current_is_stored = TRUE; g_hash_table_replace (self->configs, self->current, self->current); if (self->previous) config_free (self->previous); self->previous = NULL; meta_monitor_config_save (self); } /* * CRTC assignment */ typedef struct { MetaConfiguration *config; MetaMonitorManager *manager; GHashTable *info; } CrtcAssignment; static gboolean output_can_clone (MetaOutput *output, MetaOutput *clone) { unsigned int i; for (i = 0; i < output->n_possible_clones; i++) if (output->possible_clones[i] == clone) return TRUE; return FALSE; } static gboolean can_clone (MetaCRTCInfo *info, MetaOutput *output) { unsigned int i; for (i = 0; i < info->outputs->len; ++i) { MetaOutput *clone = info->outputs->pdata[i]; if (!output_can_clone (clone, output)) return FALSE; } return TRUE; } static gboolean crtc_can_drive_output (MetaCRTC *crtc, MetaOutput *output) { unsigned int i; for (i = 0; i < output->n_possible_crtcs; i++) if (output->possible_crtcs[i] == crtc) return TRUE; return FALSE; } static gboolean output_supports_mode (MetaOutput *output, MetaMonitorMode *mode) { unsigned int i; for (i = 0; i < output->n_modes; i++) if (output->modes[i] == mode) return TRUE; return FALSE; } static gboolean crtc_assignment_assign (CrtcAssignment *assign, MetaCRTC *crtc, MetaMonitorMode *mode, int x, int y, enum wl_output_transform transform, MetaOutput *output) { MetaCRTCInfo *info = g_hash_table_lookup (assign->info, crtc); if (!crtc_can_drive_output (crtc, output)) return FALSE; if (!output_supports_mode (output, mode)) return FALSE; if ((crtc->all_transforms & (1 << transform)) == 0) return FALSE; if (info) { if (!(info->mode == mode && info->x == x && info->y == y && info->transform == transform)) return FALSE; if (!can_clone (info, output)) return FALSE; g_ptr_array_add (info->outputs, output); return TRUE; } else { MetaCRTCInfo *info = g_slice_new0 (MetaCRTCInfo); info->crtc = crtc; info->mode = mode; info->x = x; info->y = y; info->transform = transform; info->outputs = g_ptr_array_new (); g_ptr_array_add (info->outputs, output); g_hash_table_insert (assign->info, crtc, info); return TRUE; } } static void crtc_assignment_unassign (CrtcAssignment *assign, MetaCRTC *crtc, MetaOutput *output) { MetaCRTCInfo *info = g_hash_table_lookup (assign->info, crtc); if (info) { g_ptr_array_remove (info->outputs, output); if (info->outputs->len == 0) g_hash_table_remove (assign->info, crtc); } } static MetaOutput * find_output_by_key (MetaOutput *outputs, unsigned int n_outputs, MetaOutputKey *key) { unsigned int i; for (i = 0; i < n_outputs; i++) { if (strcmp (outputs[i].name, key->connector) == 0) { /* This should be checked a lot earlier! */ g_warn_if_fail (strcmp (outputs[i].vendor, key->vendor) == 0 && strcmp (outputs[i].product, key->product) == 0 && strcmp (outputs[i].serial, key->serial) == 0); return &outputs[i]; } } /* Just to satisfy GCC - this is a fatal error if occurs */ return NULL; } /* Check whether the given set of settings can be used * at the same time -- ie. whether there is an assignment * of CRTC's to outputs. * * Brute force - the number of objects involved is small * enough that it doesn't matter. */ static gboolean real_assign_crtcs (CrtcAssignment *assignment, unsigned int output_num) { MetaMonitorMode *modes; MetaCRTC *crtcs; MetaOutput *outputs; unsigned int n_crtcs, n_modes, n_outputs; MetaOutputKey *output_key; MetaOutputConfig *output_config; unsigned int i; gboolean success; if (output_num == assignment->config->n_outputs) return TRUE; output_key = &assignment->config->keys[output_num]; output_config = &assignment->config->outputs[output_num]; /* It is always allowed for an output to be turned off */ if (!output_config->enabled) return real_assign_crtcs (assignment, output_num + 1); meta_monitor_manager_get_resources (assignment->manager, &modes, &n_modes, &crtcs, &n_crtcs, &outputs, &n_outputs); success = FALSE; for (i = 0; i < n_crtcs; i++) { MetaCRTC *crtc = &crtcs[i]; unsigned int pass; /* Make two passes, one where frequencies must match, then * one where they don't have to */ for (pass = 0; pass < 2; pass++) { MetaOutput *output = find_output_by_key (outputs, n_outputs, output_key); unsigned int j; for (j = 0; j < n_modes; j++) { MetaMonitorMode *mode = &modes[j]; int width, height; if (meta_monitor_transform_is_rotated (output_config->transform)) { width = mode->height; height = mode->width; } else { width = mode->width; height = mode->height; } if (width == output_config->rect.width && height == output_config->rect.height && (pass == 1 || mode->refresh_rate == output_config->refresh_rate)) { meta_verbose ("CRTC %ld: trying mode %dx%d@%fHz with output at %dx%d@%fHz (transform %d) (pass %d)\n", crtc->crtc_id, mode->width, mode->height, mode->refresh_rate, output_config->rect.width, output_config->rect.height, output_config->refresh_rate, output_config->transform, pass); if (crtc_assignment_assign (assignment, crtc, &modes[j], output_config->rect.x, output_config->rect.y, output_config->transform, output)) { if (real_assign_crtcs (assignment, output_num + 1)) { success = TRUE; goto out; } crtc_assignment_unassign (assignment, crtc, output); } } } } } out: return success; } static gboolean meta_monitor_config_assign_crtcs (MetaConfiguration *config, MetaMonitorManager *manager, GPtrArray *crtcs, GPtrArray *outputs) { CrtcAssignment assignment; GHashTableIter iter; MetaCRTC *crtc; MetaCRTCInfo *info; unsigned int i; MetaOutput *all_outputs; unsigned int n_outputs; assignment.config = config; assignment.manager = manager; assignment.info = g_hash_table_new_full (NULL, NULL, NULL, (GDestroyNotify)meta_crtc_info_free); if (!real_assign_crtcs (&assignment, 0)) { meta_warning ("Could not assign CRTC to outputs, ignoring configuration\n"); g_hash_table_destroy (assignment.info); return FALSE; } g_hash_table_iter_init (&iter, assignment.info); while (g_hash_table_iter_next (&iter, (void**)&crtc, (void**)&info)) { g_hash_table_iter_steal (&iter); g_ptr_array_add (crtcs, info); } all_outputs = meta_monitor_manager_get_outputs (manager, &n_outputs); g_assert (n_outputs == config->n_outputs); for (i = 0; i < n_outputs; i++) { MetaOutputInfo *output_info = g_slice_new (MetaOutputInfo); MetaOutputConfig *output_config = &config->outputs[i]; output_info->output = find_output_by_key (all_outputs, n_outputs, &config->keys[i]); output_info->is_primary = output_config->is_primary; output_info->is_presentation = output_config->is_presentation; g_ptr_array_add (outputs, output_info); } g_hash_table_destroy (assignment.info); return TRUE; } void meta_crtc_info_free (MetaCRTCInfo *info) { g_ptr_array_free (info->outputs, TRUE); g_slice_free (MetaCRTCInfo, info); } void meta_output_info_free (MetaOutputInfo *info) { g_slice_free (MetaOutputInfo, info); }