diff options
Diffstat (limited to 'redshift/location-geoclue2.c')
-rw-r--r-- | redshift/location-geoclue2.c | 451 |
1 files changed, 451 insertions, 0 deletions
diff --git a/redshift/location-geoclue2.c b/redshift/location-geoclue2.c new file mode 100644 index 0000000..9eb49fe --- /dev/null +++ b/redshift/location-geoclue2.c @@ -0,0 +1,451 @@ +/*- + * 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" + +#if defined(__clang__) +# pragma clang diagnostic push +# pragma clang diagnostic ignored "-Wreserved-identifier" +# pragma clang diagnostic ignored "-Wreserved-macro-identifier" +# pragma clang diagnostic ignored "-Wdocumentation-unknown-command" +# pragma clang diagnostic ignored "-Wdocumentation" +# pragma clang diagnostic ignored "-Wpadded" +#endif +#include <glib.h> +#include <glib/gprintf.h> +#include <gio/gio.h> +#if defined(__clang__) +# pragma clang diagnostic pop +#endif + + +/** + * D-Bus error indicating denial of access + */ +#define DBUS_ACCESS_ERROR "org.freedesktop.DBus.Error.AccessDenied" + + +/** + * Location data + */ +struct location_data { + /** + * The user's geographical location + */ + struct location location; + + /** + * Whether the location provider is available + */ + int available; + + /** + * Whether an unrecoverable error has occurred + */ + int error; +}; + + +struct location_state { + GMainLoop *loop; + + /** + * Slave thread, used to receive location updates + */ + GThread *thread; + + /** + * Read-end of piped used to send location data + * from the slave thread to the master thread + */ + int pipe_fd_read; + + /** + * Write-end of piped used to send location data + * from the slave thread to the master thread + */ + int pipe_fd_write; + + /** + * Location data available from the slave thread + */ + struct location_data data; + + /** + * Location data sent to the master thread + */ + struct location_data saved_data; +}; + + +/* Print the message explaining denial from GeoClue */ +static void +print_denial_message(void) +{ + g_printerr(_( + "Access to the current location was denied by GeoClue!\n" + "Make sure that location services are enabled and that" + " Redshift is permitted\nto use location services." + " See https://github.com/jonls/redshift#faq for more\n" + "information.\n")); +} + + +static void +send_data(struct location_state *state) +{ + while (write(state->pipe_fd_write, &state->data, sizeof(state->data)) == -1 && errno == EINTR); +} + + +/* Indicate an unrecoverable error during GeoClue2 communication */ +static void +mark_error(struct location_state *state) +{ + state->data.error = 1; + send_data(state); +} + + +/* Handle position change callbacks */ +static void +geoclue_client_signal_cb(GDBusProxy *client, gchar *sender_name, gchar *signal_name, GVariant *parameters, gpointer user_data) +{ + struct location_state *state = user_data; + const gchar *location_path; + GDBusProxy *location; + GError *error; + GVariant *lat_v, *lon_v; + + (void) sender_name; + + /* Only handle LocationUpdated signals */ + if (g_strcmp0(signal_name, "LocationUpdated")) + return; + + /* Obtain location path */ + g_variant_get_child(parameters, 1, "&o", &location_path); + + /* Obtain location */ + error = NULL; + location = g_dbus_proxy_new_sync(g_dbus_proxy_get_connection(client), G_DBUS_PROXY_FLAGS_NONE, + NULL, "org.freedesktop.GeoClue2", location_path, + "org.freedesktop.GeoClue2.Location", NULL, &error); + if (!location) { + weprintf(_("Unable to obtain location: %s."), error->message); + g_error_free(error); + mark_error(state); + return; + } + + /* Read location properties */ + lat_v = g_dbus_proxy_get_cached_property(location, "Latitude"); + state->data.location.latitude = g_variant_get_double(lat_v); + lon_v = g_dbus_proxy_get_cached_property(location, "Longitude"); + state->data.location.longitude = g_variant_get_double(lon_v); + state->data.available = 1; + + send_data(state); +} + + +/* Callback when GeoClue name appears on the bus */ +static void +on_name_appeared(GDBusConnection *conn, const gchar *name, const gchar *name_owner, gpointer user_data) +{ + struct location_state *state = user_data; + const gchar *client_path; + GDBusProxy *geoclue_client; + GVariant *client_path_v; + GDBusProxy *geoclue_manager; + GError *error; + GVariant *ret_v; + gchar *dbus_error; + + (void) name; + (void) name_owner; + + /* Obtain GeoClue Manager */ + error = NULL; + geoclue_manager = g_dbus_proxy_new_sync(conn, G_DBUS_PROXY_FLAGS_NONE, NULL, + "org.freedesktop.GeoClue2", "/org/freedesktop/GeoClue2/Manager", + "org.freedesktop.GeoClue2.Manager", NULL, &error); + if (!geoclue_manager) { + weprintf(_("Unable to obtain GeoClue Manager: %s."), error->message); + g_error_free(error); + mark_error(state); + return; + } + + /* Obtain GeoClue Client path */ + error = NULL; + client_path_v = g_dbus_proxy_call_sync(geoclue_manager, "GetClient", NULL, + G_DBUS_CALL_FLAGS_NONE, -1, NULL, &error); + if (!client_path_v) { + weprintf(_("Unable to obtain GeoClue client path: %s."), error->message); + g_error_free(error); + g_object_unref(geoclue_manager); + mark_error(state); + return; + } + + g_variant_get(client_path_v, "(&o)", &client_path); + + /* Obtain GeoClue client */ + error = NULL; + geoclue_client = g_dbus_proxy_new_sync(conn, G_DBUS_PROXY_FLAGS_NONE, NULL, "org.freedesktop.GeoClue2", + client_path, "org.freedesktop.GeoClue2.Client", NULL, &error); + if (!geoclue_client) { + weprintf(_("Unable to obtain GeoClue Client: %s."), error->message); + g_error_free(error); + g_variant_unref(client_path_v); + g_object_unref(geoclue_manager); + mark_error(state); + return; + } + + g_variant_unref(client_path_v); + + /* Set desktop id (basename of the .desktop file) */ + error = NULL; + ret_v = g_dbus_proxy_call_sync(geoclue_client, "org.freedesktop.DBus.Properties.Set", + g_variant_new("(ssv)", "org.freedesktop.GeoClue2.Client", + "DesktopId", g_variant_new("s", "redshift")), + G_DBUS_CALL_FLAGS_NONE, -1, NULL, &error); + if (!ret_v) { + /* Ignore this error for now. The property is not available + * in early versions of GeoClue2. */ + } else { + g_variant_unref(ret_v); + } + + /* Set distance threshold */ + error = NULL; + ret_v = g_dbus_proxy_call_sync(geoclue_client, "org.freedesktop.DBus.Properties.Set", + g_variant_new("(ssv)", "org.freedesktop.GeoClue2.Client", + "DistanceThreshold", g_variant_new("u", 50000)), + G_DBUS_CALL_FLAGS_NONE, -1, NULL, &error); + if (!ret_v) { + weprintf(_("Unable to set distance threshold: %s."), error->message); + g_error_free(error); + g_object_unref(geoclue_client); + g_object_unref(geoclue_manager); + mark_error(state); + return; + } + + g_variant_unref(ret_v); + + /* Attach signal callback to client */ + g_signal_connect(geoclue_client, "g-signal", G_CALLBACK(geoclue_client_signal_cb), user_data); + + /* Start GeoClue client */ + error = NULL; + ret_v = g_dbus_proxy_call_sync(geoclue_client, "Start", NULL, G_DBUS_CALL_FLAGS_NONE, -1, NULL, &error); + if (!ret_v) { + weprintf(_("Unable to start GeoClue client: %s."), error->message); + if (g_dbus_error_is_remote_error(error)) { + dbus_error = g_dbus_error_get_remote_error( error); + if (!g_strcmp0(dbus_error, DBUS_ACCESS_ERROR)) + print_denial_message(); + g_free(dbus_error); + } + g_error_free(error); + g_object_unref(geoclue_client); + g_object_unref(geoclue_manager); + mark_error(state); + return; + } + + g_variant_unref(ret_v); +} + + +/* Callback when GeoClue disappears from the bus */ +static void +on_name_vanished(GDBusConnection *connection, const gchar *name, gpointer user_data) +{ + struct location_state *state = user_data; + + (void) connection; + (void) name; + + state->data.available = 0; + send_data(state); +} + + +/* Callback when the pipe to the main thread is closed */ +static gboolean +on_pipe_closed(GIOChannel *channel, GIOCondition condition, gpointer user_data) +{ + struct location_state *state = user_data; + g_main_loop_quit(state->loop); + + (void) channel; + (void) condition; + return FALSE; +} + + +/* Run loop for location provider thread */ +static void * +run_geoclue2_loop(void *state_) +{ + struct location_state *state = state_; + GMainContext *context; + guint watcher_id; + GIOChannel *pipe_channel; + GSource *pipe_source; + + context = g_main_context_new(); + g_main_context_push_thread_default(context); + state->loop = g_main_loop_new(context, FALSE); + + watcher_id = g_bus_watch_name(G_BUS_TYPE_SYSTEM, "org.freedesktop.GeoClue2", + G_BUS_NAME_WATCHER_FLAGS_AUTO_START, + on_name_appeared, on_name_vanished, state, NULL); + + /* Listen for closure of pipe */ + pipe_channel = g_io_channel_unix_new(state->pipe_fd_write); + pipe_source = g_io_create_watch(pipe_channel, G_IO_IN | G_IO_HUP | G_IO_ERR); + g_source_set_callback(pipe_source, (GSourceFunc)on_pipe_closed, state, NULL); + g_source_attach(pipe_source, context); + + g_main_loop_run(state->loop); + + g_source_unref(pipe_source); + g_io_channel_unref(pipe_channel); + close(state->pipe_fd_write); + + g_bus_unwatch_name(watcher_id); + + g_main_loop_unref(state->loop); + g_main_context_unref(context); + + return NULL; +} + + +static int +geoclue2_create(struct location_state **state_out) +{ +#if !GLIB_CHECK_VERSION(2, 35, 0) + g_type_init(); +#endif + *state_out = emalloc(sizeof(**state_out)); + return 0; +} + + +static int +geoclue2_start(struct location_state *state) +{ + int pipefds[2]; + + state->pipe_fd_read = -1; + state->pipe_fd_write = -1; + + state->data.available = 0; + state->data.error = 0; + state->data.location.latitude = 0; + state->data.location.longitude = 0; + state->saved_data = state->data; + + pipe_rdnonblock(pipefds); + state->pipe_fd_read = pipefds[0]; + state->pipe_fd_write = pipefds[1]; + + send_data(state); /* TODO why? */ + + state->thread = g_thread_new("geoclue2", run_geoclue2_loop, state); + + return 0; +} + + +static void +geoclue2_free(struct location_state *state) +{ + if (state->pipe_fd_read >= 0) + close(state->pipe_fd_read); + + /* Closing the pipe should cause the thread to exit, but it may be blocked by I/O */ + install_forceful_exit_signal_handlers(); + g_thread_join(state->thread); + state->thread = NULL; + + free(state); +} + + +static void +geoclue2_print_help(void) +{ + printf(_("Use the location as discovered by a GeoClue2 provider.\n")); + printf("\n"); +} + + +static int +geoclue2_set_option(struct location_state *state, const char *key, const char *value) +{ + (void) state; + (void) value; + weprintf(_("Unknown provider parameter: `%s'."), key); + return -1; +} + + +static int +geoclue2_get_fd(struct location_state *state) +{ + return state->pipe_fd_read; +} + + +static int +geoclue2_fetch(struct location_state *state, struct location *location_out, int *available_out) +{ + struct location_data data; + ssize_t r; + + for (;;) { + r = read(state->pipe_fd_read, &data, sizeof(data)); + if (r == (ssize_t)sizeof(data)) { + state->saved_data = data; + } else if (r > 0) { + /* writes of 512 bytes or less are always atomic on pipes */ + weprintf("read <pipe>: %s", _("Unexpected message size")); + } else if (!r || errno == EAGAIN) { + break; + } else if (errno != EINTR) { + weprintf("read <pipe>:"); + state->saved_data.error = 1; + break; + } + } + + *location_out = state->saved_data.location; + *available_out = state->saved_data.available; + return state->saved_data.error ? -1 : 0; +} + + +const struct location_provider geoclue2_location_provider = LOCATION_PROVIDER_INIT("geoclue2", geoclue2); |