1*38e8c45fSAndroid Build Coastguard Worker /*
2*38e8c45fSAndroid Build Coastguard Worker * Copyright 2021 The Android Open Source Project
3*38e8c45fSAndroid Build Coastguard Worker *
4*38e8c45fSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*38e8c45fSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*38e8c45fSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*38e8c45fSAndroid Build Coastguard Worker *
8*38e8c45fSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*38e8c45fSAndroid Build Coastguard Worker *
10*38e8c45fSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*38e8c45fSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*38e8c45fSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*38e8c45fSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*38e8c45fSAndroid Build Coastguard Worker * limitations under the License.
15*38e8c45fSAndroid Build Coastguard Worker */
16*38e8c45fSAndroid Build Coastguard Worker
17*38e8c45fSAndroid Build Coastguard Worker #undef LOG_TAG
18*38e8c45fSAndroid Build Coastguard Worker #define LOG_TAG "FpsReporter"
19*38e8c45fSAndroid Build Coastguard Worker #define ATRACE_TAG ATRACE_TAG_GRAPHICS
20*38e8c45fSAndroid Build Coastguard Worker
21*38e8c45fSAndroid Build Coastguard Worker #include <algorithm>
22*38e8c45fSAndroid Build Coastguard Worker
23*38e8c45fSAndroid Build Coastguard Worker #include "FpsReporter.h"
24*38e8c45fSAndroid Build Coastguard Worker #include "Layer.h"
25*38e8c45fSAndroid Build Coastguard Worker #include "SurfaceFlinger.h"
26*38e8c45fSAndroid Build Coastguard Worker
27*38e8c45fSAndroid Build Coastguard Worker namespace android {
28*38e8c45fSAndroid Build Coastguard Worker
FpsReporter(frametimeline::FrameTimeline & frameTimeline,std::unique_ptr<Clock> clock)29*38e8c45fSAndroid Build Coastguard Worker FpsReporter::FpsReporter(frametimeline::FrameTimeline& frameTimeline, std::unique_ptr<Clock> clock)
30*38e8c45fSAndroid Build Coastguard Worker : mFrameTimeline(frameTimeline), mClock(std::move(clock)) {
31*38e8c45fSAndroid Build Coastguard Worker LOG_ALWAYS_FATAL_IF(mClock == nullptr, "Passed in null clock when constructing FpsReporter!");
32*38e8c45fSAndroid Build Coastguard Worker }
33*38e8c45fSAndroid Build Coastguard Worker
dispatchLayerFps(const frontend::LayerHierarchy & layerHierarchy)34*38e8c45fSAndroid Build Coastguard Worker void FpsReporter::dispatchLayerFps(const frontend::LayerHierarchy& layerHierarchy) {
35*38e8c45fSAndroid Build Coastguard Worker const auto now = mClock->now();
36*38e8c45fSAndroid Build Coastguard Worker if (now - mLastDispatch < kMinDispatchDuration) {
37*38e8c45fSAndroid Build Coastguard Worker return;
38*38e8c45fSAndroid Build Coastguard Worker }
39*38e8c45fSAndroid Build Coastguard Worker
40*38e8c45fSAndroid Build Coastguard Worker std::vector<TrackedListener> localListeners;
41*38e8c45fSAndroid Build Coastguard Worker {
42*38e8c45fSAndroid Build Coastguard Worker std::scoped_lock lock(mMutex);
43*38e8c45fSAndroid Build Coastguard Worker if (mListeners.empty()) {
44*38e8c45fSAndroid Build Coastguard Worker return;
45*38e8c45fSAndroid Build Coastguard Worker }
46*38e8c45fSAndroid Build Coastguard Worker
47*38e8c45fSAndroid Build Coastguard Worker std::transform(mListeners.begin(), mListeners.end(), std::back_inserter(localListeners),
48*38e8c45fSAndroid Build Coastguard Worker [](const std::pair<wp<IBinder>, TrackedListener>& entry) {
49*38e8c45fSAndroid Build Coastguard Worker return entry.second;
50*38e8c45fSAndroid Build Coastguard Worker });
51*38e8c45fSAndroid Build Coastguard Worker }
52*38e8c45fSAndroid Build Coastguard Worker
53*38e8c45fSAndroid Build Coastguard Worker std::unordered_set<int32_t> seenTasks;
54*38e8c45fSAndroid Build Coastguard Worker std::vector<std::pair<TrackedListener, const frontend::LayerHierarchy*>>
55*38e8c45fSAndroid Build Coastguard Worker listenersAndLayersToReport;
56*38e8c45fSAndroid Build Coastguard Worker
57*38e8c45fSAndroid Build Coastguard Worker layerHierarchy.traverse([&](const frontend::LayerHierarchy& hierarchy,
58*38e8c45fSAndroid Build Coastguard Worker const frontend::LayerHierarchy::TraversalPath& traversalPath) {
59*38e8c45fSAndroid Build Coastguard Worker if (traversalPath.variant == frontend::LayerHierarchy::Variant::Detached) {
60*38e8c45fSAndroid Build Coastguard Worker return false;
61*38e8c45fSAndroid Build Coastguard Worker }
62*38e8c45fSAndroid Build Coastguard Worker const auto& metadata = hierarchy.getLayer()->metadata;
63*38e8c45fSAndroid Build Coastguard Worker if (metadata.has(gui::METADATA_TASK_ID)) {
64*38e8c45fSAndroid Build Coastguard Worker int32_t taskId = metadata.getInt32(gui::METADATA_TASK_ID, 0);
65*38e8c45fSAndroid Build Coastguard Worker if (seenTasks.count(taskId) == 0) {
66*38e8c45fSAndroid Build Coastguard Worker // localListeners is expected to be tiny
67*38e8c45fSAndroid Build Coastguard Worker for (TrackedListener& listener : localListeners) {
68*38e8c45fSAndroid Build Coastguard Worker if (listener.taskId == taskId) {
69*38e8c45fSAndroid Build Coastguard Worker seenTasks.insert(taskId);
70*38e8c45fSAndroid Build Coastguard Worker listenersAndLayersToReport.push_back({listener, &hierarchy});
71*38e8c45fSAndroid Build Coastguard Worker break;
72*38e8c45fSAndroid Build Coastguard Worker }
73*38e8c45fSAndroid Build Coastguard Worker }
74*38e8c45fSAndroid Build Coastguard Worker }
75*38e8c45fSAndroid Build Coastguard Worker }
76*38e8c45fSAndroid Build Coastguard Worker return true;
77*38e8c45fSAndroid Build Coastguard Worker });
78*38e8c45fSAndroid Build Coastguard Worker
79*38e8c45fSAndroid Build Coastguard Worker for (const auto& [listener, hierarchy] : listenersAndLayersToReport) {
80*38e8c45fSAndroid Build Coastguard Worker std::unordered_set<int32_t> layerIds;
81*38e8c45fSAndroid Build Coastguard Worker
82*38e8c45fSAndroid Build Coastguard Worker hierarchy->traverse([&](const frontend::LayerHierarchy& hierarchy,
83*38e8c45fSAndroid Build Coastguard Worker const frontend::LayerHierarchy::TraversalPath& traversalPath) {
84*38e8c45fSAndroid Build Coastguard Worker if (traversalPath.variant == frontend::LayerHierarchy::Variant::Detached) {
85*38e8c45fSAndroid Build Coastguard Worker return false;
86*38e8c45fSAndroid Build Coastguard Worker }
87*38e8c45fSAndroid Build Coastguard Worker layerIds.insert(static_cast<int32_t>(hierarchy.getLayer()->id));
88*38e8c45fSAndroid Build Coastguard Worker return true;
89*38e8c45fSAndroid Build Coastguard Worker });
90*38e8c45fSAndroid Build Coastguard Worker
91*38e8c45fSAndroid Build Coastguard Worker listener.listener->onFpsReported(mFrameTimeline.computeFps(layerIds));
92*38e8c45fSAndroid Build Coastguard Worker }
93*38e8c45fSAndroid Build Coastguard Worker
94*38e8c45fSAndroid Build Coastguard Worker mLastDispatch = now;
95*38e8c45fSAndroid Build Coastguard Worker }
96*38e8c45fSAndroid Build Coastguard Worker
binderDied(const wp<IBinder> & who)97*38e8c45fSAndroid Build Coastguard Worker void FpsReporter::binderDied(const wp<IBinder>& who) {
98*38e8c45fSAndroid Build Coastguard Worker std::scoped_lock lock(mMutex);
99*38e8c45fSAndroid Build Coastguard Worker mListeners.erase(who);
100*38e8c45fSAndroid Build Coastguard Worker }
101*38e8c45fSAndroid Build Coastguard Worker
addListener(const sp<gui::IFpsListener> & listener,int32_t taskId)102*38e8c45fSAndroid Build Coastguard Worker void FpsReporter::addListener(const sp<gui::IFpsListener>& listener, int32_t taskId) {
103*38e8c45fSAndroid Build Coastguard Worker sp<IBinder> asBinder = IInterface::asBinder(listener);
104*38e8c45fSAndroid Build Coastguard Worker asBinder->linkToDeath(sp<DeathRecipient>::fromExisting(this));
105*38e8c45fSAndroid Build Coastguard Worker std::lock_guard lock(mMutex);
106*38e8c45fSAndroid Build Coastguard Worker mListeners.emplace(wp<IBinder>(asBinder), TrackedListener{listener, taskId});
107*38e8c45fSAndroid Build Coastguard Worker }
108*38e8c45fSAndroid Build Coastguard Worker
removeListener(const sp<gui::IFpsListener> & listener)109*38e8c45fSAndroid Build Coastguard Worker void FpsReporter::removeListener(const sp<gui::IFpsListener>& listener) {
110*38e8c45fSAndroid Build Coastguard Worker std::lock_guard lock(mMutex);
111*38e8c45fSAndroid Build Coastguard Worker mListeners.erase(wp<IBinder>(IInterface::asBinder(listener)));
112*38e8c45fSAndroid Build Coastguard Worker }
113*38e8c45fSAndroid Build Coastguard Worker
114*38e8c45fSAndroid Build Coastguard Worker } // namespace android
115