358fcad1b8
Bug: skia: Change-Id: Id73c983cc71d39fe587d355e690261627fa63aee Reviewed-on: https://skia-review.googlesource.com/c/172643 Reviewed-by: Mike Reed <reed@google.com> Commit-Queue: Mike Reed <reed@google.com>
182 lines
5.4 KiB
C++
182 lines
5.4 KiB
C++
/*
|
|
* Copyright 2011 Google Inc.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
* found in the LICENSE file.
|
|
*/
|
|
|
|
#include "Benchmark.h"
|
|
#include "Resources.h"
|
|
#include "SkCanvas.h"
|
|
#include "SkPaint.h"
|
|
#include "SkRandom.h"
|
|
#include "SkStream.h"
|
|
#include "SkString.h"
|
|
#include "SkTemplates.h"
|
|
#include "SkTypeface.h"
|
|
|
|
#ifdef SK_SUPPORT_LEGACY_PAINT_TEXTMEASURE
|
|
|
|
enum FontQuality {
|
|
kBW,
|
|
kAA,
|
|
kLCD,
|
|
};
|
|
|
|
static const char* fontQualityName(const SkPaint& paint) {
|
|
if (!paint.isAntiAlias()) {
|
|
return "BW";
|
|
}
|
|
if (paint.isLCDRenderText()) {
|
|
return "LCD";
|
|
}
|
|
return "AA";
|
|
}
|
|
|
|
/* Some considerations for performance:
|
|
short -vs- long strings (measuring overhead)
|
|
tiny -vs- large pointsize (measure blit -vs- overhead)
|
|
1 -vs- many point sizes (measure cache lookup)
|
|
normal -vs- subpixel -vs- lineartext (minor)
|
|
force purge after each draw to measure scaler
|
|
textencoding?
|
|
text -vs- postext - pathtext
|
|
*/
|
|
class TextBench : public Benchmark {
|
|
SkPaint fPaint;
|
|
SkString fText;
|
|
SkString fName;
|
|
FontQuality fFQ;
|
|
bool fDoPos;
|
|
bool fDoColorEmoji;
|
|
sk_sp<SkTypeface> fColorEmojiTypeface;
|
|
SkPoint* fPos;
|
|
public:
|
|
TextBench(const char text[], int ps,
|
|
SkColor color, FontQuality fq, bool doColorEmoji = false, bool doPos = false)
|
|
: fText(text)
|
|
, fFQ(fq)
|
|
, fDoPos(doPos)
|
|
, fDoColorEmoji(doColorEmoji)
|
|
, fPos(nullptr) {
|
|
fPaint.setAntiAlias(kBW != fq);
|
|
fPaint.setLCDRenderText(kLCD == fq);
|
|
fPaint.setTextSize(SkIntToScalar(ps));
|
|
fPaint.setColor(color);
|
|
}
|
|
|
|
~TextBench() override {
|
|
delete[] fPos;
|
|
}
|
|
|
|
protected:
|
|
void onDelayedSetup() override {
|
|
if (fDoColorEmoji) {
|
|
SkASSERT(kBW == fFQ);
|
|
fColorEmojiTypeface = MakeResourceAsTypeface("fonts/Funkster.ttf");
|
|
}
|
|
|
|
if (fDoPos) {
|
|
size_t len = fText.size();
|
|
SkScalar* adv = new SkScalar[len];
|
|
fPaint.getTextWidths(fText.c_str(), len, adv);
|
|
fPos = new SkPoint[len];
|
|
SkScalar x = 0;
|
|
for (size_t i = 0; i < len; ++i) {
|
|
fPos[i].set(x, SkIntToScalar(50));
|
|
x += adv[i];
|
|
}
|
|
delete[] adv;
|
|
}
|
|
}
|
|
|
|
|
|
const char* onGetName() override {
|
|
fName.printf("text_%g", SkScalarToFloat(fPaint.getTextSize()));
|
|
if (fDoPos) {
|
|
fName.append("_pos");
|
|
}
|
|
fName.appendf("_%s", fontQualityName(fPaint));
|
|
if (SK_ColorBLACK == fPaint.getColor()) {
|
|
fName.append("_BK");
|
|
} else if (SK_ColorWHITE == fPaint.getColor()) {
|
|
fName.append("_WT");
|
|
} else {
|
|
fName.appendf("_%02X", fPaint.getAlpha());
|
|
}
|
|
|
|
if (fDoColorEmoji) {
|
|
fName.append("_ColorEmoji");
|
|
}
|
|
|
|
return fName.c_str();
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas* canvas) override {
|
|
const SkIPoint dim = this->getSize();
|
|
SkRandom rand;
|
|
|
|
SkPaint paint(fPaint);
|
|
this->setupPaint(&paint);
|
|
// explicitly need these
|
|
paint.setColor(fPaint.getColor());
|
|
paint.setAntiAlias(kBW != fFQ);
|
|
paint.setLCDRenderText(kLCD == fFQ);
|
|
|
|
if (fDoColorEmoji && fColorEmojiTypeface) {
|
|
paint.setTypeface(fColorEmojiTypeface);
|
|
}
|
|
|
|
const SkScalar x0 = SkIntToScalar(-10);
|
|
const SkScalar y0 = SkIntToScalar(-10);
|
|
|
|
if (fDoPos) {
|
|
// realistically, the matrix is often at least translated, so we
|
|
// do that since it exercises different code in drawPosText.
|
|
canvas->translate(SK_Scalar1, SK_Scalar1);
|
|
}
|
|
|
|
for (int i = 0; i < loops; i++) {
|
|
if (fDoPos) {
|
|
canvas->drawPosText(fText.c_str(), fText.size(), fPos, paint);
|
|
} else {
|
|
SkScalar x = x0 + rand.nextUScalar1() * dim.fX;
|
|
SkScalar y = y0 + rand.nextUScalar1() * dim.fY;
|
|
canvas->drawString(fText, x, y, paint);
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
typedef Benchmark INHERITED;
|
|
};
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define STR "Hamburgefons"
|
|
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFFFFFFFF, kBW); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFF000000, kBW); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFFFF0000, kBW); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0x88FF0000, kBW); )
|
|
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFFFFFFFF, kAA); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFF000000, kAA); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFFFF0000, kAA); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0x88FF0000, kAA); )
|
|
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFFFFFFFF, kLCD); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFF000000, kLCD); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFFFF0000, kLCD); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0x88FF0000, kLCD); )
|
|
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFFFFFFFF, kBW, true); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFF000000, kBW, true); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFFFF0000, kBW, true); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0x88FF0000, kBW, true); )
|
|
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFF000000, kBW, true, true); )
|
|
DEF_BENCH( return new TextBench(STR, 16, 0xFF000000, kAA, false, true); )
|
|
|
|
#endif
|