Data Connect流程分析(基于1.5源码)
Android的数据连接是基于PPP方式的,主要步骤为:首先通过AT命令激活PDP连接,然后利用pppd通过数据端口完成拨号连接;
数据连接的核心控制类是DataConnectionTracker,存在于GSMPhone里,数据连接不需要用户的干预,在APN设置好之后,在适当的情况下就会自动激活,激活的入口点是:DataConnectionTracker.trySetupData→setupData→ PdpConnection.connetc→CommandsInterface.setupDefaultPDP,通过PdpConnection访问GSMPhone中的RIL层的setupDefaultPDP实现,setupDefaultPDP的结果由EVENT_SETUP_PDP_DONE返回,如果成功,则开始调用pppd完成实际连接,这是通过DataLink.connect实现的;
DataLink只是抽象基类,此处它的实现类是PppLink,实现DataLinkInterface接口,所以DataLink.connect实际上调用PppLink.connect,它通过SystemService.start(SERVICE_PPPD_GPRS)开始pppd服务,并通过checkPPP函数访问Linux的sys文件系统来查询pppd的连接状态,如果成功,便可以将LINKUP的消息通知出去以完成连接流程。
接入点使用在我看来主要包括接入点的创建、接入点的切换以及接入点的删除三个方面,我们下面按照android源码,按照程序调用的先后顺序依次分析其流程;
Create New APN流程分析
Android因为是以事件驱动的,因此在诸如接入点设置这样的操作的时候,都是从按键触发事件开始的:Activity.java里的onKeyDown函数;由于是基于EVENT驱动的,因此在每一个动作的时候都会触发一定Type的Message,因此对于源代码流程的分析也比较有利;
Create New APN的过程主要就是APN如name、port、proxy等的添加以及在设置的过程里状态的切换等;
一、下面为这个过程里JAVA Framework调用的过程:
1、 ActivityManagerService.java:startActivity:
说明:界面跳转,使用隐式的界面跳转,这个过程是基于事件的,在Android中,传递数据使用Intent,Intent相当于各个Activity之间的桥梁,可以传递数据,可以通过Intent启动另外一个Activity。Intent有显式和隐式之分,显式的是直接什么要启动的组件,比如Service或者Activity,隐式的通过配置的datatype、url、action来找到匹配的组件启动。
2、telephony/TelephonyProvider.java:insert函数:
说明:通过对URL 的s_urlMatcher.match,URL_TELEPHONY宏的处理,对Name,APN等的检查和容错
3、MobileDataStateTracker.java:MobileDataStateReceiver函数
说明:这个是这部分处理的一个核心函数,该函数的一个实现为onReceive(),在此函数里对于APN的各种参数如Type(isApnTypeIncluded(apnTypeList))以及状态state进行判断和转换,在这个时候,状态的切换为:old =CONNECTED and new state=DISCONNECTED
4、ConnectivityService.java:handleMessage()函数
说明:由于系统本身就是事件驱动的,因此这个handleMessage被调用来完成network state状态的改变:DISCONNECTED/DISCONNECTED,并且在这个函数实现了WIFI接入点有关判断;
5、GpsLocationProvider.java:updateNetworkState函数
说明:GPS状态的更新
6、MobileDataStateTracker.java:MobileDataStateReceiver()函数:
说明:类似(3),只是状态切换变为:state= CONNECTING, old= DISCONNECTED, reason= apnChanged
7、NetworkStateTracker.java:setDetailedState()函数
说明:该函数记录网络状态的改变,并在改变的时候发送一个notify事件:
Message msg = mTarget.obtainMessage(EVENT_STATE_CHANGED, mNetworkInfo);
8、ConnectivityService.java:handleMessage()函数:
说明:类似(4),状态:CONNECTING/CONNECTING
9、Checkin.java:updateStats函数:
说明:update statistics in the database
10、MobileDataStateTracker.java:MobileDataStateReceiver
说明:处理在onReceive里,状态为:state= CONNECTED, old= CONNECTING, reason= apnChanged
11、NetworkStateTracker.java:setDetailedState()函数
说明:(7),状态:old =CONNECTING and new state=CONNECTED
12、NetworkStateTracker.java:updateNetworkSettings函数
说明:该函数从Network TCP buffer读取network设置参数,并设置网络
13、ConnectivityService。java:handleDnsConfigurationChange
说明:从dnsList里读取预设的dns 通过writePidDns设置dns
二、下面分析RIL Java层的处理:
该部分的核心实现存在于Ril.java以及GsmDataConnectionTracker.java之中,Ril.java中RIL.RILSender负责处理命令的发送,RIL.RILReceiver用于处理命令响应以及主动上报信息的接收;
Ril.Java中一个命令发送的流程为:RILRequest.obtain(命令ID)→复制参数→通过Send()函数发送EVENT_SEND→在RILSender线程中处理EVENT_SEND→将命令写到out stream(socket);
Ril.java响应和主动上报消息的流程为:RILReceiver线程监视mSocket input→readMessage(读取完整响应)→processReponse→分别处理RESPONSE_UNSOLICITED(主动上报)与REPONSE_SOLICITED(命令响应)
RILD守护进程里的Request都是由RIL.java发起
APN 切换流程分析
1、ApnPreference.java: onCheckedChanged
说明:检查接入点切换的ID是否合法
2、MobileDataStateTracker.java:MobileDataStateReceiver
说明:处理在onReceive里,状态为:state= DISCONNECTED, old= CONNECTED
3、ConnectivityService.java:handleMessage()函数:
说明:状态:DISCONNECTED/DISCONNECTED
4、ActivityThread.java:getProvider
说明:ActivityThread.java是app里的一个实例,在main里创建了一个thread,在getProvider里 holder = ActivityManagerNative.getDefault().getContentProvider(getApplicationThread(), name);进行了判断
5、Checkin.java:updateStats函数:
说明:update statistics in the database,会对PHONE_GPRS_ATTEMPTED进行判断,在emulator里会报Can't update stat PHONE_GPRS_ATTEMPTED: java.lang.IllegalArgumentException: Unknown URL content://android.server.checkin/stats的错误
6、MobileDataStateTracker.java:MobileDataStateReceiver
说明:处理在onReceive里,state= CONNECTED, old= CONNECTING
7、NetworkStateTracker.java:setDetailedState()函数
说明:状态:old =CONNECTING and new state=CONNECTED
8、ConnectivityService.java:handleMessage()函数:
说明:状态:CONNECTED/CONNECTED
9、NetworkStateTracker.java:updateNetworkSettings函数
说明:该函数从Network TCP buffer读取network设置参数,并设置网络
10、ConnectivityService.java:handleDnsConfigurationChange
说明:从dnsList里读取预设的dns 通过writePidDns设置dns