aboutsummaryrefslogtreecommitdiffstats
path: root/src/common.h
diff options
context:
space:
mode:
authorMattias Andrée <m@maandree.se>2025-03-16 14:45:03 +0100
committerMattias Andrée <m@maandree.se>2025-03-16 15:58:16 +0100
commitb55234a74d17503ca2fecb273cfcc44549f9e43e (patch)
tree321ef06dec317ff0e92011e1d680965e058b10fc /src/common.h
parentUnlist redshift/issues/846: rejected on technical grounds (diff)
downloadredshift-ng-b55234a74d17503ca2fecb273cfcc44549f9e43e.tar.gz
redshift-ng-b55234a74d17503ca2fecb273cfcc44549f9e43e.tar.bz2
redshift-ng-b55234a74d17503ca2fecb273cfcc44549f9e43e.tar.xz
Major refactoring and some fixes
Signed-off-by: Mattias Andrée <m@maandree.se>
Diffstat (limited to 'src/common.h')
-rw-r--r--src/common.h413
1 files changed, 343 insertions, 70 deletions
diff --git a/src/common.h b/src/common.h
index ebf63e0..704e64a 100644
--- a/src/common.h
+++ b/src/common.h
@@ -1,5 +1,7 @@
-/* common.h -- Common header file for Redshift source files
- * This file is part of redshift-ng.
+/* 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
@@ -13,9 +15,6 @@
*
* You should have received a copy of the GNU General Public License
* along with redshift-ng. If not, see <http://www.gnu.org/licenses/>.
- *
- * Copyright (c) 2009-2017 Jon Lund Steffensen <jonlst@gmail.com>
- * Copyright (c) 2014, 2015, 2016, 2025 Mattias Andrée <m@maandree.se>
*/
#ifndef REDSHIFT_COMMON_H
#define REDSHIFT_COMMON_H
@@ -31,6 +30,7 @@
#include <sys/stat.h>
#include <sys/types.h>
+#include <ctype.h>
#include <dirent.h>
#include <errno.h>
#include <fcntl.h>
@@ -116,56 +116,82 @@
*/
#define WITHIN(LO, X, UP) ((LO) <= (X) && (X) <= (UP))
+/**
+ * Check whether a value is within a bounded, open range
+ *
+ * @param LO The lower bound
+ * @param X The value to check
+ * @param UP The upper bound
+ * @return :int 1 if `X` is within (`LO`, `UP`), 0 otherwise
+ */
+#define BETWEEN(LO, X, UP) ((LO) < (X) && (X) < (UP))
+
+
+/**
+ * Symbol used to delimit paths in environment
+ * variables listing multiple paths
+ */
+#ifdef WINDOWS
+# define PATH_DELIMITER ';'
+#else
+# define PATH_DELIMITER ':'
+#endif
+
+/**
+ * The number of seconds in a day
+ */
+#define ONE_DAY ((time_t)(24L * 60L * 60L))
+
/**
* Minimum valid latitude
*/
-#define MIN_LATITUDE -90.0
+#define MIN_LATITUDE -90.0
/**
* Maximum valid latitude
*/
-#define MAX_LATITUDE 90.0
+#define MAX_LATITUDE 90.0
/**
* Minimum valid longitude
*/
-#define MIN_LONGITUDE -180.0
+#define MIN_LONGITUDE -180.0
/**
* Maximum valid longitude
*/
-#define MAX_LONGITUDE 180.0
+#define MAX_LONGITUDE 180.0
/**
* Minimum allowed colour temperature
*/
-#define MIN_TEMPERATURE ((unsigned long int)LIBRED_LOWEST_TEMPERATURE)
+#define MIN_TEMPERATURE ((unsigned long int)LIBRED_LOWEST_TEMPERATURE)
/**
* Maximum allowed colour temperature
*/
-#define MAX_TEMPERATURE ULONG_MAX
+#define MAX_TEMPERATURE ULONG_MAX
/**
* Minimum allowed whitepoint brightness
*/
-#define MIN_BRIGHTNESS 0.1
+#define MIN_BRIGHTNESS 0.1
/**
* Maximum allowed whitepoint brightness
*/
-#define MAX_BRIGHTNESS 1.0
+#define MAX_BRIGHTNESS 1.0
/**
* Minimum allowed gamma
*/
-#define MIN_GAMMA 0.1
+#define MIN_GAMMA 0.1
/**
* Maximum allowed gamma
*/
-#define MAX_GAMMA 10.0
+#define MAX_GAMMA 10.0
/**
@@ -186,6 +212,49 @@
/**
+ * Default daytime colour temperature
+ */
+#define DEFAULT_DAY_TEMPERATURE 6500UL
+
+/**
+ * Default night colour temperature
+ */
+#define DEFAULT_NIGHT_TEMPERATURE 4500UL
+
+/**
+ * Default daytime whitepoint brightness level
+ */
+#define DEFAULT_DAY_BRIGHTNESS NEUTRAL_BRIGHTNESS
+
+/**
+ * Default night whitepoint brightness level
+ */
+#define DEFAULT_NIGHT_BRIGHTNESS NEUTRAL_BRIGHTNESS
+
+/**
+ * Default daytime gamma value
+ */
+#define DEFAULT_DAY_GAMMA NEUTRAL_GAMMA
+
+/**
+ * Default night gamma value
+ */
+#define DEFAULT_NIGHT_GAMMA NEUTRAL_GAMMA
+
+/**
+ * The solar elevation, in degrees, that marks the
+ * threshold to daytime
+ */
+#define DEFAULT_HIGH_ELEVATION 3.0
+
+/**
+ * The solar elevation, in degrees, that marks the
+ * threshold to night
+ */
+#define DEFAULT_LOW_ELEVATION LIBRED_SOLAR_ELEVATION_CIVIL_DUSK_DAWN
+
+
+/**
* Initialiser for `struct colour_setting`
*
* Sets all values to their neutral values (no effects applied)
@@ -230,6 +299,83 @@ enum program_mode {
/**
+ * By what the effects of the application change
+ */
+enum scheme_type {
+ /**
+ * Effects are dependent on the Sun's elevation
+ */
+ SOLAR_SCHEME,
+
+ /**
+ * Effects are dependent on the wall clock time
+ */
+ CLOCK_SCHEME,
+
+ /**
+ * Effects do not change
+ */
+ STATIC_SCHEME
+};
+
+
+/**
+ * The sources where an setting was been loaded from
+ *
+ * Higher valued sources have higher priority
+ *
+ * This is a bitmask `enum`
+ */
+enum setting_source {
+ /**
+ * No setting loaded, default value set
+ */
+ SETTING_DEFAULT = 0x00,
+
+ /**
+ * Setting loaded from configuration file
+ */
+ SETTING_CONFIGFILE = 0x01,
+
+ /**
+ * Setting loaded from command line arguments
+ */
+ SETTING_CMDLINE = 0x02
+};
+
+
+/**
+ * Specification for a path that consists of a two parts:
+ * the first being defined by the environment, and the
+ * seocnd being a static string
+ */
+struct env_path {
+ /**
+ * Whether the environment variable referenced by `.prefix`
+ * should be split at each colon (:) into multiple paths to
+ * test
+ *
+ * On Windows semicolon (;) is used instead of colon
+ */
+ int multidir_env;
+
+ /**
+ * Environment variable to use as the first part of the path
+ *
+ * `NULL` if the user's home directory should be used
+ *
+ * The empty string if `.suffix` should be used as is
+ */
+ const char *prefix_env;
+
+ /**
+ * The second part of the path
+ */
+ const char *suffix;
+};
+
+
+/**
* Geographical location, using GPS coordinates
*/
struct location {
@@ -270,22 +416,34 @@ struct colour_setting {
/* Time range.
Fields are offsets from midnight in seconds. */
struct time_range {
- int start;
- int end;
+ time_t start;
+ time_t end;
+};
+
+
+struct time_period {
+ time_t start;
+ double day_level;
+ double diff_over_duration;
+ const struct time_period *next;
};
/* Transition scheme.
- The solar elevations at which the transition begins/ends,
- and the association colour settings. */
-struct transition_scheme {
- double high;
- double low;
- int use_time; /* When enabled, ignore elevation and use time ranges. */
- struct time_range dawn;
- struct time_range dusk;
- struct colour_setting day;
- struct colour_setting night;
+ The solar elevations at which the transition begins/ends */
+union scheme {
+ enum scheme_type type;
+ struct {
+ enum scheme_type type;
+ double high;
+ double low;
+ double range;
+ } elevation;
+ struct {
+ enum scheme_type type;
+ const struct time_period *periods;
+ struct time_period periods_array[4];
+ } time;
};
@@ -308,20 +466,51 @@ struct config_ini_state {
};
-struct options {
+struct setting_i {
+ enum setting_source source;
+ int value;
+};
+
+struct setting_lu {
+ enum setting_source source;
+ unsigned long int value;
+};
+
+struct setting_f {
+ enum setting_source source;
+ double value;
+};
+
+struct setting_f3 {
+ enum setting_source source;
+ double value[3];
+};
+
+struct setting_time {
+ enum setting_source source;
+ time_t value;
+};
+
+struct settings {
/* Path to config file */
- char *config_filepath;
+ const char *config_file;
- struct transition_scheme scheme;
- enum program_mode mode;
- int verbose;
+ struct config_ini_state config;
+
+ struct {
+ struct setting_lu temperature;
+ struct setting_f brightness;
+ struct setting_f3 gamma;
+ } day, night;
+ struct setting_i preserve_gamma; /* Whether to preserve gamma ramps if supported by gamma method. */
+ struct setting_i use_fade; /* Whether to fade between large skips in colour temperature. */
- /* Temperature to set in manual mode. */
- unsigned long int temp_set;
- /* Whether to fade between large skips in colour temperature. */
- int use_fade;
- /* Whether to preserve gamma ramps if supported by gamma method. */
- int preserve_gamma;
+ struct setting_f elevation_high; /* TODO no cmdline option */
+ struct setting_f elevation_low; /* TODO no cmdline option */
+ struct {
+ struct setting_time start;
+ struct setting_time end;
+ } dawn, dusk; /* TODO no cmdline option */
/* Selected gamma method. */
const struct gamma_method *method;
@@ -332,6 +521,10 @@ struct options {
const struct location_provider *provider;
/* Arguments for location provider. */
char *provider_args;
+
+ enum program_mode mode;
+ int verbose;
+ enum scheme_type scheme_type;
};
@@ -397,6 +590,33 @@ extern const struct gamma_method *gamma_methods[];
*/
extern const struct location_provider *location_providers[];
+/**
+ * Set to 1 once the process has received a signal to terminate
+ */
+extern volatile sig_atomic_t exiting;
+
+/**
+ * Set to 1 once the process has received a signal to remove its effect
+ */
+extern volatile sig_atomic_t disable;
+
+/**
+ * The colour settings applied at daytime
+ */
+extern struct colour_setting day_settings;
+
+/**
+ * The colour settings applied at nighttime
+ */
+extern struct colour_setting night_settings;
+
+/**
+ * The colour settings applied at nighttime
+ */
+extern union scheme scheme;
+
+
+/* colour.c */
#define LIST_RAMPS_STOP_VALUE_TYPES(X, D)\
X(u8, uint8_t, UINT8_MAX, 8) D\
@@ -418,13 +638,15 @@ extern const struct location_provider *location_providers[];
* @param size_b The number of stops in `gamma_b`
* @param settings The colour settings to apply (temperature, brightness, gamma)
*/\
- void colourramp_fill_##SUFFIX(TYPE *gamma_r, TYPE *gamma_g, TYPE *gamma_b,\
- size_t size_r, size_t size_g, size_t size_b,\
- const struct colour_setting *setting)
+ void fill_ramps_##SUFFIX(TYPE *gamma_r, TYPE *gamma_g, TYPE *gamma_b,\
+ size_t size_r, size_t size_g, size_t size_b,\
+ const struct colour_setting *setting)
LIST_RAMPS_STOP_VALUE_TYPES(X, ;);
#undef X
+/* config-ini.c */
+
/**
* Load the configuration file
*
@@ -453,61 +675,112 @@ GCC_ONLY(__attribute__((__pure__)))
struct config_ini_section *config_ini_get_section(struct config_ini_state *state, const char *name);
-void options_init(struct options *options);
-void options_parse_args(struct options *options, int argc, char *argv[]);
-void options_parse_config_file(struct options *options, struct config_ini_state *config_state);
-void options_set_defaults(struct options *options);
+/* config.c */
+
+/**
+ * Load settings
+ *
+ * @param settings Output parameter for the settings
+ * @param argc Number of command line arguments
+ * @param argv `NULL` terminated list of command line arguments,
+ * including argument zero
+ */
+void load_settings(struct settings *settings, int argc, char *argv[]);
+
+
+/* gamma.c */
+
+void acquire_adjustment_method(struct settings *settings, GAMMA_STATE **method_state_out);
-void hooks_signal_period_change(enum period prev_period, enum period period);
+/* location.c */
+int get_location(const struct location_provider *provider, LOCATION_STATE *state, int timeout, struct location *loc);
+
+void acquire_location_provider(struct settings *settings, LOCATION_STATE **location_state_out);
+
+
+/* hooks.c */
/**
- * Create a pipe(7) where both ends have O_NONBLOCK applied
+ * Run hooks with a signal that the period changed
*
- * @param pipefds Output parameter for the pipe's file descriptors:
- * 0) reading file descriptor, and
- * 1) writing file descriptor.
- * @return 0 on success, -1 on failure
+ * @param prev_period The previous period
+ * @param period The new current period
*/
-int pipeutils_create_nonblocking(int pipefds[2]);
+void run_period_change_hooks(enum period prev_period, enum period period);
-void pipeutils_signal(int write_fd);
-void pipeutils_handle_signal(int read_fd);
+/* signals.c */
/**
- * Set to 1 once the process has received a signal to terminate
+ * Install signal handlers for the process
*/
-extern volatile sig_atomic_t exiting;
+void install_signal_handlers(void);
+
+
+/* util.c */
/**
- * Set to 1 once the process has received a signal to remove its effect
+ * Remove trailing whitespace
+ *
+ * @param s The string to trim, will be truncated
+ * @param end The current end of `s`; will be looked up if `NULL`
+ * @return `s`
*/
-extern volatile sig_atomic_t disable;
+char *rtrim(char *s, char *end);
/**
- * Install signal handlers for the process
+ * Remove leading whitespace
+ *
+ * @param s The string to trim (will not be modified)
+ * @return `s` with an offset
*/
-void signals_install_handlers(void);
-
+GCC_ONLY(__attribute__((__warn_unused_result__, __pure__)))
+char *ltrim(char *s);
/**
- * Get the current time in seconds since the Unix epoch
+ * Get the user's home directory
*
- * @return The current time
+ * This function looks up the user's home directory
+ * once and caches the result, later calls to this
+ * function will use the cached result
+ *
+ * @return The user's home directory; the empty string if not found
*/
-double systemtime_get_time(void);
+const char *get_home(void);
/**
- * Suspend the process for a short time
- *
- * The process may be resumed earily, specifically
- * if it receives a signal
+ * Search for a file and open it for reading
+ *
+ * @param path_spec Specification for the path to try
+ * @param path_out Output parameter for the found file
+ * @param pathbuf_out Output parameter for the memory allocation for `*path_out`;
+ * shall be free(3)d by the caller
+ * @return `FILE` object for the reading the file; `NULL` if not found
+ */
+FILE *try_path_fopen(const struct env_path *path_spec, const char **path_out, char **pathbuf_out);
+
+/**
+ * Search for a directory and open it for reading
+ *
+ * @param path_spec Specification for the path to try
+ * @param path_out Output parameter for the found directory
+ * @param pathbuf_out Output parameter for the memory allocation for `*path_out`;
+ * shall be free(3)d by the caller
+ * @return `DIR` object for the reading the directory; `NULL` if not found
+ */
+DIR *try_path_opendir(const struct env_path *path_spec, const char **path_out, char **pathbuf_out);
+
+/**
+ * Create a pipe(7) where both ends have `O_NONBLOCK` applied
*
- * @param msecs The number of milliseconds to sleep
+ * @param pipefds Output parameter for the pipe's file descriptors:
+ * 0) reading file descriptor, and
+ * 1) writing file descriptor
+ * @return 0 on success, -1 on failure
*/
-void systemtime_msleep(unsigned int msecs);
+int pipe_nonblock(int pipefds[2]);
extern const struct gamma_method dummy_gamma_method;