/* * Copyright (C) 2013 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "RenderBufferCache.h" #include "Debug.h" #include "DeviceInfo.h" #include "Properties.h" #include <utils/Log.h> #include <cstdlib> namespace android { namespace uirenderer { /////////////////////////////////////////////////////////////////////////////// // Defines /////////////////////////////////////////////////////////////////////////////// // Debug #if DEBUG_RENDER_BUFFERS #define RENDER_BUFFER_LOGD(...) ALOGD(__VA_ARGS__) #else #define RENDER_BUFFER_LOGD(...) #endif static uint32_t calculateRboCacheSize() { // TODO: Do we need to use extensions().has4BitStencil() here? // The tuning guide recommends it, but all real devices are configured // with a larger cache than necessary by 4x, so keep the 2x for now regardless return DeviceInfo::multiplyByResolution(2); } /////////////////////////////////////////////////////////////////////////////// // Constructors/destructor /////////////////////////////////////////////////////////////////////////////// RenderBufferCache::RenderBufferCache() : mSize(0), mMaxSize(calculateRboCacheSize()) {} RenderBufferCache::~RenderBufferCache() { clear(); } /////////////////////////////////////////////////////////////////////////////// // Size management /////////////////////////////////////////////////////////////////////////////// uint32_t RenderBufferCache::getSize() { return mSize; } uint32_t RenderBufferCache::getMaxSize() { return mMaxSize; } /////////////////////////////////////////////////////////////////////////////// // Caching /////////////////////////////////////////////////////////////////////////////// int RenderBufferCache::RenderBufferEntry::compare(const RenderBufferCache::RenderBufferEntry& lhs, const RenderBufferCache::RenderBufferEntry& rhs) { int deltaInt = int(lhs.mWidth) - int(rhs.mWidth); if (deltaInt != 0) return deltaInt; deltaInt = int(lhs.mHeight) - int(rhs.mHeight); if (deltaInt != 0) return deltaInt; return int(lhs.mFormat) - int(rhs.mFormat); } void RenderBufferCache::deleteBuffer(RenderBuffer* buffer) { if (buffer) { RENDER_BUFFER_LOGD("Deleted %s render buffer (%dx%d)", RenderBuffer::formatName(buffer->getFormat()), buffer->getWidth(), buffer->getHeight()); mSize -= buffer->getSize(); delete buffer; } } void RenderBufferCache::clear() { for (auto entry : mCache) { deleteBuffer(entry.mBuffer); } mCache.clear(); } RenderBuffer* RenderBufferCache::get(GLenum format, const uint32_t width, const uint32_t height) { RenderBuffer* buffer = nullptr; RenderBufferEntry entry(format, width, height); auto iter = mCache.find(entry); if (iter != mCache.end()) { entry = *iter; mCache.erase(iter); buffer = entry.mBuffer; mSize -= buffer->getSize(); RENDER_BUFFER_LOGD("Found %s render buffer (%dx%d)", RenderBuffer::formatName(format), width, height); } else { buffer = new RenderBuffer(format, width, height); RENDER_BUFFER_LOGD("Created new %s render buffer (%dx%d)", RenderBuffer::formatName(format), width, height); } buffer->bind(); buffer->allocate(); return buffer; } bool RenderBufferCache::put(RenderBuffer* buffer) { if (!buffer) return false; const uint32_t size = buffer->getSize(); if (size < mMaxSize) { while (mSize + size > mMaxSize) { RenderBuffer* victim = mCache.begin()->mBuffer; deleteBuffer(victim); mCache.erase(mCache.begin()); } RenderBufferEntry entry(buffer); mCache.insert(entry); mSize += size; RENDER_BUFFER_LOGD("Added %s render buffer (%dx%d)", RenderBuffer::formatName(buffer->getFormat()), buffer->getWidth(), buffer->getHeight()); return true; } else { RENDER_BUFFER_LOGD("Deleted %s render buffer (%dx%d) Size=%d, MaxSize=%d", RenderBuffer::formatName(buffer->getFormat()), buffer->getWidth(), buffer->getHeight(), size, mMaxSize); delete buffer; } return false; } }; // namespace uirenderer }; // namespace android