blob: 0dbb5f882411455cd3f7056f1361ff176cf67a48 [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 Hungad2faf72023-07-13 20:00:50 -070018#pragma once
Vlad Popab042ee62022-10-20 18:05:00 +020019
Vlad Popab042ee62022-10-20 18:05:00 +020020#include <mutex>
Vlad Popa2900c0a2022-10-24 13:38:00 +020021#include <sounddose/SoundDoseManager.h>
22#include <unordered_map>
Vlad Popab042ee62022-10-20 18:05:00 +020023
Andy Hungad2faf72023-07-13 20:00:50 -070024namespace android {
25
Vlad Popab042ee62022-10-20 18:05:00 +020026constexpr static int kMaxTimestampDeltaInSec = 120;
27
Andy Hungb60a2c82023-07-17 14:02:52 -070028class IAfMelReporterCallback : public virtual RefBase {
29public:
30 virtual Mutex& mutex() const = 0;
31 virtual const sp<PatchCommandThread>& getPatchCommandThread() = 0;
32 virtual sp<IAfThreadBase> checkOutputThread_l(audio_io_handle_t ioHandle) const = 0;
33};
34
Vlad Popab042ee62022-10-20 18:05:00 +020035/**
36 * Class for listening to new patches and starting the MEL computation. MelReporter is
37 * concealed within AudioFlinger, their lifetimes are the same.
38 */
39class MelReporter : public PatchCommandThread::PatchCommandListener {
40public:
Andy Hungb60a2c82023-07-17 14:02:52 -070041 explicit MelReporter(const sp<IAfMelReporterCallback>& afMelReporterCallback)
42 : mAfMelReporterCallback(afMelReporterCallback),
Vlad Popa1d5f0d52022-12-18 12:21:26 +010043 mSoundDoseManager(sp<SoundDoseManager>::make()) {}
Vlad Popab042ee62022-10-20 18:05:00 +020044
Vlad Popa1d5f0d52022-12-18 12:21:26 +010045 void onFirstRef() override;
Vlad Popab042ee62022-10-20 18:05:00 +020046
Vlad Popa1d5f0d52022-12-18 12:21:26 +010047 /**
48 * Activates the MEL reporting from the HAL sound dose interface. If the HAL
49 * does not support the sound dose interface for this module, the internal MEL
50 * calculation will be use.
51 *
Vlad Popa03bd5bc2023-01-17 16:16:51 +010052 * <p>If the device is using the audio AIDL HAL then this method will try to get the sound
53 * dose interface from IModule#getSoundDose(). Otherwise, if the legacy audio HIDL HAL is used
54 * this method will be looking for the standalone sound dose implementation. It falls back to
55 * the internal MEL computation if no valid sound dose interface can be retrieved.
Vlad Popa1d5f0d52022-12-18 12:21:26 +010056 *
Vlad Popa03bd5bc2023-01-17 16:16:51 +010057 * @return true if the MEL reporting will be done from any sound dose HAL interface
58 * implementation, false otherwise.
Vlad Popa1d5f0d52022-12-18 12:21:26 +010059 */
Vlad Popa03bd5bc2023-01-17 16:16:51 +010060 bool activateHalSoundDoseComputation(const std::string& module,
61 const sp<DeviceHalInterface>& device);
Vlad Popa1d5f0d52022-12-18 12:21:26 +010062
63 /**
64 * Activates the MEL reporting from internal framework values. These are used
65 * as a fallback when there is no sound dose interface implementation from HAL.
66 * Note: the internal CSD computation does not guarantee a certification with
67 * IEC62368-1 3rd edition or EN50332-3
68 */
69 void activateInternalSoundDoseComputation();
Vlad Popab042ee62022-10-20 18:05:00 +020070
Vlad Popae3fd1c22022-11-07 21:03:18 +010071 sp<media::ISoundDose> getSoundDoseInterface(const sp<media::ISoundDoseCallback>& callback);
Vlad Popa63f047e2022-11-05 14:09:19 +010072
Vlad Popab042ee62022-10-20 18:05:00 +020073 std::string dump();
74
75 // PatchCommandListener methods
76 void onCreateAudioPatch(audio_patch_handle_t handle,
Andy Hung8e6b62a2023-07-13 18:11:33 -070077 const IAfPatchPanel::Patch& patch) final;
78 void onReleaseAudioPatch(audio_patch_handle_t handle) final;
François Gaffie58e73af2023-02-15 11:47:24 +010079 void onUpdateAudioPatch(audio_patch_handle_t oldHandle,
80 audio_patch_handle_t newHandle,
Andy Hung8e6b62a2023-07-13 18:11:33 -070081 const IAfPatchPanel::Patch& patch) final;
Vlad Popab042ee62022-10-20 18:05:00 +020082
Vlad Popa7e81cea2023-01-19 16:34:16 +010083 /**
84 * The new metadata can determine whether we should compute MEL for the given thread.
85 * This is the case only if one of the tracks in the thread mix is using MEDIA or GAME.
86 * Otherwise, this method will disable CSD.
87 **/
88 void updateMetadataForCsd(audio_io_handle_t streamHandle,
89 const std::vector<playback_track_metadata_v7_t>& metadataVec);
Vlad Popab042ee62022-10-20 18:05:00 +020090private:
Vlad Popa7e81cea2023-01-19 16:34:16 +010091 struct ActiveMelPatch {
92 audio_io_handle_t streamHandle{AUDIO_IO_HANDLE_NONE};
93 std::vector<audio_port_handle_t> deviceHandles;
94 bool csdActive;
95 };
Vlad Popab042ee62022-10-20 18:05:00 +020096
Vlad Popa7e81cea2023-01-19 16:34:16 +010097 /** Returns true if we should compute MEL for the given device. */
98 bool shouldComputeMelForDeviceType(audio_devices_t device);
99
100 void stopInternalMelComputation();
101
102 /** Should be called with the following order of locks: mAudioFlinger.mLock -> mLock. */
Andy Hung920f6572022-10-06 12:09:49 -0700103 void stopMelComputationForPatch_l(const ActiveMelPatch& patch) REQUIRES(mLock);
Vlad Popa7e81cea2023-01-19 16:34:16 +0100104
105 /** Should be called with the following order of locks: mAudioFlinger.mLock -> mLock. */
Andy Hung920f6572022-10-06 12:09:49 -0700106 void startMelComputationForActivePatch_l(const ActiveMelPatch& patch) REQUIRES(mLock);
Vlad Popa7e81cea2023-01-19 16:34:16 +0100107
Andy Hung920f6572022-10-06 12:09:49 -0700108 std::optional<audio_patch_handle_t>
109 activePatchStreamHandle_l(audio_io_handle_t streamHandle) REQUIRES(mLock);
Vlad Popa7e81cea2023-01-19 16:34:16 +0100110
Vlad Popa69fbbee2023-04-25 12:23:24 +0200111 bool useHalSoundDoseInterface_l() REQUIRES(mLock);
Vlad Popa1d5f0d52022-12-18 12:21:26 +0100112
Andy Hungb60a2c82023-07-17 14:02:52 -0700113 const sp<IAfMelReporterCallback> mAfMelReporterCallback;
Vlad Popa1d5f0d52022-12-18 12:21:26 +0100114
115 sp<SoundDoseManager> mSoundDoseManager;
Vlad Popab042ee62022-10-20 18:05:00 +0200116
Vlad Popab042ee62022-10-20 18:05:00 +0200117 /**
118 * Lock for protecting the active mel patches. Do not mix with the AudioFlinger lock.
119 * Locking order AudioFlinger::mLock -> PatchCommandThread::mLock -> MelReporter::mLock.
120 */
121 std::mutex mLock;
Andy Hungad2faf72023-07-13 20:00:50 -0700122 std::unordered_map<audio_patch_handle_t, ActiveMelPatch> mActiveMelPatches GUARDED_BY(mLock);
123 std::unordered_map<audio_port_handle_t, int> mActiveDevices GUARDED_BY(mLock);
124 bool mUseHalSoundDoseInterface GUARDED_BY(mLock) = false;
Vlad Popab042ee62022-10-20 18:05:00 +0200125};
Andy Hungad2faf72023-07-13 20:00:50 -0700126
127} // namespace android