2016-06-01 22:49:41 +00:00
|
|
|
#include <stdio.h>
|
2016-06-28 22:18:27 +00:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
2014-10-25 00:22:14 +00:00
|
|
|
#include <math.h>
|
2018-05-18 21:45:03 +00:00
|
|
|
#include <atomic>
|
2016-04-27 21:00:14 +00:00
|
|
|
#include "projection.hpp"
|
2014-10-25 00:22:14 +00:00
|
|
|
|
2016-06-28 22:18:27 +00:00
|
|
|
struct projection projections[] = {
|
|
|
|
{"EPSG:4326", lonlat2tile, tile2lonlat, "urn:ogc:def:crs:OGC:1.3:CRS84"},
|
|
|
|
{"EPSG:3857", epsg3857totile, tiletoepsg3857, "urn:ogc:def:crs:EPSG::3857"},
|
2017-11-07 18:48:19 +00:00
|
|
|
{NULL, NULL, NULL, NULL},
|
2016-06-28 22:18:27 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct projection *projection = &projections[0];
|
|
|
|
|
2014-10-25 00:22:14 +00:00
|
|
|
// http://wiki.openstreetmap.org/wiki/Slippy_map_tilenames
|
2016-06-01 22:49:41 +00:00
|
|
|
void lonlat2tile(double lon, double lat, int zoom, long long *x, long long *y) {
|
2018-05-15 19:55:17 +00:00
|
|
|
// Place infinite and NaN coordinates off the edge of the Mercator plane
|
|
|
|
|
|
|
|
int lat_class = fpclassify(lat);
|
|
|
|
int lon_class = fpclassify(lon);
|
|
|
|
|
|
|
|
if (lat_class == FP_INFINITE || lat_class == FP_NAN) {
|
|
|
|
lat = 89.9;
|
|
|
|
}
|
|
|
|
if (lon_class == FP_INFINITE || lon_class == FP_NAN) {
|
|
|
|
lon = 360;
|
|
|
|
}
|
|
|
|
|
2015-12-09 00:24:17 +00:00
|
|
|
// Must limit latitude somewhere to prevent overflow.
|
|
|
|
// 89.9 degrees latitude is 0.621 worlds beyond the edge of the flat earth,
|
|
|
|
// hopefully far enough out that there are few expectations about the shape.
|
|
|
|
if (lat < -89.9) {
|
|
|
|
lat = -89.9;
|
|
|
|
}
|
|
|
|
if (lat > 89.9) {
|
|
|
|
lat = 89.9;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lon < -360) {
|
|
|
|
lon = -360;
|
|
|
|
}
|
|
|
|
if (lon > 360) {
|
|
|
|
lon = 360;
|
|
|
|
}
|
|
|
|
|
2014-10-25 00:22:14 +00:00
|
|
|
double lat_rad = lat * M_PI / 180;
|
|
|
|
unsigned long long n = 1LL << zoom;
|
|
|
|
|
|
|
|
long long llx = n * ((lon + 180) / 360);
|
2015-06-03 18:21:40 +00:00
|
|
|
long long lly = n * (1 - (log(tan(lat_rad) + 1 / cos(lat_rad)) / M_PI)) / 2;
|
2014-10-25 00:22:14 +00:00
|
|
|
|
|
|
|
*x = llx;
|
|
|
|
*y = lly;
|
|
|
|
}
|
|
|
|
|
|
|
|
// http://wiki.openstreetmap.org/wiki/Slippy_map_tilenames
|
2016-06-01 22:49:41 +00:00
|
|
|
void tile2lonlat(long long x, long long y, int zoom, double *lon, double *lat) {
|
2014-10-25 00:22:14 +00:00
|
|
|
unsigned long long n = 1LL << zoom;
|
|
|
|
*lon = 360.0 * x / n - 180.0;
|
2015-10-12 19:51:55 +00:00
|
|
|
*lat = atan(sinh(M_PI * (1 - 2.0 * y / n))) * 180.0 / M_PI;
|
2014-10-25 00:22:14 +00:00
|
|
|
}
|
|
|
|
|
2016-06-01 22:49:41 +00:00
|
|
|
void epsg3857totile(double ix, double iy, int zoom, long long *x, long long *y) {
|
2018-05-15 19:55:17 +00:00
|
|
|
// Place infinite and NaN coordinates off the edge of the Mercator plane
|
|
|
|
|
|
|
|
int iy_class = fpclassify(iy);
|
|
|
|
int ix_class = fpclassify(ix);
|
|
|
|
|
|
|
|
if (iy_class == FP_INFINITE || iy_class == FP_NAN) {
|
|
|
|
iy = 40000000.0;
|
|
|
|
}
|
|
|
|
if (ix_class == FP_INFINITE || ix_class == FP_NAN) {
|
|
|
|
ix = 40000000.0;
|
|
|
|
}
|
|
|
|
|
2016-06-01 22:49:41 +00:00
|
|
|
*x = ix * (1LL << 31) / 6378137.0 / M_PI + (1LL << 31);
|
|
|
|
*y = ((1LL << 32) - 1) - (iy * (1LL << 31) / 6378137.0 / M_PI + (1LL << 31));
|
|
|
|
|
|
|
|
if (zoom != 0) {
|
|
|
|
*x >>= (32 - zoom);
|
|
|
|
*y >>= (32 - zoom);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-28 22:18:27 +00:00
|
|
|
void tiletoepsg3857(long long ix, long long iy, int zoom, double *ox, double *oy) {
|
|
|
|
if (zoom != 0) {
|
|
|
|
ix <<= (32 - zoom);
|
|
|
|
iy <<= (32 - zoom);
|
|
|
|
}
|
|
|
|
|
|
|
|
*ox = (ix - (1LL << 31)) * M_PI * 6378137.0 / (1LL << 31);
|
|
|
|
*oy = ((1LL << 32) - 1 - iy - (1LL << 31)) * M_PI * 6378137.0 / (1LL << 31);
|
|
|
|
}
|
|
|
|
|
2014-10-25 00:22:14 +00:00
|
|
|
unsigned long long encode(unsigned int wx, unsigned int wy) {
|
2016-04-27 18:13:15 +00:00
|
|
|
unsigned long long out = 0;
|
2014-10-25 00:22:14 +00:00
|
|
|
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < 32; i++) {
|
2016-04-27 18:13:15 +00:00
|
|
|
unsigned long long v = ((wx >> (32 - (i + 1))) & 1) << 1;
|
2014-10-25 00:22:14 +00:00
|
|
|
v |= (wy >> (32 - (i + 1))) & 1;
|
|
|
|
v = v << (64 - 2 * (i + 1));
|
|
|
|
|
|
|
|
out |= v;
|
|
|
|
}
|
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2018-05-18 21:45:03 +00:00
|
|
|
static std::atomic<unsigned char> decodex[256];
|
|
|
|
static std::atomic<unsigned char> decodey[256];
|
2017-03-16 22:06:26 +00:00
|
|
|
|
2014-10-25 00:22:14 +00:00
|
|
|
void decode(unsigned long long index, unsigned *wx, unsigned *wy) {
|
2018-05-18 21:45:03 +00:00
|
|
|
static std::atomic<int> initialized(0);
|
2017-03-16 22:06:26 +00:00
|
|
|
if (!initialized) {
|
|
|
|
for (size_t ix = 0; ix < 256; ix++) {
|
|
|
|
size_t xx = 0, yy = 0;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < 32; i++) {
|
|
|
|
xx |= ((ix >> (64 - 2 * (i + 1) + 1)) & 1) << (32 - (i + 1));
|
|
|
|
yy |= ((ix >> (64 - 2 * (i + 1) + 0)) & 1) << (32 - (i + 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
decodex[ix] = xx;
|
|
|
|
decodey[ix] = yy;
|
|
|
|
}
|
|
|
|
|
|
|
|
initialized = 1;
|
|
|
|
}
|
|
|
|
|
2014-10-25 00:22:14 +00:00
|
|
|
*wx = *wy = 0;
|
|
|
|
|
2017-03-16 22:06:26 +00:00
|
|
|
for (size_t i = 0; i < 8; i++) {
|
|
|
|
*wx |= ((unsigned) decodex[(index >> (8 * i)) & 0xFF]) << (4 * i);
|
|
|
|
*wy |= ((unsigned) decodey[(index >> (8 * i)) & 0xFF]) << (4 * i);
|
2014-10-25 00:22:14 +00:00
|
|
|
}
|
|
|
|
}
|
2016-06-28 22:18:27 +00:00
|
|
|
|
|
|
|
void set_projection_or_exit(const char *optarg) {
|
|
|
|
struct projection *p;
|
|
|
|
for (p = projections; p->name != NULL; p++) {
|
|
|
|
if (strcmp(p->name, optarg) == 0) {
|
|
|
|
projection = p;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (strcmp(p->alias, optarg) == 0) {
|
|
|
|
projection = p;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (p->name == NULL) {
|
|
|
|
fprintf(stderr, "Unknown projection (-s): %s\n", optarg);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
}
|