aboutsummaryrefslogtreecommitdiffstats
path: root/libgeome_get_from_command.c
diff options
context:
space:
mode:
Diffstat (limited to 'libgeome_get_from_command.c')
-rw-r--r--libgeome_get_from_command.c100
1 files changed, 100 insertions, 0 deletions
diff --git a/libgeome_get_from_command.c b/libgeome_get_from_command.c
new file mode 100644
index 0000000..4d1c5af
--- /dev/null
+++ b/libgeome_get_from_command.c
@@ -0,0 +1,100 @@
+/* See LICENSE file for copyright and license details. */
+#include "common.h"
+
+
+static char *
+spawn_read(struct libgeome_context *ctx, const char *path, const char *const *argv, size_t limit, unsigned int alarm_seconds)
+{
+ struct spawn_join_info info;
+ char *ret = NULL, *new;
+ size_t size = 0;
+ size_t len = 0;
+ ssize_t r;
+ int fd, status;
+
+ if (libgeome_util_spawn_async_read(ctx, path, argv, alarm_seconds, &info, &fd))
+ return NULL;
+
+ for (;;) {
+ if (len == size) {
+ size += 512U;
+ new = realloc(ret, size);
+ if (!new) {
+ ctx->print_error(ctx, "realloc: %s\n", strerror(errno));
+ goto read_fail;
+ }
+ ret = new;
+ }
+ r = read(fd, &ret[len], size - len);
+ if (r <= 0) {
+ if (!r)
+ break;
+ if (errno == EINTR)
+ continue;
+ ctx->print_error(ctx, "read <pipe>: %s\n", strerror(errno));
+ read_fail:
+ close(fd);
+ free(ret);
+ ret = NULL;
+ libgeome_util_spawn_async_kill(ctx, &info, SIGKILL);
+ break;
+ }
+ len += (size_t)r;
+ if (len >= limit)
+ break;
+ }
+
+ if (len) {
+ if (len == size)
+ len--;
+ ret[len] = '\0';
+ } else {
+ free(ret);
+ ret = NULL;
+ }
+
+ if (libgeome_util_spawn_async_join(ctx, &info, &status) || status != 0) {
+ free(ret);
+ ret = NULL;
+ }
+ return ret;
+}
+
+
+int
+libgeome_get_from_command(struct libgeome_context *ctx, struct libgeome_data *out, size_t limit,
+ unsigned int alarm_secs, const char *path, const char *const *argv)
+{
+ struct location location;
+ char *text;
+
+ text = spawn_read(ctx, path, argv, limit ? limit : 2048U, alarm_secs);
+ if (!text)
+ return -1;
+ out->requested_data &= libgeome_util_parse_output(text, &location);
+ free(text);
+
+ if (out->requested_data & LIBGEOME_DATUM_LATITUDE) {
+ if (!isfinite(location.latitude) || location.latitude < -90 || location.latitude > -90) {
+ ctx->print_debug(ctx, "invalide latitude retrieved: %g\n", location.latitude);
+ out->requested_data ^= LIBGEOME_DATUM_LATITUDE;
+ } else {
+ out->latitude = location.latitude;
+ }
+ }
+
+ if (out->requested_data & LIBGEOME_DATUM_LONGITUDE) {
+ if (!isfinite(location.longitude) || location.longitude < -90 || location.longitude > -90) {
+ ctx->print_debug(ctx, "invalide longitude retrieved: %g\n", location.longitude);
+ out->requested_data ^= LIBGEOME_DATUM_LONGITUDE;
+ } else {
+ out->longitude = location.longitude;
+ }
+ }
+
+ return 0;
+}
+
+
+extern inline int libgeome_get_from_netservice(struct libgeome_context *ctx, struct libgeome_data *out,
+ const struct libgeome_netservice *svc);