913 lines
39 KiB
C++
913 lines
39 KiB
C++
#include "extractor/extractor.hpp"
|
|
|
|
#include "extractor/compressed_edge_container.hpp"
|
|
#include "extractor/compressed_node_based_graph_edge.hpp"
|
|
#include "extractor/edge_based_edge.hpp"
|
|
#include "extractor/extraction_containers.hpp"
|
|
#include "extractor/extraction_node.hpp"
|
|
#include "extractor/extraction_relation.hpp"
|
|
#include "extractor/extraction_way.hpp"
|
|
#include "extractor/extractor_callbacks.hpp"
|
|
#include "extractor/files.hpp"
|
|
#include "extractor/maneuver_override_relation_parser.hpp"
|
|
#include "extractor/name_table.hpp"
|
|
#include "extractor/node_based_graph_factory.hpp"
|
|
#include "extractor/node_restriction_map.hpp"
|
|
#include "extractor/restriction_graph.hpp"
|
|
#include "extractor/restriction_parser.hpp"
|
|
#include "extractor/scripting_environment.hpp"
|
|
#include "extractor/turn_path_filter.hpp"
|
|
#include "extractor/way_restriction_map.hpp"
|
|
|
|
#include "guidance/files.hpp"
|
|
#include "guidance/guidance_processing.hpp"
|
|
#include "guidance/segregated_intersection_classification.hpp"
|
|
#include "guidance/turn_data_container.hpp"
|
|
|
|
#include "util/exception.hpp"
|
|
#include "util/exception_utils.hpp"
|
|
#include "util/integer_range.hpp"
|
|
#include "util/log.hpp"
|
|
#include "util/static_graph.hpp"
|
|
#include "util/static_rtree.hpp"
|
|
#include "util/tarjan_scc.hpp"
|
|
#include "util/timing_util.hpp"
|
|
|
|
// Keep debug include to make sure the debug header is in sync with types.
|
|
#include "util/debug.hpp"
|
|
|
|
#include <boost/assert.hpp>
|
|
|
|
#include <osmium/handler/node_locations_for_ways.hpp>
|
|
#include <osmium/index/map/flex_mem.hpp>
|
|
#include <osmium/io/any_input.hpp>
|
|
#include <osmium/thread/pool.hpp>
|
|
#include <osmium/visitor.hpp>
|
|
#include <tbb/global_control.h>
|
|
#include <tbb/parallel_pipeline.h>
|
|
|
|
#include <algorithm>
|
|
#include <atomic>
|
|
#include <bitset>
|
|
#include <chrono>
|
|
#include <iostream>
|
|
#include <memory>
|
|
#include <thread>
|
|
#include <tuple>
|
|
#include <type_traits>
|
|
#include <unordered_map>
|
|
#include <vector>
|
|
|
|
namespace osrm::extractor
|
|
{
|
|
|
|
namespace
|
|
{
|
|
// Converts the class name map into a fixed mapping of index to name
|
|
void SetClassNames(const std::vector<std::string> &class_names,
|
|
ExtractorCallbacks::ClassesMap &classes_map,
|
|
ProfileProperties &profile_properties)
|
|
{
|
|
// if we get a list of class names we can validate if we set invalid classes
|
|
// and add classes that were never reference
|
|
if (!class_names.empty())
|
|
{
|
|
// add class names that were never used explicitly on a way
|
|
// this makes sure we can correctly validate unknown class names later
|
|
for (const auto &name : class_names)
|
|
{
|
|
if (!isValidClassName(name))
|
|
{
|
|
throw util::exception("Invalid class name " + name + " only [a-Z0-9] allowed.");
|
|
}
|
|
|
|
auto iter = classes_map.find(name);
|
|
if (iter == classes_map.end())
|
|
{
|
|
auto index = classes_map.size();
|
|
if (index > MAX_CLASS_INDEX)
|
|
{
|
|
throw util::exception("Maximum number of classes is " +
|
|
std::to_string(MAX_CLASS_INDEX + 1));
|
|
}
|
|
|
|
classes_map[name] = getClassData(index);
|
|
}
|
|
}
|
|
|
|
// check if class names are only from the list supplied by the user
|
|
for (const auto &pair : classes_map)
|
|
{
|
|
auto iter = std::find(class_names.begin(), class_names.end(), pair.first);
|
|
if (iter == class_names.end())
|
|
{
|
|
throw util::exception("Profile used unknown class name: " + pair.first);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const auto &pair : classes_map)
|
|
{
|
|
auto range = getClassIndexes(pair.second);
|
|
BOOST_ASSERT(range.size() == 1);
|
|
profile_properties.SetClassName(range.front(), pair.first);
|
|
}
|
|
}
|
|
|
|
// Converts the class name list to a mask list
|
|
void SetExcludableClasses(const ExtractorCallbacks::ClassesMap &classes_map,
|
|
const std::vector<std::vector<std::string>> &excludable_classes,
|
|
ProfileProperties &profile_properties)
|
|
{
|
|
if (excludable_classes.size() > MAX_EXCLUDABLE_CLASSES)
|
|
{
|
|
throw util::exception("Only " + std::to_string(MAX_EXCLUDABLE_CLASSES) +
|
|
" excludable combinations allowed.");
|
|
}
|
|
|
|
// The exclude index 0 is reserve for not excludeing anything
|
|
profile_properties.SetExcludableClasses(0, 0);
|
|
|
|
std::size_t combination_index = 1;
|
|
for (const auto &combination : excludable_classes)
|
|
{
|
|
ClassData mask = 0;
|
|
for (const auto &name : combination)
|
|
{
|
|
auto iter = classes_map.find(name);
|
|
if (iter == classes_map.end())
|
|
{
|
|
util::Log(logWARNING)
|
|
<< "Unknown class name " + name + " in excludable combination. Ignoring.";
|
|
}
|
|
else
|
|
{
|
|
mask |= iter->second;
|
|
}
|
|
}
|
|
|
|
if (mask > 0)
|
|
{
|
|
profile_properties.SetExcludableClasses(combination_index++, mask);
|
|
}
|
|
}
|
|
}
|
|
|
|
std::vector<CompressedNodeBasedGraphEdge> toEdgeList(const util::NodeBasedDynamicGraph &graph)
|
|
{
|
|
std::vector<CompressedNodeBasedGraphEdge> edges;
|
|
edges.reserve(graph.GetNumberOfEdges());
|
|
|
|
// For all nodes iterate over its edges and dump (from, to) pairs
|
|
for (const NodeID from_node : util::irange(0u, graph.GetNumberOfNodes()))
|
|
{
|
|
for (const EdgeID edge : graph.GetAdjacentEdgeRange(from_node))
|
|
{
|
|
const auto to_node = graph.GetTarget(edge);
|
|
|
|
edges.push_back({from_node, to_node});
|
|
}
|
|
}
|
|
|
|
return edges;
|
|
}
|
|
} // namespace
|
|
|
|
/**
|
|
* TODO: Refactor this function into smaller functions for better readability.
|
|
*
|
|
* This function is the entry point for the whole extraction process. The goal of the extraction
|
|
* step is to filter and convert the OSM geometry to something more fitting for routing.
|
|
* That includes:
|
|
* - extracting turn restrictions
|
|
* - splitting ways into (directional!) edge segments
|
|
* - checking if nodes are barriers or traffic signal
|
|
* - discarding all tag information: All relevant type information for nodes/ways
|
|
* is extracted at this point.
|
|
*
|
|
* The result of this process are the following files:
|
|
* .names : Names of all streets, stored as long consecutive string with prefix sum based index
|
|
* .osrm : Nodes and edges in a intermediate format that easy to digest for osrm-contract
|
|
* .restrictions : Turn restrictions that are used by osrm-contract to construct the edge-expanded
|
|
* graph
|
|
*
|
|
*/
|
|
int Extractor::run(ScriptingEnvironment &scripting_environment)
|
|
{
|
|
util::LogPolicy::GetInstance().Unmute();
|
|
|
|
const unsigned recommended_num_threads = std::thread::hardware_concurrency();
|
|
const auto number_of_threads = std::min(recommended_num_threads, config.requested_num_threads);
|
|
|
|
tbb::global_control gc(tbb::global_control::max_allowed_parallelism,
|
|
config.requested_num_threads);
|
|
|
|
auto parsed_osm_data = ParseOSMData(scripting_environment, number_of_threads);
|
|
|
|
// Transform the node-based graph that OSM is based on into an edge-based graph
|
|
// that is better for routing. Every edge becomes a node, and every valid
|
|
// movement (e.g. turn from A->B, and B->A) becomes an edge
|
|
util::Log() << "Generating edge-expanded graph representation";
|
|
|
|
TIMER_START(expansion);
|
|
|
|
EdgeBasedNodeDataContainer edge_based_nodes_container;
|
|
std::vector<EdgeBasedNodeSegment> edge_based_node_segments;
|
|
util::DeallocatingVector<EdgeBasedEdge> edge_based_edge_list;
|
|
std::vector<EdgeWeight> edge_based_node_weights;
|
|
std::vector<EdgeDuration> edge_based_node_durations;
|
|
std::vector<EdgeDistance> edge_based_node_distances;
|
|
std::uint32_t ebg_connectivity_checksum = 0;
|
|
|
|
// Create a node-based graph from the OSRM file
|
|
NodeBasedGraphFactory node_based_graph_factory(scripting_environment,
|
|
parsed_osm_data.turn_restrictions,
|
|
parsed_osm_data.unresolved_maneuver_overrides,
|
|
parsed_osm_data.traffic_signals,
|
|
std::move(parsed_osm_data.barriers),
|
|
std::move(parsed_osm_data.osm_coordinates),
|
|
std::move(parsed_osm_data.osm_node_ids),
|
|
parsed_osm_data.edge_list,
|
|
std::move(parsed_osm_data.annotation_data));
|
|
|
|
NameTable name_table;
|
|
files::readNames(config.GetPath(".osrm.names"), name_table);
|
|
|
|
util::Log() << "Find segregated edges in node-based graph ..." << std::flush;
|
|
TIMER_START(segregated);
|
|
|
|
auto segregated_edges = guidance::findSegregatedNodes(node_based_graph_factory, name_table);
|
|
|
|
TIMER_STOP(segregated);
|
|
util::Log() << "ok, after " << TIMER_SEC(segregated) << "s";
|
|
util::Log() << "Segregated edges count = " << segregated_edges.size();
|
|
|
|
util::Log() << "Writing nodes for nodes-based and edges-based graphs ...";
|
|
auto const &coordinates = node_based_graph_factory.GetCoordinates();
|
|
files::writeNodes(
|
|
config.GetPath(".osrm.nbg_nodes"), coordinates, node_based_graph_factory.GetOsmNodes());
|
|
node_based_graph_factory.ReleaseOsmNodes();
|
|
|
|
auto const &node_based_graph = node_based_graph_factory.GetGraph();
|
|
|
|
// The osrm-partition tool requires the compressed node based graph with an embedding.
|
|
//
|
|
// The `Run` function above re-numbers non-reverse compressed node based graph edges
|
|
// to a continuous range so that the nodes in the edge based graph are continuous.
|
|
//
|
|
// Luckily node based node ids still coincide with the coordinate array.
|
|
// That's the reason we can only here write out the final compressed node based graph.
|
|
files::writeCompressedNodeBasedGraph(config.GetPath(".osrm.cnbg").string(),
|
|
toEdgeList(node_based_graph));
|
|
|
|
node_based_graph_factory.GetCompressedEdges().PrintStatistics();
|
|
|
|
const auto &barrier_nodes = node_based_graph_factory.GetBarriers();
|
|
// stealing the annotation data from the node-based graph
|
|
edge_based_nodes_container =
|
|
EdgeBasedNodeDataContainer({}, std::move(node_based_graph_factory.GetAnnotationData()));
|
|
|
|
parsed_osm_data.turn_restrictions =
|
|
removeInvalidTurnPaths(std::move(parsed_osm_data.turn_restrictions), node_based_graph);
|
|
parsed_osm_data.unresolved_maneuver_overrides = removeInvalidTurnPaths(
|
|
std::move(parsed_osm_data.unresolved_maneuver_overrides), node_based_graph);
|
|
auto restriction_graph = constructRestrictionGraph(parsed_osm_data.turn_restrictions);
|
|
|
|
const auto number_of_node_based_nodes = node_based_graph.GetNumberOfNodes();
|
|
|
|
const auto number_of_edge_based_nodes =
|
|
BuildEdgeExpandedGraph(node_based_graph,
|
|
coordinates,
|
|
node_based_graph_factory.GetCompressedEdges(),
|
|
barrier_nodes,
|
|
parsed_osm_data.traffic_signals,
|
|
restriction_graph,
|
|
segregated_edges,
|
|
name_table,
|
|
parsed_osm_data.unresolved_maneuver_overrides,
|
|
parsed_osm_data.turn_lane_map,
|
|
scripting_environment,
|
|
edge_based_nodes_container,
|
|
edge_based_node_segments,
|
|
edge_based_node_weights,
|
|
edge_based_node_durations,
|
|
edge_based_node_distances,
|
|
edge_based_edge_list,
|
|
ebg_connectivity_checksum);
|
|
|
|
ProcessGuidanceTurns(node_based_graph,
|
|
edge_based_nodes_container,
|
|
coordinates,
|
|
node_based_graph_factory.GetCompressedEdges(),
|
|
barrier_nodes,
|
|
restriction_graph,
|
|
name_table,
|
|
std::move(parsed_osm_data.turn_lane_map),
|
|
scripting_environment);
|
|
|
|
TIMER_STOP(expansion);
|
|
|
|
// output the geometry of the node-based graph, needs to be done after the last usage, since it
|
|
// destroys internal containers
|
|
files::writeSegmentData(config.GetPath(".osrm.geometry"),
|
|
*node_based_graph_factory.GetCompressedEdges().ToSegmentData());
|
|
|
|
util::Log() << "Saving edge-based node weights to file.";
|
|
TIMER_START(timer_write_node_weights);
|
|
extractor::files::writeEdgeBasedNodeWeightsDurationsDistances(config.GetPath(".osrm.enw"),
|
|
edge_based_node_weights,
|
|
edge_based_node_durations,
|
|
edge_based_node_distances);
|
|
TIMER_STOP(timer_write_node_weights);
|
|
util::Log() << "Done writing. (" << TIMER_SEC(timer_write_node_weights) << ")";
|
|
|
|
util::Log() << "Computing strictly connected components ...";
|
|
FindComponents(number_of_edge_based_nodes,
|
|
edge_based_edge_list,
|
|
edge_based_node_segments,
|
|
edge_based_nodes_container);
|
|
|
|
util::Log() << "Building r-tree ...";
|
|
TIMER_START(rtree);
|
|
BuildRTree(std::move(edge_based_node_segments), coordinates);
|
|
|
|
TIMER_STOP(rtree);
|
|
|
|
files::writeNodeData(config.GetPath(".osrm.ebg_nodes"), edge_based_nodes_container);
|
|
|
|
util::Log() << "Writing edge-based-graph edges ... " << std::flush;
|
|
TIMER_START(write_edges);
|
|
files::writeEdgeBasedGraph(config.GetPath(".osrm.ebg"),
|
|
number_of_edge_based_nodes,
|
|
edge_based_edge_list,
|
|
ebg_connectivity_checksum);
|
|
TIMER_STOP(write_edges);
|
|
util::Log() << "ok, after " << TIMER_SEC(write_edges) << "s";
|
|
|
|
util::Log() << "Processed " << edge_based_edge_list.size() << " edges";
|
|
|
|
const auto nodes_per_second =
|
|
static_cast<std::uint64_t>(number_of_node_based_nodes / TIMER_SEC(expansion));
|
|
const auto edges_per_second =
|
|
static_cast<std::uint64_t>((number_of_edge_based_nodes) / TIMER_SEC(expansion));
|
|
|
|
util::Log() << "Expansion: " << nodes_per_second << " nodes/sec and " << edges_per_second
|
|
<< " edges/sec";
|
|
util::Log() << "To prepare the data for routing, run: "
|
|
<< "./osrm-contract " << config.base_path;
|
|
|
|
return 0;
|
|
}
|
|
|
|
Extractor::ParsedOSMData Extractor::ParseOSMData(ScriptingEnvironment &scripting_environment,
|
|
const unsigned number_of_threads)
|
|
{
|
|
TIMER_START(extracting);
|
|
|
|
util::Log() << "Input file: " << config.input_path.filename().string();
|
|
if (!config.profile_path.empty())
|
|
{
|
|
util::Log() << "Profile: " << config.profile_path.filename().string();
|
|
}
|
|
util::Log() << "Threads: " << number_of_threads;
|
|
|
|
const osmium::io::File input_file(config.input_path.string());
|
|
osmium::thread::Pool pool(number_of_threads);
|
|
|
|
util::Log() << "Parsing in progress..";
|
|
TIMER_START(parsing);
|
|
|
|
{ // Parse OSM header
|
|
osmium::io::Reader reader(input_file, pool, osmium::osm_entity_bits::nothing);
|
|
osmium::io::Header header = reader.header();
|
|
|
|
std::string generator = header.get("generator");
|
|
if (generator.empty())
|
|
{
|
|
generator = "unknown tool";
|
|
}
|
|
util::Log() << "input file generated by " << generator;
|
|
|
|
// write .timestamp data file
|
|
std::string timestamp = header.get("osmosis_replication_timestamp");
|
|
if (config.data_version == "osmosis")
|
|
{
|
|
files::writeTimestamp(config.GetPath(".osrm.timestamp").string(), timestamp);
|
|
}
|
|
else
|
|
{
|
|
files::writeTimestamp(config.GetPath(".osrm.timestamp").string(), config.data_version);
|
|
}
|
|
if (timestamp.empty())
|
|
{
|
|
timestamp = "n/a";
|
|
}
|
|
util::Log() << "timestamp: " << timestamp;
|
|
}
|
|
|
|
// Extraction containers and restriction parser
|
|
ExtractionContainers extraction_containers;
|
|
ExtractorCallbacks::ClassesMap classes_map;
|
|
LaneDescriptionMap turn_lane_map;
|
|
auto extractor_callbacks =
|
|
std::make_unique<ExtractorCallbacks>(extraction_containers,
|
|
classes_map,
|
|
turn_lane_map,
|
|
scripting_environment.GetProfileProperties());
|
|
|
|
// get list of supported relation types
|
|
auto relation_types = scripting_environment.GetRelations();
|
|
std::sort(relation_types.begin(), relation_types.end());
|
|
|
|
std::vector<std::string> restrictions = scripting_environment.GetRestrictions();
|
|
// setup restriction parser
|
|
const RestrictionParser restriction_parser(
|
|
scripting_environment.GetProfileProperties().use_turn_restrictions,
|
|
config.parse_conditionals,
|
|
restrictions);
|
|
|
|
const ManeuverOverrideRelationParser maneuver_override_parser;
|
|
|
|
// OSM data reader
|
|
using SharedBuffer = std::shared_ptr<osmium::memory::Buffer>;
|
|
struct ParsedBuffer
|
|
{
|
|
SharedBuffer buffer;
|
|
std::vector<std::pair<const osmium::Node &, ExtractionNode>> resulting_nodes;
|
|
std::vector<std::pair<const osmium::Way &, ExtractionWay>> resulting_ways;
|
|
std::vector<std::pair<const osmium::Relation &, ExtractionRelation>> resulting_relations;
|
|
std::vector<InputTurnRestriction> resulting_restrictions;
|
|
std::vector<InputManeuverOverride> resulting_maneuver_overrides;
|
|
};
|
|
|
|
ExtractionRelationContainer relations;
|
|
|
|
const auto buffer_reader = [](osmium::io::Reader &reader) {
|
|
return tbb::filter<void, SharedBuffer>(
|
|
tbb::filter_mode::serial_in_order, [&reader](tbb::flow_control &fc) {
|
|
if (auto buffer = reader.read())
|
|
{
|
|
return std::make_shared<osmium::memory::Buffer>(std::move(buffer));
|
|
}
|
|
else
|
|
{
|
|
fc.stop();
|
|
return SharedBuffer{};
|
|
}
|
|
});
|
|
};
|
|
|
|
// Node locations cache (assumes nodes are placed before ways)
|
|
using osmium_index_type =
|
|
osmium::index::map::FlexMem<osmium::unsigned_object_id_type, osmium::Location>;
|
|
using osmium_location_handler_type = osmium::handler::NodeLocationsForWays<osmium_index_type>;
|
|
|
|
osmium_index_type location_cache;
|
|
osmium_location_handler_type location_handler(location_cache);
|
|
|
|
tbb::filter<SharedBuffer, SharedBuffer> location_cacher(
|
|
tbb::filter_mode::serial_in_order, [&location_handler](SharedBuffer buffer) {
|
|
osmium::apply(buffer->begin(), buffer->end(), location_handler);
|
|
return buffer;
|
|
});
|
|
|
|
// OSM elements Lua parser
|
|
tbb::filter<SharedBuffer, ParsedBuffer> buffer_transformer(
|
|
tbb::filter_mode::parallel,
|
|
// NOLINTNEXTLINE(performance-unnecessary-value-param)
|
|
[&](const SharedBuffer buffer) {
|
|
ParsedBuffer parsed_buffer;
|
|
parsed_buffer.buffer = buffer;
|
|
scripting_environment.ProcessElements(*buffer,
|
|
restriction_parser,
|
|
maneuver_override_parser,
|
|
relations,
|
|
parsed_buffer.resulting_nodes,
|
|
parsed_buffer.resulting_ways,
|
|
parsed_buffer.resulting_restrictions,
|
|
parsed_buffer.resulting_maneuver_overrides);
|
|
return parsed_buffer;
|
|
});
|
|
|
|
// Parsed nodes and ways handler
|
|
unsigned number_of_nodes = 0;
|
|
unsigned number_of_ways = 0;
|
|
unsigned number_of_restrictions = 0;
|
|
unsigned number_of_maneuver_overrides = 0;
|
|
tbb::filter<ParsedBuffer, void> buffer_storage(
|
|
tbb::filter_mode::serial_in_order, [&](const ParsedBuffer &parsed_buffer) {
|
|
number_of_nodes += parsed_buffer.resulting_nodes.size();
|
|
// put parsed objects thru extractor callbacks
|
|
for (const auto &result : parsed_buffer.resulting_nodes)
|
|
{
|
|
extractor_callbacks->ProcessNode(result.first, result.second);
|
|
}
|
|
number_of_ways += parsed_buffer.resulting_ways.size();
|
|
for (const auto &result : parsed_buffer.resulting_ways)
|
|
{
|
|
extractor_callbacks->ProcessWay(result.first, result.second);
|
|
}
|
|
|
|
number_of_restrictions += parsed_buffer.resulting_restrictions.size();
|
|
for (const auto &result : parsed_buffer.resulting_restrictions)
|
|
{
|
|
extractor_callbacks->ProcessRestriction(result);
|
|
}
|
|
|
|
number_of_maneuver_overrides = parsed_buffer.resulting_maneuver_overrides.size();
|
|
for (const auto &result : parsed_buffer.resulting_maneuver_overrides)
|
|
{
|
|
extractor_callbacks->ProcessManeuverOverride(result);
|
|
}
|
|
});
|
|
|
|
tbb::filter<SharedBuffer, std::shared_ptr<ExtractionRelationContainer>> buffer_relation_cache(
|
|
tbb::filter_mode::parallel,
|
|
// NOLINTNEXTLINE(performance-unnecessary-value-param)
|
|
[&](const SharedBuffer buffer) {
|
|
if (!buffer)
|
|
return std::shared_ptr<ExtractionRelationContainer>{};
|
|
|
|
auto relations = std::make_shared<ExtractionRelationContainer>();
|
|
for (auto entity = buffer->cbegin(), end = buffer->cend(); entity != end; ++entity)
|
|
{
|
|
if (entity->type() != osmium::item_type::relation)
|
|
continue;
|
|
|
|
const auto &rel = static_cast<const osmium::Relation &>(*entity);
|
|
|
|
const char *rel_type = rel.get_value_by_key("type");
|
|
if (!rel_type || !std::binary_search(relation_types.begin(),
|
|
relation_types.end(),
|
|
std::string(rel_type)))
|
|
continue;
|
|
|
|
ExtractionRelation extracted_rel({rel.id(), osmium::item_type::relation});
|
|
for (auto const &t : rel.tags())
|
|
extracted_rel.attributes.emplace_back(std::make_pair(t.key(), t.value()));
|
|
|
|
for (auto const &m : rel.members())
|
|
{
|
|
ExtractionRelation::OsmIDTyped const mid(m.ref(), m.type());
|
|
extracted_rel.AddMember(mid, m.role());
|
|
relations->AddRelationMember(extracted_rel.id, mid);
|
|
}
|
|
|
|
relations->AddRelation(std::move(extracted_rel));
|
|
};
|
|
return relations;
|
|
});
|
|
|
|
unsigned number_of_relations = 0;
|
|
tbb::filter<std::shared_ptr<ExtractionRelationContainer>, void> buffer_storage_relation(
|
|
tbb::filter_mode::serial_in_order,
|
|
// NOLINTNEXTLINE(performance-unnecessary-value-param)
|
|
[&](const std::shared_ptr<ExtractionRelationContainer> parsed_relations) {
|
|
number_of_relations += parsed_relations->GetRelationsNum();
|
|
relations.Merge(std::move(*parsed_relations));
|
|
});
|
|
|
|
// Parse OSM elements with parallel transformer
|
|
// Number of pipeline tokens that yielded the best speedup was about 1.5 * num_cores
|
|
const auto num_threads = std::thread::hardware_concurrency() * 1.5;
|
|
const auto read_meta =
|
|
config.use_metadata ? osmium::io::read_meta::yes : osmium::io::read_meta::no;
|
|
|
|
{ // Relations reading pipeline
|
|
util::Log() << "Parse relations ...";
|
|
osmium::io::Reader reader(input_file, pool, osmium::osm_entity_bits::relation, read_meta);
|
|
tbb::parallel_pipeline(
|
|
num_threads, buffer_reader(reader) & buffer_relation_cache & buffer_storage_relation);
|
|
}
|
|
|
|
{ // Nodes and ways reading pipeline
|
|
util::Log() << "Parse ways and nodes ...";
|
|
osmium::io::Reader reader(input_file,
|
|
pool,
|
|
osmium::osm_entity_bits::node | osmium::osm_entity_bits::way |
|
|
osmium::osm_entity_bits::relation,
|
|
read_meta);
|
|
|
|
const auto pipeline =
|
|
scripting_environment.HasLocationDependentData() && config.use_locations_cache
|
|
? buffer_reader(reader) & location_cacher & buffer_transformer & buffer_storage
|
|
: buffer_reader(reader) & buffer_transformer & buffer_storage;
|
|
tbb::parallel_pipeline(num_threads, pipeline);
|
|
}
|
|
|
|
TIMER_STOP(parsing);
|
|
util::Log() << "Parsing finished after " << TIMER_SEC(parsing) << " seconds";
|
|
|
|
util::Log() << "Raw input contains " << number_of_nodes << " nodes, " << number_of_ways
|
|
<< " ways, and " << number_of_relations << " relations, " << number_of_restrictions
|
|
<< " restrictions";
|
|
|
|
extractor_callbacks.reset();
|
|
|
|
if (extraction_containers.all_edges_list.empty())
|
|
{
|
|
throw util::exception(std::string("There are no edges remaining after parsing.") +
|
|
SOURCE_REF);
|
|
}
|
|
|
|
extraction_containers.PrepareData(scripting_environment,
|
|
config.GetPath(".osrm.names").string());
|
|
|
|
auto profile_properties = scripting_environment.GetProfileProperties();
|
|
SetClassNames(scripting_environment.GetClassNames(), classes_map, profile_properties);
|
|
auto excludable_classes = scripting_environment.GetExcludableClasses();
|
|
SetExcludableClasses(classes_map, excludable_classes, profile_properties);
|
|
files::writeProfileProperties(config.GetPath(".osrm.properties").string(), profile_properties);
|
|
|
|
TIMER_STOP(extracting);
|
|
util::Log() << "extraction finished after " << TIMER_SEC(extracting) << "s";
|
|
|
|
std::vector<util::Coordinate> osm_coordinates;
|
|
extractor::PackedOSMIDs osm_node_ids;
|
|
|
|
osm_coordinates.resize(extraction_containers.used_nodes.size());
|
|
osm_node_ids.reserve(extraction_containers.used_nodes.size());
|
|
for (size_t index = 0; index < extraction_containers.used_nodes.size(); ++index)
|
|
{
|
|
const auto ¤t_node = extraction_containers.used_nodes[index];
|
|
osm_coordinates[index].lon = current_node.lon;
|
|
osm_coordinates[index].lat = current_node.lat;
|
|
osm_node_ids.push_back(current_node.node_id);
|
|
}
|
|
|
|
if (config.dump_nbg_graph)
|
|
{
|
|
storage::tar::FileWriter writer(config.GetPath(".osrm.nbg").string(),
|
|
storage::tar::FileWriter::GenerateFingerprint);
|
|
storage::serialization::write(writer, "/extractor/nodes", extraction_containers.used_nodes);
|
|
storage::serialization::write(writer, "/extractor/edges", extraction_containers.used_edges);
|
|
}
|
|
|
|
return ParsedOSMData{std::move(turn_lane_map),
|
|
std::move(extraction_containers.turn_restrictions),
|
|
std::move(extraction_containers.internal_maneuver_overrides),
|
|
std::move(extraction_containers.internal_traffic_signals),
|
|
std::move(extraction_containers.used_barrier_nodes),
|
|
std::move(osm_coordinates),
|
|
std::move(osm_node_ids),
|
|
std::move(extraction_containers.used_edges),
|
|
std::move(extraction_containers.all_edges_annotation_data_list)};
|
|
}
|
|
|
|
void Extractor::FindComponents(unsigned number_of_edge_based_nodes,
|
|
const util::DeallocatingVector<EdgeBasedEdge> &input_edge_list,
|
|
const std::vector<EdgeBasedNodeSegment> &input_node_segments,
|
|
EdgeBasedNodeDataContainer &nodes_container) const
|
|
{
|
|
using InputEdge = util::static_graph_details::SortableEdgeWithData<void>;
|
|
using UncontractedGraph = util::StaticGraph<void>;
|
|
std::vector<InputEdge> edges;
|
|
edges.reserve(input_edge_list.size() * 2);
|
|
|
|
for (const auto &edge : input_edge_list)
|
|
{
|
|
BOOST_ASSERT_MSG((std::max(edge.data.weight, EdgeWeight{1})) > EdgeWeight{0},
|
|
"edge distance < 1");
|
|
BOOST_ASSERT(edge.source < number_of_edge_based_nodes);
|
|
BOOST_ASSERT(edge.target < number_of_edge_based_nodes);
|
|
if (edge.data.forward)
|
|
{
|
|
edges.push_back({edge.source, edge.target});
|
|
}
|
|
|
|
if (edge.data.backward)
|
|
{
|
|
edges.push_back({edge.target, edge.source});
|
|
}
|
|
}
|
|
|
|
// Connect forward and backward nodes of each edge to enforce
|
|
// forward and backward edge-based nodes be in one strongly-connected component
|
|
for (const auto &segment : input_node_segments)
|
|
{
|
|
if (segment.reverse_segment_id.enabled)
|
|
{
|
|
BOOST_ASSERT(segment.forward_segment_id.id < number_of_edge_based_nodes);
|
|
BOOST_ASSERT(segment.reverse_segment_id.id < number_of_edge_based_nodes);
|
|
edges.push_back({segment.forward_segment_id.id, segment.reverse_segment_id.id});
|
|
edges.push_back({segment.reverse_segment_id.id, segment.forward_segment_id.id});
|
|
}
|
|
}
|
|
|
|
tbb::parallel_sort(edges.begin(), edges.end());
|
|
edges.erase(std::unique(edges.begin(), edges.end()), edges.end());
|
|
|
|
auto uncontracted_graph = UncontractedGraph(number_of_edge_based_nodes, edges);
|
|
|
|
util::TarjanSCC<UncontractedGraph> component_search(uncontracted_graph);
|
|
component_search.Run();
|
|
|
|
for (NodeID node_id = 0; node_id < number_of_edge_based_nodes; ++node_id)
|
|
{
|
|
const auto forward_component = component_search.GetComponentID(node_id);
|
|
const auto component_size = component_search.GetComponentSize(forward_component);
|
|
const auto is_tiny = component_size < config.small_component_size;
|
|
BOOST_ASSERT(node_id < nodes_container.NumberOfNodes());
|
|
nodes_container.nodes[node_id].component_id = {1 + forward_component, is_tiny};
|
|
}
|
|
}
|
|
|
|
/**
|
|
\brief Building an edge-expanded graph from node-based input and turn restrictions
|
|
*/
|
|
|
|
EdgeID Extractor::BuildEdgeExpandedGraph(
|
|
// input data
|
|
const util::NodeBasedDynamicGraph &node_based_graph,
|
|
const std::vector<util::Coordinate> &coordinates,
|
|
const CompressedEdgeContainer &compressed_edge_container,
|
|
const std::unordered_set<NodeID> &barrier_nodes,
|
|
const TrafficSignals &traffic_signals,
|
|
const RestrictionGraph &restriction_graph,
|
|
const std::unordered_set<EdgeID> &segregated_edges,
|
|
const NameTable &name_table,
|
|
const std::vector<UnresolvedManeuverOverride> &maneuver_overrides,
|
|
const LaneDescriptionMap &turn_lane_map,
|
|
// for calculating turn penalties
|
|
ScriptingEnvironment &scripting_environment,
|
|
// output data
|
|
EdgeBasedNodeDataContainer &edge_based_nodes_container,
|
|
std::vector<EdgeBasedNodeSegment> &edge_based_node_segments,
|
|
std::vector<EdgeWeight> &edge_based_node_weights,
|
|
std::vector<EdgeDuration> &edge_based_node_durations,
|
|
std::vector<EdgeDistance> &edge_based_node_distances,
|
|
util::DeallocatingVector<EdgeBasedEdge> &edge_based_edge_list,
|
|
std::uint32_t &connectivity_checksum)
|
|
{
|
|
EdgeBasedGraphFactory edge_based_graph_factory(node_based_graph,
|
|
edge_based_nodes_container,
|
|
compressed_edge_container,
|
|
barrier_nodes,
|
|
traffic_signals,
|
|
coordinates,
|
|
name_table,
|
|
segregated_edges,
|
|
turn_lane_map);
|
|
|
|
const auto create_edge_based_edges = [&]() {
|
|
// scoped to release intermediate data structures right after the call
|
|
RestrictionMap unconditional_node_restriction_map(restriction_graph);
|
|
ConditionalRestrictionMap conditional_node_restriction_map(restriction_graph);
|
|
WayRestrictionMap via_way_restriction_map(restriction_graph);
|
|
edge_based_graph_factory.Run(scripting_environment,
|
|
config.GetPath(".osrm.turn_weight_penalties").string(),
|
|
config.GetPath(".osrm.turn_duration_penalties").string(),
|
|
config.GetPath(".osrm.turn_penalties_index").string(),
|
|
config.GetPath(".osrm.cnbg_to_ebg").string(),
|
|
config.GetPath(".osrm.restrictions").string(),
|
|
config.GetPath(".osrm.maneuver_overrides").string(),
|
|
unconditional_node_restriction_map,
|
|
conditional_node_restriction_map,
|
|
via_way_restriction_map,
|
|
maneuver_overrides);
|
|
return edge_based_graph_factory.GetNumberOfEdgeBasedNodes();
|
|
};
|
|
|
|
const auto number_of_edge_based_nodes = create_edge_based_edges();
|
|
|
|
edge_based_graph_factory.GetEdgeBasedEdges(edge_based_edge_list);
|
|
edge_based_graph_factory.GetEdgeBasedNodeSegments(edge_based_node_segments);
|
|
edge_based_graph_factory.GetEdgeBasedNodeWeights(edge_based_node_weights);
|
|
edge_based_graph_factory.GetEdgeBasedNodeDurations(edge_based_node_durations);
|
|
edge_based_graph_factory.GetEdgeBasedNodeDistances(edge_based_node_distances);
|
|
connectivity_checksum = edge_based_graph_factory.GetConnectivityChecksum();
|
|
|
|
return number_of_edge_based_nodes;
|
|
}
|
|
|
|
/**
|
|
\brief Building rtree-based nearest-neighbor data structure
|
|
|
|
Saves tree into '.ramIndex' and leaves into '.fileIndex'.
|
|
*/
|
|
void Extractor::BuildRTree(std::vector<EdgeBasedNodeSegment> edge_based_node_segments,
|
|
const std::vector<util::Coordinate> &coordinates)
|
|
{
|
|
util::Log() << "Constructing r-tree of " << edge_based_node_segments.size()
|
|
<< " segments build on-top of " << coordinates.size() << " coordinates";
|
|
|
|
// Filter node based edges based on startpoint
|
|
auto start_point_count = std::accumulate(edge_based_node_segments.begin(),
|
|
edge_based_node_segments.end(),
|
|
0,
|
|
[](const size_t so_far, const auto &segment) {
|
|
return so_far + (segment.is_startpoint ? 1 : 0);
|
|
});
|
|
if (start_point_count == 0)
|
|
{
|
|
throw util::exception("There are no snappable edges left after processing. Are you "
|
|
"setting travel modes correctly in the profile? Cannot continue." +
|
|
SOURCE_REF);
|
|
}
|
|
|
|
TIMER_START(construction);
|
|
util::StaticRTree<EdgeBasedNodeSegment> rtree(
|
|
edge_based_node_segments, coordinates, config.GetPath(".osrm.fileIndex"));
|
|
|
|
files::writeRamIndex(config.GetPath(".osrm.ramIndex"), rtree);
|
|
|
|
TIMER_STOP(construction);
|
|
util::Log() << "finished r-tree construction in " << TIMER_SEC(construction) << " seconds";
|
|
}
|
|
|
|
template <typename Map> auto convertIDMapToVector(const Map &map)
|
|
{
|
|
std::vector<typename Map::key_type> result(map.size());
|
|
for (const auto &pair : map)
|
|
{
|
|
BOOST_ASSERT(pair.second < map.size());
|
|
result[pair.second] = pair.first;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void Extractor::ProcessGuidanceTurns(
|
|
const util::NodeBasedDynamicGraph &node_based_graph,
|
|
const extractor::EdgeBasedNodeDataContainer &edge_based_node_container,
|
|
const std::vector<util::Coordinate> &node_coordinates,
|
|
const CompressedEdgeContainer &compressed_edge_container,
|
|
const std::unordered_set<NodeID> &barrier_nodes,
|
|
const RestrictionGraph &restriction_graph,
|
|
const NameTable &name_table,
|
|
LaneDescriptionMap lane_description_map,
|
|
ScriptingEnvironment &scripting_environment)
|
|
{
|
|
// Output data
|
|
osrm::guidance::TurnDataExternalContainer turn_data_container;
|
|
util::guidance::LaneDataIdMap lane_data_map;
|
|
osrm::guidance::BearingClassesVector bearing_class_by_node_based_node;
|
|
osrm::guidance::BearingClassesMap bearing_class_hash;
|
|
osrm::guidance::EntryClassesMap entry_class_hash;
|
|
std::uint32_t connectivity_checksum = 0;
|
|
|
|
TIMER_START(turn_annotations);
|
|
|
|
{
|
|
SuffixTable street_name_suffix_table(scripting_environment);
|
|
const auto &turn_lanes_data = transformTurnLaneMapIntoArrays(lane_description_map);
|
|
|
|
RestrictionMap unconditional_node_restriction_map(restriction_graph);
|
|
WayRestrictionMap way_restriction_map(restriction_graph);
|
|
|
|
osrm::guidance::annotateTurns(node_based_graph,
|
|
edge_based_node_container,
|
|
node_coordinates,
|
|
compressed_edge_container,
|
|
barrier_nodes,
|
|
unconditional_node_restriction_map,
|
|
way_restriction_map,
|
|
name_table,
|
|
street_name_suffix_table,
|
|
turn_lanes_data,
|
|
lane_description_map,
|
|
lane_data_map,
|
|
turn_data_container,
|
|
bearing_class_by_node_based_node,
|
|
bearing_class_hash,
|
|
entry_class_hash,
|
|
connectivity_checksum);
|
|
}
|
|
|
|
TIMER_STOP(turn_annotations);
|
|
util::Log() << "Guidance turn annotations took " << TIMER_SEC(turn_annotations) << "s";
|
|
|
|
util::Log() << "Writing Intersection Classification Data";
|
|
TIMER_START(write_intersections);
|
|
files::writeIntersections(
|
|
config.GetPath(".osrm.icd").string(),
|
|
IntersectionBearingsContainer{bearing_class_by_node_based_node,
|
|
convertIDMapToVector(bearing_class_hash.data)},
|
|
convertIDMapToVector(entry_class_hash.data));
|
|
TIMER_STOP(write_intersections);
|
|
util::Log() << "ok, after " << TIMER_SEC(write_intersections) << "s";
|
|
|
|
util::Log() << "Writing Turns and Lane Data...";
|
|
TIMER_START(write_guidance_data);
|
|
|
|
{
|
|
auto turn_lane_data = convertIDMapToVector(lane_data_map.data);
|
|
files::writeTurnLaneData(config.GetPath(".osrm.tld"), turn_lane_data);
|
|
}
|
|
|
|
{ // Turn lanes handler modifies lane_description_map, so another transformation is needed
|
|
std::vector<std::uint32_t> turn_lane_offsets;
|
|
std::vector<TurnLaneType::Mask> turn_lane_masks;
|
|
std::tie(turn_lane_offsets, turn_lane_masks) =
|
|
transformTurnLaneMapIntoArrays(lane_description_map);
|
|
files::writeTurnLaneDescriptions(
|
|
config.GetPath(".osrm.tls"), turn_lane_offsets, turn_lane_masks);
|
|
}
|
|
|
|
osrm::guidance::files::writeTurnData(
|
|
config.GetPath(".osrm.edges").string(), turn_data_container, connectivity_checksum);
|
|
TIMER_STOP(write_guidance_data);
|
|
util::Log() << "ok, after " << TIMER_SEC(write_guidance_data) << "s";
|
|
}
|
|
|
|
} // namespace osrm::extractor
|