继续研究hfp相关功能。蓝牙耳机可以控制手机接听、拒接、挂断电话,拨打电话等功能。本文主要分析下起这些操作的大致流程。
在系统应用Bluetooth中com_android_bluetooth.cpp提供了多个回调方法,由hardware、协议栈回调过来。蓝牙耳机的一些控制命令都会发到这里。
本文基于Android4.3源码。
1 接通电话 |
蓝牙耳机控制手机接通电话,回掉com_android_bluetooth.cpp中的answer_call_callback()函数,该函数主要操作是调用HeadsetStateMachine的onAnswerCall()函数,代码如下:
在onAnswerCall()中发送消息(消息类型STACK_EVENT,StackEvent事件类型EVENT_TYPE_ANSWER_CALL)向状体机,此时通话尚未接通,audio没有连接,所以此时处于Connected状态。状态机收到该消息后调用processAnswerCall()函数。processAnswerCall()代码如下:
-
private void processAnswerCall() {
-
if (mPhoneProxy != null) {
-
try {
-
//mPhoneProxy是通过bindservice 获取的。
-
mPhoneProxy.answerCall();
-
} catch (RemoteException e) {
-
}
-
} else {
-
}
-
}
初始化的时候会bind service,绑定的该service为系统应用Phone下的BluetoothPhoneService(AndroidManifest中该service的action为android.bluetooth.IBluetoothHeadsetPhone),代码如下:
-
//参数为android.bluetooth.IBluetoothHeadsetPhone
-
Intent intent = new Intent(IBluetoothHeadsetPhone.class.getName());
-
//resolveSystemService该方法是hide的,由系统使用的特殊功能来解决系统应用程序的服务意图。
-
intent.setComponent(intent.resolveSystemService(context.getPackageManager(), 0));
-
if (intent.getComponent() == null || !context.bindService(intent, mConnection, 0)) {
-
Log.e(TAG, "Could not bind to Bluetooth Headset Phone Service");
-
}
绑定service成功回调mConnection,在其成功回调中设置的mPhoneProxy。通过mPhoneProxy来调用service中提供的接口。mPhoneProxy.answerCall()跳到BluetoothPhoneService中answerCall。
-
public boolean answerCall() {
-
//申请权限,修改电话状态
-
enforceCallingOrSelfPermission(MODIFY_PHONE_STATE, null);
-
return PhoneUtils.answerCall(mCM.getFirstActiveRingingCall());
-
}
PhoneUtils调用answerCall,在这里面去接通电话。answerCall()就不具体分析了。
2 拒接、挂断电话 |
蓝牙耳机控制手机拒接、挂断电话,回掉com_android_bluetooth.cpp中的hangup_call_callback()函数,该函数主要操作是调用HeadsetStateMachine的onHangupCall()函数,代码如下:
-
private void onHangupCall() {
-
StackEvent event = new StackEvent(EVENT_TYPE_HANGUP_CALL);
-
sendMessage(STACK_EVENT, event);
-
}
此时HeadsetStateMachine可能处于Conneted或AudioOn状态,这两种状态收到该消息的处理一样,都是调用processHangupCall(),代码如下:
-
private void processHangupCall() {
-
if (isVirtualCallInProgress()) {
-
//对于虚拟电话,结束。
-
terminateScoUsingVirtualVoiceCall();
-
} else {
-
if (mPhoneProxy != null) {
-
try { //挂断电话
-
mPhoneProxy.hangupCall();
-
} catch (RemoteException e) {
-
}
-
} else {
-
}
-
}
-
}
对于虚拟电话则直接将其结束。真实的通话跳到BluetoothPhoneService的hangupCall。
-
public boolean hangupCall() {
-
enforceCallingOrSelfPermission(MODIFY_PHONE_STATE, null);
-
if (mCM.hasActiveFgCall()) { //挂断正在进行的通话
-
return PhoneUtils.hangupActiveCall(mCM.getActiveFgCall());
-
} else if (mCM.hasActiveRingingCall()) { //停止正在响铃的电话
-
return PhoneUtils.hangupRingingCall(mCM.getFirstActiveRingingCall());
-
} else if (mCM.hasActiveBgCall()) { //挂断保持的电话
-
return PhoneUtils.hangupHoldingCall(mCM.getFirstActiveBgCall());
-
}
-
return false;
-
}
hangupCall中会根据状态处理通话,优先处理正在进行的通话、其次是尚未接通的电话、最后是保持的电话。
3 更改通话音量 |
蓝牙耳机更改通话的音量,回掉com_android_bluetooth.cpp中的volume_control_callback()函数,该函数主要操作是调用HeadsetStateMachine的onVolumeChnaged()函数,代码如下:
-
private void onVolumeChanged(int type, int volume) {
-
StackEvent event = new StackEvent(EVENT_TYPE_VOLUME_CHANGED);
-
event.valueInt = type;
-
event.valueInt2 = volume;
-
sendMessage(STACK_EVENT, event);
-
}
此时HeadsetStateMachine可能处于Conneted或AudioOn状态,这两种状态收到该消息的处理一样,都是调用processVolumeEvent,代码如下:
-
private void processVolumeEvent(int volumeType, int volume) {
-
if (volumeType == HeadsetHalConstants.VOLUME_TYPE_SPK) {
-
mPhoneState.setSpeakerVolume(volume);
-
//是否在ui上显示
-
int flag = (getCurrentState() == mAudioOn) ? AudioManager.FLAG_SHOW_UI : 0;
-
//设置SCO通道声音大小。
-
mAudioManager.setStreamVolume(AudioManager.STREAM_BLUETOOTH_SCO, volume, flag);
-
} else if (volumeType == HeadsetHalConstants.VOLUME_TYPE_MIC) {
-
// 只是存了下该volume值,并没有设置mic。
-
mPhoneState.setMicVolume(volume);
-
} else {
-
}
-
}
更改音量两种类型,VOLUME_TYPE_MIC类型,保存了下该值,并没有看到具体用该值的地方。对于VOLUME_TYPE_SPK类型的,会设置SCO声音大小。如果此时处于AudioOn状态,则会在UI上显示。
4 拨打电话 |
蓝牙耳机进行拨打电话,回掉com_android_bluetooth.cpp中的dial_call_callback函数,该函数主要操作是调用HeadsetStateMachine的onDialCall()函数,代码如下:
-
private void onDialCall(String number) {
-
StackEvent event = new StackEvent(EVENT_TYPE_DIAL_CALL);
-
event.valueString = number;
-
sendMessage(STACK_EVENT, event);
-
}
此时HeadsetStateMachine可能处于Conneted或AudioOn状态,这两种状态收到该消息的处理一样,都是调用processDialCall,代码如下:
-
private void processDialCall(String number) {
-
String dialNumber;
-
if ((number == null) || (number.length() == 0)) {
-
//获取最近向外打的电话号码
-
dialNumber = mPhonebook.getLastDialledNumber();
-
if (dialNumber == null) { //没有最近拨打的电话,回应error
-
atResponseCodeNative(HeadsetHalConstants.AT_RESPONSE_ERROR, 0);
-
return;
-
}
-
} else if (number.charAt(0) == '>') {
-
//测试
-
} else {
-
// Remove trailing ';'
-
if (number.charAt(number.length() - 1) == ';') {
-
number = number.substring(0, number.length() - 1);
-
}
-
dialNumber = PhoneNumberUtils.convertPreDial(number);
-
}
-
terminateScoUsingVirtualVoiceCall(); // 终止虚拟呼叫
-
Intent intent = new Intent(Intent.ACTION_CALL_PRIVILEGED,
-
Uri.fromParts(SCHEME_TEL, dialNumber, null));
-
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
-
mService.startActivity(intent); //开启拨打电话的界面
-
mDialingOut = true;
-
sendMessageDelayed(DIALING_OUT_TIMEOUT, DIALING_OUT_TIMEOUT_VALUE);
-
}
蓝牙耳机发过来的命令可能携带电话号码,也可能不带,对于没有电话号码则查询最近的拨打电话记录,拨打最近拨打的电话。对于有号码,则拨打该号码。
Intent.ACTION_CALL_PRIVILEGED(该变量是hide的,执行任何号码的呼叫,紧急或不紧急):”android.intent.action.CALL_PRIVILEGED”
通过该action打开系统应用Phone中的OutgoingCallBroadcaster界面,向外进行拨打电话。
来源:oschina
链接:https://my.oschina.net/u/920274/blog/2998997