Android SoftAp支持 (一)

Softap字面意思是用軟件實現AP的功能,讓你的移動設備能夠做爲一個路由,讓別的站點連接。好比讓別人的手機連上你的已經打開AP功能的手機,玩聯機遊戲或者上網等等 java

但事實上此功能是須要硬件以及驅動的支持才能真正的實現的。 android

Softap打開流程。 c++

        在Android系統的Setting界面的wireless配置項中會看到一個「Portable Wi-Fi hotspot」 跟一個"Configure Wi-Fi hotspot setting"選項,能夠進入系統配置AP的名稱,加密方式,密碼等。 以下圖 安全

        當你作完這些設置,系統接受的AP設置界面變化打開的響應,今後開啓了整個Android SoftAP的序幕。得意 網絡

首先./packages/apps/Settings/src/com/android/settings/TetherSettings.java 的onPreferenceChange 函數接收到Softap狀態改變信息 app

[java]  view plain copy print ?
  1. public boolean onPreferenceChange(Preference preference, Object value) {  
  2.     boolean enable = (Boolean) value;  
  3.   
  4.     if (enable) {  
  5.         startProvisioningIfNecessary(WIFI_TETHERING);  
  6.     } else {  
  7.         mWifiApEnabler.setSoftapEnabled(false);  
  8.     }  
  9.     return false;  
  10. }  

 Softap開啓時,enable 爲真,於是執行startProvisioningIfNecessary(WIFI_TETHERING); less

[java]  view plain copy print ?
  1. private void startProvisioningIfNecessary(int choice) {  
  2.     mTetherChoice = choice;  
  3.     if (isProvisioningNeeded()) {  
  4.         Intent intent = new Intent(Intent.ACTION_MAIN);  
  5.         intent.setClassName(mProvisionApp[0], mProvisionApp[1]);  
  6.         startActivityForResult(intent, PROVISION_REQUEST);  
  7.     } else {  
  8.         startTethering();  
  9.     }  
  10. }  

isProvisioningNeeded 用來檢測是否須要進行一些準備工做 socket

若是無需準備工做則執行startTethering  大戲即將上演了 期待ing 大笑 ide

[java]  view plain copy print ?
  1. private void startTethering() {  
  2.     switch (mTetherChoice) {  
  3.         case WIFI_TETHERING:  
  4.             mWifiApEnabler.setSoftapEnabled(true);  
  5.             break;  
  6.         case BLUETOOTH_TETHERING:  
  7.             // turn on Bluetooth first  
  8.             break;  
  9.         case USB_TETHERING:  
  10.             setUsbTethering(true);  
  11.             break;  
  12.         default:  
  13.             //should not happen  
  14.             break;  
  15.     }  
  16. }  

這裏 mTetherChoice == WIFI_TETHERING 因此繼而執行WiFiApEnable.java中的setSoftapEnabled(true)函數 函數

也今後處也跳出了Setting的代碼 跳入了Android WIFI 子系統的framework層

./packages/apps/Settings/src/com/android/settings/wifi/WifiApEnabler.java

[java]  view plain copy print ?
  1. public void setSoftapEnabled(boolean enable) {  
  2.     final ContentResolver cr = mContext.getContentResolver();  
  3.     /** 
  4.      * Disable Wifi if enabling tethering 
  5.      */  
  6.     int wifiState = mWifiManager.getWifiState(); //獲取當前wifi的狀態 若是開啓則關閉且保存狀態信息到變量中  
  7.     if (enable && ((wifiState == WifiManager.WIFI_STATE_ENABLING) ||  
  8.                 (wifiState == WifiManager.WIFI_STATE_ENABLED))) {  
  9.         mWifiManager.setWifiEnabled(false);  
  10.         Settings.Global.putInt(cr, Settings.Global.WIFI_SAVED_STATE, 1);  
  11.     }  
  12.   
  13.     if (mWifiManager.setWifiApEnabled(null, enable)) {  
  14.         /* Disable here, enabled on receiving success broadcast */  
  15.         mCheckBox.setEnabled(false);  
  16.     } else {  
  17.         mCheckBox.setSummary(R.string.wifi_error);  
  18.     }  
  19.   
  20.     /** 
  21.      *  If needed, restore Wifi on tether disable 
  22.      */  
  23.     if (!enable) {  
  24.         int wifiSavedState = 0;  
  25.         try {  
  26.             wifiSavedState = Settings.Global.getInt(cr, Settings.Global.WIFI_SAVED_STATE);  
  27.         } catch (Settings.SettingNotFoundException e) {  
  28.             ;  
  29.         }  
  30.         if (wifiSavedState == 1) {  
  31.             mWifiManager.setWifiEnabled(true);  
  32.             Settings.Global.putInt(cr, Settings.Global.WIFI_SAVED_STATE, 0);  
  33.         }  
  34.     }  
  35. }  

上面的代碼中咱們看到了Google人的考慮事情的周全。首先檢測Wifi當前狀態若是正在打開或者已經打開則關閉WIFI並將此狀態記錄下來,以便關閉softap時它能自動恢復到以前打開wifi的狀態。 Android代碼不愧牛X,這些都能想到...  崇拜那些大牛。

這裏調用mWifiManager.setWifiApEnabled(null, enable)    "frameworks/base/wifi/java/android/net/wifi/WifiManager.java"

[java]  view plain copy print ?
  1. public boolean setWifiApEnabled(WifiConfiguration wifiConfig, boolean enabled) {  
  2.     try {  
  3.         mService.setWifiApEnabled(wifiConfig, enabled);  
  4.         return true;  
  5.     } catch (RemoteException e) {  
  6.         return false;  
  7.     }  
  8. }  

轉向服務層的 setWifiApEnabled  "frameworks/base/services/java/com/android/server/WifiService.java"

[java]  view plain copy print ?
  1. public void setWifiApEnabled(WifiConfiguration wifiConfig, boolean enabled) {  
  2.     enforceChangePermission();  
  3.     mWifiStateMachine.setWifiApEnabled(wifiConfig, enabled);  
  4. }  

從而調用到最基礎的也是最重要的Wifi狀態機中的   setWifiApEnabled 實例 其實我真搞不懂爲何Android代碼要嵌套這麼多層去調用,爲了安全、方便... 哪一個牛人解釋一下。

"frameworks/base/wifi/java/android/net/wifi/WifiStateMachine.java"

[java]  view plain copy print ?
  1. public void setWifiApEnabled(WifiConfiguration wifiConfig, boolean enable) {  
  2.     mLastApEnableUid.set(Binder.getCallingUid());  
  3.     if (enable) {  
  4.         /* Argument is the state that is entered prior to load */  
  5.         sendMessage(obtainMessage(CMD_LOAD_DRIVER, WIFI_AP_STATE_ENABLING, 0));  
  6.         sendMessage(obtainMessage(CMD_START_AP, wifiConfig));  
  7.     } else {  
  8.         sendMessage(CMD_STOP_AP);  
  9.         /* Argument is the state that is entered upon success */  
  10.         sendMessage(obtainMessage(CMD_UNLOAD_DRIVER, WIFI_AP_STATE_DISABLED, 0));  
  11.     }  
  12. }  

        發送CMD_LOAD_DRIVER狀態遷移到mDriverLoadingState 加載AP對應的驅動 這裏把WIFI的驅動跟 AP的驅動作了區分,可見SoftAP不單單是軟件實現的,須要硬件驅動的相應支持。

[java]  view plain copy print ?
  1. class DriverLoadingState extends State {  
  2.     @Override  
  3.     public void enter() {  
  4.         new Thread(new Runnable() {  
  5.             public void run() {  
  6.                 mWakeLock.acquire();  
  7.                 //enabling state  
  8.                 switch(message.arg1) {  
  9.                     case WIFI_STATE_ENABLING:  
  10.                         setWifiState(WIFI_STATE_ENABLING);  
  11.                         break;  
  12.                     case WIFI_AP_STATE_ENABLING:  
  13.                         setWifiApState(WIFI_AP_STATE_ENABLING);  
  14.                         break;  
  15.                 }  
  16.   
  17.                 if(mWifiNative.loadDriver()) {  
  18.                     if (DBG) log("Driver load successful");  
  19.                     sendMessage(CMD_LOAD_DRIVER_SUCCESS);  
  20.                 } else {  
  21.                     loge("Failed to load driver!");  
  22.                     switch(message.arg1) {  
  23.                         case WIFI_STATE_ENABLING:  
  24.                             setWifiState(WIFI_STATE_UNKNOWN);  
  25.                             break;  
  26.                         case WIFI_AP_STATE_ENABLING:  
  27.                             setWifiApState(WIFI_AP_STATE_FAILED);  
  28.                             break;  
  29.                     }  
  30.                     sendMessage(CMD_LOAD_DRIVER_FAILURE);  
  31.                 }  
  32.                 mWakeLock.release();  
  33.             }  
  34.         }).start();  
  35.     }  
  36.   
  37.     @Override  
  38.     public boolean processMessage(Message message) {  
  39.         if (DBG) log(getName() + message.toString() + "\n");  
  40.         switch (message.what) {  
  41.             case CMD_LOAD_DRIVER_SUCCESS:  
  42.                 transitionTo(mDriverLoadedState);  
  43.                 break;  
  44.             case CMD_LOAD_DRIVER_FAILURE:  
  45.                 transitionTo(mDriverFailedState);  
  46.                 break;  
  47.             default:  
  48.                 return NOT_HANDLED;  
  49.         }  
  50.         return HANDLED;  
  51.     }  
  52. }  

加載驅動成功後 系統遷移到mDriverLoadedState 狀態

接收到 CMD_START_AP消息  狀態又被遷移至mSoftApStartingState

[java]  view plain copy print ?
  1. class DriverLoadedState extends State {  
  2.     @Override  
  3.     public void enter() {  
  4.         if (DBG) log(getName() + "\n");  
  5.         EventLog.writeEvent(EVENTLOG_WIFI_STATE_CHANGED, getName());  
  6.     }  
  7.     @Override  
  8.     public boolean processMessage(Message message) {  
  9.         if (DBG) log(getName() + message.toString() + "\n");  
  10.         switch(message.what) {  
  11.                 /*  
  12.                 ****** 
  13.                 */  
  14.             case CMD_START_AP:  
  15.                 transitionTo(mSoftApStartingState);  
  16.                 break;  
  17.             default:  
  18.                 return NOT_HANDLED;  
  19.         }  
  20.         return HANDLED;  
  21.     }  
  22. }  

 SoftApStartingState 會檢測上層傳下的參數的有效性並調用startSoftApWithConfig 配置、打開SoftAP

[java]  view plain copy print ?
  1. class SoftApStartingState extends State {  
  2.     @Override  
  3.     public void enter() {  
  4.         if (DBG) log(getName() + "\n");  
  5.         EventLog.writeEvent(EVENTLOG_WIFI_STATE_CHANGED, getName());  
  6.   
  7.         final Message message = getCurrentMessage();  
  8.         if (message.what == CMD_START_AP) {  
  9.             final WifiConfiguration config = (WifiConfiguration) message.obj;  
  10.   
  11.             if (config == null) {  
  12.                 mWifiApConfigChannel.sendMessage(CMD_REQUEST_AP_CONFIG);  
  13.             } else {  
  14.                 mWifiApConfigChannel.sendMessage(CMD_SET_AP_CONFIG, config);  
  15.                 startSoftApWithConfig(config);  
  16.             }  
  17.         } else {  
  18.             throw new RuntimeException("Illegal transition to SoftApStartingState: " + message);  
  19.         }  
  20.     }  
  21.     @Override  
  22.     public boolean processMessage(Message message) {  
  23.         if (DBG) log(getName() + message.toString() + "\n");  
  24.         switch(message.what) {  
  25.             case CMD_LOAD_DRIVER:  
  26.             case CMD_UNLOAD_DRIVER:  
  27.             //....  
  28.             case CMD_STOP_SUPPLICANT:  
  29.             case CMD_START_AP:  
  30.             //....  
  31.         }  
  32.     }  
  33. }  

獲取SoftAp的網絡配置AP名稱 加密方式密碼....

進行系統驅動(硬件)的配置。

[java]  view plain copy print ?
  1. private void startSoftApWithConfig(final WifiConfiguration config) {  
  2.     // start hostapd on a seperate thread  
  3.     new Thread(new Runnable() {  
  4.         public void run() {  
  5.             try {  
  6.                 mNwService.startAccessPoint(config, mInterfaceName);  
  7.             } catch (Exception e) {  
  8.                 loge("Exception in softap start " + e);  
  9.                 try {  
  10.                     mNwService.stopAccessPoint(mInterfaceName);  
  11.                     mNwService.startAccessPoint(config, mInterfaceName);  
  12.                 } catch (Exception e1) {  
  13.                     loge("Exception in softap re-start " + e1);  
  14.                     sendMessage(CMD_START_AP_FAILURE);  
  15.                     return;  
  16.                 }  
  17.             }  
  18.             if (DBG) log("Soft AP start successful");  
  19.             sendMessage(CMD_START_AP_SUCCESS);  
  20.         }  
  21.     }).start();  
  22. }  
  23.  //...  

這裏調用到了"frameworks/base/services/java/com/android/server/NetworkManagementService.java" 中的startAccessPoint函數

函數以下:

[java]  view plain copy print ?
  1. public void startAccessPoint(  
  2.         WifiConfiguration wifiConfig, String wlanIface) {  
  3.     mContext.enforceCallingOrSelfPermission(CONNECTIVITY_INTERNAL, TAG);  
  4.     try {  
  5.         wifiFirmwareReload(wlanIface, "AP");  
  6.         if (wifiConfig == null) {  
  7.             mConnector.execute("softap""set", wlanIface);  
  8.         } else {  
  9.             mConnector.execute("softap""set", wlanIface, wifiConfig.SSID,  
  10.                     getSecurityType(wifiConfig), wifiConfig.preSharedKey);  
  11.         }  
  12.         mConnector.execute("softap""startap");  
  13.     } catch (NativeDaemonConnectorException e) {  
  14.         throw e.rethrowAsParcelableException();  
  15.     }  
  16. }  

一、下載AP對應的 firmware

wifiFirmwareReload(wlanIface, "AP");

二、設置ap的ssid 加密方式 以及密碼

mConnector.execute("softap", "set", wlanIface, wifiConfig.SSID, getSecurityType(wifiConfig), wifiConfig.preSharedKey);

三、運行softap

mConnector.execute("softap", "startap");

 這裏經過一個NativeDaemonConnector的實例mConnector 調用c++程序 具體的實現我是沒看懂 可是知道最後實際調用的函數, 想深刻了解能夠找一些其餘的資料看

實際調用到了 "./system/netd/CommandListener.cpp" 中的CommandListener::SoftapCmd::runCommand

[java]  view plain copy print ?
  1. int CommandListener::SoftapCmd::runCommand(SocketClient *cli,  
  2.                                         int argc, char **argv) {  
  3.     int rc = 0, flag = 0;  
  4.     char *retbuf = NULL;  
  5.   
  6.     if (argc < 2) {  
  7.         cli->sendMsg(ResponseCode::CommandSyntaxError, "Softap Missing argument"false);  
  8.         return 0;  
  9.     }  
  10.   
  11.     if (!strcmp(argv[1], "startap")) {  
  12.         rc = sSoftapCtrl->startSoftap();  
  13.     } else if (!strcmp(argv[1], "stopap")) {  
  14.         rc = sSoftapCtrl->stopSoftap();  
  15.     } else if (!strcmp(argv[1], "fwreload")) {  
  16.         rc = sSoftapCtrl->fwReloadSoftap(argc, argv);  
  17.     } else if (!strcmp(argv[1], "clients")) {  
  18.         rc = sSoftapCtrl->clientsSoftap(&retbuf);  
  19.         if (!rc) {  
  20.             cli->sendMsg(ResponseCode::CommandOkay, retbuf, false);  
  21.             free(retbuf);  
  22.             return 0;  
  23.         }  
  24.     } else if (!strcmp(argv[1], "status")) {  
  25.         asprintf(&retbuf, "Softap service %s",  
  26.                  (sSoftapCtrl->isSoftapStarted() ? "started" : "stopped"));  
  27.         cli->sendMsg(ResponseCode::SoftapStatusResult, retbuf, false);  
  28.         free(retbuf);  
  29.         return 0;  
  30.     } else if (!strcmp(argv[1], "set")) {  
  31.         rc = sSoftapCtrl->setSoftap(argc, argv);  
  32.     } else {  
  33.         cli->sendMsg(ResponseCode::CommandSyntaxError, "Softap Unknown cmd"false);  
  34.         return 0;  
  35.     }  
  36.   
  37.     if (!rc) {  
  38.         cli->sendMsg(ResponseCode::CommandOkay, "Softap operation succeeded"false);  
  39.     } else {  
  40.         cli->sendMsg(ResponseCode::OperationFailed, "Softap operation failed"true);  
  41.     }  
  42.   
  43.     return 0;  
  44. }  

首先是"set「 命令, 調用到c = sSoftapCtrl->setSoftap(argc, argv); 來配置網絡

配置即將全部上層的網絡設置寫到HOSTAPD_CONF_FILE[]    = "/data/misc/wifi/hostapd.conf" 中

("system/netd/SoftapController.cpp")

[java]  view plain copy print ?
  1. /* 
  2.  * Arguments: 
  3.  *  argv[2] - wlan interface 
  4.  *  argv[3] - SSID 
  5.  *  argv[4] - Security 
  6.  *  argv[5] - Key 
  7.  *  argv[6] - Channel 
  8.  *  argv[7] - Preamble 
  9.  *  argv[8] - Max SCB 
  10.  */  
  11. int SoftapController::setSoftap(int argc, char *argv[]) {  
  12.     char psk_str[2*SHA256_DIGEST_LENGTH+1];  
  13.     int ret = 0, i = 0, fd;  
  14.     char *ssid, *iface;  
  15.   
  16.     /* ..... */  
  17.     iface = argv[2];  
  18.   
  19.     char *wbuf = NULL;  
  20.     char *fbuf = NULL;  
  21.   
  22.     if (argc > 3) {  
  23.         ssid = argv[3];  
  24.     } else {  
  25.         ssid = (char *)"AndroidAP";  
  26.     }  
  27.   
  28.     if (argc > 4) {  
  29.         if (!strcmp(argv[4], "wpa-psk")) {  
  30.             generatePsk(ssid, argv[5], psk_str);  
  31.             asprintf(&fbuf, "%swpa=1\nwpa_pairwise=TKIP CCMP\nwpa_psk=%s\n", wbuf, psk_str);  
  32.         } else if (!strcmp(argv[4], "wpa2-psk")) {  
  33.             generatePsk(ssid, argv[5], psk_str);  
  34.             asprintf(&fbuf, "%swpa=2\nrsn_pairwise=CCMP\nwpa_psk=%s\n", wbuf, psk_str);  
  35.         } else if (!strcmp(argv[4], "open")) {  
  36.             asprintf(&fbuf, "%s", wbuf);  
  37.         }  
  38.     } else {  
  39.         asprintf(&fbuf, "%s", wbuf);  
  40.     }  
  41.   
  42.     fd = open(HOSTAPD_CONF_FILE, O_CREAT | O_TRUNC | O_WRONLY | O_NOFOLLOW, 0660);  
  43.     /*............*/  
  44.     if (write(fd, fbuf, strlen(fbuf)) < 0) {  
  45.         ALOGE("Cannot write to \"%s\": %s", HOSTAPD_CONF_FILE, strerror(errno));  
  46.         ret = -1;  
  47.     }  
  48.     free(wbuf);  
  49.     free(fbuf);  
  50.   
  51.     /* Note: apparently open can fail to set permissions correctly at times */  
  52.     // .......  
  53. }  

而後是"startap"命令調用rc = sSoftapCtrl->startSoftap(); 真正開啓Softap 

[java]  view plain copy print ?
  1. int SoftapController::startSoftap() {  
  2.     pid_t pid = 1;  
  3.     int ret = 0;  
  4.   
  5.     if (mPid) {  
  6.         ALOGE("Softap already started");  
  7.         return 0;  
  8.     }  
  9.     if (mSock < 0) {  
  10.         ALOGE("Softap startap - failed to open socket");  
  11.         return -1;  
  12.     }  
  13.   
  14.     if ((pid = fork()) < 0) {  
  15.         ALOGE("fork failed (%s)", strerror(errno));  
  16.         return -1;  
  17.     }  
  18.   
  19.     if (!pid) {  
  20.         ensure_entropy_file_exists();  
  21.         if (execl("/system/bin/hostapd""/system/bin/hostapd",  
  22.                   "-e", WIFI_ENTROPY_FILE,  
  23.                   HOSTAPD_CONF_FILE, (char *) NULL)) {  
  24.             ALOGE("execl failed (%s)", strerror(errno));  
  25.         }  
  26.         ALOGE("Should never get here!");  
  27.         return -1;  
  28.     } else {  
  29.         mPid = pid;  
  30.         ALOGD("Softap startap - Ok");  
  31.         usleep(AP_BSS_START_DELAY);  
  32.     }  
  33.     return ret;  
  34.   
  35. }  

在startSoftap函數中調用了

execl("/system/bin/hostapd", "/system/bin/hostapd", "-e", WIFI_ENTROPY_FILE, HOSTAPD_CONF_FILE, (char *) NULL)

這裏hostapd就是softap的deamon 程序 相似於wifi的的wpa_supplicant

 

至此全部wifi子系統從界面打開softap 到如何運行調用到deamon程序打開Softap的流程就是這樣的

以後會介紹到Setting 界面"Portable Wi-Fi"的開啓 以及  Hostapd 的一些東東 

相關文章
相關標籤/搜索