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.c86
1 files changed, 73 insertions, 13 deletions
diff --git a/src/backend-direct.c b/src/backend-direct.c
index 62fe2fa..de58548 100644
--- a/src/backend-direct.c
+++ b/src/backend-direct.c
@@ -86,6 +86,22 @@ struct crtc_state {
};
+/**
+ * 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)
@@ -162,11 +178,11 @@ struct gamma_state {
size_t npartitions;
/**
- * Indices of selected CRTCs
+ * Selected CRTC numbers
*
* Deallocated by when no longer needed
*/
- size_t *selected_crtcs;
+ struct crtc_number *selected_crtcs;
/**
* Selected EDIDs
@@ -253,17 +269,22 @@ direct_print_help(FILE *f, int method)
if (caps.multiple_sites)
fprintf(f, " display=%s %s\n", _("NAME "), _("Display server instance to apply adjustments to"));
- if (caps.multiple_partitions && caps.partitions_are_graphics_cards)
+ if (caps.multiple_partitions && caps.partitions_are_graphics_cards) {
/* TRANSLATORS: "N" represents "ordinal"; right-pad with spaces to preserve display width */
fprintf(f, " card=%s %s\n", _("N "), _("Graphics card to apply adjustments to"));
- else if (caps.multiple_partitions)
+ } else if (caps.multiple_partitions) {
+ /* TRANSLATORS: "N" represents "ordinal"; right-pad with spaces to preserve display width */
fprintf(f, " screen=%s %s\n", _("N "), _("X screen to apply adjustments to"));
+ }
- if (caps.multiple_crtcs)
+ if (caps.multiple_crtcs) {
+ /* TRANSLATORS: "N" represents "number"; right-pad with spaces to preserve display width */
fprintf(f, " 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))
+ }
+ if (caps.multiple_crtcs && (caps.crtc_information & LIBGAMMA_CRTC_INFO_EDID)) {
fprintf(f, " 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)
fprintf(f, "\n");
@@ -356,12 +377,25 @@ direct_set_crtcs(struct gamma_state *state, const char *key, const char *value)
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) {
+ 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;
@@ -464,6 +498,25 @@ direct_start(struct gamma_state *state)
}
}
+ /* 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;
@@ -522,10 +575,11 @@ direct_start(struct gamma_state *state)
resolved_edids[i].edid, resolved_edids[i].crtc,
resolved_edids[i].partition);
}
- num = state->ncrtcs + 1U;
- state->selected_crtcs = erealloc(state->selected_crtcs, num * sizeof(*state->crtcs));
- num = state->partitions[resolved_edids[i].partition].crtc_num_offset;
- state->selected_crtcs[state->ncrtcs++] = num + resolved_edids[i].crtc;
+ 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;
}
}
@@ -546,13 +600,19 @@ direct_start(struct gamma_state *state)
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];
+ 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."));