| 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: | 
| Yichi Chen | 9c696ed | 2018-10-01 22:32:30 +0800 | [diff] [blame] | 64 |     static constexpr auto kDefaultBufferCapInByte = 100_MB; | 
 | 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 |