转自 http://blog.****.net/u012439416/article/details/75267191
4 数据业务模式
在手机以及模块中,移动/联通/电信的信号都会有类似下面的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库中也会有对应值的定义。
2.1 数据网络模式默认值
在切换信号之前,首先看看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基线都会采用第一种方法进行设置,因为这样可以根据不同的项目设置不同的配置文件,灵活度高。
2.2 开机数据网络设置
一般双卡手机开机时,会调用三次RIL的setPreferredNetworkType方法向modem发送
RIL_REQUEST_SET_PREFERRED_NETWORK_TYPE消息设置数据网络。
2.2.1第一次设置
第一次设置比较容易,是在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构造方法中赋值的,一般为系统的默认值。
2.2.2第二次设置
其实,第二次和第三次设置的过程完全一样,都在在SIM加载完设置SIM信息的时候调用的,间隔时间也很短.几秒钟甚至不到 1秒钟。
都是SubscriptionController里面的方法,例如setCarrierText, setIconTint, setMccMnc等等。
这些方法都是调用notifySubscriptionInfoChanged方法进行设置的,调用流程图如下,

整个过程比较简单直白,在TelephonyRegistry和ServiceStateTracker使用了一个简单的listener,
SstSubscriptionsChangedListener是ServiceStateTracker的内部类,继承于OnSubscriptionsChangedListener,
-
protected class SstSubscriptionsChangedListener extends OnSubscriptionsChangedListener {
并且在ServiceStateTracker调用SubscriptionManager对象的addOnSubscriptionsChangedListener方法注册的,
最后调用TelephonyRegistry的addOnSubscriptionsChangedListener方法完成注册。
2.3设置界面的设置过程
图一 界面对应的代码为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这个消息。
2.4 TelephonyManager设置
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消息。回调处理在此就不论述了。