Store SkColorSpaceXform gamma LUTs in a malloced field

In order of likelihood:
(1) Tables are never used, since gamma is recognized and named.
(2) Only use one table, since all three gammas are the same.
(3) Actually need three tables.

No reason to waste a bunch of space for these tables on
SkColorSpaceXform, when it will likely be unused.

This will be more efficient in lots of cases, but is particularly
useful when the client really only wants a gamut xform.

BUG=skia:
GOLD_TRYBOT_URL= https://gold.skia.org/search?issue=2336913005

Review-Url: https://codereview.chromium.org/2336913005
This commit is contained in:
msarett 2016-09-14 11:02:04 -07:00 committed by Commit bot
parent 896ddb706d
commit 7bbda991af
2 changed files with 58 additions and 22 deletions

View File

@ -256,7 +256,8 @@ static const GammaFns<uint8_t> kFromLinear {
// Build tables to transform src gamma to linear.
template <typename T>
static void build_gamma_tables(const T* outGammaTables[3], T* gammaTableStorage, int gammaTableSize,
const sk_sp<SkColorSpace>& space, const GammaFns<T>& fns) {
const sk_sp<SkColorSpace>& space, const GammaFns<T>& fns,
bool gammasAreMatching) {
switch (as_CSB(space)->gammaNamed()) {
case kSRGB_SkGammaNamed:
outGammaTables[0] = outGammaTables[1] = outGammaTables[2] = fns.fSRGBTable;
@ -271,18 +272,7 @@ static void build_gamma_tables(const T* outGammaTables[3], T* gammaTableStorage,
const SkGammas* gammas = as_CSB(space)->gammas();
SkASSERT(gammas);
for (int i = 0; i < 3; i++) {
if (i > 0) {
// Check if this curve matches the first curve. In this case, we can
// share the same table pointer. This should almost always be true.
// I've never seen a profile where all three gamma curves didn't match.
// But it is possible that they won't.
if (gammas->type(0) == gammas->type(i) && gammas->data(0) == gammas->data(i)) {
outGammaTables[i] = outGammaTables[0];
continue;
}
}
auto build_table = [=](int i) {
if (gammas->isNamed(i)) {
switch (gammas->data(i).fNamed) {
case kSRGB_SkGammaNamed:
@ -319,7 +309,19 @@ static void build_gamma_tables(const T* outGammaTables[3], T* gammaTableStorage,
params.fF);
outGammaTables[i] = &gammaTableStorage[i * gammaTableSize];
}
};
if (gammasAreMatching) {
build_table(0);
outGammaTables[1] = outGammaTables[0];
outGammaTables[2] = outGammaTables[0];
} else {
build_table(0);
build_table(1);
build_table(2);
}
break;
}
}
}
@ -1159,6 +1161,28 @@ static void color_xform_RGBA(void* dst, const uint32_t* src, int len,
///////////////////////////////////////////////////////////////////////////////////////////////////
static inline int num_tables(SkColorSpace* space) {
switch (as_CSB(space)->gammaNamed()) {
case kSRGB_SkGammaNamed:
case k2Dot2Curve_SkGammaNamed:
case kLinear_SkGammaNamed:
return 0;
default: {
const SkGammas* gammas = as_CSB(space)->gammas();
SkASSERT(gammas);
bool gammasAreMatching = (gammas->type(0) == gammas->type(1)) &&
(gammas->data(0) == gammas->data(1)) &&
(gammas->type(0) == gammas->type(2)) &&
(gammas->data(0) == gammas->data(2));
// It's likely that each component will have the same gamma. In this case,
// we only need to build one table.
return gammasAreMatching ? 1 : 3;
}
}
}
template <SrcGamma kSrc, DstGamma kDst, ColorSpaceMatch kCSM>
SkColorSpaceXform_Base<kSrc, kDst, kCSM>
::SkColorSpaceXform_Base(const sk_sp<SkColorSpace>& srcSpace, const SkMatrix44& srcToDst,
@ -1166,11 +1190,24 @@ SkColorSpaceXform_Base<kSrc, kDst, kCSM>
: fColorLUT(sk_ref_sp((SkColorLookUpTable*) as_CSB(srcSpace)->colorLUT()))
{
srcToDst.asColMajorf(fSrcToDst);
build_gamma_tables(fSrcGammaTables, fSrcGammaTableStorage, 256, srcSpace, kToLinear);
build_gamma_tables(fDstGammaTables, fDstGammaTableStorage, kDstGammaTableSize, dstSpace,
kFromLinear);
const int numSrcTables = num_tables(srcSpace.get());
const int numDstTables = num_tables(dstSpace.get());
const size_t srcTableBytes = numSrcTables * 256 * sizeof(float);
const size_t dstTableBytes = numDstTables * kDstGammaTableSize * sizeof(uint8_t);
fStorage.reset(srcTableBytes + dstTableBytes);
float* srcStorage = (float*) fStorage.get();
uint8_t* dstStorage = SkTAddOffset<uint8_t>(fStorage.get(), srcTableBytes);
const bool srcGammasAreMatching = (1 >= numSrcTables);
const bool dstGammasAreMatching = (1 >= numDstTables);
build_gamma_tables(fSrcGammaTables, srcStorage, 256, srcSpace, kToLinear, srcGammasAreMatching);
build_gamma_tables(fDstGammaTables, dstStorage, kDstGammaTableSize, dstSpace, kFromLinear,
dstGammasAreMatching);
}
///////////////////////////////////////////////////////////////////////////////////////////////////
template <SrcFormat kSrc, DstFormat kDst, ColorSpaceMatch kCSM, SwapRB kSwap>
static inline void apply_set_alpha(void* dst, const uint32_t* src, int len, SkAlphaType alphaType,
const float* const srcTables[3], const float matrix[16],
@ -1291,6 +1328,8 @@ const
}
}
///////////////////////////////////////////////////////////////////////////////////////////////////
std::unique_ptr<SkColorSpaceXform> SlowIdentityXform(const sk_sp<SkColorSpace>& space) {
return std::unique_ptr<SkColorSpaceXform>(new SkColorSpaceXform_Base
<kTable_SrcGamma, kTable_DstGamma, kNone_ColorSpaceMatch>

View File

@ -77,16 +77,13 @@ private:
sk_sp<SkColorLookUpTable> fColorLUT;
// May contain pointers into storage or pointers into precomputed tables.
// Contain pointers into storage or pointers into precomputed tables.
const float* fSrcGammaTables[3];
float fSrcGammaTableStorage[3 * 256];
const uint8_t* fDstGammaTables[3];
SkAutoMalloc fStorage;
float fSrcToDst[16];
// May contain pointers into storage or pointers into precomputed tables.
const uint8_t* fDstGammaTables[3];
uint8_t fDstGammaTableStorage[3 * kDstGammaTableSize];
friend class SkColorSpaceXform;
friend std::unique_ptr<SkColorSpaceXform> SlowIdentityXform(const sk_sp<SkColorSpace>& space);
};