Rewrite feature covering. Use a simpler approach and remove stream optimizer.

This commit is contained in:
Yury Melnichek 2011-04-25 03:49:24 +02:00 committed by Alex Zolotarev
parent 830281b9ee
commit 81e732a71d
12 changed files with 140 additions and 974 deletions

View file

@ -3,31 +3,26 @@
#include "../base/assert.hpp"
#include "../base/base.hpp"
#include "../base/logging.hpp"
#include "../base/set_operations.hpp"
#include "../base/stl_add.hpp"
#include "../base/math.hpp"
#include "../std/algorithm.hpp"
#include "../std/array.hpp"
#include "../std/functional.hpp"
#include "../std/iterator.hpp"
#include "../std/map.hpp"
#include "../std/queue.hpp"
#include "../std/utility.hpp"
#include "../std/vector.hpp"
namespace covering
{
enum CellTriangleIntersection
enum CellObjectIntersection
{
CELL_TRIANGLE_NO_INTERSECTION = 0,
CELL_TRIANGLE_INTERSECT = 1,
CELL_INSIDE_TRIANGLE = 2,
TRIANGLE_INSIDE_CELL = 3
CELL_OBJECT_NO_INTERSECTION = 0,
CELL_OBJECT_INTERSECT = 1,
CELL_INSIDE_OBJECT = 2,
OBJECT_INSIDE_CELL = 3
};
template <class CellIdT>
CellTriangleIntersection IntersectCellWithTriangle(
CellIdT const cell, m2::PointD const & a, m2::PointD const & b, m2::PointD const & c)
inline CellObjectIntersection IntersectCellWithLine(CellIdT const cell,
m2::PointD const & a,
m2::PointD const & b)
{
pair<uint32_t, uint32_t> const xy = cell.XY();
uint32_t const r = cell.Radius();
@ -39,308 +34,66 @@ CellTriangleIntersection IntersectCellWithTriangle(
m2::PointD(xy.first + r, xy.second - r)
};
for (int i = 0; i < 4; ++i)
{
m2::PointD const & p1 = cellCorners[i];
m2::PointD const & p2 = cellCorners[i == 0 ? 3 : i - 1];
if (m2::SegmentsIntersect(a, b, p1, p2)) return CELL_TRIANGLE_INTERSECT;
if (m2::SegmentsIntersect(b, c, p1, p2)) return CELL_TRIANGLE_INTERSECT;
if (m2::SegmentsIntersect(c, a, p1, p2)) return CELL_TRIANGLE_INTERSECT;
}
if (m2::SegmentsIntersect(a, b, cellCorners[i], cellCorners[i == 0 ? 3 : i - 1]))
return CELL_OBJECT_INTERSECT;
if (xy.first - r <= a.x && a.x <= xy.first + r && xy.second - r <= a.y && a.y <= xy.second + r)
return TRIANGLE_INSIDE_CELL;
if (m2::IsPointStrictlyInsideTriangle(cellCorners[0], a, b, c))
return CELL_INSIDE_TRIANGLE;
return CELL_TRIANGLE_NO_INTERSECTION;
}
template <class CellIdT, typename F>
void CoverTriangle(m2::PointD const & a, m2::PointD const & b, m2::PointD const & c,
F const & output, uint8_t const level = CellIdT::DEPTH_LEVELS - 1,
CellIdT const startCell = CellIdT::Root())
{
ASSERT_LESS_OR_EQUAL(startCell.Level(), level, (a, b, c));
CellTriangleIntersection intersection = IntersectCellWithTriangle(startCell, a, b, c);
if (intersection == CELL_TRIANGLE_NO_INTERSECTION)
return;
if (startCell.Level() == level || intersection == CELL_INSIDE_TRIANGLE)
{
output(startCell);
return;
}
for (uint8_t child = 0; child < 4; ++child)
CoverTriangle(a, b, c, output, level, startCell.Child(child));
return OBJECT_INSIDE_CELL;
return CELL_OBJECT_NO_INTERSECTION;
}
template <class CellIdT>
class Covering
CellObjectIntersection IntersectCellWithTriangle(
CellIdT const cell, m2::PointD const & a, m2::PointD const & b, m2::PointD const & c)
{
public:
typedef CellIdT CellId;
typedef typename CellId::LessQueueOrder LessQueueOrder;
Covering() : m_Size(0) {}
explicit Covering(CellId cell) : m_Size(1)
{
m_Covering[cell.Level()].push_back(cell);
}
explicit Covering(vector<CellId> const & v, int64_t minId = 0)
{
for (size_t i = 0; i < v.size(); ++i)
m_Covering[v[i].Level()].push_back(v[i]);
Sort();
Unique();
RemoveDuplicateChildren();
RemoveFullSquares(minId);
m_Size = CalculateSize();
}
// Cover triangle.
Covering(m2::PointD const & a, m2::PointD const & b, m2::PointD const & c,
int level = CellId::DEPTH_LEVELS - 1)
{
// TODO: ASSERT(a != b), ASSERT(b != c), ASSERT(a != c) ?
ASSERT(0 <= level && level <= CellId::DEPTH_LEVELS, (level, CellId::Root()));
CoverTriangleInfo info;
info.m_pCovering = this;
info.m_A = a;
info.m_B = b;
info.m_C = c;
info.m_Level = level;
CoverTriangleImpl(info, CellId::Root());
Sort();
RemoveFullSquares();
m_Size = CalculateSize();
}
size_t Size() const
{
ASSERT_EQUAL(m_Size, CalculateSize(), ());
return m_Size;
}
void Append(Covering<CellId> const & c)
{
AppendWithoutNormalize(c);
RemoveDuplicateChildren();
RemoveFullSquares();
m_Size = CalculateSize();
}
void OutputToVector(vector<CellId> & result) const
{
for (int level = 0; level < CellId::DEPTH_LEVELS; ++level)
result.insert(result.end(), m_Covering[level].begin(), m_Covering[level].end());
}
void OutputToVector(vector<int64_t> & result) const
{
for (int level = 0; level < CellId::DEPTH_LEVELS; ++level)
for (size_t i = 0; i < m_Covering[level].size(); ++i)
result.push_back(m_Covering[level][i].ToInt64());
}
void Simplify(int64_t minId = 0)
{
int cellsSimplified = 0;
int const initialSize = m_Size;
for (int level = CellId::DEPTH_LEVELS - 1; level > 1; --level)
{
if (m_Covering[level].size() >= 2)
{
int const initialLevelSize = static_cast<int>(m_Covering[level].size());
SimplifyLevel(level, minId);
cellsSimplified += initialLevelSize - static_cast<int>(m_Covering[level].size());
if (cellsSimplified > initialSize / 2)
break;
}
}
RemoveDuplicateChildren();
RemoveFullSquares(minId);
m_Size = CalculateSize();
}
private:
void SimplifyLevel(int level, int64_t minId)
{
map<CellId, uint32_t, LessQueueOrder> parentCellCounts;
typedef typename vector<CellId>::const_iterator ConstIteartor;
for (ConstIteartor it = m_Covering[level].begin(); it != m_Covering[level].end(); ++it)
if (it->Parent().ToInt64() >= minId)
++parentCellCounts[it->Parent()];
vector<CellId> parentCells, childCells;
for (ConstIteartor it = m_Covering[level].begin(); it != m_Covering[level].end(); ++it)
{
if (parentCellCounts[it->Parent()] > 1)
parentCells.push_back(it->Parent());
else
childCells.push_back(*it);
}
ASSERT(IsSorted(parentCells.begin(), parentCells.end(), LessQueueOrder()), (parentCells));
ASSERT(IsSorted(childCells.begin(), childCells.end(), LessQueueOrder()), (childCells));
m_Covering[level].swap(childCells);
parentCells.erase(unique(parentCells.begin(), parentCells.end()), parentCells.end());
AppendToVector(m_Covering[level - 1], parentCells);
}
static void AppendToVector(vector<CellId> & a, vector<CellId> const & b)
{
ASSERT(IsSortedAndUnique(a.begin(), a.end(), LessQueueOrder()), (a));
ASSERT(IsSortedAndUnique(b.begin(), b.end(), LessQueueOrder()), (b));
vector<CellId> merged;
set_union(a.begin(), a.end(), b.begin(), b.end(), back_inserter(merged), LessQueueOrder());
a.swap(merged);
}
template <typename CompareT>
struct CompareCellsAtLevel
{
explicit CompareCellsAtLevel(int level) : m_Level(level) {}
bool operator() (CellId id1, CellId id2) const
{
return m_Comp(id1.AncestorAtLevel(m_Level), id2.AncestorAtLevel(m_Level));
}
CompareT m_Comp;
int m_Level;
};
void AppendWithoutNormalize(Covering const & c)
{
for (int level = CellId::DEPTH_LEVELS - 1; level >= 0; --level)
AppendToVector(m_Covering[level], c.m_Covering[level]);
}
void Sort()
{
for (int level = 0; level < CellId::DEPTH_LEVELS; ++level)
sort(m_Covering[level].begin(), m_Covering[level].end(), LessQueueOrder());
}
void Unique()
{
for (int level = 0; level < CellId::DEPTH_LEVELS; ++level)
{
vector<CellId> & covering = m_Covering[level];
ASSERT(IsSorted(covering.begin(), covering.end(), LessQueueOrder()), (covering));
covering.erase(unique(covering.begin(), covering.end()), covering.end());
}
}
void RemoveDuplicateChildren()
{
RemoveDuplicateChildrenImpl();
#ifdef DEBUG
// Assert that all duplicate children were removed.
vector<int64_t> v1, v2;
OutputToVector(v1);
RemoveDuplicateChildrenImpl();
OutputToVector(v2);
ASSERT_EQUAL(v1, v2, ());
#endif
}
void RemoveDuplicateChildrenImpl()
{
for (int parentLevel = 0; parentLevel < static_cast<int>(m_Covering.size()) - 1; ++parentLevel)
{
if (m_Covering[parentLevel].empty())
continue;
for (int childLevel = parentLevel + 1; childLevel < static_cast<int>(m_Covering.size());
++childLevel)
{
vector<CellId> substracted;
CompareCellsAtLevel<LessQueueOrder> comparator(parentLevel);
ASSERT(IsSorted(m_Covering[childLevel].begin(), m_Covering[childLevel].end(), comparator),
(m_Covering[childLevel]));
ASSERT(IsSorted(m_Covering[parentLevel].begin(), m_Covering[parentLevel].end(), comparator),
(m_Covering[parentLevel]));
SetDifferenceUnlimited(m_Covering[childLevel].begin(), m_Covering[childLevel].end(),
m_Covering[parentLevel].begin(), m_Covering[parentLevel].end(),
back_inserter(substracted), comparator);
m_Covering[childLevel].swap(substracted);
}
}
}
void RemoveFullSquares(int64_t minId = 0)
{
vector<CellId> cellsToAppend;
for (int level = m_Covering.size() - 1; level >= 0; --level)
{
// a -> b + parents
vector<CellId> const & a = m_Covering[level];
vector<CellId> b;
vector<CellId> parents;
b.reserve(a.size());
for (size_t i = 0; i < a.size(); ++i)
{
if (i + 3 < a.size())
{
CellId const parent = a[i].Parent();
if (parent == a[i+1].Parent() &&
parent == a[i+2].Parent() &&
parent == a[i+3].Parent() &&
parent.ToInt64() >= minId)
{
parents.push_back(parent);
i += 3;
continue;
}
}
b.push_back(a[i]);
}
m_Covering[level].swap(b);
if (level > 0)
AppendToVector(m_Covering[level - 1], parents);
}
}
size_t CalculateSize() const
{
size_t size = 0;
for (int level = 0; level < CellId::DEPTH_LEVELS; ++level)
size += m_Covering[level].size();
return size;
}
struct CoverTriangleInfo
{
m2::PointD m_A, m_B, m_C;
int m_Level;
Covering<CellId> * m_pCovering;
};
static void CoverTriangleImpl(CoverTriangleInfo const & info, CellId const cell)
{
ASSERT_LESS_OR_EQUAL(cell.Level(), info.m_Level, (info.m_A, info.m_B, info.m_C));
CellTriangleIntersection intersection =
IntersectCellWithTriangle(cell, info.m_A, info.m_B, info.m_C);
if (intersection == CELL_TRIANGLE_NO_INTERSECTION)
return;
if (cell.Level() == info.m_Level || intersection == CELL_INSIDE_TRIANGLE)
{
info.m_pCovering->m_Covering[cell.Level()].push_back(cell);
return;
}
for (uint8_t child = 0; child < 4; ++child)
CoverTriangleImpl(info, cell.Child(child));
}
array<vector<CellId>, CellId::DEPTH_LEVELS> m_Covering; // Covering by level.
size_t m_Size;
};
CellObjectIntersection const i1 = IntersectCellWithLine(cell, a, b);
if (i1 == CELL_OBJECT_INTERSECT)
return CELL_OBJECT_INTERSECT;
CellObjectIntersection const i2 = IntersectCellWithLine(cell, b, c);
if (i2 == CELL_OBJECT_INTERSECT)
return CELL_OBJECT_INTERSECT;
CellObjectIntersection const i3 = IntersectCellWithLine(cell, c, a);
if (i3 == CELL_OBJECT_INTERSECT)
return CELL_OBJECT_INTERSECT;
ASSERT_EQUAL(i1, i2, (cell, a, b, c));
ASSERT_EQUAL(i2, i3, (cell, a, b, c));
ASSERT_EQUAL(i3, i1, (cell, a, b, c));
return i1;
}
template <class CellIdT, typename IntersectF>
void CoverObject(IntersectF const & intersect, uint64_t cellPenaltyArea, vector<CellIdT> & out,
CellIdT cell = CellIdT::Root())
{
uint64_t const cellArea = my::sq(uint64_t(1 << (CellIdT::DEPTH_LEVELS - 1 - cell.Level())));
CellObjectIntersection const intersection = intersect(cell);
if (intersection == CELL_OBJECT_NO_INTERSECTION)
return;
if (intersection == CELL_INSIDE_OBJECT ||
cell.Level() == CellIdT::DEPTH_LEVELS - 1 ||
cellPenaltyArea >= cellArea)
{
out.push_back(cell);
return;
}
vector<CellIdT> subdiv;
for (uint8_t i = 0; i < 4; ++i)
CoverObject(intersect, cellPenaltyArea, subdiv, cell.Child(i));
uint64_t subdivArea = 0;
for (size_t i = 0; i < subdiv.size(); ++i)
subdivArea += my::sq(uint64_t(1 << (CellIdT::DEPTH_LEVELS - 1 - subdiv[i].Level())));
ASSERT(!subdiv.empty(), (cellPenaltyArea, out, cell));
if (subdiv.empty() ||
cellPenaltyArea * (int(subdiv.size()) - 1) >= cellArea - subdivArea)
out.push_back(cell);
else
out.insert(out.end(), subdiv.begin(), subdiv.end());
}
} // namespace covering

View file

@ -1,121 +0,0 @@
#pragma once
#include "covering.hpp"
#include "../base/assert.hpp"
#include "../base/logging.hpp"
#include "../std/algorithm.hpp"
#include "../std/deque.hpp"
#include "../std/iterator.hpp"
#include "../std/map.hpp"
#include "../std/utility.hpp"
namespace covering
{
template <class CellIdT, typename ValueT, typename SinkT>
class CoveringStreamOptimizer
{
public:
typedef CellIdT CellId;
CoveringStreamOptimizer(SinkT & sink, uint32_t windowSize, uint32_t maxDuplicates)
: m_Sink(sink), m_WindowSize(windowSize), m_MaxDuplicates(maxDuplicates),
m_LastPopped(0), m_LastPushed(0)
{
CHECK_GREATER_OR_EQUAL(windowSize, maxDuplicates, ());
}
~CoveringStreamOptimizer()
{
Flush();
}
void Add(int64_t id, ValueT value)
{
CHECK_LESS_OR_EQUAL(m_LastPushed, id, (value));
m_LastPushed = id;
m_Buffer.push_back(make_pair(id, value));
if (++m_ValueCounts[value] >= m_MaxDuplicates && m_Buffer.size() >= m_WindowSize)
ProcessWindow(value);
while (m_Buffer.size() >= m_WindowSize)
PopFront();
}
void Flush()
{
while (!m_Buffer.empty())
PopFront();
}
static void Optimize(vector<int64_t> & ids, int64_t minId)
{
CHECK_GREATER(ids.size(), 2, (minId, ids));
CHECK(IsSortedAndUnique(ids.begin(), ids.end()), (minId, ids));
CHECK_GREATER_OR_EQUAL(ids[0], minId, (minId, ids));
vector<CellId> cells(ids.size(), CellId(""));
for (size_t i = 0; i < ids.size(); ++i)
cells[i] = CellId::FromInt64(ids[i]);
covering::Covering<CellId> covering(cells, minId);
covering.Simplify(minId);
vector<int64_t> res;
covering.OutputToVector(res);
sort(res.begin(), res.end());
CHECK(IsSortedAndUnique(res.begin(), res.end()), (minId, ids, res));
CHECK_GREATER_OR_EQUAL(res[0], minId, (minId, ids, res));
ids.swap(res);
}
private:
void PopFront()
{
CHECK_LESS_OR_EQUAL(m_LastPopped, m_Buffer.front().first, ());
m_LastPopped = m_Buffer.front().first;
m_Sink(m_Buffer.front().first, m_Buffer.front().second);
typename map<ValueT, uint32_t>::iterator it = m_ValueCounts.find(m_Buffer.front().second);
CHECK(it != m_ValueCounts.end(), (m_Buffer.front().second))
CHECK_GREATER(it->second, 0, (m_Buffer.front().second));
if (--(it->second) == 0)
m_ValueCounts.erase(it);
m_Buffer.pop_front();
}
void ProcessWindow(ValueT value)
{
CHECK(IsSortedAndUnique(m_Buffer.begin(), m_Buffer.end()), ());
vector<pair<int64_t, ValueT> > others;
vector<int64_t> ids;
for (typename BufferType::const_iterator it = m_Buffer.begin(); it != m_Buffer.end(); ++it)
{
if (it->second == value)
ids.push_back(it->first);
else
others.push_back(*it);
}
CHECK_GREATER_OR_EQUAL(ids.size(), m_MaxDuplicates, ());
Optimize(ids, m_LastPopped);
CHECK(IsSortedAndUnique(ids.begin(), ids.end()), (ids));
vector<pair<int64_t, ValueT> > optimized(ids.size(), make_pair(0LL, value));
for (size_t i = 0; i < ids.size(); ++i)
optimized[i].first = ids[i];
CHECK(IsSortedAndUnique(others.begin(), others.end()), (others));
CHECK(IsSortedAndUnique(optimized.begin(), optimized.end()), (optimized));
m_Buffer.clear();
set_union(others.begin(), others.end(),
optimized.begin(), optimized.end(),
back_inserter(m_Buffer));
m_ValueCounts[value] = optimized.size();
}
SinkT & m_Sink;
uint32_t const m_WindowSize;
uint32_t const m_MaxDuplicates;
typedef deque<pair<int64_t, ValueT> > BufferType;
BufferType m_Buffer;
map<ValueT, uint32_t> m_ValueCounts;
int64_t m_LastPopped, m_LastPushed;
};
}

View file

@ -23,7 +23,6 @@ HEADERS += \
cellid.hpp \
rect_intersect.hpp \
covering.hpp \
covering_stream_optimizer.hpp \
packer.hpp \
pointu_to_uint64.hpp \
simplification.hpp \

View file

@ -1,63 +0,0 @@
#include "../../testing/testing.hpp"
#include "../cellid.hpp"
#include "../covering.hpp"
#include "../covering_stream_optimizer.hpp"
#include "../../base/macros.hpp"
typedef m2::CellId<5> CellId;
namespace
{
struct TestSink
{
vector<pair<int64_t, int> > & m_Res;
TestSink(vector<pair<int64_t, int> > & res) : m_Res(res) {}
void operator() (int64_t cell, int value) const
{
m_Res.push_back(make_pair(cell, value));
}
};
}
UNIT_TEST(CoveringStreamOptimizer_Smoke)
{
// my::g_LogLevel = LDEBUG;
vector<pair<int64_t, int> > res1;
TestSink sink(res1);
covering::CoveringStreamOptimizer<CellId, int, TestSink> optimizer(sink, 5, 4);
optimizer.Add(CellId("01").ToInt64(), 1);
optimizer.Add(CellId("01").ToInt64(), 2);
optimizer.Add(CellId("0200").ToInt64(), 1);
optimizer.Add(CellId("0201").ToInt64(), 1);
optimizer.Add(CellId("0202").ToInt64(), 1);
optimizer.Add(CellId("021").ToInt64(), 1);
optimizer.Add(CellId("022").ToInt64(), 1);
optimizer.Flush();
vector<pair<CellId, int> > e, res;
for (size_t i = 0; i < res1.size(); ++i)
res.push_back(make_pair(CellId::FromInt64(res1[i].first), res1[i].second));
e.push_back(make_pair(CellId("01"), 1));
e.push_back(make_pair(CellId("01"), 2));
e.push_back(make_pair(CellId("02"), 1));
TEST_EQUAL(e, res, ());
}
UNIT_TEST(CoveringStreamOptimizer_MinId1)
{
int64_t const minId = 72200890973LL;
int64_t ids[] = { 72200890973, 72200890995, 72200891005, 72200891011, 72200891013,
72200891014, 72200891015, 72200891036, 72200896894, 72200896902 };
vector<int64_t> idsV(&ids[0], &ids[0] + ARRAY_SIZE(ids));
covering::CoveringStreamOptimizer<m2::CellId<19>, int, TestSink>::Optimize(idsV, minId);
}
UNIT_TEST(CoveringStreamOptimizer_MinId2)
{
int64_t const minId = 72200890973LL;
int64_t ids[] = { 72200890973, 72200890994, 72200891015, 72200891036};
vector<int64_t> idsV(&ids[0], &ids[0] + ARRAY_SIZE(ids));
covering::CoveringStreamOptimizer<m2::CellId<19>, int, TestSink>::Optimize(idsV, minId);
}

View file

@ -1,127 +1,4 @@
#include "../../testing/testing.hpp"
#include "../cellid.hpp"
#include "../covering.hpp"
#include "../point2d.hpp"
#include "../../base/stl_add.hpp"
typedef m2::CellId<5> CellId;
UNIT_TEST(CoverTriangle_Simple)
{
vector<CellId> v;
covering::Covering<CellId> c(m2::PointD(3*2, 3*2), m2::PointD(4*2, 12*2), m2::PointD(14*2, 3*2));
c.OutputToVector(v);
vector<CellId> e;
e.push_back(CellId("03"));
e.push_back(CellId("003"));
e.push_back(CellId("012"));
e.push_back(CellId("013"));
e.push_back(CellId("102"));
e.push_back(CellId("103"));
e.push_back(CellId("112"));
e.push_back(CellId("120"));
e.push_back(CellId("121"));
e.push_back(CellId("122"));
e.push_back(CellId("210"));
e.push_back(CellId("211"));
e.push_back(CellId("212"));
e.push_back(CellId("0211"));
e.push_back(CellId("0213"));
e.push_back(CellId("0231"));
e.push_back(CellId("0233"));
e.push_back(CellId("1130"));
e.push_back(CellId("1132"));
e.push_back(CellId("1230"));
e.push_back(CellId("1300"));
e.push_back(CellId("2011"));
e.push_back(CellId("2013"));
e.push_back(CellId("2031"));
e.push_back(CellId("2033"));
e.push_back(CellId("2130"));
e.push_back(CellId("2211"));
e.push_back(CellId("2300"));
e.push_back(CellId("3000"));
TEST_EQUAL(v, e, ());
}
UNIT_TEST(CoverTriangle_TriangleInsideCell)
{
vector<CellId> v;
covering::Covering<CellId> c(m2::PointD(0.1, 0.1), m2::PointD(0.2, 0.2), m2::PointD(0.2, 0.4));
c.OutputToVector(v);
vector<CellId> e;
e.push_back(CellId("0000"));
TEST_EQUAL(v, e, ());
}
UNIT_TEST(Covering_Append_Simple)
{
vector<CellId> v1, v2, v3;
v1.push_back(CellId("012"));
v2.push_back(CellId("0123"));
v3.push_back(CellId("012"));
v1.push_back(CellId("023"));
v2.push_back(CellId("023"));
v3.push_back(CellId("023"));
v1.push_back(CellId("130"));
v1.push_back(CellId("131"));
v2.push_back(CellId("133"));
v1.push_back(CellId("1320"));
v1.push_back(CellId("1321"));
v2.push_back(CellId("1322"));
v2.push_back(CellId("1323"));
v3.push_back(CellId("13"));
covering::Covering<CellId> c1(v1);
c1.Append(covering::Covering<CellId>(v2));
vector<CellId> v4;
c1.OutputToVector(v4);
sort(v4.begin(), v4.end(), CellId::LessStackOrder());
TEST_EQUAL(v3, v4, ());
}
UNIT_TEST(IntersectCellWithTriangle_EmptyTriangle)
{
m2::PointU pt(27, 31);
TEST_EQUAL(covering::CELL_TRIANGLE_NO_INTERSECTION,
covering::IntersectCellWithTriangle(CellId("0"), pt, pt, pt), ());
TEST_EQUAL(covering::CELL_TRIANGLE_NO_INTERSECTION,
covering::IntersectCellWithTriangle(CellId("1"), pt, pt, pt), ());
TEST_EQUAL(covering::CELL_TRIANGLE_NO_INTERSECTION,
covering::IntersectCellWithTriangle(CellId("2"), pt, pt, pt), ());
TEST_EQUAL(covering::TRIANGLE_INSIDE_CELL,
covering::IntersectCellWithTriangle(CellId("3"), pt, pt, pt), ());
}
UNIT_TEST(Covering_EmptyTriangle)
{
m2::PointU pt(27, 31);
CellId const expectedCellId = CellId::FromXY(pt.x, pt.y);
TEST_GREATER(expectedCellId.ToInt64(), 5, ());
covering::Covering<CellId> covering(pt, pt, pt);
vector<CellId> ids;
covering.OutputToVector(ids);
TEST_EQUAL(ids, vector<CellId>(1, expectedCellId), ());
}
UNIT_TEST(Covering_Simplify_Smoke)
{
vector<CellId> v;
v.push_back(CellId("03"));
v.push_back(CellId("020"));
v.push_back(CellId("021"));
v.push_back(CellId("022"));
v.push_back(CellId("0012"));
covering::Covering<CellId> covering(v);
v.clear();
covering.Simplify();
covering.OutputToVector(v);
vector<CellId> e;
e.push_back(CellId("02"));
e.push_back(CellId("03"));
e.push_back(CellId("0012"));
TEST_EQUAL(v, e, ());
}
// TODO: Add covering unit tests here.

View file

@ -26,7 +26,6 @@ SOURCES += \
packer_test.cpp \
segments_intersect_test.cpp \
covering_test.cpp \
covering_stream_optimizer_test.cpp \
pointu_to_uint64_test.cpp \
simplification_test.cpp \
transformations_test.cpp \

View file

@ -5,126 +5,7 @@
#include "../std/queue.hpp"
#include "../std/vector.hpp"
inline bool IntersectsHoriz(CoordT x1, CoordT y1, CoordT x2, CoordT y2,
CoordT y, CoordT l, CoordT r)
{
CoordT d = (y1 - y) * (y2 - y);
if (d > 0) return false;
CoordT x = x1 + (x2 - x1) * (y - y1) / (y2 - y1);
if ((l - x) * (r - x) <= 0) return true; else return false;
}
inline bool IntersectsVert(CoordT x1, CoordT y1, CoordT x2, CoordT y2,
CoordT x, CoordT b, CoordT t)
{
CoordT d = (x1 - x) * (x2 - x);
if (d > 0) return false;
CoordT y = y1 + (y2 - y1) * (x - x1) / (x2 - x1);
if ((b - y) * (t - y) <= 0) return true; else return false;
}
template <typename BoundsT, typename CellIdT>
inline bool CellIntersects(vector<CoordPointT> const & polyLine, CellIdT id)
{
CoordT minX, minY, maxX, maxY;
CellIdConverter<BoundsT, CellIdT>::GetCellBounds(id, minX, minY, maxX, maxY);
CoordPointT minPoint = make_pair(minX, minY);
CoordPointT maxPoint = make_pair(maxX, maxY);
for (size_t i = 0; i < polyLine.size() - 1; ++i)
{
if (IntersectsHoriz(polyLine[i].first, polyLine[i].second,
polyLine[i + 1].first, polyLine[i + 1].second,
minPoint.second, minPoint.first, maxPoint.first)) return true;
if (IntersectsHoriz(polyLine[i].first, polyLine[i].second,
polyLine[i + 1].first, polyLine[i + 1].second,
maxPoint.second, minPoint.first, maxPoint.first)) return true;
if (IntersectsVert(polyLine[i].first, polyLine[i].second,
polyLine[i + 1].first, polyLine[i + 1].second,
minPoint.first, minPoint.second, maxPoint.second)) return true;
if (IntersectsVert(polyLine[i].first, polyLine[i].second,
polyLine[i + 1].first, polyLine[i + 1].second,
maxPoint.first, minPoint.second, maxPoint.second)) return true;
}
return false;
}
template <typename BoundsT, typename CellIdT>
inline void SplitCell(vector<CoordPointT> const & polyLine, queue<CellIdT> & cellQueue)
{
CellIdT id = cellQueue.front();
cellQueue.pop();
for (size_t i = 0; i < 4; ++i)
{
CellIdT child = id.Child(i);
if (CellIntersects<BoundsT>(polyLine, child))
{
cellQueue.push(child);
}
}
}
template <typename ItT>
inline bool FindBounds(ItT begin, ItT end,
CoordT & minX, CoordT & minY, CoordT & maxX, CoordT & maxY)
{
if (begin == end) return false;
minX = begin->first;
maxX = begin->first;
minY = begin->second;
maxY = begin->second;
for (ItT it = begin; it != end; ++it)
{
if (it->first < minX) minX = it->first;
if (it->first > maxX) maxX = it->first;
if (it->second < minY) minY = it->second;
if (it->second > maxY) maxY = it->second;
}
return true;
}
template <typename BoundsT, typename CellIdT>
inline CellIdT CoverPoint(CoordPointT const & point)
{
return CellIdConverter<BoundsT, CellIdT>::ToCellId(point.first, point.second);
}
template <typename BoundsT, typename CellIdT>
inline void CoverPolyLine(vector< CoordPointT > const & polyLine, size_t cellDepth,
vector<CellIdT> & cells)
{
CoordT minX = 0, minY = 0, maxX = 0, maxY = 0;
FindBounds(polyLine.begin(), polyLine.end(), minX, minY, maxX, maxY);
CellIdT commonCell =
CellIdConverter<BoundsT, CellIdT>::Cover2PointsWithCell(minX, minY, maxX, maxY);
queue<CellIdT> cellQueue;
cellQueue.push(commonCell);
while (cellQueue.front().Level() < static_cast<int>(cellDepth)) // cellQueue.size() < cells_count
{
SplitCell<BoundsT>(polyLine, cellQueue);
}
while (!cellQueue.empty())
{
cells.push_back(cellQueue.front());
cellQueue.pop();
}
}
// TODO: Move neccessary functions to geometry/covering.hpp and delete this file.
template <typename BoundsT, typename CellIdT>
inline void SplitRectCell(CellIdT id,

View file

@ -8,90 +8,79 @@
#include "../std/algorithm.hpp"
#include "../std/bind.hpp"
// TODO: Redo polyline covering right.
namespace
{
template <class BoundsT, class CoveringT>
class TriangleCoverer
{
public:
typedef typename CoveringT::CellId CellId;
typedef CellIdConverter<BoundsT, CellId> CellIdConverterType;
explicit TriangleCoverer(CoveringT const & covering) : m_Covering(1, covering) {}
void operator () (m2::PointD const & a, m2::PointD const & b, m2::PointD const & c)
{
AddTriangle(a, b, c);
}
void AddTriangle(m2::PointD const & a, m2::PointD const & b, m2::PointD const & c)
{
m_Covering.push_back(CoveringT(
m2::PointD(CellIdConverterType::XToCellIdX(a.x), CellIdConverterType::YToCellIdY(a.y)),
m2::PointD(CellIdConverterType::XToCellIdX(b.x), CellIdConverterType::YToCellIdY(b.y)),
m2::PointD(CellIdConverterType::XToCellIdX(c.x), CellIdConverterType::YToCellIdY(c.y))));
while (m_Covering.size() > 1 &&
m_Covering[m_Covering.size() - 2].Size() < m_Covering.back().Size())
{
m_Covering[m_Covering.size() - 2].Append(m_Covering.back());
m_Covering.pop_back();
}
}
CoveringT const & GetCovering()
{
ASSERT(!m_Covering.empty(), ());
#ifdef DEBUG
// Make appends in another order and assert that result is the same.
CoveringT dbgCovering(m_Covering[0]);
for (size_t i = 1; i < m_Covering.size(); ++i)
dbgCovering.Append(m_Covering[i]);
#endif
while (m_Covering.size() > 1)
{
m_Covering[m_Covering.size() - 2].Append(m_Covering.back());
m_Covering.pop_back();
}
#ifdef DEBUG
vector<CellId> dbgIds, ids;
dbgCovering.OutputToVector(dbgIds);
m_Covering[0].OutputToVector(ids);
ASSERT_EQUAL(dbgIds, ids, ());
#endif
return m_Covering[0];
}
private:
vector<CoveringT> m_Covering;
};
}
vector<int64_t> covering::CoverFeature(FeatureType const & f, int level)
class FeatureIntersector
{
vector<CoordPointT> geometry;
f.ForEachPoint(MakeBackInsertFunctor(geometry), level);
vector<RectId> ids;
if (!geometry.empty())
public:
struct Trg
{
if (geometry.size() > 1)
{
// TODO: Tweak CoverPolyLine() depth level.
CoverPolyLine<MercatorBounds, RectId>(geometry, RectId::DEPTH_LEVELS - 1, ids);
}
else
ids.push_back(CoverPoint<MercatorBounds, RectId>(geometry[0]));
m2::PointD m_A, m_B, m_C;
Trg(m2::PointU a, m2::PointU b, m2::PointU c) : m_A(a), m_B(b), m_C(c) {}
};
vector<m2::PointD> m_Polyline;
vector<Trg> m_Trg;
covering::CellObjectIntersection operator() (RectId const & cell) const
{
for (size_t i = 1; i < m_Polyline.size(); ++i)
if (covering::IntersectCellWithLine(cell, m_Polyline[i], m_Polyline[i-1]))
return covering::CELL_OBJECT_INTERSECT;
for (size_t i = 0; i < m_Trg.size(); ++i)
if (covering::CellObjectIntersection res =
covering::IntersectCellWithTriangle(cell, m_Trg[i].m_A, m_Trg[i].m_B, m_Trg[i].m_C))
return res == covering::OBJECT_INSIDE_CELL ? covering::CELL_OBJECT_INTERSECT : res;
return covering::CELL_OBJECT_NO_INTERSECTION;
}
typedef covering::Covering<RectId> CoveringType;
typedef TriangleCoverer<MercatorBounds, CoveringType> CovererType;
CovererType coverer = CovererType(CoveringType(ids));
f.ForEachTriangleRef(coverer, level);
typedef CellIdConverter<MercatorBounds, RectId> CellIdConverterType;
static m2::PointD ConvertPoint(double x, double y)
{
return m2::PointD(CellIdConverterType::XToCellIdX(x),
CellIdConverterType::YToCellIdY(y));
}
void operator() (pair<double, double> const & pt)
{
m_Polyline.push_back(ConvertPoint(pt.first, pt.second));
}
void operator() (m2::PointD const & a, m2::PointD const & b, m2::PointD const & c)
{
m_Trg.push_back(Trg(ConvertPoint(a.x, a.y),
ConvertPoint(b.x, b.y),
ConvertPoint(c.x, c.y)));
}
};
}
vector<int64_t> covering::CoverFeature(FeatureType const & f,
uint64_t cellPenaltyAreaPerNode)
{
FeatureIntersector featureIntersector;
f.ForEachPointRef(featureIntersector, -1);
f.ForEachTriangleRef(featureIntersector, -1);
CHECK(!featureIntersector.m_Trg.empty() || !featureIntersector.m_Polyline.empty(), \
(f.DebugString(-1)));
if (featureIntersector.m_Trg.empty() && featureIntersector.m_Polyline.size() == 1)
{
m2::PointD pt = featureIntersector.m_Polyline[0];
return vector<int64_t>(
1, RectId::FromXY(static_cast<uint32_t>(pt.x), static_cast<uint32_t>(pt.y)).ToInt64());
}
vector<RectId> cells;
covering::CoverObject(featureIntersector, cellPenaltyAreaPerNode, cells);
vector<int64_t> res(cells.size());
for (size_t i = 0; i < cells.size(); ++i)
res[i] = cells[i].ToInt64();
vector<int64_t> res;
coverer.GetCovering().OutputToVector(res);
return res;
}

View file

@ -12,7 +12,8 @@ class FeatureType;
namespace covering
{
// Cover feature with RectIds and return their integer representations.
vector<int64_t> CoverFeature(FeatureType const & feature, int level);
vector<int64_t> CoverFeature(FeatureType const & feature,
uint64_t cellPenaltyAreaPerNode);
// Cover viewport with RectIds and append their RectIds as well.
vector<pair<int64_t, int64_t> > CoverViewportAndAppendLowerLevels(m2::RectD const & rect);
// Given a vector of intervals [a, b), sort them and merge overlapping intervals.

View file

@ -5,97 +5,16 @@
#include "../../coding/hex.hpp"
#include "../../base/logging.hpp"
// Unit test uses m2::CellId<30> for historical reasons, the actual production code uses RectId.
typedef m2::CellId<30> CellIdT;
typedef Bounds<-180, -90, 180, 90> OrthoBounds;
namespace
{
class CoordsPusher
{
public:
typedef vector< CoordPointT > VT;
CoordsPusher(VT & v) : m_v(v) {}
CoordsPusher & operator()(CoordT x, CoordT y)
{
m_v.push_back(make_pair(x, y));
return *this;
}
private:
VT & m_v;
};
string EnumCells(vector<CellIdT> const & v)
{
string result;
for (size_t i = 0; i < v.size(); ++i)
{
result += v[i].ToString();
if (i != v.size() - 1) result += ' ';
}
return result;
}
}
UNIT_TEST(CellIdToStringRecode)
{
char const kTest[] = "21032012203";
TEST_EQUAL(CellIdT::FromString(kTest).ToString(), kTest, ());
}
UNIT_TEST(GoldenTestCover)
{
vector<CoordPointT> coords;
CoordsPusher c(coords);
c(0.7, 0.5)
(1.5, 1.5)
(2.5, 3.5)
(5.5, 5.0);
vector<CellIdT> cells;
CoverPolyLine<Bounds<0, 0, 8, 8> >(coords, 3, cells);
TEST_EQUAL(EnumCells(cells), "000 001 003 021 030 032 033 211 300 301 303", ());
}
UNIT_TEST(GoldenTestCellIntersect)
{
vector< CoordPointT > coords;
CoordsPusher c(coords);
c (0.7, 0.5)
(1.5, 1.5)
(2.5, 3.5)
(5.5, 5.0);
vector<CellIdT> cells;
typedef Bounds<0, 0, 8, 8> BoundsT;
CoverPolyLine<BoundsT>(coords, 7, cells);
TEST(!CellIntersects<BoundsT>(coords, CellIdT::FromString("210")), ());
}
UNIT_TEST(GoldenOrthoCover)
{
vector< CoordPointT > coords;
CoordsPusher c(coords);
c
(27.545927047729492, 53.888893127441406)
(27.546476364135742, 53.888614654541016)
(27.546852111816406, 53.889347076416016)
(27.546596527099609, 53.889404296875000)
(27.545927047729492, 53.888893127441406);
vector<CellIdT> cells;
CoverPolyLine<OrthoBounds>(coords, 19, cells);
TEST_EQUAL(EnumCells(cells),
"3201221130210310103 3201221130210310120 3201221130210310121 3201221130210310122 "
"3201221130210310123 3201221130210310301 3201221130210310310", ());
}
UNIT_TEST(GoldenCoverRect)
{
vector<CellIdT> cells;
@ -123,40 +42,3 @@ UNIT_TEST(ArtificialCoverRect)
TEST_EQUAL(cells[2].ToString(), "21", ());
TEST_EQUAL(cells[3].ToString(), "30", ());
}
UNIT_TEST(CoverEmptyTriangleTest)
{
vector<int64_t> ids, expectedIds;
m2::PointD pt(8.89748, 51.974);
expectedIds.push_back(CoverPoint<MercatorBounds, RectId>(CoordPointT(pt.x, pt.y)).ToInt64());
TEST_NOT_EQUAL(expectedIds[0], 1, ());
typedef CellIdConverter<MercatorBounds, RectId> CellIdConverterType;
m2::PointD pt1(CellIdConverterType::XToCellIdX(pt.x), CellIdConverterType::YToCellIdY(pt.y));
covering::Covering<RectId> covering(pt1, pt1, pt1);
covering.OutputToVector(ids);
TEST_EQUAL(ids, expectedIds, ());
}
// TODO: UNIT_TEST(CoverPolygon)
/*
UNIT_TEST(CoverPolygon)
{
typedef Bounds<0, 0, 16, 16> TestBounds;
vector< CoordPointT > coords;
CoordsPusher c(coords);
c
(4.1, 4.1)
(6.1, 8.1)
(10.1, 10.1)
(8.1, 6.1)
(4.1, 4.1);
vector<CellIdT> cells;
CoverPolygon<TestBounds>(coords, 4, cells);
TEST_EQUAL(EnumCells(cells),
"0300 0301 0302 0303 0312 0313 0321 0330 0331 1220 0323 0332 0333 "
"1222 1223 2110 2111 3000 3001 2113 3002 3003 3012 3021 3030", ());
}
*/

View file

@ -24,7 +24,8 @@ UNIT_TEST(BuildIndexTest)
{
FeaturesVector featuresVector(originalContainer);
MemWriter<vector<char> > serialWriter(serialIndex);
indexer::BuildIndex(ScaleIndexBase::NUM_BUCKETS, featuresVector, serialWriter, "build_index_test");
indexer::BuildIndex(ScaleIndexBase::NUM_BUCKETS, featuresVector, serialWriter,
"build_index_test");
}
// Create a new mwm file.

View file

@ -5,8 +5,6 @@
#include "interval_index_builder.hpp"
#include "cell_id.hpp"
#include "../geometry/covering_stream_optimizer.hpp"
#include "../coding/dd_vector.hpp"
#include "../coding/file_sort.hpp"
#include "../coding/var_serial_vector.hpp"
@ -71,7 +69,7 @@ public:
{
if (FeatureShouldBeIndexed(f))
{
vector<int64_t> const cells = covering::CoverFeature(f, GetGeometryScale());
vector<int64_t> const cells = covering::CoverFeature(f, 1025);
for (vector<int64_t>::const_iterator it = cells.begin(); it != cells.end(); ++it)
m_Sorter.Add(CellFeaturePair(*it, offset));
++m_NumFeatures;
@ -112,31 +110,6 @@ private:
SinkT & m_Sink;
};
template <class SinkT>
class FeatureCoveringOptimizeProxySink
{
public:
FeatureCoveringOptimizeProxySink(SinkT & sink)
: m_Sink(sink), m_Optimizer(m_Sink, 100, 10)
{
}
~FeatureCoveringOptimizeProxySink()
{
m_Optimizer.Flush();
}
void operator () (CellFeaturePair const & cellFeaturePair)
{
m_Optimizer.Add(cellFeaturePair.GetCell(), cellFeaturePair.GetFeature());
}
private:
CellFeaturePairSinkAdapter<SinkT> m_Sink;
covering::CoveringStreamOptimizer<RectId, uint32_t, CellFeaturePairSinkAdapter<SinkT> >
m_Optimizer;
};
template <class FeaturesVectorT, class WriterT>
inline void IndexScales(uint32_t bucketsCount,
FeaturesVectorT const & featuresVector,
@ -156,16 +129,11 @@ inline void IndexScales(uint32_t bucketsCount,
{
FileWriter cellsToFeaturesWriter(tmpFilePrefix + ".c2f.sorted");
typedef FeatureCoveringOptimizeProxySink<FileWriter> OptimizeSink;
OptimizeSink optimizeSink(cellsToFeaturesWriter);
typedef FileSorter<CellFeaturePair, OptimizeSink> SorterType;
SorterType sorter(1024*1024, tmpFilePrefix + ".c2f.tmp", optimizeSink);
/*
typedef FileSorter<CellFeaturePair, WriterFunctor<FileWriter> > SorterType;
WriterFunctor<FileWriter> out(cellsToFeaturesWriter);
SorterType sorter(1024*1024, tmpFilePrefix + ".c2f.tmp", out);
*/
featuresVector.ForEachOffset(FeatureCoverer<SorterType>(bucket, sorter, numFeatures));
featuresVector.ForEachOffset(
FeatureCoverer<SorterType>(bucket, sorter, numFeatures));
// LOG(LINFO, ("Sorting..."));
sorter.SortAndFinish();
}