aboutsummaryrefslogtreecommitdiffstats
path: root/src/location-geoclue2.c
diff options
context:
space:
mode:
authorJon Lund Steffensen <jonlst@gmail.com>2014-12-14 16:19:25 -0500
committerJon Lund Steffensen <jonlst@gmail.com>2014-12-15 01:47:49 -0500
commitee7e91dd8692f51acfde6a534766ff391000ed82 (patch)
treecacf18fd8b6f2c1ac9a4d1951bd85de114fa5957 /src/location-geoclue2.c
parentsystemtime: Use gettimeofday if POSIX timers not available (diff)
downloadredshift-ng-ee7e91dd8692f51acfde6a534766ff391000ed82.tar.gz
redshift-ng-ee7e91dd8692f51acfde6a534766ff391000ed82.tar.bz2
redshift-ng-ee7e91dd8692f51acfde6a534766ff391000ed82.tar.xz
Fix #80: Add Geoclue2 location provider
This simply runs a GLib main loop when the location is requested and tries to get the location from GeoClue2. We need to run a main loop since the location is not immediately available, instead it becomes available through a signal.
Diffstat (limited to 'src/location-geoclue2.c')
-rw-r--r--src/location-geoclue2.c281
1 files changed, 281 insertions, 0 deletions
diff --git a/src/location-geoclue2.c b/src/location-geoclue2.c
new file mode 100644
index 0000000..4c4150e
--- /dev/null
+++ b/src/location-geoclue2.c
@@ -0,0 +1,281 @@
+/* location-geoclue2.c -- GeoClue2 location provider source
+ 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) 2014 Jon Lund Steffensen <jonlst@gmail.com>
+*/
+
+#include <stdio.h>
+#include <stdlib.h>
+
+#include <glib.h>
+#include <glib/gprintf.h>
+#include <gio/gio.h>
+
+#include "location-geoclue2.h"
+
+#ifdef ENABLE_NLS
+# include <libintl.h>
+# define _(s) gettext(s)
+#else
+# define _(s) s
+#endif
+
+
+typedef struct {
+ GMainLoop *loop;
+
+ int available;
+ float latitude;
+ float longitude;
+} get_location_data_t;
+
+
+int
+location_geoclue2_init(void *state)
+{
+#if !GLIB_CHECK_VERSION(2, 35, 0)
+ g_type_init();
+#endif
+ return 0;
+}
+
+int
+location_geoclue2_start(void *state)
+{
+ return 0;
+}
+
+void
+location_geoclue2_free(void *state)
+{
+}
+
+void
+location_geoclue2_print_help(FILE *f)
+{
+ fputs(_("Use the location as discovered by a GeoClue2 provider.\n"), f);
+ fputs("\n", f);
+
+ fputs(_("NOTE: currently redshift doesn't recheck GeoClue2 once started,\n"
+ "which means it has to be restarted to take notice after travel.\n"),
+ f);
+ fputs("\n", f);
+}
+
+int
+location_geoclue2_set_option(void *state,
+ const char *key, const char *value)
+{
+ fprintf(stderr, _("Unknown method parameter: `%s'.\n"), key);
+ return -1;
+}
+
+/* Handle position change callbacks */
+static void
+geoclue_client_signal_cb(GDBusProxy *client, gchar *sender_name,
+ gchar *signal_name, GVariant *parameters,
+ gpointer *user_data)
+{
+ get_location_data_t *data = (get_location_data_t *)user_data;
+
+ /* Only handle LocationUpdated signals */
+ if (g_strcmp0(signal_name, "LocationUpdated") != 0) {
+ return;
+ }
+
+ /* Obtain location path */
+ const gchar *location_path;
+ g_variant_get_child(parameters, 1, "&o", &location_path);
+
+ /* Obtain location */
+ GError *error = NULL;
+ GDBusProxy *location =
+ g_dbus_proxy_new_for_bus_sync(G_BUS_TYPE_SYSTEM,
+ G_DBUS_PROXY_FLAGS_NONE,
+ NULL,
+ "org.freedesktop.GeoClue2",
+ location_path,
+ "org.freedesktop.GeoClue2.Location",
+ NULL, &error);
+ if (location == NULL) {
+ g_printerr(_("Unable to obtain location: %s.\n"),
+ error->message);
+ g_error_free(error);
+ return;
+ }
+
+ /* Read location properties */
+ GVariant *lat_v = g_dbus_proxy_get_cached_property(location,
+ "Latitude");
+ data->latitude = g_variant_get_double(lat_v);
+
+ GVariant *lon_v = g_dbus_proxy_get_cached_property(location,
+ "Longitude");
+ data->longitude = g_variant_get_double(lon_v);
+
+ data->available = 1;
+
+ /* Return from main loop */
+ g_main_loop_quit(data->loop);
+}
+
+/* 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)
+{
+ get_location_data_t *data = (get_location_data_t *)user_data;
+
+ /* Obtain GeoClue Manager */
+ GError *error = NULL;
+ GDBusProxy *geoclue_manager =
+ g_dbus_proxy_new_for_bus_sync(G_BUS_TYPE_SYSTEM,
+ G_DBUS_PROXY_FLAGS_NONE,
+ NULL,
+ "org.freedesktop.GeoClue2",
+ "/org/freedesktop/GeoClue2/Manager",
+ "org.freedesktop.GeoClue2.Manager",
+ NULL, &error);
+ if (geoclue_manager == NULL) {
+ g_printerr(_("Unable to obtain GeoClue Manager: %s.\n"),
+ error->message);
+ g_error_free(error);
+ return;
+ }
+
+ /* Obtain GeoClue Client path */
+ error = NULL;
+ GVariant *client_path_v =
+ g_dbus_proxy_call_sync(geoclue_manager,
+ "GetClient",
+ NULL,
+ G_DBUS_CALL_FLAGS_NONE,
+ -1, NULL, &error);
+ if (client_path_v == NULL) {
+ g_printerr(_("Unable to obtain GeoClue client path: %s.\n"),
+ error->message);
+ g_error_free(error);
+ g_object_unref(geoclue_manager);
+ return;
+ }
+
+ const gchar *client_path;
+ g_variant_get(client_path_v, "(&o)", &client_path);
+
+ /* Obtain GeoClue client */
+ error = NULL;
+ GDBusProxy *geoclue_client =
+ g_dbus_proxy_new_for_bus_sync(G_BUS_TYPE_SYSTEM,
+ G_DBUS_PROXY_FLAGS_NONE,
+ NULL,
+ "org.freedesktop.GeoClue2",
+ client_path,
+ "org.freedesktop.GeoClue2.Client",
+ NULL, &error);
+ if (geoclue_client == NULL) {
+ g_printerr(_("Unable to obtain GeoClue Client: %s.\n"),
+ error->message);
+ g_error_free(error);
+ g_variant_unref(client_path_v);
+ g_object_unref(geoclue_manager);
+ return;
+ }
+
+ g_variant_unref(client_path_v);
+
+ /* Set distance threshold */
+ error = NULL;
+ GVariant *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 == NULL) {
+ g_printerr(_("Unable to set distance threshold: %s.\n"),
+ error->message);
+ g_error_free(error);
+ g_object_unref(geoclue_client);
+ g_object_unref(geoclue_manager);
+ return;
+ }
+
+ g_variant_unref(ret_v);
+
+ /* Attach signal callback to client */
+ g_signal_connect(geoclue_client, "g-signal",
+ G_CALLBACK(geoclue_client_signal_cb),
+ 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 == NULL) {
+ g_printerr(_("Unable to start GeoClue client: %s.\n"),
+ error->message);
+ g_error_free(error);
+ g_object_unref(geoclue_client);
+ g_object_unref(geoclue_manager);
+ 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)
+{
+ get_location_data_t *data = (get_location_data_t *)user_data;
+
+ g_fprintf(stderr, _("Unable to connect to GeoClue.\n"));
+
+ g_main_loop_quit(data->loop);
+}
+
+int
+location_geoclue2_get_location(void *state,
+ float *lat, float *lon)
+{
+ get_location_data_t data;
+ data.available = 0;
+
+ guint 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,
+ &data, NULL);
+ data.loop = g_main_loop_new(NULL, FALSE);
+ g_main_loop_run(data.loop);
+
+ g_bus_unwatch_name(watcher_id);
+
+ if (!data.available) return -1;
+
+ *lat = data.latitude;
+ *lon = data.longitude;
+
+ return 0;
+}