#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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include 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 struct ValueIndexer { private: std::vector used_values; std::unordered_map 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 &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 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; using FloatPoint = Point; using FixedLine = std::vector; using FloatLine = std::vector; // 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 point_t; typedef boost::geometry::model::linestring linestring_t; typedef boost::geometry::model::box box_t; typedef boost::geometry::model::multi_linestring 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(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 coordinatesToTileLine(const std::vector &points, const BBox &tile_bbox) { FloatLine geo_line; for (auto const &c : points) { geo_line.emplace_back(static_cast(util::toFloating(c.lon)), static_cast(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 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(util::toFloating(start.lon)), static_cast(util::toFloating(start.lat))); geo_line.emplace_back(static_cast(util::toFloating(target.lon)), static_cast(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(util::toFloating(point.lon)), static_cast(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::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::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 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 getEdgeIndex(const std::vector &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 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 getSegregatedNodes(const DataFacadeBase &facade, const std::vector &edges) { std::vector 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 &edges, const std::vector &sorted_edge_indexes, const std::vector &all_turn_data, const std::vector &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 line_int_index; ValueIndexer line_string_index; ValueIndexer point_int_index; ValueIndexer point_float_index; ValueIndexer 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(round(length / forward_weight * 10.)); std::uint32_t reverse_rate = static_cast(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(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(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(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(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 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 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(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 geometry; if (geomIndex.forward) geometry = facade.GetUncompressedForwardGeometry(geomIndex.id); else geometry = facade.GetUncompressedReverseGeometry(geomIndex.id); std::vector 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 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; } } } }