diff options
Diffstat (limited to 'src/config.c')
-rw-r--r-- | src/config.c | 1172 |
1 files changed, 0 insertions, 1172 deletions
diff --git a/src/config.c b/src/config.c deleted file mode 100644 index 29c5a53..0000000 --- a/src/config.c +++ /dev/null @@ -1,1172 +0,0 @@ -/*- - * 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> - * - * redshift-ng 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-ng 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-ng. If not, see <http://www.gnu.org/licenses/>. - */ -#include "common.h" - - -/** - * Output usage synopsis, without exiting - * - * @param f Output sink - */ -static void -usage_no_exit(FILE *f) -{ - fprintf(f, _("Usage: %s %s\n"), argv0, - _("[-b brightness] [-c config-file] [-D | +D] [-E | +E | -e elevations] " - "[-g gamma] [-H hook-file] [-l latitude:longitude | -l provider[:options]] " - "[-m method[:options]] [-P | +P] [-r | +r] [-dv] " - "[-O temperature | -o | -p | -t temperature | -x] | -h | -V")); -} - - -/** - * Output usage synopsis and exit - */ -static void -usage(void) -{ - usage_no_exit(stderr); - exit(1); -} - - -struct colour_setting day_settings; -struct colour_setting night_settings; -union scheme scheme = {.type = SOLAR_SCHEME}; -enum program_mode mode = PROGRAM_MODE_CONTINUAL; -int preserve_gamma; -int use_fade; -int verbose = 0; -char *hook_file; - - -/** - * Print general help text - */ -static void -print_help(void) -{ - usage_no_exit(stdout); - printf("\n"); - - printf(_("Automatically adjust display colour temperature according the Sun\n")); - printf("\n"); - - printf(_(" -b day:night Select whitepoint brightness (Default: 1:1)\n")); - printf(_(" -c file Load settings from specified configuration file\n")); - printf(_(" -D Start in enabled state (Default)\n")); - printf(_(" +D Start in disabled state\n")); - printf(_(" -d Keep the process alive and remove the colour effects when killed\n")); - printf(_(" -E Use wall-clock based schedule\n")); - printf(_(" +E Use solar elevation based schedule\n")); - printf(_(" -e day:night Select solar elevation thresholds for day and night (Default: %g:%g)\n"), - DEFAULT_HIGH_ELEVATION, DEFAULT_LOW_ELEVATION); - printf(_(" -g day:night Additional gamma correction (Default: 1:1:1:1:1:1)\n")); - printf(_(" -H hook-file Select hook file or directrory\n")); - printf(_(" -h Display this help message\n")); - printf(_(" -l lat:lon Specific geographical location\n")); - printf(_(" -l provider[:options] Select location provider to get geographical location\n")); - printf(_(" (Use `-l list' to list available providers)\n")); - printf(_(" -m method[:options] Select adjustment method for applying colour settings\n")); - printf(_(" (Use `-m list' to list adjustment methods)\n")); - printf(_(" -O day:night Select colour temperature and apply once\n")); - printf(_(" -o Apply colour settings once, then exit\n")); - printf(_(" -P Remove preexisting colour adjustments\n")); - printf(_(" +P Preserve preexisting colour adjustments (Default)\n")); - printf(_(" -p Print parameters and exit\n")); - printf(_(" -r Disable fading between colour adjustments\n")); - printf(_(" +r Enable fading between colour adjustments (Default)\n")); - printf(_(" -t day:night Select colour temperature and apply continually (Default: %lu:%lu\n"), - DEFAULT_DAY_TEMPERATURE, DEFAULT_NIGHT_TEMPERATURE); - printf(_(" -V Show program implementation and version\n")); - printf(_(" -v Enable verbose output\n")); - printf(_(" -x Remove adjustments from screen\n")); - - printf("\n"); - printf(_("This is a breif summary, see `%s' for more information.\n"), "man redshift"); - printf("\n"); - printf(_("The neutral temperature is %luK. Using this value will not change the color\n" - "temperature of the display. Setting the color temperature to a value higher\n" - "than this results in more blue light, and setting a lower value will result in\n" - "more red light.\n"), NEUTRAL_TEMPERATURE); - printf("\n"); -} - - -/** - * Parse a boolean string - * - * @param s The string to parse - * @param key The name of the configuration assigned the value `s` - * @return 1 if `s` represents true, 0 if `s` represents false - */ -GCC_ONLY(__attribute__((__pure__))) -static int -get_boolean(const char *s, const char *key) -{ - int ret; - if (s[0] == '0' || s[0] == '1') { - ret = s[0] - '0'; - if (s[1]) - goto bad; - } else { - bad: - eprintf(_("Value of configuration `%s' must be either `1' or `0'."), key); - } - return ret; -} - - -/** - * atof(3)-like wrapper for strtod(3) that checks that the string is valid - * - * @param s The string to parse - * @param key The name of the configuration assigned the value `s` - * @return The encoded value - */ -static double -checked_atof(const char *s, const char *key) -{ - double ret; - errno = 0; - ret = strtod(s, (void *)&s); - if (errno || *s) - eprintf(_("Value of configuration `%s' is not a value decimal value."), key); - return ret; -} - - -/** - * Split a list of values, and remove leading and trailing whitespace - * from each value - * - * @param s The string to split; will be modified - * @param count The number of despired strings - * @param strs Output array for the strings; each element will - * be set to `NULL` or `s` with an offset - * @param delim The character `s` shall be split by, cannot be - * a whitespace character - * @return 1 if `s` is valid, 0 otherwise - * - * If `count` is 1, - * the value most be specified, - * if `count` is 2, - * at least one of values must be specified - * if `count` is 3, - * all values most be specified, otherwise `s` must only contain - * on value, and no colon, which will be used from each output value, or - * if `count` is 6, - * `s` must be on one of the formats: - * `a`: - * the specified value is used for each output value, - * `a:b`: - * the three lower output values will be set to `a`, which may be empty/`NULL`, and - * the three upper output values will be set to `b`, which may be empty/`NULL`; - * at least `a` or `b` must be non-empty, - * `a:b:c`: - * each value most be specified, `s` will be interpreted as `a:b:c:a:b:c`, - * `:a:b:c`: - * each value most be specified, the lower three values be set to `NULL`, - * and `a`, `b`, `c` will be used fo the upper three values, - * `a:b:c:`: - * each value most be specified, the upper three values be set to `NULL`, - * and `a`, `b`, `c` will be used fo the lower three values, or - * `a:b:c:d:e:f`: - * each value most be specified; - * where ':' represents `delim` - * - * Summarily said, `s` may contain a scalar value or a 3-tuple, and it may - * also contain a value or one value for daytime and one value or nighttime. - * If configured to use 3-tuple but scalar is provided, the provided value is - * used for each of the 3 requested values. If configured to use daytime and - * nighttime, but only one is specified it is used for both, but if `s` - * starts with `delim`, daytime is skipped but if `s` ends with `delim`, - * nighttime, is skipped; but both cannot be skipped. - */ -static int -get_strings(char *s, int count, char *strs[], char delim) -{ - int i = 0, n; - - /* Split by colon and left-trim */ - for (i = 0; i < count;) { - strs[i++] = s = ltrim(s); - s = strchr(s, delim); - if (!s) - break; - *s++ = '\0'; - } - n = i; - - /* Confirm no excess strings */ - if (s && *ltrim(s)) - return 0; - - /* Right-trim and replace empty strings with NULL */ - for (i = 0; i < n; i++) - if (!*rtrim(strs[i], NULL)) - strs[i] = NULL; - - /* Validate NULLs */ - switch (n) { - case 1: - /* must be specified */ - if (!strs[0]) - return 0; - break; - case 2: - /* at least one most be specified */ - if (!strs[0] && !strs[1]) - return 0; - break; - case 3: - /* each most be specified */ - if (!strs[0] || !strs[1] || !strs[2]) - return 0; - break; - case 4: - /* exactly either the first or the last shall be NULL */ - if (!strs[0] == !strs[3] || !strs[1] || !strs[2]) - return 0; - break; - case 6: - /* each most be specified */ - if (!strs[0] || !strs[1] || !strs[2] || !strs[3] || !strs[4] || !strs[5]) - return 0; - break; - default: - /* n==5 is always invalid */ - return 0; - } - - /* Duplicate to fill `strs` */ - switch (count) { - case 2: - if (n == 1) - strs[1] = strs[0]; - break; - case 3: - if (n == 1) - strs[2] = strs[1] = strs[0]; - else if (n == 2) - return 0; - break; - case 6: - if (n == 1) { - strs[5] = strs[4] = strs[3] = strs[2] = strs[1] = strs[0]; - } else if (n == 2) { - strs[5] = strs[4] = strs[3] = strs[1]; - strs[2] = strs[1] = strs[0]; - } else if (n == 3) { - strs[5] = strs[2]; - strs[4] = strs[1]; - strs[3] = strs[0]; - } else if (n == 4 && !strs[0]) { - strs[5] = strs[3]; - strs[4] = strs[2]; - strs[3] = strs[1]; - strs[2] = NULL; - strs[1] = NULL; - } else if (n == 4) { - strs[5] = NULL; - strs[4] = NULL; - } - break; - default: - break; - } - - return 1; -} - - -/** - * Parse and set temperature settings - * - * @param str The temperature specification to parse - * @param day The currently specified temperature for daytime, - * will be updated; `NULL` if it shall not be set - * @param night The currently specified temperature for nighttime, - * will be updated; `NULL` if it shall not be set - * @param key The configuration file setting being parsed, - * `NULL` if parsing the command line - */ -static void -set_temperature(char *str, struct setting_lu *day, struct setting_lu *night, const char *key) -{ - struct setting_lu *settings[] = {day, night}; - enum setting_source source = key ? SETTING_CONFIGFILE : SETTING_CMDLINE; - char *strs[2], *end; - size_t i, j; - - if (!get_strings(str, !!day + !!night, strs, ':')) { - invalid: - weprintf(_("Malformed temperature argument.")); - eprintf(_("Try `-h' for more information.")); - } - - errno = 0; - for (i = 0, j = 0; i < ELEMSOF(settings); j += settings[i++] ? 1 : 0) { - if (!settings[i] || !strs[j] || settings[i]->source > source) - continue; - if (settings[i]->source & SETTING_CONFIGFILE) { - if (i == 0) - weprintf(_("Daytime temperature specified multiple times in configuration file.")); - else - weprintf(_("Night temperature specified multiple times in configuration file.")); - } - settings[i]->source |= source; - settings[i]->value = strtoul(strs[j], &end, 10); - if (errno || end[*end == 'K']) - goto invalid; - if (!WITHIN(MIN_TEMPERATURE, settings[i]->value, MAX_TEMPERATURE)) - eprintf(_("Temperature must be between %luK and %luK."), MIN_TEMPERATURE, MAX_TEMPERATURE); - } -} - - -/** - * Parse and set whitepoint brightness settings - * - * @param str The brightness specification to parse - * @param day The currently specified brightness for daytime, - * will be updated; `NULL` if it shall not be set - * @param night The currently specified brightness for nighttime, - * will be updated; `NULL` if it shall not be set - * @param key The configuration file setting being parsed, - * `NULL` if parsing the command line - */ -static void -set_brightness(char *str, struct setting_f *day, struct setting_f *night, const char *key) -{ - struct setting_f *settings[] = {day, night}; - enum setting_source source = key ? SETTING_CONFIGFILE : SETTING_CMDLINE; - char *strs[2], *end; - size_t i, j; - - if (!get_strings(str, !!day + !!night, strs, ':')) { - invalid: - weprintf(_("Malformed brightness argument.")); - eprintf(_("Try `-h' for more information.")); - } - - errno = 0; - for (i = 0, j = 0; i < ELEMSOF(settings); j += settings[i++] ? 1 : 0) { - if (!settings[i] || !strs[j] || settings[i]->source > source) - continue; - if (settings[i]->source & SETTING_CONFIGFILE) { - if (i == 0) - weprintf(_("Daytime brightness specified multiple times in configuration file.")); - else - weprintf(_("Night brightness specified multiple times in configuration file.")); - } - settings[i]->source |= source; - settings[i]->value = strtod(strs[j], &end); - if (errno || *end) - goto invalid; - if (!WITHIN(MIN_BRIGHTNESS, settings[i]->value, MAX_BRIGHTNESS)) - eprintf(_("Brightness values must be between %.1f and %.1f."), MIN_BRIGHTNESS, MAX_BRIGHTNESS); - } -} - - -/** - * Parse and set gamma settings - * - * @param str The gamma specification to parse - * @param day The currently specified gamma for daytime, - * will be updated; `NULL` if it shall not be set - * @param night The currently specified gamma for nighttime, - * will be updated; `NULL` if it shall not be set - * @param key The configuration file setting being parsed, - * `NULL` if parsing the command line - */ -static void -set_gamma(char *str, struct setting_f3 *day, struct setting_f3 *night, const char *key) -{ - struct setting_f3 *settings[] = {day, night}; - enum setting_source source = key ? SETTING_CONFIGFILE : SETTING_CMDLINE; - char *strs[6], *end; - size_t i, j, k; - - if (!get_strings(str, 3 * (!!day + !!night), strs, ':')) { - invalid: - weprintf(_("Malformed gamma argument.")); - eprintf(_("Try `-h' for more information.")); - } - - errno = 0; - for (i = 0, j = 0; i < ELEMSOF(settings); j += settings[i++] ? 3 : 0) { - if (!settings[i] || !strs[j] || settings[i]->source > source) - continue; - if (settings[i]->source & SETTING_CONFIGFILE) { - if (i == 0) - weprintf(_("Daytime gamma specified multiple times in configuration file.")); - else - weprintf(_("Night gamma specified multiple times in configuration file.")); - } - settings[i]->source |= source; - for (k = 0; k < 3; k++) { - settings[i]->value[k] = strtod(strs[j + k], &end); - if (errno || *end) - goto invalid; - if (!WITHIN(MIN_GAMMA, settings[i]->value[k], MAX_GAMMA)) - eprintf(_("Gamma values must be between %.1f and %.1f."), MIN_GAMMA, MAX_GAMMA); - } - } -} - - -/** - * Parse and set solar elevation settings - * - * The fucntion assumes that the setting source is the command line - * - * @param str The gamma specification to parse - * @param day The currently specified lowest solar elevation for - * daytime, will be updated; `NULL` if it shall not be set - * @param night The currently specified highest solar elevation for - * nighttime, will be updated; `NULL` if it shall not be set - */ -static void -set_elevations(char *str, struct setting_f *day, struct setting_f *night) -{ - struct setting_f *settings[] = {day, night}; - const enum setting_source source = SETTING_CMDLINE; - char *strs[2], *end; - size_t i, j; - - if (!get_strings(str, !!day + !!night, strs, ':')) { - invalid: - weprintf(_("Malformed solar elevation argument.")); - eprintf(_("Try `-h' for more information.")); - } - - errno = 0; - for (i = 0, j = 0; i < ELEMSOF(settings); j += settings[i++] ? 1 : 0) { - if (!settings[i] || !strs[j] || settings[i]->source > source) - continue; - settings[i]->source |= source; - settings[i]->value = strtod(strs[j], &end); - if (errno || *end) - goto invalid; - } -} - - -/** - * Parse a time string on either of the formats "HH:MM" and "HH:MM:SS" - * - * Times up to, but excluding, 48:00 are supported. - * - * Leap seconds are not supported - * - * @param str String to parse - * @return The represented time, -1 if malformed - */ -static time_t -parse_time(char *str) -{ - time_t ret; - unsigned long int v; - - errno = 0; - - if (!isdigit(*str)) - return -1; - v = strtoul(str, &str, 10); - if (errno || *str++ != ':' || v >= 48UL) - return -1; - ret = (time_t)(v * 60UL * 60UL); - - if (!isdigit(*str)) - return -1; - v = strtoul(str, &str, 10); - if (errno || v >= 60UL) - return -1; - ret += (time_t)(v * 60UL); - - if (*str) { - if (*str++ != ':') - return -1; - if (!isdigit(*str)) - return -1; - v = strtoul(str, &str, 10); - if (errno || *str || v >= 60UL) - return -1; - ret += (time_t)v; - } - - return ret; -} - - -/** - * Parse and set a transition time setting - * - * @param str The transition time to parse - * @param start The currently specified transition start, will be updated - * @param end The currently specified transition end, will be updated - * @param key The configuration file setting being parsed, - * `NULL` if parsing the command line - * @return Normally 1, 0 if `str` is malformeda - */ -static int -set_transition_time(char *str, struct setting_time *start, struct setting_time *end, const char *key) -{ - struct setting_time *settings[] = {start, end}; - enum setting_source source = key ? SETTING_CONFIGFILE : SETTING_CMDLINE; - char *strs[ELEMSOF(settings)]; - int i; - - if (!get_strings(str, ELEMSOF(strs), strs, '-')) - return 0; - - for (i = 0; i < (int)ELEMSOF(settings); i++) { - if (!strs[i] || settings[i]->source > source) - continue; - if (settings[i]->source & SETTING_CONFIGFILE) { - if (i == 0) - weprintf(_("Start value for `%s' specified multiple times in configuration file."), key); - else - weprintf(_("End value for `%s' specified multiple times in configuration file."), key); - } - settings[i]->source |= source; - settings[i]->value = parse_time(strs[i]); - if (settings[i]->value < 0) - return 0; - } - - return 1; -} - - -/** - * Print list of available adjustment methods, - * and some helpful information - */ -static void -print_method_list(void) -{ - size_t i; - printf(_("Available adjustment methods:\n")); - for (i = 0; gamma_methods[i]; i++) - if (gamma_methods[i]->is_available()) - printf(" %s\n", gamma_methods[i]->name); - - printf("\n"); - printf(_("Specify colon-separated options with `-m METHOD:OPTIONS'.\n")); - /* TRANSLATORS: `help' must not be translated. */ - printf(_("Try `-m METHOD:help' for help.\n")); -} - - -/** - * Print list of available location providers, - * and some helpful information - */ -static void -print_provider_list(void) -{ - size_t i; - printf(_("Available location providers:\n")); - for (i = 0; location_providers[i]; i++) - printf(" %s\n", location_providers[i]->name); - - printf("\n"); - printf(_("Specify colon-separated options with `-l PROVIDER:OPTIONS'.\n")); - /* TRANSLATORS: `help' must not be translated. */ - printf(_("Try `-l PROVIDER:help' for help.\n")); -} - - -/** - * Get adjustment method by name - * - * @param name The name of the adjustment method to return - * @return The adjustment method - */ -GCC_ONLY(__attribute__((__pure__, __returns_nonnull__))) -static const struct gamma_method * -find_gamma_method(const char *name) -{ - size_t i; - for (i = 0; gamma_methods[i]; i++) - if (!strcasecmp(name, gamma_methods[i]->name)) - return gamma_methods[i]; - /* TRANSLATORS: This refers to the method used to adjust colours e.g. VidMode */ - eprintf(_("Unknown adjustment method `%s'."), name); -} - - -/** - * Get location provider by name - * - * @param name The name of the location provider to return - * @return The location provider - */ -GCC_ONLY(__attribute__((__pure__, __returns_nonnull__))) -static const struct location_provider * -find_location_provider(const char *name) -{ - size_t i; - for (i = 0; location_providers[i]; i++) - if (!strcasecmp(name, location_providers[i]->name)) - return location_providers[i]; - eprintf(_("Unknown location provider `%s'."), name); -} - - -/** - * Load default settings - * - * @param settings Output parameter for the settings - */ -static void -load_defaults(struct settings *settings) -{ - memset(settings, 0, sizeof(*settings)); /* set each `.source` to `SETTING_DEFAULT` and booleans to 0 */ - - settings->config_file = NULL; - settings->scheme_type = UNSPECIFIED_SCHEME; - - settings->day.temperature.value = DEFAULT_DAY_TEMPERATURE; - settings->day.brightness.value = DEFAULT_DAY_BRIGHTNESS; - settings->day.gamma.value[0] = DEFAULT_DAY_GAMMA; - settings->day.gamma.value[1] = DEFAULT_DAY_GAMMA; - settings->day.gamma.value[2] = DEFAULT_DAY_GAMMA; - - settings->night.temperature.value = DEFAULT_NIGHT_TEMPERATURE; - settings->night.brightness.value = DEFAULT_NIGHT_BRIGHTNESS; - settings->night.gamma.value[0] = DEFAULT_NIGHT_GAMMA; - settings->night.gamma.value[1] = DEFAULT_NIGHT_GAMMA; - settings->night.gamma.value[2] = DEFAULT_NIGHT_GAMMA; - - settings->hook_file.value = NULL; - settings->preserve_gamma.value = 1; - settings->use_fade.value = 1; - - settings->elevation_high.value = DEFAULT_HIGH_ELEVATION; - settings->elevation_low.value = DEFAULT_LOW_ELEVATION; - - settings->method = NULL; - settings->method_args = NULL; - - settings->provider = NULL; - settings->provider_args = NULL; -} - - -/** - * Load settings from the command line - * - * @param settings The currently loaded settings, will be updated - * @param argc Number of command line arguments - * @param argv `NULL` terminated list of command line arguments, - * including argument zero - */ -static void -load_from_cmdline(struct settings *settings, int argc, char *argv[]) -{ - const char *provider_name; - char *s, *end, *value; - - ARGBEGIN { - case 'b': - set_brightness(ARG(), &settings->day.brightness, &settings->night.brightness, NULL); - break; - - case 'c': - settings->config_file = ARG(); - break; - - case 'D': - settings->disabled.source |= SETTING_CMDLINE; - settings->disabled.value = 0; - break; - - case 'd': - settings->until_death = 1; - break; - - case 'E': - settings->scheme_type = CLOCK_SCHEME; - break; - - case 'e': - set_elevations(ARG(), &settings->elevation_high, &settings->elevation_low); - settings->scheme_type = SOLAR_SCHEME; - break; - - case 'g': - set_gamma(ARG(), &settings->day.gamma, &settings->night.gamma, NULL); - break; - - case 'H': - settings->hook_file.source |= SETTING_CMDLINE; - free(settings->hook_file.value); - settings->hook_file.value = ARG(); - break; - - case 'h': - print_help(); - exit(0); - - case 'l': - value = ARG(); - - /* Print list of providers if argument is `list' */ - if (!strcasecmp(value, "list")) { - print_provider_list(); - exit(0); - } - - provider_name = NULL; - - /* Don't save the result of strtof(); we simply want - to know if value can be parsed as a float. */ - errno = 0; - strtof(value, &end); - if (!errno && *end == ':') { - /* Use instead as arguments to `manual'. */ - provider_name = "manual"; - settings->provider_args = value; - } else { - /* Split off provider arguments. */ - s = strchr(value, ':'); - if (s) { - *s++ = '\0'; - settings->provider_args = s; - } - - provider_name = value; - } - - /* Lookup provider from name. */ - settings->provider = find_location_provider(provider_name); - - /* Print provider help if arg is `help'. */ - if (settings->provider_args && !strcasecmp(settings->provider_args, "help")) { - settings->provider->print_help(); - exit(0); - } - break; - - case 'm': - value = ARG(); - - /* Print list of methods if argument is `list' */ - if (!strcasecmp(value, "list")) { - print_method_list(); - exit(0); - } - - /* Split off method arguments. */ - s = strchr(value, ':'); - if (s) { - *s++ = '\0'; - settings->method_args = s; - } - - /* Find adjustment method by name. */ - settings->method = find_gamma_method(value); - - /* Print method help if arg is `help'. */ - if (settings->method_args && !strcasecmp(settings->method_args, "help")) { - settings->method->print_help(); - exit(0); - } - break; - - case 'O': - mode = PROGRAM_MODE_ONE_SHOT; - set_temperature(ARG(), &settings->day.temperature, &settings->night.temperature, NULL); - break; - - case 'o': - mode = PROGRAM_MODE_ONE_SHOT; - break; - - case 'P': - settings->preserve_gamma.source |= SETTING_CMDLINE; - settings->preserve_gamma.value = 0; - break; - - case 'p': - mode = PROGRAM_MODE_PRINT; - break; - - case 'r': - settings->use_fade.source |= SETTING_CMDLINE; - settings->use_fade.value = 0; - break; - - case 't': - set_temperature(ARG(), &settings->day.temperature, &settings->night.temperature, NULL); - break; - - case 'V': - printf("%s\n", VERSION_STRING); - exit(0); - - case 'v': - verbose = 1; - break; - - case 'x': - mode = PROGRAM_MODE_RESET; - break; - - default: - usage(); - - } ARGALT('+') { - case 'D': - settings->disabled.source |= SETTING_CMDLINE; - settings->disabled.value = 1; - break; - - case 'E': - settings->scheme_type = SOLAR_SCHEME; - break; - - case 'P': - settings->preserve_gamma.source |= SETTING_CMDLINE; - settings->preserve_gamma.value = 1; - break; - - case 'r': - settings->use_fade.source |= SETTING_CMDLINE; - settings->use_fade.value = 1; - break; - - default: - usage(); - } ARGEND; - - if (argc) - usage(); -} - - -/** - * Load an setting, form the "redshift" section, from the configuration file - * - * @param settings The currently loaded settings, will be updated - * @param key The name of the configuration - * @param value The value of the configuration - */ -static void -load_from_config_ini(struct settings *settings, const char *key, char *value) -{ - if (!strcasecmp(key, "temp") || !strcasecmp(key, "temperature")) { - set_temperature(value, &settings->day.temperature, &settings->night.temperature, key); - } else if (!strcasecmp(key, "temp-day") || !strcasecmp(key, "temperature-day")) { - set_temperature(value, &settings->day.temperature, NULL, key); - } else if (!strcasecmp(key, "temp-night") || !strcasecmp(key, "temperature-night")) { - set_temperature(value, NULL, &settings->night.temperature, key); - - } else if (!strcasecmp(key, "brightness")) { - set_brightness(value, &settings->day.brightness, &settings->night.brightness, key); - } else if (!strcasecmp(key, "brightness-day")) { - set_brightness(value, &settings->day.brightness, NULL, key); - } else if (!strcasecmp(key, "brightness-night")) { - set_brightness(value, NULL, &settings->night.brightness, key); - - } else if (!strcasecmp(key, "gamma")) { - set_gamma(value, &settings->day.gamma, &settings->night.gamma, key); - } else if (!strcasecmp(key, "gamma-day")) { - set_gamma(value, &settings->day.gamma, NULL, key); - } else if (!strcasecmp(key, "gamma-night")) { - set_gamma(value, NULL, &settings->night.gamma, key); - - } else if (!strcasecmp(key, "transition")) { - weprintf(_("`transition' is deprecated and has been replaced with `fade'.")); - goto set_use_fade; - } else if (!strcasecmp(key, "fade")) { - set_use_fade: - if (settings->use_fade.source & SETTING_CONFIGFILE) - weprintf(_("`fade' setting specified multiple times in configuration file.")); - settings->use_fade.source |= SETTING_CONFIGFILE; - if (settings->use_fade.source <= SETTING_CONFIGFILE) - settings->use_fade.value = get_boolean(value, key); - - } else if (!strcasecmp(key, "hook")) { - if (settings->hook_file.source & SETTING_CONFIGFILE) - weprintf(_("`hook' setting specified multiple times in configuration file.")); - settings->hook_file.source |= SETTING_CONFIGFILE; - if (settings->hook_file.source <= SETTING_CONFIGFILE) { - free(settings->hook_file.value); - settings->hook_file.value = estrdup(value); - } - - } else if (!strcasecmp(key, "preserve-gamma")) { - if (settings->preserve_gamma.source & SETTING_CONFIGFILE) - weprintf(_("`preserve-gamma' setting specified multiple times in configuration file.")); - settings->preserve_gamma.source |= SETTING_CONFIGFILE; - if (settings->preserve_gamma.source <= SETTING_CONFIGFILE) - settings->preserve_gamma.value = get_boolean(value, key); - - } else if (!strcasecmp(key, "start-disabled")) { - if (settings->disabled.source & SETTING_CONFIGFILE) - weprintf(_("`start-disabled' setting specified multiple times in configuration file.")); - settings->disabled.source |= SETTING_CONFIGFILE; - if (settings->disabled.source <= SETTING_CONFIGFILE) - settings->disabled.value = get_boolean(value, key); - - } else if (!strcasecmp(key, "elevation-high")) { - if (settings->elevation_high.source & SETTING_CONFIGFILE) - weprintf(_("`elevation-high' setting specified multiple times in configuration file.")); - settings->elevation_high.source |= SETTING_CONFIGFILE; - if (settings->elevation_high.source <= SETTING_CONFIGFILE) - settings->elevation_high.value = checked_atof(value, key); - - } else if (!strcasecmp(key, "elevation-low")) { - if (settings->elevation_low.source & SETTING_CONFIGFILE) - weprintf(_("`elevation-low' setting specified multiple times in configuration file.")); - settings->elevation_low.source |= SETTING_CONFIGFILE; - if (settings->elevation_low.source <= SETTING_CONFIGFILE) - settings->elevation_low.value = checked_atof(value, key); - - } else if (!strcasecmp(key, "dawn-time")) { - if (!set_transition_time(value, &settings->dawn_start, &settings->dawn_end, key)) - eprintf(_("Malformed dawn-time setting `%s'."), value); - - } else if (!strcasecmp(key, "dusk-time")) { - if (!set_transition_time(value, &settings->dusk_start, &settings->dusk_end, key)) - eprintf(_("Malformed dusk-time setting `%s'."), value); - - } else if (!strcasecmp(key, "adjustment-method")) { - if (!settings->method) - settings->method = find_gamma_method(value); - - } else if (!strcasecmp(key, "location-provider")) { - if (!settings->provider) - settings->provider = find_location_provider(value); - - } else { - weprintf(_("Unknown configuration setting `%s'."), key); - } -} - - -void -load_settings(struct settings *settings, int argc, char *argv[]) -{ - struct config_ini_section *section; - struct config_ini_setting *setting; - const struct time_period *first, *current; - const char *conf_path, *p; - char *conf_pathbuf, *s; - int i, j, n; - time_t duration; - size_t len; - - /* Clear unused bit so they do not interfere with comparsion */ - memset(&day_settings, 0, sizeof(day_settings)); - memset(&night_settings, 0, sizeof(night_settings)); - - /* Load settings; some validation takes place */ - load_defaults(settings); - load_from_cmdline(settings, argc, argv); - conf_path = config_ini_init(&settings->config, settings->config_file, &conf_pathbuf); - if ((section = config_ini_get_section(&settings->config, "redshift"))) - for (setting = section->settings; setting; setting = setting->next) - load_from_config_ini(settings, setting->name, setting->value); - - /* Further validate settings and select scheme */ - n = !!settings->dawn_start.source + !!settings->dawn_end.source; - n += !!settings->dusk_start.source + !!settings->dusk_end.source; - if (settings->scheme_type == SOLAR_SCHEME) { - n = 0; - } else if (settings->scheme_type == CLOCK_SCHEME) { - if (!n) - eprintf(_("`-E' option used with a time-configuration configured.")); - } - if (n) { - scheme.type = CLOCK_SCHEME; - if (n != 4) - eprintf(_("Partial time-configuration not supported!")); - - if (settings->dawn_start.value >= ONE_DAY || settings->dusk_start.value >= ONE_DAY || - labs((long)settings->dawn_end.value - (long)settings->dawn_start.value) > (long)ONE_DAY || - labs((long)settings->dusk_end.value - (long)settings->dusk_start.value) > (long)ONE_DAY) - goto invalid_twilight; - - /* TODO deal with edge-case where one of the twilights last 24 hour */ - settings->dawn_end.value %= ONE_DAY; - settings->dusk_end.value %= ONE_DAY; - if (settings->dawn_start.value <= settings->dawn_end.value) { - if (BETWEEN(settings->dawn_start.value, settings->dusk_start.value, settings->dawn_end.value) || - BETWEEN(settings->dawn_start.value, settings->dusk_end.value, settings->dawn_end.value)) - goto invalid_twilight; - } else { - if (!WITHIN(settings->dawn_end.value, settings->dusk_start.value, settings->dawn_start.value) || - !WITHIN(settings->dawn_end.value, settings->dusk_end.value, settings->dawn_start.value)) - goto invalid_twilight; - } - } - if (settings->elevation_high.value < settings->elevation_low.value) - eprintf(_("High transition elevation cannot be lower than the low transition elevation.")); - - /* If resetting effects, use neutral colour settings (static scheme) and do not preserve gamma */ - if (mode == PROGRAM_MODE_RESET) { - scheme.type = STATIC_SCHEME; - settings->preserve_gamma.value = 0; - day_settings = COLOUR_SETTING_NEUTRAL; - night_settings = COLOUR_SETTING_NEUTRAL; - goto settings_published; - } - - /* Make hook file absolute if set from config file (relative to config file) */ - if (settings->hook_file.source == SETTING_CONFIGFILE && conf_path) { -#ifdef WINDOWS - /* Regular absolute path */ - if (isalpha(settings->hook_file.value[0]) && settings->hook_file.value[1] == ':') - goto absolute_hook_path; - /* Absolute extended path or network path (relative extended paths do not exist) */ - if (settings->hook_file.value[0] == '\\' && settings->hook_file.value[1] == '\\') - goto absolute_hook_path; - /* Path relative to root */ - if (settings->hook_file.value[0] == '\\' || settings->hook_file.value[0] == '/') { - /* This is safe as we know that `conf_path` is a valid path */ - if (isalpha(conf_path[0]) && conf_path[1] == ':') { - p = &conf_path[3]; - goto base_found; - } else if (conf_path[0] == '\\' && conf_path[1] == '\\') { - p = &conf_path[2]; - while (*p == '\\') - p++; - while (*p != '/' && *p != '\\') - p++; - goto base_found; - } - } -#else - if (settings->hook_file.value[0] == '/') - goto absolute_hook_path; -#endif - p = strrchr(conf_path, '/'); - p = p ? &p[1] : conf_path; -#ifdef WINDOWS - if (strrchr(p, '\\')) - p = &strrchr(p, '\\')[1]; - base_found: -#endif - len = (size_t)(p - conf_path); - s = emalloc(len + strlen(settings->hook_file.value) + 1U); - memcpy(s, conf_path, len); - stpcpy(&s[len], settings->hook_file.value); - free(settings->hook_file.value); - settings->hook_file.value = s; -#ifdef WINDOWS - /* Used extended path is too long */ - if (settings->hook_file.value[0] != '\\' && (len = strlen(settings->hook_file.value)) >= 260) { - /* We have already made sure the path is absolute, so \ prefix is always extended or network path */ - settings->hook_file.value = erealloc(settings->hook_file.value, len + sizeof("\\\\?\\")); - memmove(&settings->hook_file.value[4], &settings->hook_file.value, len + 1U); - settings->hook_file.value[0] = '\\'; - settings->hook_file.value[1] = '\\'; - settings->hook_file.value[2] = '?'; - settings->hook_file.value[3] = '\\'; - } -#endif - } - free(conf_pathbuf); -absolute_hook_path: - - /* Publish loaded settings */ - if (mode == PROGRAM_MODE_ONE_SHOT && settings->until_death) - mode = PROGRAM_MODE_UNTIL_DEATH; - hook_file = settings->hook_file.value, settings->hook_file.value = NULL; - preserve_gamma = settings->preserve_gamma.value; - use_fade = settings->use_fade.value; - disable ^= settings->disabled.value; - day_settings.temperature = settings->day.temperature.value; - day_settings.brightness = settings->day.brightness.value; - day_settings.gamma[0] = settings->day.gamma.value[0]; - day_settings.gamma[1] = settings->day.gamma.value[1]; - day_settings.gamma[2] = settings->day.gamma.value[2]; - night_settings.temperature = settings->night.temperature.value; - night_settings.brightness = settings->night.brightness.value; - night_settings.gamma[0] = settings->night.gamma.value[0]; - night_settings.gamma[1] = settings->night.gamma.value[1]; - night_settings.gamma[2] = settings->night.gamma.value[2]; - if (!memcmp(&day_settings, &night_settings, sizeof(day_settings))) { - /* If the effects are the same throughout the day, do not use a transition scheme */ - scheme.type = STATIC_SCHEME; - } else if (scheme.type == SOLAR_SCHEME) { - scheme.elevation.high = settings->elevation_high.value; - scheme.elevation.low = settings->elevation_low.value; - scheme.elevation.range = scheme.elevation.high - scheme.elevation.low; - } else if (scheme.type == CLOCK_SCHEME) { - scheme.time.periods = &scheme.time.periods_array[0]; - scheme.time.periods_array[0].start = settings->dawn_start.value; - scheme.time.periods_array[0].day_level = 0.0; - scheme.time.periods_array[1].start = settings->dawn_end.value; - scheme.time.periods_array[1].day_level = 1.0; - scheme.time.periods_array[2].start = settings->dusk_start.value; - scheme.time.periods_array[2].day_level = 1.0; - scheme.time.periods_array[3].start = settings->dusk_end.value; - scheme.time.periods_array[3].day_level = 0.0; - for (i = 0; i < 4; i++) { - j = (i + 1) % 4; - scheme.time.periods_array[i].next = &scheme.time.periods_array[j]; - duration = scheme.time.periods_array[j].start - scheme.time.periods_array[i].start; - if (duration < 0) - duration += ONE_DAY; - scheme.time.periods_array[i].diff_over_duration = scheme.time.periods_array[j].day_level; - scheme.time.periods_array[i].diff_over_duration -= scheme.time.periods_array[i].day_level; - scheme.time.periods_array[i].diff_over_duration /= duration ? (double)duration : 1.0; - } - } -settings_published: - - /* Output settings */ - if (verbose) { - if (scheme.type == SOLAR_SCHEME) { - /* TRANSLATORS: Append degree symbols if possible. */ - printf(_("Solar elevations: day above %.1f, night below %.1f\n"), - scheme.elevation.high, scheme.elevation.low); - } else if (scheme.type == CLOCK_SCHEME) { - printf(_("Schedule:\n")); - current = first = scheme.time.periods; - do { - printf(_(" %.2f%% day at %02u:%02u:%02u\n"), - current->day_level * 100, (unsigned)(current->start / 60 / 60 % 24), - (unsigned)(current->start / 60 % 60), (unsigned)(current->start % 60)); - } while ((current = current->next) != first); - printf(_("(End of schedule)\n")); - } - printf(_("Temperatures: %luK at day, %luK at night\n"), - day_settings.temperature, night_settings.temperature); - printf(_("Brightness: %.2f:%.2f\n"), settings->day.brightness.value, settings->night.brightness.value); - /* TRANSLATORS: The string in parenthesis is either Daytime or Night (translated). */ - printf(_("Gamma (%s): %.3f, %.3f, %.3f\n"), _("Daytime"), - day_settings.gamma[0], day_settings.gamma[1], day_settings.gamma[2]); - printf(_("Gamma (%s): %.3f, %.3f, %.3f\n"), _("Night"), - night_settings.gamma[0], night_settings.gamma[1], night_settings.gamma[2]); - } - - return; - -invalid_twilight: - eprintf(_("Invalid dawn/dusk time configuration!")); -} |