blob: cea1a3324e22afad8f9fd4b112dbcc2f415bbc96 [file] [log] [blame]
Adrian Roos1e1a1282017-11-01 19:05:31 +01001/*
2 * Copyright 2017 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#pragma once
18
Alec Mouri5793c7d2020-03-10 19:55:50 -070019#include <android-base/thread_annotations.h>
Adrian Roos1e1a1282017-11-01 19:05:31 +010020#include <layerproto/LayerProtoHeader.h>
21#include <utils/Errors.h>
Nataniel Borges2b796da2019-02-15 13:32:18 -080022#include <utils/StrongPointer.h>
Adrian Roos1e1a1282017-11-01 19:05:31 +010023
Nataniel Borges2b796da2019-02-15 13:32:18 -080024#include <condition_variable>
Chia-I Wua3e7ddc2018-09-20 11:42:46 -070025#include <memory>
Adrian Roos1e1a1282017-11-01 19:05:31 +010026#include <mutex>
Yichi Chen9c696ed2018-10-01 22:32:30 +080027#include <queue>
Nataniel Borges2b796da2019-02-15 13:32:18 -080028#include <thread>
Adrian Roos1e1a1282017-11-01 19:05:31 +010029
30using namespace android::surfaceflinger;
31
32namespace android {
33
Nataniel Borges2b796da2019-02-15 13:32:18 -080034class SurfaceFlinger;
Yichi Chen9c696ed2018-10-01 22:32:30 +080035constexpr auto operator""_MB(unsigned long long const num) {
36 return num * 1024 * 1024;
37}
Adrian Roos1e1a1282017-11-01 19:05:31 +010038/*
Vishnu Nair31a8dbc2020-10-27 17:37:49 -070039 * SurfaceTracing records layer states during surface flinging. Manages tracing state and
40 * configuration.
Adrian Roos1e1a1282017-11-01 19:05:31 +010041 */
42class SurfaceTracing {
43public:
Vishnu Nair31a8dbc2020-10-27 17:37:49 -070044 SurfaceTracing(SurfaceFlinger& flinger);
Dominik Laskowski542c9dc2020-04-10 12:42:02 -070045 bool enable();
Nataniel Borges2b796da2019-02-15 13:32:18 -080046 bool disable();
47 status_t writeToFile();
Yichi Chen9c696ed2018-10-01 22:32:30 +080048 bool isEnabled() const;
Vishnu Nair31a8dbc2020-10-27 17:37:49 -070049 /*
50 * Adds a trace entry, must be called from the drawing thread or while holding the
51 * SurfaceFlinger tracing lock.
52 */
Nataniel Borges2b796da2019-02-15 13:32:18 -080053 void notify(const char* where);
Vishnu Nair31a8dbc2020-10-27 17:37:49 -070054 /*
55 * Adds a trace entry, called while holding the SurfaceFlinger tracing lock.
56 */
57 void notifyLocked(const char* where) /* REQUIRES(mSfLock) */;
Nataniel Borges2b796da2019-02-15 13:32:18 -080058
Vishnu Nair31a8dbc2020-10-27 17:37:49 -070059 void setBufferSize(size_t bufferSizeInBytes) { mConfig.bufferSize = bufferSizeInBytes; }
Yiwei Zhang5434a782018-12-05 18:06:32 -080060 void dump(std::string& result) const;
Adrian Roos1e1a1282017-11-01 19:05:31 +010061
Vishnu Nair9245d3b2019-03-22 13:38:56 -070062 enum : uint32_t {
63 TRACE_CRITICAL = 1 << 0,
64 TRACE_INPUT = 1 << 1,
Vishnu Nair60db8c02020-04-02 11:55:16 -070065 TRACE_COMPOSITION = 1 << 2,
66 TRACE_EXTRA = 1 << 3,
67 TRACE_HWC = 1 << 4,
Vishnu Nair31a8dbc2020-10-27 17:37:49 -070068 // Add non-geometry composition changes to the trace.
69 TRACE_BUFFERS = 1 << 5,
70 // Add entries from the drawing thread post composition.
71 TRACE_SYNC = 1 << 6,
72 TRACE_ALL = TRACE_CRITICAL | TRACE_INPUT | TRACE_COMPOSITION | TRACE_EXTRA,
Vishnu Nair9245d3b2019-03-22 13:38:56 -070073 };
Vishnu Nair31a8dbc2020-10-27 17:37:49 -070074 void setTraceFlags(uint32_t flags) { mConfig.flags = flags; }
75 bool flagIsSet(uint32_t flags) { return (mConfig.flags & flags) == flags; }
chaviw66e11cf2021-08-13 10:13:01 -050076 static LayersTraceFileProto createLayersTraceFileProto();
Vishnu Nair9245d3b2019-03-22 13:38:56 -070077
Adrian Roos1e1a1282017-11-01 19:05:31 +010078private:
Vishnu Nair31a8dbc2020-10-27 17:37:49 -070079 class Runner;
80 static constexpr auto DEFAULT_BUFFER_SIZE = 5_MB;
Nataniel Borges13881062021-08-31 10:26:28 +000081 static constexpr auto DEFAULT_FILE_NAME = "/data/misc/wmtrace/layers_trace.winscope";
Yichi Chen9c696ed2018-10-01 22:32:30 +080082
Vishnu Nair31a8dbc2020-10-27 17:37:49 -070083 SurfaceFlinger& mFlinger;
84 mutable std::mutex mTraceLock;
Vishnu Nairc96ad662021-02-12 11:23:19 -080085 bool mEnabled GUARDED_BY(mTraceLock) = false;
86 std::unique_ptr<Runner> runner GUARDED_BY(mTraceLock);
Vishnu Nair31a8dbc2020-10-27 17:37:49 -070087
88 struct Config {
Vishnu Nair49353772020-12-09 17:49:57 -080089 uint32_t flags = TRACE_CRITICAL | TRACE_INPUT | TRACE_SYNC;
Vishnu Nair31a8dbc2020-10-27 17:37:49 -070090 size_t bufferSize = DEFAULT_BUFFER_SIZE;
91 } mConfig;
92
93 /*
94 * ring buffer.
95 */
96 class LayersTraceBuffer {
Yichi Chen9c696ed2018-10-01 22:32:30 +080097 public:
98 size_t size() const { return mSizeInBytes; }
99 size_t used() const { return mUsedInBytes; }
100 size_t frameCount() const { return mStorage.size(); }
101
Nataniel Borges2b796da2019-02-15 13:32:18 -0800102 void setSize(size_t newSize) { mSizeInBytes = newSize; }
Yichi Chen9c696ed2018-10-01 22:32:30 +0800103 void reset(size_t newSize);
104 void emplace(LayersTraceProto&& proto);
105 void flush(LayersTraceFileProto* fileProto);
106
107 private:
108 size_t mUsedInBytes = 0U;
Vishnu Nair31a8dbc2020-10-27 17:37:49 -0700109 size_t mSizeInBytes = DEFAULT_BUFFER_SIZE;
Yichi Chen9c696ed2018-10-01 22:32:30 +0800110 std::queue<LayersTraceProto> mStorage;
111 };
Adrian Roos1e1a1282017-11-01 19:05:31 +0100112
Vishnu Nair31a8dbc2020-10-27 17:37:49 -0700113 /*
114 * Implements a synchronous way of adding trace entries. This must be called
115 * from the drawing thread.
116 */
117 class Runner {
118 public:
119 Runner(SurfaceFlinger& flinger, SurfaceTracing::Config& config);
120 virtual ~Runner() = default;
121 virtual status_t stop();
122 virtual status_t writeToFile();
123 virtual void notify(const char* where);
124 /* Cannot be called with a synchronous runner. */
125 virtual void notifyLocked(const char* /* where */) {}
126 void dump(std::string& result) const;
Vishnu Nair9245d3b2019-03-22 13:38:56 -0700127
Vishnu Nair31a8dbc2020-10-27 17:37:49 -0700128 protected:
129 bool flagIsSet(uint32_t flags) { return (mConfig.flags & flags) == flags; }
130 SurfaceFlinger& mFlinger;
131 SurfaceTracing::Config mConfig;
132 SurfaceTracing::LayersTraceBuffer mBuffer;
133 uint32_t mMissedTraceEntries = 0;
134 LayersTraceProto traceLayers(const char* where);
135 };
Adrian Roos1e1a1282017-11-01 19:05:31 +0100136
Vishnu Nair31a8dbc2020-10-27 17:37:49 -0700137 /*
138 * Implements asynchronous way to add trace entries called from a separate thread while holding
139 * the SurfaceFlinger tracing lock. Trace entries may be missed if the tracing thread is not
140 * scheduled in time.
141 */
142 class AsyncRunner : public Runner {
143 public:
144 AsyncRunner(SurfaceFlinger& flinger, SurfaceTracing::Config& config, std::mutex& sfLock);
145 virtual ~AsyncRunner() = default;
146 status_t stop() override;
147 status_t writeToFile() override;
148 void notify(const char* where) override;
149 void notifyLocked(const char* where);
Nataniel Borges2b796da2019-02-15 13:32:18 -0800150
Vishnu Nair31a8dbc2020-10-27 17:37:49 -0700151 private:
152 std::mutex& mSfLock;
153 std::condition_variable mCanStartTrace;
154 std::thread mThread;
155 const char* mWhere = "";
156 bool mWriteToFile = false;
157 bool mEnabled = false;
158 bool mAddEntry = false;
159 void loop();
160 bool traceWhenNotified(LayersTraceProto* outProto);
161 };
Adrian Roos1e1a1282017-11-01 19:05:31 +0100162};
163
164} // namespace android