/* * Copyright (C) 2012 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. */ #define LOG_TAG "BluetoothA2dpServiceJni" #include #include #include "btif/include/btif_util.h" #include "com_android_bluetooth.h" #include "hardware/bt_av.h" namespace android { static jmethodID method_onConnectionStateChanged; static jmethodID method_onAudioStateChanged; static jmethodID method_onCodecConfigChanged; static jmethodID method_isMandatoryCodecPreferred; static struct { jclass clazz; jmethodID constructor; jmethodID getCodecType; jmethodID getCodecPriority; jmethodID getSampleRate; jmethodID getBitsPerSample; jmethodID getChannelMode; jmethodID getCodecSpecific1; jmethodID getCodecSpecific2; jmethodID getCodecSpecific3; jmethodID getCodecSpecific4; } android_bluetooth_BluetoothCodecConfig; static const btav_source_interface_t* sBluetoothA2dpInterface = nullptr; static std::vector supported_codecs; static std::shared_timed_mutex interface_mutex; static jobject mCallbacksObj = nullptr; static std::shared_timed_mutex callbacks_mutex; static void bta2dp_connection_state_callback(const RawAddress& bd_addr, btav_connection_state_t state, const btav_error_t& /* error */) { log::info("{}: state: {}", bd_addr, dump_av_conn_state(state)); std::shared_lock lock(callbacks_mutex); CallbackEnv sCallbackEnv(__func__); if (!sCallbackEnv.valid() || mCallbacksObj == nullptr) return; ScopedLocalRef addr( sCallbackEnv.get(), sCallbackEnv->NewByteArray(sizeof(RawAddress))); if (!addr.get()) { log::error("Fail to new jbyteArray bd addr"); return; } sCallbackEnv->SetByteArrayRegion( addr.get(), 0, sizeof(RawAddress), reinterpret_cast(bd_addr.address)); sCallbackEnv->CallVoidMethod(mCallbacksObj, method_onConnectionStateChanged, addr.get(), (jint)state); } static void bta2dp_audio_state_callback(const RawAddress& bd_addr, btav_audio_state_t state) { log::info("{}: state: {}", bd_addr, dump_av_audio_state(state)); std::shared_lock lock(callbacks_mutex); CallbackEnv sCallbackEnv(__func__); if (!sCallbackEnv.valid() || mCallbacksObj == nullptr) return; ScopedLocalRef addr( sCallbackEnv.get(), sCallbackEnv->NewByteArray(sizeof(RawAddress))); if (!addr.get()) { log::error("Fail to new jbyteArray bd addr"); return; } sCallbackEnv->SetByteArrayRegion( addr.get(), 0, sizeof(RawAddress), reinterpret_cast(bd_addr.address)); sCallbackEnv->CallVoidMethod(mCallbacksObj, method_onAudioStateChanged, addr.get(), (jint)state); } static void bta2dp_audio_config_callback( const RawAddress& bd_addr, btav_a2dp_codec_config_t codec_config, std::vector codecs_local_capabilities, std::vector codecs_selectable_capabilities) { log::info( "{}: codec: {}, local codecs: {}, selectable codecs: {}", bd_addr, codec_config.CodecNameStr(), btav_a2dp_codec_config_t::PrintCodecs(codecs_local_capabilities), btav_a2dp_codec_config_t::PrintCodecs(codecs_selectable_capabilities)); std::shared_lock lock(callbacks_mutex); CallbackEnv sCallbackEnv(__func__); if (!sCallbackEnv.valid() || mCallbacksObj == nullptr) return; jobject codecConfigObj = sCallbackEnv->NewObject( android_bluetooth_BluetoothCodecConfig.clazz, android_bluetooth_BluetoothCodecConfig.constructor, (jint)codec_config.codec_type, (jint)codec_config.codec_priority, (jint)codec_config.sample_rate, (jint)codec_config.bits_per_sample, (jint)codec_config.channel_mode, (jlong)codec_config.codec_specific_1, (jlong)codec_config.codec_specific_2, (jlong)codec_config.codec_specific_3, (jlong)codec_config.codec_specific_4); jsize i = 0; jobjectArray local_capabilities_array = sCallbackEnv->NewObjectArray( (jsize)codecs_local_capabilities.size(), android_bluetooth_BluetoothCodecConfig.clazz, nullptr); for (auto const& cap : codecs_local_capabilities) { jobject capObj = sCallbackEnv->NewObject( android_bluetooth_BluetoothCodecConfig.clazz, android_bluetooth_BluetoothCodecConfig.constructor, (jint)cap.codec_type, (jint)cap.codec_priority, (jint)cap.sample_rate, (jint)cap.bits_per_sample, (jint)cap.channel_mode, (jlong)cap.codec_specific_1, (jlong)cap.codec_specific_2, (jlong)cap.codec_specific_3, (jlong)cap.codec_specific_4); sCallbackEnv->SetObjectArrayElement(local_capabilities_array, i++, capObj); sCallbackEnv->DeleteLocalRef(capObj); } i = 0; jobjectArray selectable_capabilities_array = sCallbackEnv->NewObjectArray( (jsize)codecs_selectable_capabilities.size(), android_bluetooth_BluetoothCodecConfig.clazz, nullptr); for (auto const& cap : codecs_selectable_capabilities) { jobject capObj = sCallbackEnv->NewObject( android_bluetooth_BluetoothCodecConfig.clazz, android_bluetooth_BluetoothCodecConfig.constructor, (jint)cap.codec_type, (jint)cap.codec_priority, (jint)cap.sample_rate, (jint)cap.bits_per_sample, (jint)cap.channel_mode, (jlong)cap.codec_specific_1, (jlong)cap.codec_specific_2, (jlong)cap.codec_specific_3, (jlong)cap.codec_specific_4); sCallbackEnv->SetObjectArrayElement(selectable_capabilities_array, i++, capObj); sCallbackEnv->DeleteLocalRef(capObj); } ScopedLocalRef addr( sCallbackEnv.get(), sCallbackEnv->NewByteArray(RawAddress::kLength)); if (!addr.get()) { log::error("Fail to new jbyteArray bd addr"); return; } sCallbackEnv->SetByteArrayRegion( addr.get(), 0, RawAddress::kLength, reinterpret_cast(bd_addr.address)); sCallbackEnv->CallVoidMethod( mCallbacksObj, method_onCodecConfigChanged, addr.get(), codecConfigObj, local_capabilities_array, selectable_capabilities_array); } static bool bta2dp_mandatory_codec_preferred_callback( const RawAddress& bd_addr) { log::info("{}", bd_addr); std::shared_lock lock(callbacks_mutex); CallbackEnv sCallbackEnv(__func__); if (!sCallbackEnv.valid() || mCallbacksObj == nullptr) return false; ScopedLocalRef addr( sCallbackEnv.get(), sCallbackEnv->NewByteArray(RawAddress::kLength)); if (!addr.get()) { log::error("Fail to new jbyteArray bd addr"); return false; } sCallbackEnv->SetByteArrayRegion( addr.get(), 0, RawAddress::kLength, reinterpret_cast(bd_addr.address)); return sCallbackEnv->CallBooleanMethod( mCallbacksObj, method_isMandatoryCodecPreferred, addr.get()); } static btav_source_callbacks_t sBluetoothA2dpCallbacks = { sizeof(sBluetoothA2dpCallbacks), bta2dp_connection_state_callback, bta2dp_audio_state_callback, bta2dp_audio_config_callback, bta2dp_mandatory_codec_preferred_callback, }; static std::vector prepareCodecPreferences( JNIEnv* env, jobject /* object */, jobjectArray codecConfigArray) { std::vector codec_preferences; int numConfigs = env->GetArrayLength(codecConfigArray); for (int i = 0; i < numConfigs; i++) { jobject jcodecConfig = env->GetObjectArrayElement(codecConfigArray, i); if (jcodecConfig == nullptr) continue; if (!env->IsInstanceOf(jcodecConfig, android_bluetooth_BluetoothCodecConfig.clazz)) { log::error("Invalid BluetoothCodecConfig instance"); continue; } jint codecType = env->CallIntMethod( jcodecConfig, android_bluetooth_BluetoothCodecConfig.getCodecType); jint codecPriority = env->CallIntMethod( jcodecConfig, android_bluetooth_BluetoothCodecConfig.getCodecPriority); jint sampleRate = env->CallIntMethod( jcodecConfig, android_bluetooth_BluetoothCodecConfig.getSampleRate); jint bitsPerSample = env->CallIntMethod( jcodecConfig, android_bluetooth_BluetoothCodecConfig.getBitsPerSample); jint channelMode = env->CallIntMethod( jcodecConfig, android_bluetooth_BluetoothCodecConfig.getChannelMode); jlong codecSpecific1 = env->CallLongMethod( jcodecConfig, android_bluetooth_BluetoothCodecConfig.getCodecSpecific1); jlong codecSpecific2 = env->CallLongMethod( jcodecConfig, android_bluetooth_BluetoothCodecConfig.getCodecSpecific2); jlong codecSpecific3 = env->CallLongMethod( jcodecConfig, android_bluetooth_BluetoothCodecConfig.getCodecSpecific3); jlong codecSpecific4 = env->CallLongMethod( jcodecConfig, android_bluetooth_BluetoothCodecConfig.getCodecSpecific4); btav_a2dp_codec_config_t codec_config = { .codec_type = static_cast(codecType), .codec_priority = static_cast(codecPriority), .sample_rate = static_cast(sampleRate), .bits_per_sample = static_cast(bitsPerSample), .channel_mode = static_cast(channelMode), .codec_specific_1 = codecSpecific1, .codec_specific_2 = codecSpecific2, .codec_specific_3 = codecSpecific3, .codec_specific_4 = codecSpecific4}; codec_preferences.push_back(codec_config); } return codec_preferences; } static void initNative(JNIEnv* env, jobject object, jint maxConnectedAudioDevices, jobjectArray codecConfigArray, jobjectArray codecOffloadingArray) { std::unique_lock interface_lock(interface_mutex); std::unique_lock callbacks_lock(callbacks_mutex); const bt_interface_t* btInf = getBluetoothInterface(); if (btInf == nullptr) { log::error("Bluetooth module is not loaded"); return; } if (sBluetoothA2dpInterface != nullptr) { log::warn("Cleaning up A2DP Interface before initializing..."); sBluetoothA2dpInterface->cleanup(); sBluetoothA2dpInterface = nullptr; } if (mCallbacksObj != nullptr) { log::warn("Cleaning up A2DP callback object"); env->DeleteGlobalRef(mCallbacksObj); mCallbacksObj = nullptr; } if ((mCallbacksObj = env->NewGlobalRef(object)) == nullptr) { log::error("Failed to allocate Global Ref for A2DP Callbacks"); return; } android_bluetooth_BluetoothCodecConfig.clazz = (jclass)env->NewGlobalRef( env->FindClass("android/bluetooth/BluetoothCodecConfig")); if (android_bluetooth_BluetoothCodecConfig.clazz == nullptr) { log::error("Failed to allocate Global Ref for BluetoothCodecConfig class"); return; } sBluetoothA2dpInterface = (btav_source_interface_t*)btInf->get_profile_interface( BT_PROFILE_ADVANCED_AUDIO_ID); if (sBluetoothA2dpInterface == nullptr) { log::error("Failed to get Bluetooth A2DP Interface"); return; } std::vector codec_priorities = prepareCodecPreferences(env, object, codecConfigArray); std::vector codec_offloading = prepareCodecPreferences(env, object, codecOffloadingArray); bt_status_t status = sBluetoothA2dpInterface->init( &sBluetoothA2dpCallbacks, maxConnectedAudioDevices, codec_priorities, codec_offloading, &supported_codecs); if (status != BT_STATUS_SUCCESS) { log::error("Failed to initialize Bluetooth A2DP, status: {}", bt_status_text(status)); sBluetoothA2dpInterface = nullptr; return; } } static void cleanupNative(JNIEnv* env, jobject /* object */) { std::unique_lock interface_lock(interface_mutex); std::unique_lock callbacks_lock(callbacks_mutex); const bt_interface_t* btInf = getBluetoothInterface(); if (btInf == nullptr) { log::error("Bluetooth module is not loaded"); return; } if (sBluetoothA2dpInterface != nullptr) { sBluetoothA2dpInterface->cleanup(); sBluetoothA2dpInterface = nullptr; } env->DeleteGlobalRef(android_bluetooth_BluetoothCodecConfig.clazz); android_bluetooth_BluetoothCodecConfig.clazz = nullptr; if (mCallbacksObj != nullptr) { env->DeleteGlobalRef(mCallbacksObj); mCallbacksObj = nullptr; } } static jobjectArray getSupportedCodecTypesNative(JNIEnv* env) { log::info("{}", fmt::ptr(sBluetoothA2dpInterface)); jclass android_bluetooth_BluetoothCodecType_clazz = (jclass)env->NewGlobalRef( env->FindClass("android/bluetooth/BluetoothCodecType")); if (android_bluetooth_BluetoothCodecType_clazz == nullptr) { log::error("Failed to allocate Global Ref for BluetoothCodecType class"); return nullptr; } jmethodID init = env->GetMethodID(android_bluetooth_BluetoothCodecType_clazz, "", "(IJLjava/lang/String;)V"); if (init == nullptr) { log::error("Failed to find method of BluetoothCodecType class"); return nullptr; } jobjectArray result = env->NewObjectArray(supported_codecs.size(), android_bluetooth_BluetoothCodecType_clazz, nullptr); if (result == nullptr) { log::error("Failed to allocate result array of BluetoothCodecType"); return nullptr; } for (size_t index = 0; index < supported_codecs.size(); index++) { jobject codec_type = env->NewObject( android_bluetooth_BluetoothCodecType_clazz, init, (jint)supported_codecs[index].codec_type, (jlong)supported_codecs[index].codec_id, env->NewStringUTF(supported_codecs[index].codec_name.c_str())); env->SetObjectArrayElement(result, index, codec_type); } return result; } static jboolean connectA2dpNative(JNIEnv* env, jobject /* object */, jbyteArray address) { std::shared_lock lock(interface_mutex); if (!sBluetoothA2dpInterface) { log::error("Failed to get the Bluetooth A2DP Interface"); return JNI_FALSE; } jbyte* addr = env->GetByteArrayElements(address, nullptr); if (!addr) { jniThrowIOException(env, EINVAL); return JNI_FALSE; } RawAddress bd_addr; bd_addr.FromOctets(reinterpret_cast(addr)); log::info("sBluetoothA2dpInterface: {}, {}", fmt::ptr(sBluetoothA2dpInterface), bd_addr); bt_status_t status = sBluetoothA2dpInterface->connect(bd_addr); if (status != BT_STATUS_SUCCESS) { log::error("Failed A2DP connection, status: {}", bt_status_text(status)); } env->ReleaseByteArrayElements(address, addr, 0); return (status == BT_STATUS_SUCCESS) ? JNI_TRUE : JNI_FALSE; } static jboolean disconnectA2dpNative(JNIEnv* env, jobject /* object */, jbyteArray address) { std::shared_lock lock(interface_mutex); if (!sBluetoothA2dpInterface) { log::error("Failed to get the Bluetooth A2DP Interface"); return JNI_FALSE; } jbyte* addr = env->GetByteArrayElements(address, nullptr); if (!addr) { jniThrowIOException(env, EINVAL); return JNI_FALSE; } RawAddress bd_addr; bd_addr.FromOctets(reinterpret_cast(addr)); log::info("sBluetoothA2dpInterface: {}, {}", fmt::ptr(sBluetoothA2dpInterface), bd_addr); bt_status_t status = sBluetoothA2dpInterface->disconnect(bd_addr); if (status != BT_STATUS_SUCCESS) { log::error("Failed A2DP disconnection, status: {}", bt_status_text(status)); } env->ReleaseByteArrayElements(address, addr, 0); return (status == BT_STATUS_SUCCESS) ? JNI_TRUE : JNI_FALSE; } static jboolean setSilenceDeviceNative(JNIEnv* env, jobject /* object */, jbyteArray address, jboolean silence) { std::shared_lock lock(interface_mutex); if (!sBluetoothA2dpInterface) { log::error("Failed to get the Bluetooth A2DP Interface"); return JNI_FALSE; } jbyte* addr = env->GetByteArrayElements(address, nullptr); RawAddress bd_addr = RawAddress::kEmpty; if (addr) { bd_addr.FromOctets(reinterpret_cast(addr)); } if (bd_addr == RawAddress::kEmpty) { return JNI_FALSE; } log::info("sBluetoothA2dpInterface: {}, {}: silence: {}", fmt::ptr(sBluetoothA2dpInterface), bd_addr, static_cast(silence)); bt_status_t status = sBluetoothA2dpInterface->set_silence_device(bd_addr, silence); if (status != BT_STATUS_SUCCESS) { log::error("Failed A2DP set_silence_device, status: {}", bt_status_text(status)); } env->ReleaseByteArrayElements(address, addr, 0); return (status == BT_STATUS_SUCCESS) ? JNI_TRUE : JNI_FALSE; } static jboolean setActiveDeviceNative(JNIEnv* env, jobject /* object */, jbyteArray address) { std::shared_lock lock(interface_mutex); if (!sBluetoothA2dpInterface) { log::error("Failed to get the Bluetooth A2DP Interface"); return JNI_FALSE; } jbyte* addr = env->GetByteArrayElements(address, nullptr); RawAddress bd_addr = RawAddress::kEmpty; if (addr) { bd_addr.FromOctets(reinterpret_cast(addr)); } log::info("sBluetoothA2dpInterface: {}, {}", fmt::ptr(sBluetoothA2dpInterface), bd_addr); bt_status_t status = sBluetoothA2dpInterface->set_active_device(bd_addr); if (status != BT_STATUS_SUCCESS) { log::error("Failed A2DP set_active_device, status: {}", bt_status_text(status)); } env->ReleaseByteArrayElements(address, addr, 0); return (status == BT_STATUS_SUCCESS) ? JNI_TRUE : JNI_FALSE; } static jboolean setCodecConfigPreferenceNative(JNIEnv* env, jobject object, jbyteArray address, jobjectArray codecConfigArray) { std::shared_lock lock(interface_mutex); if (!sBluetoothA2dpInterface) { log::error("Failed to get the Bluetooth A2DP Interface"); return JNI_FALSE; } jbyte* addr = env->GetByteArrayElements(address, nullptr); if (!addr) { jniThrowIOException(env, EINVAL); return JNI_FALSE; } RawAddress bd_addr; bd_addr.FromOctets(reinterpret_cast(addr)); std::vector codec_preferences = prepareCodecPreferences(env, object, codecConfigArray); log::info("sBluetoothA2dpInterface: {}, {}: {}", fmt::ptr(sBluetoothA2dpInterface), bd_addr, btav_a2dp_codec_config_t::PrintCodecs(codec_preferences)); bt_status_t status = sBluetoothA2dpInterface->config_codec(bd_addr, codec_preferences); if (status != BT_STATUS_SUCCESS) { log::error("Failed codec configuration, status: {}", bt_status_text(status)); } env->ReleaseByteArrayElements(address, addr, 0); return (status == BT_STATUS_SUCCESS) ? JNI_TRUE : JNI_FALSE; } int register_com_android_bluetooth_a2dp(JNIEnv* env) { const JNINativeMethod methods[] = { {"initNative", "(I[Landroid/bluetooth/BluetoothCodecConfig;" "[Landroid/bluetooth/BluetoothCodecConfig;)V", (void*)initNative}, {"cleanupNative", "()V", (void*)cleanupNative}, {"getSupportedCodecTypesNative", "()[Landroid/bluetooth/BluetoothCodecType;", (void*)getSupportedCodecTypesNative}, {"connectA2dpNative", "([B)Z", (void*)connectA2dpNative}, {"disconnectA2dpNative", "([B)Z", (void*)disconnectA2dpNative}, {"setSilenceDeviceNative", "([BZ)Z", (void*)setSilenceDeviceNative}, {"setActiveDeviceNative", "([B)Z", (void*)setActiveDeviceNative}, {"setCodecConfigPreferenceNative", "([B[Landroid/bluetooth/BluetoothCodecConfig;)Z", (void*)setCodecConfigPreferenceNative}, }; const int result = REGISTER_NATIVE_METHODS( env, "com/android/bluetooth/a2dp/A2dpNativeInterface", methods); if (result != 0) { return result; } const JNIJavaMethod javaMethods[] = { {"onConnectionStateChanged", "([BI)V", &method_onConnectionStateChanged}, {"onAudioStateChanged", "([BI)V", &method_onAudioStateChanged}, {"onCodecConfigChanged", "([BLandroid/bluetooth/BluetoothCodecConfig;" "[Landroid/bluetooth/BluetoothCodecConfig;" "[Landroid/bluetooth/BluetoothCodecConfig;)V", &method_onCodecConfigChanged}, {"isMandatoryCodecPreferred", "([B)Z", &method_isMandatoryCodecPreferred}, }; GET_JAVA_METHODS(env, "com/android/bluetooth/a2dp/A2dpNativeInterface", javaMethods); const JNIJavaMethod codecConfigCallbacksMethods[] = { {"", "(IIIIIJJJJ)V", &android_bluetooth_BluetoothCodecConfig.constructor}, {"getCodecType", "()I", &android_bluetooth_BluetoothCodecConfig.getCodecType}, {"getCodecPriority", "()I", &android_bluetooth_BluetoothCodecConfig.getCodecPriority}, {"getSampleRate", "()I", &android_bluetooth_BluetoothCodecConfig.getSampleRate}, {"getBitsPerSample", "()I", &android_bluetooth_BluetoothCodecConfig.getBitsPerSample}, {"getChannelMode", "()I", &android_bluetooth_BluetoothCodecConfig.getChannelMode}, {"getCodecSpecific1", "()J", &android_bluetooth_BluetoothCodecConfig.getCodecSpecific1}, {"getCodecSpecific2", "()J", &android_bluetooth_BluetoothCodecConfig.getCodecSpecific2}, {"getCodecSpecific3", "()J", &android_bluetooth_BluetoothCodecConfig.getCodecSpecific3}, {"getCodecSpecific4", "()J", &android_bluetooth_BluetoothCodecConfig.getCodecSpecific4}, }; GET_JAVA_METHODS(env, "android/bluetooth/BluetoothCodecConfig", codecConfigCallbacksMethods); return 0; } }