5ec22387ff
Change-Id: I08b4a7bc088705b93a0aa680f6733d09c7ad23dd Reviewed-on: https://skia-review.googlesource.com/c/skia/+/354221 Reviewed-by: Brian Salomon <bsalomon@google.com> Commit-Queue: Mike Reed <reed@google.com>
385 lines
13 KiB
C++
385 lines
13 KiB
C++
/*
|
|
* Copyright 2012 Google Inc.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
* found in the LICENSE file.
|
|
*/
|
|
|
|
#include "bench/Benchmark.h"
|
|
#include "include/core/SkBitmap.h"
|
|
#include "include/core/SkCanvas.h"
|
|
#include "include/core/SkM44.h"
|
|
#include "include/core/SkPaint.h"
|
|
#include "include/core/SkShader.h"
|
|
#include "include/core/SkString.h"
|
|
#include "include/core/SkVertices.h"
|
|
#include "include/utils/SkRandom.h"
|
|
|
|
// This bench simulates the calls Skia sees from various HTML5 canvas
|
|
// game bench marks
|
|
class GameBench : public Benchmark {
|
|
public:
|
|
enum Type {
|
|
kScale_Type,
|
|
kTranslate_Type,
|
|
kRotate_Type
|
|
};
|
|
|
|
enum Clear {
|
|
kFull_Clear,
|
|
kPartial_Clear
|
|
};
|
|
|
|
GameBench(Type type, Clear clear,
|
|
bool aligned = false, bool useAtlas = false,
|
|
bool useDrawVertices = false)
|
|
: fType(type)
|
|
, fClear(clear)
|
|
, fAligned(aligned)
|
|
, fUseAtlas(useAtlas)
|
|
, fUseDrawVertices(useDrawVertices)
|
|
, fName("game")
|
|
, fNumSaved(0)
|
|
, fInitialized(false) {
|
|
|
|
switch (fType) {
|
|
case kScale_Type:
|
|
fName.append("_scale");
|
|
break;
|
|
case kTranslate_Type:
|
|
fName.append("_trans");
|
|
break;
|
|
case kRotate_Type:
|
|
fName.append("_rot");
|
|
break;
|
|
}
|
|
|
|
if (aligned) {
|
|
fName.append("_aligned");
|
|
}
|
|
|
|
if (kPartial_Clear == clear) {
|
|
fName.append("_partial");
|
|
} else {
|
|
fName.append("_full");
|
|
}
|
|
|
|
if (useAtlas) {
|
|
fName.append("_atlas");
|
|
}
|
|
|
|
if (useDrawVertices) {
|
|
fName.append("_drawVerts");
|
|
}
|
|
|
|
// It's HTML 5 canvas, so always AA
|
|
fName.append("_aa");
|
|
}
|
|
|
|
protected:
|
|
const char* onGetName() override {
|
|
return fName.c_str();
|
|
}
|
|
|
|
void onDelayedSetup() override {
|
|
if (!fInitialized) {
|
|
this->makeCheckerboard();
|
|
this->makeAtlas();
|
|
fInitialized = true;
|
|
}
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas* canvas) override {
|
|
SkRandom scaleRand;
|
|
SkRandom transRand;
|
|
SkRandom rotRand;
|
|
|
|
int width, height;
|
|
if (fUseAtlas) {
|
|
width = kAtlasCellWidth;
|
|
height = kAtlasCellHeight;
|
|
} else {
|
|
width = kCheckerboardWidth;
|
|
height = kCheckerboardHeight;
|
|
}
|
|
|
|
SkPaint clearPaint;
|
|
clearPaint.setColor(0xFF000000);
|
|
clearPaint.setAntiAlias(true);
|
|
|
|
SkISize size = canvas->getBaseLayerSize();
|
|
|
|
SkScalar maxTransX, maxTransY;
|
|
|
|
if (kScale_Type == fType) {
|
|
maxTransX = size.fWidth - (1.5f * width);
|
|
maxTransY = size.fHeight - (1.5f * height);
|
|
} else if (kTranslate_Type == fType) {
|
|
maxTransX = SkIntToScalar(size.fWidth - width);
|
|
maxTransY = SkIntToScalar(size.fHeight - height);
|
|
} else {
|
|
SkASSERT(kRotate_Type == fType);
|
|
// Yes, some rotations will be off the top and left sides
|
|
maxTransX = size.fWidth - SK_ScalarSqrt2 * height;
|
|
maxTransY = size.fHeight - SK_ScalarSqrt2 * height;
|
|
}
|
|
|
|
SkMatrix mat;
|
|
SkRect dst = { 0, 0, SkIntToScalar(width), SkIntToScalar(height) };
|
|
SkRect clearRect = { -1.0f, -1.0f, width+1.0f, height+1.0f };
|
|
SkPoint verts[4] = { // for drawVertices path
|
|
{ 0, 0 },
|
|
{ 0, SkIntToScalar(height) },
|
|
{ SkIntToScalar(width), SkIntToScalar(height) },
|
|
{ SkIntToScalar(width), 0 }
|
|
};
|
|
uint16_t indices[6] = { 0, 1, 2, 0, 2, 3 };
|
|
|
|
SkPaint p;
|
|
p.setColor(0xFF000000);
|
|
|
|
SkPaint p2; // for drawVertices path
|
|
p2.setColor(0xFF000000);
|
|
p2.setShader(fAtlas.makeShader(SkSamplingOptions(SkFilterMode::kLinear)));
|
|
|
|
for (int i = 0; i < loops; ++i, ++fNumSaved) {
|
|
if (0 == i % kNumBeforeClear) {
|
|
if (kPartial_Clear == fClear) {
|
|
for (int j = 0; j < fNumSaved; ++j) {
|
|
canvas->setMatrix(SkMatrix::I());
|
|
mat.setTranslate(fSaved[j][0], fSaved[j][1]);
|
|
|
|
if (kScale_Type == fType) {
|
|
mat.preScale(fSaved[j][2], fSaved[j][2]);
|
|
} else if (kRotate_Type == fType) {
|
|
mat.preRotate(fSaved[j][2]);
|
|
}
|
|
|
|
canvas->concat(mat);
|
|
canvas->drawRect(clearRect, clearPaint);
|
|
}
|
|
} else {
|
|
canvas->clear(0xFF000000);
|
|
}
|
|
|
|
fNumSaved = 0;
|
|
}
|
|
|
|
SkASSERT(fNumSaved < kNumBeforeClear);
|
|
|
|
canvas->setMatrix(SkMatrix::I());
|
|
|
|
fSaved[fNumSaved][0] = transRand.nextRangeScalar(0.0f, maxTransX);
|
|
fSaved[fNumSaved][1] = transRand.nextRangeScalar(0.0f, maxTransY);
|
|
if (fAligned) {
|
|
// make the translations integer aligned
|
|
fSaved[fNumSaved][0] = SkScalarFloorToScalar(fSaved[fNumSaved][0]);
|
|
fSaved[fNumSaved][1] = SkScalarFloorToScalar(fSaved[fNumSaved][1]);
|
|
}
|
|
|
|
mat.setTranslate(fSaved[fNumSaved][0], fSaved[fNumSaved][1]);
|
|
|
|
if (kScale_Type == fType) {
|
|
fSaved[fNumSaved][2] = scaleRand.nextRangeScalar(0.5f, 1.5f);
|
|
mat.preScale(fSaved[fNumSaved][2], fSaved[fNumSaved][2]);
|
|
} else if (kRotate_Type == fType) {
|
|
fSaved[fNumSaved][2] = rotRand.nextRangeScalar(0.0f, 360.0f);
|
|
mat.preRotate(fSaved[fNumSaved][2]);
|
|
}
|
|
|
|
canvas->concat(mat);
|
|
if (fUseAtlas) {
|
|
const int curCell = i % (kNumAtlasedX * kNumAtlasedY);
|
|
SkIRect src = fAtlasRects[curCell % (kNumAtlasedX)][curCell / (kNumAtlasedX)];
|
|
|
|
if (fUseDrawVertices) {
|
|
SkPoint uvs[4] = {
|
|
{ SkIntToScalar(src.fLeft), SkIntToScalar(src.fBottom) },
|
|
{ SkIntToScalar(src.fLeft), SkIntToScalar(src.fTop) },
|
|
{ SkIntToScalar(src.fRight), SkIntToScalar(src.fTop) },
|
|
{ SkIntToScalar(src.fRight), SkIntToScalar(src.fBottom) },
|
|
};
|
|
canvas->drawVertices(SkVertices::MakeCopy(SkVertices::kTriangles_VertexMode,
|
|
4, verts, uvs, nullptr, 6, indices),
|
|
SkBlendMode::kModulate, p2);
|
|
} else {
|
|
canvas->drawBitmapRect(fAtlas, src, dst, &p,
|
|
SkCanvas::kFast_SrcRectConstraint);
|
|
}
|
|
} else {
|
|
canvas->drawBitmapRect(fCheckerboard, dst, &p);
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
static const int kCheckerboardWidth = 64;
|
|
static const int kCheckerboardHeight = 128;
|
|
|
|
static const int kAtlasCellWidth = 48;
|
|
static const int kAtlasCellHeight = 36;
|
|
static const int kNumAtlasedX = 5;
|
|
static const int kNumAtlasedY = 5;
|
|
static const int kAtlasSpacer = 2;
|
|
static const int kTotAtlasWidth = kNumAtlasedX * kAtlasCellWidth +
|
|
(kNumAtlasedX+1) * kAtlasSpacer;
|
|
static const int kTotAtlasHeight = kNumAtlasedY * kAtlasCellHeight +
|
|
(kNumAtlasedY+1) * kAtlasSpacer;
|
|
static const int kNumBeforeClear = 100;
|
|
|
|
Type fType;
|
|
Clear fClear;
|
|
bool fAligned;
|
|
bool fUseAtlas;
|
|
bool fUseDrawVertices;
|
|
SkString fName;
|
|
int fNumSaved; // num draws stored in 'fSaved'
|
|
bool fInitialized;
|
|
|
|
// 0 & 1 are always x & y translate. 2 is either scale or rotate.
|
|
SkScalar fSaved[kNumBeforeClear][3];
|
|
|
|
SkBitmap fCheckerboard;
|
|
SkBitmap fAtlas;
|
|
SkIRect fAtlasRects[kNumAtlasedX][kNumAtlasedY];
|
|
|
|
// Note: the resulting checker board has transparency
|
|
void makeCheckerboard() {
|
|
static int kCheckSize = 16;
|
|
|
|
fCheckerboard.allocN32Pixels(kCheckerboardWidth, kCheckerboardHeight);
|
|
for (int y = 0; y < kCheckerboardHeight; ++y) {
|
|
int even = (y / kCheckSize) % 2;
|
|
|
|
SkPMColor* scanline = fCheckerboard.getAddr32(0, y);
|
|
|
|
for (int x = 0; x < kCheckerboardWidth; ++x) {
|
|
if (even == (x / kCheckSize) % 2) {
|
|
*scanline++ = 0xFFFF0000;
|
|
} else {
|
|
*scanline++ = 0x00000000;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Note: the resulting atlas has transparency
|
|
void makeAtlas() {
|
|
SkRandom rand;
|
|
|
|
SkColor colors[kNumAtlasedX][kNumAtlasedY];
|
|
|
|
for (int y = 0; y < kNumAtlasedY; ++y) {
|
|
for (int x = 0; x < kNumAtlasedX; ++x) {
|
|
colors[x][y] = rand.nextU() | 0xff000000;
|
|
fAtlasRects[x][y] = SkIRect::MakeXYWH(kAtlasSpacer + x * (kAtlasCellWidth + kAtlasSpacer),
|
|
kAtlasSpacer + y * (kAtlasCellHeight + kAtlasSpacer),
|
|
kAtlasCellWidth,
|
|
kAtlasCellHeight);
|
|
}
|
|
}
|
|
|
|
fAtlas.allocN32Pixels(kTotAtlasWidth, kTotAtlasHeight);
|
|
|
|
for (int y = 0; y < kTotAtlasHeight; ++y) {
|
|
int colorY = y / (kAtlasCellHeight + kAtlasSpacer);
|
|
bool inColorY = (y % (kAtlasCellHeight + kAtlasSpacer)) >= kAtlasSpacer;
|
|
|
|
SkPMColor* scanline = fAtlas.getAddr32(0, y);
|
|
|
|
for (int x = 0; x < kTotAtlasWidth; ++x, ++scanline) {
|
|
int colorX = x / (kAtlasCellWidth + kAtlasSpacer);
|
|
bool inColorX = (x % (kAtlasCellWidth + kAtlasSpacer)) >= kAtlasSpacer;
|
|
|
|
if (inColorX && inColorY) {
|
|
SkASSERT(colorX < kNumAtlasedX && colorY < kNumAtlasedY);
|
|
*scanline = colors[colorX][colorY];
|
|
} else {
|
|
*scanline = 0x00000000;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
using INHERITED = Benchmark;
|
|
};
|
|
|
|
// Partial clear
|
|
DEF_BENCH(return new GameBench(GameBench::kScale_Type, GameBench::kPartial_Clear);)
|
|
DEF_BENCH(return new GameBench(GameBench::kTranslate_Type, GameBench::kPartial_Clear);)
|
|
DEF_BENCH(return new GameBench(GameBench::kTranslate_Type, GameBench::kPartial_Clear, true);)
|
|
DEF_BENCH(return new GameBench(GameBench::kRotate_Type, GameBench::kPartial_Clear);)
|
|
|
|
// Full clear
|
|
DEF_BENCH(return new GameBench(GameBench::kScale_Type, GameBench::kFull_Clear);)
|
|
DEF_BENCH(return new GameBench(GameBench::kTranslate_Type, GameBench::kFull_Clear);)
|
|
DEF_BENCH(return new GameBench(GameBench::kTranslate_Type, GameBench::kFull_Clear, true);)
|
|
DEF_BENCH(return new GameBench(GameBench::kRotate_Type, GameBench::kFull_Clear);)
|
|
|
|
// Atlased
|
|
DEF_BENCH(return new GameBench(GameBench::kTranslate_Type, GameBench::kFull_Clear, false, true);)
|
|
DEF_BENCH(return new GameBench(
|
|
GameBench::kTranslate_Type, GameBench::kFull_Clear, false, true, true);)
|
|
|
|
|
|
class CanvasMatrixBench : public Benchmark {
|
|
SkString fName;
|
|
public:
|
|
enum Type {
|
|
kTranslate_Type,
|
|
kScale_Type,
|
|
k2x3_Type,
|
|
k3x3_Type,
|
|
k4x4_Type,
|
|
};
|
|
Type fType;
|
|
|
|
CanvasMatrixBench(Type t) : fType(t) {
|
|
fName.set("canvas_matrix");
|
|
switch (fType) {
|
|
case kTranslate_Type: fName.append("_trans"); break;
|
|
case kScale_Type: fName.append("_scale"); break;
|
|
case k2x3_Type: fName.append("_2x3"); break;
|
|
case k3x3_Type: fName.append("_3x3"); break;
|
|
case k4x4_Type: fName.append("_4x4"); break;
|
|
}
|
|
}
|
|
|
|
protected:
|
|
const char* onGetName() override {
|
|
return fName.c_str();
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas* canvas) override {
|
|
SkMatrix m;
|
|
m.setRotate(1);
|
|
if (fType == k3x3_Type) {
|
|
m[7] = 0.0001f;
|
|
}
|
|
SkM44 m4(m);
|
|
|
|
for (int i = 0; i < loops; ++i) {
|
|
canvas->save();
|
|
for (int j = 0; j < 10000; ++j) {
|
|
switch (fType) {
|
|
case kTranslate_Type: canvas->translate(0.0001f, 0.0001f); break;
|
|
case kScale_Type: canvas->scale(1.0001f, 0.9999f); break;
|
|
case k2x3_Type: canvas->concat(m); break;
|
|
case k3x3_Type: canvas->concat(m); break;
|
|
case k4x4_Type: canvas->concat(m4); break;
|
|
}
|
|
}
|
|
canvas->restore();
|
|
}
|
|
}
|
|
|
|
private:
|
|
using INHERITED = Benchmark;
|
|
};
|
|
|
|
DEF_BENCH(return new CanvasMatrixBench(CanvasMatrixBench::kTranslate_Type));
|
|
DEF_BENCH(return new CanvasMatrixBench(CanvasMatrixBench::kScale_Type));
|
|
DEF_BENCH(return new CanvasMatrixBench(CanvasMatrixBench::k2x3_Type));
|
|
DEF_BENCH(return new CanvasMatrixBench(CanvasMatrixBench::k3x3_Type));
|
|
DEF_BENCH(return new CanvasMatrixBench(CanvasMatrixBench::k4x4_Type));
|