五、 数据业务APN参数的开机默认使能

在DcTracker的创建及准备工作中,initApnContexts()将初始化好当前系统支持的ApnContexts类型,但是这些APN参数默认都是没有被Enable的,只有当APN被Enable之后,该APN才可以被使用,下面我们就来看如何将一个APN参数激活

protected void initApnContexts() {
    // Load device network attributes from resources
    String[] networkConfigStrings = mPhone.getContext().getResources().getStringArray(
            com.android.internal.R.array.networkAttributes);
    for (String networkConfigString : networkConfigStrings) {
        NetworkConfig networkConfig = new NetworkConfig(networkConfigString);
        ApnContext apnContext = null;

        switch (networkConfig.type) {
        case ConnectivityManager.TYPE_MOBILE:
            apnContext = addApnContext(PhoneConstants.APN_TYPE_DEFAULT, networkConfig);
            break;
        case ConnectivityManager.TYPE_MOBILE_MMS:
            apnContext = addApnContext(PhoneConstants.APN_TYPE_MMS, networkConfig);
            break;
        case ConnectivityManager.TYPE_MOBILE_SUPL:
            apnContext = addApnContext(PhoneConstants.APN_TYPE_SUPL, networkConfig);
            break;
        case ConnectivityManager.TYPE_MOBILE_DUN:
            apnContext = addApnContext(PhoneConstants.APN_TYPE_DUN, networkConfig);
            break;
        case ConnectivityManager.TYPE_MOBILE_HIPRI:
            apnContext = addApnContext(PhoneConstants.APN_TYPE_HIPRI, networkConfig);
            break;
        case ConnectivityManager.TYPE_MOBILE_FOTA:
            apnContext = addApnContext(PhoneConstants.APN_TYPE_FOTA, networkConfig);
            break;
        case ConnectivityManager.TYPE_MOBILE_IMS:
            apnContext = addApnContext(PhoneConstants.APN_TYPE_IMS, networkConfig);
            break;
        case ConnectivityManager.TYPE_MOBILE_CBS:
            apnContext = addApnContext(PhoneConstants.APN_TYPE_CBS, networkConfig);
            break;
        case ConnectivityManager.TYPE_MOBILE_IA:
            apnContext = addApnContext(PhoneConstants.APN_TYPE_IA, networkConfig);
            break;
        case ConnectivityManager.TYPE_MOBILE_EMERGENCY:
            apnContext = addApnContext(PhoneConstants.APN_TYPE_EMERGENCY, networkConfig);
            break;
        default:
            continue;
        }
    }
}

可以看出总共初始化了8种ApnContext,每一种的Type对应一种数据连接方式,他们的对应关系如下:
APN_TYPE_DEFAULT :默认网络连接,当激活时所有数据传输都使用该连接,不能与其他网络连接同时使用
APN_TYPE_MMS :彩信专用连接,此连接与default类似,用于与载体的多媒体信息服务器对话的应用程序,此连接能与default连接同时使用。适用场合:使用彩信服务时,必须有mms类型的接入点,不必选中,应用程序会自动使用此接入点
APN_TYPE_SUPL :是Secure User Plane Location“安全用户面定位”的简写,此连接与default类似,用于帮助定位设备与载体的安全用户面定位服务器对话的应用程序,此连接能与default连接同时使用,
APN_TYPE_DUN :Dial Up Networking拨号网络的简称,此连接与default连接类似,用于执行一个拨号网络网桥,使载体能知道拨号网络流量的应用程序,此连接能与default连接同时使用
APN_TYPE_HIPRI :高优先级网络,与default类似,但路由设置不同
APN_TYPE_FOTA :
APN_TYPE_IMS :
APN_TYPE_CBS :
APN_TYPE_IA :
APN_TYPE_EMERGENCY:紧急APN
addApnContext就是将相应的apnContext添加到mApnContexts这个List中去

private ApnContext addApnContext(String type, NetworkConfig networkConfig) {
    ApnContext apnContext = new ApnContext(mPhone.getContext(), type, LOG_TAG, networkConfig,
            this);
    mApnContexts.put(type, apnContext);
    mPrioritySortedApnContexts.add(apnContext);
    return apnContext;
}

ApnContext的构造如下,其中apnType被设置为了指定的8种类型之一,一一对应关系;mDataEnabled 被初始化为了false,因此是不能联网的,需要使能之后才能联网;使用的优先级priority时根据解析frameworks/base/core/res/res/values/config.xml文件得到的,如下图

public ApnContext(Context context, String apnType, String logTag, NetworkConfig config,
        DcTrackerBase tracker) {
    mContext = context;
    mApnType = apnType;
    mState = DctConstants.State.IDLE;
    setReason(Phone.REASON_DATA_ENABLED);
    mDataEnabled = new AtomicBoolean(false);
    mDependencyMet = new AtomicBoolean(config.dependencyMet);
    mWaitingApnsPermanentFailureCountDown = new AtomicInteger(0);
    priority = config.priority;
    LOG_TAG = logTag;
    mDcTracker = tracker;
}       

这里写图片描述
第一种情况是开机后默认使能APN_TYPE_DEFAULT类型的APN
第二种情况是其他的类型使能,开机不会默认使能,而是通过TelephonyNetworkFactory中的needNetworkFor()方法触发的,当当前网络的评分比其他网络高时,就会通过needNetworkFor()方法触发当前网络的建立。比如当前WIFI断开时,如果发现数据网络是打开的,此时就会通过该方法激活数据流量,从而实现从WIFI到数据的转换过程。
这里先分析开机情况时默认使能APN_TYPE_DEFAULT类型的APN,激活的条件是EVENT_DATA_ATTACHED,这个事件的注册是在DctController中,如下

private DctController(PhoneProxy[] phones) {
    for (int i = 0; i < mPhoneNum; ++i) {
        // Register for radio state change
        PhoneBase phoneBase = (PhoneBase)mPhones[i].getActivePhone();
        updatePhoneBaseForIndex(i, phoneBase);
    }
}    
private void updatePhoneBaseForIndex(int index, PhoneBase phoneBase) {
    phoneBase.getServiceStateTracker().registerForDataConnectionAttached(mRspHandler,
               EVENT_DATA_ATTACHED + index, null);
    phoneBase.getServiceStateTracker().registerForDataConnectionDetached(mRspHandler,
               EVENT_DATA_DETACHED + index, null);
}

此处只关注EVENT_DATA_ATTACHED 事件注册,这个事件的触发是由GsmServiceStateTracker或者CdmaServiceStateTracker发出的,当DataRegState的状态由STATE_OUT_OF_SERVICE -> STATE_IN_SERVICE时就会触发EVENT_DATA_ATTACHED 事件,具体的流程如下图
这里写图片描述
注意:DctController是动态数据链接的管理者,也就是和ConnectivityService耦合的部分,ConnectivityService决定了采取哪种方式,具体的细节在此不分析,另外分析这节。

猜你喜欢

转载自blog.csdn.net/xiabodan/article/details/53766729
今日推荐