diff --git a/base/stats.hpp b/base/stats.hpp index 974c5f7703..407f07e25c 100644 --- a/base/stats.hpp +++ b/base/stats.hpp @@ -20,35 +20,36 @@ template class AverageStats { public: - AverageStats() : m_Count(0), m_Sum(0) {} + AverageStats() = default; - void operator() (T const & x) + template + AverageStats(Iter begin, Iter end) { - ++m_Count; - m_Sum += x; + for (auto it = begin; it != end; ++it) + operator()(*it); + } + + void operator()(T const & x) + { + ++m_count; + m_sum += x; } std::string GetStatsStr() const { std::ostringstream out; - out << "N: " << m_Count; - if (m_Count > 0) - out << " Av: " << m_Sum / m_Count; + out << "N: " << m_count; + if (m_count > 0) + out << " Avg: " << GetAverage(); return out.str(); } - double GetAverage() const - { - return m_Count ? m_Sum / m_Count : 0.0; - } + double GetAverage() const { return m_count == 0 ? 0.0 : m_sum / static_cast(m_count); } - uint32_t GetCount() const - { - return m_Count; - } + uint32_t GetCount() const { return m_count; } private: - uint32_t m_Count; - double m_Sum; + uint32_t m_count = 0; + double m_sum = 0.0; }; } // namespace base diff --git a/geometry/geometry_tests/CMakeLists.txt b/geometry/geometry_tests/CMakeLists.txt index 736c4041f6..237e5e089b 100644 --- a/geometry/geometry_tests/CMakeLists.txt +++ b/geometry/geometry_tests/CMakeLists.txt @@ -29,7 +29,7 @@ set( polygon_test.cpp rect_test.cpp region2d_binary_op_test.cpp - region_test.cpp + region_tests.cpp robust_test.cpp screen_test.cpp segments_intersect_test.cpp diff --git a/geometry/geometry_tests/region_test.cpp b/geometry/geometry_tests/region_tests.cpp similarity index 73% rename from geometry/geometry_tests/region_test.cpp rename to geometry/geometry_tests/region_tests.cpp index fccfd24127..ed733508ff 100644 --- a/geometry/geometry_tests/region_test.cpp +++ b/geometry/geometry_tests/region_tests.cpp @@ -2,8 +2,17 @@ #include "base/macros.hpp" +#include "geometry/convex_hull.hpp" +#include "geometry/point2d.hpp" #include "geometry/region2d.hpp" +#include +#include +#include +#include + +using namespace std; + namespace { template @@ -213,7 +222,7 @@ UNIT_TEST(Region_Contains_double) UNIT_TEST(Region_ForEachPoint) { - typedef m2::PointF P; + using P = m2::PointF; P const points[] = {P(0.0, 1.0), P(1.0, 2.0), P(10.5, 11.5)}; m2::Region

region(points, points + ARRAY_SIZE(points)); @@ -225,7 +234,7 @@ UNIT_TEST(Region_ForEachPoint) UNIT_TEST(Region_point_at_border_test) { - typedef m2::PointF P; + using P = m2::PointF; P const points[] = {P(0.0, 1.0), P(0.0, 10.0), P(10.0, 10.0), P(10.0, 1.0)}; m2::Region

region(points, points + ARRAY_SIZE(points)); @@ -244,7 +253,7 @@ UNIT_TEST(Region_point_at_border_test) UNIT_TEST(Region_border_intersecion_Test) { - typedef m2::PointF P; + using P = m2::PointF; P const points[] = {P(0.0, 1.0), P(0.0, 10.0), P(10.0, 10.0), P(10.0, 1.0)}; m2::Region

region(points, points + ARRAY_SIZE(points)); @@ -262,3 +271,84 @@ UNIT_TEST(Region_border_intersecion_Test) TEST(!region.FindIntersection(P(5.0, 5.0), P(2.0, 2.0), intersection), ("This case has no intersection")); } + +UNIT_TEST(Region_Area) +{ + using P = m2::PointD; + + { + m2::Region

region; + TEST_EQUAL(region.CalculateArea(), 0.0, ()); + } + { + // Counterclockwise. + P const points[] = {P(0.0, 0.0), P(1.0, 0.0), P(1.0, 1.0), P(0.0, 1.0)}; + m2::Region

region(points, points + ARRAY_SIZE(points)); + TEST_EQUAL(region.CalculateArea(), 1.0, ()); + } + { + // Clockwise. + P const points[] = {P(0.0, 0.0), P(0.0, 1.0), P(1.0, 1.0), P(1.0, 0.0)}; + m2::Region

region(points, points + ARRAY_SIZE(points)); + TEST_EQUAL(region.CalculateArea(), 1.0, ()); + } + { + // Non-convex. + P const points[] = {P(0.0, 0.0), P(1.0, 0.0), P(1.0, 1.0), P(0.5, 0.5), P(0.0, 1.0)}; + m2::Region

region(points, points + ARRAY_SIZE(points)); + TEST_EQUAL(region.CalculateArea(), 0.75, ()); + } +} + +UNIT_TEST(Region_GetRandomPoint) +{ + using P = m2::PointD; + + // Run several iterations of Monte-Carlo and check that areas are similar. + size_t const kNumIterations = 1000; + bool const kNeedPlot = true; + + auto testConvexRegion = [&](m2::Region

const & region) { + minstd_rand rng(0); + vector

points; + points.reserve(kNumIterations); + for (size_t i = 0; i < kNumIterations; ++i) + points.emplace_back(region.GetRandomPoint(rng)); + m2::ConvexHull const hull(points, 1e-9 /* eps */); + auto const hullRegion = m2::Region

(hull.Points().begin(), hull.Points().end()); + LOG(LINFO, (hullRegion.CalculateArea())); + TEST(base::AlmostEqualRel(region.CalculateArea(), hullRegion.CalculateArea(), 0.05), ()); + + if (kNeedPlot) + { + cout << "import matplotlib.pyplot as plt" << endl; + cout << endl; + + cout << "x = ["; + for (size_t i = 0; i < points.size(); i++) + cout << points[i].x << ","; + cout << "]" << endl; + + cout << "y = ["; + for (size_t i = 0; i < points.size(); i++) + cout << points[i].y << ","; + cout << "]" << endl; + + cout << endl; + cout << "plt.scatter(x, y)" << endl; + cout << "plt.show()" << endl; + } + }; + + { + P const points[] = {P(0.0, 0.0), P(1.0, 0.0), P(1.0, 1.0), P(0.0, 1.0)}; + m2::Region

region(points, points + ARRAY_SIZE(points)); + testConvexRegion(region); + } + + { + P const points[] = {P(0.0, -1.0), P(1.0, 0.0), P(0.0, 1.0), P(-1.0, 0.0)}; + m2::Region

region(points, points + ARRAY_SIZE(points)); + testConvexRegion(region); + } +} diff --git a/geometry/rect2d.hpp b/geometry/rect2d.hpp index 1e87a7c6c4..09fba91329 100644 --- a/geometry/rect2d.hpp +++ b/geometry/rect2d.hpp @@ -160,6 +160,7 @@ public: Point Center() const { return Point((m_minX + m_maxX) / 2.0, (m_minY + m_maxY) / 2.0); } T SizeX() const { return std::max(static_cast(0), m_maxX - m_minX); } T SizeY() const { return std::max(static_cast(0), m_maxY - m_minY); } + T Area() const { return SizeX() * SizeY(); } void DivideByGreaterSize(Rect & r1, Rect & r2) const { diff --git a/geometry/region2d.hpp b/geometry/region2d.hpp index fa60739806..dbdb501e2d 100644 --- a/geometry/region2d.hpp +++ b/geometry/region2d.hpp @@ -4,9 +4,11 @@ #include "geometry/point2d.hpp" #include "geometry/rect2d.hpp" +#include "base/logging.hpp" #include "base/math.hpp" #include +#include #include #include #include @@ -301,6 +303,49 @@ public: return AtBorder(pt, delta, typename Traits::EqualType()); } + double CalculateArea() const + { + double area = 0.0; + + if (m_points.empty()) + return area; + + size_t const numPoints = m_points.size(); + Point prev = m_points[numPoints - 1]; + for (size_t i = 0; i < numPoints; ++i) + { + Point const curr = m_points[i]; + area += CrossProduct(prev, curr); + prev = curr; + } + area = 0.5 * std::fabs(area); + + return area; + } + + template + Point GetRandomPoint(Engine & engine) const + { + CHECK(m_rect.IsValid(), ()); + CHECK(!m_rect.IsEmptyInterior(), ()); + + std::uniform_real_distribution<> distrX(m_rect.minX(), m_rect.maxX()); + std::uniform_real_distribution<> distrY(m_rect.minY(), m_rect.maxY()); + + size_t const kMaxIterations = 1000; + for (size_t it = 0; it < kMaxIterations; ++it) + { + auto const x = distrX(engine); + auto const y = distrY(engine); + Point const pt(x, y); + if (Contains(pt)) + return pt; + } + + LOG(LWARNING, ("Could not sample a random point from m2::Region")); + return m_points[0]; + } + private: template friend Archive & operator<<(Archive & ar, Region const & region); diff --git a/storage/storage_tests/country_info_getter_tests.cpp b/storage/storage_tests/country_info_getter_tests.cpp index 8509b1fc28..c07ff08bc6 100644 --- a/storage/storage_tests/country_info_getter_tests.cpp +++ b/storage/storage_tests/country_info_getter_tests.cpp @@ -1,3 +1,4 @@ +#include "testing/benchmark.hpp" #include "testing/testing.hpp" #include "storage/storage_tests/helpers.hpp" @@ -16,11 +17,14 @@ #include "base/assert.hpp" #include "base/logging.hpp" +#include "base/stats.hpp" +#include "base/timer.hpp" #include #include #include #include +#include #include using namespace storage; @@ -34,6 +38,50 @@ bool IsEmptyName(map const & id2info, string const & id) TEST(it != id2info.end(), ()); return it->second.m_name.empty(); } + +// A helper class to sample random points from mwms uniformly. +class RandomPointGenerator +{ +public: + explicit RandomPointGenerator(mt19937 & randomEngine, + vector> const & allRegions) + : m_randomEngine(randomEngine), m_allRegions(allRegions) + { + size_t const n = m_allRegions.size(); + vector> allAreas(n); + vector aggregateAreas(n); + for (size_t i = 0; i < n; ++i) + { + allAreas[i].resize(allRegions[i].size()); + for (size_t j = 0; j < allRegions[i].size(); ++j) + { + double const area = allRegions[i][j].CalculateArea(); + allAreas[i][j] = area; + aggregateAreas[i] += area; + } + } + + m_distrAll = discrete_distribution(aggregateAreas.begin(), aggregateAreas.end()); + m_distrPerMwm.resize(n); + for (size_t i = 0; i < n; ++i) + m_distrPerMwm[i] = discrete_distribution(allAreas[i].begin(), allAreas[i].end()); + } + + m2::PointD operator()() + { + CHECK(!m_allRegions.empty(), ()); + auto const i = m_distrAll(m_randomEngine); + auto const j = m_distrPerMwm[i](m_randomEngine); + return m_allRegions[i][j].GetRandomPoint(m_randomEngine); + } + +private: + mt19937 m_randomEngine; + + vector> m_allRegions; + discrete_distribution m_distrAll; + vector> m_distrPerMwm; +}; } // namespace UNIT_TEST(CountryInfoGetter_GetByPoint_Smoke) @@ -257,3 +305,147 @@ UNIT_TEST(CountryInfoGetter_Countries_And_Polygons) } } } + +BENCHMARK_TEST(CountryInfoGetter_RegionsByRect) +{ + auto const getterRaw = CountryInfoReader::CreateCountryInfoReader(GetPlatform()); + CountryInfoReader * getter = static_cast(getterRaw.get()); + + Storage storage(COUNTRIES_FILE); + + auto const & countryDefs = getter->GetCountries(); + + base::Timer timer; + + double const kRectSize = 10; + + mt19937 rng(0); + + vector> allRegions; + allRegions.reserve(countryDefs.size()); + for (size_t i = 0; i < countryDefs.size(); ++i) + { + vector regions; + getter->LoadRegionsFromDisk(i, regions); + allRegions.emplace_back(move(regions)); + } + + size_t totalPoints = 0; + for (auto const & regs : allRegions) + { + for (auto const & reg : regs) + totalPoints += reg.Size(); + } + LOG(LINFO, ("Total points:", totalPoints)); + + { + size_t const kNumIterations = 1000; + + double const t0 = timer.ElapsedSeconds(); + + // Antarctica's rect is too large and skews the random point generation. + vector> regionsWithoutAnarctica; + for (size_t i = 0; i < allRegions.size(); ++i) + { + if (countryDefs[i].m_countryId == "Antarctica") + continue; + + regionsWithoutAnarctica.emplace_back(allRegions[i]); + } + + RandomPointGenerator pointGen(rng, regionsWithoutAnarctica); + vector points; + for (size_t i = 0; i < kNumIterations; i++) + points.emplace_back(pointGen()); + + map hits; + for (auto const & pt : points) + { + auto const rect = + MercatorBounds::RectByCenterXYAndSizeInMeters(pt.x, pt.y, kRectSize, kRectSize); + auto vec = getter->GetRegionsCountryIdByRect(rect, false /* rough */); + for (auto const & countryId : vec) + ++hits[countryId]; + } + double const t1 = timer.ElapsedSeconds(); + + LOG(LINFO, ("hits:", hits.size(), "/", countryDefs.size(), t1 - t0)); + } + + { + map> timesByCountry; + map avgTimeByCountry; + size_t kNumPointsPerCountry = 1; + CountryId longest; + for (size_t countryDefId = 0; countryDefId < countryDefs.size(); ++countryDefId) + { + RandomPointGenerator pointGen(rng, {allRegions[countryDefId]}); + auto const & countryId = countryDefs[countryDefId].m_countryId; + + vector & times = timesByCountry[countryId]; + times.resize(kNumPointsPerCountry); + for (size_t i = 0; i < times.size(); ++i) + { + auto const pt = pointGen(); + auto const rect = + MercatorBounds::RectByCenterXYAndSizeInMeters(pt.x, pt.y, kRectSize, kRectSize); + double const t0 = timer.ElapsedSeconds(); + auto vec = getter->GetRegionsCountryIdByRect(rect, false /* rough */); + double const t1 = timer.ElapsedSeconds(); + times[i] = t1 - t0; + } + + avgTimeByCountry[countryId] = + base::AverageStats(times.begin(), times.end()).GetAverage(); + + if (longest.empty() || avgTimeByCountry[longest] < avgTimeByCountry[countryId]) + longest = countryId; + } + + LOG(LINFO, ("Slowest country for CountryInfoGetter (random point)", longest, + avgTimeByCountry[longest])); + } + + { + map> timesByCountry; + map avgTimeByCountry; + size_t kNumSidesPerCountry = 1; + CountryId longest; + for (size_t countryDefId = 0; countryDefId < countryDefs.size(); ++countryDefId) + { + auto const & countryId = countryDefs[countryDefId].m_countryId; + + vector> sides; + for (auto const & region : allRegions[countryDefId]) + { + auto const & points = region.Data(); + for (size_t i = 0; i < points.size(); ++i) + sides.emplace_back(points[i], points[(i + 1) % points.size()]); + } + + CHECK(!sides.empty(), ()); + uniform_int_distribution distr(0, sides.size() - 1); + vector & times = timesByCountry[countryId]; + times.resize(kNumSidesPerCountry); + for (size_t i = 0; i < times.size(); ++i) + { + auto const & side = sides[distr(rng)]; + auto const pt = side.first.mid(side.second); + auto const rect = + MercatorBounds::RectByCenterXYAndSizeInMeters(pt.x, pt.y, kRectSize, kRectSize); + double const t0 = timer.ElapsedSeconds(); + auto vec = getter->GetRegionsCountryIdByRect(rect, false /* rough */); + double const t1 = timer.ElapsedSeconds(); + times[i] = t1 - t0; + } + + avgTimeByCountry[countryId] = + base::AverageStats(times.begin(), times.end()).GetAverage(); + + if (longest.empty() || avgTimeByCountry[longest] < avgTimeByCountry[countryId]) + longest = countryId; + } + LOG(LINFO, ("Slowest country for CountryInfoGetter (point on a random side)", longest, + avgTimeByCountry[longest])); + } +} diff --git a/xcode/geometry/geometry.xcodeproj/project.pbxproj b/xcode/geometry/geometry.xcodeproj/project.pbxproj index f08c598781..394c475f53 100644 --- a/xcode/geometry/geometry.xcodeproj/project.pbxproj +++ b/xcode/geometry/geometry.xcodeproj/project.pbxproj @@ -16,6 +16,7 @@ 347F33711C454205009758CC /* triangle2d.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 347F336F1C454205009758CC /* triangle2d.hpp */; }; 39396ECC210B2E590054AC20 /* parametrized_segment.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 39396ECB210B2E590054AC20 /* parametrized_segment.hpp */; }; 39396ECF210B2FB80054AC20 /* parametrized_segment_tests.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 39396ECD210B2FB30054AC20 /* parametrized_segment_tests.cpp */; }; + 3946DDEB22A668D800A3A37D /* region_tests.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3946DDE922A668D500A3A37D /* region_tests.cpp */; }; 395D1A9B207BCD8D001164A5 /* convex_hull_tests.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 395D1A99207BCD8A001164A5 /* convex_hull_tests.cpp */; }; 39B2B9701FB4680500AB85A1 /* diamond_box.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 39B2B96E1FB4680500AB85A1 /* diamond_box.hpp */; }; 39B2B9711FB4680500AB85A1 /* diamond_box.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 39B2B96F1FB4680500AB85A1 /* diamond_box.cpp */; }; @@ -53,7 +54,6 @@ 670F29911BAABDA600F2ABF4 /* point_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 670F292E1BA9D2E700F2ABF4 /* point_test.cpp */; }; 670F29931BAABDA600F2ABF4 /* polygon_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 670F29301BA9D2E700F2ABF4 /* polygon_test.cpp */; }; 670F29941BAABDA600F2ABF4 /* rect_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 670F29311BA9D2E700F2ABF4 /* rect_test.cpp */; }; - 670F29951BAABDA600F2ABF4 /* region_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 670F29321BA9D2E700F2ABF4 /* region_test.cpp */; }; 670F29961BAABDA600F2ABF4 /* region2d_binary_op_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 670F29331BA9D2E700F2ABF4 /* region2d_binary_op_test.cpp */; }; 670F29971BAABDA600F2ABF4 /* robust_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 670F29341BA9D2E700F2ABF4 /* robust_test.cpp */; }; 670F29981BAABDA600F2ABF4 /* screen_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 670F29351BA9D2E700F2ABF4 /* screen_test.cpp */; }; @@ -125,6 +125,7 @@ 34AF87CE1DBE517700E5E7DC /* common-release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = "common-release.xcconfig"; path = "../common-release.xcconfig"; sourceTree = ""; }; 39396ECB210B2E590054AC20 /* parametrized_segment.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = parametrized_segment.hpp; sourceTree = ""; }; 39396ECD210B2FB30054AC20 /* parametrized_segment_tests.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = parametrized_segment_tests.cpp; sourceTree = ""; }; + 3946DDE922A668D500A3A37D /* region_tests.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = region_tests.cpp; sourceTree = ""; }; 395D1A99207BCD8A001164A5 /* convex_hull_tests.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = convex_hull_tests.cpp; sourceTree = ""; }; 39B2B96E1FB4680500AB85A1 /* diamond_box.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = diamond_box.hpp; sourceTree = ""; }; 39B2B96F1FB4680500AB85A1 /* diamond_box.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = diamond_box.cpp; sourceTree = ""; }; @@ -156,7 +157,6 @@ 670F292E1BA9D2E700F2ABF4 /* point_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = point_test.cpp; sourceTree = ""; }; 670F29301BA9D2E700F2ABF4 /* polygon_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = polygon_test.cpp; sourceTree = ""; }; 670F29311BA9D2E700F2ABF4 /* rect_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = rect_test.cpp; sourceTree = ""; }; - 670F29321BA9D2E700F2ABF4 /* region_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = region_test.cpp; sourceTree = ""; }; 670F29331BA9D2E700F2ABF4 /* region2d_binary_op_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = region2d_binary_op_test.cpp; sourceTree = ""; }; 670F29341BA9D2E700F2ABF4 /* robust_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = robust_test.cpp; sourceTree = ""; }; 670F29351BA9D2E700F2ABF4 /* screen_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = screen_test.cpp; sourceTree = ""; }; @@ -231,6 +231,7 @@ 670F29211BA9D25D00F2ABF4 /* geometry_tests */ = { isa = PBXGroup; children = ( + 3946DDE922A668D500A3A37D /* region_tests.cpp */, 39396ECD210B2FB30054AC20 /* parametrized_segment_tests.cpp */, 395D1A99207BCD8A001164A5 /* convex_hull_tests.cpp */, 45A2D9C51F7526FE003310A0 /* bounding_box_tests.cpp */, @@ -254,7 +255,6 @@ 670F292E1BA9D2E700F2ABF4 /* point_test.cpp */, 670F29301BA9D2E700F2ABF4 /* polygon_test.cpp */, 670F29311BA9D2E700F2ABF4 /* rect_test.cpp */, - 670F29321BA9D2E700F2ABF4 /* region_test.cpp */, 670F29331BA9D2E700F2ABF4 /* region2d_binary_op_test.cpp */, 670F29341BA9D2E700F2ABF4 /* robust_test.cpp */, 670F29351BA9D2E700F2ABF4 /* screen_test.cpp */, @@ -503,7 +503,6 @@ 45A2D9CF1F75270D003310A0 /* diamond_box_tests.cpp in Sources */, 670F299B1BAABDA600F2ABF4 /* spline_test.cpp in Sources */, 670F299C1BAABDA600F2ABF4 /* transformations_test.cpp in Sources */, - 670F29951BAABDA600F2ABF4 /* region_test.cpp in Sources */, 670F29891BAABDA600F2ABF4 /* cellid_test.cpp in Sources */, 670F29961BAABDA600F2ABF4 /* region2d_binary_op_test.cpp in Sources */, 670F299D1BAABDA600F2ABF4 /* tree_test.cpp in Sources */, @@ -521,6 +520,7 @@ 670F298B1BAABDA600F2ABF4 /* covering_test.cpp in Sources */, 670F29931BAABDA600F2ABF4 /* polygon_test.cpp in Sources */, 670F298C1BAABDA600F2ABF4 /* distance_on_sphere_test.cpp in Sources */, + 3946DDEB22A668D800A3A37D /* region_tests.cpp in Sources */, 45A2D9D21F752727003310A0 /* clipping_test.cpp in Sources */, ); runOnlyForDeploymentPostprocessing = 0;