Android 2.3 拨号上网流程从源码角度进行分析

通常,如果我们想使用SIM卡拨号上网功能,我们要在设置中进行简单的配置,步骤如下
设置 -》无线和网络 -》移动网络 -》(已启用数据/数据漫游/接入点名称/仅使用2G网络/网络运营商)
我们必须选中其中的“已启用数据”选项,然后配置接入点名称后就可以上网了,当然有的设置中已经根据你的SIM卡类型默认设置了接入点,这时候你只选择“已启用数据”项后就可以完成上网功能设置。
这些设置步骤究竟做了哪些事情呢?我们现在就从源码的角度进行分析。

1. 首先,我们找到“移动网络”的设置UI-------Settings.java(/packages/apps/Phone/src/com/android/phone/Settings.java)
Settings.java:
"已启用数据"选项的相关代码如下:


代码如下:

......
else if (preference == mButtonDataEnabled) {
if (DBG) log("onPreferenceTreeClick: preference == mButtonDataEnabled.");
ConnectivityManager cm =
(ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE);
cm.setMobileDataEnabled(mButtonDataEnabled.isChecked());
return true;
}
......

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

2. ConnectivityManager.java(/frameworks/base/core/java/android/net/ConnectivityManager.java)
这个时候,数据已经进入frameworks层。
setMobileDataEnable()方法代码如下:


代码如下:

IConnectivityManager mService;
......
 public ConnectivityManager(IConnectivityManager service) {
        if (service == null) {
            throw new IllegalArgumentException(
                "ConnectivityManager() cannot be constructed with null service");
        }
        mService = service;
    }
......
public void setMobileDataEnabled(boolean enabled) {
try {
mService.setMobileDataEnabled(enabled);
} catch (RemoteException e) {
}
}

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

3. ConnectivityService.java(/frameworks/./base/services/java/com/android/server/ConnectivityService.java)
setMobileDataEnable()方法代码如下:


代码如下:

public void setMobileDataEnabled(boolean enabled) {
enforceChangePermission();
if (DBG) Slog.d(TAG, "setMobileDataEnabled(" + enabled + ")");
mHandler.sendMessage(mHandler.obtainMessage(EVENT_SET_MOBILE_DATA,
(enabled ? ENABLED : DISABLED), 0));
}

这里发送了一个消息出去,mHandler收到该消息以后:


代码如下:

case EVENT_SET_MOBILE_DATA:
{
boolean enabled = (msg.arg1 == ENABLED);
handleSetMobileData(enabled);
break;
}

收到该消息后,调用handleSetMobileData()方法:


代码如下:

private NetworkStateTracker mNetTrackers[];
......
private void handleSetMobileData(boolean enabled) {
        ......
if (enabled) {
if (mNetTrackers[ConnectivityManager.TYPE_MOBILE] != null) {
if (DBG) {
Slog.d(TAG, "starting up " + mNetTrackers[ConnectivityManager.TYPE_MOBILE]);
}
mNetTrackers[ConnectivityManager.TYPE_MOBILE].reconnect();
}
            ......
}
 }

如果“已启用数据”选项已经选择,那这个时候传进来的参数“enabled”应该是“true”,所以会处理代码中if语句块,即执行:


代码如下:

mNetTrackers[ConnectivityManager.TYPE_MOBILE].reconnect();

而在ConnectivityManager中,TYPE_MOBILE 为 0,所以这里相当于调用了


代码如下:

mNetTracker[0].reconnect()

但是,NetworkStateTracker是一个抽象类,所以具体的事情要交给它的子类MobileDataStateTracker.java来干。

4. MobileDataStateTracker.java(/frameworks/base/core/java/android/net/MobileDataStateTracker.java)
该类包含多种数据连接,包括MMS,SUPL,DUN等,
在MobileDataStateTracker.java里面的调用流程是这样的:


代码如下:

<PRE class=java name="code">mPhoneService = ITelephony.Stub.asInterface(ServiceManager.getService("phone"));</PRE>......<BR>
reconnect->mPhoneService.enableApnType(apnType);<P></P>
<PRE></PRE>
mPhoneService是电话的服务的客户端,它的server端实际上是PhoneInterfaceManager对象
<P></P>
<P>5. PhoneInterfaceManager.java(/packages/apps/Phone/src/com/android/phone/PhoneInterfaceManager.java)<BR>
</P>
<P>看PhoneInterfaceManager的enableApnType方法:</P>
<P><PRE class=java name="code"> public int enableApnType(String type) {
enforceModifyPermission();
return mPhone.enableApnType(type);
}
</PRE><P></P>
这样,就将连接apn的请求发送到telephony框架层下去了。apn在设置应用里面有指定,一般在你的工程目录下的system/etc/apns-conf.xml文件<BR>
<BR>
<P>6. 上面的mPhone是PhoneProxy对象,</P>
<P>调用流程:</P>
<P>PhoneProxy.java:<BR>
</P>
<P><PRE class=java name="code">mActivePhone.enableApnType(type)</PRE>mActivePhone是GSMPhone或者CDMAPhone的上溯接口PhoneBase对象<BR>
<P></P>
<P>PhoneBase.java:</P>
<P><PRE class=java name="code">mDataConnection.enableApnType(type);</PRE><P></P>
<P>调用到 DataConnectionTracker的enableApnType方法</P>
<P>DataConnectionTracker.java:<BR>
</P>
<P>enableApnType(String type)->setEnabled->onEnableApn->onEnableNewApn<BR>
</P>
<BR>
<P>onEnableNewApn方法在DataConnectionTracker的派生类GsmDataConnectionTracker和CdmaDataConnectionTracker中实现,从而区别不同类型PHONE的数据连接流程。<BR>
</P>
<P>以GSM为例,调用流程:onEnableNewApn->cleanUpConnection->conn.disconnect<BR>
<BR>
</P>
conn是DataConnection对象,标识一钟数据连接,可以看出这里实际上实现了一个数据连接的状态机。<BR>
<P>在DataConnection对象里面数据连接的状态分为:</P>
<P><PRE class=java name="code">DcDefaultState,默认状态。
DcInactiveState,非激活状态。
DcActivatingState,正在激活状态
DcActiveState,激活状态
DcDisconnectingState,正在断开状态
DcDisconnectingBadDnsState,断开状态(因为错误的DNS)
</PRE><P></P>
<P>连接成功以后,notifyDefaultData调用到DefaultPhoneNotifier的notifyDataConnection方法。</P>
<P>DefaultPhoneNotifier是ITelephonyRegistry接口的客户端,其服务端是TelephonyRegistry(com.android.server.TelephonyRegistry)</P>
<P>TelephonyRegistry的notifyDataConnection方法调用如下语句<BR>
<PRE class=java name="code"> r.callback.onDataConnectionStateChanged(state, networkType);</PRE><P></P>
<P>r是当前mRecords中的元素,包含有IPhoneStateListener接口的实现callback,TelephonyRegistry中的每个调用都会遍历mRecords中的元素,如果某个元素注册了对应接听,</P>
<P>则调用callback的某个函数。</P>
<P>客户端通过如下方式调用取得电话状态的监听, 以StatusBarPolicy.java中的mPhoneStateListener为例:</P>
<P>            ((TelephonyManager)mContext.getSystemService(Context.TELEPHONY_SERVICE))</P>
<P>                    .listen(mPhoneStateListener,<BR>
                              PhoneStateListener.LISTEN_SERVICE_STATE<BR>
                            | PhoneStateListener.LISTEN_SIGNAL_STRENGTHS<BR>
                            | PhoneStateListener.LISTEN_CALL_STATE<BR>
                            | PhoneStateListener.LISTEN_DATA_CONNECTION_STATE<BR>
                            | PhoneStateListener.LISTEN_DATA_ACTIVITY);<BR>
</P>
<P>mPhoneStateListener是PhoneStateListener实例,PhoneStateListener实现了IPhoneStateListener接口,假如你继承PhoneStateListener子类,首先你要确定你感兴趣的监听</P>
<P>事件,然后重写对应的方法。再像上面那样调用listen方法就可以了。</P>
<P>TelephonyRegistry的方法、监听动作、已经你要重写的方法对应关系如下:</P>
<P>TelephonyRegistry的方法  ---------------------监听动作-------------------------------------------------------PhoneStateListener子类中的中的回调<BR>
</P>
<P>notifyServiceState   ---------- PhoneStateListener.LISTEN_SERVICE_STATE       -----------------  public void onServiceStateChanged(ServiceState state) <BR>
</P>
<P>notifySignalStrength   ------- PhoneStateListener.LISTEN_SIGNAL_STRENGTHS     --------- --  public void onSignalStrengthsChanged(SignalStrength signalStrength)<BR>
</P>
<P>notifyCallState  ---------------- PhoneStateListener.LISTEN_CALL_STATE    -------------------------   public void onCallStateChanged(int state, String incomingNumber)<BR>
</P>
<P>notifyDataConnection ------- PhoneStateListener.LISTEN_DATA_CONNECTION_STATE    ---   public void onDataConnectionStateChanged(int state, int networkType)<BR>
</P>
<P>notifyDataActivity  -------------- PhoneStateListener.LISTEN_DATA_ACTIVITY -----------------------   public void onDataActivity(int direction)<BR>
</P>
<P>。。。。。。。。</P>
<P>因此整个调用链是:DefaultPhoneNotifier:notifyDataConnection ---------》 TelephonyRegistry :notifyDataConnection---------》</P>
<P>PhoneStateListener.callback:onDataConnectionStateChanged --------------》PhoneStateListener子类的onDataConnectionStateChanged</P>
<P>除此之外,TelephonyRegistry还发出一个ACTION_ANY_DATA_CONNECTION_STATE_CHANGED,包含数据连接的详细信息。</P>
<P><BR>
而Mobile Data Service里面的MobileDataStateTracker会接收到这个动作,由它的BoadcastReceiver类MobileDataStateReceiver提取出数据连接的信息,然后设置好状态</P>
<PRE class=java name="code">setDetailedState(DetailedState.CONNECTING, reason, apnName);
</PRE>
<P>MobileDataStateTracker根据状态变化给ConnectivityService发送EVENT_STATE_CHANGED消息。</P>
<P>ConnectivityService调用handleConnect去执行相关炒作,包括关闭优先级比它低的数据连接,更新状态栏等等。<BR>
</P>
<P>还有很多地方还没有搞明白,以后再续。<BR>
</P>
<P><BR>
</P>
<P><BR>
</P>
<BR>
<P><BR>
<BR>
</P>

(0)

相关推荐

  • Android设备间实现蓝牙(Bluetooth)共享上网

    Android设备之间可以除了通过wifi热点共享上网,还可以通过蓝牙共享上网,后面这个功能很少人使用,但适合某台设备没有wifi却有蓝牙的情况. 一.设置WT19i,系统设置>无线连接>网络共享>开启蓝牙共享网络(这步很多人忽略,导致无法上网) 二.开启N7 二代蓝牙并配对,返回WT19i,已配对设备>配置>开启互联网连接共享 三.设置N7 二代,已配对设备>配置>开启互联网访问(第二步主机共享没开启的话这里也无法开启) 测试效果良好,访问网络正常,Androi

  • Android 判断是否能真正上网的实例详解

    Android 判断是否能真正上网的实例详解 检测网络是否连接 实现代码: /** * 检测网络是否连接 * * @return */ private boolean isNetworkAvailable() { // 得到网络连接信息 ConnectivityManager manager = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE); // 去进行判断网络是否连接 if (manager.getA

  • Android 2.3 拨号上网流程从源码角度进行分析

    通常,如果我们想使用SIM卡拨号上网功能,我们要在设置中进行简单的配置,步骤如下: 设置 ->无线和网络 ->移动网络 ->(已启用数据/数据漫游/接入点名称/仅使用2G网络/网络运营商) 我们必须选中其中的"已启用数据"选项,然后配置接入点名称后就可以上网了,当然有的设置中已经根据你的SIM卡类型默认设置了接入点,这时候你只选择"已启用数据"项后就可以完成上网功能设置. 这些设置步骤究竟做了哪些事情呢?我们现在就从源码的角度进行分析. 1. 首先

  • Android用于加载xml的LayoutInflater源码超详细分析

    1.在view的加载和绘制流程中:文章链接 我们知道,定义在layout.xml布局中的view是通过LayoutInflate加载并解析成Java中对应的View对象的.那么具体的解析过程是哪样的. 先看onCreate方法,如果我们的Activity是继承自AppCompactActivity.android是通过getDelegate返回的对象setContentView,这个mDelegate 是AppCompatDelegateImpl的实例. @Override protected

  • Android okhttp的启动流程及源码解析

    前言 这篇文章主要讲解了okhttp的主要工作流程以及源码的解析. 什么是OKhttp 简单来说 OkHttp 就是一个客户端用来发送 HTTP 消息并对服务器的响应做出处理的应用层框架. 那么它有什么优点呢? 易使用.易扩展. 支持 HTTP/2 协议,允许对同一主机的所有请求共用同一个 socket 连接. 如果 HTTP/2 不可用, 使用连接池复用减少请求延迟. 支持 GZIP,减小了下载大小. 支持缓存处理,可以避免重复请求. 如果你的服务有多个 IP 地址,当第一次连接失败,OkHt

  • 从源码角度分析Android的消息机制

    前言 说到Android的消息机制,那么主要的就是指的Handler的运行机制.其中包括MessageQueue以及Looper的工作过程. 在开始正文之前,先抛出两个问题: 为什么更新UI的操作要在主线程中进行? Android中为什么主线程不会因为Looper.loop()里的死循环卡死? UI线程的判断是在ViewRootImpl中的checkThread方法中完成的. 对于第一个问题,这里给一个简单的回答: 如果可以在子线程中修改UI,多线程的并发访问可能会导致UI控件的不可预期性,采用

  • Android音视频开发Media FrameWork框架源码解析

    目录 一.Media FrameWork背景 二.Media Framework“路线图” 2.1 代理端 2.2 服务端 2.2.1 Source 2.2.2 Decoder 2.2.3 Renderer 2.2.4 Foundation 2.3 OMX端 2.4 Kernel端 三.media播放的流程 四.Media FrameWork源码分析 一.Media FrameWork背景 Media Framework (媒体函数库):此函数库让Android 可以播放与录制许多常见的音频与视

  • Android管理与操作Wifi简单实例源码

    因为需要一直在弄网络的问题,今天看了一下Wifi的操作,经过整理,做出来了一个类,可能不全,但是个人感觉已经完全能够满足需要了,当然,里面的方法也有可能是错误的或者是不全的,这个类我没有进行完整的测试,只测试了其中的一些方法. 其实操作Wifi也是很简单的,主要使用以下几个对象或变量: private WifiManager wifiManager;// 声明管理对象OpenWifi private WifiInfo wifiInfo;// Wifi信息 private List<ScanRes

  • Android getJSONObject与optJSONObject的区别结合源码分析

    Android getJSONObject与optJSONObject的区别结合源码分析 json解析常见问题: getJSONObject与optJSONObject的区别,下面结合源码和案例来分析当我们使用这两周方法来解析数据时,哪种比较好. 源码分析: //使用getJSONObject时,如果返回的对象不是JSONObject,抛出JSONException异常 /** * Returns the value mapped by {@code name} if it exists and

  • Spring启动流程refresh()源码深入解析

    一.Spring容器的refresh() spring  version:4.3.12  ,尚硅谷Spring注解驱动开发-源码部分 //refresh():543, AbstractApplicationContext (org.springframework.context.support) public void refresh() throws BeansException, IllegalStateException { synchronized (this.startupShutdo

  • Tomcat的类加载机制流程及源码解析

    目录 前言 1.Tomcat 的类加载器结构图: 2.Tomcat 的类加载流程说明: 3.源码解析: 4.为什么tomcat要实现自己的类加载机制: 前言 在前面 Java虚拟机:对象创建过程与类加载机制.双亲委派模型 文章中,我们介绍了 JVM 的类加载机制以及双亲委派模型,双亲委派模型的类加载过程主要分为以下几个步骤: (1)初始化 ClassLoader 时需要指定自己的 parent 是谁 (2)先检查类是否已经被加载过,如果类已经被加载了,直接返回 (3)若没有加载则调用父加载器 p

  • Netty分布式客户端接入流程初始化源码分析

    目录 前文概述: 第一节:初始化NioSockectChannelConfig 创建channel 跟到其父类DefaultChannelConfig的构造方法中 再回到AdaptiveRecvByteBufAllocator的构造方法中 继续跟到ChannelMetadata的构造方法中 回到DefaultChannelConfig的构造方法 前文概述: 之前的章节学习了server启动以及eventLoop相关的逻辑, eventLoop轮询到客户端接入事件之后是如何处理的?这一章我们循序渐

随机推荐