diff options
Diffstat (limited to 'src/redshift.c')
-rw-r--r-- | src/redshift.c | 426 |
1 files changed, 203 insertions, 223 deletions
diff --git a/src/redshift.c b/src/redshift.c index 04c6c77..77cabb8 100644 --- a/src/redshift.c +++ b/src/redshift.c @@ -1,4 +1,5 @@ -/* redshift-ng - Automatically adjust display colour temperature according the Sun +/*- + * 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> @@ -18,42 +19,54 @@ */ #include "common.h" -/* poll.h is not available on Windows but there is no Windows location provider - * using polling. On Windows, we just define some stubs to make things compile. + +/** + * The number of milliseconds to sleep normally between colour updates */ -#ifndef WINDOWS -# include <poll.h> -#else -# define POLLIN 0 -struct pollfd { - int fd; - short events; - short revents; -}; -int poll(struct pollfd *fds, int nfds, int timeout) { abort(); } -#endif +#define SLEEP_DURATION 5000U +/** + * The number of milliseconds to sleep between each step during + * fade between large colour settings + */ +#define SLEEP_DURATION_SHORT 25U -/* Duration of sleep between screen updates (milliseconds). */ -#define SLEEP_DURATION 5000 -#define SLEEP_DURATION_SHORT 100 +/** + * The fade time, for when making large changes in colour + * settings, divided by `SLEEP_DURATION_SHORT` + */ +#define FADE_LENGTH 160U -/* Length of fade in numbers of short sleep durations. */ -#define FADE_LENGTH 40 +/** + * The user's current geographical location + */ +static struct location location; -#if defined(__GNUC__) -# pragma GCC diagnostic push -# pragma GCC diagnostic ignored "-Wfloat-equal" -#endif -static int -exact_eq(double a, double b) -{ - return a == b; -} -#if defined(__GNUC__) -# pragma GCC diagnostic pop -#endif +/** + * Whether the location provider is available + */ +static int location_available; + +/** + * State of location provider, `NULL` if none + */ +static LOCATION_STATE *provider_state; + +/** + * Locaiton provider functions + */ +static const struct location_provider *provider; + +/** + * State of the gamma ramp adjustment method, `NULL` if none (print mode) + */ +static GAMMA_STATE *method_state; + +/** + * Gamma ramp adjustment functions + */ +static const struct gamma_method *method; /** @@ -124,14 +137,12 @@ print_period(enum period period, double day_level) * Get the current period of day and the colour settings * applicable to the current time of the day * - * @param location Geographical user location * @param colour_out Output parameter for the colour settings * @param period_out Output parameter for the period of the day * @param day_level_out Output parameter for the dayness level */ static void -get_colour_settings(const struct location *location, struct colour_setting *colour_out, - enum period *period_out, double *day_level_out) +get_colour_settings(struct colour_setting *colour_out, enum period *period_out, double *day_level_out) { time_t time_offset; double t, elevation; @@ -148,7 +159,7 @@ get_colour_settings(const struct location *location, struct colour_setting *colo *day_level_out = fma(t, scheme.time.periods->diff_over_duration, scheme.time.periods->day_level); } else if (scheme.type == SOLAR_SCHEME) { - if (libred_solar_elevation(location->latitude, location->longitude, &elevation)) + if (libred_solar_elevation(location.latitude, location.longitude, &elevation)) eprintf("libred_solar_elevation:"); if (verbose) { /* TRANSLATORS: Append degree symbol if possible. */ @@ -158,7 +169,7 @@ get_colour_settings(const struct location *location, struct colour_setting *colo } else { /* Static scheme, no dayness-level or peroid; day_settings == nigh_settings, use either */ - *day_level_out = NAN; + *day_level_out = FNAN; *period_out = PERIOD_NONE; *colour_out = day_settings; return; @@ -200,201 +211,167 @@ ease_fade(double t) } -/* Run continual mode loop - This is the main loop of the continual mode which keeps track of the - current time and continuously updates the screen to the appropriate - colour temperature. */ +#ifndef WINDOWS /* we don't have poll on Windows, but neither do with have any dynamic location providers */ +/** + * Get the current location + * + * The function will return once any of the following has occured: + * - the specified timeout duration has elapsed, + * - a location message has been received (could be location or error), or + * - a signal(7) was received + * + * @param timeout The number of milliseconds to wait before + * returning without updating the location + * @param location_fd File descriptor to wait on to receive input event + */ static void -run_continual_mode(const struct location_provider *provider, LOCATION_STATE *location_state, - const struct gamma_method *method, GAMMA_STATE *method_state, int use_fade, - int preserve_gamma) +pull_location(unsigned int timeout, int location_fd) { - int done = 0; - int prev_disabled = 1; - int disabled = 0; - int location_available = 1; - struct colour_setting fade_start_interp; - struct colour_setting prev_target_interp; - struct colour_setting interp; - struct location loc; - - /* Short fade parameters */ - int fade_length = 0; - int fade_time = 0; - - /* Save previous parameters so we can avoid printing status updates if - * the values did not change. */ - enum period prev_period = PERIOD_NONE; - - install_signal_handlers(); - - /* Previous target colour setting and current actual colour setting. - * Actual colour setting takes into account the current colour fade. */ - prev_target_interp = COLOUR_SETTING_NEUTRAL; - - interp = COLOUR_SETTING_NEUTRAL; - - loc = (struct location){NAN, NAN}; - if (scheme.type == SOLAR_SCHEME) { - weprintf(_("Waiting for initial location to become available...")); + struct pollfd pollfds[1]; + struct location new_location; + int r, new_available; + + /* Await new location information */ + pollfds[0].fd = location_fd; + pollfds[0].events = POLLIN; + r = poll(pollfds, 1, (int)timeout); + if (r < 0) { +#ifndef WINDOWS + if (errno == EINTR) + return; +#endif + weprintf("poll:"); + eprintf(_("Unable to get location from provider.")); + } else if (!r) { + return; + } - if (get_location(provider, location_state, -1, &loc) < 0) - eprintf(_("Unable to get location from provider.")); + /* Get new location and availability information */ + if (provider->fetch(provider_state, &new_location, &new_available) < 0) + eprintf(_("Unable to get location from provider.")); + if (new_available < location_available) { + weprintf(_("Location is temporarily unavailable; using previous" + " location until it becomes available...")); + location_available = 0; + return; + } - if (!location_is_valid(&loc)) + /* Store and announce new location */ + if (new_available > location_available || + !exact_eq(new_location.latitude, location.latitude) || + !exact_eq(new_location.longitude, location.longitude)) { + location_available = 1; + location = new_location; + print_location(&location); + if (!location_is_valid(&location)) eprintf(_("Invalid location returned from provider.")); - - print_location(&loc); } +} +#endif - if (verbose) { - printf(_("Color temperature: %luK\n"), interp.temperature); - printf(_("Brightness: %.2f\n"), interp.brightness); - } - /* Continuously adjust colour temperature */ - for (;;) { - enum period period; - double day_level; - struct colour_setting target_interp; - int delay, loc_fd; +/** + * Loop for `PROGRAM_MODE_CONTINUAL` + */ +static void +run_continual_mode(void) +{ + enum period period, prev_period = PERIOD_NONE; + double day_level = FNAN, prev_day_level = FNAN; + int disabled = 0, prev_disabled = !disabled; + int done = 0; + struct colour_setting colour; + struct colour_setting target_colour, prev_target_colour; + struct colour_setting fade_start_colour; + unsigned int fade_length = 0; + unsigned int fade_time = 0; + unsigned int delay; + double fade_progress, eased_fade_progress; +#ifndef WINDOWS + int location_fd; +#endif - /* Check to see if disable signal was caught */ + prev_target_colour = COLOUR_SETTING_NEUTRAL; + colour = COLOUR_SETTING_NEUTRAL; + + for (;;) { + /* Act on signals */ if (disable && !done) { - disabled = !disabled; + disabled ^= 1; disable = 0; } - - /* Check to see if exit signal was caught */ if (exiting) { - if (done) + if (done) /* TODO also if already disabled (fade complete) */ break; /* On second signal stop the ongoing fade */ done = 1; disabled = 1; exiting = 0; } - - /* Print status change */ if (verbose && disabled != prev_disabled) printf(_("Status: %s\n"), disabled ? _("Disabled") : _("Enabled")); - prev_disabled = disabled; - get_colour_settings(&loc, &target_interp, &period, &day_level); - + /* Get dayness level and corresponding colour settings */ if (disabled) { period = PERIOD_NONE; - target_interp = COLOUR_SETTING_NEUTRAL; + target_colour = COLOUR_SETTING_NEUTRAL; + } else { + get_colour_settings(&target_colour, &period, &day_level); } - - if (done) - period = PERIOD_NONE; - - /* Print period if it changed during this update, - * or if we are in the transition period. In transition we - * print the progress, so we always print it in - * that case. */ - if (verbose && (period != prev_period || period == PERIOD_TRANSITION)) + if (verbose && (period != prev_period || !exact_eq(day_level, prev_day_level))) print_period(period, day_level); - - /* Activate hooks if period changed */ if (period != prev_period) run_period_change_hooks(prev_period, period); + prev_period = period; + prev_day_level = day_level; + if (verbose) { + if (prev_target_colour.temperature != target_colour.temperature) + printf(_("Color temperature: %luK\n"), target_colour.temperature); + if (!exact_eq(prev_target_colour.brightness, target_colour.brightness)) + printf(_("Brightness: %.2f\n"), target_colour.brightness); + if (memcmp(prev_target_colour.gamma, target_colour.gamma, sizeof(target_colour.gamma))) { + printf(_("Gamma: %.3f, %.3f, %.3f\n"), + target_colour.gamma[0], target_colour.gamma[1], target_colour.gamma[2]); + } + } - /* Start fade if the parameter differences are too big to apply instantly */ - if (use_fade && colour_setting_diff_is_major(&target_interp, fade_length ? &prev_target_interp : &interp)) { + /* Fade if the parameter differences are too big to apply instantly */ + if (use_fade && colour_setting_diff_is_major(&target_colour, fade_length ? &prev_target_colour : &colour)) { fade_length = FADE_LENGTH; fade_time = 0; - fade_start_interp = interp; + fade_start_colour = colour; } - - /* Handle ongoing fade */ - if (fade_length != 0) { - double frac = ++fade_time / (double)fade_length; - double alpha = CLAMP(0.0, ease_fade(frac), 1.0); - - interpolate_colour_settings(&fade_start_interp, &target_interp, alpha, &interp); - - if (fade_time > fade_length) { + if (fade_length) { + fade_progress = ++fade_time / (double)fade_length; + eased_fade_progress = ease_fade(fade_progress); + interpolate_colour_settings(&fade_start_colour, &target_colour, eased_fade_progress, &colour); + if (fade_time == fade_length) { fade_time = 0; fade_length = 0; } } else { - interp = target_interp; + colour = target_colour; } + prev_target_colour = target_colour; /* Break loop when done and final fade is over */ if (done && fade_length == 0) break; - if (verbose) { - if (prev_target_interp.temperature != target_interp.temperature) - printf(_("Color temperature: %luK\n"), target_interp.temperature); - if (!exact_eq(prev_target_interp.brightness, target_interp.brightness)) - printf(_("Brightness: %.2f\n"), target_interp.brightness); - } - - /* Adjust temperature */ - if (method->apply(method_state, &interp, preserve_gamma) < 0) + /* Adjust temperature and sleep */ + if (method->apply(method_state, &colour, preserve_gamma) < 0) eprintf(_("Temperature adjustment failed.")); - - /* Save period and target colour setting as previous */ - prev_period = period; - prev_target_interp = target_interp; - - /* Sleep length depends on whether a fade is ongoing */ delay = fade_length ? SLEEP_DURATION_SHORT : SLEEP_DURATION; - - /* Update location */ - loc_fd = scheme.type == SOLAR_SCHEME ? provider->get_fd(location_state) : -1; - - if (loc_fd >= 0) { - struct pollfd pollfds[1]; - struct location new_loc; - int r, new_available; - - /* Provider is dynamic */ - pollfds[0].fd = loc_fd; - pollfds[0].events = POLLIN; - r = poll(pollfds, 1, delay); - if (r < 0) { #ifndef WINDOWS - if (errno == EINTR) - continue; + location_fd = scheme.type == SOLAR_SCHEME ? provider->get_fd(provider_state) : -1; + if (location_fd >= 0) + pull_location(delay, location_fd); + else #endif - weprintf("poll:"); - eprintf(_("Unable to get location from provider.")); - } else if (!r) { - continue; - } - - /* Get new location and availability information */ - if (provider->fetch(location_state, &new_loc, &new_available) < 0) - eprintf(_("Unable to get location from provider.")); - - if (!new_available && new_available != location_available) { - weprintf(_("Location is temporarily unavailable; using previous" - " location until it becomes available...")); - } - - if (new_available && - (!exact_eq(new_loc.latitude, loc.latitude) || - !exact_eq(new_loc.longitude, loc.longitude) || - new_available != location_available)) { - loc = new_loc; - print_location(&loc); - } - - location_available = new_available; - - if (!location_is_valid(&loc)) - eprintf(_("Invalid location returned from provider.")); - } else { millisleep(delay); - } + /* SLEEP_DURATION_SHORT is short enough for remaining time after interruption to be ignored */ } - /* Restore saved gamma ramps */ method->restore(method_state); } @@ -403,15 +380,13 @@ int main(int argc, char *argv[]) { struct settings settings; - GAMMA_STATE *method_state = NULL; - LOCATION_STATE *location_state = NULL; - struct location loc = {NAN, NAN}; double day_level; enum period period; struct colour_setting colour; argv0 = argv[0]; + /* Set up localisation */ #ifdef ENABLE_NLS setlocale(LC_CTYPE, ""); setlocale(LC_MESSAGES, ""); @@ -419,63 +394,68 @@ main(int argc, char *argv[]) textdomain(PACKAGE); #endif + /* Get configurations and configure */ load_settings(&settings, argc, argv); - if (scheme.type == SOLAR_SCHEME) - acquire_location_provider(&settings, &location_state); - if (mode != PROGRAM_MODE_PRINT) + if (scheme.type == SOLAR_SCHEME) { + acquire_location_provider(&settings, &provider_state); + provider = settings.provider; + } + if (mode != PROGRAM_MODE_PRINT) { acquire_adjustment_method(&settings, &method_state); + method = settings.method; + } config_ini_free(&settings.config); - switch (mode) { - case PROGRAM_MODE_ONE_SHOT: - case PROGRAM_MODE_PRINT: - case PROGRAM_MODE_RESET: - if (scheme.type == SOLAR_SCHEME) { - weprintf(_("Waiting for current location to become available...")); - - if (get_location(settings.provider, location_state, -1, &loc) < 0) - eprintf(_("Unable to get location from provider.")); - - if (!location_is_valid(&loc)) - exit(1); + /* Set up interprocess communication */ + install_signal_handlers(); - print_location(&loc); - } + /* Get location if required */ + if (scheme.type == SOLAR_SCHEME) { + if (provider->get_fd(provider_state) >= 0) + weprintf(_("Waiting for current location to become available...")); + if (get_location(provider, provider_state, -1, &location) < 0) + eprintf(_("Unable to get location from provider.")); + if (!location_is_valid(&location)) + eprintf(_("Invalid location returned from provider.")); + print_location(&location); + location_available = 1; + } - get_colour_settings(&loc, &colour, &period, &day_level); + /* Get and print colour to set or if continual mode the initial colour */ + get_colour_settings(&colour, &period, &day_level); /* needed in contiual mode for `period` and `day_level` */ + if (mode == PROGRAM_MODE_CONTINUAL) + colour = COLOUR_SETTING_NEUTRAL; + if (verbose || mode == PROGRAM_MODE_PRINT) { + if (scheme.type != STATIC_SCHEME) + print_period(period, day_level); + printf(_("Color temperature: %luK\n"), colour.temperature); + printf(_("Brightness: %.2f\n"), colour.brightness); + printf(_("Gamma: %.3f, %.3f, %.3f\n"), colour.gamma[0], colour.gamma[1], colour.gamma[2]); + } - if (verbose || mode == PROGRAM_MODE_PRINT) { - if (scheme.type != STATIC_SCHEME) - print_period(period, day_level); - printf(_("Color temperature: %luK\n"), colour.temperature); - printf(_("Brightness: %.2f\n"), colour.brightness); - if (mode == PROGRAM_MODE_PRINT) - break; - } + switch (mode) { + case PROGRAM_MODE_PRINT: + break; - if (settings.method->apply(method_state, &colour, settings.preserve_gamma.value) < 0) + case PROGRAM_MODE_ONE_SHOT: + case PROGRAM_MODE_RESET: + if (method->apply(method_state, &colour, preserve_gamma) < 0) eprintf(_("Temperature adjustment failed.")); - -#ifndef WINDOWS - /* In Quartz (OSX) the gamma adjustments will automatically revert when - * the process exits. Therefore, we have to loop until Ctrl+C is received. */ - if (!strcmp(settings.method->name, "quartz")) { + if (method->autoreset) { weprintf(_("Press ctrl-c to stop...")); while (!exiting) pause(); } -#endif break; case PROGRAM_MODE_CONTINUAL: - run_continual_mode(settings.provider, location_state, settings.method, method_state, - settings.use_fade.value, settings.preserve_gamma.value); + run_continual_mode(); break; } if (method_state) - settings.method->free(method_state); - if (location_state) - settings.provider->free(location_state); + method->free(method_state); + if (provider_state) + provider->free(provider_state); return 0; } |