aboutsummaryrefslogtreecommitdiffstats
path: root/libgeome_get_from_timezone.c
diff options
context:
space:
mode:
Diffstat (limited to 'libgeome_get_from_timezone.c')
-rw-r--r--libgeome_get_from_timezone.c221
1 files changed, 221 insertions, 0 deletions
diff --git a/libgeome_get_from_timezone.c b/libgeome_get_from_timezone.c
new file mode 100644
index 0000000..a9d3b69
--- /dev/null
+++ b/libgeome_get_from_timezone.c
@@ -0,0 +1,221 @@
+/* See LICENSE file for copyright and license details. */
+#include "common.h"
+
+
+static char *
+get_timezone(struct libgeome_context *ctx, int *env_determined_out)
+{
+ const char *tzdir;
+ const char *tz;
+ char *tzbuf = NULL;
+ char *ret;
+ size_t tzdirlen;
+
+ tz = getenv("TZ");
+ *env_determined_out = !!tz;
+ if (!tz) {
+ tzdir = getenv("TZDIR");
+ if (!tzdir)
+ tzdir = TZDIR;
+ tz = tzbuf = libgeome_util_areadlink(ctx, TZFILE);
+ if (!tz)
+ return NULL;
+ tzdirlen = strlen(tzdir);
+ if (!strncmp(tz, tzdir, tzdirlen) && tz[tzdirlen] == '/') {
+ tz = &tz[tzdirlen];
+ while (*tz == '/')
+ tz++;
+ }
+ }
+
+ ret = strdup(tz);
+ if (!ret)
+ ctx->print_error(ctx, "strdup: %s\n", strerror(errno));
+ free(tzbuf);
+ return ret;
+}
+
+
+static int
+get_timezone_location3(struct libgeome_context *ctx, const char *tz, const char *tabfile, struct location *location_out)
+{
+ const char *tzdir, *coords;
+ char *path, *line = NULL, *p;
+ size_t size = 0, tzlen = strlen(tz);
+ int lat_sign, lat_degrees, lat_minutes, lat_seconds = 0;
+ int lon_sign, lon_degrees, lon_minutes, lon_seconds = 0;
+ ssize_t len;
+ FILE *f;
+
+ tzdir = getenv("TZDIR");
+ if (!tzdir)
+ tzdir = TZDIR;
+
+ path = malloc(strlen(tzdir) + strlen(tabfile) + sizeof("/"));
+ if (!path) {
+ ctx->print_error(ctx, "malloc: %s\n", strerror(errno));
+ return -1;
+ }
+ stpcpy(stpcpy(stpcpy(path, tzdir), "/"), tabfile);
+
+ f = fopen(path, "r");
+ if (!f) {
+ if (errno == ENOENT)
+ ctx->print_debug(ctx, "fopen %s r: %s\n", path, strerror(errno));
+ else
+ ctx->print_error(ctx, "fopen %s r: %s\n", path, strerror(errno));
+ free(path);
+ return -1;
+ }
+
+again:
+ while (len = getline(&line, &size, f), len >= 0) {
+ p = line;
+ for (;; p++) {
+ if (!*p || *p == '\r' || *p == '\n' || *p == '#')
+ goto next_line;
+ if (*p == '\t')
+ break;
+ }
+
+ coords = ++p;
+ for (;; p++) {
+ if (!*p || *p == '\r' || *p == '\n' || *p == '#')
+ goto next_line;
+ if (*p == '\t')
+ break;
+ }
+ *p++ = '\0';
+
+ if (strncmp(p, tz, tzlen))
+ goto next_line;
+ p = &p[tzlen];
+ if (*p && *p != '\r' && *p != '\n' && *p != '#' && *p != '\t')
+ goto next_line;
+
+ goto found_coords;
+
+ next_line:;
+ }
+ if (!feof(f)) {
+ if (errno == EINTR) {
+ clearerr(f);
+ goto again;
+ }
+ ctx->print_error(ctx, "getline %s: %s\n", path, strerror(errno));
+ }
+ ctx->print_debug(ctx, "entry for timezone %s not found in %s\n", tz, path);
+
+ free(line);
+ free(path);
+ fclose(f);
+ return -1;
+
+bad_coords:
+ ctx->print_debug(ctx, "malformatted coordinates for %s in %s\n", tz, path);
+ free(line);
+ free(path);
+ fclose(f);
+ return -1;
+
+found_coords:
+ if (coords[0] != '+' && coords[0] != '-')
+ goto bad_coords;
+ if (!isdigit(coords[1]) || !isdigit(coords[2]))
+ goto bad_coords;
+ if (!isdigit(coords[3]) || !isdigit(coords[4]))
+ goto bad_coords;
+ lat_sign = coords[0] == '-' ? -1 : +1;
+ lat_degrees = (coords[1] & 15) * 10 + (coords[2] & 15);
+ lat_minutes = (coords[3] & 15) * 10 + (coords[4] & 15);
+ coords = &coords[5];
+ if (isdigit(coords[0]) && isdigit(coords[1])) {
+ lat_seconds = (coords[0] & 15) * 10 + (coords[1] & 15);
+ coords = &coords[2];
+ }
+
+ if (coords[0] != '+' && coords[0] != '-')
+ goto bad_coords;
+ if (!isdigit(coords[1]) || !isdigit(coords[2]) || !isdigit(coords[3]))
+ goto bad_coords;
+ if (!isdigit(coords[4]) || !isdigit(coords[5]))
+ goto bad_coords;
+ lon_sign = coords[0] == '-' ? -1 : +1;
+ lon_degrees = (coords[1] & 15) * 100 + (coords[2] & 15) * 10 + (coords[3] & 15);
+ lon_minutes = (coords[4] & 15) * 10 + (coords[5] & 15);
+ coords = &coords[6];
+ if (isdigit(coords[0]) && isdigit(coords[1])) {
+ lon_seconds = (coords[0] & 15) * 10 + (coords[1] & 15);
+ coords = &coords[2];
+ }
+
+ if (*coords)
+ goto bad_coords;
+
+ if (lat_degrees > 90 || lon_degrees > 180)
+ goto bad_coords;
+ if (lat_minutes >= 60 || lon_minutes >= 60)
+ goto bad_coords;
+ if (lat_seconds >= 60 || lon_seconds >= 60)
+ goto bad_coords;
+ if (lat_degrees == 90 && lat_minutes + lat_seconds)
+ goto bad_coords;
+ if (lon_degrees == 180 && lon_minutes + lon_seconds)
+ goto bad_coords;
+
+ location_out->latitude = (double)lat_degrees;
+ location_out->latitude += (double)lat_minutes / 60;
+ location_out->latitude += (double)lat_seconds / 3600;
+ location_out->latitude *= (double)lat_sign;
+
+ location_out->longitude = (double)lon_degrees;
+ location_out->longitude += (double)lon_minutes / 60;
+ location_out->longitude += (double)lon_seconds / 3600;
+ location_out->longitude *= (double)lon_sign;
+
+ free(line);
+ free(path);
+ fclose(f);
+ return 0;
+}
+
+
+static int
+get_timezone_location(struct libgeome_context *ctx, const char *tz, struct location *location_out)
+{
+ if (!get_timezone_location3(ctx, tz, "zone1970.tab", location_out))
+ return 0;
+ if (!get_timezone_location3(ctx, tz, "zone.tab", location_out))
+ return 0;
+ if (!get_timezone_location3(ctx, tz, "zonenow.tab", location_out))
+ return 0;
+ return -1;
+}
+
+
+int
+libgeome_get_from_timezone(struct libgeome_context *ctx, struct libgeome_data *out)
+{
+ struct location location;
+ char *tz;
+
+ tz = get_timezone(ctx, &(int){0});
+ if (!tz)
+ return -1;
+ if (get_timezone_location(ctx, tz, &location)) {
+ free(tz);
+ return -1;
+ }
+ free(tz);
+
+ out->requested_data &= LIBGEOME_DATUM_LATITUDE | LIBGEOME_DATUM_LONGITUDE;
+ if (out->requested_data) {
+ if (out->requested_data & LIBGEOME_DATUM_LATITUDE)
+ out->latitude = location.latitude;
+ if (out->requested_data & LIBGEOME_DATUM_LONGITUDE)
+ out->longitude = location.longitude;
+ }
+
+ return 0;
+
+}