From d359461e61bd0952b6ba794fe25d131831115034 Mon Sep 17 00:00:00 2001 From: Erica Fischer Date: Wed, 8 Nov 2023 10:51:28 -1000 Subject: [PATCH] Reduce tile-join overzooming memory usage (#162) * 16 bits is enough for tile numbers * Revert "16 bits is enough for tile numbers" This reverts commit 71a0c4e1cf81a0f5d027e65fb8460a52681eb3dd. * Check what child tiles each overzoomed tile will have and don't queue further overzooming of empty tiles * Clean up naming; use std::move to avoid copying large arrays --- CHANGELOG.md | 4 ++++ clip.cpp | 30 +++++++++++++++++++++++++++--- geometry.hpp | 6 ++++-- overzoom.cpp | 2 +- tile-join.cpp | 25 +++++++++++++++++++------ version.hpp | 2 +- 6 files changed, 56 insertions(+), 13 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index d7c08508e..8ccf0a700 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,7 @@ +# 2.37.0 + +* Speed up tile-join overzooming and make it use less memory, by not including empty child tiles in the enumeration + # 2.36.0 * Make tile-join distrust the source tilesets' metadata maxzoom and minzoom diff --git a/clip.cpp b/clip.cpp index 5e4e644f7..512ab2fd5 100644 --- a/clip.cpp +++ b/clip.cpp @@ -753,7 +753,8 @@ static std::vector> clip_poly1(std::vector const &keep, bool do_compress) { + int detail, int buffer, std::set const &keep, bool do_compress, + std::vector> *next_overzoomed_tiles) { mvt_tile tile; try { @@ -767,11 +768,12 @@ std::string overzoom(std::string s, int oz, int ox, int oy, int nz, int nx, int exit(EXIT_PROTOBUF); } - return overzoom(tile, oz, ox, oy, nz, nx, ny, detail, buffer, keep, do_compress); + return overzoom(tile, oz, ox, oy, nz, nx, ny, detail, buffer, keep, do_compress, next_overzoomed_tiles); } std::string overzoom(mvt_tile tile, int oz, int ox, int oy, int nz, int nx, int ny, - int detail, int buffer, std::set const &keep, bool do_compress) { + int detail, int buffer, std::set const &keep, bool do_compress, + std::vector> *next_overzoomed_tiles) { mvt_tile outtile; for (auto const &layer : tile.layers) { @@ -890,6 +892,28 @@ std::string overzoom(mvt_tile tile, int oz, int ox, int oy, int nz, int nx, int } } + if (next_overzoomed_tiles != NULL) { + // will any child tiles have features in them? + // find out recursively from the tile we just made. + // + // (yes, we should keep them instead of remaking them + // later, but that first requires figuring out where to + // keep them.) + + if (outtile.layers.size() > 0) { + for (size_t x = 0; x < 2; x++) { + for (size_t y = 0; y < 2; y++) { + std::string child = overzoom(outtile, nz, nx, ny, + nz + 1, nx * 2 + x, ny * 2 + y, + detail, buffer, keep, false, NULL); + if (child.size() > 0) { + next_overzoomed_tiles->emplace_back(nx * 2 + x, ny * 2 + y); + } + } + } + } + } + if (outtile.layers.size() > 0) { std::string pbf = outtile.encode(); diff --git a/geometry.hpp b/geometry.hpp index 5da7cdf33..1cbaa8ef1 100644 --- a/geometry.hpp +++ b/geometry.hpp @@ -99,9 +99,11 @@ int pnpoly(const drawvec &vert, size_t start, size_t nvert, long long testx, lon double distance_from_line(long long point_x, long long point_y, long long segA_x, long long segA_y, long long segB_x, long long segB_y); std::string overzoom(mvt_tile tile, int oz, int ox, int oy, int nz, int nx, int ny, - int detail, int buffer, std::set const &keep, bool do_compress); + int detail, int buffer, std::set const &keep, bool do_compress, + std::vector> *next_overzoomed_tiles); std::string overzoom(std::string s, int oz, int ox, int oy, int nz, int nx, int ny, - int detail, int buffer, std::set const &keep, bool do_compress); + int detail, int buffer, std::set const &keep, bool do_compress, + std::vector> *next_overzoomed_tiles); #endif diff --git a/overzoom.cpp b/overzoom.cpp index c2129a023..1a630413d 100644 --- a/overzoom.cpp +++ b/overzoom.cpp @@ -91,7 +91,7 @@ int main(int argc, char **argv) { exit(EXIT_FAILURE); } - std::string out = overzoom(tile, oz, ox, oy, nz, nx, ny, detail, buffer, keep, true); + std::string out = overzoom(tile, oz, ox, oy, nz, nx, ny, detail, buffer, keep, true, NULL); fwrite(out.c_str(), sizeof(char), out.size(), f); fclose(f); diff --git a/tile-join.cpp b/tile-join.cpp index 0a5102269..6e2f93199 100644 --- a/tile-join.cpp +++ b/tile-join.cpp @@ -434,7 +434,8 @@ struct tileset_reader { // for overzooming int maxzoom_so_far = -1; std::vector> tiles_at_maxzoom_so_far; - std::vector> overzoomed_tiles; + std::vector> overzoomed_tiles; // tiles at `zoom` + std::vector> next_overzoomed_tiles; // tiles at `zoom + 1` bool overzoom_consumed_at_this_zoom = false; // parent tile cache @@ -642,12 +643,24 @@ struct tileset_reader { long long scale = (1LL << zoom) / (1LL << maxzoom_so_far); - for (auto const &xy : tiles_at_maxzoom_so_far) { - for (long long xx = 0; xx < scale; xx++) { - for (long long yy = 0; yy < scale; yy++) { - overzoomed_tiles.push_back(std::pair(xy.first * scale + xx, xy.second * scale + yy)); + // If this is the first overzoomed level, we don't know yet + // which tiles will be useful, so spell out all 4 child tiles + // from each parent tile. + // + // If it is further overzoomed than that, we have a list of + // which child tiles will have features in them, so use that. + + if (zoom == maxzoom_so_far + 1) { + for (auto const &xy : tiles_at_maxzoom_so_far) { + for (long long xx = 0; xx < scale; xx++) { + for (long long yy = 0; yy < scale; yy++) { + overzoomed_tiles.push_back(std::pair(xy.first * scale + xx, xy.second * scale + yy)); + } } } + } else { + overzoomed_tiles = std::move(next_overzoomed_tiles); + next_overzoomed_tiles.clear(); } std::sort(overzoomed_tiles.begin(), overzoomed_tiles.end(), tilecmp); @@ -769,7 +782,7 @@ struct tileset_reader { } if (source.layers.size() != 0) { - std::string ret = overzoom(source, parent_tile.z, parent_tile.x, parent_tile.y, tile.z, tile.x, tile.y, -1, buffer, std::set(), false); + std::string ret = overzoom(source, parent_tile.z, parent_tile.x, parent_tile.y, tile.z, tile.x, tile.y, -1, buffer, std::set(), false, &next_overzoomed_tiles); return ret; } diff --git a/version.hpp b/version.hpp index 922d5ce74..5cd56cb78 100644 --- a/version.hpp +++ b/version.hpp @@ -1,6 +1,6 @@ #ifndef VERSION_HPP #define VERSION_HPP -#define VERSION "v2.36.0" +#define VERSION "v2.37.0" #endif