1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2012 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRect.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTemplates.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkRandom.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkRTree.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "tests/Test.h"
14*c8dee2aaSAndroid Build Coastguard Worker
15*c8dee2aaSAndroid Build Coastguard Worker #include <cmath>
16*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef>
17*c8dee2aaSAndroid Build Coastguard Worker #include <vector>
18*c8dee2aaSAndroid Build Coastguard Worker
19*c8dee2aaSAndroid Build Coastguard Worker using namespace skia_private;
20*c8dee2aaSAndroid Build Coastguard Worker
21*c8dee2aaSAndroid Build Coastguard Worker static const int NUM_RECTS = 200;
22*c8dee2aaSAndroid Build Coastguard Worker static const size_t NUM_ITERATIONS = 100;
23*c8dee2aaSAndroid Build Coastguard Worker static const size_t NUM_QUERIES = 50;
24*c8dee2aaSAndroid Build Coastguard Worker
random_rect(SkRandom & rand)25*c8dee2aaSAndroid Build Coastguard Worker static SkRect random_rect(SkRandom& rand) {
26*c8dee2aaSAndroid Build Coastguard Worker SkRect rect = {0,0,0,0};
27*c8dee2aaSAndroid Build Coastguard Worker while (rect.isEmpty()) {
28*c8dee2aaSAndroid Build Coastguard Worker rect.fLeft = rand.nextRangeF(0, 1000);
29*c8dee2aaSAndroid Build Coastguard Worker rect.fRight = rand.nextRangeF(0, 1000);
30*c8dee2aaSAndroid Build Coastguard Worker rect.fTop = rand.nextRangeF(0, 1000);
31*c8dee2aaSAndroid Build Coastguard Worker rect.fBottom = rand.nextRangeF(0, 1000);
32*c8dee2aaSAndroid Build Coastguard Worker rect.sort();
33*c8dee2aaSAndroid Build Coastguard Worker }
34*c8dee2aaSAndroid Build Coastguard Worker return rect;
35*c8dee2aaSAndroid Build Coastguard Worker }
36*c8dee2aaSAndroid Build Coastguard Worker
verify_query(SkRect query,SkRect rects[],const std::vector<int> & found)37*c8dee2aaSAndroid Build Coastguard Worker static bool verify_query(SkRect query, SkRect rects[], const std::vector<int>& found) {
38*c8dee2aaSAndroid Build Coastguard Worker std::vector<int> expected;
39*c8dee2aaSAndroid Build Coastguard Worker // manually intersect with every rectangle
40*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < NUM_RECTS; ++i) {
41*c8dee2aaSAndroid Build Coastguard Worker if (SkRect::Intersects(query, rects[i])) {
42*c8dee2aaSAndroid Build Coastguard Worker expected.push_back(i);
43*c8dee2aaSAndroid Build Coastguard Worker }
44*c8dee2aaSAndroid Build Coastguard Worker }
45*c8dee2aaSAndroid Build Coastguard Worker
46*c8dee2aaSAndroid Build Coastguard Worker if (expected.size() != found.size()) {
47*c8dee2aaSAndroid Build Coastguard Worker return false;
48*c8dee2aaSAndroid Build Coastguard Worker }
49*c8dee2aaSAndroid Build Coastguard Worker if (0 == expected.size()) {
50*c8dee2aaSAndroid Build Coastguard Worker return true;
51*c8dee2aaSAndroid Build Coastguard Worker }
52*c8dee2aaSAndroid Build Coastguard Worker return found == expected;
53*c8dee2aaSAndroid Build Coastguard Worker }
54*c8dee2aaSAndroid Build Coastguard Worker
run_queries(skiatest::Reporter * reporter,SkRandom & rand,SkRect rects[],const SkRTree & tree)55*c8dee2aaSAndroid Build Coastguard Worker static void run_queries(skiatest::Reporter* reporter, SkRandom& rand, SkRect rects[],
56*c8dee2aaSAndroid Build Coastguard Worker const SkRTree& tree) {
57*c8dee2aaSAndroid Build Coastguard Worker for (size_t i = 0; i < NUM_QUERIES; ++i) {
58*c8dee2aaSAndroid Build Coastguard Worker std::vector<int> hits;
59*c8dee2aaSAndroid Build Coastguard Worker SkRect query = random_rect(rand);
60*c8dee2aaSAndroid Build Coastguard Worker tree.search(query, &hits);
61*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, verify_query(query, rects, hits));
62*c8dee2aaSAndroid Build Coastguard Worker }
63*c8dee2aaSAndroid Build Coastguard Worker }
64*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(RTree,reporter)65*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(RTree, reporter) {
66*c8dee2aaSAndroid Build Coastguard Worker int expectedDepthMin = -1;
67*c8dee2aaSAndroid Build Coastguard Worker int tmp = NUM_RECTS;
68*c8dee2aaSAndroid Build Coastguard Worker while (tmp > 0) {
69*c8dee2aaSAndroid Build Coastguard Worker tmp -= static_cast<int>(pow(static_cast<double>(SkRTree::kMaxChildren),
70*c8dee2aaSAndroid Build Coastguard Worker static_cast<double>(expectedDepthMin + 1)));
71*c8dee2aaSAndroid Build Coastguard Worker ++expectedDepthMin;
72*c8dee2aaSAndroid Build Coastguard Worker }
73*c8dee2aaSAndroid Build Coastguard Worker
74*c8dee2aaSAndroid Build Coastguard Worker int expectedDepthMax = -1;
75*c8dee2aaSAndroid Build Coastguard Worker tmp = NUM_RECTS;
76*c8dee2aaSAndroid Build Coastguard Worker while (tmp > 0) {
77*c8dee2aaSAndroid Build Coastguard Worker tmp -= static_cast<int>(pow(static_cast<double>(SkRTree::kMinChildren),
78*c8dee2aaSAndroid Build Coastguard Worker static_cast<double>(expectedDepthMax + 1)));
79*c8dee2aaSAndroid Build Coastguard Worker ++expectedDepthMax;
80*c8dee2aaSAndroid Build Coastguard Worker }
81*c8dee2aaSAndroid Build Coastguard Worker
82*c8dee2aaSAndroid Build Coastguard Worker SkRandom rand;
83*c8dee2aaSAndroid Build Coastguard Worker AutoTArray<SkRect> rects(NUM_RECTS);
84*c8dee2aaSAndroid Build Coastguard Worker for (size_t i = 0; i < NUM_ITERATIONS; ++i) {
85*c8dee2aaSAndroid Build Coastguard Worker SkRTree rtree;
86*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, 0 == rtree.getCount());
87*c8dee2aaSAndroid Build Coastguard Worker
88*c8dee2aaSAndroid Build Coastguard Worker for (int j = 0; j < NUM_RECTS; j++) {
89*c8dee2aaSAndroid Build Coastguard Worker rects[j] = random_rect(rand);
90*c8dee2aaSAndroid Build Coastguard Worker }
91*c8dee2aaSAndroid Build Coastguard Worker
92*c8dee2aaSAndroid Build Coastguard Worker rtree.insert(rects.data(), NUM_RECTS);
93*c8dee2aaSAndroid Build Coastguard Worker
94*c8dee2aaSAndroid Build Coastguard Worker run_queries(reporter, rand, rects.data(), rtree);
95*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, NUM_RECTS == rtree.getCount());
96*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, expectedDepthMin <= rtree.getDepth() &&
97*c8dee2aaSAndroid Build Coastguard Worker expectedDepthMax >= rtree.getDepth());
98*c8dee2aaSAndroid Build Coastguard Worker }
99*c8dee2aaSAndroid Build Coastguard Worker }
100