GradientCache.cpp revision a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65
1c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy/* 2c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * Copyright (C) 2010 The Android Open Source Project 3c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * 4c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * Licensed under the Apache License, Version 2.0 (the "License"); 5c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * you may not use this file except in compliance with the License. 6c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * You may obtain a copy of the License at 7c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * 8c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * http://www.apache.org/licenses/LICENSE-2.0 9c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * 10c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * Unless required by applicable law or agreed to in writing, software 11c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * distributed under the License is distributed on an "AS IS" BASIS, 12c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * See the License for the specific language governing permissions and 14c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy * limitations under the License. 15c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy */ 16c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 17c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy#define LOG_TAG "OpenGLRenderer" 18c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 19c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy#include <GLES2/gl2.h> 20c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 21c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy#include <SkCanvas.h> 22c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy#include <SkGradientShader.h> 23c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 24a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy#include <utils/threads.h> 25a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy 26c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy#include "GradientCache.h" 27fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy#include "Properties.h" 28c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 29c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guynamespace android { 30c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guynamespace uirenderer { 31c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 32c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy/////////////////////////////////////////////////////////////////////////////// 33c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy// Constructors/destructor 34c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy/////////////////////////////////////////////////////////////////////////////// 35c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 36fb8b763f762ae21923c58d64caa729b012f40e05Romain GuyGradientCache::GradientCache(): 37fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy mCache(GenerationCache<SkShader*, Texture*>::kUnlimitedCapacity), 38fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy mSize(0), mMaxSize(MB(DEFAULT_GRADIENT_CACHE_SIZE)) { 39fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy char property[PROPERTY_VALUE_MAX]; 40fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy if (property_get(PROPERTY_GRADIENT_CACHE_SIZE, property, NULL) > 0) { 41fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy LOGD(" Setting gradient cache size to %sMB", property); 42fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy setMaxSize(MB(atof(property))); 43fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy } else { 44fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy LOGD(" Using default gradient cache size of %.2fMB", DEFAULT_GRADIENT_CACHE_SIZE); 45fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy } 46fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy 47fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy mCache.setOnEntryRemovedListener(this); 48fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy} 49fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy 50c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain GuyGradientCache::GradientCache(uint32_t maxByteSize): 51c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy mCache(GenerationCache<SkShader*, Texture*>::kUnlimitedCapacity), 52c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy mSize(0), mMaxSize(maxByteSize) { 53c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy mCache.setOnEntryRemovedListener(this); 54c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy} 55c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 56c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain GuyGradientCache::~GradientCache() { 57a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy Mutex::Autolock _l(mLock); 58c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy mCache.clear(); 59c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy} 60c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 61c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy/////////////////////////////////////////////////////////////////////////////// 62c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy// Size management 63c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy/////////////////////////////////////////////////////////////////////////////// 64c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 65c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guyuint32_t GradientCache::getSize() { 66a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy Mutex::Autolock _l(mLock); 67c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy return mSize; 68c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy} 69c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 70c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guyuint32_t GradientCache::getMaxSize() { 71a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy Mutex::Autolock _l(mLock); 72c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy return mMaxSize; 73c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy} 74c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 75c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guyvoid GradientCache::setMaxSize(uint32_t maxSize) { 76a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy Mutex::Autolock _l(mLock); 77c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy mMaxSize = maxSize; 78c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy while (mSize > mMaxSize) { 79c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy mCache.removeOldest(); 80c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy } 81c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy} 82c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 83c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy/////////////////////////////////////////////////////////////////////////////// 84c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy// Callbacks 85c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy/////////////////////////////////////////////////////////////////////////////// 86c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 87c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guyvoid GradientCache::operator()(SkShader*& shader, Texture*& texture) { 88a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy // Already locked here 89c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy if (shader) { 90c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy const uint32_t size = texture->width * texture->height * 4; 91c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy mSize -= size; 92c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy } 93c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 94c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy if (texture) { 95c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy glDeleteTextures(1, &texture->id); 96c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy delete texture; 97c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy } 98c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy} 99c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 100c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy/////////////////////////////////////////////////////////////////////////////// 101c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy// Caching 102c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy/////////////////////////////////////////////////////////////////////////////// 103c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 104c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain GuyTexture* GradientCache::get(SkShader* shader) { 105a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy Mutex::Autolock _l(mLock); 1069cccc2b9bdd4850a3f9679569aaec3ab98477a5dRomain Guy return mCache.get(shader); 107c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy} 108c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 109c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guyvoid GradientCache::remove(SkShader* shader) { 110a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy Mutex::Autolock _l(mLock); 111c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy mCache.remove(shader); 112c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy} 113c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 114c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guyvoid GradientCache::clear() { 115a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy Mutex::Autolock _l(mLock); 116c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy mCache.clear(); 117c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy} 118c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 119c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain GuyTexture* GradientCache::addLinearGradient(SkShader* shader, float* bounds, uint32_t* colors, 120c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy float* positions, int count, SkShader::TileMode tileMode) { 121c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy SkBitmap bitmap; 122c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy bitmap.setConfig(SkBitmap::kARGB_8888_Config, 1024, 1); 123c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy bitmap.allocPixels(); 124c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy bitmap.eraseColor(0); 125c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 126c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy SkCanvas canvas(bitmap); 127c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 128c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy SkPoint points[2]; 129c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy points[0].set(0.0f, 0.0f); 130c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy points[1].set(bitmap.width(), 0.0f); 131c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 132c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy SkShader* localShader = SkGradientShader::CreateLinear(points, 133c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy reinterpret_cast<const SkColor*>(colors), positions, count, tileMode); 134c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 135c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy SkPaint p; 136c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy p.setStyle(SkPaint::kStrokeAndFill_Style); 137c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy p.setShader(localShader)->unref(); 138c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 139c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy canvas.drawRectCoords(0.0f, 0.0f, bitmap.width(), 1.0f, p); 140c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 141a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy mLock.lock(); 142c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy // Asume the cache is always big enough 143c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy const uint32_t size = bitmap.rowBytes() * bitmap.height(); 144c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy while (mSize + size > mMaxSize) { 145c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy mCache.removeOldest(); 146c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy } 147a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy mLock.unlock(); 148c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 149c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy Texture* texture = new Texture; 150c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy generateTexture(&bitmap, texture); 151c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 152a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy mLock.lock(); 153c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy mSize += size; 154c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy mCache.put(shader, texture); 155a2341a9f6addcd79723965ec5b1a1c5ae0f8bd65Romain Guy mLock.unlock(); 156c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 157c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy return texture; 158c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy} 159c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 160c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guyvoid GradientCache::generateTexture(SkBitmap* bitmap, Texture* texture) { 161c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy SkAutoLockPixels autoLock(*bitmap); 162c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy if (!bitmap->readyToDraw()) { 163c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy LOGE("Cannot generate texture from shader"); 164c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy return; 165c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy } 166c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 167c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy texture->generation = bitmap->getGenerationID(); 168c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy texture->width = bitmap->width(); 169c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy texture->height = bitmap->height(); 170c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 171c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy glGenTextures(1, &texture->id); 172c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 173c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy glBindTexture(GL_TEXTURE_2D, texture->id); 174c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy glPixelStorei(GL_UNPACK_ALIGNMENT, bitmap->bytesPerPixel()); 175c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 176c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy texture->blend = !bitmap->isOpaque(); 177c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, bitmap->rowBytesAsPixels(), texture->height, 0, 178c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy GL_RGBA, GL_UNSIGNED_BYTE, bitmap->getPixels()); 179c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 180c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); 181c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); 182c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 183c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); 184c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); 185c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy} 186c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy 187c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy}; // namespace uirenderer 188c0ac193b9415680f0a69e20a3f5f22d16f8053beRomain Guy}; // namespace android 189