http://blog.csdn.net/wh_19910525/article/details/7392199 2012 一、WIFI的基本架構(gòu)(代碼路徑)
1、WIFI Settings應(yīng)用程序:
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/ 我不清楚是哪一個(gè)目錄 生成庫libwpaclient.so和 守護(hù)進(jìn)程wpa_supplicant。 6、kernel 驅(qū)動(dòng)
二、WIFI在Android中如何工作
Android使用一個(gè)修改版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/)發(fā)送 命令 給wifi.c。相應(yīng)的JNI實(shí)現(xiàn)位于frameworks/base/core/jni/android_net_wifi_Wifi.cpp。更高一級(jí)的網(wǎng)絡(luò)管理位于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//驅(qū)動(dòng)名字(自己定義的宏),主要在hardware/平臺(tái)名稱/wlan/芯片名/Android.mk 文件里使用
BOARD_WPA_SUPPLICANT_PRIVATE_LIB := lib_driver_cmd_wext
BOARD_WPA_SUPPLICANT_DRIVER := WEXT //驅(qū)動(dòng)類型,決定wap_supplicant的底層接口類型 這將使external/wpa_supplicant/Android.mk設(shè)置WPA_BUILD_SUPPLICANT為true,默認(rèn)使用驅(qū)動(dòng)driver_wext.c。 如果使用定制的wpa_supplicant驅(qū)動(dòng)(例如 madwifi),可以設(shè)置: BOARD_WPA_SUPPLICANT_DRIVER := MADWIFI ------- 如果 wifi 具有 SoftAP(即虛擬無線AP) 的功能,需要以下兩個(gè)宏 ------- WIFI_DRIVER_FW_STA_PATH := /system/wifi/fw.bin WIFI_DRIVER_FW_AP_PATH :=/system/wifi/fw_ap.bin --------/* 以下兩項(xiàng) 可以在 hardware/libhardware_legary/wifi/wifi.c 里邊直接定義 */ WIFI_DRIVER_MODULE_PATH := /system/wifi/ar6000.ko //wifi 驅(qū)動(dòng)路徑 WIFI_DRIVER_MODULE_NAME := ar6000//驅(qū)動(dòng)名字 WIFI_DRIVER_MODULE_ARG:=DBG=7 //該宏是用于insmod時(shí)傳參數(shù)
四、使能wpa_supplicant調(diào)試信息 默認(rèn)wpa_supplicant設(shè)置為MSG_INFO,為了輸出更多信息,可修改:
1、在/external/wpa_supplicant/common.c中設(shè)置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的,這個(gè)路徑在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
有時(shí),驅(qū)動(dòng)需要增加: ap_scan=1 如果遇到AP連接問題,需要修改ap_scan=0來讓驅(qū)動(dòng)連接,代替wpa_supplicant。
如果要連接到non-WPA or open wireless networks,要增加: network={ key_mgmt=NONE } 六、配置路徑和權(quán)限
Google修改的wpa_supplicant要運(yùn)行在wifi用戶和組下的。代碼可見/external/wpa_supplicant/os_unix.c中的os_program_init()函數(shù)。
如果配置不對(duì),會(huì)出現(xiàn)下面錯(cuò)誤: E/WifiHW ( ): Unable to open connection to supplicant on "/data/system/wpa_supplicant/wlan0": No such file or directory will appear.
確認(rèn)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根據(jù)實(shí)際配置寫入該文件 setprop wifi.interfaceeth0
#intreface名稱設(shè)置,這在framework/base/wifi/java/android/net/wifi /WifiStateTracker.java中會(huì)用到,以處理dhcp。ar6000.ko用eth0。
目錄權(quán)限的處理是為了所有用戶能對(duì)下一級(jí)進(jìn)行搜索,/data/misc/wifi/sockets目錄不僅為wifi擁有者服務(wù),還因?yàn)橥ㄐ诺脑蛞推渌脩袈?lián)系,要不然,將會(huì)出現(xiàn)Unable to open connection to supplicant on "/data/system/wpa_supplicant/ra0": Connection refused,或permission denied的錯(cuò)誤。很多人干脆將上述所有的權(quán)限都設(shè)為0777,當(dāng)然也行,但總覺得有些粗糙。 七、運(yùn)行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 //這項(xiàng)是用于UDP連接的 disabled oneshot
service
dhcpcd /system/bin/logwrapper /system/bin/dhcpcd -d -B eth0
group system dhcp
disabled oneshot 根據(jù)所用的WIFI驅(qū)動(dòng)名字,修改wlan0為自己驅(qū)動(dòng)的名字。
八、編譯WIFI驅(qū)動(dòng)為module或kernel built in 1、編譯為module
在device/rootfs-project/BoardConfig.mk中添加: WIFI_DRIVER_MODULE_PATH := "/system/lib/modules/ar6000.ko" //驅(qū)動(dòng)的具體位置 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中當(dāng)insmod/rmmod時(shí), 直接return 0。 九、WIFI需要的firmware Android不使用標(biāo)準(zhǔn)的hotplug binary,WIFI需要的firmware要復(fù)制到/etc/firmware。
或者復(fù)制到WIFI驅(qū)動(dòng)指定的位置,然后WIFI驅(qū)動(dòng)會(huì)自動(dòng)加載。 十、修改WIFI驅(qū)動(dòng)適合Android
Google修改的wpa_supplicant要求SIOCSIWPRIVioctl 發(fā)送 命令 到驅(qū)動(dòng),及接收信息,例如signal strength, mac address of the AP, link speed等。所以要正確實(shí)現(xiàn)WIFI驅(qū)動(dòng),需要從
SIOCSIWPRIV ioctl返回RSSI (signal strength)和MACADDR信息。
如果沒實(shí)現(xiàn)這個(gè)ioctl,會(huì)出現(xiàn)如下錯(cuò)誤:
E/wpa_supplicant( ): wpa_driver_priv_driver_cmd failed
wpa_driver_priv_driver_cmd RSSI len = 4096
|
|