android 上网流程分析,Android 2.3 拨号上网流程分析

通常,如果我们想使用SIM卡拨号上网功能,我们要在设置中进行简单的配置,步骤如下:

设置 -》无线和网络 -》移动网络 -》(已启用数据/数据漫游/接入点名称/仅使用2G网络/网络运营商)

我们必须选中其中的“已启用数据”选项,然后配置接入点名称后就可以上网了,当然有的设置中已经根据你的SIM卡类型默认设置了接入点,这时候你只选择“已启用数据”项后就可以完成上网功能设置。

这些设置步骤究竟做了哪些事情呢?我们现在就从源码的角度进行分析。

1. 首先,我们找到“移动网络”的设置UI-------Settings.java(/packages/apps/Phone/src/com/android/phone/Settings.java)

Settings.java:

"已启用数据"选项的相关代码如下:

......

elseif(preference == mButtonDataEnabled) {

if(DBG) log("onPreferenceTreeClick: preference == mButtonDataEnabled.");

ConnectivityManager cm =

(ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE);

cm.setMobileDataEnabled(mButtonDataEnabled.isChecked());

returntrue;

}

...... 代码中,我们得到一个ConnectivityManager对象,并调用该对象的setMobileDataEnable(boolean b)方法,根据传入的参数进行设置,我们看一下ConnectivityManager类。

2. ConnectivityManager.java(/frameworks/base/core/java/android/net/ConnectivityManager.java)

这个时候,数据已经进入frameworks层。

setMobileDataEnable()方法代码如下:

IConnectivityManager mService;

......

publicConnectivityManager(IConnectivityManager service) {

if(service ==null) {

thrownewIllegalArgumentException(

"ConnectivityManager() cannot be constructed with null service");

}

mService = service;

}

......

publicvoidsetMobileDataEnabled(booleanenabled) {

try{

mService.setMobileDataEnabled(enabled);

} catch(RemoteException e) {

}

} 这里我们要知道IConnectivityManager类,是根据IConnectivityManager.aidl接口自动生成的一个java类,而我们自己有一个Service则继承了该类的内部类:Stub,在我们自己为拨号上网实现的这个Service就是ConnectivityService,所以根据AIDL只是,我们知道,代码中的mService其实就是ConnectivityService类的对象,所以代码在这里实际上是调用了ConnectivityService对象的setMobileDataEnable()方法

接着之后的流程为:

frameworks\base\services\java\com\android\server\ ConnectivityService.java

->[MyHandler ->handleMessage ->case EVENT_SET_MOBILE_DATA ->handleSetMobileData]

frameworks\base\services\java\com\android\server\ ConnectivityService.java

->[mNetTrackers[ConnectivityManager.TYPE_MOBILE].reconnect()]

frameworks/base/core/java/android/net/MobileDataStateTracker.java

extends NetworkStateTracker

->[reconnect()]

frameworks/base/core/java/android/net/MobileDataStateTracker.java

->[setEnableApn]

..\base\telephony\java\com\android\internal\telephony\ITelephony.java

->[enableApnType]

packages\apps\phone\src\com\android\phone\

PhoneInterfaceManager.java   (extends ITelephony.Stub  )  实现

->[enableApnType-> mPhone.enableApnType(type)]

{那么这个mPhone是哪里来的,通过源代码我们发现在PhoneInterfaceManger的构造函数传进去的。在PhoneApp.java 的OnCreate 函数里创建了一个PhoneInterfaceManger对象, PhoneFactory.makeDefaultPhones(this);

phone = PhoneFactory.getDefaultPhone();  也就是说这里的Phone要么是CDMAPhone 的实例要么是GSMPhone 的实例,因为CDMAPhone extends PhoneBase,GSMPhone  extends PhoneBase.}

frameworks\base\telephony\java\com\android\internal\telephony\

PhoneBase.java  extends Handler  implements  Phone

->[enableApnType]

frameworks\base\telephony\java\com\android\internal\telephony\

DataConnectionTracker.java

->[enableApnType() -> setEnabled() -> handleMessage -> case EVENT_ENABLE_NEW_APN -> onEnableApn()]

{这里稍微解释下, CDMAPhone ,GSMPhone  都是继承 PhoneBase. 而DataConnectionTracker 是PhoneBase 的一个成员字段,该对象的初始化分别是在CDMAPhone ,GSMPhone 类里,对应CdmaDataConnectionTracker, GsmDataConnectionTracker ,因为此二类分别继承DataConnectionTracker,所以具体是调用那个类的函数,要根据前面传进的是CDMAPhone 或者GSMPhone ,这个流程以GSMPhone 示例。}

frameworks\base\telephony\java\com\android\internal\telephony\gsm\

GsmDataConnectionTracker .java       extends DataConnectionTracker

->[onEnableNewApn() -> cleanUpConnection() -> conn.disconnect(obtainMessage(EVENT_DISCONNECT_DONE, reason))]

frameworks\base\telephony\java\com\android\internal\telephony\

DataConnectionTracker.java

->[handleMessage ->case EVENT_DISCONNECT_DONE -> onDisconnectDone()]

frameworks\base\telephony\java\com\android\internal\telephony\gsm\

GsmDataConnectionTracker.java

->[onDisconnectDone() -> trySetupData(reason) -> setupData(reason)]

frameworks\base\telephony\java\com\android\internal\telephony\

DataConnection.java

->[processMessage -> case EVENT_CONNECT -> onConnect(cp)]

frameworks\base\telephony\java\com\android\internal\telephony\gsm\

GsmDataConnection .java

->[phone.mCM.setupDataCall()]

{phone.mCM  类型为CommandsInterface,也是根据CMDAPhone 或者GSMPhone 决定的,那么两者都是RIL 的实例,RIL implements CommandsInterface , 其实是根据RIL的构造函数,根据参数int networkMode 来决定。 注意:如果是Sip  就另当别论了,那么phone.mCM  就是通过SipPhoneBase  构造函数传给父类PhoneBase的构造函数,参数CommandsInterface,其实现 SipCommandInterface ,因为SipPhoneBase implements Phone, SipCommandInterface implements CommandsInterface }

RIL

->[setupDataCall()]

到这里,  java层的pppd请求就通过RIL发到C层了。

所调用的框架图:

0818b9ca8b590ca3270a3433284dd417.png

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值