Merge 692e37d33afc857db7a53d17cea74a13886a5701 on remote branch
Change-Id: Ic5c08d70f233c96905823e0f11b5784e7b461aba
diff --git a/packages_apps_bluetooth_ext/src/avrcp/Avrcp_ext.java b/packages_apps_bluetooth_ext/src/avrcp/Avrcp_ext.java
index 923ad45..f4412cc 100644
--- a/packages_apps_bluetooth_ext/src/avrcp/Avrcp_ext.java
+++ b/packages_apps_bluetooth_ext/src/avrcp/Avrcp_ext.java
@@ -176,7 +176,9 @@
private int changePathDepth;
private byte changePathDirection;
HashMap<BluetoothDevice, Integer> mVolumeMap = new HashMap();
+ HashMap<BluetoothDevice, Boolean> mDeviceAbsVolMap = new HashMap();
public static final String VOLUME_MAP = "bluetooth_volume_map";
+ public static final String ABS_VOL_MAP = "bluetooth_ABS_VOL_map";
private boolean isShoActive = false;
AudioManagerAudioDeviceCallback mAudioManagerAudioDeviceCallback;
private boolean twsShoEnabled = false;
@@ -842,8 +844,6 @@
BluetoothDevice device = mA2dpService.getActiveDevice();
if(device == null)
break;
- mAudioManager.avrcpSupportsAbsoluteVolume(device.getAddress(),
- isAbsoluteVolumeSupported(deviceIndex));
if (mAbsVolThreshold > 0 && mAbsVolThreshold < mAudioStreamMax &&
vol > mAbsVolThreshold) {
if (DEBUG) Log.v(TAG, "remote inital volume too high " + vol + ">" +
@@ -905,6 +905,20 @@
deviceFeatures[deviceIndex].isAbsoluteVolumeSupportingDevice =
((deviceFeatures[deviceIndex].mFeatures &
BTRC_FEAT_ABSOLUTE_VOLUME) != 0);
+ //store Absolute volume support
+ Log.d(TAG,"absolute volume support device is present " + mDeviceAbsVolMap.containsKey(deviceFeatures[deviceIndex].mCurrentDevice));
+ if (!mDeviceAbsVolMap.containsKey(deviceFeatures[deviceIndex].mCurrentDevice) ||
+ (mDeviceAbsVolMap.containsKey(deviceFeatures[deviceIndex].mCurrentDevice) &&
+ mDeviceAbsVolMap.get(deviceFeatures[deviceIndex].mCurrentDevice) !=
+ deviceFeatures[deviceIndex].isAbsoluteVolumeSupportingDevice)) {
+ SharedPreferences.Editor absVolumeMapEditor = getAbsVolumeMap().edit();
+ mDeviceAbsVolMap.put(deviceFeatures[deviceIndex].mCurrentDevice, (boolean)deviceFeatures[deviceIndex].isAbsoluteVolumeSupportingDevice);
+ absVolumeMapEditor.putBoolean(deviceFeatures[deviceIndex].mCurrentDevice.getAddress(), deviceFeatures[deviceIndex].isAbsoluteVolumeSupportingDevice);
+ // Always use apply() since it is asynchronous, otherwise the call can hang waiting for
+ // storage to be written.
+ absVolumeMapEditor.apply();
+ Log.d(TAG,"absolute volume support data saved");
+ }
BATService mBatService = BATService.getBATService();
if ((mBatService != null) && mBatService.isBATActive()) {
Log.d(TAG,"MSG_NATIVE_REQ_GET_RC_FEATURES BA Active, update absvol support as true ");
@@ -1198,6 +1212,7 @@
}
if (msg.arg2 == AVRC_RSP_INTERIM && areMultipleDevicesConnected() &&
+ !(activeDevice != null && Objects.equals(deviceFeatures[deviceIndex].mCurrentDevice, activeDevice)) &&
deviceFeatures[deviceIndex].mInitialRemoteVolume == -1 &&
deviceFeatures[deviceIndex].mCurrentDevice.isTwsPlusDevice()) {
device = deviceFeatures[deviceIndex].mCurrentDevice;
@@ -1251,26 +1266,12 @@
/*Avoid send set absolute volume for store volume untill volume registration
complete and making synchronization to send only one setAbsolute volume
during connection*/
- if(getVolume(deviceFeatures[deviceIndex].mCurrentDevice) != -1) {
+ int val = getVolume(deviceFeatures[deviceIndex].mCurrentDevice);
+ if (val != volIndex) {
deviceFeatures[deviceIndex].mRemoteVolume = absVol;
setAbsVolumeFlag(deviceFeatures[deviceIndex].mCurrentDevice);
break;
}
- /*if volume is stored than no need to send setAbsoluteVolume use register volume*/
- else if(mAbsVolThreshold > 0 && mAbsVolThreshold < mAudioStreamMax &&
- volIndex > mAbsVolThreshold) {
- //To handle volume when device volume is not stored during
- // paring
- if (DEBUG) Log.v(TAG, "remote inital volume too high " + volIndex + ">" +
- mAbsVolThreshold);
- Message msg1 = mHandler.obtainMessage(MSG_SET_ABSOLUTE_VOLUME,
- mAbsVolThreshold , 0);
- mHandler.sendMessage(msg1);
- deviceFeatures[deviceIndex].mRemoteVolume = absVol;
- deviceFeatures[deviceIndex].mLocalVolume = volIndex;
- deviceFeatures[deviceIndex].mLastRequestedVolume = -1;
- break;
- }
}
}
if (deviceFeatures[deviceIndex].mLocalVolume != volIndex &&
@@ -3321,6 +3322,12 @@
SharedPreferences.Editor editor = pref.edit();
editor.putBoolean(mAddress, true);
editor.commit();
+ SharedPreferences.Editor absVolumeMapEditor = getAbsVolumeMap().edit();
+ mDeviceAbsVolMap.put(deviceFeatures[i].mCurrentDevice, false);
+ absVolumeMapEditor.putBoolean(mAddress, false);
+ // Always use apply() since it is asynchronous, otherwise the call can hang waiting for
+ // storage to be written.
+ absVolumeMapEditor.apply();
Log.v(TAG, "Exit blackListCurrentDevice");
}
@@ -3342,9 +3349,26 @@
public void resetBlackList(String address) {
SharedPreferences pref = mContext.getSharedPreferences(ABSOLUTE_VOLUME_BLACKLIST,
Context.MODE_PRIVATE);
+ if (!pref.contains(address)) {
+ Log.v(TAG, "resetBlackList : device is not added in blacklist");
+ return;
+ }
SharedPreferences.Editor editor = pref.edit();
editor.remove(address);
editor.apply();
+ SharedPreferences.Editor absVolumeMapEditor = getAbsVolumeMap().edit();
+ BluetoothDevice device = BluetoothAdapter.getDefaultAdapter().getRemoteDevice(address);
+ if (device != null && device.getBondState() == BluetoothDevice.BOND_NONE) {
+ Log.v(TAG, "resetBlackList : device state is unbonded");
+ return;
+ }
+ mDeviceAbsVolMap.put(device, true);
+ absVolumeMapEditor.putBoolean(address, true);
+ // Always use apply() since it is asynchronous, otherwise the call can hang waiting for
+ // storage to be written.
+ absVolumeMapEditor.apply();
+ //updating audio for absolute volume support
+ mAudioManager.avrcpSupportsAbsoluteVolume(address, true);
}
/**
@@ -5162,7 +5186,34 @@
}
return INVALID_DEVICE_INDEX;
}
+
+ public void removeVolumeForDevice(BluetoothDevice device) {
+ if (device == null || device.getBondState() != BluetoothDevice.BOND_NONE)
+ return;
+ SharedPreferences.Editor pref = getVolumeMap().edit();
+ SharedPreferences.Editor prefAbs = getAbsVolumeMap().edit();
+ Log.i(TAG, "RemoveStoredVolume: Remove stored stream volume level and abs volume support for device " + device);
+ mVolumeMap.remove(device);
+ pref.remove(device.getAddress());
+ // Always use apply() since it is asynchronous, otherwise the call can hang waiting for
+ // storage to be written.
+ pref.apply();
+ //remove absoute volume support
+ mDeviceAbsVolMap.remove(device);
+ prefAbs.remove(device.getAddress());
+ // Always use apply() since it is asynchronous, otherwise the call can hang waiting for
+ // storage to be written.
+ prefAbs.apply();
+ }
+
public void storeVolumeForDevice(BluetoothDevice device) {
+ /*Due to race condtion between active device change and bond state change, if bond state
+ change event handled first and active device change later than volume will not stored
+ if bond state is not paired */
+ if (device != null && device.getBondState() == BluetoothDevice.BOND_NONE) {
+ Log.i(TAG, "storeVolume: not saved because bond state is none");
+ return;
+ }
SharedPreferences.Editor pref = getVolumeMap().edit();
int index = getIndexForDevice(device);
int storeVolume = mAudioManager.getStreamVolume(AudioManager.STREAM_MUSIC);
@@ -5280,6 +5331,10 @@
return mContext.getSharedPreferences(VOLUME_MAP, Context.MODE_PRIVATE);
}
+ private SharedPreferences getAbsVolumeMap() {
+ return mContext.getSharedPreferences(ABS_VOL_MAP, Context.MODE_PRIVATE);
+ }
+
private void Avrcp_extVolumeManager() {
// Load the stored volume preferences into a hash map since shared preferences are slow
// to poll and update. If the device has been unbonded since last start remove it from
@@ -5300,12 +5355,29 @@
}
}
volumeMapEditor.apply();
+
+ allKeys = getAbsVolumeMap().getAll();
+ SharedPreferences.Editor absVolumeMapEditor = getAbsVolumeMap().edit();
+ for (Map.Entry<String, ?> entry : allKeys.entrySet()) {
+ String key = entry.getKey();
+ Object value = entry.getValue();
+ BluetoothDevice d = BluetoothAdapter.getDefaultAdapter().getRemoteDevice(key);
+
+ if (value instanceof Boolean && d.getBondState() == BluetoothDevice.BOND_BONDED) {
+ mDeviceAbsVolMap.put(d, (Boolean) value);
+ Log.w(TAG, "Address " + key + " from the absolute volume map support :" + value);
+ } else {
+ Log.w(TAG, "Removing " + key + " from the absolute volume map support state" + d.getBondState());
+ absVolumeMapEditor.remove(key);
+ }
+ }
+ absVolumeMapEditor.apply();
}
public int getVolume(BluetoothDevice device) {
if (!mVolumeMap.containsKey(device)) {
Log.w(TAG, "getVolume: Couldn't find volume preference for device: " + device);
- return -1;
+ return mAudioStreamMax/2;
}
Log.d(TAG, "getVolume: Returning volume " + mVolumeMap.get(device));
return mVolumeMap.get(device);
@@ -5313,19 +5385,26 @@
public void setAbsVolumeFlag(BluetoothDevice device) {
int deviceIndex = getIndexForDevice(device);
+ int volume = getVolume(device);
+ Log.d(TAG, " device allocated for abs volume support " + mDeviceAbsVolMap.containsKey(device));
+ if (mDeviceAbsVolMap.containsKey(device)) {
+ //updating abs volume supported or not to audio when active device change is success
+ Log.d(TAG, " Returning abs volume support " + mDeviceAbsVolMap.get(device));
+ mAudioManager.avrcpSupportsAbsoluteVolume(device.getAddress(), mDeviceAbsVolMap.get(device));
+ } else {
+ //absolute volume is not stored upadtion false by default
+ Log.d(TAG, "Returning abs volume support false ");
+ mAudioManager.avrcpSupportsAbsoluteVolume(device.getAddress(), false);
+ }
if (deviceIndex == INVALID_DEVICE_INDEX) {
Log.e(TAG,"Invalid device index for setAbsVolumeFlag");
return;
}
- int volume = getVolume(device);
- //updating abs volume supported or not to audio when active device change is success
- mAudioManager.avrcpSupportsAbsoluteVolume(device.getAddress(),
- isAbsoluteVolumeSupported(deviceIndex));
if(deviceFeatures[deviceIndex].isAbsoluteVolumeSupportingDevice == false) {
Log.d(TAG,"isAbsoluteVolumeSupportingDevice is false or volume is not stored");
return;
}
- if(deviceFeatures[deviceIndex].mInitialRemoteVolume == -1 || volume == -1) {
+ if(deviceFeatures[deviceIndex].mInitialRemoteVolume == -1) {
Log.e(TAG,"intial volume is not updated or volume is not stored");
return;
}
diff --git a/system_bt_ext/bta/swb/bta_ag_swb.cc b/system_bt_ext/bta/swb/bta_ag_swb.cc
index 496f110..1514bc8 100644
--- a/system_bt_ext/bta/swb/bta_ag_swb.cc
+++ b/system_bt_ext/bta/swb/bta_ag_swb.cc
@@ -34,6 +34,7 @@
#include "utl.h"
#include "device/include/interop.h"
#include <hardware/vendor_hf.h>
+#include <cutils/properties.h>
#if (SWB_ENABLED == TRUE)
@@ -138,15 +139,17 @@
*
******************************************************************************/
void bta_ag_send_qac(tBTA_AG_SCB* p_scb, tBTA_AG_DATA* p_data) {
- /* send +BCS */
- APPL_TRACE_DEBUG("send +QAC codecs suuported");
- if (!get_swb_codec_status()) {
+ char value[PROPERTY_VALUE_MAX];
+ /* send +QAC */
+ APPL_TRACE_DEBUG("send +QAC codec response");
+
+ if (property_get("persist.vendor.qcom.bluetooth.enable.swb", value, "false")
+ && strncmp(value, "true", sizeof("true")) != 0) {
bta_ag_send_result(p_scb, BTA_AG_LOCAL_RES_QAC, SWB_CODECS_UNSUPPORTD, 0);
return;
- } else{
- bta_ag_send_result(p_scb, BTA_AG_LOCAL_RES_QAC, SWB_CODECS_SUPPORTED, 0);
}
- if (p_scb->sco_codec == BTA_AG_SCO_SWB_SETTINGS_Q0) {
+ bta_ag_send_result(p_scb, BTA_AG_LOCAL_RES_QAC, SWB_CODECS_SUPPORTED, 0);
+ if (get_swb_codec_status() && p_scb->sco_codec == BTA_AG_SCO_SWB_SETTINGS_Q0) {
p_scb->is_swb_codec = true;
}
}
diff --git a/system_bt_ext/bta/tws_plus/ag/bta_ag_twsp_sco.cc b/system_bt_ext/bta/tws_plus/ag/bta_ag_twsp_sco.cc
index 5d579ed..aee4c0c 100644
--- a/system_bt_ext/bta/tws_plus/ag/bta_ag_twsp_sco.cc
+++ b/system_bt_ext/bta/tws_plus/ag/bta_ag_twsp_sco.cc
@@ -308,7 +308,8 @@
p_sco->state = BTA_AG_SCO_OPEN_ST;
other_scb = get_other_twsp_scb((p_scb->peer_addr));
if (other_scb && twsp_sco_active(other_scb) == false &&
- get_twsp_state(other_scb) == TWSPLUS_EB_STATE_INEAR) {
+ get_twsp_state(other_scb) != TWSPLUS_EB_STATE_OUT_OF_EAR &&
+ get_twsp_state(other_scb) != TWSPLUS_EB_STATE_INCASE) {
APPL_TRACE_WARNING("Calling SCO open");
dispatch_event_primary_peer_device(p_scb, BTA_AG_SCO_OPEN_E);
}
@@ -319,7 +320,8 @@
p_sco->state = BTA_AG_SCO_LISTEN_ST;
other_scb = get_other_twsp_scb((p_scb->peer_addr));
if (other_scb && twsp_sco_active(other_scb) == false &&
- get_twsp_state(other_scb) == TWSPLUS_EB_STATE_INEAR) {
+ get_twsp_state(other_scb) != TWSPLUS_EB_STATE_OUT_OF_EAR &&
+ get_twsp_state(other_scb) != TWSPLUS_EB_STATE_INCASE) {
//Atleast try bringing up the other EB eSCO
APPL_TRACE_WARNING("Calling SCO open for other EB");
dispatch_event_primary_peer_device(p_scb, BTA_AG_SCO_OPEN_E);
diff --git a/system_bt_ext/btif/include/btif_twsp_hf.h b/system_bt_ext/btif/include/btif_twsp_hf.h
index cd7da30..dc89956 100644
--- a/system_bt_ext/btif/include/btif_twsp_hf.h
+++ b/system_bt_ext/btif/include/btif_twsp_hf.h
@@ -51,5 +51,7 @@
void btif_hf_twsp_send_bvra_update(int idx, tBTA_AG_RES_DATA* ag_res);
+bool btif_hf_check_twsp_state_for_sco(int idx);
+
} // namespace headset
} // namespace bluetooth
diff --git a/system_bt_ext/btif/src/btif_twsp_hf.cc b/system_bt_ext/btif/src/btif_twsp_hf.cc
index 029df6d..9dddeca 100644
--- a/system_bt_ext/btif/src/btif_twsp_hf.cc
+++ b/system_bt_ext/btif/src/btif_twsp_hf.cc
@@ -33,8 +33,10 @@
*/
#include <hardware/bluetooth.h>
+#include "btif_common.h"
#include "btif_hf.h"
#include "btif_tws_plus.h"
+#include "btif_twsp_hf.h"
#include "vendor.h"
#include <cutils/properties.h>
@@ -135,13 +137,32 @@
void btif_hf_twsp_send_bvra_update(int current_index, tBTA_AG_RES_DATA* ag_res) {
int peer_eb_idx = btif_hf_get_other_connected_twsp_index(current_index);
- if ((peer_eb_idx >= 0) && (peer_eb_idx < BTA_AG_MAX_NUM_CLIENTS) &&
+ if ((peer_eb_idx >= 0) && (peer_eb_idx < btif_max_hf_clients) &&
(btif_hf_cb[peer_eb_idx].peer_feat & BTA_AG_PEER_FEAT_VREC)) {
BTIF_TRACE_DEBUG("%s: Send BVRA update for peer earbud, idx: %d", __func__, peer_eb_idx);
BTA_AgResult(btif_hf_cb[peer_eb_idx].handle, BTA_AG_BVRA_RES, ag_res);
}
}
+bool btif_hf_check_twsp_state_for_sco(int idx) {
+ BTIF_TRACE_DEBUG("%s: Current earbud state: %d", __func__, btif_hf_cb[idx].twsp_state);
+ if (btif_hf_cb[idx].twsp_state == TWSPLUS_EB_STATE_OUT_OF_EAR ||
+ btif_hf_cb[idx].twsp_state == TWSPLUS_EB_STATE_INCASE) {
+ int peer_eb_idx = btif_hf_get_other_connected_twsp_index(idx);
+ if ((peer_eb_idx < 0) || (peer_eb_idx >= btif_max_hf_clients) ||
+ (btif_hf_cb[peer_eb_idx].twsp_state == TWSPLUS_EB_STATE_OUT_OF_EAR) ||
+ (btif_hf_cb[peer_eb_idx].twsp_state == TWSPLUS_EB_STATE_INCASE)) {
+ BTIF_TRACE_DEBUG("%s: No any connected earbud in ear", __func__);
+ btif_transfer_context(btif_in_hf_generic_evt, BTIF_HFP_CB_AUDIO_CONNECTING,
+ (char*)(&btif_hf_cb[idx].connected_bda), sizeof(RawAddress), NULL);
+ btif_transfer_context(btif_in_hf_generic_evt, BTIF_HFP_CB_AUDIO_DISCONNECTED,
+ (char*)(&btif_hf_cb[idx].connected_bda), sizeof(RawAddress), NULL);
+ }
+ return false;
+ }
+ return true;
+}
+
} // namespace headset
} // namespace bluetooth