1 /*
2  * Copyright 2021 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #define LOG_TAG "SurfaceFlingerPuller"
18 
19 #include "SurfaceFlingerPuller.h"
20 
21 #include <gui/SurfaceComposerClient.h>
22 #include <log/log.h>
23 #include <statslog.h>
24 #include <timestatsatomsproto/TimeStatsAtomsProtoHeader.h>
25 
26 #include <vector>
27 
28 namespace android {
29 namespace server {
30 namespace stats {
31 
32 using android::util::BytesField;
33 using std::optional;
34 
35 namespace {
getBytes(const google::protobuf::MessageLite & proto,std::string & data)36 optional<BytesField> getBytes(const google::protobuf::MessageLite& proto, std::string& data) {
37     if (!proto.SerializeToString(&data)) {
38         ALOGW("Unable to serialize surface flinger bytes field");
39         return std::nullopt;
40     }
41     return {BytesField(data.data(), data.size())};
42 }
43 } // namespace
44 
pull(int32_t atomTag,AStatsEventList * data)45 AStatsManager_PullAtomCallbackReturn SurfaceFlingerPuller::pull(int32_t atomTag,
46                                                                 AStatsEventList* data) {
47     // Don't need mutexes here, since there is no global state.
48     // SurfaceComposerClient is thread safe, and surfaceflinger is internally thread safe.
49 
50     bool success = false;
51     std::string pullDataProto;
52     status_t err = SurfaceComposerClient::onPullAtom(atomTag, &pullDataProto, &success);
53     if (!success || err != NO_ERROR) {
54         ALOGW("Failed to pull atom %" PRId32
55               " from surfaceflinger. Success is %d, binder status is %s",
56               atomTag, (int)success, binder::Status::exceptionToString(err).c_str());
57         return AStatsManager_PULL_SKIP;
58     }
59 
60     switch (atomTag) {
61         case android::util::SURFACEFLINGER_STATS_GLOBAL_INFO:
62             return parseGlobalInfoPull(pullDataProto, data);
63         case android::util::SURFACEFLINGER_STATS_LAYER_INFO:
64             return parseLayerInfoPull(pullDataProto, data);
65         default:
66             ALOGW("Invalid atom id for surfaceflinger pullers: %" PRId32, atomTag);
67             return AStatsManager_PULL_SKIP;
68     }
69 }
70 
parseGlobalInfoPull(const std::string & protoData,AStatsEventList * data)71 AStatsManager_PullAtomCallbackReturn SurfaceFlingerPuller::parseGlobalInfoPull(
72         const std::string& protoData, AStatsEventList* data) {
73     android::surfaceflinger::SurfaceflingerStatsGlobalInfoWrapper atomList;
74     if (!atomList.ParseFromString(protoData)) {
75         ALOGW("Error parsing surface flinger global stats to proto");
76         return AStatsManager_PULL_SKIP;
77     }
78 
79     for (const auto& atom : atomList.atom()) {
80         // The strings must outlive the BytesFields, which only have a pointer to the data.
81         std::string frameDurationStr, renderEngineTimeStr, deadlineMissesStr, predictionErrorsStr;
82         optional<BytesField> frameDuration = getBytes(atom.frame_duration(), frameDurationStr);
83         optional<BytesField> renderEngineTime =
84                 getBytes(atom.render_engine_timing(), renderEngineTimeStr);
85         optional<BytesField> deadlineMisses =
86                 getBytes(atom.sf_deadline_misses(), deadlineMissesStr);
87         optional<BytesField> predictionErrors =
88                 getBytes(atom.sf_prediction_errors(), predictionErrorsStr);
89 
90         // Fail if any serialization to bytes failed.
91         if (!frameDuration || !renderEngineTime || !deadlineMisses || !predictionErrors) {
92             return AStatsManager_PULL_SKIP;
93         }
94 
95         android::util::addAStatsEvent(data, android::util::SURFACEFLINGER_STATS_GLOBAL_INFO,
96                                       atom.total_frames(), atom.missed_frames(),
97                                       atom.client_composition_frames(), atom.display_on_millis(),
98                                       atom.animation_millis(), atom.event_connection_count(),
99                                       frameDuration.value(), renderEngineTime.value(),
100                                       atom.total_timeline_frames(), atom.total_janky_frames(),
101                                       atom.total_janky_frames_with_long_cpu(),
102                                       atom.total_janky_frames_with_long_gpu(),
103                                       atom.total_janky_frames_sf_unattributed(),
104                                       atom.total_janky_frames_app_unattributed(),
105                                       atom.total_janky_frames_sf_scheduling(),
106                                       atom.total_jank_frames_sf_prediction_error(),
107                                       atom.total_jank_frames_app_buffer_stuffing(),
108                                       atom.display_refresh_rate_bucket(), deadlineMisses.value(),
109                                       predictionErrors.value(), atom.render_rate_bucket());
110     }
111     return AStatsManager_PULL_SUCCESS;
112 }
113 
parseLayerInfoPull(const std::string & protoData,AStatsEventList * data)114 AStatsManager_PullAtomCallbackReturn SurfaceFlingerPuller::parseLayerInfoPull(
115         const std::string& protoData, AStatsEventList* data) {
116     android::surfaceflinger::SurfaceflingerStatsLayerInfoWrapper atomList;
117     if (!atomList.ParseFromString(protoData)) {
118         ALOGW("Error parsing surface flinger layer stats to proto");
119         return AStatsManager_PULL_SKIP;
120     }
121 
122     for (const auto& atom : atomList.atom()) {
123         // The strings must outlive the BytesFields, which only have a pointer to the data.
124         std::string present2PresentStr, post2presentStr, acquire2PresentStr, latch2PresentStr,
125                 desired2PresentStr, post2AcquireStr, frameRateVoteStr, appDeadlineMissesStr,
126                 present2PresentDeltaStr;
127         optional<BytesField> present2Present =
128                 getBytes(atom.present_to_present(), present2PresentStr);
129         optional<BytesField> present2PresentDelta =
130                 getBytes(atom.present_to_present_delta(), present2PresentDeltaStr);
131         optional<BytesField> post2present = getBytes(atom.post_to_present(), post2presentStr);
132         optional<BytesField> acquire2Present =
133                 getBytes(atom.acquire_to_present(), acquire2PresentStr);
134         optional<BytesField> latch2Present = getBytes(atom.latch_to_present(), latch2PresentStr);
135         optional<BytesField> desired2Present =
136                 getBytes(atom.desired_to_present(), desired2PresentStr);
137         optional<BytesField> post2Acquire = getBytes(atom.post_to_acquire(), post2AcquireStr);
138         optional<BytesField> frameRateVote = getBytes(atom.set_frame_rate_vote(), frameRateVoteStr);
139         optional<BytesField> appDeadlineMisses =
140                 getBytes(atom.app_deadline_misses(), appDeadlineMissesStr);
141 
142         // Fail if any serialization to bytes failed.
143         if (!present2Present || !post2present || !acquire2Present || !latch2Present ||
144             !desired2Present || !post2Acquire || !frameRateVote || !appDeadlineMisses ||
145             !present2PresentDelta) {
146             return AStatsManager_PULL_SKIP;
147         }
148 
149         android::util::addAStatsEvent(data, android::util::SURFACEFLINGER_STATS_LAYER_INFO,
150                                       atom.layer_name().c_str(), atom.total_frames(),
151                                       atom.dropped_frames(), present2Present.value(),
152                                       post2present.value(), acquire2Present.value(),
153                                       latch2Present.value(), desired2Present.value(),
154                                       post2Acquire.value(), atom.late_acquire_frames(),
155                                       atom.bad_desired_present_frames(), atom.uid(),
156                                       atom.total_timeline_frames(), atom.total_janky_frames(),
157                                       atom.total_janky_frames_with_long_cpu(),
158                                       atom.total_janky_frames_with_long_gpu(),
159                                       atom.total_janky_frames_sf_unattributed(),
160                                       atom.total_janky_frames_app_unattributed(),
161                                       atom.total_janky_frames_sf_scheduling(),
162                                       atom.total_jank_frames_sf_prediction_error(),
163                                       atom.total_jank_frames_app_buffer_stuffing(),
164                                       atom.display_refresh_rate_bucket(), atom.render_rate_bucket(),
165                                       frameRateVote.value(), appDeadlineMisses.value(),
166                                       atom.game_mode(), present2PresentDelta.value());
167     }
168     return AStatsManager_PULL_SUCCESS;
169 }
170 
171 } // namespace stats
172 } // namespace server
173 } // namespace android
174