diff options
author | Mattias Andrée <m@maandree.se> | 2025-03-24 22:58:14 +0100 |
---|---|---|
committer | Mattias Andrée <m@maandree.se> | 2025-03-24 22:58:14 +0100 |
commit | 12d7f9b45303fb458cb21a3e0e430af96c781d8b (patch) | |
tree | 2b9c674f474d8ab4d6b35c657d4c005b4089192d /libgeome_get_from_command.c | |
download | libgeome-12d7f9b45303fb458cb21a3e0e430af96c781d8b.tar.gz libgeome-12d7f9b45303fb458cb21a3e0e430af96c781d8b.tar.bz2 libgeome-12d7f9b45303fb458cb21a3e0e430af96c781d8b.tar.xz |
First commit
Signed-off-by: Mattias Andrée <m@maandree.se>
Diffstat (limited to 'libgeome_get_from_command.c')
-rw-r--r-- | libgeome_get_from_command.c | 100 |
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); |