diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/Makefile | 1 | ||||
-rw-r--r-- | src/backend-direct.c | 608 | ||||
-rw-r--r-- | src/colour.c | 2 | ||||
-rw-r--r-- | src/common.h | 88 | ||||
-rw-r--r-- | src/config.mk | 4 | ||||
-rw-r--r-- | src/gamma-coopgamma.c | 6 | ||||
-rw-r--r-- | src/gamma-randr.c | 381 | ||||
-rw-r--r-- | src/gamma-vidmode.c | 157 |
8 files changed, 716 insertions, 531 deletions
diff --git a/src/Makefile b/src/Makefile index fabc63e..a674b2e 100644 --- a/src/Makefile +++ b/src/Makefile @@ -8,6 +8,7 @@ VERSION_STRING = redshift-ng 1.13 OBJ =\ + backend-direct.o\ colour.o\ config.o\ config-ini.o\ diff --git a/src/backend-direct.c b/src/backend-direct.c new file mode 100644 index 0000000..d0a53eb --- /dev/null +++ b/src/backend-direct.c @@ -0,0 +1,608 @@ +/*- + * 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; +}; + + +struct gamma_state { + /** + * libgamma state for the site (e.g. X display) + */ + struct libgamma_site_state site; + + /** + * 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 partition_selected : 1; + + /** + * Whether CRTCs have been selected + */ + unsigned crtcs_selected : 1; + + /** + * Selected parition (if `partition_selected`) + */ + size_t selected_partition; + + /** + * Number of selected CRTCs + */ + size_t ncrtcs; + + /** + * Number of selected parition + */ + size_t npartitions; + + /** + * Indices of selected CRTCs + */ + size_t *selected_crtcs; + + /** + * 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->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; + + 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; +} + + +int +direct_set_partitions(struct gamma_state *state, const char *key, const char *value) +{ + const char *end; + uintmax_t num; + + /* Check previously unspecified */ + if (state->partition_selected) + weprintf(_("`%s' option specified multiple times, using last selection."), key); + state->partition_selected = 1; + + /* Parse number */ + errno = 0; + num = strtoumax(value, (void *)&end, 10); + state->selected_partition = (size_t)num; + if (num > (uintmax_t)SIZE_MAX || *end || !isdigit(*value) || errno) + eprintf(_("Invalid value of `%s' option: `%s'."), key, value); + + 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++] = (size_t)num; + if (num > (uintmax_t)SIZE_MAX || (*end && *end != ',') || !isdigit(*p) || errno) + eprintf(_("Invalid value of `%s' option: `%s'."), key, value); + end = &end[*end == ',']; + } + + return 0; +} + + +int +direct_start(struct gamma_state *state) +{ + + struct libgamma_crtc_information crtc_info; + size_t crtc_num_offset = 0; + size_t crtcs_removed = 0; + size_t i, j, k, num, part; + int err; + + /* Allocate partition states */ + if (state->partition_selected) { + state->partitions = ecalloc(1, sizeof(*state->partitions)); + state->partitions[0].state.partition = state->selected_partition; + state->partitions[0].state.data = NULL; + state->npartitions = 1; + } 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; + } + } + + /* 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; + } + + /* 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.crtc = state->selected_crtcs[i]; + } + for (i = 0; i < state->ncrtcs; i++) { + 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; + } + } 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; + } + + return 0; +} + + +int +direct_apply(struct gamma_state *state, const struct colour_setting *setting, int preserve /* TODO */) +{ + 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,\ + 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_crtcs); + libgamma_site_destroy(&state->site); + free(state); +} diff --git a/src/colour.c b/src/colour.c index 88acf64..ad65b5f 100644 --- a/src/colour.c +++ b/src/colour.c @@ -44,7 +44,7 @@ colour_setting_diff_is_major(const struct colour_setting *a, const struct colour } -#define X(SUFFIX, TYPE, MAX, DEPTH)\ +#define X(SUFFIX, RAMPS, TYPE, MAX, DEPTH)\ /** * Fill a gamma ramp * diff --git a/src/common.h b/src/common.h index b7cf872..e79c6f4 100644 --- a/src/common.h +++ b/src/common.h @@ -62,6 +62,7 @@ # include <time.h> #endif +#include <libgamma.h> #include <libred.h> @@ -1178,6 +1179,77 @@ extern int use_fade; extern int verbose; +/* backend-direct.c */ + +/** + * Create an initialised state object for direct gamma adjustments + * + * @param state_out Output parameter for the state object + * @return 0 on success, -1 on failure + * + * `*state_out` is set (potentially to `NULL`) on failure + */ +int direct_create(GAMMA_STATE **state_out, int method, const char *method_name); + +/** + * Select partitions to apply adjustments to using direct gamma adjustments + * + * @param state State object for the adjustment method + * @param key Option to configure + * @param value Option value to set + * @return 0 on success, -1 on failure + */ +int direct_set_partitions(GAMMA_STATE *state, const char *key, const char *value); + +/** + * Select CRTCs to apply adjustments to using direct gamma adjustments + * + * @param state State object for the adjustment method + * @param key Option to configure + * @param value Option value to set + * @return 0 on success, -1 on failure + */ +int direct_set_crtcs(GAMMA_STATE *state, const char *key, const char *value); + +/** + * Finalise option-dependent initialisation and connections + * for direct gamma adjustments + * + * @param state State object for the adjustment method + * @return 0 on success, -1 on failure + */ +int direct_start(GAMMA_STATE *state); + +/** + * Apply colour settings using direct gamma adjustments + * + * @param state State object for the adjustment method + * @param settings The colour settings to apply + * @param preserve Whether currently applied adjustments (assumed + * to be colour calibration) shall remain applied + * @return 0 on success, -1 on failure + */ +int direct_apply(GAMMA_STATE *state, const struct colour_setting *setting, int preserve); + +/** + * Restore the adjustments to the `.state` before start was called + * using direct gamma adjustments + * + * @param state State object for the adjustment method + */ +void direct_restore(GAMMA_STATE *state); + +/** + * Close connections and deallocate all state resources + * for direct gamma adjustments + * + * @param state The state to terminate + * + * The pointer `state` will become invalid + */ +void direct_free(GAMMA_STATE *state); + + /* colour.c */ /** @@ -1203,14 +1275,14 @@ GCC_ONLY(__attribute__((__pure__))) int colour_setting_diff_is_major(const struct colour_setting *a, const struct colour_setting *b); #define LIST_RAMPS_STOP_VALUE_TYPES(X, D)\ - X(u8, uint8_t, UINT8_MAX, 8) D\ - X(u16, uint16_t, UINT16_MAX, 16) D\ - X(u32, uint32_t, UINT32_MAX, 32) D\ - X(u64, uint64_t, UINT64_MAX, 64) D\ - X(float, float, 1, -1) D\ - X(double, double, 1, -2) - -#define X(SUFFIX, TYPE, MAX, DEPTH)\ + X(u8, ramps8, uint8_t, UINT8_MAX, 8) D \ + X(u16, ramps16, uint16_t, UINT16_MAX, 16) D\ + X(u32, ramps32, uint32_t, UINT32_MAX, 32) D\ + X(u64, ramps64, uint64_t, UINT64_MAX, 64) D\ + X(float, rampsf, float, 1, -1) D\ + X(double, rampsd, double, 1, -2) + +#define X(SUFFIX, RAMPS, TYPE, MAX, DEPTH)\ /** * Fill the gamma ramps * diff --git a/src/config.mk b/src/config.mk index 9c0e131..d0418bb 100644 --- a/src/config.mk +++ b/src/config.mk @@ -12,10 +12,8 @@ PKGCONFIG_LDFLAGS = $(PKGCONFIG) --libs DRM_LIBS = libdrm GEOCLUE_LIBS = glib-2.0 gio-2.0 -RANDR_LIBS = xcb xcb-randr -VIDMODE_LIBS = x11 xxf86vm -LIBS_PKGCONFIG = $(DRM_LIBS) $(GEOCLUE_LIBS) $(RANDR_LIBS) $(VIDMODE_LIBS) +LIBS_PKGCONFIG = $(DRM_LIBS) $(GEOCLUE_LIBS) libgamma CPPFLAGS = -D_DEFAULT_SOURCE -D_BSD_SOURCE -D_XOPEN_SOURCE=700 -D_GNU_SOURCE\ -DENABLE_DRM -DENABLE_GEOCLUE2 -DENABLE_RANDR -DENABLE_VIDMODE\ diff --git a/src/gamma-coopgamma.c b/src/gamma-coopgamma.c index 5343e08..7a33746 100644 --- a/src/gamma-coopgamma.c +++ b/src/gamma-coopgamma.c @@ -278,7 +278,7 @@ coopgamma_start(struct gamma_state *state) /* Get total size of the ramps */ switch (info.depth) { -#define X(SUFFIX, TYPE, MAX, DEPTH)\ +#define X(SUFFIX, RAMPS, TYPE, MAX, DEPTH)\ case DEPTH:\ crtc->rampsize = sizeof(TYPE);\ break @@ -310,7 +310,7 @@ coopgamma_start(struct gamma_state *state) #define float f #define double d switch (info.depth) { -#define X(SUFFIX, TYPE, MAX, DEPTH)\ +#define X(SUFFIX, RAMPS, TYPE, MAX, DEPTH)\ case DEPTH:\ r = libcoopgamma_ramps_initialise(&crtc->filter.ramps.SUFFIX);\ if (r < 0) {\ @@ -505,7 +505,7 @@ coopgamma_apply(struct gamma_state *state, const struct colour_setting *setting, /* Otherwise, create calculate the ramps */ memcpy(filter->ramps.u8.red, state->crtcs[i].plain_ramps.u8.red, state->crtcs[i].rampsize); switch (filter->depth) { -#define X(SUFFIX, TYPE, MAX, DEPTH)\ +#define X(SUFFIX, RAMPS, TYPE, MAX, DEPTH)\ case DEPTH:\ fill_ramps_##SUFFIX((void *)(filter->ramps.u8.red),\ (void *)(filter->ramps.u8.green),\ diff --git a/src/gamma-randr.c b/src/gamma-randr.c index 9438d17..1132960 100644 --- a/src/gamma-randr.c +++ b/src/gamma-randr.c @@ -19,244 +19,11 @@ */ #include "common.h" -#include <xcb/xcb.h> -#include <xcb/randr.h> -#if defined(__GNUC__) -# pragma GCC diagnostic ignored "-Waggregate-return" -#endif - - -#define RANDR_VERSION_MAJOR 1 -#define RANDR_VERSION_MINOR 3 - - -struct randr_crtc_state { - xcb_randr_crtc_t crtc; - unsigned int ramp_size; - uint16_t *saved_ramps; -}; - - -struct gamma_state { - xcb_connection_t *conn; - xcb_screen_t *screen; - int preferred_screen; - int screen_num; - int crtc_num_count; - int *crtc_num; - unsigned int crtc_count; - struct randr_crtc_state *crtcs; -}; - static int randr_create(struct gamma_state **state_out) { - xcb_randr_query_version_cookie_t ver_cookie; - xcb_randr_query_version_reply_t *ver_reply; - xcb_generic_error_t *error; - struct gamma_state *state; - int ec; - - /* Initialize state */ - state = *state_out = emalloc(sizeof(**state_out)); - state->screen_num = -1; - state->crtc_num = NULL; - state->crtc_num_count = 0; - state->crtc_count = 0; - state->crtcs = NULL; - - /* Open X server connection */ - state->conn = xcb_connect(NULL, &state->preferred_screen); - - /* Query RandR version */ - ver_cookie = xcb_randr_query_version(state->conn, RANDR_VERSION_MAJOR, RANDR_VERSION_MINOR); - ver_reply = xcb_randr_query_version_reply(state->conn, ver_cookie, &error); - - /* TODO What does it mean when both error and ver_reply is NULL? - Apparently, we have to check both to avoid seg faults. */ - if (error || ver_reply == NULL) { - ec = (error != 0) ? error->error_code : -1; - weprintf(_("`%s' returned error %i."), "RANDR Query Version", ec); - goto fail; - } - - if (ver_reply->major_version != RANDR_VERSION_MAJOR || - ver_reply->minor_version < RANDR_VERSION_MINOR) { - weprintf(_("Unsupported RANDR version (%u.%u)."), ver_reply->major_version, ver_reply->minor_version); - free(ver_reply); - goto fail; - } - - free(ver_reply); - - return 0; - -fail: - xcb_disconnect(state->conn); - free(state); - *state_out = NULL; - return -1; -} - - -static int -randr_start(struct gamma_state *state) -{ - xcb_generic_error_t *error; - const xcb_setup_t *setup; - xcb_screen_iterator_t iter; - int i, screen_num; - xcb_randr_get_screen_resources_current_cookie_t res_cookie; - xcb_randr_get_screen_resources_current_reply_t *res_reply; - xcb_randr_crtc_t *crtcs; - - screen_num = state->screen_num; - if (screen_num < 0) - screen_num = state->preferred_screen; - - /* Get screen */ - setup = xcb_get_setup(state->conn); - iter = xcb_setup_roots_iterator(setup); - state->screen = NULL; - - for (i = 0; iter.rem > 0; i++) { - if (i == screen_num) { - state->screen = iter.data; - break; - } - xcb_screen_next(&iter); - } - - if (!state->screen) { - weprintf(_("Screen %i could not be found."), screen_num); - return -1; - } - - /* Get list of CRTCs for the screen */ - res_cookie = xcb_randr_get_screen_resources_current(state->conn, state->screen->root); - res_reply = xcb_randr_get_screen_resources_current_reply(state->conn, res_cookie, &error); - - if (error) { - weprintf(_("`%s' returned error %i."), "RANDR Get Screen Resources Current", error->error_code); - return -1; - } - - state->crtc_count = res_reply->num_crtcs; - state->crtcs = ecalloc(state->crtc_count, sizeof(struct randr_crtc_state)); - - crtcs = xcb_randr_get_screen_resources_current_crtcs(res_reply); - - /* Save CRTC identifier in state */ - for (i = 0; i < state->crtc_count; i++) - state->crtcs[i].crtc = crtcs[i]; - - free(res_reply); - - /* Save size and gamma ramps of all CRTCs. - Current gamma ramps are saved so we can restore them - at program exit. */ - for (i = 0; i < state->crtc_count; i++) { - xcb_randr_crtc_t crtc = state->crtcs[i].crtc; - xcb_randr_get_crtc_gamma_size_cookie_t gamma_size_cookie; - xcb_randr_get_crtc_gamma_size_reply_t *gamma_size_reply; - xcb_randr_get_crtc_gamma_cookie_t gamma_get_cookie; - xcb_randr_get_crtc_gamma_reply_t *gamma_get_reply; - uint16_t *gamma_r, *gamma_g, *gamma_b; - unsigned int ramp_size; - - /* Request size of gamma ramps */ - gamma_size_cookie = xcb_randr_get_crtc_gamma_size(state->conn, crtc); - gamma_size_reply = xcb_randr_get_crtc_gamma_size_reply(state->conn, gamma_size_cookie, &error); - - if (error) { - weprintf(_("`%s' returned error %i."), "RANDR Get CRTC Gamma Size", error->error_code); - return -1; - } - - ramp_size = gamma_size_reply->size; - state->crtcs[i].ramp_size = ramp_size; - - free(gamma_size_reply); - - if (ramp_size == 0) { - weprintf(_("Gamma ramp size too small: %zu"), (size_t)ramp_size); - return -1; - } - - /* Request current gamma ramps */ - gamma_get_cookie = xcb_randr_get_crtc_gamma(state->conn, crtc); - gamma_get_reply = xcb_randr_get_crtc_gamma_reply(state->conn, gamma_get_cookie, &error); - - if (error) { - weprintf(_("`%s' returned error %i."), "RANDR Get CRTC Gamma", error->error_code); - return -1; - } - - gamma_r = xcb_randr_get_crtc_gamma_red(gamma_get_reply); - gamma_g = xcb_randr_get_crtc_gamma_green(gamma_get_reply); - gamma_b = xcb_randr_get_crtc_gamma_blue(gamma_get_reply); - - /* Allocate space for saved gamma ramps */ - state->crtcs[i].saved_ramps = emalloc(3 * ramp_size * sizeof(uint16_t)); - - /* Copy gamma ramps into CRTC state */ - memcpy(&state->crtcs[i].saved_ramps[0 * ramp_size], gamma_r, ramp_size * sizeof(uint16_t)); - memcpy(&state->crtcs[i].saved_ramps[1 * ramp_size], gamma_g, ramp_size * sizeof(uint16_t)); - memcpy(&state->crtcs[i].saved_ramps[2 * ramp_size], gamma_b, ramp_size * sizeof(uint16_t)); - - free(gamma_get_reply); - } - - return 0; -} - - -static void -randr_restore(struct gamma_state *state) -{ - xcb_generic_error_t *error; - int i; - - /* Restore CRTC gamma ramps */ - for (i = 0; i < state->crtc_count; i++) { - xcb_randr_crtc_t crtc = state->crtcs[i].crtc; - - unsigned int ramp_size = state->crtcs[i].ramp_size; - uint16_t *gamma_r = &state->crtcs[i].saved_ramps[0*ramp_size]; - uint16_t *gamma_g = &state->crtcs[i].saved_ramps[1*ramp_size]; - uint16_t *gamma_b = &state->crtcs[i].saved_ramps[2*ramp_size]; - - /* Set gamma ramps */ - xcb_void_cookie_t gamma_set_cookie = - xcb_randr_set_crtc_gamma_checked(state->conn, crtc, - ramp_size, gamma_r, - gamma_g, gamma_b); - error = xcb_request_check(state->conn, gamma_set_cookie); - - if (error) { - weprintf(_("`%s' returned error %i."), "RANDR Set CRTC Gamma", error->error_code); - weprintf(_("Unable to restore CRTC %i."), i); - } - } -} - - -static void -randr_free(struct gamma_state *state) -{ - int i; - - /* Free CRTC state */ - for (i = 0; i < state->crtc_count; i++) - free(state->crtcs[i].saved_ramps); - free(state->crtcs); - free(state->crtc_num); - - /* Close connection */ - xcb_disconnect(state->conn); - - free(state); + return direct_create(state_out, LIBGAMMA_METHOD_X_RANDR, "randr"); } @@ -266,12 +33,9 @@ randr_print_help(FILE *f) fputs(_("Adjust gamma ramps with the X RANDR extension.\n"), f); fputs("\n", f); - /* TRANSLATORS: RANDR help output - left column must not be translated */ - fputs(_(" screen=N\t\tX screen to apply adjustments to\n" - " crtc=N\tList of comma separated CRTCs to apply" - " adjustments to\n"), - f); + /* TRANSLATORS: RANDR help output left column must not be translated */ + fputs(_(" screen=N X screen to apply adjustments to\n"), f); + fputs(_(" crtc=N List of comma-separated CRTCs to apply adjustments to\n"), f); fputs("\n", f); } @@ -280,140 +44,21 @@ static int randr_set_option(struct gamma_state *state, const char *key, const char *value) { if (!strcasecmp(key, "screen")) { - state->screen_num = atoi(value); + return direct_set_partitions(state, key, value); } else if (!strcasecmp(key, "crtc")) { - char *tail; - int i, parsed; - - /* Check how many crtcs are configured */ - const char *local_value = value; - if (!*local_value || !strcasecmp(local_value, "all")) { - state->crtc_num_count = 0; - free(state->crtc_num); - state->crtc_num = NULL; - return 0; - } - for (;;) { - errno = 0; - parsed = strtol(local_value, &tail, 0); - if (!parsed && (errno || tail == local_value)) { - weprintf(_("Unable to read screen number: `%s'."), value); - return -1; - } - state->crtc_num_count += 1; - local_value = tail; - - if (*local_value == ',') - local_value += 1; - else if (!*local_value) - break; - } - - /* Configure all given crtcs */ - state->crtc_num = calloc(state->crtc_num_count, sizeof(int)); - local_value = value; - for (i = 0; i < state->crtc_num_count; i++) { - errno = 0; - parsed = strtol(local_value, &tail, 0); - if (parsed == 0 && (errno != 0 || tail == local_value)) - return -1; - state->crtc_num[i] = parsed; - local_value = tail; - - if (*local_value == ',') - local_value += 1; - else if (!*local_value) - break; - } - } else if (strcasecmp(key, "preserve") == 0) { - weprintf(_("Parameter `%s' is now always on; use the `%s' command-line option to disable."), key, "-P"); + return direct_set_crtcs(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; } - - return 0; -} - - -static int -randr_apply_for_crtc(struct gamma_state *state, int crtc_num, const struct colour_setting *setting, int preserve) -{ - xcb_randr_crtc_t crtc; - xcb_void_cookie_t gamma_set_cookie; - xcb_generic_error_t *error; - unsigned int i, ramp_size; - uint16_t *gamma_ramps, *gamma_r, *gamma_g, *gamma_b, value; - - if (crtc_num >= state->crtc_count || crtc_num < 0) { - if (state->crtc_count > 1) - weprintf(_("CRTC %i does not exist, valid CRTCs are [0, %i]."), crtc_num, (int)state->crtc_count - 1); - else - weprintf(_("CRTC %i does not exist, only CRTC 0 exists."), crtc_num); - - return -1; - } - - crtc = state->crtcs[crtc_num].crtc; - ramp_size = state->crtcs[crtc_num].ramp_size; - - /* Create new gamma ramps */ - gamma_ramps = emalloc(3 * ramp_size * sizeof(uint16_t)); - - gamma_r = &gamma_ramps[0 * ramp_size]; - gamma_g = &gamma_ramps[1 * ramp_size]; - gamma_b = &gamma_ramps[2 * ramp_size]; - - if (preserve) { - /* Initialize gamma ramps from saved state */ - memcpy(gamma_ramps, state->crtcs[crtc_num].saved_ramps, - 3 * ramp_size * sizeof(uint16_t)); - } else { - /* Initialize gamma ramps to pure state */ - for (i = 0; i < ramp_size; i++) { - value = (double)i / (ramp_size - 1) * UINT16_MAX; - gamma_r[i] = value; - gamma_g[i] = value; - gamma_b[i] = value; - } - } - - fill_ramps_u16(gamma_r, gamma_g, gamma_b, ramp_size, ramp_size, ramp_size, setting); - - /* Set new gamma ramps */ - gamma_set_cookie = xcb_randr_set_crtc_gamma_checked(state->conn, crtc, ramp_size, gamma_r, gamma_g, gamma_b); - error = xcb_request_check(state->conn, gamma_set_cookie); - - if (error) { - weprintf(_("`%s' returned error %i."), "RANDR Set CRTC Gamma", error->error_code); - free(gamma_ramps); - return -1; - } - - free(gamma_ramps); - - return 0; -} - - -static int -randr_apply(struct gamma_state *state, const struct colour_setting *setting, int preserve) -{ - int i; - - /* If no CRTC numbers have been specified, set temperature on all CRTCs. */ - if (!state->crtc_num_count) { - for (i = 0; i < state->crtc_count; i++) - if (randr_apply_for_crtc(state, i, setting, preserve) < 0) - return -1; - } else { - for (i = 0; i < state->crtc_num_count; ++i) - if (randr_apply_for_crtc(state, state->crtc_num[i], setting, preserve) < 0) - return -1; - } - - return 0; } +#define randr_start direct_start +#define randr_apply direct_apply +#define randr_restore direct_restore +#define randr_free direct_free const struct gamma_method randr_gamma_method = GAMMA_METHOD_INIT("randr", 1, 0, randr); diff --git a/src/gamma-vidmode.c b/src/gamma-vidmode.c index 7c5321a..e38de97 100644 --- a/src/gamma-vidmode.c +++ b/src/gamma-vidmode.c @@ -19,93 +19,11 @@ */ #include "common.h" -#include <X11/Xlib.h> -#include <X11/extensions/xf86vmode.h> - - -struct gamma_state { - Display *display; - int screen_num; - int ramp_size; - uint16_t *saved_ramps; -}; - static int vidmode_create(struct gamma_state **state_out) { - struct gamma_state *state; - - state = *state_out = emalloc(sizeof(**state_out)); - state->screen_num = -1; - state->saved_ramps = NULL; - - state->display = XOpenDisplay(NULL); - if (!state->display) { - weprintf(_("X request failed: %s"), "XOpenDisplay"); - return -1; - } - - return 0; -} - - -static int -vidmode_start(struct gamma_state *state) -{ - int r; - int screen_num = state->screen_num; - int major, minor; - uint16_t *gamma_r, *gamma_g, *gamma_b; - - if (screen_num < 0) - screen_num = DefaultScreen(state->display); - state->screen_num = screen_num; - - /* Query extension version */ - r = XF86VidModeQueryVersion(state->display, &major, &minor); - if (!r) { - weprintf(_("X request failed: %s"), "XF86VidModeQueryVersion"); - return -1; - } - - /* Request size of gamma ramps */ - r = XF86VidModeGetGammaRampSize(state->display, state->screen_num, &state->ramp_size); - if (!r) { - weprintf(_("X request failed: %s"), "XF86VidModeGetGammaRampSize"); - return -1; - } - - if (!state->ramp_size) { - weprintf(_("Gamma ramp size too small: %zu"), (size_t)state->ramp_size); - return -1; - } - - /* Allocate space for saved gamma ramps */ - state->saved_ramps = emalloc(3 * state->ramp_size * sizeof(uint16_t)); - - gamma_r = &state->saved_ramps[0 * state->ramp_size]; - gamma_g = &state->saved_ramps[1 * state->ramp_size]; - gamma_b = &state->saved_ramps[2 * state->ramp_size]; - - /* Save current gamma ramps so we can restore them at program exit. */ - r = XF86VidModeGetGammaRamp(state->display, state->screen_num, - state->ramp_size, gamma_r, gamma_g, gamma_b); - if (!r) { - weprintf(_("X request failed: %s"), "XF86VidModeGetGammaRamp"); - return -1; - } - - return 0; -} - - -static void -vidmode_free(struct gamma_state *state) -{ - free(state->saved_ramps); - XCloseDisplay(state->display); - free(state); + return direct_create(state_out, LIBGAMMA_METHOD_X_VIDMODE, "vidmode"); } @@ -115,8 +33,7 @@ vidmode_print_help(FILE *f) fputs(_("Adjust gamma ramps with the X VidMode extension.\n"), f); fputs("\n", f); - /* TRANSLATORS: VidMode help output left column must not be translated */ - fputs(_(" screen=N\t\tX screen to apply adjustments to\n"), f); + fputs(_(" screen=N X screen to apply adjustments to\n"), f); fputs("\n", f); } @@ -125,75 +42,19 @@ static int vidmode_set_option(struct gamma_state *state, const char *key, const char *value) { if (!strcasecmp(key, "screen")) { - state->screen_num = atoi(value); + return direct_set_partitions(state, key, value); } else if (!strcasecmp(key, "preserve")) { - weprintf(_("Parameter `%s' is now always on; use the `%s' command-line option to disable."), key, "-P"); + weprintf(_("Deprecated method parameter ignored: `%s'."), key); + return 0; } else { weprintf(_("Unknown method parameter: `%s'."), key); return -1; } - - return 0; -} - - -static void -vidmode_restore(struct gamma_state *state) -{ - uint16_t *gamma_r = &state->saved_ramps[0 * state->ramp_size]; - uint16_t *gamma_g = &state->saved_ramps[1 * state->ramp_size]; - uint16_t *gamma_b = &state->saved_ramps[2 * state->ramp_size]; - int r; - - /* Restore gamma ramps */ - r = XF86VidModeSetGammaRamp(state->display, state->screen_num, - state->ramp_size, gamma_r, gamma_g, gamma_b); - if (!r) - weprintf(_("X request failed: %s"), "XF86VidModeSetGammaRamp"); -} - - -static int -vidmode_apply(struct gamma_state *state, const struct colour_setting *setting, int preserve) -{ - uint16_t value, *gamma_ramps, *gamma_r, *gamma_g, *gamma_b; - int r, i; - - /* Create new gamma ramps */ - gamma_ramps = emalloc(3 * state->ramp_size * sizeof(uint16_t)); - - gamma_r = &gamma_ramps[0 * state->ramp_size]; - gamma_g = &gamma_ramps[1 * state->ramp_size]; - gamma_b = &gamma_ramps[2 * state->ramp_size]; - - if (preserve) { - /* Initialize gamma ramps from saved state */ - memcpy(gamma_ramps, state->saved_ramps, 3 * state->ramp_size * sizeof(uint16_t)); - } else { - /* Initialize gamma ramps to pure state */ - for (i = 0; i < state->ramp_size; i++) { - value = (double)i / (state->ramp_size - 1) * UINT16_MAX; - gamma_r[i] = value; - gamma_g[i] = value; - gamma_b[i] = value; - } - } - - fill_ramps_u16(gamma_r, gamma_g, gamma_b, state->ramp_size, state->ramp_size, state->ramp_size, setting); - - /* Set new gamma ramps */ - r = XF86VidModeSetGammaRamp(state->display, state->screen_num, - state->ramp_size, gamma_r, gamma_g, gamma_b); - if (!r) { - weprintf(_("X request failed: %s"), "XF86VidModeSetGammaRamp"); - free(gamma_ramps); - return -1; - } - - free(gamma_ramps); - - return 0; } +#define vidmode_start direct_start +#define vidmode_apply direct_apply +#define vidmode_restore direct_restore +#define vidmode_free direct_free const struct gamma_method vidmode_gamma_method = GAMMA_METHOD_INIT("vidmode", 1, 0, vidmode); |