/* 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 */ #ifdef HAVE_CONFIG_H # include "config.h" #endif #include #include #ifndef WINVER # define WINVER 0x0500 #endif #include #include #ifdef ENABLE_NLS # include # define _(s) gettext(s) #else # define _(s) s #endif #include "gamma-w32gdi.h" #include "colorramp.h" #define GAMMA_RAMP_SIZE 256 #define MAX_ATTEMPTS 10 typedef struct { WORD *saved_ramps; } w32gdi_state_t; static int w32gdi_init(w32gdi_state_t **state) { *state = malloc(sizeof(w32gdi_state_t)); if (state == NULL) return -1; w32gdi_state_t *s = *state; s->saved_ramps = NULL; return 0; } static int w32gdi_start(w32gdi_state_t *state, program_mode_t mode) { BOOL r; /* Open device context */ HDC hDC = GetDC(NULL); if (hDC == NULL) { fputs(_("Unable to open device context.\n"), stderr); return -1; } /* Check support for gamma ramps */ int cmcap = GetDeviceCaps(hDC, COLORMGMTCAPS); if (cmcap != CM_GAMMA_RAMP) { fputs(_("Display device does not support gamma ramps.\n"), stderr); return -1; } /* Allocate space for saved gamma ramps */ state->saved_ramps = malloc(3*GAMMA_RAMP_SIZE*sizeof(WORD)); if (state->saved_ramps == NULL) { perror("malloc"); ReleaseDC(NULL, hDC); return -1; } /* Save current gamma ramps so we can restore them at program exit */ r = GetDeviceGammaRamp(hDC, state->saved_ramps); if (!r) { fputs(_("Unable to save current gamma ramp.\n"), stderr); ReleaseDC(NULL, hDC); return -1; } /* Release device context */ ReleaseDC(NULL, hDC); return 0; } static void w32gdi_free(w32gdi_state_t *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(w32gdi_state_t *state, const char *key, const char *value) { if (strcasecmp(key, "preserve") == 0) { fprintf(stderr, _("Parameter `%s` is now always on; " " Use the `%s` command-line option" " to disable.\n"), key, "-P"); } else { fprintf(stderr, _("Unknown method parameter: `%s'.\n"), key); return -1; } return 0; } static void w32gdi_restore(w32gdi_state_t *state) { /* Open device context */ HDC hDC = GetDC(NULL); if (hDC == NULL) { fputs(_("Unable to open device context.\n"), stderr); return; } /* Restore gamma ramps */ BOOL r = FALSE; for (int i = 0; i < MAX_ATTEMPTS && !r; 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. */ r = SetDeviceGammaRamp(hDC, state->saved_ramps); } if (!r) fputs(_("Unable to restore gamma ramps.\n"), stderr); /* Release device context */ ReleaseDC(NULL, hDC); } static int w32gdi_set_temperature( w32gdi_state_t *state, const color_setting_t *setting, int preserve) { BOOL r; /* Open device context */ HDC hDC = GetDC(NULL); if (hDC == NULL) { fputs(_("Unable to open device context.\n"), stderr); return -1; } /* Create new gamma ramps */ WORD *gamma_ramps = malloc(3*GAMMA_RAMP_SIZE*sizeof(WORD)); if (gamma_ramps == NULL) { perror("malloc"); ReleaseDC(NULL, hDC); return -1; } WORD *gamma_r = &gamma_ramps[0*GAMMA_RAMP_SIZE]; WORD *gamma_g = &gamma_ramps[1*GAMMA_RAMP_SIZE]; WORD *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 (int i = 0; i < GAMMA_RAMP_SIZE; i++) { WORD value = (double)i/GAMMA_RAMP_SIZE * (UINT16_MAX+1); 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 */ r = FALSE; for (int i = 0; i < MAX_ATTEMPTS && !r; 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. */ r = SetDeviceGammaRamp(hDC, gamma_ramps); } if (!r) { fputs(_("Unable to set gamma ramps.\n"), stderr); free(gamma_ramps); ReleaseDC(NULL, hDC); return -1; } free(gamma_ramps); /* Release device context */ ReleaseDC(NULL, hDC); return 0; } const gamma_method_t w32gdi_gamma_method = { "wingdi", 1, (gamma_method_init_func *)w32gdi_init, (gamma_method_start_func *)w32gdi_start, (gamma_method_free_func *)w32gdi_free, (gamma_method_print_help_func *)w32gdi_print_help, (gamma_method_set_option_func *)w32gdi_set_option, (gamma_method_restore_func *)w32gdi_restore, (gamma_method_set_temperature_func *)w32gdi_set_temperature };