1 /*
2  * Copyright 2013 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 // GM to stress the GPU font cache
9 // It's not necessary to run this with CPU configs
10 
11 #include "gm.h"
12 
13 #include "GrContext.h"
14 #include "GrContextPriv.h"
15 #include "GrContextOptions.h"
16 #include "SkCanvas.h"
17 #include "SkGraphics.h"
18 #include "SkImage.h"
19 #include "SkTypeface.h"
20 #include "gm.h"
21 #include "sk_tool_utils.h"
22 
draw_string(SkCanvas * canvas,const SkString & text,SkScalar x,SkScalar y,const SkFont & font)23 static SkScalar draw_string(SkCanvas* canvas, const SkString& text, SkScalar x,
24                            SkScalar y, const SkFont& font) {
25     SkPaint paint;
26     canvas->drawString(text, x, y, font, paint);
27     return x + font.measureText(text.c_str(), text.size(), kUTF8_SkTextEncoding);
28 }
29 
30 class FontCacheGM : public skiagm::GpuGM {
31 public:
FontCacheGM(GrContextOptions::Enable allowMultipleTextures)32     FontCacheGM(GrContextOptions::Enable allowMultipleTextures)
33         : fAllowMultipleTextures(allowMultipleTextures) {
34         this->setBGColor(SK_ColorLTGRAY);
35     }
36 
modifyGrContextOptions(GrContextOptions * options)37     void modifyGrContextOptions(GrContextOptions* options) override {
38         options->fGlyphCacheTextureMaximumBytes = 0;
39         options->fAllowMultipleGlyphCacheTextures = fAllowMultipleTextures;
40     }
41 
42 protected:
onShortName()43     SkString onShortName() override {
44         SkString name("fontcache");
45         if (GrContextOptions::Enable::kYes == fAllowMultipleTextures) {
46             name.append("-mt");
47         }
48         return name;
49     }
50 
onISize()51     SkISize onISize() override { return SkISize::Make(kSize, kSize); }
52 
onOnceBeforeDraw()53     void onOnceBeforeDraw() override {
54         fTypefaces[0] = sk_tool_utils::create_portable_typeface("serif", SkFontStyle::Italic());
55         fTypefaces[1] = sk_tool_utils::create_portable_typeface("sans-serif",SkFontStyle::Italic());
56         fTypefaces[2] = sk_tool_utils::create_portable_typeface("serif", SkFontStyle::Normal());
57         fTypefaces[3] =
58                 sk_tool_utils::create_portable_typeface("sans-serif", SkFontStyle::Normal());
59         fTypefaces[4] = sk_tool_utils::create_portable_typeface("serif", SkFontStyle::Bold());
60         fTypefaces[5] = sk_tool_utils::create_portable_typeface("sans-serif", SkFontStyle::Bold());
61     }
62 
onDraw(GrContext *,GrRenderTargetContext *,SkCanvas * canvas)63     void onDraw(GrContext*, GrRenderTargetContext*, SkCanvas* canvas) override {
64         this->drawText(canvas);
65         //  Debugging tool for GPU.
66         static const bool kShowAtlas = false;
67         if (kShowAtlas) {
68             if (auto ctx = canvas->getGrContext()) {
69                 auto img = ctx->priv().testingOnly_getFontAtlasImage(kA8_GrMaskFormat);
70                 canvas->drawImage(img, 0, 0);
71             }
72         }
73     }
74 
75 private:
drawText(SkCanvas * canvas)76     void drawText(SkCanvas* canvas) {
77         static const int kSizes[] = {8, 9, 10, 11, 12, 13, 18, 20, 25};
78 
79         static const SkString kTexts[] = {SkString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"),
80                                           SkString("abcdefghijklmnopqrstuvwxyz"),
81                                           SkString("0123456789"),
82                                           SkString("!@#$%^&*()<>[]{}")};
83         SkFont font;
84         font.setEdging(SkFont::Edging::kAntiAlias);
85         font.setSubpixel(true);
86 
87         static const SkScalar kSubPixelInc = 1 / 2.f;
88         SkScalar x = 0;
89         SkScalar y = 10;
90         SkScalar subpixelX = 0;
91         SkScalar subpixelY = 0;
92         bool offsetX = true;
93 
94         if (GrContextOptions::Enable::kYes == fAllowMultipleTextures) {
95             canvas->scale(10, 10);
96         }
97 
98         do {
99             for (auto s : kSizes) {
100                 auto size = 2 * s;
101                 font.setSize(size);
102                 for (const auto& typeface : fTypefaces) {
103                     font.setTypeface(typeface);
104                     for (const auto& text : kTexts) {
105                         x = size + draw_string(canvas, text, x + subpixelX, y + subpixelY, font);
106                         x = SkScalarCeilToScalar(x);
107                         if (x + 100 > kSize) {
108                             x = 0;
109                             y += SkScalarCeilToScalar(size + 3);
110                             if (y > kSize) {
111                                 return;
112                             }
113                         }
114                     }
115                 }
116                 (offsetX ? subpixelX : subpixelY) += kSubPixelInc;
117                 offsetX = !offsetX;
118             }
119         } while (true);
120     }
121 
122     static constexpr SkScalar kSize = 1280;
123 
124     GrContextOptions::Enable fAllowMultipleTextures;
125     sk_sp<SkTypeface> fTypefaces[6];
126     typedef GM INHERITED;
127 };
128 
129 constexpr SkScalar FontCacheGM::kSize;
130 
131 //////////////////////////////////////////////////////////////////////////////
132 
133 DEF_GM(return new FontCacheGM(GrContextOptions::Enable::kNo))
134 DEF_GM(return new FontCacheGM(GrContextOptions::Enable::kYes))
135