978ccd40f9
On platform release T, the legacy `audio.a2dp.default` HAL no longer exists, and cannot be trivially restored for generic system-side A2DP audio support. Rather than trying to cling on to its existence (which does not even work anymore even if one did so), it is time to come up with a new solution. This commit introduces a system-side implementation of the generic bluetooth audio HAL. The HAL itself is modelled after the default bluetooth audio HAL, while substantial components of the default audio HAL are also included as it has to also implement the audio HAL interfaces. The audio HAL implementation is delegated to `audio.sysbta.default`, forked from `audio.bluetooth.default` from the Bluetooth apex package. It then communicates with the bluetooth audio HAL interfaces, which need to live in the same process (in this case, the process of the sysbta HAL). This is why we cannot just load `audio.sysbta.default` into the default audio HAL or the audioserver process directly, but rather have to include our own simplistic audio HAL implementation. For now, the audio HAL implementation only includes one for the 6.0 version, but technically, an interface for *every* audio HAL version needs to exist. This, along with other messiness in the sysbta implementation, will be addressed in a future commit. Note that to actually make use of the sysbta audio hal, patches in frameworks/av and packages/modules/Bluetooth are required to introduce support for the property `persist.bluetooth.system_audio_hal.enabled`.
192 lines
7.7 KiB
C++
192 lines
7.7 KiB
C++
/*
|
|
* Copyright 2018 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <mutex>
|
|
#include <unordered_map>
|
|
|
|
#include <android/hardware/bluetooth/audio/2.0/IBluetoothAudioPort.h>
|
|
#include <fmq/MessageQueue.h>
|
|
#include <hardware/audio.h>
|
|
#include <hidl/MQDescriptor.h>
|
|
|
|
namespace android {
|
|
namespace bluetooth {
|
|
namespace audio {
|
|
|
|
using ::android::sp;
|
|
using ::android::hardware::kSynchronizedReadWrite;
|
|
using ::android::hardware::MessageQueue;
|
|
using ::android::hardware::bluetooth::audio::V2_0::AudioConfiguration;
|
|
using ::android::hardware::bluetooth::audio::V2_0::BitsPerSample;
|
|
using ::android::hardware::bluetooth::audio::V2_0::ChannelMode;
|
|
using ::android::hardware::bluetooth::audio::V2_0::CodecConfiguration;
|
|
using ::android::hardware::bluetooth::audio::V2_0::IBluetoothAudioPort;
|
|
using ::android::hardware::bluetooth::audio::V2_0::PcmParameters;
|
|
using ::android::hardware::bluetooth::audio::V2_0::SampleRate;
|
|
using ::android::hardware::bluetooth::audio::V2_0::SessionType;
|
|
|
|
using BluetoothAudioStatus =
|
|
::android::hardware::bluetooth::audio::V2_0::Status;
|
|
|
|
using DataMQ = MessageQueue<uint8_t, kSynchronizedReadWrite>;
|
|
|
|
static constexpr uint16_t kObserversCookieSize = 0x0010; // 0x0000 ~ 0x000f
|
|
constexpr uint16_t kObserversCookieUndefined =
|
|
(static_cast<uint16_t>(SessionType::UNKNOWN) << 8 & 0xff00);
|
|
inline SessionType ObserversCookieGetSessionType(uint16_t cookie) {
|
|
return static_cast<SessionType>(cookie >> 8 & 0x00ff);
|
|
}
|
|
inline uint16_t ObserversCookieGetInitValue(SessionType session_type) {
|
|
return (static_cast<uint16_t>(session_type) << 8 & 0xff00);
|
|
}
|
|
inline uint16_t ObserversCookieGetUpperBound(SessionType session_type) {
|
|
return (static_cast<uint16_t>(session_type) << 8 & 0xff00) +
|
|
kObserversCookieSize;
|
|
}
|
|
|
|
// This presents the callbacks of started / suspended and session changed,
|
|
// and the bluetooth_audio module uses to receive the status notification
|
|
struct PortStatusCallbacks {
|
|
// control_result_cb_ - when the Bluetooth stack reports results of
|
|
// streamStarted or streamSuspended, the BluetoothAudioProvider will invoke
|
|
// this callback to report to the bluetooth_audio module.
|
|
// @param: cookie - indicates which bluetooth_audio output should handle
|
|
// @param: start_resp - this report is for startStream or not
|
|
// @param: status - the result of startStream
|
|
std::function<void(uint16_t cookie, bool start_resp,
|
|
const BluetoothAudioStatus& status)>
|
|
control_result_cb_;
|
|
// session_changed_cb_ - when the Bluetooth stack start / end session, the
|
|
// BluetoothAudioProvider will invoke this callback to notify to the
|
|
// bluetooth_audio module.
|
|
// @param: cookie - indicates which bluetooth_audio output should handle
|
|
std::function<void(uint16_t cookie)> session_changed_cb_;
|
|
};
|
|
|
|
class BluetoothAudioSession {
|
|
friend class BluetoothAudioSession_2_1;
|
|
friend class BluetoothAudioSession_2_2;
|
|
|
|
private:
|
|
// using recursive_mutex to allow hwbinder to re-enter agian.
|
|
std::recursive_mutex mutex_;
|
|
SessionType session_type_;
|
|
|
|
// audio control path to use for both software and offloading
|
|
sp<IBluetoothAudioPort> stack_iface_;
|
|
// audio data path (FMQ) for software encoding
|
|
std::unique_ptr<DataMQ> mDataMQ;
|
|
// audio data configuration for both software and offloading
|
|
AudioConfiguration audio_config_;
|
|
|
|
static AudioConfiguration invalidSoftwareAudioConfiguration;
|
|
static AudioConfiguration invalidOffloadAudioConfiguration;
|
|
|
|
// saving those registered bluetooth_audio's callbacks
|
|
std::unordered_map<uint16_t, std::shared_ptr<struct PortStatusCallbacks>>
|
|
observers_;
|
|
|
|
bool UpdateDataPath(const DataMQ::Descriptor* dataMQ);
|
|
bool UpdateAudioConfig(const AudioConfiguration& audio_config);
|
|
// invoking the registered session_changed_cb_
|
|
void ReportSessionStatus();
|
|
|
|
public:
|
|
BluetoothAudioSession(const SessionType& session_type);
|
|
|
|
// The function helps to check if this session is ready or not
|
|
// @return: true if the Bluetooth stack has started the specified session
|
|
bool IsSessionReady();
|
|
|
|
// The report function is used to report that the Bluetooth stack has started
|
|
// this session without any failure, and will invoke session_changed_cb_ to
|
|
// notify those registered bluetooth_audio outputs
|
|
void OnSessionStarted(const sp<IBluetoothAudioPort> stack_iface,
|
|
const DataMQ::Descriptor* dataMQ,
|
|
const AudioConfiguration& audio_config);
|
|
|
|
// The report function is used to report that the Bluetooth stack has ended
|
|
// the session, and will invoke session_changed_cb_ to notify registered
|
|
// bluetooth_audio outputs
|
|
void OnSessionEnded();
|
|
|
|
// The report function is used to report that the Bluetooth stack has notified
|
|
// the result of startStream or suspendStream, and will invoke
|
|
// control_result_cb_ to notify registered bluetooth_audio outputs
|
|
void ReportControlStatus(bool start_resp, const BluetoothAudioStatus& status);
|
|
|
|
// The control function helps the bluetooth_audio module to register
|
|
// PortStatusCallbacks
|
|
// @return: cookie - the assigned number to this bluetooth_audio output
|
|
uint16_t RegisterStatusCback(const PortStatusCallbacks& cbacks);
|
|
|
|
// The control function helps the bluetooth_audio module to unregister
|
|
// PortStatusCallbacks
|
|
// @param: cookie - indicates which bluetooth_audio output is
|
|
void UnregisterStatusCback(uint16_t cookie);
|
|
|
|
// The control function is for the bluetooth_audio module to get the current
|
|
// AudioConfiguration
|
|
const AudioConfiguration& GetAudioConfig();
|
|
|
|
// Those control functions are for the bluetooth_audio module to start,
|
|
// suspend, stop stream, to check position, and to update metadata.
|
|
bool StartStream();
|
|
bool SuspendStream();
|
|
void StopStream();
|
|
bool GetPresentationPosition(uint64_t* remote_delay_report_ns,
|
|
uint64_t* total_bytes_readed,
|
|
timespec* data_position);
|
|
void UpdateTracksMetadata(const struct source_metadata* source_metadata);
|
|
|
|
// The control function writes stream to FMQ
|
|
size_t OutWritePcmData(const void* buffer, size_t bytes);
|
|
// The control function read stream from FMQ
|
|
size_t InReadPcmData(void* buffer, size_t bytes);
|
|
|
|
static constexpr PcmParameters kInvalidPcmParameters = {
|
|
.sampleRate = SampleRate::RATE_UNKNOWN,
|
|
.channelMode = ChannelMode::UNKNOWN,
|
|
.bitsPerSample = BitsPerSample::BITS_UNKNOWN,
|
|
};
|
|
// can't be constexpr because of non-literal type
|
|
static const CodecConfiguration kInvalidCodecConfiguration;
|
|
|
|
static constexpr AudioConfiguration& kInvalidSoftwareAudioConfiguration =
|
|
invalidSoftwareAudioConfiguration;
|
|
static constexpr AudioConfiguration& kInvalidOffloadAudioConfiguration =
|
|
invalidOffloadAudioConfiguration;
|
|
};
|
|
|
|
class BluetoothAudioSessionInstance {
|
|
public:
|
|
// The API is to fetch the specified session of A2DP / Hearing Aid
|
|
static std::shared_ptr<BluetoothAudioSession> GetSessionInstance(
|
|
const SessionType& session_type);
|
|
|
|
private:
|
|
static std::unique_ptr<BluetoothAudioSessionInstance> instance_ptr;
|
|
std::mutex mutex_;
|
|
std::unordered_map<SessionType, std::shared_ptr<BluetoothAudioSession>>
|
|
sessions_map_;
|
|
};
|
|
|
|
} // namespace audio
|
|
} // namespace bluetooth
|
|
} // namespace android
|