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

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

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

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

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

Settings.java:

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



1. ......  
2. else if (preference == mButtonDataEnabled) {  
3. if (DBG) log("onPreferenceTreeClick: preference == mButtonDataEnabled.");  
4.             ConnectivityManager cm =  
5.                     (ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE);  
6.   
7.             cm.setMobileDataEnabled(mButtonDataEnabled.isChecked());  
8. return true;  
9.         }  
10. ......

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

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

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

setMobileDataEnable()方法代码如下:


1. IConnectivityManager mService;  
2. ......  
3. public ConnectivityManager(IConnectivityManager service) {  
4. if (service == null) {  
5. throw new IllegalArgumentException(  
6. "ConnectivityManager() cannot be constructed with null service");  
7.         }  
8.         mService = service;  
9.     }  
10. ......  
11. public void setMobileDataEnabled(boolean enabled) {  
12. try {  
13.             mService.setMobileDataEnabled(enabled);  
14. catch (RemoteException e) {  
15.         }  
16.     }

这里我们要知道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层了。



 



所调用的框架图:

 

android modem拨号 android拨号上网_frameworks