xref: /aosp_15_r20/external/skia/tests/RTreeTest.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
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