/* gamma-w32gdi.c -- Windows GDI gamma adjustment source
This file is part of Redshift.
Redshift 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 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. If not, see .
Copyright (c) 2010-2017 Jon Lund Steffensen
Copyright (c) 2025 Mattias Andrée
*/
#ifndef WINVER
# define WINVER 0x0500
#endif
#include "common.h"
#include
#define GAMMA_RAMP_SIZE 256
#define MAX_ATTEMPTS 10
struct gamma_state {
WORD *saved_ramps;
};
static int
w32gdi_init(struct gamma_state **state)
{
*state = emalloc(sizeof(**state));
(*state)->saved_ramps = NULL;
return 0;
}
static int
w32gdi_start(struct gamma_state *state, program_mode_t mode)
{
HDC hDC;
int cmcap;
/* Open device context */
hDC = GetDC(NULL);
if (!hDC) {
weprintf(_("Unable to open device context.\n"));
return -1;
}
/* Check support for gamma ramps */
cmcap = GetDeviceCaps(hDC, COLORMGMTCAPS);
if (cmcap != CM_GAMMA_RAMP) {
weprintf(_("Display device does not support gamma ramps.\n"));
return -1;
}
/* Allocate space for saved gamma ramps */
state->saved_ramps = emalloc(3 * GAMMA_RAMP_SIZE * sizeof(WORD));
/* Save current gamma ramps so we can restore them at program exit */
if (!GetDeviceGammaRamp(hDC, state->saved_ramps)) {
weprintf(_("Unable to save current gamma ramp.\n"));
ReleaseDC(NULL, hDC);
return -1;
}
/* Release device context */
ReleaseDC(NULL, hDC);
return 0;
}
static void
w32gdi_free(struct gamma_state *state)
{
/* Free saved ramps */
free(state->saved_ramps);
free(state);
}
static void
w32gdi_print_help(FILE *f)
{
fputs(_("Adjust gamma ramps with the Windows GDI.\n"), f);
fputs("\n", f);
}
static int
w32gdi_set_option(struct gamma_state *state, const char *key, const char *value)
{
if (!strcasecmp(key, "preserve")) {
weprintf(_("Parameter `%s` is now always on; Use the `%s` "
"command-line option to disable.\n"), key, "-P");
} else {
weprintf(_("Unknown method parameter: `%s'.\n"), key);
return -1;
}
return 0;
}
static void
w32gdi_restore(struct gamma_state *state)
{
HDC hDC;
int i;
/* Open device context */
hDC = GetDC(NULL);
if (!hDC) {
fputs(_("Unable to open device context.\n"), stderr);
return;
}
/* Restore gamma ramps */
for (i = 0; i < MAX_ATTEMPTS; i++) {
/* We retry a few times before giving up because some
buggy drivers fail on the first invocation of
SetDeviceGammaRamp just to succeed on the second. */
if (SetDeviceGammaRamp(hDC, state->saved_ramps))
goto done;
}
weprintf(_("Unable to restore gamma ramps.\n"));
done:
/* Release device context */
ReleaseDC(NULL, hDC);
}
static int
w32gdi_set_temperature(
struct gamma_state *state, const color_setting_t *setting, int preserve)
{
WORD *gamma_ramps, *gamma_r, *gamma_b, *gamma_g, value;
HDC hDC;
int i;
/* Open device context */
hDC = GetDC(NULL);
if (!hDC) {
weprintf(_("Unable to open device context.\n"));
return -1;
}
/* Create new gamma ramps */
gamma_ramps = emalloc(3 * GAMMA_RAMP_SIZE * sizeof(WORD));
gamma_r = &gamma_ramps[0 * GAMMA_RAMP_SIZE];
gamma_g = &gamma_ramps[1 * GAMMA_RAMP_SIZE];
gamma_b = &gamma_ramps[2 * GAMMA_RAMP_SIZE];
if (preserve) {
/* Initialize gamma ramps from saved state */
memcpy(gamma_ramps, state->saved_ramps, 3 * GAMMA_RAMP_SIZE * sizeof(WORD));
} else {
/* Initialize gamma ramps to pure state */
for (i = 0; i < GAMMA_RAMP_SIZE; i++) {
value = (double)i / (GAMMA_RAMP_SIZE - 1) * UINT16_MAX;
gamma_r[i] = value;
gamma_g[i] = value;
gamma_b[i] = value;
}
}
colorramp_fill_u16(gamma_r, gamma_g, gamma_b, GAMMA_RAMP_SIZE,
GAMMA_RAMP_SIZE, GAMMA_RAMP_SIZE, setting);
/* Set new gamma ramps */
for (i = 0; i < MAX_ATTEMPTS; i++) {
/* We retry a few times before giving up because some
buggy drivers fail on the first invocation of
SetDeviceGammaRamp just to succeed on the second. */
if (SetDeviceGammaRamp(hDC, gamma_ramps))
goto done;
}
weprintf(_("Unable to set gamma ramps.\n"));
free(gamma_ramps);
ReleaseDC(NULL, hDC);
return -1;
done:
free(gamma_ramps);
/* Release device context */
ReleaseDC(NULL, hDC);
return 0;
}
const struct gamma_method w32gdi_gamma_method = {
"wingdi", 1,
&w32gdi_init,
&w32gdi_start,
&w32gdi_free,
&w32gdi_print_help,
&w32gdi_set_option,
&w32gdi_restore,
&w32gdi_set_temperature
};