1 /*
2  * Copyright (C) 2019 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 <android-base/thread_annotations.h>
20 #include "AnalyticsActions.h"
21 #include "AnalyticsState.h"
22 #include "AudioPowerUsage.h"
23 #include "HeatMap.h"
24 #include "StatsdLog.h"
25 #include "TimedAction.h"
26 #include "Wrap.h"
27 
28 namespace android::mediametrics {
29 
30 class AudioAnalytics
31 {
32     // AudioAnalytics action / state helper classes
33     friend AudioPowerUsage;
34 
35 public:
36     explicit AudioAnalytics(const std::shared_ptr<StatsdLog>& statsdLog);
37     ~AudioAnalytics();
38 
39     /**
40      * Returns success if AudioAnalytics recognizes item.
41      *
42      * AudioAnalytics requires the item key to start with "audio.".
43      *
44      * A trusted source can create a new key, an untrusted source
45      * can only modify the key if the uid will match that authorized
46      * on the existing key.
47      *
48      * \param item the item to be submitted.
49      * \param isTrusted whether the transaction comes from a trusted source.
50      *        In this case, a trusted source is verified by binder
51      *        UID to be a system service by MediaMetrics service.
52      *        Do not use true if you haven't really checked!
53      *
54      * \return NO_ERROR on success,
55      *         PERMISSION_DENIED if the item cannot be put into the AnalyticsState,
56      *         BAD_VALUE if the item key does not start with "audio.".
57      */
58     status_t submit(const std::shared_ptr<const mediametrics::Item>& item, bool isTrusted);
59 
60     /**
61      * Returns a pair consisting of the dump string, and the number of lines in the string.
62      *
63      * The number of lines in the returned pair is used as an optimization
64      * for subsequent line limiting.
65      *
66      * The TimeMachine and the TransactionLog are dumped separately under
67      * different locks, so may not be 100% consistent with the last data
68      * delivered.
69      *
70      * \param lines the maximum number of lines in the string returned.
71      * \param sinceNs the nanoseconds since Unix epoch to start dump (0 shows all)
72      * \param prefix the desired key prefix to match (nullptr shows all)
73      */
74     std::pair<std::string, int32_t> dump(
75             int32_t lines = INT32_MAX, int64_t sinceNs = 0, const char *prefix = nullptr) const;
76 
77     /**
78      * Returns a pair consisting of the dump string and the number of lines in the string.
79      *
80      * HeatMap dump.
81      */
82     std::pair<std::string, int32_t> dumpHeatMap(int32_t lines = INT32_MAX) const {
83         return mHeatMap.dump(lines);
84     }
85 
86     /**
87      * Returns a pair consisting of the dump string and the number of lines in the string.
88      *
89      * Health dump.
90      */
91     std::pair<std::string, int32_t> dumpHealth(int32_t lines = INT32_MAX) const {
92         return mHealth.dump(lines);
93     }
94 
95     /**
96      * Returns a pair consisting of the dump string and the number of lines in the string.
97      *
98      * Spatializer dump.
99      */
100     std::pair<std::string, int32_t> dumpSpatializer(int32_t lines = INT32_MAX) const {
101         return mSpatializer.dump(lines);
102     }
103 
clear()104     void clear() {
105         // underlying state is locked.
106         mPreviousAnalyticsState->clear();
107         mAnalyticsState->clear();
108 
109         // Clears the status map
110         mHeatMap.clear();
111 
112         // Clear power usage state.
113         mAudioPowerUsage.clear();
114     }
115 
116 private:
117 
118     /*
119      * AudioAnalytics class does not contain a monitor mutex.
120      * Instead, all of its variables are individually locked for access.
121      * Since data and items are generally added only (gc removes it), this is a reasonable
122      * compromise for availability/concurrency versus consistency.
123      *
124      * It is possible for concurrent threads to be reading and writing inside of AudioAnalytics.
125      * Reads based on a prior time (e.g. one second) in the past from the TimeMachine can be
126      * used to achieve better consistency if needed.
127      */
128 
129     /**
130      * Processes any pending actions for a particular item.
131      *
132      * \param item to check against the current AnalyticsActions.
133      */
134     void processActions(const std::shared_ptr<const mediametrics::Item>& item);
135 
136     /**
137      * Processes status information contained in the item.
138      *
139      * \param item to check against for status handling
140      */
141     void processStatus(const std::shared_ptr<const mediametrics::Item>& item);
142 
143     // Specific reporting methods
144     bool reportAudioRecordStatus(
145             const std::shared_ptr<const mediametrics::Item>& item,
146             const std::string& key, const std::string& eventStr,
147             const std::string& statusString, uid_t uid, const std::string& message,
148             int32_t subCode) const;
149 
150     bool reportAudioTrackStatus(
151             const std::shared_ptr<const mediametrics::Item>& item,
152             const std::string& key, const std::string& eventStr,
153             const std::string& statusString, uid_t uid, const std::string& message,
154             int32_t subCode) const;
155 
156     // HELPER METHODS
157     /**
158      * Return the audio thread associated with an audio track name.
159      * e.g. "audio.track.32" -> "audio.thread.10" if the associated
160      * threadId for the audio track is 10.
161      */
162     std::string getThreadFromTrack(const std::string& track) const;
163 
164     /**
165      * return the device name, if present.
166      *
167      * This is currently enabled only for Bluetooth output devices.
168      */
169     std::string getDeviceNamesFromOutputDevices(std::string_view devices) const;
170 
171     const bool mDeliverStatistics;
172 
173     // Actions is individually locked
174     AnalyticsActions mActions;
175 
176     // AnalyticsState is individually locked, and we use SharedPtrWrap
177     // to allow safe access even if the shared pointer changes underneath.
178     // These wrap pointers always point to a valid state object.
179     SharedPtrWrap<AnalyticsState> mAnalyticsState;
180     SharedPtrWrap<AnalyticsState> mPreviousAnalyticsState;
181 
182     TimedAction mTimedAction; // locked internally
183     const std::shared_ptr<StatsdLog> mStatsdLog; // locked internally, ok for multiple threads.
184 
185     static constexpr size_t kHeatEntries = 100;
186     HeatMap mHeatMap{kHeatEntries}; // locked internally, ok for multiple threads.
187 
188     // DeviceUse is a nested class which handles audio device usage accounting.
189     // We define this class at the end to ensure prior variables all properly constructed.
190     // TODO: Track / Thread interaction
191     // TODO: Consider statistics aggregation.
192     class DeviceUse {
193     public:
194         enum ItemType {
195             RECORD = 0,
196             THREAD = 1,
197             TRACK = 2,
198         };
199 
DeviceUse(AudioAnalytics & audioAnalytics)200         explicit DeviceUse(AudioAnalytics &audioAnalytics) : mAudioAnalytics{audioAnalytics} {}
201 
202         // Called every time an endAudioIntervalGroup message is received.
203         void endAudioIntervalGroup(
204                 const std::shared_ptr<const android::mediametrics::Item> &item,
205                 ItemType itemType) const;
206 
207     private:
208         AudioAnalytics &mAudioAnalytics;
209     } mDeviceUse{*this};
210 
211     // DeviceConnected is a nested class which handles audio device connection
212     // We define this class at the end to ensure prior variables all properly constructed.
213     // TODO: Track / Thread interaction
214     // TODO: Consider statistics aggregation.
215     class DeviceConnection {
216     public:
DeviceConnection(AudioAnalytics & audioAnalytics)217         explicit DeviceConnection(AudioAnalytics &audioAnalytics)
218             : mAudioAnalytics{audioAnalytics} {}
219 
220         // Called every time an endAudioIntervalGroup message is received.
221         void a2dpConnected(
222                 const std::shared_ptr<const android::mediametrics::Item> &item);
223 
224         // Called when we have an AudioFlinger createPatch
225         void createPatch(
226                 const std::shared_ptr<const android::mediametrics::Item> &item);
227 
228         // Called through AudioManager when the BT service wants to notify connection
229         void postBluetoothA2dpDeviceConnectionStateSuppressNoisyIntent(
230                 const std::shared_ptr<const android::mediametrics::Item> &item);
231 
232         // When the timer expires.
233         void expire();
234 
235     private:
236         AudioAnalytics &mAudioAnalytics;
237 
238         mutable std::mutex mLock;
239         std::string mA2dpDeviceName;
240         int64_t mA2dpConnectionRequestNs GUARDED_BY(mLock) = 0;  // Time for BT service request.
241         int64_t mA2dpConnectionServiceNs GUARDED_BY(mLock) = 0;  // Time audio service agrees.
242 
243         int32_t mA2dpConnectionRequests GUARDED_BY(mLock) = 0;
244         int32_t mA2dpConnectionServices GUARDED_BY(mLock) = 0;
245 
246         // See the statsd atoms.proto
247         int32_t mA2dpConnectionSuccesses GUARDED_BY(mLock) = 0;
248         int32_t mA2dpConnectionJavaServiceCancels GUARDED_BY(mLock) = 0;
249         int32_t mA2dpConnectionUnknowns GUARDED_BY(mLock) = 0;
250     } mDeviceConnection{*this};
251 
252     // AAudioStreamInfo is a nested class which collect aaudio stream info from both client and
253     // server side.
254     class AAudioStreamInfo {
255     public:
256         // All the enum here must be kept the same as the ones defined in atoms.proto
257         enum CallerPath {
258             CALLER_PATH_UNKNOWN = 0,
259             CALLER_PATH_LEGACY = 1,
260             CALLER_PATH_MMAP = 2,
261         };
262 
AAudioStreamInfo(AudioAnalytics & audioAnalytics)263         explicit AAudioStreamInfo(AudioAnalytics &audioAnalytics)
264             : mAudioAnalytics(audioAnalytics) {}
265 
266         void endAAudioStream(
267                 const std::shared_ptr<const android::mediametrics::Item> &item,
268                 CallerPath path) const;
269 
270     private:
271 
272         AudioAnalytics &mAudioAnalytics;
273     } mAAudioStreamInfo{*this};
274 
275     // Create new state, typically occurs after an AudioFlinger ctor event.
276     void newState();
277 
278     // Health is a nested class that tracks audioserver health properties
279     class Health {
280     public:
Health(AudioAnalytics & audioAnalytics)281         explicit Health(AudioAnalytics &audioAnalytics)
282             : mAudioAnalytics(audioAnalytics) {}
283 
284         enum class Module {
285             AUDIOFLINGER,
286             AUDIOPOLICY,
287         };
288 
getModuleName(Module module)289         const char *getModuleName(Module module) {
290             switch (module) {
291                 case Module::AUDIOFLINGER: return "AudioFlinger";
292                 case Module::AUDIOPOLICY: return "AudioPolicy";
293             }
294             return "Unknown";
295         }
296 
297         // Called when we believe audioserver starts (AudioFlinger ctor)
298         void onAudioServerStart(Module module,
299                 const std::shared_ptr<const android::mediametrics::Item> &item);
300 
301         // Called when we believe audioserver crashes (TimeCheck timeouts).
302         void onAudioServerTimeout(Module module,
303                 const std::shared_ptr<const android::mediametrics::Item> &item);
304 
305         std::pair<std::string, int32_t> dump(
306                 int32_t lines = INT32_MAX, const char *prefix = nullptr) const;
307 
308     private:
309         AudioAnalytics& mAudioAnalytics;
310 
311         mutable std::mutex mLock;
312 
313         // Life cycle of AudioServer
314         // mAudioFlingerCtorTime
315         // mAudioPolicyCtorTime
316         // mAudioPolicyCtorDoneTime
317         // ...
318         // possibly mStopTime  (if TimeCheck thread)
319         //
320         // UpTime is measured from mStopTime - mAudioFlingerCtorTime.
321         //
322         // The stop events come from TimeCheck timeout aborts.  There may be other
323         // uncaught signals, e.g. SIGSEGV, that cause missing stop events.
324         std::chrono::system_clock::time_point mAudioFlingerCtorTime GUARDED_BY(mLock);
325         std::chrono::system_clock::time_point mAudioPolicyCtorTime GUARDED_BY(mLock);
326         std::chrono::system_clock::time_point mAudioPolicyCtorDoneTime GUARDED_BY(mLock);
327         std::chrono::system_clock::time_point mStopTime GUARDED_BY(mLock);
328 
329         // mStartCount and mStopCount track the audioserver start and stop events.
330         int64_t mStartCount GUARDED_BY(mLock) = 0;
331         int64_t mStopCount GUARDED_BY(mLock) = 0;
332 
GUARDED_BY(mLock)333         SimpleLog mSimpleLog GUARDED_BY(mLock) {64};
334     } mHealth{*this};
335 
336     // Spatializer is a nested class that tracks related messages.
337     class Spatializer {
338     public:
Spatializer(AudioAnalytics & audioAnalytics)339         explicit Spatializer(AudioAnalytics &audioAnalytics)
340             : mAudioAnalytics(audioAnalytics) {}
341 
342         // an item that starts with "audio.spatializer"
343         void onEvent(const std::shared_ptr<const android::mediametrics::Item> &item);
344 
345         std::pair<std::string, int32_t> dump(
346                 int32_t lines = INT32_MAX, const char *prefix = nullptr) const;
347 
348     private:
349 
350         // Current device state as strings:
351         // "" means unknown, "true" or "false".
352         struct DeviceState {
353             std::string enabled;
354             std::string hasHeadTracker;
355             std::string headTrackerEnabled;
356         };
357 
358         AudioAnalytics& mAudioAnalytics;
359         static constexpr int64_t kBootDurationThreshold = 120 /* seconds */ * 1e9;
360         mutable std::mutex mLock;
361         int64_t mFirstCreateTimeNs GUARDED_BY(mLock) = 0;
362         std::map<std::string, DeviceState> mDeviceStateMap GUARDED_BY(mLock);
GUARDED_BY(mLock)363         SimpleLog mSimpleLog GUARDED_BY(mLock) {64};
364     } mSpatializer{*this};
365 
366     // MidiLogging collects info whenever a MIDI device is closed.
367     class MidiLogging {
368     public:
MidiLogging(AudioAnalytics & audioAnalytics)369         explicit MidiLogging(AudioAnalytics &audioAnalytics)
370             : mAudioAnalytics(audioAnalytics) {}
371 
372         void onEvent(
373                 const std::shared_ptr<const android::mediametrics::Item> &item) const;
374 
375     private:
376 
377         AudioAnalytics &mAudioAnalytics;
378     } mMidiLogging{*this};
379 
380     AudioPowerUsage mAudioPowerUsage;
381 };
382 
383 } // namespace android::mediametrics
384