在手机以及模块中,移动/联通/电信的信号都会有类似下面的2G/3G/4G切换,
图一 信号模式切换图
这些值的定义都在RILConstants.java中,如下,
/* NETWORK_MODE_* See ril.h RIL_REQUEST_SET_PREFERRED_NETWORK_TYPE */
int NETWORK_MODE_WCDMA_PREF = 0; /* GSM/WCDMA (WCDMA preferred) */
int NETWORK_MODE_GSM_ONLY = 1; /* GSM only */
int NETWORK_MODE_WCDMA_ONLY = 2; /* WCDMA only */
int NETWORK_MODE_GSM_UMTS = 3; /* GSM/WCDMA (auto mode, according to PRL)
int NETWORK_MODE_CDMA = 4; /* CDMA and EvDo (auto mode, according to PRL)
int NETWORK_MODE_CDMA_NO_EVDO = 5; /* CDMA only */
int NETWORK_MODE_EVDO_NO_CDMA = 6; /* EvDo only */
int NETWORK_MODE_GLOBAL = 7; /* GSM/WCDMA, CDMA, and EvDo*/
int NETWORK_MODE_LTE_CDMA_EVDO = 8; /* LTE, CDMA and EvDo */
int NETWORK_MODE_LTE_GSM_WCDMA = 9; /* LTE, GSM/WCDMA */
int NETWORK_MODE_LTE_CDMA_EVDO_GSM_WCDMA = 10;
/* LTE, CDMA, EvDo, GSM/WCDMA */
int NETWORK_MODE_LTE_ONLY = 11; /* LTE Only mode. */
int NETWORK_MODE_LTE_WCDMA = 12; /* LTE/WCDMA */
当然,ril库中也会有对应值的定义。
在切换信号之前,首先看看android系统中信号的默认值。在加载完系统的SIM卡之后,然后才会设置信号的默认值。
主要流程图如下,
图二 获取数据网络默认值流程图
该流程图分为2个进程,左半部分为系统服务所在的进程,又半部分为phone进程。
主要的步骤如下,
1,android系统启动加载sim卡完成之后,发送ACTION_INTERNAL_SIM_STATE_CHANGED广播;
2, 系统服务所在的进程的SubscriptionInfoUpdater的构造方法中会注册该广播, sReceiver对该广播的处理如下,
else if (action.equals(IccCardProxy.ACTION_INTERNAL_SIM_STATE_CHANGED)) {
if (IccCardConstants.INTENT_VALUE_ICC_LOCKED.equals(simStatus)) {//sim被锁
String reason = intent.getStringExtra(
IccCardConstants.INTENT_KEY_LOCKED_REASON);
sendMessage(obtainMessage(EVENT_SIM_LOCKED, slotId, -1, reason));
} else if (IccCardConstants.INTENT_VALUE_ICC_LOADED.equals(simStatus)){//sim加载完
sendMessage(obtainMessage(EVENT_SIM_LOADED, slotId, -1));//发送消息
}
3, 系统服务所在的进程处理完成之后,通过binder机制跨进程调用phone进程设置数据网络默认sim卡以及信号值。
PhoneFactory的calculatePreferredNetworkType方法如下,
public static int calculatePreferredNetworkType(Context context, int phoneSubId) {
int networkType = android.provider.Settings.Global.getInt(context.getContentResolver(),
android.provider.Settings.Global.PREFERRED_NETWORK_MODE + phoneSubId,
RILConstants.PREFERRED_NETWORK_MODE);
return networkType;
}
当然,首先读取setting数据库中的PREFERRED_NETWORK_MODE 字段值,如果为空就设置RILConstants的
PREFERRED_NETWORK_MODE值。
在刚刷机时,该值肯定为空,只有在设置之后,信号值才会保存在PREFERRED_NETWORK_MODE 字段中。
也就是说,如果不是刷机启动,机器开机后会默认加载上次设置的值。
RILConstants的PREFERRED_NETWORK_MODE值如下,
int PREFERRED_NETWORK_MODE = TelephonyManager
.getDefaultPreferredNetworkType(0, NETWORK_MODE_WCDMA_PREF);
TelephonyManager的getDefaultPreferredNetworkType方法如下,
public static int getDefaultPreferredNetworkType(int phoneId, int defaultValue) {
String mode = getTelephonyProperty(phoneId, "ro.telephony.default_network", null);
if (mode != null) {
return Integer.parseInt(mode);
}
return defaultValue;
}
这2个方法的目的都是为设置数据网络信号值,
1,首先通过配置文件设置,android系统中一般都有设置,虽然设置的文件不同,但是语句完全一样,
ro.telephony.default_network=5
2,如果配置文件未设置,就会在RILConstants中进行设置, getDefaultPreferredNetworkType方法第二个参数就是默认值。
当然,几乎所有的android基线都会采用第一种方法进行设置,因为这样可以根据不同的项目设置不同的配置文件,灵活度高。
一般双卡手机开机时,会调用三次RIL的setPreferredNetworkType方法向modem发送
RIL_REQUEST_SET_PREFERRED_NETWORK_TYPE消息设置数据网络。
第一次设置比较容易,是在phone进程的RIL和ril守护进程连接之后,收到守护进程上报的RIL_UNSOL_RIL_CONNECTED
消息就直接设置网络制式,
case RIL_UNSOL_RIL_CONNECTED: {
if (RILJ_LOGD) unsljLogRet(response, ret);
getRadioCapability(mSupportedRafHandler.obtainMessage());
// Initial conditions
setRadioPower(false, null);
setPreferredNetworkType(mPreferredNetworkType, null);
setCdmaSubscriptionSource(mCdmaSubscription, null);
setCellInfoListRate(Integer.MAX_VALUE, null);
notifyRegistrantsRilConnectionChanged(((int[])ret)[0]);
break;
}
mPreferredNetworkType的值是在RIL构造方法中赋值的,一般为系统的默认值。
其实,第二次和第三次设置的过程完全一样,都在在SIM加载完设置SIM信息的时候调用的,间隔时间也很短.几秒钟甚至不到 1秒钟。
都是SubscriptionController里面的方法,例如setCarrierText, setIconTint, setMccMnc等等。
这些方法都是调用notifySubscriptionInfoChanged方法进行设置的,调用流程图如下,
整个过程比较简单直白,在TelephonyRegistry和ServiceStateTracker使用了一个简单的listener,
SstSubscriptionsChangedListener是ServiceStateTracker的内部类,继承于OnSubscriptionsChangedListener,
protected class SstSubscriptionsChangedListener extends OnSubscriptionsChangedListener {
并且在ServiceStateTracker调用SubscriptionManager对象的addOnSubscriptionsChangedListener方法注册的,
最后调用TelephonyRegistry的addOnSubscriptionsChangedListener方法完成注册。
图一 界面对应的代码为MobileNetworkSettings.java,数据网络设置的流程图如下,
重点分析以下三个点:
1,MobileNetworkSettings的onPreferenceChange方法,
MobileNetworkSettings的onPreferenceChange方法如下,
public boolean onPreferenceChange(Preference preference, Object objValue) {
final int phoneSubId = mPhone.getSubId(); //获取卡槽
if (preference == mButtonPreferredNetworkMode) {
//NOTE onPreferenceChange seems to be called even if there is no change
//Check if the button value is changed from the System.Setting
mButtonPreferredNetworkMode.setValue((String) objValue);
int buttonNetworkMode;
buttonNetworkMode = Integer.valueOf((String) objValue).intValue();
int settingsNetworkMode = android.provider.Settings.Global.getInt(
mPhone.getContext().getContentResolver(),
android.provider.Settings.Global.PREFERRED_NETWORK_MODE + phoneSubId,
preferredNetworkMode);//获取当前的数据网络模式
if (buttonNetworkMode != settingsNetworkMode) {
int modemNetworkMode;
// if new mode is invalid ignore it
switch (buttonNetworkMode) {
case Phone.NT_MODE_WCDMA_PREF:
case Phone.NT_MODE_GSM_ONLY:
case Phone.NT_MODE_WCDMA_ONLY:
case Phone.NT_MODE_GSM_UMTS:
case Phone.NT_MODE_CDMA:
case Phone.NT_MODE_CDMA_NO_EVDO:
case Phone.NT_MODE_EVDO_NO_CDMA:
case Phone.NT_MODE_GLOBAL:
case Phone.NT_MODE_LTE_CDMA_AND_EVDO:
case Phone.NT_MODE_LTE_GSM_WCDMA:
case Phone.NT_MODE_LTE_CDMA_EVDO_GSM_WCDMA:
case Phone.NT_MODE_LTE_ONLY:
case Phone.NT_MODE_LTE_WCDMA:
// This is one of the modes we recognize
modemNetworkMode = buttonNetworkMode;
break;
default:
return true;
}
android.provider.Settings.Global.putInt(mPhone.getContext().getContentResolver(),
android.provider.Settings.Global.PREFERRED_NETWORK_MODE + phoneSubId,
buttonNetworkMode );// 保存到数据库中
//Set the modem network mode
mPhone.setPreferredNetworkType(modemNetworkMode, mHandler
.obtainMessage(MyHandler.MESSAGE_SET_PREFERRED_NETWORK_TYPE));
}
}
由此可见,主要步骤如下,
A,首先从数据库中获取上次的信号模式,如果和当前的值相等就没有设置的必要;
B,然后检查模式是否正确,将模式的值保存在设置数据库中;
C,调用PhoneProxy的setPreferredNetworkType方法设置网络模式。
其中需要注意的是MESSAGE_SET_PREFERRED_NETWORK_TYPE这个消息。
TelephonyManager是有phone进程有关的API接口,当然也可以进行网络设置,调用其setPreferredNetworkType方法就可以了,
当然需要添加MODIFY_PHONE_STATE权限,
调用流程图如下,
整个过程和数据业务的打开调用过程差不多。
TelephonyManager的 setPreferredNetworkType方法如下,
try {
return getITelephony().setPreferredNetworkType(networkType);
}
跨进程调用phone进程的PhoneInterfaceManager的setPreferredNetworkType方法,如下,
首先检查权限,
enforceModifyPermissionOrCarrierPrivilege();
然后发送CMD_SET_PREFERRED_NETWORK_TYPE消息切换到主线程中执行,
Boolean success = (Boolean) sendRequest(CMD_SET_PREFERRED_NETWORK_TYPE, networkType, subId);
如果切换成功,将切换后的值保存到数据库,
if (success) {
Settings.Global.putInt(mPhone.getContext().getContentResolver(),
Settings.Global.PREFERRED_NETWORK_MODE + subId, networkType);
}
PhoneInterfaceManager对CMD_SET_PREFERRED_NETWORK_TYPE消息处理如下,
request = (MainThreadRequest) msg.obj;
onCompleted = obtainMessage(EVENT_SET_PREFERRED_NETWORK_TYPE_DONE, request);
int networkType = (Integer) request.argument;
getPhoneFromRequest(request).setPreferredNetworkType(networkType, onCompleted);
break;
首先封装EVENT_SET_PREFERRED_NETWORK_TYPE_DONE消息,
然后调用getPhoneFromRequest获取实际的phone对象,当然哪个对象都继承于PhoneBase,
最后的setPreferredNetworkType在PhoneBase中实现,
PhoneBase直接调用RIL的setPreferredNetworkType方法进行设置。
mCi.setPreferredNetworkType(networkType, response);
最后注意,当RIL收到modem的处理消息时,会向PhoneInterfaceManager发送已经封装的
EVENT_SET_PREFERRED_NETWORK_TYPE_DONE消息。回调处理在此就不论述了。