xref: /aosp_15_r20/external/skia/bench/PathOpsBench.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2018 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 "bench/Benchmark.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPath.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkShader.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkString.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/pathops/SkPathOps.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTArray.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkRandom.h"
15*c8dee2aaSAndroid Build Coastguard Worker 
16*c8dee2aaSAndroid Build Coastguard Worker class PathOpsBench : public Benchmark {
17*c8dee2aaSAndroid Build Coastguard Worker     SkString    fName;
18*c8dee2aaSAndroid Build Coastguard Worker     SkPath      fPath1, fPath2;
19*c8dee2aaSAndroid Build Coastguard Worker     SkPathOp    fOp;
20*c8dee2aaSAndroid Build Coastguard Worker 
21*c8dee2aaSAndroid Build Coastguard Worker public:
PathOpsBench(const char suffix[],SkPathOp op)22*c8dee2aaSAndroid Build Coastguard Worker     PathOpsBench(const char suffix[], SkPathOp op) : fOp(op) {
23*c8dee2aaSAndroid Build Coastguard Worker         fName.printf("pathops_%s", suffix);
24*c8dee2aaSAndroid Build Coastguard Worker 
25*c8dee2aaSAndroid Build Coastguard Worker         fPath1.addOval({-10, -20, 10, 20});
26*c8dee2aaSAndroid Build Coastguard Worker         fPath2.addOval({-20, -10, 20, 10});
27*c8dee2aaSAndroid Build Coastguard Worker     }
28*c8dee2aaSAndroid Build Coastguard Worker 
isSuitableFor(Backend backend)29*c8dee2aaSAndroid Build Coastguard Worker     bool isSuitableFor(Backend backend) override {
30*c8dee2aaSAndroid Build Coastguard Worker         return backend == Backend::kNonRendering;
31*c8dee2aaSAndroid Build Coastguard Worker     }
32*c8dee2aaSAndroid Build Coastguard Worker 
33*c8dee2aaSAndroid Build Coastguard Worker protected:
onGetName()34*c8dee2aaSAndroid Build Coastguard Worker     const char* onGetName() override {
35*c8dee2aaSAndroid Build Coastguard Worker         return fName.c_str();
36*c8dee2aaSAndroid Build Coastguard Worker     }
37*c8dee2aaSAndroid Build Coastguard Worker 
onDraw(int loops,SkCanvas * canvas)38*c8dee2aaSAndroid Build Coastguard Worker     void onDraw(int loops, SkCanvas* canvas) override {
39*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < loops; i++) {
40*c8dee2aaSAndroid Build Coastguard Worker             for (int j = 0; j < 1000; ++j) {
41*c8dee2aaSAndroid Build Coastguard Worker                 SkPath result;
42*c8dee2aaSAndroid Build Coastguard Worker                 Op(fPath1, fPath2, fOp, &result);
43*c8dee2aaSAndroid Build Coastguard Worker             }
44*c8dee2aaSAndroid Build Coastguard Worker         }
45*c8dee2aaSAndroid Build Coastguard Worker     }
46*c8dee2aaSAndroid Build Coastguard Worker 
47*c8dee2aaSAndroid Build Coastguard Worker private:
48*c8dee2aaSAndroid Build Coastguard Worker     using INHERITED = Benchmark;
49*c8dee2aaSAndroid Build Coastguard Worker };
50*c8dee2aaSAndroid Build Coastguard Worker 
51*c8dee2aaSAndroid Build Coastguard Worker class PathOpsSimplifyBench : public Benchmark {
52*c8dee2aaSAndroid Build Coastguard Worker     SkString    fName;
53*c8dee2aaSAndroid Build Coastguard Worker     SkPath      fPath;
54*c8dee2aaSAndroid Build Coastguard Worker 
55*c8dee2aaSAndroid Build Coastguard Worker public:
PathOpsSimplifyBench(const char suffix[],const SkPath & path)56*c8dee2aaSAndroid Build Coastguard Worker     PathOpsSimplifyBench(const char suffix[], const SkPath& path) : fPath(path) {
57*c8dee2aaSAndroid Build Coastguard Worker         fName.printf("pathops_simplify_%s", suffix);
58*c8dee2aaSAndroid Build Coastguard Worker     }
59*c8dee2aaSAndroid Build Coastguard Worker 
isSuitableFor(Backend backend)60*c8dee2aaSAndroid Build Coastguard Worker     bool isSuitableFor(Backend backend) override {
61*c8dee2aaSAndroid Build Coastguard Worker         return backend == Backend::kNonRendering;
62*c8dee2aaSAndroid Build Coastguard Worker     }
63*c8dee2aaSAndroid Build Coastguard Worker 
64*c8dee2aaSAndroid Build Coastguard Worker protected:
onGetName()65*c8dee2aaSAndroid Build Coastguard Worker     const char* onGetName() override {
66*c8dee2aaSAndroid Build Coastguard Worker         return fName.c_str();
67*c8dee2aaSAndroid Build Coastguard Worker     }
68*c8dee2aaSAndroid Build Coastguard Worker 
onDraw(int loops,SkCanvas * canvas)69*c8dee2aaSAndroid Build Coastguard Worker     void onDraw(int loops, SkCanvas* canvas) override {
70*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < loops; i++) {
71*c8dee2aaSAndroid Build Coastguard Worker             for (int j = 0; j < 100; ++j) {
72*c8dee2aaSAndroid Build Coastguard Worker                 SkPath result;
73*c8dee2aaSAndroid Build Coastguard Worker                 Simplify(fPath, &result);
74*c8dee2aaSAndroid Build Coastguard Worker             }
75*c8dee2aaSAndroid Build Coastguard Worker         }
76*c8dee2aaSAndroid Build Coastguard Worker     }
77*c8dee2aaSAndroid Build Coastguard Worker 
78*c8dee2aaSAndroid Build Coastguard Worker private:
79*c8dee2aaSAndroid Build Coastguard Worker     using INHERITED = Benchmark;
80*c8dee2aaSAndroid Build Coastguard Worker };
81*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return new PathOpsBench("sect", kIntersect_SkPathOp); )
82*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return new PathOpsBench("join", kUnion_SkPathOp); )
83*c8dee2aaSAndroid Build Coastguard Worker 
makerects()84*c8dee2aaSAndroid Build Coastguard Worker static SkPath makerects() {
85*c8dee2aaSAndroid Build Coastguard Worker     SkRandom rand;
86*c8dee2aaSAndroid Build Coastguard Worker     SkPath path;
87*c8dee2aaSAndroid Build Coastguard Worker     SkScalar scale = 100;
88*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 0; i < 20; ++i) {
89*c8dee2aaSAndroid Build Coastguard Worker         SkScalar x = rand.nextUScalar1() * scale;
90*c8dee2aaSAndroid Build Coastguard Worker         SkScalar y = rand.nextUScalar1() * scale;
91*c8dee2aaSAndroid Build Coastguard Worker         path.addRect({x, y, x + scale, y + scale});
92*c8dee2aaSAndroid Build Coastguard Worker     }
93*c8dee2aaSAndroid Build Coastguard Worker     return path;
94*c8dee2aaSAndroid Build Coastguard Worker }
95*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return new PathOpsSimplifyBench("rects", makerects()); )
96*c8dee2aaSAndroid Build Coastguard Worker 
97*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPathBuilder.h"
98*c8dee2aaSAndroid Build Coastguard Worker 
99*c8dee2aaSAndroid Build Coastguard Worker template <size_t N> struct ArrayPath {
100*c8dee2aaSAndroid Build Coastguard Worker     SkPoint fPts[N];
101*c8dee2aaSAndroid Build Coastguard Worker     uint8_t fVbs[N];
102*c8dee2aaSAndroid Build Coastguard Worker     int fPIndex = 0, fVIndex = 0;
103*c8dee2aaSAndroid Build Coastguard Worker 
moveToArrayPath104*c8dee2aaSAndroid Build Coastguard Worker     void moveTo(float x, float y) {
105*c8dee2aaSAndroid Build Coastguard Worker         fVbs[fVIndex++] = (uint8_t)SkPathVerb::kMove;
106*c8dee2aaSAndroid Build Coastguard Worker         fPts[fPIndex++] = {x, y};
107*c8dee2aaSAndroid Build Coastguard Worker     }
lineToArrayPath108*c8dee2aaSAndroid Build Coastguard Worker     void lineTo(float x, float y) {
109*c8dee2aaSAndroid Build Coastguard Worker         fVbs[fVIndex++] = (uint8_t)SkPathVerb::kLine;
110*c8dee2aaSAndroid Build Coastguard Worker         fPts[fPIndex++] = {x, y};
111*c8dee2aaSAndroid Build Coastguard Worker     }
quadToArrayPath112*c8dee2aaSAndroid Build Coastguard Worker     void quadTo(float x, float y, float x1, float y1) {
113*c8dee2aaSAndroid Build Coastguard Worker         fVbs[fVIndex++] = (uint8_t)SkPathVerb::kQuad;
114*c8dee2aaSAndroid Build Coastguard Worker         fPts[fPIndex++] = {x, y};
115*c8dee2aaSAndroid Build Coastguard Worker         fPts[fPIndex++] = {x1, y1};
116*c8dee2aaSAndroid Build Coastguard Worker     }
cubicToArrayPath117*c8dee2aaSAndroid Build Coastguard Worker     void cubicTo(float x, float y, float x1, float y1, float x2, float y2) {
118*c8dee2aaSAndroid Build Coastguard Worker         fVbs[fVIndex++] = (uint8_t)SkPathVerb::kCubic;
119*c8dee2aaSAndroid Build Coastguard Worker         fPts[fPIndex++] = {x, y};
120*c8dee2aaSAndroid Build Coastguard Worker         fPts[fPIndex++] = {x1, y1};
121*c8dee2aaSAndroid Build Coastguard Worker         fPts[fPIndex++] = {x2, y2};
122*c8dee2aaSAndroid Build Coastguard Worker     }
incReserveArrayPath123*c8dee2aaSAndroid Build Coastguard Worker     void incReserve(int) {}
124*c8dee2aaSAndroid Build Coastguard Worker };
125*c8dee2aaSAndroid Build Coastguard Worker 
run_builder(T & b,bool useReserve,int N)126*c8dee2aaSAndroid Build Coastguard Worker template <typename T> void run_builder(T& b, bool useReserve, int N) {
127*c8dee2aaSAndroid Build Coastguard Worker     if (useReserve) {
128*c8dee2aaSAndroid Build Coastguard Worker         b.incReserve(N * 12);
129*c8dee2aaSAndroid Build Coastguard Worker     }
130*c8dee2aaSAndroid Build Coastguard Worker 
131*c8dee2aaSAndroid Build Coastguard Worker     float x = 0, y = 0;
132*c8dee2aaSAndroid Build Coastguard Worker     b.moveTo(x, y);
133*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 1; i < N; ++i) {
134*c8dee2aaSAndroid Build Coastguard Worker         b.lineTo(x, y);
135*c8dee2aaSAndroid Build Coastguard Worker         b.quadTo(x, y, x, y);
136*c8dee2aaSAndroid Build Coastguard Worker         b.cubicTo(x, y, x, y, x, y);
137*c8dee2aaSAndroid Build Coastguard Worker     }
138*c8dee2aaSAndroid Build Coastguard Worker }
139*c8dee2aaSAndroid Build Coastguard Worker 
140*c8dee2aaSAndroid Build Coastguard Worker enum class MakeType {
141*c8dee2aaSAndroid Build Coastguard Worker     kPath,
142*c8dee2aaSAndroid Build Coastguard Worker     kSnapshot,
143*c8dee2aaSAndroid Build Coastguard Worker     kDetach,
144*c8dee2aaSAndroid Build Coastguard Worker     kArray,
145*c8dee2aaSAndroid Build Coastguard Worker };
146*c8dee2aaSAndroid Build Coastguard Worker 
147*c8dee2aaSAndroid Build Coastguard Worker class PathBuilderBench : public Benchmark {
148*c8dee2aaSAndroid Build Coastguard Worker     SkString    fName;
149*c8dee2aaSAndroid Build Coastguard Worker     MakeType    fMakeType;
150*c8dee2aaSAndroid Build Coastguard Worker     bool        fUseReserve;
151*c8dee2aaSAndroid Build Coastguard Worker 
152*c8dee2aaSAndroid Build Coastguard Worker     enum { N = 100 };
153*c8dee2aaSAndroid Build Coastguard Worker     ArrayPath<N*12> fArrays;
154*c8dee2aaSAndroid Build Coastguard Worker 
155*c8dee2aaSAndroid Build Coastguard Worker public:
PathBuilderBench(MakeType mt,bool reserve)156*c8dee2aaSAndroid Build Coastguard Worker     PathBuilderBench(MakeType mt, bool reserve) : fMakeType(mt), fUseReserve(reserve) {
157*c8dee2aaSAndroid Build Coastguard Worker         const char* typenames[] = { "path", "snapshot", "detach", "arrays" };
158*c8dee2aaSAndroid Build Coastguard Worker 
159*c8dee2aaSAndroid Build Coastguard Worker         fName.printf("makepath_%s_%s", typenames[(int)mt], reserve ? "reserve" : "noreserve");
160*c8dee2aaSAndroid Build Coastguard Worker     }
161*c8dee2aaSAndroid Build Coastguard Worker 
isSuitableFor(Backend backend)162*c8dee2aaSAndroid Build Coastguard Worker     bool isSuitableFor(Backend backend) override {
163*c8dee2aaSAndroid Build Coastguard Worker         return backend == Backend::kNonRendering;
164*c8dee2aaSAndroid Build Coastguard Worker     }
165*c8dee2aaSAndroid Build Coastguard Worker 
166*c8dee2aaSAndroid Build Coastguard Worker protected:
onGetName()167*c8dee2aaSAndroid Build Coastguard Worker     const char* onGetName() override {
168*c8dee2aaSAndroid Build Coastguard Worker         return fName.c_str();
169*c8dee2aaSAndroid Build Coastguard Worker     }
170*c8dee2aaSAndroid Build Coastguard Worker 
onDelayedSetup()171*c8dee2aaSAndroid Build Coastguard Worker     void onDelayedSetup() override {
172*c8dee2aaSAndroid Build Coastguard Worker         run_builder(fArrays, false, N);
173*c8dee2aaSAndroid Build Coastguard Worker     }
174*c8dee2aaSAndroid Build Coastguard Worker 
build()175*c8dee2aaSAndroid Build Coastguard Worker     SkPath build() {
176*c8dee2aaSAndroid Build Coastguard Worker         switch (fMakeType) {
177*c8dee2aaSAndroid Build Coastguard Worker             case MakeType::kSnapshot:
178*c8dee2aaSAndroid Build Coastguard Worker             case MakeType::kDetach: {
179*c8dee2aaSAndroid Build Coastguard Worker                 SkPathBuilder b;
180*c8dee2aaSAndroid Build Coastguard Worker                 run_builder(b, fUseReserve, N);
181*c8dee2aaSAndroid Build Coastguard Worker                 return MakeType::kSnapshot == fMakeType ? b.snapshot() : b.detach();
182*c8dee2aaSAndroid Build Coastguard Worker             }
183*c8dee2aaSAndroid Build Coastguard Worker             case MakeType::kPath: {
184*c8dee2aaSAndroid Build Coastguard Worker                 SkPath p;
185*c8dee2aaSAndroid Build Coastguard Worker                 run_builder(p, fUseReserve, N);
186*c8dee2aaSAndroid Build Coastguard Worker                 return p;
187*c8dee2aaSAndroid Build Coastguard Worker             }
188*c8dee2aaSAndroid Build Coastguard Worker             case MakeType::kArray: {
189*c8dee2aaSAndroid Build Coastguard Worker             //    ArrayPath<N*12> arrays;
190*c8dee2aaSAndroid Build Coastguard Worker             //    run_builder(arrays, false, N);
191*c8dee2aaSAndroid Build Coastguard Worker                 return SkPath::Make(fArrays.fPts, fArrays.fPIndex,
192*c8dee2aaSAndroid Build Coastguard Worker                                     fArrays.fVbs, fArrays.fVIndex,
193*c8dee2aaSAndroid Build Coastguard Worker                                     nullptr, 0, SkPathFillType::kWinding);
194*c8dee2aaSAndroid Build Coastguard Worker             }
195*c8dee2aaSAndroid Build Coastguard Worker         }
196*c8dee2aaSAndroid Build Coastguard Worker         return SkPath();
197*c8dee2aaSAndroid Build Coastguard Worker     }
198*c8dee2aaSAndroid Build Coastguard Worker 
onDraw(int loops,SkCanvas * canvas)199*c8dee2aaSAndroid Build Coastguard Worker     void onDraw(int loops, SkCanvas* canvas) override {
200*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < loops; i++) {
201*c8dee2aaSAndroid Build Coastguard Worker             for (int j = 0; j < 100; ++j) {
202*c8dee2aaSAndroid Build Coastguard Worker                 SkPath result = this->build();
203*c8dee2aaSAndroid Build Coastguard Worker                 // force bounds calc as part of the test
204*c8dee2aaSAndroid Build Coastguard Worker                 if (!result.getBounds().isFinite()) {
205*c8dee2aaSAndroid Build Coastguard Worker                     SkDebugf("should never get here!\n");
206*c8dee2aaSAndroid Build Coastguard Worker                     return;
207*c8dee2aaSAndroid Build Coastguard Worker                 }
208*c8dee2aaSAndroid Build Coastguard Worker             }
209*c8dee2aaSAndroid Build Coastguard Worker         }
210*c8dee2aaSAndroid Build Coastguard Worker     }
211*c8dee2aaSAndroid Build Coastguard Worker 
212*c8dee2aaSAndroid Build Coastguard Worker private:
213*c8dee2aaSAndroid Build Coastguard Worker     using INHERITED = Benchmark;
214*c8dee2aaSAndroid Build Coastguard Worker };
215*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return new PathBuilderBench(MakeType::kPath, false); )
216*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return new PathBuilderBench(MakeType::kSnapshot, false); )
217*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return new PathBuilderBench(MakeType::kDetach, false); )
218*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return new PathBuilderBench(MakeType::kPath, true); )
219*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return new PathBuilderBench(MakeType::kSnapshot, true); )
220*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return new PathBuilderBench(MakeType::kDetach, true); )
221*c8dee2aaSAndroid Build Coastguard Worker 
222*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return new PathBuilderBench(MakeType::kArray, true); )
223