forked from organicmaps/organicmaps-tmp
[OpenLR] Add PathsConnector.
This commit is contained in:
parent
9092296d88
commit
c4320eb43e
8 changed files with 285 additions and 33 deletions
|
@ -18,6 +18,8 @@ set(
|
|||
openlr_model.hpp
|
||||
openlr_model_xml.cpp
|
||||
openlr_model_xml.hpp
|
||||
paths_connector.cpp
|
||||
paths_connector.hpp
|
||||
road_info_getter.cpp
|
||||
road_info_getter.hpp
|
||||
road_type_checkers.cpp
|
||||
|
|
|
@ -169,8 +169,6 @@ bool CandidatePathsGetter::GetLineCandidatesForPoints(
|
|||
void CandidatePathsGetter::GetStartLines(vector<m2::PointD> const & points, bool const isLastPoint,
|
||||
Graph::EdgeVector & edges)
|
||||
{
|
||||
ScopedTimer t(m_stats.m_startLinesTime);
|
||||
|
||||
for (auto const & pc : points)
|
||||
{
|
||||
if (!isLastPoint)
|
||||
|
@ -188,8 +186,6 @@ void CandidatePathsGetter::GetAllSuitablePaths(Graph::EdgeVector const & startLi
|
|||
FunctionalRoadClass const frc,
|
||||
vector<LinkPtr> & allPaths)
|
||||
{
|
||||
ScopedTimer t(m_stats.m_allStatPathsTime);
|
||||
|
||||
queue<LinkPtr> q;
|
||||
|
||||
for (auto const & e : startLines)
|
||||
|
@ -246,8 +242,6 @@ void CandidatePathsGetter::GetBestCandidatePaths(
|
|||
vector<LinkPtr> const & allPaths, bool const isLastPoint, uint32_t const requiredBearing,
|
||||
uint32_t const bearDistM, m2::PointD const & startPoint, vector<Graph::EdgeVector> & candidates)
|
||||
{
|
||||
ScopedTimer t(m_stats.m_bestCandidatePathsTime);
|
||||
|
||||
set<CandidatePath> candidatePaths;
|
||||
set<CandidatePath> fakeEndingsCandidatePaths;
|
||||
|
||||
|
@ -330,8 +324,6 @@ void CandidatePathsGetter::GetLineCandidates(openlr::LocationReferencePoint cons
|
|||
vector<m2::PointD> const & pointCandidates,
|
||||
vector<Graph::EdgeVector> & candidates)
|
||||
{
|
||||
ScopedTimer t(m_stats.m_lineCandidatesTime);
|
||||
|
||||
uint32_t const kDefaultBearDistM = 25;
|
||||
uint32_t const bearDistM = min(kDefaultBearDistM, distanceToNextPoint);
|
||||
|
||||
|
|
|
@ -13,8 +13,6 @@ namespace openlr
|
|||
void CandidatePointsGetter::GetJunctionPointCandidates(m2::PointD const & p,
|
||||
vector<m2::PointD> & candidates)
|
||||
{
|
||||
ScopedTimer t(m_stat.m_pointCadtidatesTime);
|
||||
|
||||
// TODO(mgsergio): Get optimal value using experiments on a sample.
|
||||
// Or start with small radius and scale it up when there are too few points.
|
||||
size_t const kRectSideMeters = 110;
|
||||
|
|
|
@ -6,8 +6,6 @@
|
|||
|
||||
#include "geometry/mercator.hpp"
|
||||
|
||||
#include "base/timer.hpp"
|
||||
|
||||
#include <sstream>
|
||||
#include <string>
|
||||
#include <type_traits>
|
||||
|
|
|
@ -5,8 +5,6 @@
|
|||
|
||||
#include "geometry/point2d.hpp"
|
||||
|
||||
#include "base/timer.hpp"
|
||||
|
||||
#include <type_traits>
|
||||
|
||||
namespace openlr
|
||||
|
@ -32,18 +30,6 @@ typename std::common_type<T, U>::type AbsDifference(T const a, U const b)
|
|||
return a >= b ? a - b : b - a;
|
||||
}
|
||||
|
||||
// TODO(mgsergio): Remove when/if unused.
|
||||
class ScopedTimer : private my::Timer
|
||||
{
|
||||
public:
|
||||
ScopedTimer(std::chrono::milliseconds & ms) : m_ms(ms) {}
|
||||
~ScopedTimer() { m_ms += TimeElapsedAs<std::chrono::milliseconds>(); }
|
||||
|
||||
private:
|
||||
std::chrono::milliseconds & m_ms;
|
||||
};
|
||||
|
||||
bool PassesRestriction(Graph::Edge const & e, FunctionalRoadClass const restriction,
|
||||
int const frcThreshold, RoadInfoGetter & infoGetter);
|
||||
|
||||
} // namespace openlr
|
||||
|
|
247
openlr/paths_connector.cpp
Normal file
247
openlr/paths_connector.cpp
Normal file
|
@ -0,0 +1,247 @@
|
|||
#include "openlr/paths_connector.hpp"
|
||||
#include "openlr/helpers.hpp"
|
||||
|
||||
#include "base/checked_cast.hpp"
|
||||
#include "base/stl_iterator.hpp"
|
||||
|
||||
#include <algorithm>
|
||||
#include <map>
|
||||
#include <numeric>
|
||||
#include <queue>
|
||||
#include <tuple>
|
||||
|
||||
using namespace std;
|
||||
|
||||
namespace openlr
|
||||
{
|
||||
namespace
|
||||
{
|
||||
size_t IntersectionLen(Graph::EdgeVector a, Graph::EdgeVector b)
|
||||
{
|
||||
sort(begin(a), end(a));
|
||||
sort(begin(b), end(b));
|
||||
return set_intersection(begin(a), end(a), begin(b), end(b), CounterIterator()).GetCount();
|
||||
}
|
||||
|
||||
bool PrefEqualsSuff(Graph::EdgeVector const & a, Graph::EdgeVector const & b, size_t const len)
|
||||
{
|
||||
ASSERT_LESS_OR_EQUAL(len, a.size(), ());
|
||||
ASSERT_LESS_OR_EQUAL(len, b.size(), ());
|
||||
return equal(end(a) - len, end(a), begin(b));
|
||||
}
|
||||
|
||||
// Returns a length of the longest suffix of |a| that matches any prefix of |b|.
|
||||
// Neither |a| nor |b| can contain several repetitions of any edge.
|
||||
// Returns -1 if |a| intersection |b| is not equal to some suffix of |a| and some prefix of |b|.
|
||||
int32_t PathOverlappingLen(Graph::EdgeVector const & a, Graph::EdgeVector const & b)
|
||||
{
|
||||
auto const len = IntersectionLen(a, b);
|
||||
if (PrefEqualsSuff(a, b, len))
|
||||
return base::checked_cast<int32_t>(len);
|
||||
return -1;
|
||||
}
|
||||
|
||||
bool ValidatePath(Graph::EdgeVector const & path,
|
||||
uint32_t const distanceToNextPoint,
|
||||
double const pathLengthTolerance)
|
||||
{
|
||||
|
||||
double pathLen = 0.0;
|
||||
for (auto const & e : path)
|
||||
pathLen += EdgeLength(e);
|
||||
|
||||
double pathDiffPercent = AbsDifference(static_cast<double>(distanceToNextPoint), pathLen) /
|
||||
static_cast<double>(distanceToNextPoint);
|
||||
|
||||
LOG(LDEBUG, ("Validating path:", LogAs2GisPath(path)));
|
||||
|
||||
if (pathDiffPercent > pathLengthTolerance)
|
||||
{
|
||||
LOG(LDEBUG,
|
||||
("Shortest path doest not meet required length constraints, error:", pathDiffPercent));
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
} // namespace
|
||||
|
||||
PathsConnector::PathsConnector(double const pathLengthTolerance, Graph const & graph,
|
||||
v2::Stats & stat)
|
||||
: m_pathLengthTolerance(pathLengthTolerance), m_graph(graph), m_stat(stat)
|
||||
{
|
||||
}
|
||||
|
||||
bool PathsConnector::ConnectCandidates(vector<LocationReferencePoint> const & points,
|
||||
vector<vector<Graph::EdgeVector>> const & lineCandidates,
|
||||
vector<Graph::EdgeVector> & resultPath)
|
||||
{
|
||||
ASSERT(!points.empty(), ());
|
||||
|
||||
resultPath.resize(points.size() - 1);
|
||||
|
||||
// TODO(mgsergio): Discard last point on failure.
|
||||
// TODO(mgsergio): Do not iterate more than kMaxRetries times.
|
||||
// TODO(mgserigio): Make kMaxRetries depend on points number in the segment.
|
||||
for (size_t i = 1; i < points.size(); ++i)
|
||||
{
|
||||
bool found = false;
|
||||
|
||||
auto const & point = points[i - 1];
|
||||
auto const distanceToNextPoint = point.m_distanceToNextPoint;
|
||||
auto const & fromCandidates = lineCandidates[i - 1];
|
||||
auto const & toCandidates = lineCandidates[i];
|
||||
auto & resultPathPart = resultPath[i - 1];
|
||||
|
||||
Graph::EdgeVector fakePath;
|
||||
|
||||
for (size_t fromInd = 0; fromInd < fromCandidates.size() && !found; ++fromInd)
|
||||
{
|
||||
for (size_t toInd = 0; toInd < toCandidates.size() && !found; ++toInd)
|
||||
{
|
||||
resultPathPart.clear();
|
||||
|
||||
found = ConnectAdjacentCandidateLines(fromCandidates[fromInd], toCandidates[toInd],
|
||||
point.m_functionalRoadClass, distanceToNextPoint,
|
||||
resultPathPart);
|
||||
|
||||
if (!found)
|
||||
continue;
|
||||
|
||||
found = ValidatePath(resultPathPart, distanceToNextPoint, m_pathLengthTolerance);
|
||||
if (fakePath.empty() && found &&
|
||||
(resultPathPart.front().IsFake() || resultPathPart.back().IsFake()))
|
||||
{
|
||||
fakePath = resultPathPart;
|
||||
found = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (!fakePath.empty() && !found)
|
||||
{
|
||||
found = true;
|
||||
resultPathPart = fakePath;
|
||||
}
|
||||
|
||||
if (!found)
|
||||
{
|
||||
LOG(LDEBUG, ("No shortest path found"));
|
||||
++m_stat.m_noShortestPathFound;
|
||||
resultPathPart.clear();
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
ASSERT_EQUAL(resultPath.size(), points.size() - 1, ());
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool PathsConnector::FindShortestPath(Graph::Edge const & from, Graph::Edge const & to,
|
||||
FunctionalRoadClass const frc, uint32_t const maxPathLength,
|
||||
Graph::EdgeVector & path)
|
||||
{
|
||||
// TODO(mgsergio): Turn Dijkstra to A*.
|
||||
|
||||
uint32_t const kLengthToleranceM = 10;
|
||||
|
||||
struct State
|
||||
{
|
||||
State(Graph::Edge const & e, uint32_t const s) : m_edge(e), m_score(s) {}
|
||||
|
||||
bool operator>(State const & o) const
|
||||
{
|
||||
return make_tuple(m_score, m_edge) > make_tuple(o.m_score, o.m_edge);
|
||||
}
|
||||
|
||||
Graph::Edge m_edge;
|
||||
uint32_t m_score;
|
||||
};
|
||||
|
||||
priority_queue<State, vector<State>, greater<State>> q;
|
||||
map<Graph::Edge, uint32_t> scores;
|
||||
map<Graph::Edge, Graph::Edge> links;
|
||||
|
||||
q.emplace(from, 0);
|
||||
scores[from] = 0;
|
||||
|
||||
while (!q.empty())
|
||||
{
|
||||
auto const state = q.top();
|
||||
q.pop();
|
||||
|
||||
auto const & u = state.m_edge;
|
||||
// TODO(mgsergio): Unify names: use either score or distance.
|
||||
auto const us = state.m_score;
|
||||
|
||||
if (us > maxPathLength + kLengthToleranceM)
|
||||
continue;
|
||||
|
||||
if (us > scores[u])
|
||||
continue;
|
||||
|
||||
if (u == to)
|
||||
{
|
||||
for (auto e = u; e != from; e = links[e])
|
||||
path.push_back(e);
|
||||
path.push_back(from);
|
||||
reverse(begin(path), end(path));
|
||||
return true;
|
||||
}
|
||||
|
||||
Graph::EdgeVector edges;
|
||||
m_graph.GetOutgoingEdges(u.GetEndJunction(), edges);
|
||||
for (auto const & e : edges)
|
||||
{
|
||||
// TODO(mgsergio): Use frc to filter edges.
|
||||
|
||||
auto const it = scores.find(e);
|
||||
auto const eScore = us + EdgeLength(e);
|
||||
if (it == end(scores) || it->second > eScore)
|
||||
{
|
||||
scores[e] = eScore;
|
||||
links[e] = u;
|
||||
q.emplace(e, eScore);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
bool PathsConnector::ConnectAdjacentCandidateLines(Graph::EdgeVector const & from,
|
||||
Graph::EdgeVector const & to,
|
||||
FunctionalRoadClass const frc,
|
||||
uint32_t const distanceToNextPoint,
|
||||
Graph::EdgeVector & resultPath)
|
||||
|
||||
{
|
||||
ASSERT(!to.empty(), ());
|
||||
|
||||
if (auto const skip = PathOverlappingLen(from, to))
|
||||
{
|
||||
if (skip == -1)
|
||||
return false;
|
||||
copy(begin(from), end(from), back_inserter(resultPath));
|
||||
copy(begin(to) + skip, end(to), back_inserter(resultPath));
|
||||
return true;
|
||||
}
|
||||
|
||||
ASSERT(from.back() != to.front(), ());
|
||||
|
||||
Graph::EdgeVector shortestPath;
|
||||
auto const found =
|
||||
FindShortestPath(from.back(), to.front(), frc, distanceToNextPoint, shortestPath);
|
||||
if (!found)
|
||||
return false;
|
||||
|
||||
// Skip the last edge from |from| because it already took its place at begin(shortestPath).
|
||||
copy(begin(from), prev(end(from)), back_inserter(resultPath));
|
||||
copy(begin(shortestPath), end(shortestPath), back_inserter(resultPath));
|
||||
// Skip the first edge from |to| because it already took its place at prev(end(shortestPath)).
|
||||
copy(next(begin(to)), end(to), back_inserter(resultPath));
|
||||
|
||||
return found;
|
||||
}
|
||||
} // namespace openlr
|
36
openlr/paths_connector.hpp
Normal file
36
openlr/paths_connector.hpp
Normal file
|
@ -0,0 +1,36 @@
|
|||
#pragma once
|
||||
|
||||
#include "openlr/graph.hpp"
|
||||
#include "openlr/openlr_model.hpp"
|
||||
#include "openlr/stats.hpp"
|
||||
|
||||
#include <cstddef>
|
||||
#include <cstdint>
|
||||
#include <vector>
|
||||
|
||||
namespace openlr
|
||||
{
|
||||
class PathsConnector
|
||||
{
|
||||
public:
|
||||
PathsConnector(double const pathLengthTolerance, Graph const & graph, v2::Stats & stat);
|
||||
|
||||
bool ConnectCandidates(std::vector<LocationReferencePoint> const & points,
|
||||
std::vector<std::vector<Graph::EdgeVector>> const & lineCandidates,
|
||||
std::vector<Graph::EdgeVector> & resultPath);
|
||||
|
||||
private:
|
||||
bool FindShortestPath(Graph::Edge const & from, Graph::Edge const & to,
|
||||
FunctionalRoadClass const frc, uint32_t const maxPathLength,
|
||||
Graph::EdgeVector & path);
|
||||
|
||||
bool ConnectAdjacentCandidateLines(Graph::EdgeVector const & from, Graph::EdgeVector const & to,
|
||||
FunctionalRoadClass const frc,
|
||||
uint32_t const distanceToNextPoint,
|
||||
Graph::EdgeVector & resultPath);
|
||||
|
||||
double const m_pathLengthTolerance;
|
||||
Graph const & m_graph;
|
||||
v2::Stats & m_stat;
|
||||
};
|
||||
} // namespace openlr
|
|
@ -16,13 +16,6 @@ struct Stats
|
|||
uint32_t m_wrongOffsets = 0;
|
||||
// Number of zeroed distance-to-next point values in the input.
|
||||
uint32_t m_dnpIsZero = 0;
|
||||
|
||||
std::chrono::milliseconds m_pointCadtidatesTime{0};
|
||||
std::chrono::milliseconds m_startLinesTime{0};
|
||||
std::chrono::milliseconds m_lineCandidatesTime{0};
|
||||
std::chrono::milliseconds m_allStatPathsTime{0};
|
||||
std::chrono::milliseconds m_bestCandidatePathsTime{0};
|
||||
std::chrono::milliseconds m_shortestPathsTime{0};
|
||||
};
|
||||
} // namespace V2
|
||||
} // namespace openlr
|
||||
|
|
Loading…
Add table
Reference in a new issue