/*
* Copyright 2021 HIMSA II K/S - www.himsa.com.
* Represented by EHIMA - www.ehima.com
*
* 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.
*/
/* Volume Control Interface */
#include
#include
#include
#include
#include
#include "bta_vc_api.h"
#include "btif_common.h"
#include "btif_profile_storage.h"
#include "stack/include/main_thread.h"
#include "types/raw_address.h"
using base::Bind;
using base::Unretained;
using bluetooth::vc::ConnectionState;
using bluetooth::vc::VolumeControlCallbacks;
using bluetooth::vc::VolumeControlInterface;
using namespace bluetooth;
namespace {
std::unique_ptr vc_instance;
std::atomic_bool initialized = false;
class VolumeControlInterfaceImpl : public VolumeControlInterface,
public VolumeControlCallbacks {
~VolumeControlInterfaceImpl() override = default;
void Init(VolumeControlCallbacks* callbacks) override {
this->callbacks_ = callbacks;
do_in_main_thread(
FROM_HERE,
Bind(&VolumeControl::Initialize, this,
jni_thread_wrapper(
Bind(&btif_storage_load_bonded_volume_control_devices))));
/* It might be not yet initialized, but setting this flag here is safe,
* because other calls will check this and the native instance
*/
initialized = true;
}
void OnConnectionState(ConnectionState state,
const RawAddress& address) override {
do_in_jni_thread(Bind(&VolumeControlCallbacks::OnConnectionState,
Unretained(callbacks_), state, address));
}
void OnVolumeStateChanged(const RawAddress& address, uint8_t volume,
bool mute, bool isAutonomous) override {
do_in_jni_thread(Bind(&VolumeControlCallbacks::OnVolumeStateChanged,
Unretained(callbacks_), address, volume, mute,
isAutonomous));
}
void OnGroupVolumeStateChanged(int group_id, uint8_t volume, bool mute,
bool isAutonomous) override {
do_in_jni_thread(Bind(&VolumeControlCallbacks::OnGroupVolumeStateChanged,
Unretained(callbacks_), group_id, volume, mute,
isAutonomous));
}
void OnDeviceAvailable(const RawAddress& address,
uint8_t num_offset) override {
do_in_jni_thread(Bind(&VolumeControlCallbacks::OnDeviceAvailable,
Unretained(callbacks_), address, num_offset));
}
/* Callbacks for Volume Offset Control Service (VOCS) - Extended Audio Outputs
*/
void OnExtAudioOutVolumeOffsetChanged(const RawAddress& address,
uint8_t ext_output_id,
int16_t offset) override {
do_in_jni_thread(
Bind(&VolumeControlCallbacks::OnExtAudioOutVolumeOffsetChanged,
Unretained(callbacks_), address, ext_output_id, offset));
}
void OnExtAudioOutLocationChanged(const RawAddress& address,
uint8_t ext_output_id,
uint32_t location) override {
do_in_jni_thread(Bind(&VolumeControlCallbacks::OnExtAudioOutLocationChanged,
Unretained(callbacks_), address, ext_output_id,
location));
}
void OnExtAudioOutDescriptionChanged(const RawAddress& address,
uint8_t ext_output_id,
std::string descr) override {
do_in_jni_thread(
Bind(&VolumeControlCallbacks::OnExtAudioOutDescriptionChanged,
Unretained(callbacks_), address, ext_output_id, descr));
}
void Connect(const RawAddress& address) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
do_in_main_thread(FROM_HERE,
Bind(&VolumeControl::Connect,
Unretained(VolumeControl::Get()), address));
}
void Disconnect(const RawAddress& address) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
do_in_main_thread(FROM_HERE,
Bind(&VolumeControl::Disconnect,
Unretained(VolumeControl::Get()), address));
}
void SetVolume(std::variant addr_or_group_id,
uint8_t volume) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
do_in_main_thread(FROM_HERE, Bind(&VolumeControl::SetVolume,
Unretained(VolumeControl::Get()),
std::move(addr_or_group_id), volume));
}
void Mute(std::variant addr_or_group_id) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
do_in_main_thread(
FROM_HERE, Bind(&VolumeControl::Mute, Unretained(VolumeControl::Get()),
std::move(addr_or_group_id)));
}
void Unmute(std::variant addr_or_group_id) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
do_in_main_thread(FROM_HERE, Bind(&VolumeControl::UnMute,
Unretained(VolumeControl::Get()),
std::move(addr_or_group_id)));
}
void RemoveDevice(const RawAddress& address) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
/* RemoveDevice can be called on devices that don't have HA enabled */
if (VolumeControl::IsVolumeControlRunning()) {
do_in_main_thread(FROM_HERE,
Bind(&VolumeControl::Remove,
Unretained(VolumeControl::Get()), address));
}
}
void GetExtAudioOutVolumeOffset(const RawAddress& address,
uint8_t ext_output_id) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
do_in_main_thread(
FROM_HERE,
Bind(&VolumeControl::GetExtAudioOutVolumeOffset,
Unretained(VolumeControl::Get()), address, ext_output_id));
}
void SetExtAudioOutVolumeOffset(const RawAddress& address,
uint8_t ext_output_id,
int16_t offset_val) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
do_in_main_thread(FROM_HERE,
Bind(&VolumeControl::SetExtAudioOutVolumeOffset,
Unretained(VolumeControl::Get()), address,
ext_output_id, offset_val));
}
void GetExtAudioOutLocation(const RawAddress& address,
uint8_t ext_output_id) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
do_in_main_thread(FROM_HERE, Bind(&VolumeControl::GetExtAudioOutLocation,
Unretained(VolumeControl::Get()), address,
ext_output_id));
}
void SetExtAudioOutLocation(const RawAddress& address, uint8_t ext_output_id,
uint32_t location) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
do_in_main_thread(FROM_HERE, Bind(&VolumeControl::SetExtAudioOutLocation,
Unretained(VolumeControl::Get()), address,
ext_output_id, location));
}
void GetExtAudioOutDescription(const RawAddress& address,
uint8_t ext_output_id) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
do_in_main_thread(FROM_HERE, Bind(&VolumeControl::GetExtAudioOutDescription,
Unretained(VolumeControl::Get()), address,
ext_output_id));
}
void SetExtAudioOutDescription(const RawAddress& address,
uint8_t ext_output_id,
std::string descr) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
do_in_main_thread(FROM_HERE, Bind(&VolumeControl::SetExtAudioOutDescription,
Unretained(VolumeControl::Get()), address,
ext_output_id, descr));
}
void Cleanup(void) override {
if (!initialized || !VolumeControl::IsVolumeControlRunning()) {
log::verbose(
"call ignored, due to already started cleanup procedure or service "
"being not read");
return;
}
initialized = false;
do_in_main_thread(FROM_HERE, Bind(&VolumeControl::CleanUp));
}
private:
VolumeControlCallbacks* callbacks_;
};
} /* namespace */
VolumeControlInterface* btif_volume_control_get_interface(void) {
if (!vc_instance) vc_instance.reset(new VolumeControlInterfaceImpl());
return vc_instance.get();
}