2edf0c6a71
BUG=skia: GOLD_TRYBOT_URL= https://gold.skia.org/search?issue=2130183003 CQ_EXTRA_TRYBOTS=client.skia:Test-Ubuntu-GCC-GCE-CPU-AVX2-x86_64-Release-SKNX_NO_SIMD-Trybot Review-Url: https://codereview.chromium.org/2130183003
221 lines
7.0 KiB
C++
221 lines
7.0 KiB
C++
/*
|
|
* Copyright 2016 Google Inc.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
* found in the LICENSE file.
|
|
*/
|
|
|
|
#include <tuple>
|
|
|
|
#include "Benchmark.h"
|
|
#include "Resources.h"
|
|
#include "SkCpu.h"
|
|
#include "SkImage.h"
|
|
#include "SkImage_Base.h"
|
|
#include "SkNx.h"
|
|
#include "SkOpts.h"
|
|
#include "SkPM4fPriv.h"
|
|
#include "SkString.h"
|
|
|
|
#define INNER_LOOPS 10
|
|
|
|
static void brute_force_srcover_srgb_srgb(
|
|
uint32_t* dst, const uint32_t* const src, int ndst, const int nsrc) {
|
|
while (ndst > 0) {
|
|
int n = SkTMin(ndst, nsrc);
|
|
|
|
for (int i = 0; i < n; i++) {
|
|
srcover_blend_srgb8888_srgb_1(dst++, srgb_to_linear(to_4f(src[i])));
|
|
}
|
|
ndst -= n;
|
|
}
|
|
}
|
|
|
|
static void best_non_simd_srcover_srgb_srgb(
|
|
uint32_t* dst, const uint32_t* const src, int ndst, const int nsrc) {
|
|
uint64_t* ddst = reinterpret_cast<uint64_t*>(dst);
|
|
|
|
auto srcover_srgb_srgb_2 = [](uint32_t* dst, const uint32_t* src) {
|
|
srcover_srgb8888_srgb_1(dst++, *src++);
|
|
srcover_srgb8888_srgb_1(dst, *src);
|
|
};
|
|
|
|
while (ndst >0) {
|
|
int count = SkTMin(ndst, nsrc);
|
|
ndst -= count;
|
|
const uint64_t* dsrc = reinterpret_cast<const uint64_t*>(src);
|
|
const uint64_t* end = dsrc + (count >> 1);
|
|
do {
|
|
if ((~*dsrc & 0xFF000000FF000000) == 0) {
|
|
do {
|
|
*ddst++ = *dsrc++;
|
|
} while (dsrc < end && (~*dsrc & 0xFF000000FF000000) == 0);
|
|
} else if ((*dsrc & 0xFF000000FF000000) == 0) {
|
|
do {
|
|
dsrc++;
|
|
ddst++;
|
|
} while (dsrc < end && (*dsrc & 0xFF000000FF000000) == 0);
|
|
} else {
|
|
srcover_srgb_srgb_2(reinterpret_cast<uint32_t*>(ddst++),
|
|
reinterpret_cast<const uint32_t*>(dsrc++));
|
|
}
|
|
} while (dsrc < end);
|
|
|
|
if ((count & 1) != 0) {
|
|
srcover_srgb8888_srgb_1(reinterpret_cast<uint32_t*>(ddst),
|
|
*reinterpret_cast<const uint32_t*>(dsrc));
|
|
}
|
|
}
|
|
}
|
|
|
|
static void trivial_srcover_srgb_srgb(
|
|
uint32_t* dst, const uint32_t* const src, int ndst, const int nsrc) {
|
|
while (ndst > 0) {
|
|
int n = SkTMin(ndst, nsrc);
|
|
|
|
for (int i = 0; i < n; i++) {
|
|
srcover_srgb8888_srgb_1(dst++, src[i]);
|
|
}
|
|
ndst -= n;
|
|
}
|
|
}
|
|
|
|
class SrcOverVSkOptsBruteForce {
|
|
public:
|
|
static SkString Name() { return SkString{"VSkOptsBruteForce"}; }
|
|
static bool WorksOnCpu() { return true; }
|
|
static void BlendN(uint32_t* dst, const uint32_t* src, int count) {
|
|
brute_force_srcover_srgb_srgb(dst, src, count, count);
|
|
}
|
|
};
|
|
|
|
class SrcOverVSkOptsTrivial {
|
|
public:
|
|
static SkString Name() { return SkString{"VSkOptsTrivial"}; }
|
|
static bool WorksOnCpu() { return true; }
|
|
static void BlendN(uint32_t* dst, const uint32_t* src, int count) {
|
|
trivial_srcover_srgb_srgb(dst, src, count, count);
|
|
}
|
|
};
|
|
|
|
class SrcOverVSkOptsNonSimdCore {
|
|
public:
|
|
static SkString Name() { return SkString{"VSkOptsNonSimdCore"}; }
|
|
static bool WorksOnCpu() { return true; }
|
|
static void BlendN(uint32_t* dst, const uint32_t* src, int count) {
|
|
best_non_simd_srcover_srgb_srgb(dst, src, count, count);
|
|
}
|
|
};
|
|
|
|
namespace sk_default {
|
|
extern void srcover_srgb_srgb(
|
|
uint32_t* dst, const uint32_t* const srcStart, int ndst, const int nsrc);
|
|
}
|
|
|
|
class SrcOverVSkOptsDefault {
|
|
public:
|
|
static SkString Name() { return SkString{"VSkOptsDefault"}; }
|
|
static bool WorksOnCpu() { return true; }
|
|
static void BlendN(uint32_t* dst, const uint32_t* src, int count) {
|
|
sk_default::srcover_srgb_srgb(dst, src, count, count);
|
|
}
|
|
};
|
|
|
|
namespace sk_sse41 {
|
|
extern void srcover_srgb_srgb(
|
|
uint32_t* dst, const uint32_t* const srcStart, int ndst, const int nsrc);
|
|
}
|
|
|
|
class SrcOverVSkOptsSSE41 {
|
|
public:
|
|
static SkString Name() { return SkString{"VSkOptsSSE41"}; }
|
|
static bool WorksOnCpu() { return SkCpu::Supports(SkCpu::SSE41); }
|
|
static void BlendN(uint32_t* dst, const uint32_t* src, int count) {
|
|
sk_sse41::srcover_srgb_srgb(dst, src, count, count);
|
|
}
|
|
};
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
template <typename Blender>
|
|
class LinearSrcOverBench : public Benchmark {
|
|
public:
|
|
LinearSrcOverBench(const char* fileName) : fFileName(fileName) {
|
|
fName = "LinearSrcOver_";
|
|
fName.append(fileName);
|
|
fName.append(Blender::Name());
|
|
}
|
|
|
|
protected:
|
|
bool isSuitableFor(Backend backend) override {
|
|
return backend == kNonRendering_Backend && Blender::WorksOnCpu();
|
|
}
|
|
const char* onGetName() override { return fName.c_str(); }
|
|
|
|
void onPreDraw(SkCanvas*) override {
|
|
if (!fPixmap.addr()) {
|
|
sk_sp<SkImage> image = GetResourceAsImage(fFileName.c_str());
|
|
SkBitmap bm;
|
|
if (!as_IB(image)->getROPixels(&bm)) {
|
|
SkFAIL("Could not read resource");
|
|
}
|
|
bm.peekPixels(&fPixmap);
|
|
fCount = fPixmap.rowBytesAsPixels();
|
|
fDst.reset(fCount);
|
|
sk_bzero(fDst.get(), fPixmap.rowBytes());
|
|
}
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas*) override {
|
|
SkASSERT(fPixmap.colorType() == kN32_SkColorType);
|
|
|
|
const int width = fPixmap.rowBytesAsPixels();
|
|
|
|
for (int i = 0; i < loops * INNER_LOOPS; ++i) {
|
|
const uint32_t* src = fPixmap.addr32();
|
|
for (int y = 0; y < fPixmap.height(); y++) {
|
|
Blender::BlendN(fDst.get(), src, width);
|
|
src += width;
|
|
}
|
|
}
|
|
}
|
|
|
|
void onPostDraw(SkCanvas*) override {
|
|
// Make sure the compiler does not optimize away the operation.
|
|
volatile uint32_t v = 0;
|
|
for (int i = 0; i < fCount; i++) {
|
|
v ^= fDst[i];
|
|
}
|
|
}
|
|
|
|
private:
|
|
int fCount;
|
|
SkAutoTArray<uint32_t> fDst;
|
|
SkString fFileName;
|
|
SkString fName;
|
|
SkPixmap fPixmap;
|
|
|
|
typedef Benchmark INHERITED;
|
|
};
|
|
|
|
#if defined(SK_CPU_X86) && !defined(SK_BUILD_NO_OPTS)
|
|
#define BENCHES(fileName) \
|
|
DEF_BENCH( return new LinearSrcOverBench<SrcOverVSkOptsBruteForce>(fileName); ) \
|
|
DEF_BENCH( return new LinearSrcOverBench<SrcOverVSkOptsTrivial>(fileName); ) \
|
|
DEF_BENCH( return new LinearSrcOverBench<SrcOverVSkOptsNonSimdCore>(fileName); ) \
|
|
DEF_BENCH( return new LinearSrcOverBench<SrcOverVSkOptsDefault>(fileName); ) \
|
|
DEF_BENCH( return new LinearSrcOverBench<SrcOverVSkOptsSSE41>(fileName); )
|
|
#else
|
|
#define BENCHES(fileName) \
|
|
DEF_BENCH( return new LinearSrcOverBench<SrcOverVSkOptsBruteForce>(fileName); ) \
|
|
DEF_BENCH( return new LinearSrcOverBench<SrcOverVSkOptsTrivial>(fileName); ) \
|
|
DEF_BENCH( return new LinearSrcOverBench<SrcOverVSkOptsNonSimdCore>(fileName); ) \
|
|
DEF_BENCH( return new LinearSrcOverBench<SrcOverVSkOptsDefault>(fileName); )
|
|
#endif
|
|
|
|
BENCHES("yellow_rose.png")
|
|
BENCHES("baby_tux.png")
|
|
BENCHES("plane.png")
|
|
BENCHES("mandrill_512.png")
|
|
BENCHES("iconstrip.png")
|