diff options
author | Jon Lund Steffensen <jonlst@gmail.com> | 2010-10-15 12:43:48 +0200 |
---|---|---|
committer | Jon Lund Steffensen <jonlst@gmail.com> | 2010-10-15 12:43:48 +0200 |
commit | 2b92b8685727d708793756158493beb30719c875 (patch) | |
tree | 7cbcc0c33e3722fa05943fabe6163737bac6a933 /src | |
parent | Minor optimizations in selection of location provider and adjustment (diff) | |
download | redshift-ng-2b92b8685727d708793756158493beb30719c875.tar.gz redshift-ng-2b92b8685727d708793756158493beb30719c875.tar.bz2 redshift-ng-2b92b8685727d708793756158493beb30719c875.tar.xz |
Add configuration file support.
Diffstat (limited to 'src')
-rw-r--r-- | src/Makefile.am | 5 | ||||
-rw-r--r-- | src/config-ini.c | 235 | ||||
-rw-r--r-- | src/config-ini.h | 49 | ||||
-rw-r--r-- | src/redshift.c | 273 |
4 files changed, 502 insertions, 60 deletions
diff --git a/src/Makefile.am b/src/Makefile.am index 3d872ed..3680f4a 100644 --- a/src/Makefile.am +++ b/src/Makefile.am @@ -11,9 +11,10 @@ bin_PROGRAMS = redshift redshift_SOURCES = \ redshift.c redshift.h \ colorramp.c colorramp.h \ + config-ini.c config-ini.h \ + location-manual.c location-manual.h \ solar.c solar.h \ - systemtime.c systemtime.h \ - location-manual.c location-manual.h + systemtime.c systemtime.h EXTRA_redshift_SOURCES = \ gamma-randr.c gamma-randr.h \ diff --git a/src/config-ini.c b/src/config-ini.c new file mode 100644 index 0000000..01ba99c --- /dev/null +++ b/src/config-ini.c @@ -0,0 +1,235 @@ +/* config-ini.c -- INI config file parser + 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 <http://www.gnu.org/licenses/>. + + Copyright (c) 2010 Jon Lund Steffensen <jonlst@gmail.com> +*/ + + +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <errno.h> + +#include "config-ini.h" + +#ifdef ENABLE_NLS +# include <libintl.h> +# define _(s) gettext(s) +#else +# define _(s) s +#endif + +#define MAX_CONFIG_PATH 4096 +#define MAX_LINE_LENGTH 512 + + +static FILE * +open_config_file(const char *filepath) +{ + if (filepath == NULL) { + char cp[MAX_CONFIG_PATH]; + char *env; + + if ((env = getenv("XDG_CONFIG_HOME")) != NULL && + env[0] != '\0') { + snprintf(cp, sizeof(cp), "%s/redshift.conf", env); + filepath = cp; + } else if ((env = getenv("HOME")) != NULL && env[0] != '\0') { + snprintf(cp, sizeof(cp), + "%s/.config/redshift.conf", env); + filepath = cp; + } + + if (filepath != NULL) { + FILE *f = fopen(filepath, "r"); + if (f != NULL) return f; + else if (f == NULL && errno != ENOENT) return NULL; + } + + /* TODO look in getenv("XDG_CONFIG_DIRS") */ + } else { + FILE *f = fopen(filepath, "r"); + if (f == NULL) { + perror("fopen"); + return NULL; + } + } + + return NULL; +} + +int +config_ini_init(config_ini_state_t *state, const char *filepath) +{ + config_ini_section_t *section = NULL; + state->sections = NULL; + + FILE *f = open_config_file(filepath); + if (f == NULL) { + /* Only a serious error if a file was explicitly requested. */ + if (filepath != NULL) return -1; + return 0; + } + + char line[MAX_LINE_LENGTH]; + char *s; + + while (1) { + /* Handle the file input linewise. */ + char *r = fgets(line, sizeof(line), f); + if (r == NULL) break; + + /* Strip leading blanks and trailing newline. */ + s = line + strspn(line, " \t"); + s[strcspn(s, "\r\n")] = '\0'; + + /* Skip comments and empty lines. */ + if (s[0] == ';' || s[0] == '\0') continue; + + if (s[0] == '[') { + /* Read name of section. */ + const char *name = s+1; + char *end = strchr(s, ']'); + if (end == NULL || end[1] != '\0' || end == name) { + fputs(_("Malformed section header in config" + " file.\n"), stderr); + fclose(f); + config_ini_free(state); + return -1; + } + + *end = '\0'; + + /* Create section. */ + section = malloc(sizeof(config_ini_section_t)); + if (section == NULL) { + fclose(f); + config_ini_free(state); + return -1; + } + + /* Insert into section list. */ + section->name = NULL; + section->settings = NULL; + section->next = state->sections; + state->sections = section; + + /* Copy section name. */ + section->name = malloc(end - name + 1); + if (section->name == NULL) { + fclose(f); + config_ini_free(state); + return -1; + } + + memcpy(section->name, name, end - name + 1); + } else { + /* Split assignment at equals character. */ + char *end = strchr(s, '='); + if (end == NULL || end == s) { + fputs(_("Malformed assignment in config" + " file.\n"), stderr); + fclose(f); + config_ini_free(state); + return -1; + } + + *end = '\0'; + char *value = end + 1; + + if (section == NULL) { + fputs(_("Assignment outside section in config" + " file.\n"), stderr); + fclose(f); + config_ini_free(state); + return -1; + } + + /* Create section. */ + config_ini_setting_t *setting = + malloc(sizeof(config_ini_setting_t)); + if (setting == NULL) { + fclose(f); + config_ini_free(state); + return -1; + } + + /* Insert into section list. */ + setting->name = NULL; + setting->value = NULL; + setting->next = section->settings; + section->settings = setting; + + /* Copy name of setting. */ + setting->name = malloc(end - s + 1); + if (setting->name == NULL) { + fclose(f); + config_ini_free(state); + return -1; + } + + memcpy(setting->name, s, end - s + 1); + + /* Copy setting value. */ + size_t value_len = strlen(value) + 1; + setting->value = malloc(value_len); + if (setting->value == NULL) { + fclose(f); + config_ini_free(state); + return -1; + } + + memcpy(setting->value, value, value_len); + } + } + + fclose(f); + + return 0; +} + +void +config_ini_free(config_ini_state_t *state) +{ + config_ini_section_t *section = state->sections; + + while (section != NULL) { + config_ini_setting_t *setting = section->settings; + + while (setting != NULL) { + free(setting->name); + free(setting->value); + setting = setting->next; + } + + free(section->name); + section = section->next; + } +} + +config_ini_section_t * +config_ini_get_section(config_ini_state_t *state, const char *name) +{ + config_ini_section_t *section = state->sections; + while (section != NULL) { + if (strcasecmp(section->name, name) == 0) { + return section; + } + section = section->next; + } + + return NULL; +} diff --git a/src/config-ini.h b/src/config-ini.h new file mode 100644 index 0000000..e1bff73 --- /dev/null +++ b/src/config-ini.h @@ -0,0 +1,49 @@ +/* config-ini.h -- INI config file parser header + 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 <http://www.gnu.org/licenses/>. + + Copyright (c) 2010 Jon Lund Steffensen <jonlst@gmail.com> +*/ + +#ifndef _REDSHIFT_CONFIG_INI_H +#define _REDSHIFT_CONFIG_INI_H + +typedef struct _config_ini_section config_ini_section_t; +typedef struct _config_ini_setting config_ini_setting_t; + +struct _config_ini_setting { + config_ini_setting_t *next; + char *name; + char *value; +}; + +struct _config_ini_section { + config_ini_section_t *next; + char *name; + config_ini_setting_t *settings; +}; + +typedef struct { + config_ini_section_t *sections; +} config_ini_state_t; + + +int config_ini_init(config_ini_state_t *state, const char *filepath); +void config_ini_free(config_ini_state_t *state); + +config_ini_section_t *config_ini_get_section(config_ini_state_t *state, + const char *name); + +#endif /* ! _REDSHIFT_CONFIG_INI_H */ diff --git a/src/redshift.c b/src/redshift.c index 6871045..c75729a 100644 --- a/src/redshift.c +++ b/src/redshift.c @@ -42,6 +42,7 @@ #endif #include "redshift.h" +#include "config-ini.h" #include "solar.h" #include "systemtime.h" @@ -355,7 +356,8 @@ print_provider_list() static int provider_try_start(const location_provider_t *provider, - location_state_t *state, char *args) + location_state_t *state, + config_ini_state_t *config, char *args) { int r; @@ -366,7 +368,31 @@ provider_try_start(const location_provider_t *provider, return -1; } - /* Set provider options. */ + /* Set provider options from config file. */ + config_ini_section_t *section = + config_ini_get_section(config, provider->name); + if (section != NULL) { + config_ini_setting_t *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. */ while (args != NULL) { char *next_arg = strchr(args, ':'); if (next_arg != NULL) *(next_arg++) = '\0'; @@ -408,7 +434,8 @@ provider_try_start(const location_provider_t *provider, static int method_try_start(const gamma_method_t *method, - gamma_state_t *state, char *args) + gamma_state_t *state, + config_ini_state_t *config, char *args) { int r; @@ -419,7 +446,31 @@ method_try_start(const gamma_method_t *method, return -1; } - /* Set method options. */ + /* Set method options from config file. */ + config_ini_section_t *section = + config_ini_get_section(config, method->name); + if (section != NULL) { + config_ini_setting_t *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'; @@ -459,6 +510,60 @@ method_try_start(const gamma_method_t *method, return 0; } +/* A gamma string contains either one floating point value, + or three values separated by colon. */ +static int +parse_gamma_string(const char *str, float gamma[]) +{ + char *s = strchr(str, ':'); + if (s == NULL) { + /* Use value for all channels */ + float g = atof(str); + gamma[0] = gamma[1] = gamma[2] = g; + } else { + /* Parse separate value for each channel */ + *(s++) = '\0'; + char *g_s = s; + s = strchr(s, ':'); + if (s == NULL) return -1; + + *(s++) = '\0'; + gamma[0] = atof(str); /* Red */ + gamma[1] = atof(g_s); /* Blue */ + gamma[2] = atof(s); /* Green */ + } +} + +static const gamma_method_t * +find_gamma_method(const char *name) +{ + const gamma_method_t *method = NULL; + for (int i = 0; gamma_methods[i].name != NULL; i++) { + const gamma_method_t *m = &gamma_methods[i]; + if (strcasecmp(name, m->name) == 0) { + method = m; + break; + } + } + + return method; +} + +static const location_provider_t * +find_location_provider(const char *name) +{ + const location_provider_t *provider = NULL; + for (int i = 0; location_providers[i].name != NULL; i++) { + const location_provider_t *p = &location_providers[i]; + if (strcasecmp(name, p->name) == 0) { + provider = p; + break; + } + } + + return provider; +} + int main(int argc, char *argv[]) @@ -475,10 +580,10 @@ main(int argc, char *argv[]) textdomain(PACKAGE); #endif - /* Initialize to defaults */ - int temp_day = DEFAULT_DAY_TEMP; - int temp_night = DEFAULT_NIGHT_TEMP; - float gamma[3] = { DEFAULT_GAMMA, DEFAULT_GAMMA, DEFAULT_GAMMA }; + /* Initialize to NULL values. */ + int temp_day = -1; + int temp_night = -1; + float gamma[3] = { NAN, NAN, NAN }; const gamma_method_t *method = NULL; char *method_args = NULL; @@ -486,7 +591,7 @@ main(int argc, char *argv[]) const location_provider_t *provider = NULL; char *provider_args = NULL; - int transition = 1; + int transition = -1; program_mode_t mode = PROGRAM_MODE_CONTINUAL; int verbose = 0; char *s; @@ -496,28 +601,13 @@ main(int argc, char *argv[]) while ((opt = getopt(argc, argv, "g:hl:m:ort:vx")) != -1) { switch (opt) { case 'g': - s = strchr(optarg, ':'); - if (s == NULL) { - /* Use value for all channels */ - float g = atof(optarg); - gamma[0] = gamma[1] = gamma[2] = g; - } else { - /* Parse separate value for each channel */ - *(s++) = '\0'; - gamma[0] = atof(optarg); /* Red */ - char *g_s = s; - s = strchr(s, ':'); - if (s == NULL) { - fputs(_("Malformed gamma argument.\n"), - stderr); - fputs(_("Try `-h' for more" - " information.\n"), stderr); - exit(EXIT_FAILURE); - } - - *(s++) = '\0'; - gamma[1] = atof(g_s); /* Blue */ - gamma[2] = atof(s); /* Green */ + r = parse_gamma_string(optarg, gamma); + if (r < 0) { + fputs(_("Malformed gamma argument.\n"), + stderr); + fputs(_("Try `-h' for more" + " information.\n"), stderr); + exit(EXIT_FAILURE); } break; case 'h': @@ -553,17 +643,8 @@ main(int argc, char *argv[]) provider_name = optarg; } - /* Lookup argument in location provider table */ - for (int i = 0; location_providers[i].name != NULL; - i++) { - const location_provider_t *p = - &location_providers[i]; - if (strcasecmp(provider_name, p->name) == 0) { - provider = p; - break; - } - } - + /* Lookup provider from name. */ + provider = find_location_provider(provider_name); if (provider == NULL) { fprintf(stderr, _("Unknown location provider" " `%s'.\n"), provider_name); @@ -591,21 +672,13 @@ main(int argc, char *argv[]) method_args = s; } - /* Lookup argument in gamma methods table */ - for (int i = 0; gamma_methods[i].name != NULL; i++) { - const gamma_method_t *m = - &gamma_methods[i]; - if (strcasecmp(optarg, m->name) == 0) { - method = m; - break; - } - } - + /* Find adjustment method by name. */ + method = find_gamma_method(optarg); if (method == NULL) { /* TRANSLATORS: This refers to the method used to adjust colors e.g VidMode */ - fprintf(stderr, _("Unknown method `%s'.\n"), - optarg); + fprintf(stderr, _("Unknown adjustment method" + " `%s'.\n"), optarg); exit(EXIT_FAILURE); } @@ -648,6 +721,89 @@ main(int argc, char *argv[]) } } + /* Load settings from config file. */ + config_ini_state_t config_state; + r = config_ini_init(&config_state, NULL); + if (r < 0) { + fputs("Unable to load config file.\n", stderr); + exit(EXIT_FAILURE); + } + + /* Read global config settings. */ + config_ini_section_t *section = config_ini_get_section(&config_state, + "redshift"); + if (section != NULL) { + config_ini_setting_t *setting = section->settings; + while (setting != NULL) { + if (strcasecmp(setting->name, "temp-day") == 0) { + if (temp_day < 0) { + temp_day = atoi(setting->value); + } + } else if (strcasecmp(setting->name, + "temp-night") == 0) { + if (temp_night < 0) { + temp_night = atoi(setting->value); + } + } else if (strcasecmp(setting->name, + "transition") == 0) { + if (transition < 0) { + transition = !!atoi(setting->value); + } + } else if (strcasecmp(setting->name, "gamma") == 0) { + if (isnan(gamma[0])) { + r = parse_gamma_string(setting->value, + gamma); + if (r < 0) { + fputs(_("Malformed gamma" + " setting.\n"), + stderr); + exit(EXIT_FAILURE); + } + } + } else if (strcasecmp(setting->name, + "adjustment-method") == 0) { + if (method == NULL) { + method = find_gamma_method( + setting->value); + if (method == NULL) { + fprintf(stderr, _("Unknown" + " adjustment" + " method" + " `%s'.\n"), + setting->value); + exit(EXIT_FAILURE); + } + } + } else if (strcasecmp(setting->name, + "location-provider") == 0) { + if (provider == NULL) { + provider = find_location_provider( + setting->value); + if (provider == NULL) { + fprintf(stderr, _("Unknown" + " location" + " provider" + " `%s'.\n"), + setting->value); + exit(EXIT_FAILURE); + } + } + } else { + fprintf(stderr, _("Unknown configuration" + " setting `%s'.\n"), + setting->name); + } + setting = setting->next; + } + } + + /* Use default values for settings that were neither defined in + the config file nor on the command line. */ + if (temp_day < 0) temp_day = DEFAULT_DAY_TEMP; + if (temp_night < 0) temp_night = DEFAULT_NIGHT_TEMP; + if (isnan(gamma[0])) gamma[0] = gamma[1] = gamma[2] = DEFAULT_GAMMA; + if (transition < 0) transition = 1; + /* Initialize location provider. If provider is NULL try all providers until one that works is found. */ location_state_t location_state; @@ -657,7 +813,7 @@ main(int argc, char *argv[]) if (provider != NULL) { /* Use provider specified on command line. */ r = provider_try_start(provider, &location_state, - provider_args); + &config_state, provider_args); if (r < 0) exit(EXIT_FAILURE); } else { /* Try all providers, use the first that works. */ @@ -666,7 +822,7 @@ main(int argc, char *argv[]) const location_provider_t *p = &location_providers[i]; r = provider_try_start(p, &location_state, - NULL); + &config_state, NULL); if (r < 0) { fputs(_("Trying next provider...\n"), stderr); @@ -764,13 +920,14 @@ main(int argc, char *argv[]) if (method != NULL) { /* Use method specified on command line. */ - r = method_try_start(method, &state, method_args); + r = method_try_start(method, &state, &config_state, + 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 gamma_method_t *m = &gamma_methods[i]; - r = method_try_start(m, &state, NULL); + r = method_try_start(m, &state, &config_state, NULL); if (r < 0) { fputs(_("Trying next method...\n"), stderr); continue; |