/* redshift.c -- Main program 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) 2009-2017 Jon Lund Steffensen
*/
#include "common.h"
#include "config-ini.h"
#include "solar.h"
#include "options.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.
*/
#ifndef _WIN32
# include
#else
#define POLLIN 0
struct pollfd {
int fd;
short events;
short revents;
};
int poll(struct pollfd *fds, int nfds, int timeout) { abort(); return -1; }
#endif
/* pause() is not defined on windows platform but is not needed either.
Use a noop macro instead. */
#ifdef __WIN32__
# define pause()
#endif
#undef CLAMP
#define CLAMP(lo,mid,up) (((lo) > (mid)) ? (lo) : (((mid) < (up)) ? (mid) : (up)))
/* Bounds for parameters. */
#define MIN_LAT -90.0
#define MAX_LAT 90.0
#define MIN_LON -180.0
#define MAX_LON 180.0
#define MIN_TEMP 1000
#define MAX_TEMP 25000
#define MIN_BRIGHTNESS 0.1
#define MAX_BRIGHTNESS 1.0
#define MIN_GAMMA 0.1
#define MAX_GAMMA 10.0
/* Duration of sleep between screen updates (milliseconds). */
#define SLEEP_DURATION 5000
#define SLEEP_DURATION_SHORT 100
/* Length of fade in numbers of short sleep durations. */
#define FADE_LENGTH 40
/* Names of periods of day */
static const char *period_names[] = {
/* TRANSLATORS: Name printed when period of day is unknown */
N_("None"),
N_("Daytime"),
N_("Night"),
N_("Transition")
};
/* Determine which period we are currently in based on time offset. */
static enum period
get_period_from_time(const struct transition_scheme *transition, int time_offset)
{
if (time_offset < transition->dawn.start ||
time_offset >= transition->dusk.end) {
return PERIOD_NIGHT;
} else if (time_offset >= transition->dawn.end &&
time_offset < transition->dusk.start) {
return PERIOD_DAYTIME;
} else {
return PERIOD_TRANSITION;
}
}
/* Determine which period we are currently in based on solar elevation. */
static enum period
get_period_from_elevation(
const struct transition_scheme *transition, double elevation)
{
if (elevation < transition->low) {
return PERIOD_NIGHT;
} else if (elevation < transition->high) {
return PERIOD_TRANSITION;
} else {
return PERIOD_DAYTIME;
}
}
/* Determine how far through the transition we are based on time offset. */
static double
get_transition_progress_from_time(
const struct transition_scheme *transition, int time_offset)
{
if (time_offset < transition->dawn.start ||
time_offset >= transition->dusk.end) {
return 0.0;
} else if (time_offset < transition->dawn.end) {
return (transition->dawn.start - time_offset) /
(double)(transition->dawn.start -
transition->dawn.end);
} else if (time_offset > transition->dusk.start) {
return (transition->dusk.end - time_offset) /
(double)(transition->dusk.end -
transition->dusk.start);
} else {
return 1.0;
}
}
/* Determine how far through the transition we are based on elevation. */
static double
get_transition_progress_from_elevation(
const struct transition_scheme *transition, double elevation)
{
if (elevation < transition->low) {
return 0.0;
} else if (elevation < transition->high) {
return (transition->low - elevation) /
(transition->low - transition->high);
} else {
return 1.0;
}
}
/* Return number of seconds since midnight from timestamp. */
static int
get_seconds_since_midnight(double timestamp)
{
time_t t = (time_t)timestamp;
struct tm tm;
#ifdef _WIN32
localtime_s(&tm, &t);
#else
localtime_r(&t, &tm);
#endif
return tm.tm_sec + tm.tm_min * 60 + tm.tm_hour * 3600;
}
/* Print verbose description of the given period. */
static void
print_period(enum period period, double transition)
{
switch (period) {
case PERIOD_NONE:
case PERIOD_NIGHT:
case PERIOD_DAYTIME:
printf(_("Period: %s\n"), gettext(period_names[period]));
break;
case PERIOD_TRANSITION:
printf(_("Period: %s (%.2f%% day)\n"),
gettext(period_names[period]),
transition*100);
break;
}
}
/* Print location */
static void
print_location(const struct location *location)
{
/* TRANSLATORS: Abbreviation for `north' */
const char *north = _("N");
/* TRANSLATORS: Abbreviation for `south' */
const char *south = _("S");
/* TRANSLATORS: Abbreviation for `east' */
const char *east = _("E");
/* TRANSLATORS: Abbreviation for `west' */
const char *west = _("W");
/* TRANSLATORS: Append degree symbols after %f if possible.
The string following each number is an abreviation for
north, source, east or west (N, S, E, W). */
printf(_("Location: %.2f %s, %.2f %s\n"),
fabs(location->lat), location->lat >= 0.f ? north : south,
fabs(location->lon), location->lon >= 0.f ? east : west);
}
/* Interpolate color setting structs given alpha. */
static void
interpolate_color_settings(
const struct color_setting *first,
const struct color_setting *second,
double alpha,
struct color_setting *result)
{
alpha = CLAMP(0.0, alpha, 1.0);
result->temperature = (1.0-alpha)*first->temperature +
alpha*second->temperature;
result->brightness = (1.0-alpha)*first->brightness +
alpha*second->brightness;
for (int i = 0; i < 3; i++) {
result->gamma[i] = (1.0-alpha)*first->gamma[i] +
alpha*second->gamma[i];
}
}
/* Interpolate color setting structs transition scheme. */
static void
interpolate_transition_scheme(
const struct transition_scheme *transition,
double alpha,
struct color_setting *result)
{
const struct color_setting *day = &transition->day;
const struct color_setting *night = &transition->night;
alpha = CLAMP(0.0, alpha, 1.0);
interpolate_color_settings(night, day, alpha, result);
}
/* Return 1 if color settings have major differences, otherwise 0.
Used to determine if a fade should be applied in continual mode. */
static int
color_setting_diff_is_major(
const struct color_setting *first,
const struct color_setting *second)
{
return (abs(first->temperature - second->temperature) > 25 ||
fabsf(first->brightness - second->brightness) > 0.1 ||
fabsf(first->gamma[0] - second->gamma[0]) > 0.1 ||
fabsf(first->gamma[1] - second->gamma[1]) > 0.1 ||
fabsf(first->gamma[2] - second->gamma[2]) > 0.1);
}
/* Reset color setting to default values. */
static void
color_setting_reset(struct color_setting *color)
{
color->temperature = NEUTRAL_TEMP;
color->gamma[0] = 1.0;
color->gamma[1] = 1.0;
color->gamma[2] = 1.0;
color->brightness = 1.0;
}
static int
provider_try_start(const struct location_provider *provider,
LOCATION_STATE **state, struct config_ini_state *config,
char *args)
{
int r;
r = provider->init(state);
if (r < 0) {
fprintf(stderr, _("Initialization of %s failed.\n"),
provider->name);
return -1;
}
/* Set provider options from config file. */
struct config_ini_section *section =
config_ini_get_section(config, provider->name);
if (section != NULL) {
struct config_ini_setting *setting = section->settings;
while (setting != NULL) {
r = provider->set_option(*state, setting->name,
setting->value);
if (r < 0) {
provider->free(*state);
fprintf(stderr, _("Failed to set %s"
" option.\n"),
provider->name);
/* TRANSLATORS: `help' must not be
translated. */
fprintf(stderr, _("Try `-l %s:help' for more"
" information.\n"),
provider->name);
return -1;
}
setting = setting->next;
}
}
/* Set provider options from command line. */
const char *manual_keys[] = { "lat", "lon" };
int i = 0;
while (args != NULL) {
char *next_arg = strchr(args, ':');
if (next_arg != NULL) *(next_arg++) = '\0';
const char *key = args;
char *value = strchr(args, '=');
if (value == NULL) {
/* The options for the "manual" method can be set
without keys on the command line for convencience
and for backwards compatability. We add the proper
keys here before calling set_option(). */
if (strcmp(provider->name, "manual") == 0 &&
i < sizeof(manual_keys)/sizeof(manual_keys[0])) {
key = manual_keys[i];
value = args;
} else {
fprintf(stderr, _("Failed to parse option `%s'.\n"),
args);
return -1;
}
} else {
*(value++) = '\0';
}
r = provider->set_option(*state, key, value);
if (r < 0) {
provider->free(*state);
fprintf(stderr, _("Failed to set %s option.\n"),
provider->name);
/* TRANSLATORS: `help' must not be translated. */
fprintf(stderr, _("Try `-l %s:help' for more"
" information.\n"), provider->name);
return -1;
}
args = next_arg;
i += 1;
}
/* Start provider. */
r = provider->start(*state);
if (r < 0) {
provider->free(*state);
fprintf(stderr, _("Failed to start provider %s.\n"),
provider->name);
return -1;
}
return 0;
}
static int
method_try_start(const struct gamma_method *method, GAMMA_STATE **state,
enum program_mode mode, struct config_ini_state *config, char *args)
{
int r;
r = method->init(state);
if (r < 0) {
fprintf(stderr, _("Initialization of %s failed.\n"),
method->name);
return -1;
}
/* Set method options from config file. */
struct config_ini_section *section =
config_ini_get_section(config, method->name);
if (section != NULL) {
struct config_ini_setting *setting = section->settings;
while (setting != NULL) {
r = method->set_option(
*state, setting->name, setting->value);
if (r < 0) {
method->free(*state);
fprintf(stderr, _("Failed to set %s"
" option.\n"),
method->name);
/* TRANSLATORS: `help' must not be
translated. */
fprintf(stderr, _("Try `-m %s:help' for more"
" information.\n"),
method->name);
return -1;
}
setting = setting->next;
}
}
/* Set method options from command line. */
while (args != NULL) {
char *next_arg = strchr(args, ':');
if (next_arg != NULL) *(next_arg++) = '\0';
const char *key = args;
char *value = strchr(args, '=');
if (value == NULL) {
fprintf(stderr, _("Failed to parse option `%s'.\n"),
args);
return -1;
} else {
*(value++) = '\0';
}
r = method->set_option(*state, key, value);
if (r < 0) {
method->free(*state);
fprintf(stderr, _("Failed to set %s option.\n"),
method->name);
/* TRANSLATORS: `help' must not be translated. */
fprintf(stderr, _("Try -m %s:help' for more"
" information.\n"), method->name);
return -1;
}
args = next_arg;
}
/* Start method. */
r = method->start(*state, mode);
if (r < 0) {
method->free(*state);
fprintf(stderr, _("Failed to start adjustment method %s.\n"),
method->name);
return -1;
}
return 0;
}
/* Check whether gamma is within allowed levels. */
static int
gamma_is_valid(const float gamma[3])
{
return !(gamma[0] < MIN_GAMMA ||
gamma[0] > MAX_GAMMA ||
gamma[1] < MIN_GAMMA ||
gamma[1] > MAX_GAMMA ||
gamma[2] < MIN_GAMMA ||
gamma[2] > MAX_GAMMA);
}
/* Check whether location is valid.
Prints error message on stderr and returns 0 if invalid, otherwise
returns 1. */
static int
location_is_valid(const struct location *location)
{
/* Latitude */
if (location->lat < MIN_LAT || location->lat > MAX_LAT) {
/* TRANSLATORS: Append degree symbols if possible. */
fprintf(stderr,
_("Latitude must be between %.1f and %.1f.\n"),
MIN_LAT, MAX_LAT);
return 0;
}
/* Longitude */
if (location->lon < MIN_LON || location->lon > MAX_LON) {
/* TRANSLATORS: Append degree symbols if possible. */
fprintf(stderr,
_("Longitude must be between"
" %.1f and %.1f.\n"), MIN_LON, MAX_LON);
return 0;
}
return 1;
}
/* Wait for location to become available from provider.
Waits until timeout (milliseconds) has elapsed or forever if timeout
is -1. Writes location to loc. Returns -1 on error,
0 if timeout was reached, 1 if location became available. */
static int
provider_get_location(
const struct location_provider *provider, LOCATION_STATE *state,
int timeout, struct location *loc)
{
int available = 0;
struct pollfd pollfds[1];
while (!available) {
int loc_fd = provider->get_fd(state);
if (loc_fd >= 0) {
/* Provider is dynamic. */
/* TODO: This should use a monotonic time source. */
double now;
int r = systemtime_get_time(&now);
if (r < 0) {
fputs(_("Unable to read system time.\n"),
stderr);
return -1;
}
/* Poll on file descriptor until ready. */
pollfds[0].fd = loc_fd;
pollfds[0].events = POLLIN;
r = poll(pollfds, 1, timeout);
if (r < 0) {
perror("poll");
return -1;
} else if (r == 0) {
return 0;
}
double later;
r = systemtime_get_time(&later);
if (r < 0) {
fputs(_("Unable to read system time.\n"),
stderr);
return -1;
}
/* Adjust timeout by elapsed time */
if (timeout >= 0) {
timeout -= (later - now) * 1000;
timeout = timeout < 0 ? 0 : timeout;
}
}
int r = provider->handle(state, loc, &available);
if (r < 0) return -1;
}
return 1;
}
/* Easing function for fade.
See https://github.com/mietek/ease-tween */
static double
ease_fade(double t)
{
if (t <= 0) return 0;
if (t >= 1) return 1;
return 1.0042954579734844 * exp(
-6.4041738958415664 * exp(-7.2908241330981340 * 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
color temperature. */
static int
run_continual_mode(const struct location_provider *provider,
LOCATION_STATE *location_state,
const struct transition_scheme *scheme,
const struct gamma_method *method,
GAMMA_STATE *method_state,
int use_fade, int preserve_gamma, int verbose)
{
int r;
/* Short fade parameters */
int fade_length = 0;
int fade_time = 0;
struct color_setting fade_start_interp;
r = signals_install_handlers();
if (r < 0) {
return r;
}
/* Save previous parameters so we can avoid printing status updates if
the values did not change. */
enum period prev_period = PERIOD_NONE;
/* Previous target color setting and current actual color setting.
Actual color setting takes into account the current color fade. */
struct color_setting prev_target_interp;
color_setting_reset(&prev_target_interp);
struct color_setting interp;
color_setting_reset(&interp);
struct location loc = { NAN, NAN };
int need_location = !scheme->use_time;
if (need_location) {
fputs(_("Waiting for initial location"
" to become available...\n"), stderr);
/* Get initial location from provider */
r = provider_get_location(provider, location_state, -1, &loc);
if (r < 0) {
fputs(_("Unable to get location"
" from provider.\n"), stderr);
return -1;
}
if (!location_is_valid(&loc)) {
fputs(_("Invalid location returned from provider.\n"),
stderr);
return -1;
}
print_location(&loc);
}
if (verbose) {
printf(_("Color temperature: %uK\n"), interp.temperature);
printf(_("Brightness: %.2f\n"), interp.brightness);
}
/* Continuously adjust color temperature */
int done = 0;
int prev_disabled = 1;
int disabled = 0;
int location_available = 1;
while (1) {
/* Check to see if disable signal was caught */
if (disable && !done) {
disabled = !disabled;
disable = 0;
}
/* Check to see if exit signal was caught */
if (exiting) {
if (done) {
/* On second signal stop the ongoing fade. */
break;
} else {
done = 1;
disabled = 1;
}
exiting = 0;
}
/* Print status change */
if (verbose && disabled != prev_disabled) {
printf(_("Status: %s\n"), disabled ?
_("Disabled") : _("Enabled"));
}
prev_disabled = disabled;
/* Read timestamp */
double now;
r = systemtime_get_time(&now);
if (r < 0) {
fputs(_("Unable to read system time.\n"), stderr);
return -1;
}
enum period period;
double transition_prog;
if (scheme->use_time) {
int time_offset = get_seconds_since_midnight(now);
period = get_period_from_time(scheme, time_offset);
transition_prog = get_transition_progress_from_time(
scheme, time_offset);
} else {
/* Current angular elevation of the sun */
double elevation = solar_elevation(
now, loc.lat, loc.lon);
period = get_period_from_elevation(scheme, elevation);
transition_prog =
get_transition_progress_from_elevation(
scheme, elevation);
}
/* Use transition progress to get target color
temperature. */
struct color_setting target_interp;
interpolate_transition_scheme(
scheme, transition_prog, &target_interp);
if (disabled) {
period = PERIOD_NONE;
color_setting_reset(&target_interp);
}
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)) {
print_period(period, transition_prog);
}
/* Activate hooks if period changed */
if (period != prev_period) {
hooks_signal_period_change(prev_period, period);
}
/* Start fade if the parameter differences are too big to apply
instantly. */
if (use_fade) {
if ((fade_length == 0 &&
color_setting_diff_is_major(
&interp,
&target_interp)) ||
(fade_length != 0 &&
color_setting_diff_is_major(
&target_interp,
&prev_target_interp))) {
fade_length = FADE_LENGTH;
fade_time = 0;
fade_start_interp = interp;
}
}
/* Handle ongoing fade */
if (fade_length != 0) {
fade_time += 1;
double frac = fade_time / (double)fade_length;
double alpha = CLAMP(0.0, ease_fade(frac), 1.0);
interpolate_color_settings(
&fade_start_interp, &target_interp, alpha,
&interp);
if (fade_time > fade_length) {
fade_time = 0;
fade_length = 0;
}
} else {
interp = target_interp;
}
/* 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: %uK\n"),
target_interp.temperature);
}
if (prev_target_interp.brightness !=
target_interp.brightness) {
printf(_("Brightness: %.2f\n"),
target_interp.brightness);
}
}
/* Adjust temperature */
r = method->set_temperature(
method_state, &interp, preserve_gamma);
if (r < 0) {
fputs(_("Temperature adjustment failed.\n"),
stderr);
return -1;
}
/* Save period and target color setting as previous */
prev_period = period;
prev_target_interp = target_interp;
/* Sleep length depends on whether a fade is ongoing. */
int delay = SLEEP_DURATION;
if (fade_length != 0) {
delay = SLEEP_DURATION_SHORT;
}
/* Update location. */
int loc_fd = -1;
if (need_location) {
loc_fd = provider->get_fd(location_state);
}
if (loc_fd >= 0) {
/* Provider is dynamic. */
struct pollfd pollfds[1];
pollfds[0].fd = loc_fd;
pollfds[0].events = POLLIN;
int r = poll(pollfds, 1, delay);
if (r < 0) {
if (errno == EINTR) continue;
perror("poll");
fputs(_("Unable to get location"
" from provider.\n"), stderr);
return -1;
} else if (r == 0) {
continue;
}
/* Get new location and availability
information. */
struct location new_loc;
int new_available;
r = provider->handle(
location_state, &new_loc,
&new_available);
if (r < 0) {
fputs(_("Unable to get location"
" from provider.\n"), stderr);
return -1;
}
if (!new_available &&
new_available != location_available) {
fputs(_("Location is temporarily"
" unavailable; Using previous"
" location until it becomes"
" available...\n"), stderr);
}
if (new_available &&
(new_loc.lat != loc.lat ||
new_loc.lon != loc.lon ||
new_available != location_available)) {
loc = new_loc;
print_location(&loc);
}
location_available = new_available;
if (!location_is_valid(&loc)) {
fputs(_("Invalid location returned"
" from provider.\n"), stderr);
return -1;
}
} else {
systemtime_msleep(delay);
}
}
/* Restore saved gamma ramps */
method->restore(method_state);
return 0;
}
int
main(int argc, char *argv[])
{
int r;
#ifdef ENABLE_NLS
/* Init locale */
setlocale(LC_CTYPE, "");
setlocale(LC_MESSAGES, "");
/* Internationalisation */
bindtextdomain(PACKAGE, LOCALEDIR);
textdomain(PACKAGE);
#endif
/* List of gamma methods. */
const struct gamma_method gamma_methods[] = {
#ifdef ENABLE_COOPGAMMA
coopgamma_gamma_method,
#endif
#ifdef ENABLE_DRM
drm_gamma_method,
#endif
#ifdef ENABLE_RANDR
randr_gamma_method,
#endif
#ifdef ENABLE_VIDMODE
vidmode_gamma_method,
#endif
#ifdef ENABLE_QUARTZ
quartz_gamma_method,
#endif
#ifdef ENABLE_WINGDI
w32gdi_gamma_method,
#endif
dummy_gamma_method,
{ NULL }
};
/* List of location providers. */
const struct location_provider location_providers[] = {
#ifdef ENABLE_GEOCLUE2
geoclue2_location_provider,
#endif
#ifdef ENABLE_CORELOCATION
corelocation_location_provider,
#endif
manual_location_provider,
{ NULL }
};
/* Flush messages consistently even if redirected to a pipe or
file. Change the flush behaviour to line-buffered, without
changing the actual buffers being used. */
setvbuf(stdout, NULL, _IOLBF, 0);
setvbuf(stderr, NULL, _IOLBF, 0);
struct options options;
options_init(&options);
options_parse_args(
&options, argc, argv, gamma_methods, location_providers);
/* Load settings from config file. */
struct config_ini_state config_state;
r = config_ini_init(&config_state, options.config_filepath);
if (r < 0) {
fputs("Unable to load config file.\n", stderr);
exit(EXIT_FAILURE);
}
free(options.config_filepath);
options_parse_config_file(
&options, &config_state, gamma_methods, location_providers);
options_set_defaults(&options);
if (options.scheme.dawn.start >= 0 || options.scheme.dawn.end >= 0 ||
options.scheme.dusk.start >= 0 || options.scheme.dusk.end >= 0) {
if (options.scheme.dawn.start < 0 ||
options.scheme.dawn.end < 0 ||
options.scheme.dusk.start < 0 ||
options.scheme.dusk.end < 0) {
fputs(_("Partial time-configuration not"
" supported!\n"), stderr);
exit(EXIT_FAILURE);
}
if (options.scheme.dawn.start > options.scheme.dawn.end ||
options.scheme.dawn.end > options.scheme.dusk.start ||
options.scheme.dusk.start > options.scheme.dusk.end) {
fputs(_("Invalid dawn/dusk time configuration!\n"),
stderr);
exit(EXIT_FAILURE);
}
options.scheme.use_time = 1;
}
/* Initialize location provider if needed. If provider is NULL
try all providers until one that works is found. */
LOCATION_STATE *location_state;
/* Location is not needed for reset mode and manual mode. */
int need_location =
options.mode != PROGRAM_MODE_RESET &&
options.mode != PROGRAM_MODE_MANUAL &&
!options.scheme.use_time;
if (need_location) {
if (options.provider != NULL) {
/* Use provider specified on command line. */
r = provider_try_start(
options.provider, &location_state,
&config_state, options.provider_args);
if (r < 0) exit(EXIT_FAILURE);
} else {
/* Try all providers, use the first that works. */
for (int i = 0;
location_providers[i].name != NULL; i++) {
const struct location_provider *p =
&location_providers[i];
fprintf(stderr,
_("Trying location provider `%s'...\n"),
p->name);
r = provider_try_start(p, &location_state,
&config_state, NULL);
if (r < 0) {
fputs(_("Trying next provider...\n"),
stderr);
continue;
}
/* Found provider that works. */
printf(_("Using provider `%s'.\n"), p->name);
options.provider = p;
break;
}
/* Failure if no providers were successful at this
point. */
if (options.provider == NULL) {
fputs(_("No more location providers"
" to try.\n"), stderr);
exit(EXIT_FAILURE);
}
}
/* Solar elevations */
if (options.scheme.high < options.scheme.low) {
fprintf(stderr,
_("High transition elevation cannot be lower than"
" the low transition elevation.\n"));
exit(EXIT_FAILURE);
}
if (options.verbose) {
/* TRANSLATORS: Append degree symbols if possible. */
printf(_("Solar elevations: day above %.1f, night below %.1f\n"),
options.scheme.high, options.scheme.low);
}
}
if (options.mode != PROGRAM_MODE_RESET &&
options.mode != PROGRAM_MODE_MANUAL) {
if (options.verbose) {
printf(_("Temperatures: %dK at day, %dK at night\n"),
options.scheme.day.temperature,
options.scheme.night.temperature);
}
/* Color temperature */
if (options.scheme.day.temperature < MIN_TEMP ||
options.scheme.day.temperature > MAX_TEMP ||
options.scheme.night.temperature < MIN_TEMP ||
options.scheme.night.temperature > MAX_TEMP) {
fprintf(stderr,
_("Temperature must be between %uK and %uK.\n"),
MIN_TEMP, MAX_TEMP);
exit(EXIT_FAILURE);
}
}
if (options.mode == PROGRAM_MODE_MANUAL) {
/* Check color temperature to be set */
if (options.temp_set < MIN_TEMP ||
options.temp_set > MAX_TEMP) {
fprintf(stderr,
_("Temperature must be between %uK and %uK.\n"),
MIN_TEMP, MAX_TEMP);
exit(EXIT_FAILURE);
}
}
/* Brightness */
if (options.scheme.day.brightness < MIN_BRIGHTNESS ||
options.scheme.day.brightness > MAX_BRIGHTNESS ||
options.scheme.night.brightness < MIN_BRIGHTNESS ||
options.scheme.night.brightness > MAX_BRIGHTNESS) {
fprintf(stderr,
_("Brightness values must be between %.1f and %.1f.\n"),
MIN_BRIGHTNESS, MAX_BRIGHTNESS);
exit(EXIT_FAILURE);
}
if (options.verbose) {
printf(_("Brightness: %.2f:%.2f\n"),
options.scheme.day.brightness,
options.scheme.night.brightness);
}
/* Gamma */
if (!gamma_is_valid(options.scheme.day.gamma) ||
!gamma_is_valid(options.scheme.night.gamma)) {
fprintf(stderr,
_("Gamma value must be between %.1f and %.1f.\n"),
MIN_GAMMA, MAX_GAMMA);
exit(EXIT_FAILURE);
}
if (options.verbose) {
/* TRANSLATORS: The string in parenthesis is either
Daytime or Night (translated). */
printf(_("Gamma (%s): %.3f, %.3f, %.3f\n"),
_("Daytime"), options.scheme.day.gamma[0],
options.scheme.day.gamma[1],
options.scheme.day.gamma[2]);
printf(_("Gamma (%s): %.3f, %.3f, %.3f\n"),
_("Night"), options.scheme.night.gamma[0],
options.scheme.night.gamma[1],
options.scheme.night.gamma[2]);
}
struct transition_scheme *scheme = &options.scheme;
/* Initialize gamma adjustment method. If method is NULL
try all methods until one that works is found. */
GAMMA_STATE *method_state;
/* Gamma adjustment not needed for print mode */
if (options.mode != PROGRAM_MODE_PRINT) {
if (options.method != NULL) {
/* Use method specified on command line. */
r = method_try_start(
options.method, &method_state, options.mode, &config_state,
options.method_args);
if (r < 0) exit(EXIT_FAILURE);
} else {
/* Try all methods, use the first that works. */
for (int i = 0; gamma_methods[i].name != NULL; i++) {
const struct gamma_method *m = &gamma_methods[i];
if (!m->autostart) continue;
r = method_try_start(
m, &method_state, options.mode, &config_state, NULL);
if (r < 0) {
fputs(_("Trying next method...\n"), stderr);
continue;
}
/* Found method that works. */
printf(_("Using method `%s'.\n"), m->name);
options.method = m;
break;
}
/* Failure if no methods were successful at this point. */
if (options.method == NULL) {
fputs(_("No more methods to try.\n"), stderr);
exit(EXIT_FAILURE);
}
}
}
config_ini_free(&config_state);
switch (options.mode) {
case PROGRAM_MODE_ONE_SHOT:
case PROGRAM_MODE_PRINT:
{
struct location loc = { NAN, NAN };
if (need_location) {
fputs(_("Waiting for current location"
" to become available...\n"), stderr);
/* Wait for location provider. */
int r = provider_get_location(
options.provider, location_state, -1, &loc);
if (r < 0) {
fputs(_("Unable to get location"
" from provider.\n"), stderr);
exit(EXIT_FAILURE);
}
if (!location_is_valid(&loc)) {
exit(EXIT_FAILURE);
}
print_location(&loc);
}
double now;
r = systemtime_get_time(&now);
if (r < 0) {
fputs(_("Unable to read system time.\n"), stderr);
options.method->free(method_state);
exit(EXIT_FAILURE);
}
enum period period;
double transition_prog;
if (options.scheme.use_time) {
int time_offset = get_seconds_since_midnight(now);
period = get_period_from_time(scheme, time_offset);
transition_prog = get_transition_progress_from_time(
scheme, time_offset);
} else {
/* Current angular elevation of the sun */
double elevation = solar_elevation(
now, loc.lat, loc.lon);
if (options.verbose) {
/* TRANSLATORS: Append degree symbol if
possible. */
printf(_("Solar elevation: %f\n"), elevation);
}
period = get_period_from_elevation(scheme, elevation);
transition_prog =
get_transition_progress_from_elevation(
scheme, elevation);
}
/* Use transition progress to set color temperature */
struct color_setting interp;
interpolate_transition_scheme(
scheme, transition_prog, &interp);
if (options.verbose || options.mode == PROGRAM_MODE_PRINT) {
print_period(period, transition_prog);
printf(_("Color temperature: %uK\n"),
interp.temperature);
printf(_("Brightness: %.2f\n"),
interp.brightness);
}
if (options.mode != PROGRAM_MODE_PRINT) {
/* Adjust temperature */
r = options.method->set_temperature(
method_state, &interp, options.preserve_gamma);
if (r < 0) {
fputs(_("Temperature adjustment failed.\n"),
stderr);
options.method->free(method_state);
exit(EXIT_FAILURE);
}
/* In Quartz (macOS) the gamma adjustments will
automatically revert when the process exits.
Therefore, we have to loop until CTRL-C is received.
*/
if (strcmp(options.method->name, "quartz") == 0) {
fputs(_("Press ctrl-c to stop...\n"), stderr);
pause();
}
}
}
break;
case PROGRAM_MODE_MANUAL:
{
if (options.verbose) {
printf(_("Color temperature: %uK\n"),
options.temp_set);
}
/* Adjust temperature */
struct color_setting manual = scheme->day;
manual.temperature = options.temp_set;
r = options.method->set_temperature(
method_state, &manual, options.preserve_gamma);
if (r < 0) {
fputs(_("Temperature adjustment failed.\n"), stderr);
options.method->free(method_state);
exit(EXIT_FAILURE);
}
/* 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(options.method->name, "quartz") == 0) {
fputs(_("Press ctrl-c to stop...\n"), stderr);
pause();
}
}
break;
case PROGRAM_MODE_RESET:
{
/* Reset screen */
struct color_setting reset;
color_setting_reset(&reset);
r = options.method->set_temperature(method_state, &reset, 0);
if (r < 0) {
fputs(_("Temperature adjustment failed.\n"), stderr);
options.method->free(method_state);
exit(EXIT_FAILURE);
}
/* 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(options.method->name, "quartz") == 0) {
fputs(_("Press ctrl-c to stop...\n"), stderr);
pause();
}
}
break;
case PROGRAM_MODE_CONTINUAL:
{
r = run_continual_mode(
options.provider, location_state, scheme,
options.method, method_state,
options.use_fade, options.preserve_gamma,
options.verbose);
if (r < 0) exit(EXIT_FAILURE);
}
break;
}
/* Clean up gamma adjustment state */
if (options.mode != PROGRAM_MODE_PRINT) {
options.method->free(method_state);
}
/* Clean up location provider state */
if (need_location) {
options.provider->free(location_state);
}
return EXIT_SUCCESS;
}