/* * Copyright 2018 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. */ #define ATRACE_TAG ATRACE_TAG_GRAPHICS #include "LayerHistory.h" #include <cinttypes> #include <cstdint> #include <limits> #include <numeric> #include <string> #include <unordered_map> #include <cutils/properties.h> #include <utils/Log.h> #include <utils/Timers.h> #include <utils/Trace.h> #include "SchedulerUtils.h" namespace android { namespace scheduler { std::atomic<int64_t> LayerHistory::sNextId = 0; LayerHistory::LayerHistory() { char value[PROPERTY_VALUE_MAX]; property_get("debug.sf.layer_history_trace", value, "0"); mTraceEnabled = bool(atoi(value)); } LayerHistory::~LayerHistory() = default; std::unique_ptr<LayerHistory::LayerHandle> LayerHistory::createLayer(const std::string name, float maxRefreshRate) { const int64_t id = sNextId++; std::lock_guard lock(mLock); mInactiveLayerInfos.emplace(id, std::make_shared<LayerInfo>(name, maxRefreshRate)); return std::make_unique<LayerHistory::LayerHandle>(*this, id); } void LayerHistory::destroyLayer(const int64_t id) { std::lock_guard lock(mLock); auto it = mActiveLayerInfos.find(id); if (it != mActiveLayerInfos.end()) { mActiveLayerInfos.erase(it); } it = mInactiveLayerInfos.find(id); if (it != mInactiveLayerInfos.end()) { mInactiveLayerInfos.erase(it); } } void LayerHistory::insert(const std::unique_ptr<LayerHandle>& layerHandle, nsecs_t presentTime, bool isHdr) { std::shared_ptr<LayerInfo> layerInfo; { std::lock_guard lock(mLock); auto layerInfoIterator = mInactiveLayerInfos.find(layerHandle->mId); if (layerInfoIterator != mInactiveLayerInfos.end()) { layerInfo = layerInfoIterator->second; mInactiveLayerInfos.erase(layerInfoIterator); mActiveLayerInfos.insert({layerHandle->mId, layerInfo}); } else { layerInfoIterator = mActiveLayerInfos.find(layerHandle->mId); if (layerInfoIterator != mActiveLayerInfos.end()) { layerInfo = layerInfoIterator->second; } else { ALOGW("Inserting information about layer that is not registered: %" PRId64, layerHandle->mId); return; } } } layerInfo->setLastPresentTime(presentTime); layerInfo->setHDRContent(isHdr); } void LayerHistory::setVisibility(const std::unique_ptr<LayerHandle>& layerHandle, bool visible) { std::shared_ptr<LayerInfo> layerInfo; { std::lock_guard lock(mLock); auto layerInfoIterator = mInactiveLayerInfos.find(layerHandle->mId); if (layerInfoIterator != mInactiveLayerInfos.end()) { layerInfo = layerInfoIterator->second; if (visible) { mInactiveLayerInfos.erase(layerInfoIterator); mActiveLayerInfos.insert({layerHandle->mId, layerInfo}); } } else { layerInfoIterator = mActiveLayerInfos.find(layerHandle->mId); if (layerInfoIterator != mActiveLayerInfos.end()) { layerInfo = layerInfoIterator->second; } else { ALOGW("Inserting information about layer that is not registered: %" PRId64, layerHandle->mId); return; } } } layerInfo->setVisibility(visible); } std::pair<float, bool> LayerHistory::getDesiredRefreshRateAndHDR() { bool isHDR = false; float newRefreshRate = 0.f; std::lock_guard lock(mLock); removeIrrelevantLayers(); // Iterate through all layers that have been recently updated, and find the max refresh rate. for (const auto& [layerId, layerInfo] : mActiveLayerInfos) { const float layerRefreshRate = layerInfo->getDesiredRefreshRate(); if (mTraceEnabled) { // Store the refresh rate in traces for easy debugging. std::string layerName = "LFPS " + layerInfo->getName(); ATRACE_INT(layerName.c_str(), std::round(layerRefreshRate)); ALOGD("%s: %f", layerName.c_str(), std::round(layerRefreshRate)); } if (layerInfo->isRecentlyActive() && layerRefreshRate > newRefreshRate) { newRefreshRate = layerRefreshRate; } isHDR |= layerInfo->getHDRContent(); } if (mTraceEnabled) { ALOGD("LayerHistory DesiredRefreshRate: %.2f", newRefreshRate); } return {newRefreshRate, isHDR}; } void LayerHistory::removeIrrelevantLayers() { const int64_t obsoleteEpsilon = systemTime() - scheduler::OBSOLETE_TIME_EPSILON_NS.count(); // Iterator pointing to first element in map auto it = mActiveLayerInfos.begin(); while (it != mActiveLayerInfos.end()) { // If last updated was before the obsolete time, remove it. // Keep HDR layer around as long as they are visible. if (!it->second->isVisible() || (!it->second->getHDRContent() && it->second->getLastUpdatedTime() < obsoleteEpsilon)) { // erase() function returns the iterator of the next // to last deleted element. if (mTraceEnabled) { ALOGD("Layer %s obsolete", it->second->getName().c_str()); // Make sure to update systrace to indicate that the layer was erased. std::string layerName = "LFPS " + it->second->getName(); ATRACE_INT(layerName.c_str(), 0); } auto id = it->first; auto layerInfo = it->second; layerInfo->clearHistory(); mInactiveLayerInfos.insert({id, layerInfo}); it = mActiveLayerInfos.erase(it); } else { ++it; } } } } // namespace scheduler } // namespace android