blob: 4ddb6df83ecb8b9b7957f4664e184f0081e81011 [file] [log] [blame]
/*
* Copyright (C) 2016 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#pragma once
#include "Color.h"
#include "IntRect.h"
#include "IntSize.h"
#include "NativeImage.h"
#include "SharedBuffer.h"
namespace WebCore {
#if USE(CAIRO)
// Due to the pixman 16.16 floating point representation, cairo is not able to handle
// images whose size is bigger than 32768.
static const int cairoMaxImageSize = 32768;
#endif
class ImageBackingStore {
WTF_MAKE_FAST_ALLOCATED;
public:
static std::unique_ptr<ImageBackingStore> create(const IntSize& size, bool premultiplyAlpha = true)
{
return std::unique_ptr<ImageBackingStore>(new ImageBackingStore(size, premultiplyAlpha));
}
static std::unique_ptr<ImageBackingStore> create(const ImageBackingStore& other)
{
return std::unique_ptr<ImageBackingStore>(new ImageBackingStore(other));
}
PlatformImagePtr image() const;
bool setSize(const IntSize& size)
{
if (size.isEmpty())
return false;
Vector<uint8_t> buffer;
size_t bufferSize = size.area() * sizeof(uint32_t);
if (!buffer.tryReserveCapacity(bufferSize))
return false;
buffer.grow(bufferSize);
m_pixels = SharedBuffer::DataSegment::create(WTFMove(buffer));
m_pixelsPtr = reinterpret_cast<uint32_t*>(const_cast<uint8_t*>(m_pixels->data()));
m_size = size;
m_frameRect = IntRect(IntPoint(), m_size);
clear();
return true;
}
void setFrameRect(const IntRect& frameRect)
{
ASSERT(!m_size.isEmpty());
ASSERT(inBounds(frameRect));
m_frameRect = frameRect;
}
const IntSize& size() const { return m_size; }
const IntRect& frameRect() const { return m_frameRect; }
void clear()
{
memset(m_pixelsPtr, 0, m_size.area() * sizeof(uint32_t));
}
void clearRect(const IntRect& rect)
{
if (rect.isEmpty() || !inBounds(rect))
return;
size_t rowBytes = rect.width() * sizeof(uint32_t);
uint32_t* start = pixelAt(rect.x(), rect.y());
for (int i = 0; i < rect.height(); ++i) {
memset(start, 0, rowBytes);
start += m_size.width();
}
}
void fillRect(const IntRect& rect, uint8_t r, uint8_t g, uint8_t b, uint8_t a)
{
if (rect.isEmpty() || !inBounds(rect))
return;
uint32_t* start = pixelAt(rect.x(), rect.y());
uint32_t pixelValue = this->pixelValue(r, g, b, a);
for (int i = 0; i < rect.height(); ++i) {
for (int j = 0; j < rect.width(); ++j)
start[j] = pixelValue;
start += m_size.width();
}
}
void repeatFirstRow(const IntRect& rect)
{
if (rect.isEmpty() || !inBounds(rect))
return;
size_t rowBytes = rect.width() * sizeof(uint32_t);
uint32_t* src = pixelAt(rect.x(), rect.y());
uint32_t* dest = src + m_size.width();
for (int i = 1; i < rect.height(); ++i) {
memcpy(dest, src, rowBytes);
dest += m_size.width();
}
}
uint32_t* pixelAt(int x, int y) const
{
ASSERT(inBounds(IntPoint(x, y)));
return m_pixelsPtr + y * m_size.width() + x;
}
void setPixel(uint32_t* dest, uint8_t r, uint8_t g, uint8_t b, uint8_t a)
{
ASSERT(dest);
*dest = pixelValue(r, g, b, a);
}
void setPixel(int x, int y, uint8_t r, uint8_t g, uint8_t b, uint8_t a)
{
setPixel(pixelAt(x, y), r, g, b, a);
}
void blendPixel(uint32_t* dest, uint8_t r, uint8_t g, uint8_t b, uint8_t a)
{
if (!a)
return;
auto pixel = asSRGBA(PackedColor::ARGB { *dest });
if (a >= 255 || !pixel.alpha) {
setPixel(dest, r, g, b, a);
return;
}
if (!m_premultiplyAlpha)
pixel = premultipliedFlooring(pixel);
uint8_t d = 255 - a;
r = fastDivideBy255(r * a + pixel.red * d);
g = fastDivideBy255(g * a + pixel.green * d);
b = fastDivideBy255(b * a + pixel.blue * d);
a += fastDivideBy255(d * pixel.alpha);
auto result = SRGBA<uint8_t> { r, g, b, a };
if (!m_premultiplyAlpha)
result = unpremultiplied(result);
*dest = PackedColor::ARGB { result }.value;
}
static bool isOverSize(const IntSize& size)
{
#if USE(CAIRO)
// FIXME: this is a workaround to avoid the cairo image size limit, but we should implement support for
// bigger images. See https://bugs.webkit.org/show_bug.cgi?id=177227.
//
// If the image is bigger than the cairo limit it can't be displayed, so we don't even try to decode it.
if (size.width() > cairoMaxImageSize || size.height() > cairoMaxImageSize)
return true;
#endif
static unsigned long long MaxPixels = ((1 << 29) - 1);
unsigned long long pixels = static_cast<unsigned long long>(size.width()) * static_cast<unsigned long long>(size.height());
return pixels > MaxPixels;
}
private:
ImageBackingStore(const IntSize& size, bool premultiplyAlpha = true)
: m_premultiplyAlpha(premultiplyAlpha)
{
ASSERT(!size.isEmpty() && !isOverSize(size));
setSize(size);
}
ImageBackingStore(const ImageBackingStore& other)
: m_size(other.m_size)
, m_premultiplyAlpha(other.m_premultiplyAlpha)
{
ASSERT(!m_size.isEmpty() && !isOverSize(m_size));
Vector<uint8_t> buffer { other.m_pixels->data(), other.m_pixels->size() };
m_pixels = SharedBuffer::DataSegment::create(WTFMove(buffer));
m_pixelsPtr = reinterpret_cast<uint32_t*>(const_cast<uint8_t*>(m_pixels->data()));
}
bool inBounds(const IntPoint& point) const
{
return IntRect(IntPoint(), m_size).contains(point);
}
bool inBounds(const IntRect& rect) const
{
return IntRect(IntPoint(), m_size).contains(rect);
}
uint32_t pixelValue(uint8_t r, uint8_t g, uint8_t b, uint8_t a) const
{
if (m_premultiplyAlpha && !a)
return 0;
auto result = SRGBA<uint8_t> { r, g, b, a };
if (m_premultiplyAlpha && a < 255)
result = premultipliedFlooring(result);
return PackedColor::ARGB { result }.value;
}
RefPtr<SharedBuffer::DataSegment> m_pixels;
uint32_t* m_pixelsPtr { nullptr };
IntSize m_size;
IntRect m_frameRect; // This will always just be the entire buffer except for GIF and PNG frames
bool m_premultiplyAlpha { true };
};
}