blob: 11343dc3347bbb6a5e7a4a8e54fcc4a904aa4de5 [file] [log] [blame]
msarett2cee9022016-06-03 08:25:21 -07001/*
2 * Copyright 2016 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 "ColorCodecBench.h"
9#include "Resources.h"
10#include "SkCodec.h"
msarettc0444612016-09-16 11:45:58 -070011#include "SkCodecPriv.h"
raftias94888332016-10-18 10:02:51 -070012#include "SkColorSpace_XYZ.h"
msarett2cee9022016-06-03 08:25:21 -070013#include "SkColorSpaceXform.h"
14#include "SkCommandLineFlags.h"
15
msarett469f1c52016-06-06 08:20:37 -070016#if defined(SK_TEST_QCMS)
msarettd2809572016-06-20 06:07:45 -070017DEFINE_bool(qcms, false, "Bench qcms color conversion");
msarett2cee9022016-06-03 08:25:21 -070018#endif
19DEFINE_bool(xform_only, false, "Only time the color xform, do not include the decode time");
msarettd2809572016-06-20 06:07:45 -070020DEFINE_bool(srgb, false, "Convert to srgb dst space");
Matt Sarettf4898862016-10-16 10:20:41 -040021DEFINE_bool(nonstd, false, "Convert to non-standard dst space");
msarett6bdbf442016-07-19 09:07:55 -070022DEFINE_bool(half, false, "Convert to half floats");
msarett2cee9022016-06-03 08:25:21 -070023
24ColorCodecBench::ColorCodecBench(const char* name, sk_sp<SkData> encoded)
25 : fEncoded(std::move(encoded))
msarett469f1c52016-06-06 08:20:37 -070026#if defined(SK_TEST_QCMS)
msarett2cee9022016-06-03 08:25:21 -070027 , fDstSpaceQCMS(nullptr)
28#endif
29{
30 fName.appendf("Color%s", FLAGS_xform_only ? "Xform" : "Codec");
msarett469f1c52016-06-06 08:20:37 -070031#if defined(SK_TEST_QCMS)
msarett2cee9022016-06-03 08:25:21 -070032 fName.appendf("%s", FLAGS_qcms ? "QCMS" : "");
33#endif
34 fName.appendf("_%s", name);
35}
36
37const char* ColorCodecBench::onGetName() {
38 return fName.c_str();
39}
40
41bool ColorCodecBench::isSuitableFor(Backend backend) {
42 return kNonRendering_Backend == backend;
43}
44
45void ColorCodecBench::decodeAndXform() {
reed42943c82016-09-12 12:01:44 -070046 SkAutoTDelete<SkCodec> codec(SkCodec::NewFromData(fEncoded));
msarett50ce1f22016-07-29 06:23:33 -070047 SkASSERT(codec);
48
msarett2cee9022016-06-03 08:25:21 -070049#ifdef SK_DEBUG
msarett50ce1f22016-07-29 06:23:33 -070050 SkCodec::Result result =
msarett2cee9022016-06-03 08:25:21 -070051#endif
msarett50ce1f22016-07-29 06:23:33 -070052 codec->getPixels(fDstInfo, fDst.get(), fDstInfo.minRowBytes());
msarett2cee9022016-06-03 08:25:21 -070053 SkASSERT(SkCodec::kSuccess == result);
msarett2cee9022016-06-03 08:25:21 -070054}
55
msarett469f1c52016-06-06 08:20:37 -070056#if defined(SK_TEST_QCMS)
msarett2cee9022016-06-03 08:25:21 -070057void ColorCodecBench::decodeAndXformQCMS() {
reed42943c82016-09-12 12:01:44 -070058 SkAutoTDelete<SkCodec> codec(SkCodec::NewFromData(fEncoded));
msarett2cee9022016-06-03 08:25:21 -070059#ifdef SK_DEBUG
60 const SkCodec::Result result =
61#endif
msarett6bdbf442016-07-19 09:07:55 -070062 codec->startScanlineDecode(fSrcInfo);
msarett2cee9022016-06-03 08:25:21 -070063 SkASSERT(SkCodec::kSuccess == result);
64
65 SkAutoTCallVProc<qcms_profile, qcms_profile_release>
66 srcSpace(qcms_profile_from_memory(fSrcData->data(), fSrcData->size()));
67 SkASSERT(srcSpace);
68
69 SkAutoTCallVProc<qcms_transform, qcms_transform_release>
70 transform (qcms_transform_create(srcSpace, QCMS_DATA_RGBA_8, fDstSpaceQCMS.get(),
71 QCMS_DATA_RGBA_8, QCMS_INTENT_PERCEPTUAL));
72 SkASSERT(transform);
73
74#ifdef SK_PMCOLOR_IS_RGBA
75 qcms_output_type outType = QCMS_OUTPUT_RGBX;
76#else
77 qcms_output_type outType = QCMS_OUTPUT_BGRX;
78#endif
79
80 void* dst = fDst.get();
msarett6bdbf442016-07-19 09:07:55 -070081 for (int y = 0; y < fSrcInfo.height(); y++) {
msarett2cee9022016-06-03 08:25:21 -070082#ifdef SK_DEBUG
83 const int rows =
84#endif
85 codec->getScanlines(fSrc.get(), 1, 0);
86 SkASSERT(1 == rows);
87
msarett6bdbf442016-07-19 09:07:55 -070088 qcms_transform_data_type(transform, fSrc.get(), dst, fSrcInfo.width(), outType);
89 dst = SkTAddOffset<void>(dst, fDstInfo.minRowBytes());
msarett2cee9022016-06-03 08:25:21 -070090 }
91}
92#endif
93
94void ColorCodecBench::xformOnly() {
Brian Osman526972e2016-10-24 09:24:02 -040095 sk_sp<SkColorSpace> srcSpace = SkColorSpace::MakeICC(fSrcData->data(), fSrcData->size());
msarett2cee9022016-06-03 08:25:21 -070096 if (!srcSpace) {
Brian Osman526972e2016-10-24 09:24:02 -040097 srcSpace = SkColorSpace::MakeNamed(SkColorSpace::kSRGB_Named);
msarett2cee9022016-06-03 08:25:21 -070098 }
msarett4be0e7c2016-09-22 07:02:24 -070099 std::unique_ptr<SkColorSpaceXform> xform = SkColorSpaceXform::New(srcSpace.get(),
100 fDstSpace.get());
msarett2cee9022016-06-03 08:25:21 -0700101 SkASSERT(xform);
102
103 void* dst = fDst.get();
104 void* src = fSrc.get();
msarett6bdbf442016-07-19 09:07:55 -0700105 for (int y = 0; y < fSrcInfo.height(); y++) {
msarett31d097e82016-10-11 12:15:03 -0700106 SkAssertResult(xform->apply(select_xform_format(fDstInfo.colorType()), dst,
107 SkColorSpaceXform::kRGBA_8888_ColorFormat, src,
108 fSrcInfo.width(), fDstInfo.alphaType()));
msarett6bdbf442016-07-19 09:07:55 -0700109 dst = SkTAddOffset<void>(dst, fDstInfo.minRowBytes());
110 src = SkTAddOffset<void>(src, fSrcInfo.minRowBytes());
msarett2cee9022016-06-03 08:25:21 -0700111 }
112}
113
msarett469f1c52016-06-06 08:20:37 -0700114#if defined(SK_TEST_QCMS)
msarett2cee9022016-06-03 08:25:21 -0700115void ColorCodecBench::xformOnlyQCMS() {
116 SkAutoTCallVProc<qcms_profile, qcms_profile_release>
117 srcSpace(qcms_profile_from_memory(fSrcData->data(), fSrcData->size()));
118 SkASSERT(srcSpace);
119
120 SkAutoTCallVProc<qcms_transform, qcms_transform_release>
121 transform (qcms_transform_create(srcSpace, QCMS_DATA_RGBA_8, fDstSpaceQCMS.get(),
122 QCMS_DATA_RGBA_8, QCMS_INTENT_PERCEPTUAL));
123 SkASSERT(transform);
124
125#ifdef SK_PMCOLOR_IS_RGBA
126 qcms_output_type outType = QCMS_OUTPUT_RGBX;
127#else
128 qcms_output_type outType = QCMS_OUTPUT_BGRX;
129#endif
130
131 void* dst = fDst.get();
132 void* src = fSrc.get();
msarett6bdbf442016-07-19 09:07:55 -0700133 for (int y = 0; y < fSrcInfo.height(); y++) {
msarett2cee9022016-06-03 08:25:21 -0700134 // Transform in place
msarett6bdbf442016-07-19 09:07:55 -0700135 qcms_transform_data_type(transform, src, dst, fSrcInfo.width(), outType);
136 dst = SkTAddOffset<void>(dst, fDstInfo.minRowBytes());
137 src = SkTAddOffset<void>(src, fSrcInfo.minRowBytes());
msarett2cee9022016-06-03 08:25:21 -0700138 }
139}
140#endif
141
142void ColorCodecBench::onDelayedSetup() {
reed42943c82016-09-12 12:01:44 -0700143 SkAutoTDelete<SkCodec> codec(SkCodec::NewFromData(fEncoded));
msarett2cee9022016-06-03 08:25:21 -0700144 fSrcData = codec->getICCData();
145 sk_sp<SkData> dstData = SkData::MakeFromFileName(
msaretta714bc32016-07-29 08:58:33 -0700146 GetResourcePath("icc_profiles/HP_ZR30w.icc").c_str());
msarett2cee9022016-06-03 08:25:21 -0700147 SkASSERT(dstData);
148
msarett50ce1f22016-07-29 06:23:33 -0700149 fDstSpace = nullptr;
msarett469f1c52016-06-06 08:20:37 -0700150#if defined(SK_TEST_QCMS)
msarett2cee9022016-06-03 08:25:21 -0700151 if (FLAGS_qcms) {
msarettd2809572016-06-20 06:07:45 -0700152 fDstSpaceQCMS.reset(FLAGS_srgb ?
153 qcms_profile_sRGB() :
154 qcms_profile_from_memory(dstData->data(), dstData->size()));
msarett2cee9022016-06-03 08:25:21 -0700155 SkASSERT(fDstSpaceQCMS);
msaretta9e878c2016-06-08 14:43:53 -0700156
157 // This call takes a non-trivial amount of time, but I think it's the most fair to
158 // treat it as overhead. It only needs to happen once.
159 qcms_profile_precache_output_transform(fDstSpaceQCMS);
msarett2cee9022016-06-03 08:25:21 -0700160 } else
161#endif
162 {
Brian Osman526972e2016-10-24 09:24:02 -0400163 fDstSpace = FLAGS_srgb ? SkColorSpace::MakeNamed(SkColorSpace::kSRGB_Named) :
164 SkColorSpace::MakeICC(dstData->data(), dstData->size());
msarett2cee9022016-06-03 08:25:21 -0700165 SkASSERT(fDstSpace);
166 }
msarett50ce1f22016-07-29 06:23:33 -0700167
168 fSrcInfo = codec->getInfo().makeColorType(kRGBA_8888_SkColorType);
msarett8bbcd5a2016-09-14 07:06:08 -0700169 fDstInfo = fSrcInfo;
msarett50ce1f22016-07-29 06:23:33 -0700170
msarett50ce1f22016-07-29 06:23:33 -0700171 if (FLAGS_half) {
172 fDstInfo = fDstInfo.makeColorType(kRGBA_F16_SkColorType);
raftias94888332016-10-18 10:02:51 -0700173 SkASSERT(SkColorSpace_Base::Type::kXYZ == as_CSB(fDstSpace)->type());
174 fDstSpace = static_cast<SkColorSpace_XYZ*>(fDstSpace.get())->makeLinearGamma();
msarett50ce1f22016-07-29 06:23:33 -0700175 }
msarett2ecc35f2016-09-08 11:55:16 -0700176
Matt Sarettf4898862016-10-16 10:20:41 -0400177 if (FLAGS_nonstd) {
178 float gammas[3] = { 1.8f, 2.0f, 2.5f, };
179 SkMatrix44 matrix = SkMatrix44(SkMatrix44::kUninitialized_Constructor);
180 matrix.set3x3(0.30f, 0.31f, 0.28f, 0.32f, 0.33f, 0.29f, 0.27f, 0.30f, 0.30f);
Brian Osman526972e2016-10-24 09:24:02 -0400181 fDstSpace = SkColorSpace::MakeRGB(gammas, matrix);
Matt Sarettf4898862016-10-16 10:20:41 -0400182 }
183
msarett8bbcd5a2016-09-14 07:06:08 -0700184 fDstInfo = fDstInfo.makeColorSpace(fDstSpace);
msarett2ecc35f2016-09-08 11:55:16 -0700185
msarett50ce1f22016-07-29 06:23:33 -0700186 fDst.reset(fDstInfo.getSafeSize(fDstInfo.minRowBytes()));
187
188 if (FLAGS_xform_only) {
189 fSrc.reset(fSrcInfo.getSafeSize(fSrcInfo.minRowBytes()));
190 codec->getPixels(fSrcInfo, fSrc.get(), fSrcInfo.minRowBytes());
191 }
192#if defined(SK_TEST_QCMS)
193 else if (FLAGS_qcms) {
194 // Set-up a row buffer to decode into before transforming to dst.
195 fSrc.reset(fSrcInfo.minRowBytes());
196 }
197#endif
msarett2cee9022016-06-03 08:25:21 -0700198}
199
200void ColorCodecBench::onDraw(int n, SkCanvas*) {
msarett575b2a32016-07-19 13:00:35 -0700201#if defined(SK_TEST_QCMS)
msarett200877e2016-08-15 08:10:44 -0700202 if (FLAGS_qcms && FLAGS_half) {
msarett6bdbf442016-07-19 09:07:55 -0700203 SkDebugf("Error: Contradicting flags.\n");
204 return;
205 }
msarett200877e2016-08-15 08:10:44 -0700206#endif
msarett6bdbf442016-07-19 09:07:55 -0700207
msarett2cee9022016-06-03 08:25:21 -0700208 for (int i = 0; i < n; i++) {
msarett469f1c52016-06-06 08:20:37 -0700209#if defined(SK_TEST_QCMS)
msarett2cee9022016-06-03 08:25:21 -0700210 if (FLAGS_qcms) {
211 if (FLAGS_xform_only) {
212 this->xformOnlyQCMS();
213 } else {
214 this->decodeAndXformQCMS();
215 }
216 } else
217#endif
218 {
219 if (FLAGS_xform_only) {
220 this->xformOnly();
221 } else {
222 this->decodeAndXform();
223 }
224 }
225 }
226}