xref: /aosp_15_r20/frameworks/native/services/surfaceflinger/FpsReporter.cpp (revision 38e8c45f13ce32b0dcecb25141ffecaf386fa17f)
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