aboutsummaryrefslogtreecommitdiffstats
path: root/src/backend-direct.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/backend-direct.c')
-rw-r--r--src/backend-direct.c941
1 files changed, 0 insertions, 941 deletions
diff --git a/src/backend-direct.c b/src/backend-direct.c
deleted file mode 100644
index 9f62c4d..0000000
--- a/src/backend-direct.c
+++ /dev/null
@@ -1,941 +0,0 @@
-/*-
- * redshift-ng - Automatically adjust display colour temperature according the Sun
- *
- * Copyright (c) 2009-2018 Jon Lund Steffensen <jonlst@gmail.com>
- * Copyright (c) 2014-2016, 2025 Mattias Andrée <m@maandree.se>
- *
- * redshift-ng 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 3 of the License, or
- * (at your option) any later version.
- *
- * redshift-ng 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 redshift-ng. If not, see <http://www.gnu.org/licenses/>.
- */
-#include "common.h"
-
-
-/**
- * Union of libgamma gamma ramp structures
- */
-union gamma_ramps {
- /**
- * Gamma ramp sizes
- */
- struct {
- size_t red; /**< Size of the gamma ramp for the red channel */
- size_t green; /**< Size of the gamma ramp for the green channel */
- size_t blue; /**< Size of the gamma ramp for the blue channel */
- } size;
-
- /* Ramp structures */
-#define X(SUFFIX, RAMPS, TYPE, MAX, DEPTH)\
- struct libgamma_gamma_##RAMPS RAMPS
- LIST_RAMPS_STOP_VALUE_TYPES(X, ;);
-#undef X
-};
-
-
-/**
- * State for partition (e.g. X screen or graphics card)
- */
-struct partition_state {
- /**
- * libgamma state
- */
- struct libgamma_partition_state state;
-
- /**
- * Index of CRTC when indexing spans multiple X screens
- */
- size_t crtc_num_offset;
-};
-
-
-/**
- * CRTC state
- */
-struct crtc_state {
- /**
- * libgamma state
- */
- struct libgamma_crtc_state state;
-
- /**
- * Gamma ramp depth, 0 if uninitialised,
- * -1 if single-precision float,
- * -2 if double-precision float,
- * otherwise the number of bits (of unsigned integer)
- */
- signed gamma_depth;
-
- /**
- * Gamma ramps used by the gamma adjustment function
- */
- union gamma_ramps gamma_ramps;
-
- /**
- * Original state of gamma ramps
- */
- union gamma_ramps saved_gamma_ramps;
-};
-
-
-/**
- * CRTC number, either overall number or partion and number within partition
- */
-struct crtc_number {
- /**
- * The partition of the EDID, -1 if using overall ordinal
- */
- ssize_t partition;
-
- /**
- * CRTC number within the partition, or overall if no partition is specified
- */
- size_t crtc;
-};
-
-
-struct gamma_state {
- /**
- * libgamma state for the site (e.g. X display)
- */
- struct libgamma_site_state site;
-
- /**
- * Whether the adjustment method supports fetching
- * EDIDs from the outputs
- */
- unsigned supports_edid : 1;
-
- /**
- * Whether the adjustment method supports multiple
- * sites rather than just the default site
- */
- unsigned multiple_sites : 1;
-
- /**
- * Whether the adjustment method supports multiple partitions
- * per site
- */
- unsigned multiple_partitions : 1;
-
- /**
- * Whether the adjustment method supports multiple CRTC:s
- * per partition per site
- */
- unsigned multiple_crtcs : 1;
-
- /**
- * Whether partitions are graphics cards
- */
- unsigned partitions_are_graphics_cards : 1;
-
- /**
- * Whether a parition has been selected
- */
- unsigned partitions_selected : 1;
-
- /**
- * Whether CRTCs have been selected
- */
- unsigned crtcs_selected : 1;
-
- /**
- * Whether the program has connected to the site
- */
- unsigned connected : 1;
-
- /**
- * The libgamma constant for the adjustment method
- */
- int method;
-
- /**
- * Number of selected CRTCs
- */
- size_t ncrtcs;
-
- /**
- * Number of selected EDIDs
- */
- size_t nedids;
-
- /**
- * Number of selected parition
- */
- size_t npartitions;
-
- /**
- * Selected paritions (if `partitions_selected`)
- */
- size_t *selected_partitions;
-
- /**
- * Selected CRTC numbers
- *
- * Deallocated by when no longer needed
- */
- struct crtc_number *selected_crtcs;
-
- /**
- * Selected EDIDs
- *
- * Deallocated by when no longer needed
- */
- char **selected_edids;
-
- /**
- * Name of site to connect to, `NULL` if not selected
- * or once connected to the site
- */
- char *site_name;
-
- /**
- * State for selected paritions
- */
- struct partition_state *partitions;
-
- /**
- * State for selected CRTCs
- */
- struct crtc_state *crtcs;
-};
-
-
-int
-direct_create(struct gamma_state **state_out, int method, const char *method_name)
-{
- struct libgamma_method_capabilities caps;
- struct gamma_state *state;
- int err;
-
- *state_out = NULL;
-
- if (!libgamma_is_method_available(method)) {
- weprintf(_("Adjustment method `%s' is not available"), method_name);
- return -1;
- }
-
- err = libgamma_method_capabilities(&caps, sizeof(caps), method);
- if (err) {
- weprintf("libgamma_method_capabilities %s: %s", libgamma_const_of_method(method), libgamma_strerror(err));
- return -1;
- }
-
- state = *state_out = ecalloc(1, sizeof(**state_out));
- state->selected_crtcs = NULL;
- state->partitions = NULL;
- state->site_name = NULL;
- state->method = method;
- state->supports_edid = (caps.crtc_information & LIBGAMMA_CRTC_INFO_EDID) ? 1 : 0;
- state->multiple_sites = caps.multiple_sites;
- state->multiple_partitions = caps.multiple_partitions;
- state->multiple_crtcs = caps.multiple_crtcs;
- state->partitions_are_graphics_cards = caps.partitions_are_graphics_cards;
-
- if (state->multiple_sites)
- return 0;
-
- err = libgamma_site_initialise(&state->site, method, NULL);
- if (err) {
- weprintf("libgamma_site_initialise %s NULL: %s", libgamma_const_of_method(method), libgamma_strerror(err));
- free(state);
- *state_out = NULL;
- return -1;
- }
-
- return 0;
-}
-
-
-void
-direct_print_help(int method)
-{
- struct libgamma_method_capabilities caps;
-
- if (libgamma_method_capabilities(&caps, sizeof(caps), method)) {
- printf(_("Adjustment method not available\n"));
- printf("\n");
- return;
- }
-
- if (caps.multiple_sites)
- printf(" display=%s %s\n", _("NAME "), _("Display server instance to apply adjustments to"));
-
- if (caps.multiple_partitions && caps.partitions_are_graphics_cards) {
- /* TRANSLATORS: "N" represents "ordinal"; right-pad with spaces to preserve display width */
- printf(" card=%s %s\n", _("N "), _("Graphics card to apply adjustments to"));
- } else if (caps.multiple_partitions) {
- /* TRANSLATORS: "N" represents "ordinal"; right-pad with spaces to preserve display width */
- printf(" screen=%s %s\n", _("N "), _("List of comma-separated X screens to apply adjustments to"));
- }
-
- if (caps.multiple_crtcs) {
- /* TRANSLATORS: "N" represents "number"; right-pad with spaces to preserve display width */
- printf(" crtc=%s %s\n", _("N "), _("List of comma-separated CRTCs to apply adjustments to"));
- }
- if (caps.multiple_crtcs && (caps.crtc_information & LIBGAMMA_CRTC_INFO_EDID)) {
- printf(" edid=%s %s\n", _("EDID "), _("List of comma-separated EDIDS of monitors to apply "
- "adjustments to, enter `list' to list available monitors"));
- }
-
- if (caps.multiple_sites || caps.multiple_partitions || caps.multiple_crtcs)
- printf("\n");
-}
-
-
-int
-direct_set_option(struct gamma_state *state, const char *key, const char *value)
-{
- if (state->multiple_sites && !strcasecmp(key, "display")) {
- return direct_set_site(state, key, value);
- } else if (state->multiple_partitions && !strcasecmp(key, state->partitions_are_graphics_cards ? "card" : "screen")) {
- return direct_set_partitions(state, key, value);
- } else if (state->multiple_crtcs && !strcasecmp(key, "crtc")) {
- return direct_set_crtcs(state, key, value);
- } else if (state->multiple_crtcs && state->supports_edid && !strcasecmp(key, "edid")) {
- return direct_set_edids(state, key, value);
- } else if (!strcasecmp(key, "preserve")) {
- weprintf(_("Deprecated method parameter ignored: `%s'."), key);
- return 0;
- } else {
- weprintf(_("Unknown method parameter: `%s'."), key);
- return -1;
- }
-}
-
-
-int
-direct_set_site(struct gamma_state *state, const char *key, const char *value)
-{
- if (state->site_name) {
- weprintf(_("`%s' option specified multiple times, using last selection."), key);
- free(state->site_name);
- }
- state->site_name = estrdup(value);
- return 0;
-}
-
-
-int
-direct_set_partitions(struct gamma_state *state, const char *key, const char *value)
-{
- const char *end, *p;
- uintmax_t num;
- size_t count;
-
- /* Check previously unspecified */
- if (state->partitions_selected)
- weprintf(_("`%s' option specified multiple times, using last selection."), key);
- state->partitions_selected = 1;
-
- /* Check if all are selected */
- state->npartitions = 0;
- free(state->selected_partitions);
- state->selected_partitions = NULL;
- if (!*value || !strcasecmp(value, "all"))
- return 0;
-
- /* Get number count */
- for (p = value, count = 1; *p; p++)
- if (*p == ',')
- count++;
- state->selected_partitions = ecalloc(count, sizeof(*state->selected_partitions));
-
- /* Parse numbers */
- errno = 0;
- for (p = value; *p; p = end) {
- num = strtoumax(p, (void *)&end, 10);
- state->selected_partitions[state->npartitions++] = (size_t)num;
- if (num > (uintmax_t)SIZE_MAX || (*end && *end != ',') || !isdigit(*p) || errno) {
- weprintf(_("Invalid value of `%s' option: `%s'."), key, value);
- return -1;
- }
- end = &end[*end == ','];
- }
-
- return 0;
-}
-
-
-int
-direct_set_crtcs(struct gamma_state *state, const char *key, const char *value)
-{
- const char *end, *p;
- uintmax_t num;
- size_t count;
-
- /* Check previously unspecified */
- if (state->crtcs_selected)
- weprintf(_("`%s' option specified multiple times, using last selection."), key);
- state->crtcs_selected = 1;
-
- /* Check if all are selected */
- state->ncrtcs = 0;
- free(state->selected_crtcs);
- state->selected_crtcs = NULL;
- if (!*value || !strcasecmp(value, "all"))
- return 0;
-
- /* Get number count */
- for (p = value, count = 1; *p; p++)
- if (*p == ',')
- count++;
- state->selected_crtcs = ecalloc(count, sizeof(*state->selected_crtcs));
-
- /* Parse numbers */
- errno = 0;
- for (p = value; *p; p = end) {
- num = strtoumax(p, (void *)&end, 10);
- state->selected_crtcs[state->ncrtcs].partition = -1;
- state->selected_crtcs[state->ncrtcs].crtc = (size_t)num;
- if (num > (uintmax_t)SIZE_MAX || (*end && *end != ',' && *end != '.') || !isdigit(*p) || errno) {
- invalid:
- weprintf(_("Invalid value of `%s' option: `%s'."), key, value);
- return -1;
- }
- if (*end == '.') {
- if (!state->multiple_partitions || num > (uintmax_t)SSIZE_MAX)
- goto invalid;
- state->selected_crtcs[state->ncrtcs].partition = (ssize_t)num;
- p = &end[1];
- num = strtoumax(p, (void *)&end, 10);
- state->selected_crtcs[state->ncrtcs].crtc = (size_t)num;
- if (num > (uintmax_t)SIZE_MAX || (*end && *end != ',') || !isdigit(*p) || errno)
- goto invalid;
- }
- end = &end[*end == ','];
- state->ncrtcs++;
- }
-
- return 0;
-}
-
-
-int
-direct_set_edids(GAMMA_STATE *state, const char *key, const char *value)
-{
- const char *end, *p;
- size_t count, len;
-
- /* Check previously unspecified */
- if (state->nedids) {
- weprintf(_("`%s' option specified multiple times, using last selection."), key);
- while (state->nedids)
- free(state->selected_edids[--state->nedids]);
- free(state->selected_edids);
- state->selected_edids = NULL;
- state->nedids = 0;
- }
-
- /* Get number count */
- for (p = value, count = 1; *p; p++)
- if (*p == ',')
- count++;
- state->selected_edids = ecalloc(count, sizeof(*state->selected_edids));
-
- /* Split */
- for (p = value;; p = end) {
- end = strchr(p, ',');
- if (!end) {
- state->selected_edids[state->nedids++] = estrdup(p);
- break;
- }
- len = (size_t)(end++ - p);
- state->selected_edids[state->nedids] = emalloc(len + 1U);
- memcpy(state->selected_edids[state->nedids], p, len);
- state->selected_edids[state->nedids][len] = '\0';
- state->nedids++;
- }
-
- return 0;
-}
-
-
-int
-direct_start(struct gamma_state *state)
-{
- struct {
- size_t partition;
- size_t crtc;
- char *edid;
- } *resolved_edids = NULL;
- struct libgamma_crtc_information crtc_info;
- struct libgamma_crtc_state crtc_state;
- size_t crtc_num_offset = 0;
- size_t crtcs_removed = 0;
- size_t nresolved_edids = 0;
- size_t i, j, k, num, part, count;
- int err;
-
- /* Connect to display server */
- if (state->multiple_sites) {
- if (state->site_name && !*state->site_name) {
- free(state->site_name);
- state->site_name = NULL;
- }
- err = libgamma_site_initialise(&state->site, state->method, state->site_name);
- state->site_name = NULL;
- if (err) {
- weprintf("libgamma_site_initialise %s %s: %s",
- libgamma_const_of_method(state->method),
- state->site_name ? state->site_name : "NULL",
- libgamma_strerror(err));
- return -1;
- }
- state->connected = 1;
- }
-
- /* Allocate partition states */
- if (state->npartitions) {
- state->partitions = ecalloc(state->npartitions, sizeof(*state->partitions));
- for (i = 0; i < state->npartitions; i++) {
- state->partitions[i].state.partition = state->selected_partitions[i];
- state->partitions[i].state.data = NULL;
- }
- free(state->selected_partitions);
- state->selected_partitions = NULL;
- } else if (!state->site.partitions_available) {
- if (state->partitions_are_graphics_cards)
- weprintf(_("No graphics card found."));
- else
- weprintf(_("No X screen found."));
- return -1;
- } else {
- state->npartitions = state->site.partitions_available;
- state->partitions = ecalloc(state->npartitions, sizeof(*state->partitions));
- for (i = 0; i < state->npartitions; i++) {
- state->partitions[i].state.partition = i;
- state->partitions[i].state.data = NULL;
- }
- }
-
- /* Map the partition indices in CRTC numbers from indices of selected
- * partitions, and allocate partitions specified via CRTC numbers */
- for (i = 0; i < state->ncrtcs; i++) {
- if (state->selected_crtcs[i].partition < 0)
- continue;
- for (j = 0; j < state->npartitions; j++) {
- if ((size_t)state->selected_crtcs[i].partition == state->partitions[j].state.partition) {
- state->selected_crtcs[i].partition = (ssize_t)j;
- break;
- }
- }
- if (j == state->npartitions) {
- state->partitions = realloc(state->partitions, (state->npartitions + 1U) * sizeof(*state->partitions));
- state->partitions[state->npartitions].state.partition = (size_t)state->selected_crtcs[i].partition;
- state->partitions[state->npartitions].state.data = NULL;
- state->npartitions++;
- }
- }
-
- /* Initialise partitions */
- for (i = 0; i < state->npartitions; i++) {
- num = state->partitions[i].state.partition;
- err = libgamma_partition_initialise(&state->partitions[i].state, &state->site, num);
- if (err) {
- weprintf("libgamma_partition_initialise %zu: %s", num, libgamma_strerror(err));
- return -1;
- }
- state->partitions[i].crtc_num_offset = crtc_num_offset;
- crtc_num_offset += state->partitions[i].state.crtcs_available;
- }
-
- /* Resolve EDIDs */
- if (state->nedids) {
- for (i = 0; i < state->npartitions; i++) {
- count = state->partitions[i].state.crtcs_available;
- if (!count)
- continue;
- resolved_edids = erealloc(resolved_edids, (nresolved_edids + count) + sizeof(*resolved_edids));
- for (j = 0; j < count; j++) {
- if (libgamma_crtc_initialise(&crtc_state, &state->partitions[i].state, j))
- continue;
- libgamma_get_crtc_information(&crtc_info, sizeof(crtc_info), &crtc_state, LIBGAMMA_CRTC_INFO_EDID);
- if (crtc_info.edid_error)
- goto next_crtc;
- resolved_edids[nresolved_edids].partition = i;
- resolved_edids[nresolved_edids].crtc = j;
- resolved_edids[nresolved_edids].edid = libgamma_behex_edid(crtc_info.edid, crtc_info.edid_length);
- if (!resolved_edids[nresolved_edids].edid)
- eprintf("libgamma_behex_edid:");
- nresolved_edids++;
- next_crtc:
- libgamma_crtc_information_destroy(&crtc_info);
- libgamma_crtc_destroy(&crtc_state);
- }
- }
- for (i = 0; i < state->nedids; i++) {
- if (!strcasecmp(state->selected_edids[i], "list")) {
- printf(_("Available outputs:\n"));
- for (i = 0; i < nresolved_edids; i++)
- printf(" %s\n", resolved_edids[i].edid);
- direct_free(state);
- exit(0);
- }
- for (j = 0; j < nresolved_edids; j++) {
- if (!strcasecmp(state->selected_edids[i], resolved_edids[i].edid)) {
- if (!state->multiple_partitions) {
- weprintf("Resolved output `%s' to CRTC %zu",
- resolved_edids[i].edid, resolved_edids[i].crtc);
- } else if (state->partitions_are_graphics_cards) {
- weprintf("Resolved output `%s' to CRTC %zu on graphics card %zu",
- resolved_edids[i].edid, resolved_edids[i].crtc,
- resolved_edids[i].partition);
- } else {
- weprintf("Resolved output `%s' to CRTC %zu on X screen %zu",
- resolved_edids[i].edid, resolved_edids[i].crtc,
- resolved_edids[i].partition);
- }
- count = state->ncrtcs + 1U;
- state->selected_crtcs = erealloc(state->selected_crtcs, count * sizeof(*state->crtcs));
- state->selected_crtcs[state->ncrtcs].partition = (ssize_t)resolved_edids[i].partition;
- state->selected_crtcs[state->ncrtcs].crtc = resolved_edids[i].crtc;
- state->ncrtcs++;
- goto next_edid;
- }
- }
- weprintf(_("Output `%s' found."), state->selected_edids[i]);
- next_edid:
- free(state->selected_edids[i]);
- state->selected_edids[i] = NULL;
- }
- while (nresolved_edids)
- free(resolved_edids[--nresolved_edids].edid);
- free(resolved_edids);
- free(state->selected_edids);
- state->selected_edids = NULL;
- }
-
- /* Allocate CRTCs states and map to partition–CRTC pairs */
- if (state->ncrtcs) {
- state->crtcs = ecalloc(state->ncrtcs, sizeof(*state->crtcs));
- for (i = 0; i < state->ncrtcs; i++) {
- state->crtcs[i].state.data = NULL;
- state->crtcs[i].state.partition = NULL;
- if (state->selected_crtcs[i].partition >= 0)
- state->crtcs[i].state.partition = &state->partitions[state->selected_crtcs[i].partition].state;
- state->crtcs[i].state.crtc = state->selected_crtcs[i].crtc;
- }
- for (i = 0; i < state->ncrtcs; i++) {
- if (state->crtcs[i].state.partition)
- continue;
- for (j = 1; j < state->npartitions; j++)
- if (state->crtcs[i].state.crtc < state->partitions[j].crtc_num_offset)
- break;
- state->crtcs[i].state.partition = &state->partitions[j - 1U].state;
- state->crtcs[i].state.crtc -= state->partitions[j - 1U].crtc_num_offset;
- }
- } else if (!crtc_num_offset) {
- weprintf(_("No CRTCs found."));
- return -1;
- } else {
- state->ncrtcs = crtc_num_offset;
- state->crtcs = ecalloc(state->ncrtcs, sizeof(*state->crtcs));
- for (j = 0, i = 0; j < state->npartitions; j++) {
- for (k = 0; k < state->partitions[j].state.crtcs_available; k++, i++) {
- state->crtcs[i].state.data = NULL;
- state->crtcs[i].state.partition = &state->partitions[j].state;
- state->crtcs[i].state.crtc = k;
- }
- }
- }
-
- /* Initialise CRTCs and fetch original gamma adjustments */
- for (i = 0; i < state->ncrtcs; i++) {
- /* Get CRTC */
- part = state->crtcs[i].state.partition->partition;
- num = state->crtcs[i].state.crtc;
- err = libgamma_crtc_initialise(&state->crtcs[i].state, state->crtcs[i].state.partition, num);
- if (err) {
- weprintf("libgamma_crtc_initialise %zu %zu: %s", part, num, libgamma_strerror(err));
- return -1;
- }
-
- /* Get gamma ramp inforamtion for CRTC */
- libgamma_get_crtc_information(&crtc_info, sizeof(crtc_info), &state->crtcs[i].state,
- LIBGAMMA_CRTC_INFO_GAMMA_SIZE |
- LIBGAMMA_CRTC_INFO_GAMMA_DEPTH |
- LIBGAMMA_CRTC_INFO_GAMMA_SUPPORT);
- if (crtc_info.gamma_size_error || crtc_info.gamma_depth_error) {
- libgamma_crtc_destroy(&state->crtcs[i].state);
- state->crtcs[i].state.data = NULL;
- crtcs_removed++;
- goto no_gamma_support;
- }
- if (!crtc_info.gamma_support_error && crtc_info.gamma_support == LIBGAMMA_NO) {
- no_gamma_support:
- if (state->multiple_crtcs) {
- if (!state->multiple_partitions)
- weprintf(_("Adjustments are not supported on CRTC %zu."), num);
- else if (state->partitions_are_graphics_cards)
- weprintf(_("Adjustments are not supported on CRTC %zu on graphics card %zu."), num, part);
- else
- weprintf(_("Adjustments are not supported on CRTC %zu on X screen %zu."), num, part);
- } else {
- if (!state->multiple_partitions)
- weprintf(_("Adjustments are not supported."));
- else if (state->partitions_are_graphics_cards)
- weprintf(_("Adjustments are not supported on graphics card %zu."), part);
- else
- weprintf(_("Adjustments are not supported on X screen %zu."), part);
- }
- if (!state->crtcs[i].state.data)
- goto next;
- }
-
- /* Initialise and fetch gamma adjustments */
- state->crtcs[i].gamma_ramps.size.red = crtc_info.red_gamma_size;
- state->crtcs[i].gamma_ramps.size.green = crtc_info.green_gamma_size;
- state->crtcs[i].gamma_ramps.size.blue = crtc_info.blue_gamma_size;
- state->crtcs[i].saved_gamma_ramps.size.red = crtc_info.red_gamma_size;
- state->crtcs[i].saved_gamma_ramps.size.green = crtc_info.green_gamma_size;
- state->crtcs[i].saved_gamma_ramps.size.blue = crtc_info.blue_gamma_size;
- switch (crtc_info.gamma_depth) {
-#define X(SUFFIX, RAMPS, TYPE, MAX, DEPTH)\
- case DEPTH:\
- if (libgamma_gamma_##RAMPS##_initialise(&state->crtcs[i].gamma_ramps.RAMPS) ||\
- libgamma_gamma_##RAMPS##_initialise(&state->crtcs[i].saved_gamma_ramps.RAMPS))\
- eprintf("libgamma_gamma_"#RAMPS"_initialise:");\
- j = 0;\
- do {\
- err = libgamma_crtc_get_gamma_##RAMPS(&state->crtcs[i].state,\
- &state->crtcs[i].saved_gamma_ramps.RAMPS);\
- } while (err && j++ < 10);\
- if (!err) {\
- state->crtcs[i].gamma_depth = DEPTH;\
- break;\
- }\
- if (state->multiple_crtcs) {\
- if (!state->multiple_partitions)\
- weprintf(_("Could not get current adjustments for CRTC %zu."), num);\
- else if (state->partitions_are_graphics_cards)\
- weprintf(_("Could not get current adjustments for CRTC %zu on graphics card %zu."),\
- num, part);\
- else\
- weprintf(_("Could not get current adjustments for CRTC %zu on X screen %zu."), num, part);\
- } else {\
- if (!state->multiple_partitions)\
- weprintf(_("Could not get current adjustments."));\
- else if (state->partitions_are_graphics_cards)\
- weprintf(_("Could not get current adjustments for graphics card %zu."), part);\
- else\
- weprintf(_("Could not get current adjustments for X screen %zu."), part);\
- }\
- libgamma_gamma_##RAMPS##_destroy(&state->crtcs[i].gamma_ramps.RAMPS);\
- libgamma_gamma_##RAMPS##_destroy(&state->crtcs[i].saved_gamma_ramps.RAMPS);\
- libgamma_crtc_destroy(&state->crtcs[i].state);\
- state->crtcs[i].state.data = NULL;\
- crtcs_removed++;\
- goto next
-
- LIST_RAMPS_STOP_VALUE_TYPES(X, ;);
-#undef X
-
- default:
- if (state->multiple_crtcs) {
- if (!state->multiple_partitions)
- weprintf(_("Unsupported gamma ramp type on CRTC %zu."), num);
- else if (state->partitions_are_graphics_cards)
- weprintf(_("Unsupported gamma ramp type on CRTC %zu on graphics card %zu."), num, part);
- else
- weprintf(_("Unsupported gamma ramp type on CRTC %zu on X screen %zu."), num, part);
- } else {
- if (!state->multiple_partitions)
- weprintf(_("Unsupported gamma ramp type."));
- else if (state->partitions_are_graphics_cards)
- weprintf(_("Unsupported gamma ramp type on graphics card %zu."), part);
- else
- weprintf(_("Unsupported gamma ramp type on X screen %zu."), part);
- }
- return -1;
- }
-
- next:
- libgamma_crtc_information_destroy(&crtc_info);
- }
-
- /* Unlist removed CRTCs */
- if (crtcs_removed) {
- num = state->ncrtcs - crtcs_removed;
- for (i = j = 0; crtcs_removed; i++, j++)
- if (!state->crtcs[i].state.data)
- break;
- for (; crtcs_removed; i++) {
- if (state->crtcs[i].state.data)
- crtcs_removed--;
- else
- memmove(&state->crtcs[j++], &state->crtcs[i], sizeof(*state->crtcs));
- }
- memmove(&state->crtcs[j], &state->crtcs[i], (state->ncrtcs - i) * sizeof(*state->crtcs));
- state->ncrtcs = num;
- }
- if (!state->ncrtcs) {
- weprintf(_("No usable CRTCs available."));
- return -1;
- }
-
- free(state->selected_crtcs);
- state->selected_crtcs = NULL;
- return 0;
-}
-
-
-int
-direct_apply(struct gamma_state *state, const struct colour_setting *setting, int preserve)
-{
- size_t i, err_count = 0, crtc, part;
- const char *errstr;
- int err;
-
- for (i = 0; i < state->ncrtcs; i++) {
- switch (state->crtcs[i].gamma_depth) {
-#define X(SUFFIX, RAMPS, TYPE, MAX, DEPTH)\
- case DEPTH:\
- fill_ramps_##SUFFIX(state->crtcs[i].gamma_ramps.RAMPS.red,\
- state->crtcs[i].gamma_ramps.RAMPS.green,\
- state->crtcs[i].gamma_ramps.RAMPS.blue,\
- preserve ? state->crtcs[i].saved_gamma_ramps.RAMPS.red : NULL,\
- preserve ? state->crtcs[i].saved_gamma_ramps.RAMPS.green : NULL,\
- preserve ? state->crtcs[i].saved_gamma_ramps.RAMPS.blue : NULL,\
- state->crtcs[i].gamma_ramps.size.red,\
- state->crtcs[i].gamma_ramps.size.green,\
- state->crtcs[i].gamma_ramps.size.blue,\
- setting);\
- err = libgamma_crtc_set_gamma_##RAMPS(&state->crtcs[i].state, &state->crtcs[i].gamma_ramps.RAMPS);\
- break
-
- LIST_RAMPS_STOP_VALUE_TYPES(X, ;);
-#undef X
-
- default:
- err_count++;
- err = 0;
- break;
- }
-
- if (err) {
- err_count++;
- crtc = state->crtcs[i].state.crtc;
- part = state->crtcs[i].state.partition->partition;
- errstr = libgamma_strerror(err);
- if (state->multiple_crtcs) {
- if (!state->multiple_partitions)
- weprintf(_("Unable to set adjustments for CRTC %zu: %s."), crtc, errstr);
- else if (state->partitions_are_graphics_cards)
- weprintf(_("Unable to set adjustments for CRTC %zu on graphics card %zu: %s."),
- crtc, part, errstr);
- else
- weprintf(_("Unable to set adjustments for CRTC %zu on X screen %zu: %s."),
- crtc, part, errstr);
- } else {
- if (!state->multiple_partitions)
- weprintf(_("Unable to set adjustments: %s."), errstr);
- else if (state->partitions_are_graphics_cards)
- weprintf(_("Unable to set adjustments for graphics card %zu: %s."), part, errstr);
- else
- weprintf(_("Unable to set adjustments for X screen %zu: %s."), part, errstr);
- }
- }
- }
-
- return err_count == state->ncrtcs ? -1 : 0;
-}
-
-
-void
-direct_restore(struct gamma_state *state)
-{
- size_t i, crtc, part;
- const char *errstr;
- int err;
-
- for (i = 0; i < state->ncrtcs; i++) {
- switch (state->crtcs[i].gamma_depth) {
-#define X(SUFFIX, RAMPS, TYPE, MAX, DEPTH)\
- case DEPTH:\
- err = libgamma_crtc_set_gamma_##RAMPS(&state->crtcs[i].state, &state->crtcs[i].saved_gamma_ramps.RAMPS);\
- break
-
- LIST_RAMPS_STOP_VALUE_TYPES(X, ;);
-#undef X
-
- default:
- err = 0;
- break;
- }
-
- if (err) {
- crtc = state->crtcs[i].state.crtc;
- part = state->crtcs[i].state.partition->partition;
- errstr = libgamma_strerror(err);
- if (state->multiple_crtcs) {
- if (!state->multiple_partitions)
- weprintf(_("Unable to restore adjustments for CRTC %zu: %s."), crtc, errstr);
- else if (state->partitions_are_graphics_cards)
- weprintf(_("Unable to restore adjustments for CRTC %zu on graphics card %zu: %s."),
- crtc, part, errstr);
- else
- weprintf(_("Unable to restore adjustments for CRTC %zu on X screen %zu: %s."),
- crtc, part, errstr);
- } else {
- if (!state->multiple_partitions)
- weprintf(_("Unable to restore adjustments: %s."), errstr);
- else if (state->partitions_are_graphics_cards)
- weprintf(_("Unable to restore adjustments for graphics card %zu: %s."), part, errstr);
- else
- weprintf(_("Unable to restore adjustments for X screen %zu: %s."), part, errstr);
- }
- }
- }
-}
-
-
-void
-direct_free(struct gamma_state *state)
-{
- size_t i;
- if (state->crtcs) {
- for (i = 0; i < state->ncrtcs; i++) {
- switch (state->crtcs[i].gamma_depth) {
-#define X(SUFFIX, RAMPS, TYPE, MAX, DEPTH)\
- case DEPTH:\
- libgamma_gamma_##RAMPS##_destroy(&state->crtcs[i].gamma_ramps.RAMPS);\
- libgamma_gamma_##RAMPS##_destroy(&state->crtcs[i].saved_gamma_ramps.RAMPS);\
- break
- LIST_RAMPS_STOP_VALUE_TYPES(X, ;);
-#undef X
- default:
- case 0: /* not initialised */
- break;
- }
- if (state->crtcs[i].state.data)
- libgamma_crtc_destroy(&state->crtcs[i].state);
- }
- free(state->crtcs);
- }
- if (state->partitions) {
- for (i = 0; i < state->npartitions; i++)
- if (state->partitions[i].state.data)
- libgamma_partition_destroy(&state->partitions[i].state);
- free(state->partitions);
- }
- free(state->selected_partitions);
- free(state->selected_crtcs);
- if (state->selected_edids) {
- for (i = 0; i < state->nedids; i++)
- free(state->selected_edids[i]);
- free(state->selected_edids);
- }
- if (state->connected)
- libgamma_site_destroy(&state->site);
- free(state->site_name);
- free(state);
-}