2016-03-01 16:30:31 -05:00
|
|
|
#ifndef OSRM_GUIDANCE_TOOLKIT_HPP_
|
|
|
|
#define OSRM_GUIDANCE_TOOLKIT_HPP_
|
2016-02-24 04:29:23 -05:00
|
|
|
|
2016-08-02 09:43:29 -04:00
|
|
|
#include "util/attributes.hpp"
|
2016-02-24 08:03:51 -05:00
|
|
|
#include "util/bearing.hpp"
|
2016-02-24 04:29:23 -05:00
|
|
|
#include "util/coordinate.hpp"
|
|
|
|
#include "util/coordinate_calculation.hpp"
|
2016-04-18 07:41:19 -04:00
|
|
|
#include "util/guidance/toolkit.hpp"
|
2016-06-15 08:38:24 -04:00
|
|
|
#include "util/guidance/turn_lanes.hpp"
|
|
|
|
#include "util/typedefs.hpp"
|
2016-02-24 04:29:23 -05:00
|
|
|
|
|
|
|
#include "extractor/compressed_edge_container.hpp"
|
|
|
|
#include "extractor/query_node.hpp"
|
|
|
|
|
2016-03-23 08:04:23 -04:00
|
|
|
#include "extractor/guidance/discrete_angle.hpp"
|
2016-04-11 06:51:06 -04:00
|
|
|
#include "extractor/guidance/intersection.hpp"
|
2016-03-01 16:30:31 -05:00
|
|
|
#include "extractor/guidance/turn_instruction.hpp"
|
2016-02-24 04:29:23 -05:00
|
|
|
|
2016-03-16 10:47:33 -04:00
|
|
|
#include <algorithm>
|
2016-02-24 04:29:23 -05:00
|
|
|
#include <cmath>
|
2016-03-03 11:24:59 -05:00
|
|
|
#include <cstdint>
|
2016-03-23 08:04:23 -04:00
|
|
|
#include <map>
|
2016-03-16 10:47:33 -04:00
|
|
|
#include <string>
|
2016-06-15 08:38:24 -04:00
|
|
|
#include <unordered_map>
|
2016-04-28 04:54:14 -04:00
|
|
|
#include <utility>
|
2016-02-24 04:29:23 -05:00
|
|
|
|
2016-04-22 05:31:46 -04:00
|
|
|
#include <boost/algorithm/string.hpp>
|
2016-06-15 08:38:24 -04:00
|
|
|
#include <boost/functional/hash.hpp>
|
2016-07-08 04:44:49 -04:00
|
|
|
#include <boost/tokenizer.hpp>
|
2016-04-21 07:59:03 -04:00
|
|
|
|
2016-02-24 04:29:23 -05:00
|
|
|
namespace osrm
|
|
|
|
{
|
2016-03-01 16:30:31 -05:00
|
|
|
namespace extractor
|
2016-02-24 04:29:23 -05:00
|
|
|
{
|
|
|
|
namespace guidance
|
|
|
|
{
|
|
|
|
|
2016-06-15 08:38:24 -04:00
|
|
|
using util::guidance::LaneTupelIdPair;
|
|
|
|
using LaneDataIdMap = std::unordered_map<LaneTupelIdPair, LaneDataID, boost::hash<LaneTupelIdPair>>;
|
|
|
|
|
2016-04-18 07:41:19 -04:00
|
|
|
using util::guidance::angularDeviation;
|
2016-07-01 05:34:44 -04:00
|
|
|
using util::guidance::entersRoundabout;
|
|
|
|
using util::guidance::leavesRoundabout;
|
2016-04-18 07:41:19 -04:00
|
|
|
|
2016-02-24 04:29:23 -05:00
|
|
|
namespace detail
|
|
|
|
{
|
|
|
|
const constexpr double DESIRED_SEGMENT_LENGTH = 10.0;
|
|
|
|
|
|
|
|
template <typename IteratorType>
|
|
|
|
util::Coordinate
|
|
|
|
getCoordinateFromCompressedRange(util::Coordinate current_coordinate,
|
2016-03-03 11:24:59 -05:00
|
|
|
const IteratorType compressed_geometry_begin,
|
2016-02-24 04:29:23 -05:00
|
|
|
const IteratorType compressed_geometry_end,
|
|
|
|
const util::Coordinate final_coordinate,
|
|
|
|
const std::vector<extractor::QueryNode> &query_nodes)
|
|
|
|
{
|
2016-03-23 08:04:23 -04:00
|
|
|
const auto extractCoordinateFromNode =
|
|
|
|
[](const extractor::QueryNode &node) -> util::Coordinate {
|
2016-02-24 04:29:23 -05:00
|
|
|
return {node.lon, node.lat};
|
|
|
|
};
|
|
|
|
double distance_to_current_coordinate = 0;
|
|
|
|
double distance_to_next_coordinate = 0;
|
|
|
|
|
|
|
|
// get the length that is missing from the current segment to reach DESIRED_SEGMENT_LENGTH
|
2016-03-23 08:04:23 -04:00
|
|
|
const auto getFactor = [](const double first_distance, const double second_distance) {
|
2016-02-24 04:29:23 -05:00
|
|
|
BOOST_ASSERT(first_distance < detail::DESIRED_SEGMENT_LENGTH);
|
|
|
|
double segment_length = second_distance - first_distance;
|
|
|
|
BOOST_ASSERT(segment_length > 0);
|
|
|
|
BOOST_ASSERT(second_distance >= detail::DESIRED_SEGMENT_LENGTH);
|
|
|
|
double missing_distance = detail::DESIRED_SEGMENT_LENGTH - first_distance;
|
2016-04-10 20:23:22 -04:00
|
|
|
return std::max(0., std::min(missing_distance / segment_length, 1.0));
|
2016-02-24 04:29:23 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
for (auto compressed_geometry_itr = compressed_geometry_begin;
|
2016-05-27 15:05:04 -04:00
|
|
|
compressed_geometry_itr != compressed_geometry_end;
|
|
|
|
++compressed_geometry_itr)
|
2016-02-24 04:29:23 -05:00
|
|
|
{
|
|
|
|
const auto next_coordinate =
|
|
|
|
extractCoordinateFromNode(query_nodes[compressed_geometry_itr->node_id]);
|
|
|
|
distance_to_next_coordinate =
|
|
|
|
distance_to_current_coordinate +
|
|
|
|
util::coordinate_calculation::haversineDistance(current_coordinate, next_coordinate);
|
|
|
|
|
|
|
|
// reached point where coordinates switch between
|
|
|
|
if (distance_to_next_coordinate >= detail::DESIRED_SEGMENT_LENGTH)
|
|
|
|
return util::coordinate_calculation::interpolateLinear(
|
|
|
|
getFactor(distance_to_current_coordinate, distance_to_next_coordinate),
|
2016-05-27 15:05:04 -04:00
|
|
|
current_coordinate,
|
|
|
|
next_coordinate);
|
2016-02-24 04:29:23 -05:00
|
|
|
|
|
|
|
// prepare for next iteration
|
|
|
|
current_coordinate = next_coordinate;
|
|
|
|
distance_to_current_coordinate = distance_to_next_coordinate;
|
|
|
|
}
|
|
|
|
|
|
|
|
distance_to_next_coordinate =
|
|
|
|
distance_to_current_coordinate +
|
|
|
|
util::coordinate_calculation::haversineDistance(current_coordinate, final_coordinate);
|
|
|
|
|
|
|
|
// reached point where coordinates switch between
|
2016-04-10 20:23:22 -04:00
|
|
|
if (distance_to_current_coordinate < detail::DESIRED_SEGMENT_LENGTH &&
|
|
|
|
distance_to_next_coordinate >= detail::DESIRED_SEGMENT_LENGTH)
|
2016-02-24 04:29:23 -05:00
|
|
|
return util::coordinate_calculation::interpolateLinear(
|
|
|
|
getFactor(distance_to_current_coordinate, distance_to_next_coordinate),
|
2016-05-27 15:05:04 -04:00
|
|
|
current_coordinate,
|
|
|
|
final_coordinate);
|
2016-02-24 04:29:23 -05:00
|
|
|
else
|
|
|
|
return final_coordinate;
|
|
|
|
}
|
|
|
|
} // namespace detail
|
|
|
|
|
2016-04-29 03:48:13 -04:00
|
|
|
// Finds a (potentially interpolated) coordinate that is DESIRED_SEGMENT_LENGTH away
|
2016-02-24 04:29:23 -05:00
|
|
|
// from the start of an edge
|
|
|
|
inline util::Coordinate
|
|
|
|
getRepresentativeCoordinate(const NodeID from_node,
|
|
|
|
const NodeID to_node,
|
|
|
|
const EdgeID via_edge_id,
|
|
|
|
const bool traverse_in_reverse,
|
|
|
|
const extractor::CompressedEdgeContainer &compressed_geometries,
|
|
|
|
const std::vector<extractor::QueryNode> &query_nodes)
|
|
|
|
{
|
2016-03-23 08:04:23 -04:00
|
|
|
const auto extractCoordinateFromNode =
|
|
|
|
[](const extractor::QueryNode &node) -> util::Coordinate {
|
2016-02-24 04:29:23 -05:00
|
|
|
return {node.lon, node.lat};
|
|
|
|
};
|
|
|
|
|
|
|
|
// Uncompressed roads are simple, return the coordinate at the end
|
|
|
|
if (!compressed_geometries.HasEntryForID(via_edge_id))
|
|
|
|
{
|
|
|
|
return extractCoordinateFromNode(traverse_in_reverse ? query_nodes[from_node]
|
|
|
|
: query_nodes[to_node]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const auto &geometry = compressed_geometries.GetBucketReference(via_edge_id);
|
|
|
|
|
|
|
|
const auto base_node_id = (traverse_in_reverse) ? to_node : from_node;
|
|
|
|
const auto base_coordinate = extractCoordinateFromNode(query_nodes[base_node_id]);
|
|
|
|
|
|
|
|
const auto final_node = (traverse_in_reverse) ? from_node : to_node;
|
|
|
|
const auto final_coordinate = extractCoordinateFromNode(query_nodes[final_node]);
|
|
|
|
|
|
|
|
if (traverse_in_reverse)
|
|
|
|
return detail::getCoordinateFromCompressedRange(
|
|
|
|
base_coordinate, geometry.rbegin(), geometry.rend(), final_coordinate, query_nodes);
|
|
|
|
else
|
|
|
|
return detail::getCoordinateFromCompressedRange(
|
|
|
|
base_coordinate, geometry.begin(), geometry.end(), final_coordinate, query_nodes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-11 06:51:06 -04:00
|
|
|
// To simplify handling of Left/Right hand turns, we can mirror turns and write an intersection
|
|
|
|
// handler only for one side. The mirror function turns a left-hand turn in a equivalent right-hand
|
|
|
|
// turn and vice versa.
|
2016-08-02 09:43:29 -04:00
|
|
|
OSRM_ATTR_WARN_UNUSED
|
2016-04-11 06:51:06 -04:00
|
|
|
inline ConnectedRoad mirror(ConnectedRoad road)
|
|
|
|
{
|
2016-05-27 15:05:04 -04:00
|
|
|
const constexpr DirectionModifier::Enum mirrored_modifiers[] = {DirectionModifier::UTurn,
|
|
|
|
DirectionModifier::SharpLeft,
|
|
|
|
DirectionModifier::Left,
|
|
|
|
DirectionModifier::SlightLeft,
|
|
|
|
DirectionModifier::Straight,
|
|
|
|
DirectionModifier::SlightRight,
|
|
|
|
DirectionModifier::Right,
|
|
|
|
DirectionModifier::SharpRight};
|
2016-04-11 06:51:06 -04:00
|
|
|
|
|
|
|
if (angularDeviation(road.turn.angle, 0) > std::numeric_limits<double>::epsilon())
|
|
|
|
{
|
|
|
|
road.turn.angle = 360 - road.turn.angle;
|
|
|
|
road.turn.instruction.direction_modifier =
|
|
|
|
mirrored_modifiers[road.turn.instruction.direction_modifier];
|
|
|
|
}
|
|
|
|
return road;
|
|
|
|
}
|
|
|
|
|
2016-06-21 08:37:20 -04:00
|
|
|
inline bool hasRoundaboutType(const TurnInstruction instruction)
|
|
|
|
{
|
|
|
|
using namespace extractor::guidance::TurnType;
|
|
|
|
const constexpr TurnType::Enum valid_types[] = {TurnType::EnterRoundabout,
|
|
|
|
TurnType::EnterAndExitRoundabout,
|
|
|
|
TurnType::EnterRotary,
|
|
|
|
TurnType::EnterAndExitRotary,
|
|
|
|
TurnType::EnterRoundaboutIntersection,
|
|
|
|
TurnType::EnterAndExitRoundaboutIntersection,
|
|
|
|
TurnType::EnterRoundaboutAtExit,
|
|
|
|
TurnType::ExitRoundabout,
|
|
|
|
TurnType::EnterRotaryAtExit,
|
|
|
|
TurnType::ExitRotary,
|
|
|
|
TurnType::EnterRoundaboutIntersectionAtExit,
|
|
|
|
TurnType::ExitRoundaboutIntersection,
|
|
|
|
TurnType::StayOnRoundabout};
|
2016-07-08 06:39:29 -04:00
|
|
|
|
|
|
|
const auto *first = valid_types;
|
|
|
|
const auto *last = first + sizeof(valid_types) / sizeof(valid_types[0]);
|
|
|
|
|
|
|
|
return std::find(first, last, instruction.type) != last;
|
2016-06-22 07:20:55 -04:00
|
|
|
}
|
2016-06-21 08:37:20 -04:00
|
|
|
|
2016-05-13 13:18:00 -04:00
|
|
|
// Public service vehicle lanes and similar can introduce additional lanes into the lane string that
|
|
|
|
// are not specifically marked for left/right turns. This function can be used from the profile to
|
|
|
|
// trim the lane string appropriately
|
|
|
|
//
|
|
|
|
// left|throught|
|
|
|
|
// in combination with lanes:psv:forward=1
|
|
|
|
// will be corrected to left|throught, since the final lane is not drivable.
|
|
|
|
// This is in contrast to a situation with lanes:psv:forward=0 (or not set) where left|through|
|
|
|
|
// represents left|through|through
|
2016-08-02 09:43:29 -04:00
|
|
|
OSRM_ATTR_WARN_UNUSED
|
2016-05-13 13:18:00 -04:00
|
|
|
inline std::string
|
|
|
|
trimLaneString(std::string lane_string, std::int32_t count_left, std::int32_t count_right)
|
|
|
|
{
|
|
|
|
if (count_left)
|
|
|
|
{
|
|
|
|
bool sane = count_left < static_cast<std::int32_t>(lane_string.size());
|
|
|
|
for (std::int32_t i = 0; i < count_left; ++i)
|
|
|
|
// this is adjusted for our fake pipe. The moment cucumber can handle multiple escaped
|
|
|
|
// pipes, the '&' part can be removed
|
2016-08-12 11:04:40 -04:00
|
|
|
if (lane_string[i] != '|')
|
2016-05-13 13:18:00 -04:00
|
|
|
{
|
|
|
|
sane = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sane)
|
|
|
|
{
|
|
|
|
lane_string.erase(lane_string.begin(), lane_string.begin() + count_left);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (count_right)
|
|
|
|
{
|
|
|
|
bool sane = count_right < static_cast<std::int32_t>(lane_string.size());
|
|
|
|
for (auto itr = lane_string.rbegin();
|
|
|
|
itr != lane_string.rend() && itr != lane_string.rbegin() + count_right;
|
|
|
|
++itr)
|
|
|
|
{
|
2016-08-12 11:04:40 -04:00
|
|
|
if (*itr != '|')
|
2016-05-13 13:18:00 -04:00
|
|
|
{
|
|
|
|
sane = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (sane)
|
|
|
|
lane_string.resize(lane_string.size() - count_right);
|
|
|
|
}
|
|
|
|
return lane_string;
|
|
|
|
}
|
|
|
|
|
2016-07-08 04:44:49 -04:00
|
|
|
// https://github.com/Project-OSRM/osrm-backend/issues/2638
|
|
|
|
// It can happen that some lanes are not drivable by car. Here we handle this tagging scheme
|
|
|
|
// (vehicle:lanes) to filter out not-allowed roads
|
|
|
|
// lanes=3
|
|
|
|
// turn:lanes=left|through|through|right
|
|
|
|
// vehicle:lanes=yes|yes|no|yes
|
|
|
|
// bicycle:lanes=yes|no|designated|yes
|
2016-08-02 09:43:29 -04:00
|
|
|
OSRM_ATTR_WARN_UNUSED
|
2016-07-26 05:23:14 -04:00
|
|
|
inline std::string applyAccessTokens(std::string lane_string, const std::string &access_tokens)
|
2016-07-08 04:44:49 -04:00
|
|
|
{
|
|
|
|
typedef boost::tokenizer<boost::char_separator<char>> tokenizer;
|
|
|
|
boost::char_separator<char> sep("|", "", boost::keep_empty_tokens);
|
|
|
|
tokenizer tokens(lane_string, sep);
|
|
|
|
tokenizer access(access_tokens, sep);
|
|
|
|
|
|
|
|
// strings don't match, don't do anything
|
|
|
|
if (std::distance(std::begin(tokens), std::end(tokens)) !=
|
|
|
|
std::distance(std::begin(access), std::end(access)))
|
|
|
|
return lane_string;
|
|
|
|
|
|
|
|
std::string result_string = "";
|
|
|
|
const static std::string yes = "yes";
|
|
|
|
|
|
|
|
for (auto token_itr = std::begin(tokens), access_itr = std::begin(access);
|
|
|
|
token_itr != std::end(tokens);
|
|
|
|
++token_itr, ++access_itr)
|
|
|
|
{
|
|
|
|
if (*access_itr == yes)
|
|
|
|
{
|
|
|
|
// we have to add this in front, because the next token could be invalid. Doing this on
|
|
|
|
// non-empty strings makes sure that the token string will be valid in the end
|
|
|
|
if (!result_string.empty())
|
|
|
|
result_string += '|';
|
|
|
|
|
|
|
|
result_string += *token_itr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result_string;
|
|
|
|
}
|
|
|
|
|
2016-02-24 04:29:23 -05:00
|
|
|
} // namespace guidance
|
2016-03-01 16:30:31 -05:00
|
|
|
} // namespace extractor
|
2016-02-24 04:29:23 -05:00
|
|
|
} // namespace osrm
|
|
|
|
|
2016-03-01 16:30:31 -05:00
|
|
|
#endif // OSRM_GUIDANCE_TOOLKIT_HPP_
|