1 /*
2  * Copyright (C) 2018 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 <chrono>
20 #include <vector>
21 
22 #include <mediautils/TimerThread.h>
23 
24 namespace android::mediautils {
25 
26 // A class monitoring execution time for a code block (scoped variable) and causing an assert
27 // if it exceeds a certain time
28 
29 class TimeCheck {
30   public:
31 
32     // Duration for TimeCheck is based on steady_clock, typically nanoseconds.
33     using Duration = std::chrono::steady_clock::duration;
34 
35     // Duration for printing is in milliseconds, using float for additional precision.
36     using FloatMs = std::chrono::duration<float, std::milli>;
37 
38     // OnTimerFunc is the callback function with 2 parameters.
39     //  bool timeout  (which is true when the TimeCheck object
40     //                 times out, false when the TimeCheck object is
41     //                 destroyed or leaves scope before the timer expires.)
42     //  float elapsedMs (the elapsed time to this event).
43     using OnTimerFunc = std::function<void(bool /* timeout */, float /* elapsedMs */ )>;
44 
45     // The default timeout is chosen to be less than system server watchdog timeout
46     // Note: kDefaultTimeOutMs should be no less than 2 seconds, otherwise spurious timeouts
47     // may occur with system suspend.
48     static constexpr TimeCheck::Duration kDefaultTimeoutDuration = std::chrono::milliseconds(3000);
49 
50     // Due to suspend abort not incrementing the monotonic clock,
51     // we allow another second chance timeout after the first timeout expires.
52     //
53     // The total timeout is therefore kDefaultTimeoutDuration + kDefaultSecondChanceDuration,
54     // and the result is more stable when the monotonic clock increments during suspend.
55     //
56     static constexpr TimeCheck::Duration kDefaultSecondChanceDuration =
57             std::chrono::milliseconds(2000);
58 
59     /**
60      * TimeCheck is a RAII object which will notify a callback
61      * on timer expiration or when the object is deallocated.
62      *
63      * TimeCheck is used as a watchdog and aborts by default on timer expiration.
64      * When it aborts, it will also send a debugger signal to pids passed in through
65      * setAudioHalPids().
66      *
67      * If the callback function returns for timeout it will not be called again for
68      * the deallocation.
69      *
70      * \param tag       string associated with the TimeCheck object.
71      * \param onTimer   callback function with 2 parameters (described above in OnTimerFunc).
72      *                  The callback when timeout is true will be called on a different thread.
73      *                  This will cancel the callback on the destructor but is not guaranteed
74      *                  to block for callback completion if it is already in progress
75      *                  (for maximum concurrency and reduced deadlock potential), so use proper
76      *                  lifetime analysis (e.g. shared or weak pointers).
77      * \param requestedTimeoutDuration timeout in milliseconds.
78      *                  A zero timeout means no timeout is set -
79      *                  the callback is called only when
80      *                  the TimeCheck object is destroyed or leaves scope.
81      * \param secondChanceDuration additional milliseconds to wait if the first timeout expires.
82      *                  This is used to prevent false timeouts if the steady (monotonic)
83      *                  clock advances on aborted suspend.
84      * \param crashOnTimeout true if the object issues an abort on timeout.
85      */
86     explicit TimeCheck(std::string_view tag, OnTimerFunc&& onTimer,
87             Duration requestedTimeoutDuration, Duration secondChanceDuration,
88             bool crashOnTimeout);
89 
90     TimeCheck() = default;
91     // Remove copy constructors as there should only be one call to the destructor.
92     // Move is kept implicitly disabled, but would be logically consistent if enabled.
93     TimeCheck(const TimeCheck& other) = delete;
94     TimeCheck& operator=(const TimeCheck&) = delete;
95 
96     ~TimeCheck();
97     static std::string toString();
98     static void setAudioHalPids(const std::vector<pid_t>& pids);
99     static std::vector<pid_t> getAudioHalPids();
100 
101   private:
102     // Helper class for handling events.
103     // The usage here is const safe.
104     class TimeCheckHandler {
105     public:
106         template <typename S, typename F>
TimeCheckHandler(S && _tag,F && _onTimer,bool _crashOnTimeout,Duration _timeoutDuration,Duration _secondChanceDuration,std::chrono::system_clock::time_point _startSystemTime,pid_t _tid)107         TimeCheckHandler(S&& _tag, F&& _onTimer, bool _crashOnTimeout,
108             Duration _timeoutDuration, Duration _secondChanceDuration,
109             std::chrono::system_clock::time_point _startSystemTime,
110             pid_t _tid)
111             : tag(std::forward<S>(_tag))
112             , onTimer(std::forward<F>(_onTimer))
113             , crashOnTimeout(_crashOnTimeout)
114             , timeoutDuration(_timeoutDuration)
115             , secondChanceDuration(_secondChanceDuration)
116             , startSystemTime(_startSystemTime)
117             , tid(_tid)
118             {}
119         const FixedString62 tag;
120         const OnTimerFunc onTimer;
121         const bool crashOnTimeout;
122         const Duration timeoutDuration;
123         const Duration secondChanceDuration;
124         const std::chrono::system_clock::time_point startSystemTime;
125         const pid_t tid;
126         void onCancel(TimerThread::Handle handle) const;
127         void onTimeout(TimerThread::Handle handle) const;
128     };
129 
130     // Returns a string that represents the timeout vs elapsed time,
131     // and diagnostics if there are any potential issues.
132     static std::string analyzeTimeouts(
133             float timeoutMs, float elapsedSteadyMs, float elapsedSystemMs);
134 
135     static TimerThread& getTimeCheckThread();
136     static void accessAudioHalPids(std::vector<pid_t>* pids, bool update);
137 
138     // mTimeCheckHandler is immutable, prefer to be first initialized, last destroyed.
139     // Technically speaking, we do not need a shared_ptr here because TimerThread::cancelTask()
140     // is mutually exclusive of the callback, but the price paid for lifetime safety is minimal.
141     const std::shared_ptr<const TimeCheckHandler> mTimeCheckHandler;
142     const TimerThread::Handle mTimerHandle = TimerThread::INVALID_HANDLE;
143 };
144 
145 // Returns a TimeCheck object that sends info to MethodStatistics
146 // obtained from getStatisticsForClass(className).
147 TimeCheck makeTimeCheckStatsForClassMethod(
148         std::string_view className, std::string_view methodName);
149 
150 // A handy statement-like macro to put at the beginning of almost every method
151 // which calls into HAL. Note that it requires the class to implement 'getClassName'.
152 #define TIME_CHECK() auto timeCheck = \
153             mediautils::makeTimeCheckStatsForClassMethod(getClassName(), __func__)
154 
155 }  // namespace android::mediautils
156