blob: ce89b24a6e6727113d57a75fb3cb4eba274d996e [file] [log] [blame]
Vlad Popab042ee62022-10-20 18:05:00 +02001/*
2**
3** Copyright 2022, The Android Open Source Project
4**
5** Licensed under the Apache License, Version 2.0 (the "License");
6** you may not use this file except in compliance with the License.
7** You may obtain a copy of the License at
8**
9** http://www.apache.org/licenses/LICENSE-2.0
10**
11** Unless required by applicable law or agreed to in writing, software
12** distributed under the License is distributed on an "AS IS" BASIS,
13** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14** See the License for the specific language governing permissions and
15** limitations under the License.
16*/
17
Andy Hung81ce7602023-07-13 20:00:50 -070018#pragma once
Vlad Popab042ee62022-10-20 18:05:00 +020019
Andy Hung4f274952023-07-18 18:55:52 -070020#include "IAfPatchPanel.h"
21#include "PatchCommandThread.h"
22
Andy Hung85a07452023-08-28 18:36:53 -070023#include <audio_utils/mutex.h>
Vlad Popa2900c0a2022-10-24 13:38:00 +020024#include <sounddose/SoundDoseManager.h>
Andy Hung4f274952023-07-18 18:55:52 -070025
Vlad Popa2900c0a2022-10-24 13:38:00 +020026#include <unordered_map>
Vlad Popab042ee62022-10-20 18:05:00 +020027
Andy Hung81ce7602023-07-13 20:00:50 -070028namespace android {
29
Vlad Popab042ee62022-10-20 18:05:00 +020030constexpr static int kMaxTimestampDeltaInSec = 120;
31
Andy Hungf9e248b2023-07-17 14:02:52 -070032class IAfMelReporterCallback : public virtual RefBase {
33public:
Andy Hung5b726572023-08-31 15:30:19 -070034 virtual audio_utils::mutex& mutex() const
35 RETURN_CAPABILITY(audio_utils::AudioFlinger_Mutex) = 0;
Andy Hungf9e248b2023-07-17 14:02:52 -070036 virtual const sp<PatchCommandThread>& getPatchCommandThread() = 0;
Andy Hung5b726572023-08-31 15:30:19 -070037 virtual sp<IAfThreadBase> checkOutputThread_l(audio_io_handle_t ioHandle) const
38 REQUIRES(mutex()) = 0;
Andy Hungf9e248b2023-07-17 14:02:52 -070039};
40
Vlad Popab042ee62022-10-20 18:05:00 +020041/**
42 * Class for listening to new patches and starting the MEL computation. MelReporter is
43 * concealed within AudioFlinger, their lifetimes are the same.
44 */
Vlad Popa197faf82023-07-27 18:27:59 -070045class MelReporter : public PatchCommandThread::PatchCommandListener,
46 public IMelReporterCallback {
Vlad Popab042ee62022-10-20 18:05:00 +020047public:
Andy Hungf9e248b2023-07-17 14:02:52 -070048 explicit MelReporter(const sp<IAfMelReporterCallback>& afMelReporterCallback)
49 : mAfMelReporterCallback(afMelReporterCallback) {}
Vlad Popab042ee62022-10-20 18:05:00 +020050
Vlad Popa1d5f0d52022-12-18 12:21:26 +010051 void onFirstRef() override;
Vlad Popab042ee62022-10-20 18:05:00 +020052
Vlad Popa1d5f0d52022-12-18 12:21:26 +010053 /**
54 * Activates the MEL reporting from the HAL sound dose interface. If the HAL
55 * does not support the sound dose interface for this module, the internal MEL
56 * calculation will be use.
57 *
Vlad Popa03bd5bc2023-01-17 16:16:51 +010058 * <p>If the device is using the audio AIDL HAL then this method will try to get the sound
59 * dose interface from IModule#getSoundDose(). Otherwise, if the legacy audio HIDL HAL is used
60 * this method will be looking for the standalone sound dose implementation. It falls back to
61 * the internal MEL computation if no valid sound dose interface can be retrieved.
Vlad Popa1d5f0d52022-12-18 12:21:26 +010062 *
Vlad Popa03bd5bc2023-01-17 16:16:51 +010063 * @return true if the MEL reporting will be done from any sound dose HAL interface
64 * implementation, false otherwise.
Vlad Popa1d5f0d52022-12-18 12:21:26 +010065 */
Vlad Popa03bd5bc2023-01-17 16:16:51 +010066 bool activateHalSoundDoseComputation(const std::string& module,
Andy Hungcd88bfe2023-09-06 17:52:01 -070067 const sp<DeviceHalInterface>& device) EXCLUDES_MelReporter_Mutex;
Vlad Popa1d5f0d52022-12-18 12:21:26 +010068
69 /**
70 * Activates the MEL reporting from internal framework values. These are used
71 * as a fallback when there is no sound dose interface implementation from HAL.
72 * Note: the internal CSD computation does not guarantee a certification with
73 * IEC62368-1 3rd edition or EN50332-3
74 */
Andy Hungcd88bfe2023-09-06 17:52:01 -070075 void activateInternalSoundDoseComputation() EXCLUDES_MelReporter_Mutex;
Vlad Popab042ee62022-10-20 18:05:00 +020076
Vlad Popae3fd1c22022-11-07 21:03:18 +010077 sp<media::ISoundDose> getSoundDoseInterface(const sp<media::ISoundDoseCallback>& callback);
Vlad Popa63f047e2022-11-05 14:09:19 +010078
Vlad Popab042ee62022-10-20 18:05:00 +020079 std::string dump();
80
Vlad Popa197faf82023-07-27 18:27:59 -070081 // IMelReporterCallback methods
Andy Hungcd88bfe2023-09-06 17:52:01 -070082 void stopMelComputationForDeviceId(audio_port_handle_t deviceId) final
83 EXCLUDES_MelReporter_Mutex;
84 void startMelComputationForDeviceId(audio_port_handle_t deviceId) final
85 EXCLUDES_MelReporter_Mutex;
Vlad Popa197faf82023-07-27 18:27:59 -070086
Vlad Popab042ee62022-10-20 18:05:00 +020087 // PatchCommandListener methods
88 void onCreateAudioPatch(audio_patch_handle_t handle,
Andy Hungcd88bfe2023-09-06 17:52:01 -070089 const IAfPatchPanel::Patch& patch) final
90 EXCLUDES_AudioFlinger_Mutex;
91 void onReleaseAudioPatch(audio_patch_handle_t handle) final EXCLUDES_AudioFlinger_Mutex;
Vlad Popab042ee62022-10-20 18:05:00 +020092
Vlad Popa7e81cea2023-01-19 16:34:16 +010093 /**
94 * The new metadata can determine whether we should compute MEL for the given thread.
95 * This is the case only if one of the tracks in the thread mix is using MEDIA or GAME.
96 * Otherwise, this method will disable CSD.
97 **/
98 void updateMetadataForCsd(audio_io_handle_t streamHandle,
Andy Hungcd88bfe2023-09-06 17:52:01 -070099 const std::vector<playback_track_metadata_v7_t>& metadataVec)
100 EXCLUDES_AudioFlinger_Mutex;
101
Vlad Popab042ee62022-10-20 18:05:00 +0200102private:
Vlad Popa7e81cea2023-01-19 16:34:16 +0100103 struct ActiveMelPatch {
104 audio_io_handle_t streamHandle{AUDIO_IO_HANDLE_NONE};
Vlad Popa197faf82023-07-27 18:27:59 -0700105 /**
106 * Stores device ids and whether they are compatible for CSD calculation.
107 * The boolean value can change since BT audio device types are user-configurable
108 * to headphones/headsets or other device types.
109 */
110 std::vector<std::pair<audio_port_handle_t,bool>> deviceStates;
Vlad Popa7e81cea2023-01-19 16:34:16 +0100111 bool csdActive;
112 };
Vlad Popab042ee62022-10-20 18:05:00 +0200113
Vlad Popa7e81cea2023-01-19 16:34:16 +0100114 void stopInternalMelComputation();
Andy Hungcd88bfe2023-09-06 17:52:01 -0700115 audio_utils::mutex& mutex() const RETURN_CAPABILITY(audio_utils::MelReporter_Mutex) {
116 return mMutex;
117 }
Vlad Popa7e81cea2023-01-19 16:34:16 +0100118
Andy Hungd65f1d82023-08-28 19:12:14 -0700119 /** Should be called with the following order of locks: mAudioFlinger.mutex() -> mutex(). */
120 void stopMelComputationForPatch_l(const ActiveMelPatch& patch) REQUIRES(mutex());
Vlad Popa7e81cea2023-01-19 16:34:16 +0100121
Andy Hungd65f1d82023-08-28 19:12:14 -0700122 /** Should be called with the following order of locks: mAudioFlinger.mutex() -> mutex(). */
123 void startMelComputationForActivePatch_l(const ActiveMelPatch& patch) REQUIRES(mutex());
Vlad Popa7e81cea2023-01-19 16:34:16 +0100124
Andy Hung920f6572022-10-06 12:09:49 -0700125 std::optional<audio_patch_handle_t>
Andy Hungd65f1d82023-08-28 19:12:14 -0700126 activePatchStreamHandle_l(audio_io_handle_t streamHandle) REQUIRES(mutex());
Vlad Popa7e81cea2023-01-19 16:34:16 +0100127
Andy Hungd65f1d82023-08-28 19:12:14 -0700128 bool useHalSoundDoseInterface_l() REQUIRES(mutex());
Vlad Popa1d5f0d52022-12-18 12:21:26 +0100129
Andy Hungf9e248b2023-07-17 14:02:52 -0700130 const sp<IAfMelReporterCallback> mAfMelReporterCallback;
Vlad Popa1d5f0d52022-12-18 12:21:26 +0100131
Andy Hungd65f1d82023-08-28 19:12:14 -0700132 /* const */ sp<SoundDoseManager> mSoundDoseManager; // set onFirstRef
Vlad Popab042ee62022-10-20 18:05:00 +0200133
Vlad Popab042ee62022-10-20 18:05:00 +0200134 /**
135 * Lock for protecting the active mel patches. Do not mix with the AudioFlinger lock.
Andy Hungd65f1d82023-08-28 19:12:14 -0700136 * Locking order AudioFlinger::mutex() -> PatchCommandThread::mutex() -> MelReporter::mutex().
Vlad Popab042ee62022-10-20 18:05:00 +0200137 */
Andy Hungd65f1d82023-08-28 19:12:14 -0700138 mutable audio_utils::mutex mMutex;
139 std::unordered_map<audio_patch_handle_t, ActiveMelPatch> mActiveMelPatches
140 GUARDED_BY(mutex());
141 std::unordered_map<audio_port_handle_t, int> mActiveDevices GUARDED_BY(mutex());
142 bool mUseHalSoundDoseInterface GUARDED_BY(mutex()) = false;
Vlad Popab042ee62022-10-20 18:05:00 +0200143};
Andy Hung81ce7602023-07-13 20:00:50 -0700144
145} // namespace android