1、WIFI的基本架構(代碼路徑) java
一、WIFI Settings應用程序:
packages/apps/Settings/src/com/android/settings/wifi/
二、JAVA部分(framework):
frameworks/base/services/java/com/android/server/
frameworks/base/wifi/java/android/net/wifi/
三、JNI部分:
frameworks/base/core/jni/android_net_wifi_Wifi.cpp
四、wpa_supplicant的適配器 部分
hardware/libhardware_legary/wifi/主要是wifi.c。
五、wifi用戶空間的程序和庫:
external/wpa_supplicant/ 和 external/wap_supplicant_6/ 我不清楚是哪個目錄
生成庫libwpaclient.so和 守護進程wpa_supplicant。
六、kernel 驅動
2、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。
3、配置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 android
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時傳參數
4、使能wpa_supplicant調試信息
默認wpa_supplicant設置爲MSG_INFO,爲了輸出更多信息,可修改:
一、在/external/wpa_supplicant/common.c中設置wpa_debug_level = MSG_DEBUG;
二、在common.c中把#define wpa_printf宏中的
if ((level) >= MSG_INFO)
改成
if ((level) >= MSG_DEBUG)
5、修改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
}
6、配置路徑和權限
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,固然也行,但總以爲有些粗糙。
7、
運行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爲本身驅動的名字。
8、編譯WIFI驅動爲module或kernel built in
一、編譯爲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 := ""
二、編譯爲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。
9、WIFI須要的firmware
Android不使用標準的hotplug binary,WIFI須要的firmware要複製到/etc/firmware。
或者複製到WIFI驅動指定的位置,而後WIFI驅動會自動加載。
10、修改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
11、設置dhcpcd.conf
通常/system/etc/dhcpcd/dhcpcd.conf的配置爲:
interface eth0
option subnet_mask, routers, domain_name_servers 架構