Index: src/core/SkBitmap.cpp |
diff --git a/src/core/SkBitmap.cpp b/src/core/SkBitmap.cpp |
index 122996a31229cdb28d9120903a4b080a0684f0e0..5f8d04e09e56252184b992a23899e7d2a07c9c77 100644 |
--- a/src/core/SkBitmap.cpp |
+++ b/src/core/SkBitmap.cpp |
@@ -268,7 +268,8 @@ void SkBitmap::getBounds(SkIRect* bounds) const { |
/////////////////////////////////////////////////////////////////////////////// |
-void SkBitmap::setConfig(Config c, int width, int height, size_t rowBytes) { |
+void SkBitmap::setConfig(Config c, int width, int height, size_t rowBytes, |
+ bool premul) { |
this->freePixels(); |
if ((width | height) < 0) { |
@@ -286,6 +287,14 @@ void SkBitmap::setConfig(Config c, int width, int height, size_t rowBytes) { |
fWidth = width; |
fHeight = height; |
fRowBytes = SkToU32(rowBytes); |
+ if (premul) { |
+ fFlags &= ~kColorsAreNotPremultiplied_Flag; |
+ } else { |
+ if (c != kARGB_8888_Config) { |
+ goto err; |
+ } |
+ fFlags |= kColorsAreNotPremultiplied_Flag; |
+ } |
fBytesPerPixel = (uint8_t)ComputeBytesPerPixel(c); |
@@ -651,8 +660,18 @@ SkColor SkBitmap::getColor(int x, int y) const { |
return SkUnPreMultiply::PMColorToColor(c); |
} |
case SkBitmap::kARGB_8888_Config: { |
- uint32_t* addr = this->getAddr32(x, y); |
- return SkUnPreMultiply::PMColorToColor(addr[0]); |
+ if (this->premultiplied()) { |
+ uint32_t* addr = this->getAddr32(x, y); |
+ return SkUnPreMultiply::PMColorToColor(addr[0]); |
+ } else { |
+ SkUnPMColor* addr = this->getAddr32(x, y); |
+ SkUnPMColor c = *addr; |
+ U8CPU a = SkGetPackedA32(c); |
+ U8CPU r = SkGetPackedR32(c); |
+ U8CPU g = SkGetPackedG32(c); |
+ U8CPU b = SkGetPackedB32(c); |
+ return SkColorSetARGB(a, r, g, b); |
+ } |
} |
case kRLE_Index8_Config: { |
uint8_t dst; |
@@ -768,6 +787,8 @@ void SkBitmap::eraseARGB(U8CPU a, U8CPU r, U8CPU g, U8CPU b) const { |
// make rgb premultiplied |
if (255 != a) { |
+ // Unpremultiplied exits above at readyToDraw. |
+ SkASSERT(this->premultiplied()); |
r = SkAlphaMul(r, a); |
g = SkAlphaMul(g, a); |
b = SkAlphaMul(b, a); |
@@ -928,7 +949,8 @@ bool SkBitmap::extractSubset(SkBitmap* result, const SkIRect& subset) const { |
SkPixelRef* pixelRef = fPixelRef->deepCopy(this->config(), &subset); |
if (pixelRef != NULL) { |
SkBitmap dst; |
- dst.setConfig(this->config(), subset.width(), subset.height()); |
+ dst.setConfig(this->config(), subset.width(), subset.height(), 0, |
+ this->premultiplied()); |
dst.setIsVolatile(this->isVolatile()); |
dst.setIsOpaque(this->isOpaque()); |
dst.setPixelRef(pixelRef)->unref(); |
@@ -947,6 +969,8 @@ bool SkBitmap::extractSubset(SkBitmap* result, const SkIRect& subset) const { |
} |
SkBitmap bm; |
+ // kRLE_Index8_Config is not compatible with unpremultiplied. |
+ SkASSERT(this->premultiplied()); |
bm.setConfig(kIndex8_Config, r.width(), r.height()); |
bm.allocPixels(this->getColorTable()); |
if (NULL == bm.getPixels()) { |
@@ -977,7 +1001,7 @@ bool SkBitmap::extractSubset(SkBitmap* result, const SkIRect& subset) const { |
} |
SkBitmap dst; |
- dst.setConfig(this->config(), r.width(), r.height(), this->rowBytes()); |
+ dst.setConfig(this->config(), r.width(), r.height(), this->rowBytes(), this->premultiplied()); |
dst.setIsVolatile(this->isVolatile()); |
dst.setIsOpaque(this->isOpaque()); |
@@ -1024,6 +1048,12 @@ bool SkBitmap::canCopyTo(Config dstConfig) const { |
return false; |
} |
+ if (!this->premultiplied()) { |
+ // Cannot copy unpremultiplied, which returns false for readyToDraw. |
+ SkASSERT(kARGB_8888_Config == this->config()); |
+ return false; |
+ } |
+ |
return true; |
} |
@@ -1031,6 +1061,9 @@ bool SkBitmap::copyTo(SkBitmap* dst, Config dstConfig, Allocator* alloc) const { |
if (!this->canCopyTo(dstConfig)) { |
return false; |
} |
+ // Cannot copy an unpremultiplied bitmap. |
+ // canCopyTo should have returned false. |
+ SkASSERT(this->premultiplied()); |
// if we have a texture, first get those pixels |
SkBitmap tmpSrc; |
@@ -1564,6 +1597,12 @@ void SkBitmap::flatten(SkFlattenableWriteBuffer& buffer) const { |
buffer.writeInt(fHeight); |
buffer.writeInt(fRowBytes); |
buffer.writeInt(fConfig); |
+#ifdef BUMP_PICTURE_VERSION |
+ // FIXME: Storing whether the colors are premultiplied |
+ // should be written to the stream, but breaks picture |
+ // serialization without changing PICTURE_VERSION |
+ buffer.writeBool(this->premultiplied()); |
+#endif |
buffer.writeBool(this->isOpaque()); |
if (fPixelRef) { |
@@ -1587,8 +1626,13 @@ void SkBitmap::unflatten(SkFlattenableReadBuffer& buffer) { |
int height = buffer.readInt(); |
int rowBytes = buffer.readInt(); |
int config = buffer.readInt(); |
+#ifdef BUMP_PICTURE_VERSION |
+ bool premul = buffer.readBool(); |
+#else |
+ bool premul = true; |
+#endif |
- this->setConfig((Config)config, width, height, rowBytes); |
+ this->setConfig((Config)config, width, height, rowBytes, premul); |
this->setIsOpaque(buffer.readBool()); |
int reftype = buffer.readInt(); |
@@ -1625,7 +1669,12 @@ SkBitmap::RLEPixels::~RLEPixels() { |
void SkBitmap::validate() const { |
SkASSERT(fConfig < kConfigCount); |
SkASSERT(fRowBytes >= (unsigned)ComputeRowBytes((Config)fConfig, fWidth)); |
- SkASSERT(fFlags <= (kImageIsOpaque_Flag | kImageIsVolatile_Flag | kImageIsImmutable_Flag)); |
+ int8_t allFlags = kImageIsOpaque_Flag | kImageIsVolatile_Flag | kImageIsImmutable_Flag |
+ | kColorsAreNotPremultiplied_Flag; |
+#ifdef SK_BUILD_FOR_ANDROID |
+ allFlags |= kHasHardwareMipMap_Flag; |
+#endif |
+ SkASSERT(fFlags <= allFlags); |
SkASSERT(fPixelLockCount >= 0); |
SkASSERT(NULL == fColorTable || (unsigned)fColorTable->getRefCnt() < 10000); |
SkASSERT((uint8_t)ComputeBytesPerPixel((Config)fConfig) == fBytesPerPixel); |