aboutsummaryrefslogtreecommitdiffstats
path: root/src/location.c
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/location.c
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/location.c')
-rw-r--r--src/location.c187
1 files changed, 187 insertions, 0 deletions
diff --git a/src/location.c b/src/location.c
new file mode 100644
index 0000000..216b7a9
--- /dev/null
+++ b/src/location.c
@@ -0,0 +1,187 @@
+/* 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"
+
+
+const struct location_provider *location_providers[] = {
+#ifdef ENABLE_GEOCLUE2
+ &geoclue2_location_provider,
+#endif
+#ifdef ENABLE_CORELOCATION
+ &corelocation_location_provider,
+#endif
+ &manual_location_provider,
+ NULL
+};
+
+
+
+static int
+try_start(const struct location_provider *provider, LOCATION_STATE **state, struct config_ini_state *config, char *args)
+{
+ const char *manual_keys[] = {"lat", "lon"};
+ struct config_ini_section *section;
+ struct config_ini_setting *setting;
+ char *next_arg, *value;
+ const char *key;
+ int i;
+
+ if (provider->init(state) < 0) {
+ weprintf(_("Initialization of %s failed."), provider->name);
+ return -1;
+ }
+
+ /* Set provider options from config file. */
+ if ((section = config_ini_get_section(config, provider->name)))
+ for (setting = section->settings; setting; setting = setting->next)
+ if (provider->set_option(*state, setting->name, setting->value) < 0)
+ goto set_option_fail;
+
+ /* Set provider options from command line. */
+ for (i = 0; args; i++) {
+ next_arg = strchr(args, ':');
+ if (next_arg)
+ *next_arg++ = '\0';
+
+ key = args;
+ value = strchr(args, '=');
+ if (!value) {
+ /* 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") && i < ELEMSOF(manual_keys)) {
+ key = manual_keys[i];
+ value = args;
+ } else {
+ weprintf(_("Failed to parse option `%s'."), args);
+ return -1;
+ }
+ } else {
+ *value++ = '\0';
+ }
+
+ if (provider->set_option(*state, key, value) < 0)
+ goto set_option_fail;
+
+ args = next_arg;
+ }
+
+ /* Start provider. */
+ if (provider->start(*state) < 0) {
+ provider->free(*state);
+ weprintf(_("Failed to start provider %s."), provider->name);
+ return -1;
+ }
+
+ return 0;
+
+set_option_fail:
+ provider->free(*state);
+ weprintf(_("Failed to set %s option."), provider->name);
+ /* TRANSLATORS: `help' must not be translated. */
+ weprintf(_("Try `-l %s:help' for more information."), provider->name);
+ return -1;
+}
+
+
+static long long int
+get_monotonic_millis(void)
+{
+#if defined(WINDOWS)
+ return (long long int)GetTickCount64();
+#else
+ struct timespec now;
+ if (clock_gettime(CLOCK_MONOTONIC, &now))
+ eprintf("clock_gettime CLOCK_MONOTONIC:");
+ return (long long int)now.tv_sec * 1000LL + (long long int)now.tv_nsec / 1000000LL;
+#endif
+}
+
+
+/* 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. */
+int
+get_location(const struct location_provider *provider, LOCATION_STATE *state, int timeout, struct location *loc)
+{
+ int r, available;
+ struct pollfd pollfds[1];
+ long long int now = get_monotonic_millis();
+ long long int end = now + (long long int)timeout;
+
+ do {
+ pollfds[0].fd = provider->get_fd(state);
+ if (pollfds[0].fd >= 0) {
+ /* Poll on file descriptor until ready. */
+ pollfds[0].events = POLLIN;
+ timeout = (int)MAX(end - now, 0);
+ r = poll(pollfds, 1, timeout);
+ if (r > 0) {
+ now = get_monotonic_millis();
+ } else if (r < 0) {
+#ifndef WINDOWS
+ if (errno == EINTR)
+ continue;
+#endif
+ weprintf("poll {{.fd=<location provider>, .events=EPOLLIN}} 1 %i:", timeout);
+ return -1;
+ } else {
+ return 0;
+ }
+ }
+
+ if (provider->handle(state, loc, &available) < 0)
+ return -1;
+ } while (!available);
+
+ return 1;
+}
+
+
+void
+acquire_location_provider(struct settings *settings, LOCATION_STATE **location_state_out)
+{
+ size_t i;
+
+ if (settings->provider) {
+ /* Use provider specified on command line. */
+ if (try_start(settings->provider, location_state_out, &settings->config, settings->provider_args) < 0)
+ exit(1);
+ } else {
+ /* Try all providers, use the first that works. */
+ for (i = 0; location_providers[i]; i++) {
+ weprintf(_("Trying location provider `%s'..."), location_providers[i]->name);
+ if (try_start(location_providers[i], location_state_out, &settings->config, NULL) < 0) {
+ weprintf(_("Trying next provider..."));
+ continue;
+ }
+
+ /* Found provider that works. */
+ printf(_("Using provider `%s'.\n"), location_providers[i]->name);
+ settings->provider = location_providers[i];
+ break;
+ }
+
+ /* Failure if no providers were successful at this point. */
+ if (!settings->provider)
+ eprintf(_("No more location providers to try."));
+ }
+}