TextDropShadowCache.cpp revision 059e12ccd20f5c249724a8362d6bac325334ea76
11e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy/*
21e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy * Copyright (C) 2010 The Android Open Source Project
31e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy *
41e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy * Licensed under the Apache License, Version 2.0 (the "License");
51e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy * you may not use this file except in compliance with the License.
61e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy * You may obtain a copy of the License at
71e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy *
81e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy *      http://www.apache.org/licenses/LICENSE-2.0
91e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy *
101e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy * Unless required by applicable law or agreed to in writing, software
111e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy * distributed under the License is distributed on an "AS IS" BASIS,
121e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
131e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy * See the License for the specific language governing permissions and
141e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy * limitations under the License.
151e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy */
161e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
171e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy#define LOG_TAG "OpenGLRenderer"
181e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
19059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy#include <utils/JenkinsHash.h>
20059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
21c9855a53edfac818dc68714557185977556f849dRomain Guy#include "Debug.h"
221e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy#include "TextDropShadowCache.h"
23b45c0c9774bd19a9dbe77d149abae4e124b08bf6Romain Guy#include "Properties.h"
241e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
251e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guynamespace android {
261e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guynamespace uirenderer {
271e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
281e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy///////////////////////////////////////////////////////////////////////////////
29059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy// Cache support
30059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy///////////////////////////////////////////////////////////////////////////////
31059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
32059e12ccd20f5c249724a8362d6bac325334ea76Romain Guyhash_t ShadowText::hash() const {
33059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    uint32_t charCount = len * sizeof(char16_t);
34059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    uint32_t hash = JenkinsHashMix(0, len);
35059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    hash = JenkinsHashMix(hash, android::hash_type(radius));
36059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    hash = JenkinsHashMix(hash, android::hash_type(textSize));
37059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    hash = JenkinsHashMix(hash, android::hash_type(typeface));
38059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    hash = JenkinsHashMix(hash, flags);
39059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    hash = JenkinsHashMix(hash, android::hash_type(italicStyle));
40059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    hash = JenkinsHashMix(hash, android::hash_type(scaleX));
41059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    hash = JenkinsHashMixShorts(hash, text, charCount);
42059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    for (uint32_t i = 0; i < charCount * 2; i++) {
43059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy        hash = JenkinsHashMix(hash, android::hash_type(positions[i]));
44059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    }
45059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    return JenkinsHashWhiten(hash);
46059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy}
47059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
48059e12ccd20f5c249724a8362d6bac325334ea76Romain Guyint ShadowText::compare(const ShadowText& lhs, const ShadowText& rhs) {
49059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    int deltaInt = int(lhs.len) - int(rhs.len);
50059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (deltaInt != 0) return deltaInt;
51059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
52059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    deltaInt = lhs.flags - rhs.flags;
53059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (deltaInt != 0) return deltaInt;
54059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
55059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (lhs.radius < rhs.radius) return -1;
56059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (lhs.radius > rhs.radius) return +1;
57059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
58059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (lhs.typeface < rhs.typeface) return -1;
59059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (lhs.typeface > rhs.typeface) return +1;
60059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
61059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (lhs.textSize < rhs.textSize) return -1;
62059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (lhs.textSize > rhs.textSize) return +1;
63059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
64059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (lhs.italicStyle < rhs.italicStyle) return -1;
65059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (lhs.italicStyle > rhs.italicStyle) return +1;
66059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
67059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (lhs.scaleX < rhs.scaleX) return -1;
68059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (lhs.scaleX > rhs.scaleX) return +1;
69059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
70059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (lhs.text != rhs.text) {
71059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy        if (!lhs.text) return -1;
72059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy        if (!rhs.text) return +1;
73059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
74059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy        deltaInt = memcmp(lhs.text, rhs.text, lhs.len);
75059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy        if (deltaInt != 0) return deltaInt;
76059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    }
77059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
78059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    if (lhs.positions != rhs.positions) {
79059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy        if (!lhs.positions) return -1;
80059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy        if (!rhs.positions) return +1;
81059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
82059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy        return memcmp(lhs.positions, rhs.positions, lhs.len << 2);
83059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    }
84059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
85059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy    return 0;
86059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy}
87059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy
88059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy///////////////////////////////////////////////////////////////////////////////
891e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy// Constructors/destructor
901e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy///////////////////////////////////////////////////////////////////////////////
911e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
92fb8b763f762ae21923c58d64caa729b012f40e05Romain GuyTextDropShadowCache::TextDropShadowCache():
93059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy        mCache(LruCache<ShadowText, ShadowTexture*>::kUnlimitedCapacity),
94fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy        mSize(0), mMaxSize(MB(DEFAULT_DROP_SHADOW_CACHE_SIZE)) {
95fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy    char property[PROPERTY_VALUE_MAX];
96fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy    if (property_get(PROPERTY_DROP_SHADOW_CACHE_SIZE, property, NULL) > 0) {
97c9855a53edfac818dc68714557185977556f849dRomain Guy        INIT_LOGD("  Setting drop shadow cache size to %sMB", property);
98fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy        setMaxSize(MB(atof(property)));
99fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy    } else {
100c9855a53edfac818dc68714557185977556f849dRomain Guy        INIT_LOGD("  Using default drop shadow cache size of %.2fMB",
101c9855a53edfac818dc68714557185977556f849dRomain Guy                DEFAULT_DROP_SHADOW_CACHE_SIZE);
102fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy    }
103fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy
10425dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy    init();
105fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy}
106fb8b763f762ae21923c58d64caa729b012f40e05Romain Guy
1071e45aae5de003657e5d18f74d34998f5de5db5b7Romain GuyTextDropShadowCache::TextDropShadowCache(uint32_t maxByteSize):
108059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy        mCache(LruCache<ShadowText, ShadowTexture*>::kUnlimitedCapacity),
1091e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        mSize(0), mMaxSize(maxByteSize) {
11025dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy    init();
1111e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy}
1121e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1131e45aae5de003657e5d18f74d34998f5de5db5b7Romain GuyTextDropShadowCache::~TextDropShadowCache() {
1141e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    mCache.clear();
1151e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy}
1161e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
11725dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guyvoid TextDropShadowCache::init() {
11825dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy    mCache.setOnEntryRemovedListener(this);
11925dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy    mDebugEnabled = readDebugLevel() & kDebugMoreCaches;
12025dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy}
12125dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy
1221e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy///////////////////////////////////////////////////////////////////////////////
1231e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy// Size management
1241e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy///////////////////////////////////////////////////////////////////////////////
1251e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1261e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guyuint32_t TextDropShadowCache::getSize() {
1271e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    return mSize;
1281e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy}
1291e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1301e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guyuint32_t TextDropShadowCache::getMaxSize() {
1311e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    return mMaxSize;
1321e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy}
1331e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1341e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guyvoid TextDropShadowCache::setMaxSize(uint32_t maxSize) {
1351e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    mMaxSize = maxSize;
1361e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    while (mSize > mMaxSize) {
1371e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        mCache.removeOldest();
1381e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    }
1391e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy}
1401e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1411e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy///////////////////////////////////////////////////////////////////////////////
1421e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy// Callbacks
1431e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy///////////////////////////////////////////////////////////////////////////////
1441e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1451e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guyvoid TextDropShadowCache::operator()(ShadowText& text, ShadowTexture*& texture) {
1461e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    if (texture) {
14725dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy        mSize -= texture->bitmapSize;
14825dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy
14925dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy        if (mDebugEnabled) {
1505baa3a62a97544669fba6d65a11c07f252e654ddSteve Block            ALOGD("Shadow texture deleted, size = %d", texture->bitmapSize);
15125dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy        }
152f70a7e34953438dd70573dbb6fd3dd34a1bce831Romain Guy
1531e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        glDeleteTextures(1, &texture->id);
1541e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        delete texture;
1551e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    }
1561e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy}
1571e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1581e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy///////////////////////////////////////////////////////////////////////////////
1591e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy// Caching
1601e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy///////////////////////////////////////////////////////////////////////////////
1611e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1621e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guyvoid TextDropShadowCache::clear() {
1631e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    mCache.clear();
1641e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy}
1651e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1661e45aae5de003657e5d18f74d34998f5de5db5b7Romain GuyShadowTexture* TextDropShadowCache::get(SkPaint* paint, const char* text, uint32_t len,
167059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy        int numGlyphs, float radius, const float* positions) {
168416a847633680d94efb926837efdc18726d54918Raph Levien    ShadowText entry(paint, radius, len, text, positions);
1691e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    ShadowTexture* texture = mCache.get(entry);
1701e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1711e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    if (!texture) {
1728b4072d3fb9bb49d774d97689a065204beca1752Raph Levien        SkPaint paintCopy(*paint);
1738b4072d3fb9bb49d774d97689a065204beca1752Raph Levien        paintCopy.setTextAlign(SkPaint::kLeft_Align);
1748b4072d3fb9bb49d774d97689a065204beca1752Raph Levien        FontRenderer::DropShadow shadow = mRenderer->renderDropShadow(&paintCopy, text, 0,
175416a847633680d94efb926837efdc18726d54918Raph Levien                len, numGlyphs, radius, positions);
1761e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1771e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        texture = new ShadowTexture;
1781e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        texture->left = shadow.penX;
1791e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        texture->top = shadow.penY;
1801e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        texture->width = shadow.width;
1811e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        texture->height = shadow.height;
1821e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        texture->generation = 0;
1831e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        texture->blend = true;
1841e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1851e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        const uint32_t size = shadow.width * shadow.height;
18625dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy        texture->bitmapSize = size;
18725dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy
1881e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        // Don't even try to cache a bitmap that's bigger than the cache
1891e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        if (size < mMaxSize) {
1901e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy            while (mSize + size > mMaxSize) {
1911e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy                mCache.removeOldest();
1921e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy            }
1931e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        }
1941e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1951e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        glGenTextures(1, &texture->id);
1961e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
1971e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        glBindTexture(GL_TEXTURE_2D, texture->id);
1981e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        // Textures are Alpha8
1991e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
2001e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
2011e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        glTexImage2D(GL_TEXTURE_2D, 0, GL_ALPHA, texture->width, texture->height, 0,
2021e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy                GL_ALPHA, GL_UNSIGNED_BYTE, shadow.image);
2031e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
20439d252a6632d057d5077f7eaf1b8ed7a142f3397Romain Guy        texture->setFilter(GL_LINEAR);
20539d252a6632d057d5077f7eaf1b8ed7a142f3397Romain Guy        texture->setWrap(GL_CLAMP_TO_EDGE);
2061e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
2071e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        if (size < mMaxSize) {
20825dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy            if (mDebugEnabled) {
2095baa3a62a97544669fba6d65a11c07f252e654ddSteve Block                ALOGD("Shadow texture created, size = %d", texture->bitmapSize);
21025dc3a7dbac2f90f5144035e9c8ed99c09cc3132Romain Guy            }
211321dce646dc3c2ecfbd72a693d8d9294a6119736Romain Guy
212321dce646dc3c2ecfbd72a693d8d9294a6119736Romain Guy            entry.copyTextLocally();
213321dce646dc3c2ecfbd72a693d8d9294a6119736Romain Guy
2141e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy            mSize += size;
2151e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy            mCache.put(entry, texture);
2161e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        } else {
2171e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy            texture->cleanup = true;
2181e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        }
2191e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
2201e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        // Cleanup shadow
2211e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy        delete[] shadow.image;
2221e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    }
2231e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
2241e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy    return texture;
2251e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy}
2261e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy
2271e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy}; // namespace uirenderer
2281e45aae5de003657e5d18f74d34998f5de5db5b7Romain Guy}; // namespace android
229