aboutsummaryrefslogtreecommitdiffstats
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
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.
-rw-r--r--.travis.yml3
-rw-r--r--configure.ac26
-rw-r--r--po/POTFILES.in1
-rw-r--r--src/Makefile.am8
-rw-r--r--src/location-geoclue2.c281
-rw-r--r--src/location-geoclue2.h38
-rw-r--r--src/redshift.c20
7 files changed, 375 insertions, 2 deletions
diff --git a/.travis.yml b/.travis.yml
index 9cca098..eef7b6c 100644
--- a/.travis.yml
+++ b/.travis.yml
@@ -8,5 +8,6 @@ install:
- sudo apt-get install -qq libxcb1-dev libxcb-randr0-dev
- sudo apt-get install -qq libx11-dev libxxf86vm-dev
- sudo apt-get install -qq libgeoclue-dev
+ - sudo apt-get install -qq libglib2.0-dev
- sudo apt-get install -qq python3
-script: ./bootstrap && ./configure --enable-drm --enable-vidmode --enable-randr --enable-geoclue --enable-gui && make -j2 distcheck
+script: ./bootstrap && ./configure --enable-drm --enable-vidmode --enable-randr --enable-geoclue --enable-geoclue2 --enable-gui && make -j2 distcheck
diff --git a/configure.ac b/configure.ac
index 9fe8b13..1bb5424 100644
--- a/configure.ac
+++ b/configure.ac
@@ -32,6 +32,7 @@ PKG_CHECK_MODULES([XCB_RANDR], [xcb-randr],
PKG_CHECK_MODULES([GLIB], [glib-2.0 gobject-2.0], [have_glib=yes], [have_glib=no])
PKG_CHECK_MODULES([GEOCLUE], [geoclue], [have_geoclue=yes], [have_geoclue=no])
+PKG_CHECK_MODULES([GEOCLUE2], [glib-2.0 gio-2.0 >= 2.26], [have_geoclue2=yes], [have_geoclue2=no])
AC_CHECK_HEADER([windows.h], [have_windows_h=yes], [have_windows_h=no])
@@ -159,6 +160,30 @@ AS_IF([test "x$enable_geoclue" != xno], [
])
AM_CONDITIONAL([ENABLE_GEOCLUE], [test "x$enable_geoclue" = xyes])
+# Check Geoclue2 location provider
+AC_MSG_CHECKING([whether to enable Geoclue2 location provider])
+AC_ARG_ENABLE([geoclue2], [AC_HELP_STRING([--enable-geoclue2],
+ [enable Geoclue2 location provider])],
+ [enable_geoclue2=$enableval],[enable_geoclue2=maybe])
+AS_IF([test "x$enable_geoclue2" != xno], [
+ AS_IF([test "x$have_geoclue2" = xyes], [
+ AC_DEFINE([ENABLE_GEOCLUE2], 1,
+ [Define to 1 to enable Geoclue2 location provider])
+ AC_MSG_RESULT([yes])
+ enable_geoclue2=yes
+ ], [
+ AC_MSG_RESULT([missing dependencies])
+ AS_IF([test "x$enable_geoclue2" = xyes], [
+ AC_MSG_ERROR([missing dependencies for Geoclue2 location provider])
+ ])
+ enable_geoclue2=no
+ ])
+], [
+ AC_MSG_RESULT([no])
+ enable_geoclue2=no
+])
+AM_CONDITIONAL([ENABLE_GEOCLUE2], [test "x$enable_geoclue2" = xyes])
+
# Check for GUI status icon
AC_MSG_CHECKING([whether to enable GUI status icon])
@@ -249,6 +274,7 @@ echo "
Location providers:
Geoclue: ${enable_geoclue}
+ Geoclue2: ${enable_geoclue2}
GUI: ${enable_gui}
Ubuntu icons: ${enable_ubuntu}
diff --git a/po/POTFILES.in b/po/POTFILES.in
index 1a4c735..1f1a30e 100644
--- a/po/POTFILES.in
+++ b/po/POTFILES.in
@@ -13,6 +13,7 @@ src/gamma-w32gdi.c
src/gamma-dummy.c
src/location-geoclue.c
+src/location-geoclue2.c
src/location-manual.c
src/redshift-gtk/statusicon.py
diff --git a/src/Makefile.am b/src/Makefile.am
index 37a0308..eca9217 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -66,3 +66,11 @@ redshift_LDADD += \
$(GEOCLUE_LIBS) $(GEOCLUE_CFLAGS)
$(GLIB_LIBS) $(GLIB_CFLAGS)
endif
+
+if ENABLE_GEOCLUE2
+redshift_SOURCES += location-geoclue2.c location-geoclue2.h
+AM_CFLAGS += \
+ $(GEOCLUE2_CFLAGS)
+redshift_LDADD += \
+ $(GEOCLUE2_LIBS) $(GEOCLUE2_CFLAGS)
+endif
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;
+}
diff --git a/src/location-geoclue2.h b/src/location-geoclue2.h
new file mode 100644
index 0000000..f0553e4
--- /dev/null
+++ b/src/location-geoclue2.h
@@ -0,0 +1,38 @@
+/* location-geoclue2.h -- Geoclue2 location provider 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) 2014 Jon Lund Steffensen <jonlst@gmail.com>
+*/
+
+#ifndef REDSHIFT_LOCATION_GEOCLUE2_H
+#define REDSHIFT_LOCATION_GEOCLUE2_H
+
+#include <stdio.h>
+
+
+int location_geoclue2_init(void *state);
+int location_geoclue2_start(void *state);
+void location_geoclue2_free(void *state);
+
+void location_geoclue2_print_help(FILE *f);
+int location_geoclue2_set_option(void *state,
+ const char *key, const char *value);
+
+int location_geoclue2_get_location(void *state,
+ float *lat, float *lon);
+
+
+#endif /* ! REDSHIFT_LOCATION_GEOCLUE2_H */
diff --git a/src/redshift.c b/src/redshift.c
index 3f22f56..0c9a6bd 100644
--- a/src/redshift.c
+++ b/src/redshift.c
@@ -14,7 +14,7 @@
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) 2013 Jon Lund Steffensen <jonlst@gmail.com>
+ Copyright (c) 2009-2014 Jon Lund Steffensen <jonlst@gmail.com>
*/
#ifdef HAVE_CONFIG_H
@@ -76,6 +76,10 @@
# include "location-geoclue.h"
#endif
+#ifdef ENABLE_GEOCLUE2
+# include "location-geoclue2.h"
+#endif
+
/* Union of state data for gamma adjustment methods */
typedef union {
@@ -183,6 +187,20 @@ static const location_provider_t location_providers[] = {
location_geoclue_get_location
},
#endif
+#ifdef ENABLE_GEOCLUE2
+ {
+ "geoclue2",
+ (location_provider_init_func *)location_geoclue2_init,
+ (location_provider_start_func *)location_geoclue2_start,
+ (location_provider_free_func *)location_geoclue2_free,
+ (location_provider_print_help_func *)
+ location_geoclue2_print_help,
+ (location_provider_set_option_func *)
+ location_geoclue2_set_option,
+ (location_provider_get_location_func *)
+ location_geoclue2_get_location
+ },
+#endif
{
"manual",
(location_provider_init_func *)location_manual_init,