Intersection analysis occupy in osrm::extractor::intersection namespace and guidance code osrm::guidance
1023 lines
47 KiB
C++
1023 lines
47 KiB
C++
#include "guidance/turn_instruction.hpp"
|
|
|
|
#include "engine/plugins/plugin_base.hpp"
|
|
#include "engine/plugins/tile.hpp"
|
|
|
|
#include "util/coordinate_calculation.hpp"
|
|
#include "util/string_view.hpp"
|
|
#include "util/vector_tile.hpp"
|
|
#include "util/web_mercator.hpp"
|
|
|
|
#include "engine/api/json_factory.hpp"
|
|
|
|
#include <boost/geometry.hpp>
|
|
#include <boost/geometry/geometries/geometries.hpp>
|
|
#include <boost/geometry/geometries/point_xy.hpp>
|
|
#include <boost/geometry/multi/geometries/multi_linestring.hpp>
|
|
|
|
#include <protozero/pbf_writer.hpp>
|
|
#include <protozero/varint.hpp>
|
|
|
|
#include <algorithm>
|
|
#include <numeric>
|
|
#include <string>
|
|
#include <unordered_map>
|
|
#include <unordered_set>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include <cmath>
|
|
#include <cstdint>
|
|
|
|
namespace osrm
|
|
{
|
|
namespace engine
|
|
{
|
|
namespace plugins
|
|
{
|
|
|
|
constexpr const static int MIN_ZOOM_FOR_TURNS = 15;
|
|
|
|
namespace
|
|
{
|
|
|
|
// Creates an indexed lookup table for values - used to encoded the vector tile
|
|
// which uses a lookup table and index pointers for encoding
|
|
template <typename T> struct ValueIndexer
|
|
{
|
|
private:
|
|
std::vector<T> used_values;
|
|
std::unordered_map<T, std::size_t> value_offsets;
|
|
|
|
public:
|
|
std::size_t add(const T &value)
|
|
{
|
|
const auto found = value_offsets.find(value);
|
|
std::size_t offset;
|
|
|
|
if (found == value_offsets.end())
|
|
{
|
|
used_values.push_back(value);
|
|
offset = used_values.size() - 1;
|
|
value_offsets[value] = offset;
|
|
}
|
|
else
|
|
{
|
|
offset = found->second;
|
|
}
|
|
|
|
return offset;
|
|
}
|
|
|
|
std::size_t indexOf(const T &value) { return value_offsets[value]; }
|
|
|
|
const std::vector<T> &values() { return used_values; }
|
|
|
|
std::size_t size() const { return used_values.size(); }
|
|
};
|
|
|
|
using RTreeLeaf = datafacade::BaseDataFacade::RTreeLeaf;
|
|
// TODO: Port all this encoding logic to https://github.com/mapbox/vector-tile, which wasn't
|
|
// available when this code was originally written.
|
|
|
|
// Simple container class for WGS84 coordinates
|
|
template <typename T> struct Point final
|
|
{
|
|
Point(T _x, T _y) : x(_x), y(_y) {}
|
|
|
|
const T x;
|
|
const T y;
|
|
};
|
|
|
|
// Simple container to hold a bounding box
|
|
struct BBox final
|
|
{
|
|
BBox(const double _minx, const double _miny, const double _maxx, const double _maxy)
|
|
: minx(_minx), miny(_miny), maxx(_maxx), maxy(_maxy)
|
|
{
|
|
}
|
|
|
|
double width() const { return maxx - minx; }
|
|
double height() const { return maxy - miny; }
|
|
|
|
const double minx;
|
|
const double miny;
|
|
const double maxx;
|
|
const double maxy;
|
|
};
|
|
|
|
// Simple container for integer coordinates (i.e. pixel coords)
|
|
struct point_type_i final
|
|
{
|
|
point_type_i(std::int64_t _x, std::int64_t _y) : x(_x), y(_y) {}
|
|
|
|
const std::int64_t x;
|
|
const std::int64_t y;
|
|
};
|
|
|
|
using FixedPoint = Point<std::int32_t>;
|
|
using FloatPoint = Point<double>;
|
|
|
|
using FixedLine = std::vector<FixedPoint>;
|
|
using FloatLine = std::vector<FloatPoint>;
|
|
|
|
// We use boost::geometry to clip lines/points that are outside or cross the boundary
|
|
// of the tile we're rendering. We need these types defined to use boosts clipping
|
|
// logic
|
|
typedef boost::geometry::model::point<double, 2, boost::geometry::cs::cartesian> point_t;
|
|
typedef boost::geometry::model::linestring<point_t> linestring_t;
|
|
typedef boost::geometry::model::box<point_t> box_t;
|
|
typedef boost::geometry::model::multi_linestring<linestring_t> multi_linestring_t;
|
|
const static box_t clip_box(point_t(-util::vector_tile::BUFFER, -util::vector_tile::BUFFER),
|
|
point_t(util::vector_tile::EXTENT + util::vector_tile::BUFFER,
|
|
util::vector_tile::EXTENT + util::vector_tile::BUFFER));
|
|
|
|
// from mapnik-vector-tile
|
|
// Encodes a linestring using protobuf zigzag encoding
|
|
inline bool encodeLinestring(const FixedLine &line,
|
|
protozero::packed_field_uint32 &geometry,
|
|
std::int32_t &start_x,
|
|
std::int32_t &start_y)
|
|
{
|
|
const std::size_t line_size = line.size();
|
|
if (line_size < 2)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
const unsigned lineto_count = static_cast<const unsigned>(line_size) - 1;
|
|
|
|
auto pt = line.begin();
|
|
const constexpr int MOVETO_COMMAND = 9;
|
|
geometry.add_element(MOVETO_COMMAND); // move_to | (1 << 3)
|
|
geometry.add_element(protozero::encode_zigzag32(pt->x - start_x));
|
|
geometry.add_element(protozero::encode_zigzag32(pt->y - start_y));
|
|
start_x = pt->x;
|
|
start_y = pt->y;
|
|
// This means LINETO repeated N times
|
|
// See: https://github.com/mapbox/vector-tile-spec/tree/master/2.1#example-command-integers
|
|
geometry.add_element((lineto_count << 3u) | 2u);
|
|
// Now that we've issued the LINETO REPEAT N command, we append
|
|
// N coordinate pairs immediately after the command.
|
|
for (++pt; pt != line.end(); ++pt)
|
|
{
|
|
const std::int32_t dx = pt->x - start_x;
|
|
const std::int32_t dy = pt->y - start_y;
|
|
geometry.add_element(protozero::encode_zigzag32(dx));
|
|
geometry.add_element(protozero::encode_zigzag32(dy));
|
|
start_x = pt->x;
|
|
start_y = pt->y;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// from mapnik-vctor-tile
|
|
// Encodes a point
|
|
inline void encodePoint(const FixedPoint &pt, protozero::packed_field_uint32 &geometry)
|
|
{
|
|
const constexpr int MOVETO_COMMAND = 9;
|
|
geometry.add_element(MOVETO_COMMAND);
|
|
const std::int32_t dx = pt.x;
|
|
const std::int32_t dy = pt.y;
|
|
// Manual zigzag encoding.
|
|
geometry.add_element(protozero::encode_zigzag32(dx));
|
|
geometry.add_element(protozero::encode_zigzag32(dy));
|
|
}
|
|
|
|
linestring_t floatLineToTileLine(const FloatLine &geo_line, const BBox &tile_bbox)
|
|
{
|
|
linestring_t unclipped_line;
|
|
|
|
for (auto const &pt : geo_line)
|
|
{
|
|
double px_merc = pt.x * util::web_mercator::DEGREE_TO_PX;
|
|
double py_merc = util::web_mercator::latToY(util::FloatLatitude{pt.y}) *
|
|
util::web_mercator::DEGREE_TO_PX;
|
|
// convert lon/lat to tile coordinates
|
|
const auto px = std::round(
|
|
((px_merc - tile_bbox.minx) * util::web_mercator::TILE_SIZE / tile_bbox.width()) *
|
|
util::vector_tile::EXTENT / util::web_mercator::TILE_SIZE);
|
|
const auto py = std::round(
|
|
((tile_bbox.maxy - py_merc) * util::web_mercator::TILE_SIZE / tile_bbox.height()) *
|
|
util::vector_tile::EXTENT / util::web_mercator::TILE_SIZE);
|
|
|
|
boost::geometry::append(unclipped_line, point_t(px, py));
|
|
}
|
|
|
|
return unclipped_line;
|
|
}
|
|
|
|
std::vector<FixedLine> coordinatesToTileLine(const std::vector<util::Coordinate> &points,
|
|
const BBox &tile_bbox)
|
|
{
|
|
FloatLine geo_line;
|
|
for (auto const &c : points)
|
|
{
|
|
geo_line.emplace_back(static_cast<double>(util::toFloating(c.lon)),
|
|
static_cast<double>(util::toFloating(c.lat)));
|
|
}
|
|
|
|
linestring_t unclipped_line = floatLineToTileLine(geo_line, tile_bbox);
|
|
|
|
multi_linestring_t clipped_line;
|
|
boost::geometry::intersection(clip_box, unclipped_line, clipped_line);
|
|
|
|
std::vector<FixedLine> result;
|
|
|
|
// b::g::intersection might return a line with one point if the
|
|
// original line was very short and coords were dupes
|
|
for (auto const &cl : clipped_line)
|
|
{
|
|
if (cl.size() < 2)
|
|
continue;
|
|
|
|
FixedLine tile_line;
|
|
for (const auto &p : cl)
|
|
tile_line.emplace_back(p.get<0>(), p.get<1>());
|
|
|
|
result.emplace_back(std::move(tile_line));
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Return the x1,y1,x2,y2 pixel coordinates of a line in a given
|
|
* tile.
|
|
*
|
|
* @param start the first coordinate of the line
|
|
* @param target the last coordinate of the line
|
|
* @param tile_bbox the boundaries of the tile, in mercator coordinates
|
|
* @return a FixedLine with coordinates relative to the tile_bbox.
|
|
*/
|
|
FixedLine coordinatesToTileLine(const util::Coordinate start,
|
|
const util::Coordinate target,
|
|
const BBox &tile_bbox)
|
|
{
|
|
FloatLine geo_line;
|
|
geo_line.emplace_back(static_cast<double>(util::toFloating(start.lon)),
|
|
static_cast<double>(util::toFloating(start.lat)));
|
|
geo_line.emplace_back(static_cast<double>(util::toFloating(target.lon)),
|
|
static_cast<double>(util::toFloating(target.lat)));
|
|
|
|
linestring_t unclipped_line = floatLineToTileLine(geo_line, tile_bbox);
|
|
|
|
multi_linestring_t clipped_line;
|
|
boost::geometry::intersection(clip_box, unclipped_line, clipped_line);
|
|
|
|
FixedLine tile_line;
|
|
|
|
// b::g::intersection might return a line with one point if the
|
|
// original line was very short and coords were dupes
|
|
if (!clipped_line.empty() && clipped_line[0].size() == 2)
|
|
{
|
|
for (const auto &p : clipped_line[0])
|
|
{
|
|
tile_line.emplace_back(p.get<0>(), p.get<1>());
|
|
}
|
|
}
|
|
|
|
return tile_line;
|
|
}
|
|
|
|
/**
|
|
* Converts lon/lat into coordinates inside a Mercator projection tile (x/y pixel values)
|
|
*
|
|
* @param point the lon/lat you want the tile coords for
|
|
* @param tile_bbox the mercator boundaries of the tile
|
|
* @return a point (x,y) on the tile defined by tile_bbox
|
|
*/
|
|
FixedPoint coordinatesToTilePoint(const util::Coordinate point, const BBox &tile_bbox)
|
|
{
|
|
const FloatPoint geo_point{static_cast<double>(util::toFloating(point.lon)),
|
|
static_cast<double>(util::toFloating(point.lat))};
|
|
|
|
const double px_merc = geo_point.x * util::web_mercator::DEGREE_TO_PX;
|
|
const double py_merc = util::web_mercator::latToY(util::FloatLatitude{geo_point.y}) *
|
|
util::web_mercator::DEGREE_TO_PX;
|
|
|
|
const auto px = static_cast<std::int32_t>(std::round(
|
|
((px_merc - tile_bbox.minx) * util::web_mercator::TILE_SIZE / tile_bbox.width()) *
|
|
util::vector_tile::EXTENT / util::web_mercator::TILE_SIZE));
|
|
const auto py = static_cast<std::int32_t>(std::round(
|
|
((tile_bbox.maxy - py_merc) * util::web_mercator::TILE_SIZE / tile_bbox.height()) *
|
|
util::vector_tile::EXTENT / util::web_mercator::TILE_SIZE));
|
|
|
|
return FixedPoint{px, py};
|
|
}
|
|
|
|
std::vector<RTreeLeaf> getEdges(const DataFacadeBase &facade, unsigned x, unsigned y, unsigned z)
|
|
{
|
|
double min_lon, min_lat, max_lon, max_lat;
|
|
|
|
// Convert the z,x,y mercator tile coordinates into WGS84 lon/lat values
|
|
//
|
|
util::web_mercator::xyzToWGS84(
|
|
x, y, z, min_lon, min_lat, max_lon, max_lat, util::web_mercator::TILE_SIZE * 0.10);
|
|
|
|
util::Coordinate southwest{util::FloatLongitude{min_lon}, util::FloatLatitude{min_lat}};
|
|
util::Coordinate northeast{util::FloatLongitude{max_lon}, util::FloatLatitude{max_lat}};
|
|
|
|
// Fetch all the segments that are in our bounding box.
|
|
// This hits the OSRM StaticRTree
|
|
return facade.GetEdgesInBox(southwest, northeast);
|
|
}
|
|
|
|
std::vector<std::size_t> getEdgeIndex(const std::vector<RTreeLeaf> &edges)
|
|
{
|
|
// In order to ensure consistent tile encoding, we need to process
|
|
// all edges in the same order. Differences in OSX/Linux/Windows
|
|
// sorting methods mean that GetEdgesInBox doesn't return the same
|
|
// ordered array on all platforms.
|
|
// GetEdgesInBox is marked `const`, so we can't sort the array itself,
|
|
// instead we create an array of indexes and sort that instead.
|
|
std::vector<std::size_t> sorted_edge_indexes(edges.size(), 0);
|
|
std::iota(
|
|
sorted_edge_indexes.begin(), sorted_edge_indexes.end(), 0); // fill with 0,1,2,3,...N-1
|
|
|
|
// Now, sort that array based on the edges list, using the u/v node IDs
|
|
// as the sort condition
|
|
std::sort(sorted_edge_indexes.begin(),
|
|
sorted_edge_indexes.end(),
|
|
[&edges](const std::size_t &left, const std::size_t &right) -> bool {
|
|
return (edges[left].u != edges[right].u) ? edges[left].u < edges[right].u
|
|
: edges[left].v < edges[right].v;
|
|
});
|
|
|
|
return sorted_edge_indexes;
|
|
}
|
|
|
|
std::vector<NodeID> getSegregatedNodes(const DataFacadeBase &facade,
|
|
const std::vector<RTreeLeaf> &edges)
|
|
{
|
|
std::vector<NodeID> result;
|
|
|
|
for (RTreeLeaf const &e : edges)
|
|
{
|
|
if (e.forward_segment_id.enabled && facade.IsSegregated(e.forward_segment_id.id))
|
|
result.push_back(e.forward_segment_id.id);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
void encodeVectorTile(const DataFacadeBase &facade,
|
|
unsigned x,
|
|
unsigned y,
|
|
unsigned z,
|
|
const std::vector<RTreeLeaf> &edges,
|
|
const std::vector<std::size_t> &sorted_edge_indexes,
|
|
const std::vector<routing_algorithms::TurnData> &all_turn_data,
|
|
const std::vector<NodeID> &segregated_nodes,
|
|
std::string &pbf_buffer)
|
|
{
|
|
|
|
std::uint8_t max_datasource_id = 0;
|
|
|
|
// Vector tiles encode properties on features as indexes into a layer-specific
|
|
// lookup table. These ValueIndexer's act as memoizers for values as we discover
|
|
// them during edge explioration, and are then used to generate the lookup
|
|
// tables for each tile layer.
|
|
ValueIndexer<int> line_int_index;
|
|
ValueIndexer<util::StringView> line_string_index;
|
|
ValueIndexer<int> point_int_index;
|
|
ValueIndexer<float> point_float_index;
|
|
ValueIndexer<std::string> point_string_index;
|
|
|
|
const auto get_geometry_id = [&facade](auto edge) {
|
|
return facade.GetGeometryIndex(edge.forward_segment_id.id).id;
|
|
};
|
|
|
|
// Vector tiles encode feature properties as indexes into a lookup table. So, we need
|
|
// to "pre-loop" over all the edges to create the lookup tables. Once we have those, we
|
|
// can then encode the features, and we'll know the indexes that feature properties
|
|
// need to refer to.
|
|
for (const auto &edge_index : sorted_edge_indexes)
|
|
{
|
|
const auto &edge = edges[edge_index];
|
|
|
|
const auto geometry_id = get_geometry_id(edge);
|
|
const auto forward_datasource_vector =
|
|
facade.GetUncompressedForwardDatasources(geometry_id);
|
|
const auto reverse_datasource_vector =
|
|
facade.GetUncompressedReverseDatasources(geometry_id);
|
|
|
|
BOOST_ASSERT(edge.fwd_segment_position < forward_datasource_vector.size());
|
|
const auto forward_datasource = forward_datasource_vector[edge.fwd_segment_position];
|
|
BOOST_ASSERT(edge.fwd_segment_position < reverse_datasource_vector.size());
|
|
const auto reverse_datasource = reverse_datasource_vector[reverse_datasource_vector.size() -
|
|
edge.fwd_segment_position - 1];
|
|
|
|
// Keep track of the highest datasource seen so that we don't write unnecessary
|
|
// data to the layer attribute values
|
|
max_datasource_id = std::max(max_datasource_id, forward_datasource);
|
|
max_datasource_id = std::max(max_datasource_id, reverse_datasource);
|
|
}
|
|
|
|
// Convert tile coordinates into mercator coordinates
|
|
double min_mercator_lon, min_mercator_lat, max_mercator_lon, max_mercator_lat;
|
|
util::web_mercator::xyzToMercator(
|
|
x, y, z, min_mercator_lon, min_mercator_lat, max_mercator_lon, max_mercator_lat);
|
|
const BBox tile_bbox{min_mercator_lon, min_mercator_lat, max_mercator_lon, max_mercator_lat};
|
|
|
|
// Protobuf serializes blocks when objects go out of scope, hence
|
|
// the extra scoping below.
|
|
protozero::pbf_writer tile_writer{pbf_buffer};
|
|
{
|
|
{
|
|
// Add a layer object to the PBF stream. 3=='layer' from the vector tile spec
|
|
// (2.1)
|
|
protozero::pbf_writer line_layer_writer(tile_writer, util::vector_tile::LAYER_TAG);
|
|
// TODO: don't write a layer if there are no features
|
|
|
|
line_layer_writer.add_uint32(util::vector_tile::VERSION_TAG, 2); // version
|
|
// Field 1 is the "layer name" field, it's a string
|
|
line_layer_writer.add_string(util::vector_tile::NAME_TAG, "speeds"); // name
|
|
// Field 5 is the tile extent. It's a uint32 and should be set to 4096
|
|
// for normal vector tiles.
|
|
line_layer_writer.add_uint32(util::vector_tile::EXTENT_TAG,
|
|
util::vector_tile::EXTENT); // extent
|
|
|
|
// Because we need to know the indexes into the vector tile lookup table,
|
|
// we need to do an initial pass over the data and create the complete
|
|
// index of used values.
|
|
for (const auto &edge_index : sorted_edge_indexes)
|
|
{
|
|
const auto &edge = edges[edge_index];
|
|
const auto geometry_id = get_geometry_id(edge);
|
|
|
|
// Get coordinates for start/end nodes of segment (NodeIDs u and v)
|
|
const auto a = facade.GetCoordinateOfNode(edge.u);
|
|
const auto b = facade.GetCoordinateOfNode(edge.v);
|
|
// Calculate the length in meters
|
|
const double length = osrm::util::coordinate_calculation::haversineDistance(a, b);
|
|
|
|
// Weight values
|
|
const auto forward_weight_vector =
|
|
facade.GetUncompressedForwardWeights(geometry_id);
|
|
const auto reverse_weight_vector =
|
|
facade.GetUncompressedReverseWeights(geometry_id);
|
|
const auto forward_weight = forward_weight_vector[edge.fwd_segment_position];
|
|
const auto reverse_weight = reverse_weight_vector[reverse_weight_vector.size() -
|
|
edge.fwd_segment_position - 1];
|
|
line_int_index.add(forward_weight);
|
|
line_int_index.add(reverse_weight);
|
|
|
|
std::uint32_t forward_rate =
|
|
static_cast<std::uint32_t>(round(length / forward_weight * 10.));
|
|
std::uint32_t reverse_rate =
|
|
static_cast<std::uint32_t>(round(length / reverse_weight * 10.));
|
|
|
|
line_int_index.add(forward_rate);
|
|
line_int_index.add(reverse_rate);
|
|
|
|
// Duration values
|
|
const auto forward_duration_vector =
|
|
facade.GetUncompressedForwardDurations(geometry_id);
|
|
const auto reverse_duration_vector =
|
|
facade.GetUncompressedReverseDurations(geometry_id);
|
|
const auto forward_duration = forward_duration_vector[edge.fwd_segment_position];
|
|
const auto reverse_duration =
|
|
reverse_duration_vector[reverse_duration_vector.size() -
|
|
edge.fwd_segment_position - 1];
|
|
line_int_index.add(forward_duration);
|
|
line_int_index.add(reverse_duration);
|
|
}
|
|
|
|
// Begin the layer features block
|
|
{
|
|
// Each feature gets a unique id, starting at 1
|
|
unsigned id = 1;
|
|
for (const auto &edge_index : sorted_edge_indexes)
|
|
{
|
|
const auto &edge = edges[edge_index];
|
|
const auto geometry_id = get_geometry_id(edge);
|
|
|
|
// Get coordinates for start/end nodes of segment (NodeIDs u and v)
|
|
const auto a = facade.GetCoordinateOfNode(edge.u);
|
|
const auto b = facade.GetCoordinateOfNode(edge.v);
|
|
// Calculate the length in meters
|
|
const double length =
|
|
osrm::util::coordinate_calculation::haversineDistance(a, b);
|
|
|
|
const auto forward_weight_vector =
|
|
facade.GetUncompressedForwardWeights(geometry_id);
|
|
const auto reverse_weight_vector =
|
|
facade.GetUncompressedReverseWeights(geometry_id);
|
|
const auto forward_duration_vector =
|
|
facade.GetUncompressedForwardDurations(geometry_id);
|
|
const auto reverse_duration_vector =
|
|
facade.GetUncompressedReverseDurations(geometry_id);
|
|
const auto forward_datasource_vector =
|
|
facade.GetUncompressedForwardDatasources(geometry_id);
|
|
const auto reverse_datasource_vector =
|
|
facade.GetUncompressedReverseDatasources(geometry_id);
|
|
const auto forward_weight = forward_weight_vector[edge.fwd_segment_position];
|
|
const auto reverse_weight =
|
|
reverse_weight_vector[reverse_weight_vector.size() -
|
|
edge.fwd_segment_position - 1];
|
|
const auto forward_duration =
|
|
forward_duration_vector[edge.fwd_segment_position];
|
|
const auto reverse_duration =
|
|
reverse_duration_vector[reverse_duration_vector.size() -
|
|
edge.fwd_segment_position - 1];
|
|
const auto forward_datasource_idx =
|
|
forward_datasource_vector[edge.fwd_segment_position];
|
|
const auto reverse_datasource_idx =
|
|
reverse_datasource_vector[reverse_datasource_vector.size() -
|
|
edge.fwd_segment_position - 1];
|
|
|
|
const auto component_id = facade.GetComponentID(edge.forward_segment_id.id);
|
|
const auto name_id = facade.GetNameIndex(edge.forward_segment_id.id);
|
|
auto name = facade.GetNameForID(name_id);
|
|
|
|
line_string_index.add(name);
|
|
|
|
const auto encode_tile_line = [&line_layer_writer,
|
|
&edge,
|
|
&component_id,
|
|
&id,
|
|
&max_datasource_id,
|
|
&line_int_index](
|
|
const FixedLine &tile_line,
|
|
const std::uint32_t speed_kmh_idx,
|
|
const std::uint32_t rate_idx,
|
|
const std::size_t weight_idx,
|
|
const std::size_t duration_idx,
|
|
const DatasourceID datasource_idx,
|
|
const std::size_t name_idx,
|
|
std::int32_t &start_x,
|
|
std::int32_t &start_y) {
|
|
// Here, we save the two attributes for our feature: the speed and
|
|
// the is_small boolean. We only serve up speeds from 0-139, so all we
|
|
// do is save the first
|
|
protozero::pbf_writer feature_writer(line_layer_writer,
|
|
util::vector_tile::FEATURE_TAG);
|
|
// Field 3 is the "geometry type" field. Value 2 is "line"
|
|
feature_writer.add_enum(
|
|
util::vector_tile::GEOMETRY_TAG,
|
|
util::vector_tile::GEOMETRY_TYPE_LINE); // geometry type
|
|
// Field 1 for the feature is the "id" field.
|
|
feature_writer.add_uint64(util::vector_tile::ID_TAG, id++); // id
|
|
{
|
|
// When adding attributes to a feature, we have to write
|
|
// pairs of numbers. The first value is the index in the
|
|
// keys array (written later), and the second value is the
|
|
// index into the "values" array (also written later). We're
|
|
// not writing the actual speed or bool value here, we're saving
|
|
// an index into the "values" array. This means many features
|
|
// can share the same value data, leading to smaller tiles.
|
|
protozero::packed_field_uint32 field(
|
|
feature_writer, util::vector_tile::FEATURE_ATTRIBUTES_TAG);
|
|
|
|
field.add_element(0); // "speed" tag key offset
|
|
field.add_element(std::min(
|
|
speed_kmh_idx, 127u)); // save the speed value, capped at 127
|
|
field.add_element(1); // "is_small" tag key offset
|
|
field.add_element(
|
|
128 + (component_id.is_tiny ? 0 : 1)); // is_small feature offset
|
|
field.add_element(2); // "datasource" tag key offset
|
|
field.add_element(130 + datasource_idx); // datasource value offset
|
|
field.add_element(3); // "weight" tag key offset
|
|
field.add_element(130 + max_datasource_id + 1 +
|
|
weight_idx); // weight value offset
|
|
field.add_element(4); // "duration" tag key offset
|
|
field.add_element(130 + max_datasource_id + 1 +
|
|
duration_idx); // duration value offset
|
|
field.add_element(5); // "name" tag key offset
|
|
|
|
field.add_element(130 + max_datasource_id + 1 +
|
|
line_int_index.values().size() + name_idx);
|
|
|
|
field.add_element(6); // rate tag key offset
|
|
field.add_element(130 + max_datasource_id + 1 + rate_idx);
|
|
}
|
|
{
|
|
|
|
// Encode the geometry for the feature
|
|
protozero::packed_field_uint32 geometry(
|
|
feature_writer, util::vector_tile::FEATURE_GEOMETRIES_TAG);
|
|
encodeLinestring(tile_line, geometry, start_x, start_y);
|
|
}
|
|
};
|
|
|
|
// If this is a valid forward edge, go ahead and add it to the tile
|
|
if (forward_duration != 0 && edge.forward_segment_id.enabled)
|
|
{
|
|
std::int32_t start_x = 0;
|
|
std::int32_t start_y = 0;
|
|
|
|
// Calculate the speed for this line
|
|
// Speeds are looked up in a simple 1:1 table, so the speed value == lookup
|
|
// table index
|
|
std::uint32_t speed_kmh_idx =
|
|
static_cast<std::uint32_t>(round(length / forward_duration * 10 * 3.6));
|
|
|
|
// Rate values are in meters per weight-unit - and similar to speeds, we
|
|
// present 1 decimal place of precision (these values are added as
|
|
// double/10) lower down
|
|
std::uint32_t forward_rate =
|
|
static_cast<std::uint32_t>(round(length / forward_weight * 10.));
|
|
|
|
auto tile_line = coordinatesToTileLine(a, b, tile_bbox);
|
|
if (!tile_line.empty())
|
|
{
|
|
encode_tile_line(tile_line,
|
|
speed_kmh_idx,
|
|
line_int_index.indexOf(forward_rate),
|
|
line_int_index.indexOf(forward_weight),
|
|
line_int_index.indexOf(forward_duration),
|
|
forward_datasource_idx,
|
|
line_string_index.indexOf(name),
|
|
start_x,
|
|
start_y);
|
|
}
|
|
}
|
|
|
|
// Repeat the above for the coordinates reversed and using the `reverse`
|
|
// properties
|
|
if (reverse_duration != 0 && edge.reverse_segment_id.enabled)
|
|
{
|
|
std::int32_t start_x = 0;
|
|
std::int32_t start_y = 0;
|
|
|
|
// Calculate the speed for this line
|
|
// Speeds are looked up in a simple 1:1 table, so the speed value == lookup
|
|
// table index
|
|
std::uint32_t speed_kmh_idx =
|
|
static_cast<std::uint32_t>(round(length / reverse_duration * 10 * 3.6));
|
|
|
|
// Rate values are in meters per weight-unit - and similar to speeds, we
|
|
// present 1 decimal place of precision (these values are added as
|
|
// double/10) lower down
|
|
std::uint32_t reverse_rate =
|
|
static_cast<std::uint32_t>(round(length / reverse_weight * 10.));
|
|
|
|
auto tile_line = coordinatesToTileLine(b, a, tile_bbox);
|
|
if (!tile_line.empty())
|
|
{
|
|
encode_tile_line(tile_line,
|
|
speed_kmh_idx,
|
|
line_int_index.indexOf(reverse_rate),
|
|
line_int_index.indexOf(reverse_weight),
|
|
line_int_index.indexOf(reverse_duration),
|
|
reverse_datasource_idx,
|
|
line_string_index.indexOf(name),
|
|
start_x,
|
|
start_y);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Field id 3 is the "keys" attribute
|
|
// We need two "key" fields, these are referred to with 0 and 1 (their array
|
|
// indexes) earlier
|
|
line_layer_writer.add_string(util::vector_tile::KEY_TAG, "speed");
|
|
line_layer_writer.add_string(util::vector_tile::KEY_TAG, "is_small");
|
|
line_layer_writer.add_string(util::vector_tile::KEY_TAG, "datasource");
|
|
line_layer_writer.add_string(util::vector_tile::KEY_TAG, "weight");
|
|
line_layer_writer.add_string(util::vector_tile::KEY_TAG, "duration");
|
|
line_layer_writer.add_string(util::vector_tile::KEY_TAG, "name");
|
|
line_layer_writer.add_string(util::vector_tile::KEY_TAG, "rate");
|
|
|
|
// Now, we write out the possible speed value arrays and possible is_tiny
|
|
// values. Field type 4 is the "values" field. It's a variable type field,
|
|
// so requires a two-step write (create the field, then write its value)
|
|
for (std::size_t i = 0; i < 128; i++)
|
|
{
|
|
// Writing field type 4 == variant type
|
|
protozero::pbf_writer values_writer(line_layer_writer,
|
|
util::vector_tile::VARIANT_TAG);
|
|
// Attribute value 5 == uint64 type
|
|
values_writer.add_uint64(util::vector_tile::VARIANT_TYPE_UINT64, i);
|
|
}
|
|
{
|
|
protozero::pbf_writer values_writer(line_layer_writer,
|
|
util::vector_tile::VARIANT_TAG);
|
|
// Attribute value 7 == bool type
|
|
values_writer.add_bool(util::vector_tile::VARIANT_TYPE_BOOL, true);
|
|
}
|
|
{
|
|
protozero::pbf_writer values_writer(line_layer_writer,
|
|
util::vector_tile::VARIANT_TAG);
|
|
// Attribute value 7 == bool type
|
|
values_writer.add_bool(util::vector_tile::VARIANT_TYPE_BOOL, false);
|
|
}
|
|
for (std::size_t i = 0; i <= max_datasource_id; i++)
|
|
{
|
|
// Writing field type 4 == variant type
|
|
protozero::pbf_writer values_writer(line_layer_writer,
|
|
util::vector_tile::VARIANT_TAG);
|
|
// Attribute value 1 == string type
|
|
values_writer.add_string(util::vector_tile::VARIANT_TYPE_STRING,
|
|
facade.GetDatasourceName(i).to_string());
|
|
}
|
|
for (auto value : line_int_index.values())
|
|
{
|
|
// Writing field type 4 == variant type
|
|
protozero::pbf_writer values_writer(line_layer_writer,
|
|
util::vector_tile::VARIANT_TAG);
|
|
// Attribute value 2 == float type
|
|
// Durations come out of OSRM in integer deciseconds, so we convert them
|
|
// to seconds with a simple /10 for display
|
|
values_writer.add_double(util::vector_tile::VARIANT_TYPE_DOUBLE, value / 10.);
|
|
}
|
|
|
|
for (const auto &name : line_string_index.values())
|
|
{
|
|
// Writing field type 4 == variant type
|
|
protozero::pbf_writer values_writer(line_layer_writer,
|
|
util::vector_tile::VARIANT_TAG);
|
|
// Attribute value 1 == string type
|
|
values_writer.add_string(
|
|
util::vector_tile::VARIANT_TYPE_STRING, name.data(), name.size());
|
|
}
|
|
}
|
|
|
|
// Only add the turn layer to the tile if it has some features (we sometimes won't
|
|
// for tiles of z<16, and tiles that don't show any intersections)
|
|
if (!all_turn_data.empty())
|
|
{
|
|
|
|
struct EncodedTurnData
|
|
{
|
|
util::Coordinate coordinate;
|
|
std::size_t angle_index;
|
|
std::size_t turn_index;
|
|
std::size_t duration_index;
|
|
std::size_t weight_index;
|
|
std::size_t turntype_index;
|
|
std::size_t turnmodifier_index;
|
|
};
|
|
// we need to pre-encode all values here because we need the full offsets later
|
|
// for encoding the actual features.
|
|
std::vector<EncodedTurnData> encoded_turn_data(all_turn_data.size());
|
|
std::transform(
|
|
all_turn_data.begin(),
|
|
all_turn_data.end(),
|
|
encoded_turn_data.begin(),
|
|
[&](const routing_algorithms::TurnData &t) {
|
|
auto angle_idx = point_int_index.add(t.in_angle);
|
|
auto turn_idx = point_int_index.add(t.turn_angle);
|
|
auto duration_idx =
|
|
point_float_index.add(t.duration / 10.0); // Note conversion to float here
|
|
auto weight_idx =
|
|
point_float_index.add(t.weight / 10.0); // Note conversion to float here
|
|
|
|
auto turntype_idx = point_string_index.add(
|
|
osrm::guidance::internalInstructionTypeToString(t.turn_instruction.type));
|
|
auto turnmodifier_idx =
|
|
point_string_index.add(osrm::guidance::instructionModifierToString(
|
|
t.turn_instruction.direction_modifier));
|
|
return EncodedTurnData{t.coordinate,
|
|
angle_idx,
|
|
turn_idx,
|
|
duration_idx,
|
|
weight_idx,
|
|
turntype_idx,
|
|
turnmodifier_idx};
|
|
});
|
|
|
|
// Now write the points layer for turn penalty data:
|
|
// Add a layer object to the PBF stream. 3=='layer' from the vector tile spec
|
|
// (2.1)
|
|
protozero::pbf_writer point_layer_writer(tile_writer, util::vector_tile::LAYER_TAG);
|
|
point_layer_writer.add_uint32(util::vector_tile::VERSION_TAG, 2); // version
|
|
point_layer_writer.add_string(util::vector_tile::NAME_TAG, "turns"); // name
|
|
point_layer_writer.add_uint32(util::vector_tile::EXTENT_TAG,
|
|
util::vector_tile::EXTENT); // extent
|
|
|
|
// Begin writing the set of point features
|
|
{
|
|
// Start each features with an ID starting at 1
|
|
int id = 1;
|
|
|
|
// Helper function to encode a new point feature on a vector tile.
|
|
const auto encode_tile_point = [&](const FixedPoint &tile_point,
|
|
const auto &point_turn_data) {
|
|
protozero::pbf_writer feature_writer(point_layer_writer,
|
|
util::vector_tile::FEATURE_TAG);
|
|
// Field 3 is the "geometry type" field. Value 1 is "point"
|
|
feature_writer.add_enum(
|
|
util::vector_tile::GEOMETRY_TAG,
|
|
util::vector_tile::GEOMETRY_TYPE_POINT); // geometry type
|
|
feature_writer.add_uint64(util::vector_tile::ID_TAG, id++); // id
|
|
{
|
|
// Write out the 4 properties we want on the feature. These
|
|
// refer to indexes in the properties lookup table, which we
|
|
// add to the tile after we add all features.
|
|
protozero::packed_field_uint32 field(
|
|
feature_writer, util::vector_tile::FEATURE_ATTRIBUTES_TAG);
|
|
field.add_element(0); // "bearing_in" tag key offset
|
|
field.add_element(point_turn_data.angle_index);
|
|
field.add_element(1); // "turn_angle" tag key offset
|
|
field.add_element(point_turn_data.turn_index);
|
|
field.add_element(2); // "cost" tag key offset
|
|
field.add_element(point_int_index.size() + point_turn_data.duration_index);
|
|
field.add_element(3); // "weight" tag key offset
|
|
field.add_element(point_int_index.size() + point_turn_data.weight_index);
|
|
field.add_element(4); // "type" tag key offset
|
|
field.add_element(point_int_index.size() + point_float_index.size() +
|
|
point_turn_data.turntype_index);
|
|
field.add_element(5); // "modifier" tag key offset
|
|
field.add_element(point_int_index.size() + point_float_index.size() +
|
|
point_turn_data.turnmodifier_index);
|
|
}
|
|
{
|
|
// Add the geometry as the last field in this feature
|
|
protozero::packed_field_uint32 geometry(
|
|
feature_writer, util::vector_tile::FEATURE_GEOMETRIES_TAG);
|
|
encodePoint(tile_point, geometry);
|
|
}
|
|
};
|
|
|
|
// Loop over all the turns we found and add them as features to the layer
|
|
for (const auto &turndata : encoded_turn_data)
|
|
{
|
|
const auto tile_point = coordinatesToTilePoint(turndata.coordinate, tile_bbox);
|
|
if (!boost::geometry::within(point_t(tile_point.x, tile_point.y), clip_box))
|
|
{
|
|
continue;
|
|
}
|
|
encode_tile_point(tile_point, turndata);
|
|
}
|
|
}
|
|
|
|
// Add the names of the three attributes we added to all the turn penalty
|
|
// features previously. The indexes used there refer to these keys.
|
|
point_layer_writer.add_string(util::vector_tile::KEY_TAG, "bearing_in");
|
|
point_layer_writer.add_string(util::vector_tile::KEY_TAG, "turn_angle");
|
|
point_layer_writer.add_string(util::vector_tile::KEY_TAG, "cost");
|
|
point_layer_writer.add_string(util::vector_tile::KEY_TAG, "weight");
|
|
point_layer_writer.add_string(util::vector_tile::KEY_TAG, "type");
|
|
point_layer_writer.add_string(util::vector_tile::KEY_TAG, "modifier");
|
|
|
|
// Now, save the lists of integers and floats that our features refer to.
|
|
for (const auto &value : point_int_index.values())
|
|
{
|
|
protozero::pbf_writer values_writer(point_layer_writer,
|
|
util::vector_tile::VARIANT_TAG);
|
|
values_writer.add_sint64(util::vector_tile::VARIANT_TYPE_SINT64, value);
|
|
}
|
|
for (const auto &value : point_float_index.values())
|
|
{
|
|
protozero::pbf_writer values_writer(point_layer_writer,
|
|
util::vector_tile::VARIANT_TAG);
|
|
values_writer.add_float(util::vector_tile::VARIANT_TYPE_FLOAT, value);
|
|
}
|
|
for (const auto &value : point_string_index.values())
|
|
{
|
|
protozero::pbf_writer values_writer(point_layer_writer,
|
|
util::vector_tile::VARIANT_TAG);
|
|
values_writer.add_string(util::vector_tile::VARIANT_TYPE_STRING, value);
|
|
}
|
|
}
|
|
|
|
// OSM Node tile layer
|
|
{
|
|
protozero::pbf_writer point_layer_writer(tile_writer, util::vector_tile::LAYER_TAG);
|
|
point_layer_writer.add_uint32(util::vector_tile::VERSION_TAG, 2); // version
|
|
point_layer_writer.add_string(util::vector_tile::NAME_TAG, "osmnodes"); // name
|
|
point_layer_writer.add_uint32(util::vector_tile::EXTENT_TAG,
|
|
util::vector_tile::EXTENT); // extent
|
|
|
|
std::vector<NodeID> internal_nodes;
|
|
internal_nodes.reserve(edges.size() * 2);
|
|
for (const auto &edge : edges)
|
|
{
|
|
internal_nodes.push_back(edge.u);
|
|
internal_nodes.push_back(edge.v);
|
|
}
|
|
std::sort(internal_nodes.begin(), internal_nodes.end());
|
|
auto new_end = std::unique(internal_nodes.begin(), internal_nodes.end());
|
|
internal_nodes.resize(new_end - internal_nodes.begin());
|
|
|
|
for (const auto &internal_node : internal_nodes)
|
|
{
|
|
const auto coord = facade.GetCoordinateOfNode(internal_node);
|
|
const auto tile_point = coordinatesToTilePoint(coord, tile_bbox);
|
|
if (!boost::geometry::within(point_t(tile_point.x, tile_point.y), clip_box))
|
|
{
|
|
continue;
|
|
}
|
|
protozero::pbf_writer feature_writer(point_layer_writer,
|
|
util::vector_tile::FEATURE_TAG);
|
|
// Field 3 is the "geometry type" field. Value 1 is "point"
|
|
feature_writer.add_enum(util::vector_tile::GEOMETRY_TAG,
|
|
util::vector_tile::GEOMETRY_TYPE_POINT); // geometry type
|
|
const auto osmid =
|
|
static_cast<OSMNodeID::value_type>(facade.GetOSMNodeIDOfNode(internal_node));
|
|
feature_writer.add_uint64(util::vector_tile::ID_TAG, osmid); // id
|
|
// There are no additional properties, just the ID and the geometry
|
|
{
|
|
// Add the geometry as the last field in this feature
|
|
protozero::packed_field_uint32 geometry(
|
|
feature_writer, util::vector_tile::FEATURE_GEOMETRIES_TAG);
|
|
encodePoint(tile_point, geometry);
|
|
}
|
|
}
|
|
}
|
|
|
|
{
|
|
protozero::pbf_writer line_layer_writer(tile_writer, util::vector_tile::LAYER_TAG);
|
|
line_layer_writer.add_uint32(util::vector_tile::VERSION_TAG, 2); // version
|
|
line_layer_writer.add_string(util::vector_tile::NAME_TAG, "internal-nodes"); // name
|
|
line_layer_writer.add_uint32(util::vector_tile::EXTENT_TAG,
|
|
util::vector_tile::EXTENT); // extent
|
|
|
|
unsigned id = 0;
|
|
for (auto edgeNodeID : segregated_nodes)
|
|
{
|
|
auto const geomIndex = facade.GetGeometryIndex(edgeNodeID);
|
|
std::vector<NodeID> geometry;
|
|
|
|
if (geomIndex.forward)
|
|
geometry = facade.GetUncompressedForwardGeometry(geomIndex.id);
|
|
else
|
|
geometry = facade.GetUncompressedReverseGeometry(geomIndex.id);
|
|
|
|
std::vector<util::Coordinate> points;
|
|
for (auto const nodeID : geometry)
|
|
points.push_back(facade.GetCoordinateOfNode(nodeID));
|
|
|
|
const auto encode_tile_line = [&line_layer_writer, &id](
|
|
const FixedLine &tile_line, std::int32_t &start_x, std::int32_t &start_y) {
|
|
|
|
protozero::pbf_writer feature_writer(line_layer_writer,
|
|
util::vector_tile::FEATURE_TAG);
|
|
|
|
feature_writer.add_enum(util::vector_tile::GEOMETRY_TAG,
|
|
util::vector_tile::GEOMETRY_TYPE_LINE); // geometry type
|
|
|
|
feature_writer.add_uint64(util::vector_tile::ID_TAG, id++); // id
|
|
{
|
|
|
|
protozero::packed_field_uint32 field(
|
|
feature_writer, util::vector_tile::FEATURE_ATTRIBUTES_TAG);
|
|
}
|
|
{
|
|
|
|
// Encode the geometry for the feature
|
|
protozero::packed_field_uint32 geometry(
|
|
feature_writer, util::vector_tile::FEATURE_GEOMETRIES_TAG);
|
|
encodeLinestring(tile_line, geometry, start_x, start_y);
|
|
}
|
|
};
|
|
|
|
std::int32_t start_x = 0;
|
|
std::int32_t start_y = 0;
|
|
|
|
auto tile_lines = coordinatesToTileLine(points, tile_bbox);
|
|
if (!tile_lines.empty())
|
|
{
|
|
for (auto const &tl : tile_lines)
|
|
{
|
|
encode_tile_line(tl, start_x, start_y);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// protozero serializes data during object destructors, so once the scope closes,
|
|
// our result buffer will have all the tile data encoded into it.
|
|
}
|
|
}
|
|
|
|
Status TilePlugin::HandleRequest(const RoutingAlgorithmsInterface &algorithms,
|
|
const api::TileParameters ¶meters,
|
|
std::string &pbf_buffer) const
|
|
{
|
|
BOOST_ASSERT(parameters.IsValid());
|
|
|
|
const auto &facade = algorithms.GetFacade();
|
|
auto edges = getEdges(facade, parameters.x, parameters.y, parameters.z);
|
|
auto segregated_nodes = getSegregatedNodes(facade, edges);
|
|
|
|
auto edge_index = getEdgeIndex(edges);
|
|
|
|
std::vector<routing_algorithms::TurnData> turns;
|
|
|
|
// If we're zooming into 16 or higher, include turn data. Why? Because turns make the map
|
|
// really cramped, so we don't bother including the data for tiles that span a large area.
|
|
if (parameters.z >= MIN_ZOOM_FOR_TURNS && algorithms.HasGetTileTurns())
|
|
{
|
|
turns = algorithms.GetTileTurns(edges, edge_index);
|
|
}
|
|
|
|
encodeVectorTile(facade,
|
|
parameters.x,
|
|
parameters.y,
|
|
parameters.z,
|
|
edges,
|
|
edge_index,
|
|
turns,
|
|
segregated_nodes,
|
|
pbf_buffer);
|
|
|
|
return Status::Ok;
|
|
}
|
|
}
|
|
}
|
|
}
|