/* * Copyright (c) 2010, Google 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: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 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. * * Neither the name of Google Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "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 THE COPYRIGHT * OWNER 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. */ #include "config.h" #if ENABLE(ACCELERATED_2D_CANVAS) #include "Texture.h" #include "Extensions3D.h" #include "FloatRect.h" #include "GraphicsContext3D.h" #include "IntRect.h" #include #include using namespace std; namespace WebCore { Texture::Texture(GraphicsContext3D* context, PassOwnPtr > tileTextureIds, Format format, int width, int height, int maxTextureSize) : m_context(context) , m_format(format) , m_tiles(maxTextureSize, width, height, true) , m_tileTextureIds(tileTextureIds) { } Texture::~Texture() { for (unsigned int i = 0; i < m_tileTextureIds->size(); i++) m_context->deleteTexture(m_tileTextureIds->at(i)); } static void convertFormat(GraphicsContext3D* context, Texture::Format format, unsigned int* glFormat, unsigned int* glType, bool* swizzle) { *swizzle = false; switch (format) { case Texture::RGBA8: *glFormat = GraphicsContext3D::RGBA; *glType = GraphicsContext3D::UNSIGNED_BYTE; break; case Texture::BGRA8: if (context->getExtensions()->supports("GL_EXT_texture_format_BGRA8888")) { *glFormat = Extensions3D::BGRA_EXT; *glType = GraphicsContext3D::UNSIGNED_BYTE; } else { *glFormat = GraphicsContext3D::RGBA; *glType = GraphicsContext3D::UNSIGNED_BYTE; *swizzle = true; } break; default: ASSERT_NOT_REACHED(); break; } } PassRefPtr Texture::create(GraphicsContext3D* context, Format format, int width, int height) { int maxTextureSize = 0; context->getIntegerv(GraphicsContext3D::MAX_TEXTURE_SIZE, &maxTextureSize); TilingData tiling(maxTextureSize, width, height, true); int numTiles = tiling.numTiles(); OwnPtr > textureIds(new Vector(numTiles)); textureIds->fill(0, numTiles); for (int i = 0; i < numTiles; i++) { int textureId = context->createTexture(); if (!textureId) { for (int i = 0; i < numTiles; i++) context->deleteTexture(textureIds->at(i)); return 0; } textureIds->at(i) = textureId; IntRect tileBoundsWithBorder = tiling.tileBoundsWithBorder(i); unsigned int glFormat = 0; unsigned int glType = 0; bool swizzle; convertFormat(context, format, &glFormat, &glType, &swizzle); context->bindTexture(GraphicsContext3D::TEXTURE_2D, textureId); context->texImage2DResourceSafe(GraphicsContext3D::TEXTURE_2D, 0, glFormat, tileBoundsWithBorder.width(), tileBoundsWithBorder.height(), 0, glFormat, glType); } return adoptRef(new Texture(context, textureIds.leakPtr(), format, width, height, maxTextureSize)); } template static uint32_t* copySubRect(uint32_t* src, int srcX, int srcY, uint32_t* dst, int width, int height, int srcStride) { uint32_t* srcOffset = src + srcX + srcY * srcStride; if (!swizzle && width == srcStride) return srcOffset; if (swizzle) { uint32_t* dstPixel = dst; for (int y = 0; y < height; ++y) { for (int x = 0; x < width ; ++x) { uint32_t pixel = srcOffset[x + y * srcStride]; *dstPixel = (pixel & 0xFF00FF00) | ((pixel & 0x00FF0000) >> 16) | ((pixel & 0x000000FF) << 16); dstPixel++; } } } else { for (int y = 0; y < height; ++y) { memcpy(dst + y * width, srcOffset + y * srcStride, 4 * width); } } return dst; } void Texture::load(void* pixels) { updateSubRect(pixels, IntRect(0, 0, m_tiles.totalSizeX(), m_tiles.totalSizeY())); } void Texture::updateSubRect(void* pixels, const IntRect& updateRect) { IntRect updateRectSanitized(updateRect); updateRectSanitized.intersect(IntRect(0, 0, m_tiles.totalSizeX(), m_tiles.totalSizeY())); uint32_t* pixels32 = static_cast(pixels); unsigned int glFormat = 0; unsigned int glType = 0; bool swizzle; convertFormat(m_context, m_format, &glFormat, &glType, &swizzle); if (swizzle) { ASSERT(glFormat == GraphicsContext3D::RGBA && glType == GraphicsContext3D::UNSIGNED_BYTE); // FIXME: This could use PBO's to save doing an extra copy here. } int tempBuffSize = // Temporary buffer size is the smaller of the max texture size or the updateRectSanitized min(m_tiles.maxTextureSize(), m_tiles.borderTexels() + updateRectSanitized.width()) * min(m_tiles.maxTextureSize(), m_tiles.borderTexels() + updateRectSanitized.height()); OwnArrayPtr tempBuff = adoptArrayPtr(new uint32_t[tempBuffSize]); for (int tile = 0; tile < m_tiles.numTiles(); tile++) { // Intersect with tile IntRect tileBoundsWithBorder = m_tiles.tileBoundsWithBorder(tile); IntRect updateRectIntersected = updateRectSanitized; updateRectIntersected.intersect(tileBoundsWithBorder); IntRect dstRect = updateRectIntersected; dstRect.move(-tileBoundsWithBorder.x(), -tileBoundsWithBorder.y()); if (updateRectIntersected.isEmpty()) continue; // Copy sub rectangle out of larger pixel data uint32_t* uploadBuff = 0; if (swizzle) { uploadBuff = copySubRect( pixels32, updateRectIntersected.x(), updateRectIntersected.y(), tempBuff.get(), updateRectIntersected.width(), updateRectIntersected.height(), m_tiles.totalSizeX()); } else { uploadBuff = copySubRect( pixels32, updateRectIntersected.x(), updateRectIntersected.y(), tempBuff.get(), updateRectIntersected.width(), updateRectIntersected.height(), m_tiles.totalSizeX()); } m_context->bindTexture(GraphicsContext3D::TEXTURE_2D, m_tileTextureIds->at(tile)); m_context->texSubImage2D(GraphicsContext3D::TEXTURE_2D, 0 /* level */, dstRect.x(), dstRect.y(), updateRectIntersected.width(), updateRectIntersected.height(), glFormat, glType, uploadBuff); } } void Texture::bindTile(int tile) { m_context->bindTexture(GraphicsContext3D::TEXTURE_2D, m_tileTextureIds->at(tile)); m_context->texParameteri(GraphicsContext3D::TEXTURE_2D, GraphicsContext3D::TEXTURE_MIN_FILTER, GraphicsContext3D::LINEAR); m_context->texParameteri(GraphicsContext3D::TEXTURE_2D, GraphicsContext3D::TEXTURE_MAG_FILTER, GraphicsContext3D::LINEAR); m_context->texParameteri(GraphicsContext3D::TEXTURE_2D, GraphicsContext3D::TEXTURE_WRAP_S, GraphicsContext3D::CLAMP_TO_EDGE); m_context->texParameteri(GraphicsContext3D::TEXTURE_2D, GraphicsContext3D::TEXTURE_WRAP_T, GraphicsContext3D::CLAMP_TO_EDGE); } } #endif