blob: f0105b278276450e1add413eb20869d8290caabc [file] [log] [blame]
/*
* Copyright (C) 2017 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 <compositionengine/impl/HwcBufferCache.h>
#include <gui/BufferQueue.h>
#include <ui/GraphicBuffer.h>
namespace android::compositionengine::impl {
HwcBufferCache::HwcBufferCache() {
for (uint32_t i = kMaxLayerBufferCount; i-- > 0;) {
mFreeSlots.push(i);
}
}
HwcSlotAndBuffer HwcBufferCache::getHwcSlotAndBuffer(const sp<GraphicBuffer>& buffer) {
if (auto i = mCacheByBufferId.find(buffer->getId()); i != mCacheByBufferId.end()) {
Cache& cache = i->second;
// mark this cache slot as more recently used so it won't get evicted anytime soon
cache.lruCounter = mLeastRecentlyUsedCounter++;
return {cache.slot, nullptr};
}
return {cache(buffer), buffer};
}
HwcSlotAndBuffer HwcBufferCache::getOverrideHwcSlotAndBuffer(const sp<GraphicBuffer>& buffer) {
if (buffer == mLastOverrideBuffer) {
return {kOverrideBufferSlot, nullptr};
}
mLastOverrideBuffer = buffer;
return {kOverrideBufferSlot, buffer};
}
uint32_t HwcBufferCache::uncache(uint64_t bufferId) {
if (auto i = mCacheByBufferId.find(bufferId); i != mCacheByBufferId.end()) {
uint32_t slot = i->second.slot;
mCacheByBufferId.erase(i);
mFreeSlots.push(slot);
return slot;
}
if (mLastOverrideBuffer && bufferId == mLastOverrideBuffer->getId()) {
mLastOverrideBuffer = nullptr;
return kOverrideBufferSlot;
}
return UINT32_MAX;
}
uint32_t HwcBufferCache::cache(const sp<GraphicBuffer>& buffer) {
Cache cache;
cache.slot = getLeastRecentlyUsedSlot();
cache.lruCounter = mLeastRecentlyUsedCounter++;
cache.buffer = buffer;
mCacheByBufferId.emplace(buffer->getId(), cache);
return cache.slot;
}
uint32_t HwcBufferCache::getLeastRecentlyUsedSlot() {
if (mFreeSlots.empty()) {
assert(!mCacheByBufferId.empty());
// evict the least recently used cache entry
auto cacheToErase = mCacheByBufferId.begin();
for (auto i = cacheToErase; i != mCacheByBufferId.end(); ++i) {
if (i->second.lruCounter < cacheToErase->second.lruCounter) {
cacheToErase = i;
}
}
uint32_t slot = cacheToErase->second.slot;
mCacheByBufferId.erase(cacheToErase);
mFreeSlots.push(slot);
}
uint32_t slot = mFreeSlots.top();
mFreeSlots.pop();
return slot;
}
} // namespace android::compositionengine::impl