1. 程式人生 > >8 Android平臺開發-WIFI 驅動移植 -- 詳細

8 Android平臺開發-WIFI 驅動移植 -- 詳細

一、WIFI的基本架構(程式碼路徑)

    1、WIFI Settings應用程式:
       packages/apps/Settings/src/com/android/settings/wifi/
    2、JAVA部分(framework):
         frameworks/base/services/java/com/android/server/
         frameworks/base/wifi/java/android/net/wifi/
    3、JNI部分:
         frameworks/base/core/jni/android_net_wifi_Wifi.cpp
    4、wpa_supplicant的介面卡 部分
         hardware/libhardware_legary/wifi/主要是wifi.c。     5、wifi使用者空間的程式和庫:
         external/wpa_supplicant/ 和 external/wap_supplicant_6/ 我不清楚是哪一個目錄
         生成庫libwpaclient.so和 守護程序wpa_supplicant。
    6、kernel 驅動 二、WIFI在Android中如何工作    Android使用一個修改版wpa_supplicant作為daemon來控制WIFI,程式碼位於external/wpa_supplicant。wpa_supplicant是通過socket與
hardware/libhardware_legacy/wifi/wifi.c通訊。UI(APP)通過android.net.wifi package(frameworks/base/wifi/java/android/net/wifi/)傳送 命令 給wifi.c。相應的JNI實現位於frameworks/base/core/jni/android_net_wifi_Wifi.cpp。更高一級的網路管理位於frameworks/base/core/java/android/net。   
三、配置Android支援WIFI
   *device/rootfs-project/BoardConfig.mk中新增: WPA_SUPPLICANT_VERSION := VER_0_8_X    //wpa_supplicant的版本
WIFI_DRIVER := ar6003//驅動名字(自己定義的巨集),主要在hardware/平臺名稱/wlan/晶片名/Android.mk 檔案裡使用 BOARD_WPA_SUPPLICANT_PRIVATE_LIB := lib_driver_cmd_wext
BOARD_WPA_SUPPLICANT_DRIVER := WEXT  //驅動型別,決定wap_supplicant的底層介面型別
這將使external/wpa_supplicant/Android.mk設定WPA_BUILD_SUPPLICANT為true預設使用驅動driver_wext.c
如果使用定製的wpa_supplicant驅動(例如 madwifi),可以設定:
       BOARD_WPA_SUPPLICANT_DRIVER := MADWIFI

------- 如果 wifi 具有 SoftAP(即虛擬無線AP) 的功能,需要以下兩個巨集 -------

WIFI_DRIVER_FW_STA_PATH := /system/wifi/fw.bin

WIFI_DRIVER_FW_AP_PATH :=/system/wifi/fw_ap.bin

--------
/* 以下兩項 可以在 hardware/libhardware_legary/wifi/wifi.c 裡邊直接定義 */
WIFI_DRIVER_MODULE_PATH := /system/wifi/ar6000.ko  //wifi 驅動路徑
WIFI_DRIVER_MODULE_NAME := ar6000//驅動名字
WIFI_DRIVER_MODULE_ARG:=DBG=7 //該巨集是用於insmod時傳引數



四、使能wpa_supplicant除錯資訊    預設wpa_supplicant設定為MSG_INFO,為了輸出更多資訊,可修改:
   1、在/external/wpa_supplicant/common.c中設定wpa_debug_level = MSG_DEBUG;
   2、在common.c中把#define wpa_printf巨集中的
      if ((level) >= MSG_INFO)
      改為
      if ((level) >= MSG_DEBUG)
五、修改system/etc/wifi/wpa_supplicant.conf (在原始碼中是修改external/wpa_supplicant/wpa_supplicant.conf)
  
   wpa_supplicant是通過 rootfs-src/external/wpa_supplicant/wpa_supplicant.conf中的ctrl_interface=來指定控制socket的
這個路徑在wifi.c中用到 一般的/external/wpa_supplicant/wpa_supplicant.conf配置為:
 ctrl_interface=DIR=/data/system/wpa_supplicant GROUP=wifi
      update_config=1
      fast_reauth=1  
      eapol_version=1
   有時,驅動需要增加:
      ap_scan=1
   如果遇到AP連線問題,需要修改ap_scan=0來讓驅動連線,代替wpa_supplicant。
  
   如果要連線到non-WPA or open wireless networks,要增加:
      network={
              key_mgmt=NONE
      }
六、配置路徑和許可權    Google修改的wpa_supplicant要執行在wifi使用者和組下的。程式碼可見/external/wpa_supplicant/os_unix.c中的os_program_init()函式   
   如果配置不對,會出現下面錯誤:
      E/WifiHW  (  ): Unable to open connection to supplicant on
      "/data/system/wpa_supplicant/wlan0": No such file or directory will appear.
 
   確認init.rc中有如下配置:
       mkdir /system/etc/wifi 0771 wifi wifi
       chmod 0771 /system/etc/wifi
       chmod 0660 /system/etc/wifi/wpa_supplicant.conf
       chown wifi wifi /system/etc/wifi/wpa_supplicant.conf #wifi的原始配置檔案
       # wpa_supplicant socket
       mkdir /data/system/wpa_supplicant 0771 wifi wifi
       chmod 0771 /data/system/wpa_supplicant #放置wifiinterface的地方

       #wpa_supplicant control socket for android wifi.c
       mkdir /data/misc/wifi 0770 wifi wifi
       mkdir /data/misc/wifi/sockets 0770 wifi wifi  #與上層通過socket通訊的路徑
       chmod 0770 /data/misc/wifi
       chmod 0660 /data/misc/wifi/wpa_supplicant.conf
 #wifi的配置檔案,將由wpa_supplicant根據實際配置寫入該檔案 setprop wifi.interfaceeth0  #intreface名稱設定,這在framework/base/wifi/java/android/net/wifi /WifiStateTracker.java中會用到,以處理dhcp。ar6000.ko用eth0
目錄許可權的處理是為了所有使用者能對下一級進行搜尋,/data/misc/wifi/sockets目錄不僅為wifi擁有者服務,還因為通訊的原因要和其他使用者聯絡,要不然,將會出現Unable to open connection to supplicant on "/data/system/wpa_supplicant/ra0": Connection refused,或permission denied的錯誤。很多人乾脆將上述所有的許可權都設為0777,當然也行,但總覺得有些粗糙。
七、執行wpa_supplicant和dhcpcd

   在init.rc中確保有如下語句:
  service wpa_supplicant /system/bin/wpa_supplicant  -Dwext -iwlan0 -c /data/misc/wifi/wpa_supplicant.conf          user root
         group system wifi 
         socket wpa_wlan0 dgram 0666 wifi wifi   //這項是用於UDP連線的
         disabled          oneshot       service dhcpcd /system/bin/logwrapper /system/bin/dhcpcd -d -B eth0          group system dhcp
         disabled
         oneshot
   根據所用的WIFI驅動名字,修改wlan0為自己驅動的名字
   
八、編譯WIFI驅動為module或kernel built in
   1、編譯為module
      在device/rootfs-project/BoardConfig.mk中新增:
         WIFI_DRIVER_MODULE_PATH :=
"/system/lib/modules/ar6000.ko"  //驅動的具體位置
         WIFI_DRIVER_MODULE_ARG := ""  #for example nohwcrypt
         WIFI_DRIVER_MODULE_NAME := "ar6000"  #for example wlan0 
         WIFI_FIRMWARE_LOADER := ""         
  
   2、編譯為kernel built in  
     1)在hardware/libhardware_legacy/wifi/wifi.c要修改interface名字,
     2)在init.rc中新增:
       
setprop wifi.interface "wlan0"
     3)在hardware/libhardware_legacy/wifi/wifi.c中當insmod/rmmod時,
        直接return 0。

 
九、WIFI需要的firmware
   Android不使用標準的hotplug binary,WIFI需要的firmware要複製到/etc/firmware。
  
   或者複製到WIFI驅動指定的位置,然後WIFI驅動會自動載入。
十、修改WIFI驅動適合Android    Google修改的wpa_supplicant要求SIOCSIWPRIVioctl 傳送 命令 到驅動,及接收資訊,例如signal strength, mac address of the AP, link speed等。所以要正確實現WIFI驅動,需要從  SIOCSIWPRIV ioctl返回RSSI (signal strength)和MACADDR資訊。    如果沒實現這個ioctl,會出現如下錯誤:
     E/wpa_supplicant(  ): wpa_driver_priv_driver_cmd failed

                               wpa_driver_priv_driver_cmd RSSI len = 4096 
     E/wpa_supplicant(  ): wpa_driver_priv_driver_cmd failed 
     D/wpa_supplicant(  ): wpa_driver_priv_driver_cmd LINKSPEED len = 4096
     E/wpa_supplicant(  ): wpa_driver_priv_driver_cmd failed 
     I/wpa_supplicant(  ): CTRL-EVENT-DRIVER-STATE HANGED
   
十一、設定dhcpcd.conf
  
   一般/system/etc/dhcpcd/dhcpcd.conf的配置為:
      interface eth0
      option subnet_mask, routers, domain_name_servers