1 /*
2  * Copyright (C) 2018 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License
15  */
16 
17 package com.android.server.telecom.bluetooth;
18 
19 import android.bluetooth.BluetoothDevice;
20 import android.bluetooth.BluetoothHeadset;
21 import android.bluetooth.BluetoothHearingAid;
22 import android.bluetooth.BluetoothProfile;
23 import android.content.BroadcastReceiver;
24 import android.content.Context;
25 import android.content.Intent;
26 import android.content.IntentFilter;
27 import android.telecom.Log;
28 import android.telecom.Logging.Session;
29 
30 import com.android.internal.os.SomeArgs;
31 
32 import static com.android.server.telecom.bluetooth.BluetoothRouteManager.BT_AUDIO_IS_ON;
33 import static com.android.server.telecom.bluetooth.BluetoothRouteManager.BT_AUDIO_LOST;
34 
35 
36 public class BluetoothStateReceiver extends BroadcastReceiver {
37     private static final String LOG_TAG = BluetoothStateReceiver.class.getSimpleName();
38     public static final IntentFilter INTENT_FILTER;
39     static {
40         INTENT_FILTER = new IntentFilter();
41         INTENT_FILTER.addAction(BluetoothHeadset.ACTION_CONNECTION_STATE_CHANGED);
42         INTENT_FILTER.addAction(BluetoothHeadset.ACTION_AUDIO_STATE_CHANGED);
43         INTENT_FILTER.addAction(BluetoothHeadset.ACTION_ACTIVE_DEVICE_CHANGED);
44         INTENT_FILTER.addAction(BluetoothHearingAid.ACTION_CONNECTION_STATE_CHANGED);
45         INTENT_FILTER.addAction(BluetoothHearingAid.ACTION_ACTIVE_DEVICE_CHANGED);
46     }
47 
48     // If not in a call, BSR won't listen to the Bluetooth stack's HFP on/off messages, since
49     // other apps could be turning it on and off. We don't want to interfere.
50     private boolean mIsInCall = false;
51     private final BluetoothRouteManager mBluetoothRouteManager;
52     private final BluetoothDeviceManager mBluetoothDeviceManager;
53 
54     public void onReceive(Context context, Intent intent) {
55         Log.startSession("BSR.oR");
56         try {
57             String action = intent.getAction();
58             switch (action) {
59                 case BluetoothHeadset.ACTION_AUDIO_STATE_CHANGED:
60                     handleAudioStateChanged(intent);
61                     break;
62                 case BluetoothHearingAid.ACTION_CONNECTION_STATE_CHANGED:
63                 case BluetoothHeadset.ACTION_CONNECTION_STATE_CHANGED:
64                     handleConnectionStateChanged(intent);
65                     break;
66                 case BluetoothHearingAid.ACTION_ACTIVE_DEVICE_CHANGED:
67                 case BluetoothHeadset.ACTION_ACTIVE_DEVICE_CHANGED:
68                     handleActiveDeviceChanged(intent);
69                     break;
70             }
71         } finally {
72             Log.endSession();
73         }
74     }
75 
76     private void handleAudioStateChanged(Intent intent) {
77         int bluetoothHeadsetAudioState =
78                 intent.getIntExtra(BluetoothHeadset.EXTRA_STATE,
79                         BluetoothHeadset.STATE_AUDIO_DISCONNECTED);
80         BluetoothDevice device =
81                 intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
82         if (device == null) {
83             Log.w(LOG_TAG, "Got null device from broadcast. " +
84                     "Ignoring.");
85             return;
86         }
87 
88         Log.i(LOG_TAG, "Device %s transitioned to audio state %d",
89                 device.getAddress(), bluetoothHeadsetAudioState);
90         Session session = Log.createSubsession();
91         SomeArgs args = SomeArgs.obtain();
92         args.arg1 = session;
93         args.arg2 = device.getAddress();
94         switch (bluetoothHeadsetAudioState) {
95             case BluetoothHeadset.STATE_AUDIO_CONNECTED:
96                 if (!mIsInCall) {
97                     Log.i(LOG_TAG, "Ignoring BT audio on since we're not in a call");
98                     return;
99                 }
100                 mBluetoothRouteManager.sendMessage(BT_AUDIO_IS_ON, args);
101                 break;
102             case BluetoothHeadset.STATE_AUDIO_DISCONNECTED:
103                 mBluetoothRouteManager.sendMessage(BT_AUDIO_LOST, args);
104                 break;
105         }
106     }
107 
108     private void handleConnectionStateChanged(Intent intent) {
109         int bluetoothHeadsetState = intent.getIntExtra(BluetoothHeadset.EXTRA_STATE,
110                 BluetoothHeadset.STATE_DISCONNECTED);
111         BluetoothDevice device =
112                 intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
113 
114         if (device == null) {
115             Log.w(LOG_TAG, "Got null device from broadcast. " +
116                     "Ignoring.");
117             return;
118         }
119 
120         Log.i(LOG_TAG, "Device %s changed state to %d",
121                 device.getAddress(), bluetoothHeadsetState);
122 
123         boolean isHearingAid = BluetoothHearingAid.ACTION_CONNECTION_STATE_CHANGED
124                 .equals(intent.getAction());
125         if (bluetoothHeadsetState == BluetoothProfile.STATE_CONNECTED) {
126             mBluetoothDeviceManager.onDeviceConnected(device, isHearingAid);
127         } else if (bluetoothHeadsetState == BluetoothProfile.STATE_DISCONNECTED
128                 || bluetoothHeadsetState == BluetoothProfile.STATE_DISCONNECTING) {
129             mBluetoothDeviceManager.onDeviceDisconnected(device, isHearingAid);
130         }
131     }
132 
133     private void handleActiveDeviceChanged(Intent intent) {
134         BluetoothDevice device =
135                 intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
136         boolean isHearingAid =
137                 BluetoothHearingAid.ACTION_ACTIVE_DEVICE_CHANGED.equals(intent.getAction());
138         Log.i(LOG_TAG, "Device %s is now the preferred BT device for %s", device,
139                 isHearingAid ? "hearing aid" : "HFP");
140 
141         mBluetoothRouteManager.onActiveDeviceChanged(device, isHearingAid);
142         if (isHearingAid) {
143             Session session = Log.createSubsession();
144             SomeArgs args = SomeArgs.obtain();
145             args.arg1 = session;
146             if (device == null) {
147                 mBluetoothRouteManager.sendMessage(BT_AUDIO_LOST, args);
148             } else {
149                 if (!mIsInCall) {
150                     Log.i(LOG_TAG, "Ignoring hearing aid audio on since we're not in a call");
151                     return;
152                 }
153                 args.arg2 = device.getAddress();
154                 mBluetoothRouteManager.sendMessage(BT_AUDIO_IS_ON, args);
155             }
156         }
157     }
158 
159     public BluetoothDeviceManager getBluetoothDeviceManager() {
160         return mBluetoothDeviceManager;
161     }
162 
163     public BluetoothStateReceiver(BluetoothDeviceManager deviceManager,
164             BluetoothRouteManager routeManager) {
165         mBluetoothDeviceManager = deviceManager;
166         mBluetoothRouteManager = routeManager;
167     }
168 
169     public void setIsInCall(boolean isInCall) {
170         mIsInCall = isInCall;
171     }
172 }
173