aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/core
diff options
context:
space:
mode:
Diffstat (limited to 'src/core')
-rw-r--r--src/core/SkMaskGamma.cpp94
-rw-r--r--src/core/SkMaskGamma.h54
-rw-r--r--src/core/SkPaint.cpp69
3 files changed, 76 insertions, 141 deletions
diff --git a/src/core/SkMaskGamma.cpp b/src/core/SkMaskGamma.cpp
index 2070dda17e..9066fb7d9f 100644
--- a/src/core/SkMaskGamma.cpp
+++ b/src/core/SkMaskGamma.cpp
@@ -11,45 +11,61 @@
#include "SkFloatingPoint.h"
#include "SkMaskGamma.h"
-SkScalar SkSRGBLuminance::toLuma(SkScalar luminance) const {
- //The magic numbers are derived from the sRGB specification.
- //See http://www.color.org/chardata/rgb/srgb.xalter .
- if (luminance <= SkFloatToScalar(0.04045f)) {
- return luminance / SkFloatToScalar(12.92f);
+class SkLinearColorSpaceLuminance : public SkColorSpaceLuminance {
+ virtual SkScalar toLuma(SkScalar SkDEBUGCODE(gamma), SkScalar luminance) const SK_OVERRIDE {
+ SkASSERT(SK_Scalar1 == gamma);
+ return luminance;
}
- return SkScalarPow((luminance + SkFloatToScalar(0.055f)) / SkFloatToScalar(1.055f),
- SkFloatToScalar(2.4f));
-}
-
-SkScalar SkSRGBLuminance::fromLuma(SkScalar luma) const {
- //The magic numbers are derived from the sRGB specification.
- //See http://www.color.org/chardata/rgb/srgb.xalter .
- if (luma <= SkFloatToScalar(0.0031308f)) {
- return luma * SkFloatToScalar(12.92f);
+ virtual SkScalar fromLuma(SkScalar SkDEBUGCODE(gamma), SkScalar luma) const SK_OVERRIDE {
+ SkASSERT(SK_Scalar1 == gamma);
+ return luma;
}
- return SkFloatToScalar(1.055f) * SkScalarPow(luma, SkScalarInvert(SkFloatToScalar(2.4f)))
- - SkFloatToScalar(0.055f);
-}
-
-SkGammaLuminance::SkGammaLuminance(SkScalar gamma)
- : fGamma(gamma)
- , fGammaInverse(SkScalarInvert(gamma)) {
-}
+};
-SkScalar SkGammaLuminance::toLuma(SkScalar luminance) const {
- return SkScalarPow(luminance, fGamma);
-}
-
-SkScalar SkGammaLuminance::fromLuma(SkScalar luma) const {
- return SkScalarPow(luma, fGammaInverse);
-}
+class SkGammaColorSpaceLuminance : public SkColorSpaceLuminance {
+ virtual SkScalar toLuma(SkScalar gamma, SkScalar luminance) const SK_OVERRIDE {
+ return SkScalarPow(luminance, gamma);
+ }
+ virtual SkScalar fromLuma(SkScalar gamma, SkScalar luma) const SK_OVERRIDE {
+ return SkScalarPow(luma, SkScalarInvert(gamma));
+ }
+};
+
+class SkSRGBColorSpaceLuminance : public SkColorSpaceLuminance {
+ virtual SkScalar toLuma(SkScalar SkDEBUGCODE(gamma), SkScalar luminance) const SK_OVERRIDE {
+ SkASSERT(0 == gamma);
+ //The magic numbers are derived from the sRGB specification.
+ //See http://www.color.org/chardata/rgb/srgb.xalter .
+ if (luminance <= SkFloatToScalar(0.04045f)) {
+ return luminance / SkFloatToScalar(12.92f);
+ }
+ return SkScalarPow((luminance + SkFloatToScalar(0.055f)) / SkFloatToScalar(1.055f),
+ SkFloatToScalar(2.4f));
+ }
+ virtual SkScalar fromLuma(SkScalar SkDEBUGCODE(gamma), SkScalar luma) const SK_OVERRIDE {
+ SkASSERT(0 == gamma);
+ //The magic numbers are derived from the sRGB specification.
+ //See http://www.color.org/chardata/rgb/srgb.xalter .
+ if (luma <= SkFloatToScalar(0.0031308f)) {
+ return luma * SkFloatToScalar(12.92f);
+ }
+ return SkFloatToScalar(1.055f) * SkScalarPow(luma, SkScalarInvert(SkFloatToScalar(2.4f)))
+ - SkFloatToScalar(0.055f);
+ }
+};
-SkScalar SkLinearLuminance::toLuma(SkScalar luminance) const {
- return luminance;
-}
+/*static*/ const SkColorSpaceLuminance& SkColorSpaceLuminance::Fetch(SkScalar gamma) {
+ static SkLinearColorSpaceLuminance gSkLinearColorSpaceLuminance;
+ static SkGammaColorSpaceLuminance gSkGammaColorSpaceLuminance;
+ static SkSRGBColorSpaceLuminance gSkSRGBColorSpaceLuminance;
-SkScalar SkLinearLuminance::fromLuma(SkScalar luma) const {
- return luma;
+ if (0 == gamma) {
+ return gSkSRGBColorSpaceLuminance;
+ } else if (SK_Scalar1 == gamma) {
+ return gSkLinearColorSpaceLuminance;
+ } else {
+ return gSkGammaColorSpaceLuminance;
+ }
}
static float apply_contrast(float srca, float contrast) {
@@ -57,16 +73,16 @@ static float apply_contrast(float srca, float contrast) {
}
void SkTMaskGamma_build_correcting_lut(uint8_t table[256], U8CPU srcI, SkScalar contrast,
- const SkColorSpaceLuminance& srcConvert,
- const SkColorSpaceLuminance& dstConvert) {
+ const SkColorSpaceLuminance& srcConvert, SkScalar srcGamma,
+ const SkColorSpaceLuminance& dstConvert, SkScalar dstGamma) {
const float src = (float)srcI / 255.0f;
- const float linSrc = srcConvert.toLuma(src);
+ const float linSrc = srcConvert.toLuma(srcGamma, src);
//Guess at the dst. The perceptual inverse provides smaller visual
//discontinuities when slight changes to desaturated colors cause a channel
//to map to a different correcting lut with neighboring srcI.
//See https://code.google.com/p/chromium/issues/detail?id=141425#c59 .
const float dst = 1.0f - src;
- const float linDst = dstConvert.toLuma(dst);
+ const float linDst = dstConvert.toLuma(dstGamma, dst);
//Contrast value tapers off to 0 as the src luminance becomes white
const float adjustedContrast = SkScalarToFloat(contrast) * linDst;
@@ -96,7 +112,7 @@ void SkTMaskGamma_build_correcting_lut(uint8_t table[256], U8CPU srcI, SkScalar
//Calculate the output we want.
float linOut = (linSrc * srca + dsta * linDst);
SkASSERT(linOut <= 1.0f);
- float out = dstConvert.fromLuma(linOut);
+ float out = dstConvert.fromLuma(dstGamma, linOut);
//Undo what the blit blend will do.
float result = (out - dst) / (src - dst);
diff --git a/src/core/SkMaskGamma.h b/src/core/SkMaskGamma.h
index 961c1b1924..a092b209ee 100644
--- a/src/core/SkMaskGamma.h
+++ b/src/core/SkMaskGamma.h
@@ -22,46 +22,28 @@
*/
class SkColorSpaceLuminance : SkNoncopyable {
public:
- virtual ~SkColorSpaceLuminance() {};
+ virtual ~SkColorSpaceLuminance() { }
/** Converts a color component luminance in the color space to a linear luma. */
- virtual SkScalar toLuma(SkScalar luminance) const = 0;
+ virtual SkScalar toLuma(SkScalar gamma, SkScalar luminance) const = 0;
/** Converts a linear luma to a color component luminance in the color space. */
- virtual SkScalar fromLuma(SkScalar luma) const = 0;
+ virtual SkScalar fromLuma(SkScalar gamma, SkScalar luma) const = 0;
/** Converts a color to a luminance value. */
- U8CPU computeLuminance(SkColor c) const {
- SkScalar r = toLuma(SkIntToScalar(SkColorGetR(c)) / 255);
- SkScalar g = toLuma(SkIntToScalar(SkColorGetG(c)) / 255);
- SkScalar b = toLuma(SkIntToScalar(SkColorGetB(c)) / 255);
+ static U8CPU computeLuminance(SkScalar gamma, SkColor c) {
+ const SkColorSpaceLuminance& luminance = Fetch(gamma);
+ SkScalar r = luminance.toLuma(gamma, SkIntToScalar(SkColorGetR(c)) / 255);
+ SkScalar g = luminance.toLuma(gamma, SkIntToScalar(SkColorGetG(c)) / 255);
+ SkScalar b = luminance.toLuma(gamma, SkIntToScalar(SkColorGetB(c)) / 255);
SkScalar luma = r * SkFloatToScalar(SK_LUM_COEFF_R) +
g * SkFloatToScalar(SK_LUM_COEFF_G) +
b * SkFloatToScalar(SK_LUM_COEFF_B);
SkASSERT(luma <= SK_Scalar1);
- return SkScalarRoundToInt(fromLuma(luma) * 255);
+ return SkScalarRoundToInt(luminance.fromLuma(gamma, luma) * 255);
}
-};
-class SkSRGBLuminance : public SkColorSpaceLuminance {
-public:
- SkScalar toLuma(SkScalar luminance) const SK_OVERRIDE;
- SkScalar fromLuma(SkScalar luma) const SK_OVERRIDE;
-};
-
-class SkGammaLuminance : public SkColorSpaceLuminance {
-public:
- SkGammaLuminance(SkScalar gamma);
- SkScalar toLuma(SkScalar luminance) const SK_OVERRIDE;
- SkScalar fromLuma(SkScalar luma) const SK_OVERRIDE;
-private:
- SkScalar fGamma;
- SkScalar fGammaInverse;
-};
-
-class SkLinearLuminance : public SkColorSpaceLuminance {
-public:
- SkScalar toLuma(SkScalar luminance) const SK_OVERRIDE;
- SkScalar fromLuma(SkScalar luma) const SK_OVERRIDE;
+ /** Retrieves the SkColorSpaceLuminance for the given gamma. */
+ static const SkColorSpaceLuminance& Fetch(SkScalar gamma);
};
///@{
@@ -95,8 +77,8 @@ template<> /*static*/ inline U8CPU sk_t_scale255<8>(U8CPU base) {
template <int R_LUM_BITS, int G_LUM_BITS, int B_LUM_BITS> class SkTMaskPreBlend;
void SkTMaskGamma_build_correcting_lut(uint8_t table[256], U8CPU srcI, SkScalar contrast,
- const SkColorSpaceLuminance& srcConvert,
- const SkColorSpaceLuminance& dstConvert);
+ const SkColorSpaceLuminance& srcConvert, SkScalar srcGamma,
+ const SkColorSpaceLuminance& dstConvert, SkScalar dstGamma);
/**
* A regular mask contains linear alpha values. A gamma correcting mask
@@ -125,12 +107,14 @@ public:
* @param paint The color space in which the paint color was chosen.
* @param device The color space of the target device.
*/
- SkTMaskGamma(SkScalar contrast,
- const SkColorSpaceLuminance& paint,
- const SkColorSpaceLuminance& device) : fIsLinear(false) {
+ SkTMaskGamma(SkScalar contrast, SkScalar paintGamma, SkScalar deviceGamma) : fIsLinear(false) {
+ const SkColorSpaceLuminance& paintConvert = SkColorSpaceLuminance::Fetch(paintGamma);
+ const SkColorSpaceLuminance& deviceConvert = SkColorSpaceLuminance::Fetch(deviceGamma);
for (U8CPU i = 0; i < (1 << kLuminanceBits_Max); ++i) {
U8CPU lum = sk_t_scale255<kLuminanceBits_Max>(i);
- SkTMaskGamma_build_correcting_lut(fGammaTables[i], lum, contrast, paint, device);
+ SkTMaskGamma_build_correcting_lut(fGammaTables[i], lum, contrast,
+ paintConvert, paintGamma,
+ deviceConvert, deviceGamma);
}
}
diff --git a/src/core/SkPaint.cpp b/src/core/SkPaint.cpp
index 7088a94ac1..f611e6d847 100644
--- a/src/core/SkPaint.cpp
+++ b/src/core/SkPaint.cpp
@@ -1634,58 +1634,6 @@ void SkScalerContext::MakeRec(const SkPaint& paint,
*/
SK_DECLARE_STATIC_MUTEX(gMaskGammaCacheMutex);
-static SkColorSpaceLuminance* gLinearLuminance = NULL;
-
-static SkColorSpaceLuminance* gDeviceLuminance = NULL;
-static SkScalar gDeviceGammaExponent = SK_ScalarMin;
-/**
- * The caller must hold the gMaskGammaCacheMutex and continue to hold it until
- * the returned SkColorSpaceLuminance pointer is forgotten.
- */
-static SkColorSpaceLuminance* cachedDeviceLuminance(SkScalar gammaExponent) {
- if (SK_Scalar1 == gammaExponent) {
- if (NULL == gLinearLuminance) {
- gLinearLuminance = SkNEW(SkLinearLuminance);
- }
- return gLinearLuminance;
- }
- if (gDeviceGammaExponent != gammaExponent) {
- SkDELETE(gDeviceLuminance);
- if (0 == gammaExponent) {
- gDeviceLuminance = SkNEW(SkSRGBLuminance);
- } else {
- gDeviceLuminance = SkNEW_ARGS(SkGammaLuminance, (gammaExponent));
- }
- gDeviceGammaExponent = gammaExponent;
- }
- return gDeviceLuminance;
-}
-
-static SkColorSpaceLuminance* gPaintLuminance = NULL;
-static SkScalar gPaintGammaExponent = SK_ScalarMin;
-/**
- * The caller must hold the gMaskGammaCacheMutex and continue to hold it until
- * the returned SkColorSpaceLuminance pointer is forgotten.
- */
-static SkColorSpaceLuminance* cachedPaintLuminance(SkScalar gammaExponent) {
- if (SK_Scalar1 == gammaExponent) {
- if (NULL == gLinearLuminance) {
- gLinearLuminance = SkNEW(SkLinearLuminance);
- }
- return gLinearLuminance;
- }
- if (gPaintGammaExponent != gammaExponent) {
- SkDELETE(gPaintLuminance);
- if (0 == gammaExponent) {
- gPaintLuminance = SkNEW(SkSRGBLuminance);
- } else {
- gPaintLuminance = SkNEW_ARGS(SkGammaLuminance, (gammaExponent));
- }
- gPaintGammaExponent = gammaExponent;
- }
- return gPaintLuminance;
-}
-
static SkMaskGamma* gLinearMaskGamma = NULL;
static SkMaskGamma* gMaskGamma = NULL;
static SkScalar gContrast = SK_ScalarMin;
@@ -1704,9 +1652,7 @@ static SkMaskGamma* cachedMaskGamma(SkScalar contrast, SkScalar paintGamma, SkSc
}
if (gContrast != contrast || gPaintGamma != paintGamma || gDeviceGamma != deviceGamma) {
SkSafeUnref(gMaskGamma);
- SkColorSpaceLuminance* paintLuminance = cachedPaintLuminance(paintGamma);
- SkColorSpaceLuminance* deviceLuminance = cachedDeviceLuminance(deviceGamma);
- gMaskGamma = SkNEW_ARGS(SkMaskGamma, (contrast, *paintLuminance, *deviceLuminance));
+ gMaskGamma = SkNEW_ARGS(SkMaskGamma, (contrast, paintGamma, deviceGamma));
gContrast = contrast;
gPaintGamma = paintGamma;
gDeviceGamma = deviceGamma;
@@ -1724,16 +1670,6 @@ static SkMaskGamma* cachedMaskGamma(SkScalar contrast, SkScalar paintGamma, SkSc
SkDEBUGCODE(gContrast = SK_ScalarMin;)
SkDEBUGCODE(gPaintGamma = SK_ScalarMin;)
SkDEBUGCODE(gDeviceGamma = SK_ScalarMin;)
-
- SkDELETE(gLinearLuminance);
-
- SkDELETE(gPaintLuminance);
- SkDEBUGCODE(gPaintLuminance = NULL;)
- SkDEBUGCODE(gPaintGammaExponent = SK_ScalarMin;)
-
- SkDELETE(gDeviceLuminance);
- SkDEBUGCODE(gDeviceLuminance = NULL;)
- SkDEBUGCODE(gDeviceGammaExponent = SK_ScalarMin;)
}
/**
@@ -1758,8 +1694,7 @@ void SkScalerContext::PostMakeRec(const SkPaint& paint, SkScalerContext::Rec* re
// use per-component information
SkColor color = rec->getLuminanceColor();
- SkAutoMutexAcquire ama(gMaskGammaCacheMutex);
- U8CPU lum = cachedPaintLuminance(rec->getPaintGamma())->computeLuminance(color);
+ U8CPU lum = SkColorSpaceLuminance::computeLuminance(rec->getPaintGamma(), color);
//If we are asked to look like LCD, look like LCD.
if (!(rec->fFlags & SkScalerContext::kGenA8FromLCD_Flag)) {
// HACK: Prevents green from being pre-blended as white.