xref: /aosp_15_r20/external/skia/bench/VertBench.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1 /*
2  * Copyright 2011 Google Inc.
3  *
4  * Use of this source code is governed by a BSD-style license that can be
5  * found in the LICENSE file.
6  */
7 
8 #include "bench/Benchmark.h"
9 #include "include/core/SkCanvas.h"
10 #include "include/core/SkPaint.h"
11 #include "include/core/SkShader.h"
12 #include "include/core/SkString.h"
13 #include "include/core/SkVertices.h"
14 #include "src/base/SkRandom.h"
15 #include "tools/DecodeUtils.h"
16 #include "tools/Resources.h"
17 
18 // Just want to trigger perspective handling, not dramatically change size
tiny_persp_effect(SkCanvas * canvas)19 static void tiny_persp_effect(SkCanvas* canvas) {
20     SkMatrix m;
21     m.reset();
22     m[7] = 0.000001f;
23     canvas->concat(m);
24 }
25 
26 enum VertFlags {
27     kColors_VertFlag  = 1 << 0,
28     kTexture_VertFlag = 1 << 1,
29     kPersp_VertFlag   = 1 << 2,
30     kBilerp_VertFlag  = 1 << 3,
31 };
32 
33 class VertBench : public Benchmark {
34     SkString fName;
35 
36     static constexpr int W = 64*2;
37     static constexpr int H = 48*2;
38     static constexpr int ROW = 20;
39     static constexpr int COL = 20;
40     static constexpr int PTS = (ROW + 1) * (COL + 1);
41     static constexpr int IDX = ROW * COL * 6;
42 
43     sk_sp<SkShader> fShader;
44     SkPoint fPts[PTS], fTex[PTS];
45     SkColor fColors[PTS];
46     uint16_t fIdx[IDX];
47     unsigned fFlags;
48 
load_2_tris(uint16_t idx[],int x,int y,int rb)49     static void load_2_tris(uint16_t idx[], int x, int y, int rb) {
50         int n = y * rb + x;
51         idx[0] = n; idx[1] = n + 1; idx[2] = rb + n + 1;
52         idx[3] = n; idx[4] = rb + n + 1; idx[5] = n + rb;
53     }
54 
onDelayedSetup()55     void onDelayedSetup() override {
56         if (fFlags & kTexture_VertFlag) {
57             auto img = ToolUtils::GetResourceAsImage("images/mandrill_256.png");
58             if (img) {
59                 SkFilterMode fm = (fFlags & kBilerp_VertFlag) ? SkFilterMode::kLinear
60                                                               : SkFilterMode::kNearest;
61                 fShader = img->makeShader(SkSamplingOptions(fm));
62             }
63         }
64     }
65 
66 public:
VertBench(unsigned flags)67     VertBench(unsigned flags) : fFlags(flags) {
68         const SkScalar dx = SkIntToScalar(W) / COL;
69         const SkScalar dy = SkIntToScalar(H) / COL;
70 
71         SkPoint* pts = fPts;
72         uint16_t* idx = fIdx;
73 
74         SkScalar yy = 0;
75         for (int y = 0; y <= ROW; y++) {
76             SkScalar xx = 0;
77             for (int x = 0; x <= COL; ++x) {
78                 pts->set(xx, yy);
79                 pts += 1;
80                 xx += dx;
81 
82                 if (x < COL && y < ROW) {
83                     load_2_tris(idx, x, y, COL + 1);
84                     for (int i = 0; i < 6; i++) {
85                         SkASSERT(idx[i] < PTS);
86                     }
87                     idx += 6;
88                 }
89             }
90             yy += dy;
91         }
92         SkASSERT(PTS == pts - fPts);
93         SkASSERT(IDX == idx - fIdx);
94 
95         // We want to store texs in a separate array, so the blitters don't "cheat" and
96         // skip the (normal) step of computing the new local-matrix. This is the common case
97         // we think in the wild (where the texture coordinates are different from the positions.
98         memcpy(fTex, fPts, sizeof(fPts));
99 
100         SkRandom rand;
101         for (int i = 0; i < PTS; ++i) {
102             fColors[i] = rand.nextU() | (0xFF << 24);
103         }
104 
105         fName.set("verts");
106         if (fFlags & kTexture_VertFlag) {
107             fName.append("_textures");
108         }
109         if (fFlags & kColors_VertFlag) {
110             fName.append("_colors");
111         }
112         if (fFlags & kPersp_VertFlag) {
113             fName.append("_persp");
114         }
115         if (fFlags & kBilerp_VertFlag) {
116             fName.append("_bilerp");
117         }
118     }
119 
120 protected:
onGetName()121     const char* onGetName() override { return fName.c_str(); }
onDraw(int loops,SkCanvas * canvas)122     void onDraw(int loops, SkCanvas* canvas) override {
123         SkPaint paint;
124         this->setupPaint(&paint);
125         paint.setShader(fShader);
126 
127         if (fFlags & kPersp_VertFlag) {
128             tiny_persp_effect(canvas);
129         }
130 
131         const SkPoint* texs = (fFlags & kTexture_VertFlag) ? fTex    : nullptr;
132         const SkColor* cols = (fFlags & kColors_VertFlag)  ? fColors : nullptr;
133         auto verts = SkVertices::MakeCopy(SkVertices::kTriangles_VertexMode, PTS,
134                                           fPts, texs, cols, IDX, fIdx);
135         for (int i = 0; i < loops; i++) {
136             canvas->drawVertices(verts, SkBlendMode::kModulate, paint);
137         }
138     }
139 private:
140     using INHERITED = Benchmark;
141 };
142 DEF_BENCH(return new VertBench(kTexture_VertFlag | kPersp_VertFlag);)
143 DEF_BENCH(return new VertBench(kTexture_VertFlag | kPersp_VertFlag | kBilerp_VertFlag);)
144 DEF_BENCH(return new VertBench(kColors_VertFlag  | kPersp_VertFlag);)
145 DEF_BENCH(return new VertBench(kTexture_VertFlag);)
146 DEF_BENCH(return new VertBench(kTexture_VertFlag | kBilerp_VertFlag);)
147 DEF_BENCH(return new VertBench(kColors_VertFlag);)
148 DEF_BENCH(return new VertBench(kColors_VertFlag | kTexture_VertFlag);)
149 DEF_BENCH(return new VertBench(kColors_VertFlag | kTexture_VertFlag | kBilerp_VertFlag);)
150 
151 /////////////////////////////////////////////////////////////////////////////////////////////////
152 
153 #include "include/core/SkRSXform.h"
154 #include "src/base/SkRandom.h"
155 #include "tools/Resources.h"
156 
157 enum AtlasFlags {
158     kColors_Flag = 1 << 0,
159     kRotate_Flag = 1 << 1,
160     kPersp_Flag  = 1 << 2,
161 };
162 
163 class AtlasBench : public Benchmark {
164     unsigned fFlags;
165     SkString fName;
166 
167     static constexpr int W = 640;
168     static constexpr int H = 480;
169     static constexpr int N = 10*1000;
170 
171     sk_sp<SkImage>  fAtlas;
172     SkRSXform       fXforms[N];
173     SkRect          fRects[N];
174     SkColor         fColors[N];
175 
176 public:
AtlasBench(unsigned flags)177     AtlasBench(unsigned flags) : fFlags(flags) {
178         fName.printf("drawAtlas");
179         if (flags & kColors_Flag) {
180             fName.append("_colors");
181         }
182         if (flags & kRotate_Flag) {
183             fName.append("_rotated");
184         }
185         if (flags & kPersp_Flag) {
186             fName.append("_persp");
187         }
188     }
~AtlasBench()189     ~AtlasBench() override {}
190 
191 protected:
onGetName()192     const char* onGetName() override { return fName.c_str(); }
onDelayedSetup()193     void onDelayedSetup() override {
194         fAtlas = ToolUtils::GetResourceAsImage("images/mandrill_256.png");
195         if (fAtlas) {
196             fAtlas = fAtlas->makeRasterImage();
197         }
198 
199         const SkScalar imageW = fAtlas->width();
200         const SkScalar imageH = fAtlas->height();
201         SkScalar scos = 1;
202         SkScalar ssin = 0;
203         if (fFlags & kRotate_Flag) {
204             scos = 0.866025403784439f;  // sqrt(3)/2
205             ssin = 0.5f;
206         }
207 
208         SkRandom rand;
209         for (int i = 0; i < N; ++i) {
210             fRects[i] = SkRect::MakeXYWH(rand.nextF() * (imageW - 8),
211                                          rand.nextF() * (imageH - 8), 8, 8);
212             fColors[i] = rand.nextU() | 0xFF000000;
213             fXforms[i] = SkRSXform::Make(scos, ssin, rand.nextF() * W, rand.nextF() * H);
214         }
215     }
onDraw(int loops,SkCanvas * canvas)216     void onDraw(int loops, SkCanvas* canvas) override {
217         const SkRect* cullRect = nullptr;
218         const SkPaint* paintPtr = nullptr;
219         const SkColor* colors = nullptr;
220         if (fFlags & kColors_Flag) {
221             colors = fColors;
222         }
223         if (fFlags & kPersp_Flag) {
224             tiny_persp_effect(canvas);
225         }
226         for (int i = 0; i < loops; i++) {
227             canvas->drawAtlas(fAtlas.get(), fXforms, fRects, colors, N, SkBlendMode::kModulate,
228                               SkSamplingOptions(), cullRect, paintPtr);
229         }
230     }
231 private:
232     using INHERITED = Benchmark;
233 };
234 //DEF_BENCH(return new AtlasBench(0);)
235 //DEF_BENCH(return new AtlasBench(kColors_Flag);)
236 DEF_BENCH(return new AtlasBench(0);)
237 DEF_BENCH(return new AtlasBench(kRotate_Flag);)
238 DEF_BENCH(return new AtlasBench(kPersp_Flag);)
239 DEF_BENCH(return new AtlasBench(kColors_Flag);)
240 DEF_BENCH(return new AtlasBench(kColors_Flag | kRotate_Flag);)
241 
242