1 /*
2 * Copyright (C) 2010 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include <GLES2/gl2.h>
18
19 #include <utils/Mutex.h>
20
21 #include "Caches.h"
22 #include "DeviceInfo.h"
23 #include "Properties.h"
24 #include "Texture.h"
25 #include "TextureCache.h"
26 #include "hwui/Bitmap.h"
27 #include "utils/TraceUtils.h"
28
29 namespace android {
30 namespace uirenderer {
31
32 ///////////////////////////////////////////////////////////////////////////////
33 // Constructors/destructor
34 ///////////////////////////////////////////////////////////////////////////////
35
TextureCache()36 TextureCache::TextureCache()
37 : mCache(LruCache<uint32_t, Texture*>::kUnlimitedCapacity)
38 , mSize(0)
39 , mMaxSize(DeviceInfo::multiplyByResolution(4 * 6)) // 6 screen-sized RGBA_8888 bitmaps
40 , mFlushRate(.4f) {
41 mCache.setOnEntryRemovedListener(this);
42 mMaxTextureSize = DeviceInfo::get()->maxTextureSize();
43 mDebugEnabled = Properties::debugLevel & kDebugCaches;
44 }
45
~TextureCache()46 TextureCache::~TextureCache() {
47 this->clear();
48 }
49
50 ///////////////////////////////////////////////////////////////////////////////
51 // Size management
52 ///////////////////////////////////////////////////////////////////////////////
53
getSize()54 uint32_t TextureCache::getSize() {
55 return mSize;
56 }
57
getMaxSize()58 uint32_t TextureCache::getMaxSize() {
59 return mMaxSize;
60 }
61
62 ///////////////////////////////////////////////////////////////////////////////
63 // Callbacks
64 ///////////////////////////////////////////////////////////////////////////////
65
operator ()(uint32_t &,Texture * & texture)66 void TextureCache::operator()(uint32_t&, Texture*& texture) {
67 // This will be called already locked
68 if (texture) {
69 mSize -= texture->bitmapSize;
70 TEXTURE_LOGD("TextureCache::callback: name, removed size, mSize = %d, %d, %d", texture->id,
71 texture->bitmapSize, mSize);
72 if (mDebugEnabled) {
73 ALOGD("Texture deleted, size = %d", texture->bitmapSize);
74 }
75 texture->deleteTexture();
76 delete texture;
77 }
78 }
79
80 ///////////////////////////////////////////////////////////////////////////////
81 // Caching
82 ///////////////////////////////////////////////////////////////////////////////
83
resetMarkInUse(void * ownerToken)84 void TextureCache::resetMarkInUse(void* ownerToken) {
85 LruCache<uint32_t, Texture*>::Iterator iter(mCache);
86 while (iter.next()) {
87 if (iter.value()->isInUse == ownerToken) {
88 iter.value()->isInUse = nullptr;
89 }
90 }
91 }
92
canMakeTextureFromBitmap(Bitmap * bitmap)93 bool TextureCache::canMakeTextureFromBitmap(Bitmap* bitmap) {
94 if (bitmap->width() > mMaxTextureSize || bitmap->height() > mMaxTextureSize) {
95 ALOGW("Bitmap too large to be uploaded into a texture (%dx%d, max=%dx%d)", bitmap->width(),
96 bitmap->height(), mMaxTextureSize, mMaxTextureSize);
97 return false;
98 }
99 return true;
100 }
101
createTexture(Bitmap * bitmap)102 Texture* TextureCache::createTexture(Bitmap* bitmap) {
103 Texture* texture = new Texture(Caches::getInstance());
104 texture->bitmapSize = bitmap->rowBytes() * bitmap->height();
105 texture->generation = bitmap->getGenerationID();
106 texture->upload(*bitmap);
107 return texture;
108 }
109
110 // Returns a prepared Texture* that either is already in the cache or can fit
111 // in the cache (and is thus added to the cache)
getCachedTexture(Bitmap * bitmap)112 Texture* TextureCache::getCachedTexture(Bitmap* bitmap) {
113 if (bitmap->isHardware()) {
114 auto textureIterator = mHardwareTextures.find(bitmap->getStableID());
115 if (textureIterator == mHardwareTextures.end()) {
116 Texture* texture = createTexture(bitmap);
117 mHardwareTextures.insert(
118 std::make_pair(bitmap->getStableID(), std::unique_ptr<Texture>(texture)));
119 if (mDebugEnabled) {
120 ALOGD("Texture created for hw bitmap size = %d", texture->bitmapSize);
121 }
122 return texture;
123 }
124 return textureIterator->second.get();
125 }
126
127 Texture* texture = mCache.get(bitmap->getStableID());
128
129 if (!texture) {
130 if (!canMakeTextureFromBitmap(bitmap)) {
131 return nullptr;
132 }
133
134 const uint32_t size = bitmap->rowBytes() * bitmap->height();
135 bool canCache = size < mMaxSize;
136 // Don't even try to cache a bitmap that's bigger than the cache
137 while (canCache && mSize + size > mMaxSize) {
138 Texture* oldest = mCache.peekOldestValue();
139 if (oldest && !oldest->isInUse) {
140 mCache.removeOldest();
141 } else {
142 canCache = false;
143 }
144 }
145
146 if (canCache) {
147 texture = createTexture(bitmap);
148 mSize += size;
149 TEXTURE_LOGD("TextureCache::get: create texture(%p): name, size, mSize = %d, %d, %d",
150 bitmap, texture->id, size, mSize);
151 if (mDebugEnabled) {
152 ALOGD("Texture created, size = %d", size);
153 }
154 mCache.put(bitmap->getStableID(), texture);
155 }
156 } else if (!texture->isInUse && bitmap->getGenerationID() != texture->generation) {
157 // Texture was in the cache but is dirty, re-upload
158 // TODO: Re-adjust the cache size if the bitmap's dimensions have changed
159 texture->upload(*bitmap);
160 texture->generation = bitmap->getGenerationID();
161 }
162
163 return texture;
164 }
165
prefetchAndMarkInUse(void * ownerToken,Bitmap * bitmap)166 bool TextureCache::prefetchAndMarkInUse(void* ownerToken, Bitmap* bitmap) {
167 Texture* texture = getCachedTexture(bitmap);
168 if (texture) {
169 texture->isInUse = ownerToken;
170 }
171 return texture;
172 }
173
prefetch(Bitmap * bitmap)174 bool TextureCache::prefetch(Bitmap* bitmap) {
175 return getCachedTexture(bitmap);
176 }
177
get(Bitmap * bitmap)178 Texture* TextureCache::get(Bitmap* bitmap) {
179 Texture* texture = getCachedTexture(bitmap);
180
181 if (!texture) {
182 if (!canMakeTextureFromBitmap(bitmap)) {
183 return nullptr;
184 }
185 texture = createTexture(bitmap);
186 texture->cleanup = true;
187 }
188
189 return texture;
190 }
191
destroyTexture(uint32_t pixelRefStableID)192 bool TextureCache::destroyTexture(uint32_t pixelRefStableID) {
193 auto hardwareIter = mHardwareTextures.find(pixelRefStableID);
194 if (hardwareIter != mHardwareTextures.end()) {
195 hardwareIter->second->deleteTexture();
196 mHardwareTextures.erase(hardwareIter);
197 return true;
198 }
199 return mCache.remove(pixelRefStableID);
200 }
201
clear()202 void TextureCache::clear() {
203 mCache.clear();
204 for (auto& iter : mHardwareTextures) {
205 iter.second->deleteTexture();
206 }
207 mHardwareTextures.clear();
208 TEXTURE_LOGD("TextureCache:clear(), mSize = %d", mSize);
209 }
210
flush()211 void TextureCache::flush() {
212 if (mFlushRate >= 1.0f || mCache.size() == 0) return;
213 if (mFlushRate <= 0.0f) {
214 clear();
215 return;
216 }
217
218 uint32_t targetSize = uint32_t(mSize * mFlushRate);
219 TEXTURE_LOGD("TextureCache::flush: target size: %d", targetSize);
220
221 while (mSize > targetSize) {
222 mCache.removeOldest();
223 }
224 }
225
226 }; // namespace uirenderer
227 }; // namespace android
228