| Adrian Roos | 1e1a128 | 2017-11-01 19:05:31 +0100 | [diff] [blame] | 1 | /* | 
|  | 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 |  | 
|  | 19 | #include <layerproto/LayerProtoHeader.h> | 
|  | 20 | #include <utils/Errors.h> | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 21 | #include <utils/StrongPointer.h> | 
| Adrian Roos | 1e1a128 | 2017-11-01 19:05:31 +0100 | [diff] [blame] | 22 |  | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 23 | #include <android-base/thread_annotations.h> | 
|  | 24 | #include <condition_variable> | 
| Chia-I Wu | a3e7ddc | 2018-09-20 11:42:46 -0700 | [diff] [blame] | 25 | #include <memory> | 
| Adrian Roos | 1e1a128 | 2017-11-01 19:05:31 +0100 | [diff] [blame] | 26 | #include <mutex> | 
| Yichi Chen | 9c696ed | 2018-10-01 22:32:30 +0800 | [diff] [blame] | 27 | #include <queue> | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 28 | #include <thread> | 
| Adrian Roos | 1e1a128 | 2017-11-01 19:05:31 +0100 | [diff] [blame] | 29 |  | 
|  | 30 | using namespace android::surfaceflinger; | 
|  | 31 |  | 
|  | 32 | namespace android { | 
|  | 33 |  | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 34 | class SurfaceFlinger; | 
|  | 35 |  | 
| Yichi Chen | 9c696ed | 2018-10-01 22:32:30 +0800 | [diff] [blame] | 36 | constexpr auto operator""_MB(unsigned long long const num) { | 
|  | 37 | return num * 1024 * 1024; | 
|  | 38 | } | 
| Adrian Roos | 1e1a128 | 2017-11-01 19:05:31 +0100 | [diff] [blame] | 39 | /* | 
|  | 40 | * SurfaceTracing records layer states during surface flinging. | 
|  | 41 | */ | 
|  | 42 | class SurfaceTracing { | 
|  | 43 | public: | 
| Dominik Laskowski | 9dab343 | 2019-03-27 13:21:10 -0700 | [diff] [blame] | 44 | explicit SurfaceTracing(SurfaceFlinger& flinger); | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 45 | void enable(); | 
|  | 46 | bool disable(); | 
|  | 47 | status_t writeToFile(); | 
| Yichi Chen | 9c696ed | 2018-10-01 22:32:30 +0800 | [diff] [blame] | 48 | bool isEnabled() const; | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 49 | void notify(const char* where); | 
|  | 50 |  | 
|  | 51 | void setBufferSize(size_t bufferSizeInByte); | 
|  | 52 | void writeToFileAsync(); | 
| Yiwei Zhang | 5434a78 | 2018-12-05 18:06:32 -0800 | [diff] [blame] | 53 | void dump(std::string& result) const; | 
| Adrian Roos | 1e1a128 | 2017-11-01 19:05:31 +0100 | [diff] [blame] | 54 |  | 
| Vishnu Nair | 9245d3b | 2019-03-22 13:38:56 -0700 | [diff] [blame] | 55 | enum : uint32_t { | 
|  | 56 | TRACE_CRITICAL = 1 << 0, | 
|  | 57 | TRACE_INPUT = 1 << 1, | 
|  | 58 | TRACE_EXTRA = 1 << 2, | 
|  | 59 | TRACE_ALL = 0xffffffff | 
|  | 60 | }; | 
|  | 61 | void setTraceFlags(uint32_t flags); | 
|  | 62 |  | 
| Adrian Roos | 1e1a128 | 2017-11-01 19:05:31 +0100 | [diff] [blame] | 63 | private: | 
| Robert Delgado | 626670f | 2019-06-27 15:55:27 -0700 | [diff] [blame] | 64 | static constexpr auto kDefaultBufferCapInByte = 5_MB; | 
| Yichi Chen | 9c696ed | 2018-10-01 22:32:30 +0800 | [diff] [blame] | 65 | static constexpr auto kDefaultFileName = "/data/misc/wmtrace/layers_trace.pb"; | 
|  | 66 |  | 
|  | 67 | class LayersTraceBuffer { // ring buffer | 
|  | 68 | public: | 
|  | 69 | size_t size() const { return mSizeInBytes; } | 
|  | 70 | size_t used() const { return mUsedInBytes; } | 
|  | 71 | size_t frameCount() const { return mStorage.size(); } | 
|  | 72 |  | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 73 | void setSize(size_t newSize) { mSizeInBytes = newSize; } | 
| Yichi Chen | 9c696ed | 2018-10-01 22:32:30 +0800 | [diff] [blame] | 74 | void reset(size_t newSize); | 
|  | 75 | void emplace(LayersTraceProto&& proto); | 
|  | 76 | void flush(LayersTraceFileProto* fileProto); | 
|  | 77 |  | 
|  | 78 | private: | 
|  | 79 | size_t mUsedInBytes = 0U; | 
|  | 80 | size_t mSizeInBytes = 0U; | 
|  | 81 | std::queue<LayersTraceProto> mStorage; | 
|  | 82 | }; | 
| Adrian Roos | 1e1a128 | 2017-11-01 19:05:31 +0100 | [diff] [blame] | 83 |  | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 84 | void mainLoop(); | 
| Vishnu Nair | 9245d3b | 2019-03-22 13:38:56 -0700 | [diff] [blame] | 85 | void addFirstEntry(); | 
|  | 86 | LayersTraceProto traceWhenNotified(); | 
|  | 87 | LayersTraceProto traceLayersLocked(const char* where) REQUIRES(mSfLock); | 
|  | 88 |  | 
|  | 89 | // Returns true if trace is enabled. | 
|  | 90 | bool addTraceToBuffer(LayersTraceProto& entry); | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 91 | void writeProtoFileLocked() REQUIRES(mTraceLock); | 
| Adrian Roos | 1e1a128 | 2017-11-01 19:05:31 +0100 | [diff] [blame] | 92 |  | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 93 | const SurfaceFlinger& mFlinger; | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 94 | status_t mLastErr = NO_ERROR; | 
|  | 95 | std::thread mThread; | 
| Vishnu Nair | 9245d3b | 2019-03-22 13:38:56 -0700 | [diff] [blame] | 96 | std::condition_variable mCanStartTrace; | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 97 |  | 
| Vishnu Nair | 9245d3b | 2019-03-22 13:38:56 -0700 | [diff] [blame] | 98 | std::mutex& mSfLock; | 
|  | 99 | uint32_t mTraceFlags GUARDED_BY(mSfLock) = TRACE_ALL; | 
|  | 100 | const char* mWhere GUARDED_BY(mSfLock) = ""; | 
|  | 101 |  | 
|  | 102 | mutable std::mutex mTraceLock; | 
| Nataniel Borges | 2b796da | 2019-02-15 13:32:18 -0800 | [diff] [blame] | 103 | LayersTraceBuffer mBuffer GUARDED_BY(mTraceLock); | 
|  | 104 | size_t mBufferSize GUARDED_BY(mTraceLock) = kDefaultBufferCapInByte; | 
|  | 105 | bool mEnabled GUARDED_BY(mTraceLock) = false; | 
|  | 106 | bool mWriteToFile GUARDED_BY(mTraceLock) = false; | 
| Adrian Roos | 1e1a128 | 2017-11-01 19:05:31 +0100 | [diff] [blame] | 107 | }; | 
|  | 108 |  | 
|  | 109 | } // namespace android |