转自 http://blog.csdn.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消息。回调处理在此就不论述了。