不久前移植过ranlink公司的wifi无线网卡,发现网上的一些文档不完整或者对一些碰到的问题没有说明,在此详细说明一,本人将其移植到android系统上碰到了不少的问题,修改了驱动及框架代码,费了两周多的时间才完全搞好。
一. Android平台Wifi的基本代码路径
1. Wpa_supplicant源码部分
external/wpa_supplicant_6/
生成库libwpa_client.so和守护进程wpa_supplicant
2. Wifi的HAL层代码
位于hardware/libhardware_legary/wifi/
3. Wifi的JNI部分
位于frameworks/base/core/jni/android_net_wifi_Wifi.cpp
4. JAVA部分
位于frameworks/base/services/java/com/android/server/
frameworks/base/wifi/java/android/net/wifi/
5. wifi设置部分
位于packages/apps/Settings/src/com/android/settings/wifi/
二. linux内核配置
在原有android内核支持情况下,增加wifi内核配置,具体配置如下:
1. Networkingsupport --->Wireless下增加802.11 协议栈的支持
2. USB 支持WIFI的配置
USB 支持WIFI 的配置选项位于Device Drivers>USB support 配置菜单下USB Wireless
Device Management support。
3. 用户空间的mdev 和firmware 支持配置
进入DeviceDriver > Generic Driver Options 配置菜单,按照下图所示配置用户空间
的mdev 和firmware支持。
4. WIFI 设备支持配置
DeviceDrivers ---> Network devicesupport ---> Wireless LAN ---> Ralink driver support--->Ralinkrt2800 (USB) support (EXPERIMENTAL) --->rt2800usb - Include support forrt30xx (USB) devices
以及Wireless LAN 目录里IEEE 802.11 for Host AP (Prism2/2.5/3 andWEP/TKIP/CCMP)都选择上,目的是打开CONFIG_WIRELESS_EXT=y CONFIG_WEXT_PRIV=y
三. 驱动配置与编译
1. 修改驱动SDK包中的配置文件
1.1 修改env.mk,将RT28xx_DIR 设为当前目录,RT28xx_DIR = $(shell pwd)。
1.2 修改makefile中对应的kernel与交叉编译器路径
1.3 修改os/linux目录下config.mk中gcc 与 ld变量
1.4 打开os/linux目录下config.mk中HAS_WPA_SUPPLICANT与HAS_NATIVE_WPA_SUPPLICANT_SUPPORT宏
2. 修改驱动SDK包中的驱动源码
2.1 将rt_linux.h中的RTUSB_URB_ALLOC_BUFFER与RTUSB_URB_FREE_BUFFER宏修改,定义如下
#define RTUSB_URB_ALLOC_BUFFER(pUsb_Dev,BufSize, pDma_addr) usb_alloc_coherent(pUsb_Dev,BufSize, GFP_ATOMIC, pDma_addr)
#defineRTUSB_URB_FREE_BUFFER(pUsb_Dev, BufSize, pTransferBuf, Dma_addr) usb_free_coherent(pUsb_Dev, BufSize,pTransferBuf, Dma_addr)
2.2 修改rt_main_dev.c,直接将MainVirtualIF_close函数放空,return 0,解决不能反复关闭wifi问题。
2.3 修改rt_linux.c里RtmpOSNetDevAttach函数里增加devname赋值。strcpy( pNetDev->name, "mlan0");注:(此处所用的名字要与上层使用的节点名保持一致,在此说明一下上层主要有这几处用到节点名:
1,\frameworks\base\wifi\java\android\net\wifiWifiStateTracker.java
2,init.rc启动wpa_supplicant守护进程里面与启动dhcpcd服务
3,dhcpcd服务配置文件,dhcpcd.conf里面
4,init.rc设置setprop wifi.interface "mlan0")
3. 编译方法
Source env.mk;make;即可,驱动是在的路径为os/linux下的rt3070sta.ko。
此处所用的驱动名字应与HAL层wifi.c所指定驱动名保持一致
四. Wap_supplicant相关配置
3.1 Wpa_supplicant.conf配置文件的修改
ctrl_interface=DIR=/data/system/wpa_supplicantGROUP=wifi #这个路径在wifi.c中用到
3.2 整个环境必须要让wext类型相关代码进行编译。也就是要打开wext相关的宏CONFIG_DRIVER_WEXT。 即在device/hisi/Hi3716C/BoardConfig.mk中添加:
BOARD_HAVE_WIFI := true
BOARD_WPA_SUPPLICANT_DRIVER := WEXT
该配置的作用是使external/wpa_supplicant/Android.mk设置WPA_BUILD_SUPPLICANT为true。
3.3 在init.rc里面增加启动wpa_supplicant守护进程及dhcpcd进程
3.4 在init.rc里面增加wifi相关文件的权限设定,设置如下:
chmod 0771 /system/etc/wifi
chmod 0660/system/etc/wifi/wpa_supplicant.conf
chown wifiwifi /system/etc/wifi/wpa_supplicant.conf #wifi的原始配置文件
#wpa_supplicantsocket
mkdir/data/system/wpa_supplicant 0770 wifi wifi
chmod 0771/data/system/wpa_supplicant #放置wifiinterface的地方
mkdir/data/misc/wifi 0770 wifi wifi
chmod 0771/data/misc/wifi
chmod 0660 /data/misc/wifi/wpa_supplicant.conf #wifi的配置文件,将由wpa_supplicant根据实际配置写入该文件
chown wifiwifi /data/misc/wifi
chown wifiwifi /data/misc/wifi/wpa_supplicant.conf
mkdir/data/misc/wifi/sockets 0770 wifi wifi #与上层通过socket通信的路径
cp/system/etc/wifi/wpa_supplicant.conf /data/misc/wifi/
mkdir/data/misc/dhcp 0777 dhcp dhcp
chown dhcpdhcp /data/misc/dhcp
# Preparefor wifi
setpropwifi.interface "mlan0"
setprop wlan.driver.status "ok"
3.5 启动wpa_supplicant守护进程与dhcpcd服务
在init.rc里面添加wpa_supplicant启动:
service wpa_supplicant/system/bin/logwrapper /system/bin/wpa_supplicant \
-Dwext -imlan0 -c/data/misc/wifi/wpa_supplicant.conf -dd
user root
group system wifi inet
socket wpa_mlan0 dgram 660 wifi wifi
disable
oneshot
在init.rc里面添加dhcpcd启动:
service dhcpcd /system/bin/logwrapper/system/bin/dhcpcd -d -B wlan0
disabled
oneshot
3.6 在init.godbox.rc里增加dns设置
Setprop net.dns1 192.168.10.247
Setprop net.dns2 192.168.10.248
五. wifi移植所需在android系统添加的一些文件
4.1 添加wifi的wpa_supplicant.conf配置文件
放置目录与hardware/libhardware_legacy/wifi/wifi.c中的目录保持一致
4.2 添加驱动的配置文件
在system/etc/Wireless/RT2870STA目录放置配置文件RT2870STA.dat,与rt_linux.h中配置文件的路径保持一致。
4.3 添加dhcpcd启动配置文件
设置/system/etc/dhcpcd/dhcpcd.conf的配置为:
interface mlan0
option subnet_mask, routers,domain_name_servers
七. 其它平台移植记录
6.1 内核
内核的修改如上述第二大点内核配置
6.2 Wpa_supplicant
将wpa_supplicant_6编译打开
Wpa_supplicant 主要是在device/hisi/Hi3716C/BoardConfig.mk中添加:
BOARD_HAVE_WIFI := true
BOARD_WPA_SUPPLICANT_DRIVER := WEXT
以及在wpa_supplicant_6 里面的.config增加ANDROID=y
对于刚入手android没多久的人来说,android wifi 驱动的移植确实还是有难度的,不过参考了网友的相关帖子后,最终还是移植成功了,,以便自己和他人查看学习: &&&& WIFI的基本架构 &&&& (这一部分比较重要,是一直wifi驱动的基础) 1、wifi用户空间的程序和库: external/wpa_supplicant/ 生成库libwpaclient.so和守护进程wpa_supplicant。 2、hardware/libhardware_legary/wifi/是wifi管理库。 3、JNI部分: frameworks/base/core/jni/android_net_wifi_Wifi.cpp 4、JAVA部分: frameworks/base/services/java/com/android/server/ frameworks/base/wifi/java/android/net/wifi/ 5、WIFI Settings应用程序位于: packages/apps/Settings/src/com/android/settings/wifi/ &&&& WIFI在Android中如何工作 &&&& (理解wifi的工作机制有助于调试wifi的相关错误) Android使用一个修改版wpa_supplicant作为daemon来控制WIFI,代码位于 external/wpa_supplicant。wpa_supplicant是通过socket与 hardware/libhardware_legacy/wifi/wifi.c通信。UI通过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。 (了解完wifi的框架和工作机制,下面就正式开始wifi驱动的移植) &&&& 配置Android支持WIFI &&&& 在BoardConfig.mk中添加: BOARD_HAVE_WIFI := true BOARD_WPA_SUPPLICANT_DRIVER := WEXT 这将在external/wpa_supplicant/Android.mk设置WPA_BUILD_SUPPLICANT为true, 默认使用驱动driver_wext.c。 如果使用定制的wpa_supplicant驱动(例如 madwifi),可以设置: BOARD_WPA_SUPPLICANT_DRIVER := MADWIFI &&&& 使能wpa_supplicant调试信息 &&&& 默认wpa_supplicant设置为MSG_INFO,为了输出更多信息,可修改: 1、在common.c中设置wpa_debug_level = MSG_DEBUG; 2、在common.c中把#define wpa_printf宏中的 if level) >= MSG_INFO) 改为 if level) >= MSG_DEBUG) &&&& 配置wpa_supplicant.conf &&&& wpa_supplicant是通过wpa_supplicant.conf中的ctrl_interface=来指定控制socket的设备,应该在 AndroidBoard.mk中配置好复制到$(TARGET_OUT_ETC)/wifi(也就是 /system/etc/wifi/wpa_supplicant.conf) 这个位置会在init.rc中再次检测的。 一般的wpa_supplicant.conf配置为: ctrl_interface=DIR=/data/system/wpa_supplicant GROUP=wifi (其实这里直接指定接口就可以了,即ctrl_interface=wlan0或eth0) update_config=1 fast_reauth=1 有时,驱动需要增加: ap_scan=1 如果遇到AP连接问题,需要修改ap_scan=0来让驱动连接,代替wpa_supplicant。 如果要连接到non-WPA or open wireless networks,要增加: network={ key_mgmt=NONE } &&&& 配置路径和权限&&&& Google修改的wpa_supplicant要运行在wifi用户和组下的。代码可见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 0770 wifi wifi chmod 0770 /system/etc/wifi chmod 0660 /system/etc/wifi/wpa_supplicant.conf chown wifi wifi /system/etc/wifi/wpa_supplicant.conf # wpa_supplicant socket mkdir /data/system/wpa_supplicant 0771 wifi wifi chmod 0771 /data/system/wpa_supplicant #wpa_supplicant control socket for android wifi.c mkdir /data/misc/wifi 0770 wifi wifi mkdir /data/misc/wifi/sockets 0770 wifi wifi chmod 0770 /data/misc/wifi chmod 0660 /data/misc/wifi/wpa_supplicant.conf 如果系统的/system目录为只读,那应该使用路径/data/misc/wifi/wpa_supplicant.conf。 &&&& 运行wpa_supplicant和dhcpcd &&&& 在init.rc中确保有如下语句: service wpa_supplicant /system/bin/logwrapper /system/bin/wpa_supplicant -dd -Dwext -iwlan0 -c /data/misc/wifi/wpa_supplicant.conf user root group wifi inet socket wpa_wlan0 dgram 660 wifi wifi oneshot service dhcpcd /system/bin/logwrapper /system/bin/dhcpcd -d -B wlan0 disabled oneshot 根据所用的WIFI驱动名字,修改wlan0为自己驱动的名字。 &&&& 编译WIFI驱动为module或kernel built in &&&& 1、编译为module 在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要求SIOCSIWPRIV ioctl发送命令到驱动,及接收信息,例如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 wlan0 option subnet_mask, routers, domain_name_servers |