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