OSRM router refactoring

This commit is contained in:
Lev Dragunov 2014-12-25 17:24:49 +03:00 committed by Alex Zolotarev
parent cbd4aee3e2
commit f1cd832523
2 changed files with 297 additions and 206 deletions

View file

@ -92,29 +92,25 @@ class Point2PhantomNode : private noncopyable
Candidate() : m_dist(numeric_limits<double>::max()), m_fid(OsrmFtSegMapping::FtSeg::INVALID_FID) {}
};
size_t m_ptIdx;
buffer_vector<Candidate, 128> m_candidates[2];
buffer_vector<Candidate, 128> m_candidates;
uint32_t m_mwmId;
Index const * m_pIndex;
public:
Point2PhantomNode(OsrmFtSegMapping const & mapping, Index const * pIndex, m2::PointD const & direction)
: m_direction(direction), m_mapping(mapping), m_ptIdx(0),
: m_direction(direction), m_mapping(mapping),
m_mwmId(numeric_limits<uint32_t>::max()), m_pIndex(pIndex)
{
}
void SetPoint(m2::PointD const & pt, size_t idx)
void SetPoint(m2::PointD const & pt)
{
ASSERT_LESS(idx, 2, ());
m_point = pt;
m_ptIdx = idx;
}
bool HasCandidates(uint32_t idx) const
bool HasCandidates() const
{
ASSERT_LESS(idx, 2, ());
return !m_candidates[idx].empty();
return !m_candidates.empty();
}
void operator() (FeatureType const & ft)
@ -151,7 +147,7 @@ public:
}
if (res.m_fid != OsrmFtSegMapping::FtSeg::INVALID_FID)
m_candidates[m_ptIdx].push_back(res);
m_candidates.push_back(res);
}
double CalculateDistance(OsrmFtSegMapping::FtSeg const & s) const
@ -246,7 +242,7 @@ public:
offset = max(static_cast<int>(distance), 1);
}
void CalculateOffsets(OsrmRouter::FeatureGraphNode & node) const
void CalculateOffsets(FeatureGraphNode & node) const
{
CalculateOffset(node.m_seg, node.m_segPt, node.m_node.forward_node_id, node.m_node.forward_offset, true);
CalculateOffset(node.m_seg, node.m_segPt, node.m_node.reverse_node_id, node.m_node.reverse_offset, false);
@ -257,8 +253,7 @@ public:
node.m_node.reverse_weight = 0;
}
void MakeResult(OsrmRouter::FeatureGraphNodeVecT & res, size_t maxCount, uint32_t & mwmId,
bool needFinal, volatile bool const & requestCancel)
void MakeResult(FeatureGraphNodeVecT & res, size_t maxCount, uint32_t & mwmId, volatile bool const & requestCancel)
{
mwmId = m_mwmId;
if (mwmId == numeric_limits<uint32_t>::max())
@ -266,41 +261,36 @@ public:
vector<OsrmFtSegMapping::FtSeg> segments;
size_t const processPtCount = needFinal ? 2 : 1;
segments.resize(maxCount * processPtCount);
segments.resize(maxCount);
OsrmFtSegMapping::FtSegSetT segmentSet;
for (size_t i = 0; i < processPtCount; ++i)
sort(m_candidates.begin(), m_candidates.end(), [] (Candidate const & r1, Candidate const & r2)
{
sort(m_candidates[i].begin(), m_candidates[i].end(), [] (Candidate const & r1, Candidate const & r2)
{
return (r1.m_dist < r2.m_dist);
});
return (r1.m_dist < r2.m_dist);
});
size_t const n = min(m_candidates[i].size(), maxCount);
for (size_t j = 0; j < n; ++j)
{
OsrmFtSegMapping::FtSeg & seg = segments[i * maxCount + j];
Candidate const & c = m_candidates[i][j];
size_t const n = min(m_candidates.size(), maxCount);
for (size_t j = 0; j < n; ++j)
{
OsrmFtSegMapping::FtSeg & seg = segments[j];
Candidate const & c = m_candidates[j];
seg.m_fid = c.m_fid;
seg.m_pointStart = c.m_segIdx;
seg.m_pointEnd = c.m_segIdx + 1;
seg.m_fid = c.m_fid;
seg.m_pointStart = c.m_segIdx;
seg.m_pointEnd = c.m_segIdx + 1;
segmentSet.insert(&seg);
}
segmentSet.insert(&seg);
}
OsrmFtSegMapping::OsrmNodesT nodes;
m_mapping.GetOsrmNodes(segmentSet, nodes, requestCancel);
res.clear();
res.resize(maxCount * 2);
res.resize(maxCount);
for (size_t i = 0; i < processPtCount; ++i)
for (size_t j = 0; j < maxCount; ++j)
{
size_t const idx = i * maxCount + j;
size_t const idx = j;
if (!segments[idx].IsValid())
continue;
@ -308,7 +298,7 @@ public:
auto it = nodes.find(segments[idx].Store());
if (it != nodes.end())
{
OsrmRouter::FeatureGraphNode & node = res[idx];
FeatureGraphNode & node = res[idx];
if (!m_direction.IsAlmostZero())
{
@ -338,7 +328,7 @@ public:
}
node.m_seg = segments[idx];
node.m_segPt = m_candidates[i][j].m_point;
node.m_segPt = m_candidates[j].m_point;
CalculateOffsets(node);
}
@ -348,8 +338,34 @@ public:
} // namespace
RoutingMapping::RoutingMapping(string fName): map_counter(0), facade_counter(0), m_base_name(fName+DATA_FILE_EXTENSION)
{
Platform & pl = GetPlatform();
fName += DATA_FILE_EXTENSION;
string const fPath = pl.WritablePathForFile(fName + ROUTING_FILE_EXTENSION);
if (!pl.IsFileExistsByFullPath(fPath))
throw IRouter::ResultCode::RouteFileNotExist;
// Open new container and check that mwm and routing have equal timestamp.
LOG(LDEBUG, ("Load routing index for file:", fPath));
m_container.Open(fPath);
{
FileReader r1 = m_container.GetReader(VERSION_FILE_TAG);
ReaderSrc src1(r1);
ModelReaderPtr r2 = FilesContainerR(pl.GetReader(fName)).GetReader(VERSION_FILE_TAG);
ReaderSrc src2(r2.GetPtr());
if (ver::ReadTimestamp(src1) != ver::ReadTimestamp(src2))
{
m_container.Close();
throw IRouter::ResultCode::InconsistentMWMandRoute;
}
mapping.Load(m_container);
}
}
OsrmRouter::OsrmRouter(Index const * index, CountryFileFnT const & fn)
: m_countryFn(fn), m_pIndex(index), m_isFinalChanged(false),
: m_pIndex(index), m_indexManager(fn), m_isFinalChanged(false),
m_requestCancel(false)
{
m_isReadyThread.clear();
@ -369,10 +385,7 @@ void OsrmRouter::ClearState()
m_cachedFinalNodes.clear();
m_dataFacade.Clear();
m_mapping.Clear();
m_container.Close();
m_indexManager.Clear();
}
void OsrmRouter::SetFinalPoint(m2::PointD const & finalPt)
@ -462,11 +475,7 @@ void OsrmRouter::CalculateRouteAsync(ReadyCallback const & callback)
LOG(LERROR, ("Routing index absent or incorrect. Error while loading routing index:", e.Msg()));
code = InternalError;
// Clear data while m_container is valid.
m_dataFacade.Clear();
m_mapping.Clear();
m_container.Close();
m_indexManager.Clear();
}
GetPlatform().RunOnGuiThread(bind(callback, route, code));
@ -484,125 +493,115 @@ bool IsRouteExist(RawRouteData const & r)
}
OsrmRouter::ResultCode OsrmRouter::CalculateRouteImpl(m2::PointD const & startPt, m2::PointD const & startDr, m2::PointD const & finalPt, Route & route)
bool OsrmRouter::FindSingleRoute(FeatureGraphNodeVecT const & source, FeatureGraphNodeVecT const & target, DataFacadeT & facade,
RawRouteData& outputPath, FeatureGraphNodeVecT::const_iterator & sourceEdge, FeatureGraphNodeVecT::const_iterator & targetEdge)
{
// 1. Find country file name and check that we are in the same MWM.
string fName = m_countryFn(startPt);
if (fName != m_countryFn(finalPt))
return PointsInDifferentMWM;
// 2. Open routing file index if needed.
Platform & pl = GetPlatform();
fName += DATA_FILE_EXTENSION;
string const fPath = pl.WritablePathForFile(fName + ROUTING_FILE_EXTENSION);
if (!pl.IsFileExistsByFullPath(fPath))
return RouteFileNotExist;
if (NeedReload(fPath))
{
LOG(LDEBUG, ("Load routing index for file:", fPath));
// Clear data while m_container is valid.
m_dataFacade.Clear();
m_mapping.Clear();
// Open new container and check that mwm and routing have equal timestamp.
m_container.Open(fPath);
{
FileReader r1 = m_container.GetReader(VERSION_FILE_TAG);
ReaderSrc src1(r1);
ModelReaderPtr r2 = FilesContainerR(pl.GetReader(fName)).GetReader(VERSION_FILE_TAG);
ReaderSrc src2(r2.GetPtr());
if (ver::ReadTimestamp(src1) != ver::ReadTimestamp(src2))
{
m_container.Close();
return InconsistentMWMandRoute;
}
}
m_mapping.Load(m_container);
}
// 3. Find start/end nodes.
if (!m_mapping.IsMapped())
m_mapping.Map(m_container);
FeatureGraphNodeVecT graphNodes;
uint32_t mwmId = -1;
ResultCode const code = FindPhantomNodes(fName, startPt, startDr, finalPt, graphNodes, MAX_NODE_CANDIDATES, mwmId);
if (m_requestCancel)
return Cancelled;
m_mapping.Unmap();
if (code != NoError)
return code;
// 4. Find route.
m_dataFacade.Load(m_container);
ASSERT_EQUAL(graphNodes.size(), MAX_NODE_CANDIDATES * 2, ());
SearchEngineData engineData;
ShortestPathRouting<DataFacadeT> pathFinder(&m_dataFacade, engineData);
RawRouteData rawRoute;
ShortestPathRouting<DataFacadeT> pathFinder(&facade, engineData);
size_t ni = 0, nj = 0;
while (ni < MAX_NODE_CANDIDATES && nj < MAX_NODE_CANDIDATES)
sourceEdge = source.cbegin();
targetEdge = target.cbegin();
while (sourceEdge < source.cend() && targetEdge < target.cend())
{
if (m_requestCancel)
break;
PhantomNodes nodes;
nodes.source_phantom = graphNodes[ni].m_node;
nodes.target_phantom = graphNodes[nj + MAX_NODE_CANDIDATES].m_node;
nodes.source_phantom = sourceEdge->m_node;
nodes.target_phantom = targetEdge->m_node;
rawRoute = RawRouteData();
outputPath = RawRouteData();
if ((nodes.source_phantom.forward_node_id != INVALID_NODE_ID ||
nodes.source_phantom.reverse_node_id != INVALID_NODE_ID) &&
(nodes.target_phantom.forward_node_id != INVALID_NODE_ID ||
nodes.target_phantom.reverse_node_id != INVALID_NODE_ID))
{
rawRoute.segment_end_coordinates.push_back(nodes);
outputPath.segment_end_coordinates.push_back(nodes);
pathFinder({nodes}, {}, rawRoute);
pathFinder({nodes}, {}, outputPath);
}
if (!IsRouteExist(rawRoute))
/// @todo: make more complex nearest edge turnaround
if (!IsRouteExist(outputPath))
{
++ni;
if (ni == MAX_NODE_CANDIDATES)
++sourceEdge;
if (sourceEdge == source.cend())
{
++nj;
ni = 0;
++targetEdge;
sourceEdge = source.cbegin();
}
}
else
break;
return true;
}
//m_dataFacade.Clear();
return IsRouteExist(outputPath);
}
if (!IsRouteExist(rawRoute))
return RouteNotFound;
OsrmRouter::ResultCode OsrmRouter::CalculateRouteImpl(m2::PointD const & startPt, m2::PointD const & startDr, m2::PointD const & finalPt, Route & route)
{
RoutingMappingPtrT startMapping;
RoutingMappingPtrT finalMapping;
try
{
startMapping = m_indexManager.GetMappingByPoint(startPt);
finalMapping = m_indexManager.GetMappingByPoint(finalPt);
}
catch (OsrmRouter::ResultCode e)
{
return e;
}
// 3. Find start/end nodes.
MultiroutingTaskPointT startTask;
startTask.resize(1);
uint32_t mwmId = -1;
{
startMapping->Map();
ResultCode const code = FindPhantomNodes(startMapping->GetName(), startPt, startDr, startTask[0], MAX_NODE_CANDIDATES, mwmId, startMapping->mapping);
if (code != NoError)
return code;
startMapping->Unmap();
}
{
if (finalPt != m_CachedTargetPoint)
{
m_CachedTargetTask.resize(1);
finalMapping->Map();
ResultCode const code = FindPhantomNodes(finalMapping->GetName(), finalPt, m2::PointD::Zero(), m_CachedTargetTask[0], MAX_NODE_CANDIDATES, mwmId, finalMapping->mapping);
if (code != NoError)
return code;
m_CachedTargetPoint = finalPt;
finalMapping->Unmap();
}
}
if (m_requestCancel)
return Cancelled;
// 4. Find route.
RawRouteData rawRoute;
FeatureGraphNodeVecT::const_iterator sourceEdge;
FeatureGraphNodeVecT::const_iterator targetEdge;
if (startMapping->GetName() == finalMapping->GetName())
{
startMapping->LoadFacade();
if (!FindSingleRoute(startTask[0], m_CachedTargetTask[0], startMapping->dataFacade, rawRoute, sourceEdge, targetEdge))
return RouteNotFound;
}
if (m_requestCancel)
return Cancelled;
// 5. Restore route.
m_mapping.Map(m_container);
startMapping->Map();
ASSERT_LESS(ni, MAX_NODE_CANDIDATES, ());
ASSERT_LESS(nj, MAX_NODE_CANDIDATES, ());
ASSERT(sourceEdge != startTask[0].cend(), ());
ASSERT(targetEdge != m_CachedTargetTask[0].cend(), ());
typedef OsrmFtSegMapping::FtSeg SegT;
FeatureGraphNode const & sNode = graphNodes[ni];
FeatureGraphNode const & eNode = graphNodes[nj + MAX_NODE_CANDIDATES];
FeatureGraphNode const & sNode = *sourceEdge;
FeatureGraphNode const & eNode = *targetEdge;
SegT const & segBegin = sNode.m_seg;
SegT const & segEnd = eNode.m_seg;
@ -640,7 +639,7 @@ OsrmRouter::ResultCode OsrmRouter::CalculateRouteImpl(m2::PointD const & startPt
GetTurnDirection(rawRoute.unpacked_path_segments[i][j - 1],
rawRoute.unpacked_path_segments[i][j],
mwmId, t, fName);
mwmId, startMapping, t);
if (t.m_turn != turns::NoTurn)
turnsDir.push_back(t);
@ -659,7 +658,7 @@ OsrmRouter::ResultCode OsrmRouter::CalculateRouteImpl(m2::PointD const & startPt
}
buffer_vector<SegT, 8> buffer;
m_mapping.ForEachFtSeg(path_data.node, MakeBackInsertFunctor(buffer));
startMapping->mapping.ForEachFtSeg(path_data.node, MakeBackInsertFunctor(buffer));
auto correctFn = [&buffer] (SegT const & seg, size_t & ind)
{
@ -794,21 +793,21 @@ m2::PointD OsrmRouter::GetPointForTurnAngle(OsrmFtSegMapping::FtSeg const &seg,
return nextPnt;
}
NodeID OsrmRouter::GetTurnTargetNode(NodeID src, NodeID trg, QueryEdge::EdgeData const & edgeData)
NodeID OsrmRouter::GetTurnTargetNode(NodeID src, NodeID trg, QueryEdge::EdgeData const & edgeData, RoutingMappingPtrT const & routingMapping)
{
ASSERT_NOT_EQUAL(src, SPECIAL_NODEID, ());
ASSERT_NOT_EQUAL(trg, SPECIAL_NODEID, ());
if (!edgeData.shortcut)
return trg;
ASSERT_LESS(edgeData.id, m_dataFacade.GetNumberOfNodes(), ());
ASSERT_LESS(edgeData.id, routingMapping->dataFacade.GetNumberOfNodes(), ());
EdgeID edge = SPECIAL_EDGEID;
QueryEdge::EdgeData d;
for (EdgeID e : m_dataFacade.GetAdjacentEdgeRange(edgeData.id))
for (EdgeID e : routingMapping->dataFacade.GetAdjacentEdgeRange(edgeData.id))
{
if (m_dataFacade.GetTarget(e) == src )
if (routingMapping->dataFacade.GetTarget(e) == src )
{
d = m_dataFacade.GetEdgeData(e, edgeData.id);
d = routingMapping->dataFacade.GetEdgeData(e, edgeData.id);
if (d.backward)
{
edge = e;
@ -819,11 +818,11 @@ NodeID OsrmRouter::GetTurnTargetNode(NodeID src, NodeID trg, QueryEdge::EdgeData
if (edge == SPECIAL_EDGEID)
{
for (EdgeID e : m_dataFacade.GetAdjacentEdgeRange(src))
for (EdgeID e : routingMapping->dataFacade.GetAdjacentEdgeRange(src))
{
if (m_dataFacade.GetTarget(e) == edgeData.id)
if (routingMapping->dataFacade.GetTarget(e) == edgeData.id)
{
d = m_dataFacade.GetEdgeData(e, src);
d = routingMapping->dataFacade.GetEdgeData(e, src);
if (d.forward)
{
edge = e;
@ -835,7 +834,7 @@ NodeID OsrmRouter::GetTurnTargetNode(NodeID src, NodeID trg, QueryEdge::EdgeData
ASSERT_NOT_EQUAL(edge, SPECIAL_EDGEID, ());
if (d.shortcut)
return GetTurnTargetNode(src, edgeData.id, d);
return GetTurnTargetNode(src, edgeData.id, d, routingMapping);
return edgeData.id;
}
@ -844,21 +843,22 @@ void OsrmRouter::GetPossibleTurns(NodeID node,
m2::PointD const & p1,
m2::PointD const & p,
uint32_t mwmId,
RoutingMappingPtrT const & routingMapping,
OsrmRouter::TurnCandidatesT & candidates)
{
for (EdgeID e : m_dataFacade.GetAdjacentEdgeRange(node))
for (EdgeID e : routingMapping->dataFacade.GetAdjacentEdgeRange(node))
{
QueryEdge::EdgeData const data = m_dataFacade.GetEdgeData(e, node);
QueryEdge::EdgeData const data = routingMapping->dataFacade.GetEdgeData(e, node);
if (!data.forward)
continue;
NodeID trg = GetTurnTargetNode(node, m_dataFacade.GetTarget(e), data);
NodeID trg = GetTurnTargetNode(node, routingMapping->dataFacade.GetTarget(e), data, routingMapping);
ASSERT_NOT_EQUAL(trg, SPECIAL_NODEID, ());
auto const range = m_mapping.GetSegmentsRange(trg);
auto const range = routingMapping->mapping.GetSegmentsRange(trg);
OsrmFtSegMapping::FtSeg seg;
m_mapping.GetSegmentByIndex(range.first, seg);
routingMapping->mapping.GetSegmentByIndex(range.first, seg);
FeatureType ft;
Index::FeaturesLoaderGuard loader(*m_pIndex, mwmId);
@ -1021,16 +1021,16 @@ turns::TurnDirection OsrmRouter::RoundaboutDirection(bool isRound1, bool isRound
void OsrmRouter::GetTurnDirection(PathData const & node1,
PathData const & node2,
uint32_t mwmId, Route::TurnItem & turn, string const & fName)
uint32_t mwmId, RoutingMappingPtrT const & routingMapping, Route::TurnItem & turn)
{
auto nSegs1 = m_mapping.GetSegmentsRange(node1.node);
auto nSegs2 = m_mapping.GetSegmentsRange(node2.node);
auto nSegs1 = routingMapping->mapping.GetSegmentsRange(node1.node);
auto nSegs2 = routingMapping->mapping.GetSegmentsRange(node2.node);
ASSERT_GREATER(nSegs1.second, 0, ());
OsrmFtSegMapping::FtSeg seg1, seg2;
m_mapping.GetSegmentByIndex(nSegs1.second - 1, seg1);
m_mapping.GetSegmentByIndex(nSegs2.first, seg2);
routingMapping->mapping.GetSegmentByIndex(nSegs1.second - 1, seg1);
routingMapping->mapping.GetSegmentByIndex(nSegs2.first, seg2);
FeatureType ft1, ft2;
Index::FeaturesLoaderGuard loader1(*m_pIndex, mwmId);
@ -1059,11 +1059,11 @@ void OsrmRouter::GetTurnDirection(PathData const & node1,
m2::PointD const p1OneSeg = ft1.GetPoint(seg1.m_pointStart < seg1.m_pointEnd ? seg1.m_pointEnd - 1 : seg1.m_pointEnd + 1);
TurnCandidatesT nodes;
GetPossibleTurns(node1.node, p1OneSeg, p, mwmId, nodes);
GetPossibleTurns(node1.node, p1OneSeg, p, mwmId, routingMapping, nodes);
#ifdef _DEBUG
GeomTurnCandidateT geoNodes;
GetTurnGeometry(p, p1OneSeg, geoNodes, fName);
GetTurnGeometry(p, p1OneSeg, geoNodes, routingMapping->GetName());
m2::PointD const p2OneSeg = ft2.GetPoint(seg2.m_pointStart < seg2.m_pointEnd ? seg2.m_pointStart + 1 : seg2.m_pointStart - 1);
@ -1096,7 +1096,7 @@ void OsrmRouter::GetTurnDirection(PathData const & node1,
bool const isRound2 = ftypes::IsRoundAboutChecker::Instance()(ft2);
if (!hasMultiTurns
&& !KeepOnewayOutgoingTurnIncomingEdges(nodes, turn, p, p1OneSeg, fName)
&& !KeepOnewayOutgoingTurnIncomingEdges(nodes, turn, p, p1OneSeg, routingMapping->GetName())
&& !KeepOnewayOutgoingTurnRoundabout(isRound1, isRound2))
{
turn.m_turn = turns::NoTurn;
@ -1226,44 +1226,22 @@ void OsrmRouter::FixupTurns(vector<m2::PointD> const & points, Route::TurnsT & t
}
}
IRouter::ResultCode OsrmRouter::FindPhantomNodes(string const & fName, m2::PointD const & startPt, m2::PointD const & startDr, m2::PointD const & finalPt,
FeatureGraphNodeVecT & res, size_t maxCount, uint32_t & mwmId)
IRouter::ResultCode OsrmRouter::FindPhantomNodes(string const & fName, m2::PointD const & point, m2::PointD const & direction,
FeatureGraphNodeVecT & res, size_t maxCount, uint32_t & mwmId, OsrmFtSegMapping const & mapping)
{
Point2PhantomNode getter(m_mapping, m_pIndex, startDr);
Point2PhantomNode getter(mapping, m_pIndex, direction);
getter.SetPoint(point);
auto processPt = [&] (m2::PointD const & p, size_t idx)
{
getter.SetPoint(p, idx);
LOG(LINFO, ("MAKE FOR EACH OF", fName));
m_pIndex->ForEachInRectForMWM(getter,
MercatorBounds::RectByCenterXYAndSizeInMeters(point, FEATURE_BY_POINT_RADIUS_M),
scales::GetUpperScale(), fName);
m_pIndex->ForEachInRectForMWM(getter,
MercatorBounds::RectByCenterXYAndSizeInMeters(p, FEATURE_BY_POINT_RADIUS_M),
scales::GetUpperScale(), fName);
};
processPt(startPt, 0);
if (!getter.HasCandidates(0))
if (!getter.HasCandidates())
return StartPointNotFound;
bool const hasFinalCache = !m_cachedFinalNodes.empty();
if (!hasFinalCache)
{
processPt(finalPt, 1);
if (!getter.HasCandidates(1))
return EndPointNotFound;
}
getter.MakeResult(res, maxCount, mwmId, !hasFinalCache, m_requestCancel);
if (hasFinalCache)
copy(m_cachedFinalNodes.begin(), m_cachedFinalNodes.end(), res.begin() + maxCount);
else
m_cachedFinalNodes.assign(res.begin() + maxCount, res.end());
getter.MakeResult(res, maxCount, mwmId, m_requestCancel);
return NoError;
}
bool OsrmRouter::NeedReload(string const & fPath) const
{
return (m_container.GetName() != fPath);
}
}

View file

@ -18,24 +18,121 @@ class Index;
struct PhantomNode;
struct PathData;
class FeatureType;
class RawRouteData;
namespace routing
{
class OsrmRouter : public IRouter
typedef function<string (m2::PointD const &)> CountryFileFnT;
typedef OsrmDataFacade<QueryEdge::EdgeData> DataFacadeT;
///Single graph node representation for routing task
struct FeatureGraphNode
{
PhantomNode m_node;
OsrmFtSegMapping::FtSeg m_seg;
m2::PointD m_segPt;
};
/// All edges available for start route while routing
typedef vector<FeatureGraphNode> FeatureGraphNodeVecT;
/// Points vector to calculate several routes
typedef vector<FeatureGraphNodeVecT> MultiroutingTaskPointT;
///Datamapping and facade for single MWM and MWM.routing file
struct RoutingMapping
{
DataFacadeT dataFacade;
OsrmFtSegMapping mapping;
///@param fName: mwm file path
RoutingMapping(string fName);
~RoutingMapping()
{
// Clear data while m_container is valid.
dataFacade.Clear();
mapping.Clear();
m_container.Close();
}
void Map()
{
++map_counter;
if (!mapping.IsMapped())
mapping.Map(m_container);
}
void Unmap()
{
--map_counter;
if (map_counter<1 && mapping.IsMapped())
mapping.Unmap();
}
void LoadFacade()
{
if (!facade_counter)
dataFacade.Load(m_container);
++facade_counter;
}
void FreeFacade()
{
--facade_counter;
if (!facade_counter)
dataFacade.Clear();
}
string GetName() {return m_base_name;}
private:
size_t map_counter;
size_t facade_counter;
string m_base_name;
FilesMappingContainer m_container;
};
typedef shared_ptr<RoutingMapping> RoutingMappingPtrT;
/*! Manager for loading, cashing and building routing indexes.
* Builds and shares special routing contexts.
*/
class RoutingIndexManager
{
typedef function<string (m2::PointD const &)> CountryFileFnT;
CountryFileFnT m_countryFn;
public:
std::map<string, RoutingMappingPtrT> m_mapping;
struct FeatureGraphNode
public:
RoutingIndexManager(CountryFileFnT const & fn): m_countryFn(fn) {}
RoutingMappingPtrT GetMappingByPoint(m2::PointD point)
{
PhantomNode m_node;
OsrmFtSegMapping::FtSeg m_seg;
m2::PointD m_segPt;
};
typedef vector<FeatureGraphNode> FeatureGraphNodeVecT;
string fName = m_countryFn(point);
//Check if we have already load this file
auto mapIter = m_mapping.find(fName);
if (mapIter != m_mapping.end())
return mapIter->second;
//Or load and check file
RoutingMappingPtrT new_mapping = RoutingMappingPtrT(new RoutingMapping(fName));
m_mapping.insert(std::make_pair(fName, new_mapping));
return new_mapping;
}
void Clear()
{
m_mapping.clear();
}
};
class OsrmRouter : public IRouter
{
public:
struct TurnCandidate
{
@ -59,24 +156,39 @@ public:
virtual void CalculateRoute(m2::PointD const & startPt, ReadyCallback const & callback, m2::PointD const & direction = m2::PointD::Zero());
protected:
IRouter::ResultCode FindPhantomNodes(string const & fName, m2::PointD const & startPt, const m2::PointD &startDr, m2::PointD const & finalPt,
FeatureGraphNodeVecT & res, size_t maxCount, uint32_t & mwmId);
IRouter::ResultCode FindPhantomNodes(string const & fName, m2::PointD const & point, const m2::PointD &direction,
FeatureGraphNodeVecT & res, size_t maxCount, uint32_t & mwmId, OsrmFtSegMapping const & mapping);
bool NeedReload(string const & fPath) const;
/*! Find single shortest path in single MWM between 2 sets of edges
* @param source: vector of source edgest to make path
* @param taget: vector of target edges to make path
* @param facade: OSRM routing data facade to recover graph information
* @param outputPath: result path data
* @param sourceEdge: iterator to src edge from source vector
* @param sourceEdge: iterator to target edge from target vector
* @return true if path exists. False otherwise
*/
bool FindSingleRoute(FeatureGraphNodeVecT const & source, FeatureGraphNodeVecT const & target, DataFacadeT & facade,
RawRouteData& outputPath,
FeatureGraphNodeVecT::const_iterator & sourceEdge,
FeatureGraphNodeVecT::const_iterator & targetEdge);
void CalculateRouteAsync(ReadyCallback const & callback);
ResultCode CalculateRouteImpl(m2::PointD const & startPt, m2::PointD const & startDr, m2::PointD const & finalPt, Route & route);
private:
NodeID GetTurnTargetNode(NodeID src, NodeID trg, QueryEdge::EdgeData const & edgeData);
NodeID GetTurnTargetNode(NodeID src, NodeID trg, QueryEdge::EdgeData const & edgeData, RoutingMappingPtrT const & routingMapping);
void GetPossibleTurns(NodeID node,
m2::PointD const & p1,
m2::PointD const & p,
uint32_t mwmId,
RoutingMappingPtrT const & routingMapping,
TurnCandidatesT & candidates);
void GetTurnDirection(PathData const & node1,
PathData const & node2,
uint32_t mwmId, Route::TurnItem & turn, string const & fName);
uint32_t mwmId,
RoutingMappingPtrT const & routingMapping,
Route::TurnItem & turn);
void CalculateTurnGeometry(vector<m2::PointD> const & points, Route::TurnsT const & turnsDir, turns::TurnsGeomT & turnsGeom) const;
void FixupTurns(vector<m2::PointD> const & points, Route::TurnsT & turnsDir) const;
m2::PointD GetPointForTurnAngle(OsrmFtSegMapping::FtSeg const &seg,
@ -96,11 +208,12 @@ private:
Index const * m_pIndex;
typedef OsrmDataFacade<QueryEdge::EdgeData> DataFacadeT;
DataFacadeT m_dataFacade;
OsrmFtSegMapping m_mapping;
FeatureGraphNodeVecT graphNodes;
FilesMappingContainer m_container;
MultiroutingTaskPointT m_CachedTargetTask;
m2::PointD m_CachedTargetPoint;
RoutingIndexManager m_indexManager;
bool m_isFinalChanged;
m2::PointD m_startPt, m_finalPt, m_startDr;