2016-12-08 20:33:02 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
#define _DARWIN_UNLIMITED_STREAMS
|
|
|
|
#endif
|
|
|
|
|
2016-12-01 00:13:15 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2016-12-07 01:25:23 +00:00
|
|
|
#include <string.h>
|
2016-12-01 00:13:15 +00:00
|
|
|
#include <vector>
|
|
|
|
#include <string>
|
|
|
|
#include <map>
|
2016-12-12 23:12:41 +00:00
|
|
|
#include <set>
|
2016-12-01 20:04:49 +00:00
|
|
|
#include <pthread.h>
|
|
|
|
#include <unistd.h>
|
2016-12-07 19:25:42 +00:00
|
|
|
#include <fcntl.h>
|
2016-12-07 00:47:56 +00:00
|
|
|
#include <cmath>
|
2016-12-07 01:25:23 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/wait.h>
|
2016-12-12 23:12:41 +00:00
|
|
|
#include <sqlite3.h>
|
2016-12-13 00:12:22 +00:00
|
|
|
#include "main.hpp"
|
2016-12-01 00:13:15 +00:00
|
|
|
#include "mvt.hpp"
|
2016-12-12 23:12:41 +00:00
|
|
|
#include "mbtiles.hpp"
|
2016-12-01 00:13:15 +00:00
|
|
|
#include "projection.hpp"
|
|
|
|
#include "geometry.hpp"
|
2016-12-09 23:35:57 +00:00
|
|
|
#include "serial.hpp"
|
2016-12-01 00:13:15 +00:00
|
|
|
|
2016-12-05 22:12:39 +00:00
|
|
|
extern "C" {
|
|
|
|
#include "jsonpull/jsonpull.h"
|
|
|
|
}
|
|
|
|
|
2016-12-09 23:35:57 +00:00
|
|
|
#include "plugin.hpp"
|
2016-12-05 23:18:27 +00:00
|
|
|
#include "write_json.hpp"
|
|
|
|
#include "read_json.hpp"
|
|
|
|
|
2016-12-01 20:04:49 +00:00
|
|
|
struct writer_arg {
|
2016-12-08 23:43:52 +00:00
|
|
|
int write_to;
|
2016-12-01 20:04:49 +00:00
|
|
|
mvt_layer *layer;
|
|
|
|
unsigned z;
|
|
|
|
unsigned x;
|
|
|
|
unsigned y;
|
|
|
|
};
|
|
|
|
|
|
|
|
void *run_writer(void *a) {
|
|
|
|
writer_arg *wa = (writer_arg *) a;
|
|
|
|
|
|
|
|
// XXX worry about SIGPIPE?
|
|
|
|
|
2016-12-08 23:43:52 +00:00
|
|
|
FILE *fp = fdopen(wa->write_to, "w");
|
2016-12-01 20:04:49 +00:00
|
|
|
if (fp == NULL) {
|
|
|
|
perror("fdopen (pipe writer)");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2016-12-08 23:13:38 +00:00
|
|
|
layer_to_geojson(fp, *(wa->layer), wa->z, wa->x, wa->y, false, false);
|
2016-12-01 20:04:49 +00:00
|
|
|
|
|
|
|
if (fclose(fp) != 0) {
|
|
|
|
perror("fclose output to filter");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2016-12-07 00:33:39 +00:00
|
|
|
// XXX deduplicate
|
|
|
|
static std::vector<mvt_geometry> to_feature(drawvec &geom) {
|
|
|
|
std::vector<mvt_geometry> out;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < geom.size(); i++) {
|
|
|
|
out.push_back(mvt_geometry(geom[i].op, geom[i].x, geom[i].y));
|
|
|
|
}
|
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2016-12-13 01:00:45 +00:00
|
|
|
mvt_layer parse_layer(int fd, int z, unsigned x, unsigned y, mvt_layer const &olayer, std::vector<std::map<std::string, layermap_entry>> *layermaps, size_t tiling_seg, std::vector<std::vector<std::string>> *layer_unmaps) {
|
2016-12-05 22:12:39 +00:00
|
|
|
mvt_layer ret;
|
2016-12-07 00:33:39 +00:00
|
|
|
ret.name = olayer.name;
|
|
|
|
ret.version = olayer.version;
|
|
|
|
ret.extent = olayer.extent;
|
2016-12-05 22:12:39 +00:00
|
|
|
|
2016-12-12 23:12:41 +00:00
|
|
|
std::string layername = olayer.name;
|
|
|
|
|
2016-12-05 22:12:39 +00:00
|
|
|
FILE *f = fdopen(fd, "r");
|
|
|
|
if (f == NULL) {
|
|
|
|
perror("fdopen filter output");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
json_pull *jp = json_begin_file(f);
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
json_object *j = json_read(jp);
|
|
|
|
if (j == NULL) {
|
|
|
|
if (jp->error != NULL) {
|
|
|
|
fprintf(stderr, "Filter output:%d: %s\n", jp->line, jp->error);
|
|
|
|
if (jp->root != NULL) {
|
|
|
|
json_context(jp->root);
|
|
|
|
}
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
json_free(jp->root);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
json_object *type = json_hash_get(j, "type");
|
|
|
|
if (type == NULL || type->type != JSON_STRING) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (strcmp(type->string, "Feature") != 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
json_object *geometry = json_hash_get(j, "geometry");
|
|
|
|
if (geometry == NULL) {
|
|
|
|
fprintf(stderr, "Filter output:%d: filtered feature with no geometry\n", jp->line);
|
|
|
|
json_context(j);
|
|
|
|
json_free(j);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
json_object *properties = json_hash_get(j, "properties");
|
|
|
|
if (properties == NULL || (properties->type != JSON_HASH && properties->type != JSON_NULL)) {
|
|
|
|
fprintf(stderr, "Filter output:%d: feature without properties hash\n", jp->line);
|
|
|
|
json_context(j);
|
|
|
|
json_free(j);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
json_object *geometry_type = json_hash_get(geometry, "type");
|
|
|
|
if (geometry_type == NULL) {
|
|
|
|
fprintf(stderr, "Filter output:%d: null geometry (additional not reported)\n", jp->line);
|
|
|
|
json_context(j);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (geometry_type->type != JSON_STRING) {
|
|
|
|
fprintf(stderr, "Filter output:%d: geometry type is not a string\n", jp->line);
|
|
|
|
json_context(j);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
json_object *coordinates = json_hash_get(geometry, "coordinates");
|
|
|
|
if (coordinates == NULL || coordinates->type != JSON_ARRAY) {
|
|
|
|
fprintf(stderr, "Filter output:%d: feature without coordinates array\n", jp->line);
|
|
|
|
json_context(j);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
int t;
|
|
|
|
for (t = 0; t < GEOM_TYPES; t++) {
|
|
|
|
if (strcmp(geometry_type->string, geometry_names[t]) == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (t >= GEOM_TYPES) {
|
|
|
|
fprintf(stderr, "Filter output:%d: Can't handle geometry type %s\n", jp->line, geometry_type->string);
|
|
|
|
json_context(j);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2016-12-07 00:33:39 +00:00
|
|
|
drawvec dv;
|
|
|
|
parse_geometry(t, coordinates, dv, VT_MOVETO, "Filter output", jp->line, j);
|
|
|
|
if (mb_geometry[t] == VT_POLYGON) {
|
|
|
|
dv = fix_polygon(dv);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scale and offset geometry from global to tile
|
|
|
|
for (size_t i = 0; i < dv.size(); i++) {
|
|
|
|
long long scale = 1LL << (32 - z);
|
2016-12-07 00:47:56 +00:00
|
|
|
dv[i].x = std::round((dv[i].x - scale * x) * olayer.extent / (double) scale);
|
|
|
|
dv[i].y = std::round((dv[i].y - scale * y) * olayer.extent / (double) scale);
|
2016-12-07 00:33:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mb_geometry[t] == VT_POLYGON) {
|
|
|
|
dv = clean_or_clip_poly(dv, 0, 0, 0, false);
|
|
|
|
if (dv.size() < 3) {
|
|
|
|
dv.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dv = remove_noop(dv, mb_geometry[t], 0);
|
|
|
|
if (mb_geometry[t] == VT_POLYGON) {
|
|
|
|
dv = close_poly(dv);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dv.size() > 0) {
|
|
|
|
mvt_feature feature;
|
|
|
|
feature.type = mb_geometry[t];
|
|
|
|
feature.geometry = to_feature(dv);
|
|
|
|
|
|
|
|
json_object *id = json_hash_get(j, "id");
|
|
|
|
if (id != NULL) {
|
|
|
|
feature.id = atoll(id->string);
|
|
|
|
feature.has_id = true;
|
|
|
|
}
|
|
|
|
|
2016-12-12 23:12:41 +00:00
|
|
|
std::map<std::string, layermap_entry> &layermap = (*layermaps)[tiling_seg];
|
|
|
|
|
2016-12-07 00:33:39 +00:00
|
|
|
for (size_t i = 0; i < properties->length; i++) {
|
2016-12-07 23:54:58 +00:00
|
|
|
int tp = -1;
|
|
|
|
std::string s;
|
2016-12-07 00:33:39 +00:00
|
|
|
|
2016-12-07 23:54:58 +00:00
|
|
|
stringify_value(properties->values[i], tp, s, "Filter output", jp->line, j);
|
|
|
|
if (tp >= 0) {
|
|
|
|
mvt_value v = stringified_to_mvt_value(tp, s.c_str());
|
|
|
|
ret.tag(feature, std::string(properties->keys[i]->string), v);
|
2016-12-12 23:12:41 +00:00
|
|
|
|
|
|
|
if (layermap.count(layername) == 0) {
|
|
|
|
layermap_entry lme = layermap_entry(layermap.size());
|
|
|
|
lme.minzoom = z;
|
|
|
|
lme.maxzoom = z;
|
|
|
|
|
|
|
|
layermap.insert(std::pair<std::string, layermap_entry>(layername, lme));
|
2016-12-13 01:00:45 +00:00
|
|
|
|
|
|
|
if (lme.id >= (*layer_unmaps)[tiling_seg].size()) {
|
|
|
|
(*layer_unmaps)[tiling_seg].resize(lme.id + 1);
|
|
|
|
(*layer_unmaps)[tiling_seg][lme.id] = layername;
|
|
|
|
}
|
2016-12-12 23:12:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type_and_string tas;
|
|
|
|
tas.string = std::string(properties->keys[i]->string);
|
|
|
|
tas.type = tp;
|
|
|
|
|
|
|
|
auto fk = layermap.find(layername);
|
|
|
|
if (z < fk->second.minzoom) {
|
|
|
|
fk->second.minzoom = z;
|
|
|
|
}
|
|
|
|
if (z > fk->second.maxzoom) {
|
|
|
|
fk->second.maxzoom = z;
|
|
|
|
}
|
|
|
|
fk->second.file_keys.insert(tas);
|
2016-12-07 23:54:58 +00:00
|
|
|
}
|
2016-12-07 00:33:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ret.features.push_back(feature);
|
|
|
|
}
|
|
|
|
|
2016-12-05 22:12:39 +00:00
|
|
|
json_free(j);
|
|
|
|
}
|
|
|
|
|
|
|
|
json_end(jp);
|
2016-12-12 23:12:41 +00:00
|
|
|
if (fclose(f) != 0) {
|
|
|
|
perror("fclose postfilter output");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2016-12-05 22:12:39 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-12-13 01:00:45 +00:00
|
|
|
serial_feature parse_feature(json_pull *jp, int z, unsigned x, unsigned y, std::vector<std::map<std::string, layermap_entry>> *layermaps, size_t tiling_seg, std::vector<std::vector<std::string>> *layer_unmaps) {
|
2016-12-09 23:35:57 +00:00
|
|
|
serial_feature sf;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
json_object *j = json_read(jp);
|
|
|
|
if (j == NULL) {
|
|
|
|
if (jp->error != NULL) {
|
|
|
|
fprintf(stderr, "Filter output:%d: %s\n", jp->line, jp->error);
|
|
|
|
if (jp->root != NULL) {
|
|
|
|
json_context(jp->root);
|
|
|
|
}
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
json_free(jp->root);
|
|
|
|
sf.t = -1;
|
|
|
|
return sf;
|
|
|
|
}
|
|
|
|
|
|
|
|
json_object *type = json_hash_get(j, "type");
|
|
|
|
if (type == NULL || type->type != JSON_STRING) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (strcmp(type->string, "Feature") != 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
json_object *geometry = json_hash_get(j, "geometry");
|
|
|
|
if (geometry == NULL) {
|
|
|
|
fprintf(stderr, "Filter output:%d: filtered feature with no geometry\n", jp->line);
|
|
|
|
json_context(j);
|
|
|
|
json_free(j);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
json_object *properties = json_hash_get(j, "properties");
|
|
|
|
if (properties == NULL || (properties->type != JSON_HASH && properties->type != JSON_NULL)) {
|
|
|
|
fprintf(stderr, "Filter output:%d: feature without properties hash\n", jp->line);
|
|
|
|
json_context(j);
|
|
|
|
json_free(j);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
json_object *geometry_type = json_hash_get(geometry, "type");
|
|
|
|
if (geometry_type == NULL) {
|
|
|
|
fprintf(stderr, "Filter output:%d: null geometry (additional not reported)\n", jp->line);
|
|
|
|
json_context(j);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (geometry_type->type != JSON_STRING) {
|
|
|
|
fprintf(stderr, "Filter output:%d: geometry type is not a string\n", jp->line);
|
|
|
|
json_context(j);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
json_object *coordinates = json_hash_get(geometry, "coordinates");
|
|
|
|
if (coordinates == NULL || coordinates->type != JSON_ARRAY) {
|
|
|
|
fprintf(stderr, "Filter output:%d: feature without coordinates array\n", jp->line);
|
|
|
|
json_context(j);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
int t;
|
|
|
|
for (t = 0; t < GEOM_TYPES; t++) {
|
|
|
|
if (strcmp(geometry_type->string, geometry_names[t]) == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (t >= GEOM_TYPES) {
|
|
|
|
fprintf(stderr, "Filter output:%d: Can't handle geometry type %s\n", jp->line, geometry_type->string);
|
|
|
|
json_context(j);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
drawvec dv;
|
|
|
|
parse_geometry(t, coordinates, dv, VT_MOVETO, "Filter output", jp->line, j);
|
|
|
|
if (mb_geometry[t] == VT_POLYGON) {
|
|
|
|
dv = fix_polygon(dv);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scale and offset geometry from global to tile
|
2016-12-13 00:12:22 +00:00
|
|
|
double scale = 1LL << geometry_scale;
|
2016-12-09 23:35:57 +00:00
|
|
|
for (size_t i = 0; i < dv.size(); i++) {
|
|
|
|
unsigned sx = 0, sy = 0;
|
|
|
|
if (z != 0) {
|
|
|
|
sx = x << (32 - z);
|
|
|
|
sy = y << (32 - z);
|
|
|
|
}
|
2016-12-13 00:12:22 +00:00
|
|
|
dv[i].x = std::round(dv[i].x / scale) * scale - sx;
|
|
|
|
dv[i].y = std::round(dv[i].y / scale) * scale - sy;
|
2016-12-09 23:35:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (dv.size() > 0) {
|
2016-12-13 00:08:08 +00:00
|
|
|
std::string layername = "unknown";
|
|
|
|
json_object *tippecanoe = json_hash_get(j, "tippecanoe");
|
|
|
|
json_object *layer = NULL;
|
|
|
|
if (tippecanoe != NULL) {
|
|
|
|
layer = json_hash_get(tippecanoe, "layer");
|
|
|
|
if (layer != NULL && layer->type == JSON_STRING) {
|
|
|
|
layername = std::string(layer->string);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-09 23:35:57 +00:00
|
|
|
sf.t = mb_geometry[t];
|
2016-12-13 01:00:45 +00:00
|
|
|
sf.segment = tiling_seg;
|
2016-12-09 23:35:57 +00:00
|
|
|
sf.geometry = dv;
|
|
|
|
sf.seq = 0; // XXX
|
|
|
|
sf.index = 0; // XXX
|
|
|
|
sf.bbox[0] = sf.bbox[1] = sf.bbox[2] = sf.bbox[3] = 0; // XXX
|
|
|
|
sf.extent = 0; // XXX
|
|
|
|
sf.m = 0; // XXX
|
|
|
|
sf.metapos = 0; // XXX
|
|
|
|
sf.has_id = false;
|
|
|
|
|
|
|
|
json_object *id = json_hash_get(j, "id");
|
|
|
|
if (id != NULL) {
|
|
|
|
sf.id = atoll(id->string);
|
|
|
|
sf.has_id = true;
|
|
|
|
}
|
|
|
|
|
2016-12-13 00:08:08 +00:00
|
|
|
std::map<std::string, layermap_entry> &layermap = (*layermaps)[tiling_seg];
|
|
|
|
|
2016-12-09 23:35:57 +00:00
|
|
|
for (size_t i = 0; i < properties->length; i++) {
|
|
|
|
serial_val v;
|
|
|
|
v.type = -1;
|
|
|
|
|
|
|
|
stringify_value(properties->values[i], v.type, v.s, "Filter output", jp->line, j);
|
|
|
|
|
|
|
|
if (v.type >= 0) {
|
|
|
|
sf.kv.insert(std::pair<std::string, serial_val>(std::string(properties->keys[i]->string), v));
|
2016-12-13 00:08:08 +00:00
|
|
|
|
|
|
|
if (layermap.count(layername) == 0) {
|
|
|
|
layermap_entry lme = layermap_entry(layermap.size());
|
|
|
|
lme.minzoom = z;
|
|
|
|
lme.maxzoom = z;
|
|
|
|
|
|
|
|
layermap.insert(std::pair<std::string, layermap_entry>(layername, lme));
|
2016-12-13 01:00:45 +00:00
|
|
|
|
|
|
|
if (lme.id >= (*layer_unmaps)[tiling_seg].size()) {
|
|
|
|
(*layer_unmaps)[tiling_seg].resize(lme.id + 1);
|
|
|
|
(*layer_unmaps)[tiling_seg][lme.id] = layername;
|
|
|
|
}
|
2016-12-13 00:08:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type_and_string tas;
|
|
|
|
tas.string = std::string(properties->keys[i]->string);
|
|
|
|
tas.type = v.type;
|
|
|
|
|
|
|
|
auto fk = layermap.find(layername);
|
|
|
|
if (z < fk->second.minzoom) {
|
|
|
|
fk->second.minzoom = z;
|
|
|
|
}
|
|
|
|
if (z > fk->second.maxzoom) {
|
|
|
|
fk->second.maxzoom = z;
|
|
|
|
}
|
|
|
|
fk->second.file_keys.insert(tas);
|
2016-12-13 01:00:45 +00:00
|
|
|
|
|
|
|
sf.layer = fk->second.id;
|
2016-12-09 23:35:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
json_free(j);
|
|
|
|
return sf;
|
|
|
|
}
|
|
|
|
|
|
|
|
json_free(j);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-07 19:16:34 +00:00
|
|
|
static pthread_mutex_t pipe_lock = PTHREAD_MUTEX_INITIALIZER;
|
|
|
|
|
2016-12-08 23:43:52 +00:00
|
|
|
void setup_filter(const char *filter, int *write_to, int *read_from, pid_t *pid, unsigned z, unsigned x, unsigned y) {
|
2016-12-01 20:04:49 +00:00
|
|
|
// This will create two pipes, a new thread, and a new process.
|
|
|
|
//
|
|
|
|
// The new process will read from one pipe and write to the other, and execute the filter.
|
|
|
|
// The new thread will write the GeoJSON to the pipe that leads to the filter.
|
|
|
|
// The original thread will read the GeoJSON from the filter and convert it back into vector tiles.
|
|
|
|
|
2016-12-07 19:16:34 +00:00
|
|
|
if (pthread_mutex_lock(&pipe_lock) != 0) {
|
|
|
|
perror("pthread_mutex_lock (pipe)");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2016-12-01 20:04:49 +00:00
|
|
|
int pipe_orig[2], pipe_filtered[2];
|
|
|
|
if (pipe(pipe_orig) < 0) {
|
|
|
|
perror("pipe (original features)");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (pipe(pipe_filtered) < 0) {
|
|
|
|
perror("pipe (filtered features)");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2016-12-07 20:15:57 +00:00
|
|
|
std::string z_str = std::to_string(z);
|
|
|
|
std::string x_str = std::to_string(x);
|
|
|
|
std::string y_str = std::to_string(y);
|
|
|
|
|
2016-12-08 23:43:52 +00:00
|
|
|
*pid = fork();
|
|
|
|
if (*pid < 0) {
|
2016-12-01 20:04:49 +00:00
|
|
|
perror("fork");
|
|
|
|
exit(EXIT_FAILURE);
|
2016-12-08 23:43:52 +00:00
|
|
|
} else if (*pid == 0) {
|
2016-12-01 20:04:49 +00:00
|
|
|
// child
|
|
|
|
|
|
|
|
if (dup2(pipe_orig[0], 0) < 0) {
|
|
|
|
perror("dup child stdin");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (dup2(pipe_filtered[1], 1) < 0) {
|
|
|
|
perror("dup child stdout");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (close(pipe_orig[1]) != 0) {
|
|
|
|
perror("close output to filter");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (close(pipe_filtered[0]) != 0) {
|
|
|
|
perror("close input from filter");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2016-12-07 19:16:34 +00:00
|
|
|
if (close(pipe_orig[0]) != 0) {
|
|
|
|
perror("close dup input of filter");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (close(pipe_filtered[1]) != 0) {
|
|
|
|
perror("close dup output of filter");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2016-12-01 20:04:49 +00:00
|
|
|
|
|
|
|
// XXX close other fds?
|
|
|
|
|
2016-12-07 20:15:57 +00:00
|
|
|
if (execlp("sh", "sh", "-c", filter, "sh", z_str.c_str(), x_str.c_str(), y_str.c_str(), NULL) != 0) {
|
2016-12-01 20:04:49 +00:00
|
|
|
perror("exec");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// parent
|
|
|
|
|
|
|
|
if (close(pipe_orig[0]) != 0) {
|
|
|
|
perror("close filter-side reader");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (close(pipe_filtered[1]) != 0) {
|
|
|
|
perror("close filter-side writer");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2016-12-07 19:25:42 +00:00
|
|
|
if (fcntl(pipe_orig[1], F_SETFD, FD_CLOEXEC) != 0) {
|
|
|
|
perror("cloxec output to filter");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (fcntl(pipe_filtered[0], F_SETFD, FD_CLOEXEC) != 0) {
|
|
|
|
perror("cloxec input from filter");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2016-12-01 20:04:49 +00:00
|
|
|
|
2016-12-07 19:16:34 +00:00
|
|
|
if (pthread_mutex_unlock(&pipe_lock) != 0) {
|
|
|
|
perror("pthread_mutex_unlock (pipe_lock)");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2016-12-08 23:43:52 +00:00
|
|
|
*write_to = pipe_orig[1];
|
|
|
|
*read_from = pipe_filtered[0];
|
|
|
|
}
|
|
|
|
}
|
2016-12-01 20:04:49 +00:00
|
|
|
|
2016-12-13 01:00:45 +00:00
|
|
|
mvt_layer filter_layer(const char *filter, mvt_layer &layer, unsigned z, unsigned x, unsigned y, std::vector<std::map<std::string, layermap_entry>> *layermaps, size_t tiling_seg, std::vector<std::vector<std::string>> *layer_unmaps) {
|
2016-12-08 23:43:52 +00:00
|
|
|
int write_to, read_from;
|
|
|
|
pid_t pid;
|
|
|
|
setup_filter(filter, &write_to, &read_from, &pid, z, x, y);
|
|
|
|
|
|
|
|
writer_arg wa;
|
|
|
|
wa.write_to = write_to;
|
|
|
|
wa.layer = &layer;
|
|
|
|
wa.z = z;
|
|
|
|
wa.x = x;
|
|
|
|
wa.y = y;
|
|
|
|
|
|
|
|
pthread_t writer;
|
|
|
|
if (pthread_create(&writer, NULL, run_writer, &wa) != 0) {
|
|
|
|
perror("pthread_create (filter writer)");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2016-12-01 20:04:49 +00:00
|
|
|
|
2016-12-13 01:00:45 +00:00
|
|
|
layer = parse_layer(read_from, z, x, y, layer, layermaps, tiling_seg, layer_unmaps);
|
2016-12-01 20:04:49 +00:00
|
|
|
|
2016-12-08 23:43:52 +00:00
|
|
|
while (1) {
|
2016-12-01 20:04:49 +00:00
|
|
|
int stat_loc;
|
|
|
|
if (waitpid(pid, &stat_loc, 0) < 0) {
|
|
|
|
perror("waitpid for filter\n");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2016-12-08 23:43:52 +00:00
|
|
|
if (WIFEXITED(stat_loc) || WIFSIGNALED(stat_loc)) {
|
|
|
|
break;
|
2016-12-01 20:04:49 +00:00
|
|
|
}
|
2016-12-08 23:43:52 +00:00
|
|
|
}
|
2016-12-01 20:04:49 +00:00
|
|
|
|
2016-12-08 23:43:52 +00:00
|
|
|
void *ret;
|
|
|
|
if (pthread_join(writer, &ret) != 0) {
|
|
|
|
perror("pthread_join filter writer");
|
|
|
|
exit(EXIT_FAILURE);
|
2016-12-01 20:04:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return layer;
|
|
|
|
}
|