Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1522051
  • 博文数量: 230
  • 博客积分: 474
  • 博客等级: 下士
  • 技术积分: 1955
  • 用 户 组: 普通用户
  • 注册时间: 2010-03-19 18:40
文章分类

全部博文(230)

文章存档

2020年(3)

2019年(3)

2018年(12)

2017年(13)

2016年(11)

2015年(55)

2014年(74)

2013年(39)

2012年(2)

2011年(18)

我的朋友

分类: LINUX

2014-12-04 14:56:22

转自:http://blog.csdn.net/dybinx/article/details/7380903
硬件环境     
     WIFI模块:Marvell8686 SDIO WIFI
     开发板:S5PC100     
     软件环境: Linux 2.6.29  Android 2.1

项目目标:实现WIFI上网功能、并对WIFI休眠进行改善。
项目开发流程:
硬件分析:对Marvell8686 SDIO WIFI模块硬件工作特性了解和分析
工作原理分析:对Marvel8686 SDIO WIFI工作原理进行分析
Android WIFI框架分析:对Android WIFI系统框架分析
进行源码分析、编写、修改及编译
调试、并完善

一、1.编译内核,生成驱动模块
     在内核的根目录下执行make
     生成libertas.ko 和libertas_sdio.ko
     [*] Networking support  --->  
         [*]   Wireless  --->  
                 --- Wireless                                                
                <*>   Improved wireless configuration API                   
                 [*]     cfg80211 regulatory debugging                        
                 [*]     nl80211 new netlink interface support                                                                              
                 -*-   Common routines for IEEE802.11 drivers    
         Device Drivers  --->   
         [*] Network device support  --->  
                Wireless LAN  --->   
               <*>   Marvell 8xxx Libertas WLAN driver support              (注:编译进内核,若模块则是:libertas.ko)       
                    Marvell Libertas 8385 and 8686 SDIO 802.11b/g cards  (注:编译成libertas_sdio.ko)
     
     2.编译内核,确保支持sd卡
        Device Drivers  ---
           <*> MMC/SD/SDIO card support  ---> 
             --- MMC/SD/SDIO card support                                     
             [*]   MMC debugging                                              
             [ ]   Allow unsafe resume (DANGEROUS)                            
             [*]   MMC embedded SDIO device support (EXPERIMENTAL)            
             [ ]   Enable paranoid SD card initialization (EXPERIMENTAL)      
                   *** MMC/SD/SDIO Card Drivers ***                           
             <*>   MMC block device driver                                    
             [*]     Use bounce buffer for simple hosts                       
             [ ]     Deferr MMC layer resume until I/O is requested           
             < >   SDIO UART/GPS class support                               
             < >   MMC host test driver                                        
                   *** MMC/SD/SDIO Host Controller Drivers ***                
             <*>   Secure Digital Host  
       3.make zImage
         make modules
        (libertas_sdio.ko在drivers/net/wireless/libertas目录下)
二、制作测试工具(使用静态编译) 
    现在,sd卡支持了,驱动以内核自带的以模块的方式编译得到。
    接下来先编译几个测试工具。
    测试无线网卡用无线工具iwconfig iwlist等命令是通过开源软件wireless_tools_29.rar编译得到。         
    步骤:
    1.解压。由于这的是win32的压缩包,先在windows底下解压,再拷贝到Ubuntu下。
    2.修改Makefile:
           8   PREFIX = ./tools  //指定安装路径                   
           12  CC = arm-linux-gnu-gcc (和编译你的文件系统所用保持一致,需要注意的是,如果在执行sudo make 就要该编译链的绝对路 径)                                                                                                      
           14  AR = arm-linux-gnu-ar   
           
           #BUILD_STATIC = y          -->    BUILD_STATIC = y                                       
           #BUILD_STRIPPING = y       -->    BUILD_STRIPPING = y   
           
 
           CFLAGS=-Os -W -Wall -Wstrict-prototypes -Wmissing-prototypes -Wshadow \         
                 -Wpointer-arith -Wcast-qual -Winline -I. 
           在这后面添加新行:                              
           CFLAGS += -static 
           
    3.make
    4.make install
      由于我是在当前目录下(PREFIX = ./tools) 产生的这个tools文件夹,
      查看有:
      lib sbin usr
      lib目录,libiw.a静态库 ;而sbin目录为一些网络工具,如:iwlist、iwconfig等
      到里面的sbin目录底下,
      $file iwlist  输出信息:iwlist: ELF 32-bit LSB executable, ARM, version 1, statically linked, for GNU/Linux 2.4.3, strippe
      确保是静态的。 
      
      然后将这些命令复制到android文件系统的/system/busybox/bin(就是out/target/prodruct/fs100/system/busybox/bin)
      或者system/bin(就是out/target/prodruct/fs100/system/bin)目录下就可以使用这些命令了。  
 
 三、将固件放到指定目录下。
     1.mkdir /system/etc/firmware(就是out/target/prodruct/fs100/system/etc/firmware)
     2.cp sd8686.bin sd8686_helper.bin  out/target/prodruct/fs100/system/etc/firmware
 
 四、测试,sdio卡是否可用。
     如果已经编译好了文件系统,就可以跳过第一步。
     1.编译android系统。
               1)在文件系统根目录下执行:
                . ./build/envsetup.sh
             注意这里两个’.’  之间有一个空格,第一个’.’指定用当前 shell 解析这个脚本,否则不能执行。
            2)配置板级信息:
             tapas
            就是上一步执行结束之后导出到环境变量里的命令,专门用来配置板级信息的。
             Build for the simulator or the device? 
                    1. Device 
                   2. Simulator 
            Which would you like? [1] 1 
            Build type choices are: 
                   1. release 
                   2. debug 
            Which would you like? [1] 1 
            Which product would you like? [fs100] fs100 
            Variant choices are: 
                   1. user 
                   2. userdebug 
                   3. eng 
            Which would you like? [eng] eng 
            
            确保输出的配置信息为:
            PLATFORM_VERSION_CODENAME=REL 
            PLATFORM_VERSION=2.1-update1  
            TARGET_PRODUCT=fs_s5pc100     
            TARGET_BUILD_VARIANT=eng      
            TARGET_SIMULATOR=false        
            TARGET_BUILD_TYPE=release     
            TARGET_ARCH=arm               
            HOST_ARCH=x86                 
            HOST_OS=linux                 
            HOST_BUILD_TYPE=release       
            BUILD_ID= ERE27               
            
            3)开始编译(如果不能找到 mm,执行”source build/envsetup.sh”): 
             mm
    
   2.重新生成文件系统镜像。
      ./make_fs100_yaffs2_image.sh
     就会在 Android 源码根目录下生成目录“fs100_root”,这个目录就是编译生成的 Android 文件系统,  
     调试时可以直接把这个目录作为 NFS-Server 的目录。还会生成一个“fs100_root.img”文件,这个文件就
     是 Android的 yaffs2 格式的镜像,可以烧写到 Nand Flash 上。
       这里我是烧写上fs100_root.img.
     
     3.手动加载驱动。
       由于sdio8686。需要加载两个.ko(libertas.ko、libertas_sdio.ko) 文件。
       上面linertas.ko的模块,我已经编进内核。
       所以我只需执行:
       1)/# insmod libertas_sdio.ko   
       输出信息:
       libertas_sdio: Libertas SDIO driver                                 
     libertas_sdio: Copyright Pierre Ossman                              
     __func__ = if_sdio_prog_helper                                      
     libertas_sdio mmc1:0001:1: firmware: requesting sd8686_helper.bin   
     init: untracked pid 2137 exited                                     
     libertas_sdio mmc1:0001:1: firmware: requesting sd8686.bin          
     init: untracked pid 2140 exited                                     
     libertas: 00:0b:6c:91:a3:f6, fw 9.70.3p24, cap 0x00000303           
     libertas: unidentified region code; using the default (USA)         
     wlan0 (libertas_sdio): not using net_device_ops yet                 
     libertas: PREP_CMD: command 0x00a3 failed: 2                        
     libertas: PREP_CMD: command 0x00a3 failed: 2                        
     libertas: wlan0: Marvell WLAN 802.11 adapter                        
    2)/# ifconfig -a  
       (要是提示:-a: No such device,则输入命令:system/busybox/sbin/ifconfig -a)    
    查看是否检测到无线网卡                           
           有如下信息说明检测到无线网卡                                    
             wlan0     Link encap:Ethernet  HWaddr 00:0B:6C:91:A3:F6       
                       BROADCAST MULTICAST  MTU:1500  Metric:1             
                       RX packets:0 errors:0 dropped:0 overruns:0 frame:0  
                       TX packets:0 errors:0 dropped:0 overruns:0 carrier:0
                       collisions:0 txqueuelen:1000                        
                       RX bytes:0 (0.0 B)  TX bytes:0 (0.0 B)              
    3)测试:  
         /#ifconfig wlan0 up                                                                                                          
        /#iwlist wlan0 scanning                                                                                                         
        /#iwconfig wlan0 essid "fs100"                                                                                                  
        /#ifconfig wlan0 192.168.1.99 netmask 255.255.255.0 up (静态分配ip,
         也可以动态分配,用DHCP)
         /#route add default gw 192.168.1.1(无线路由)
         /#ping 192.168.1.1            
    4)能够ping通后,说明sdio8686 wifi网卡没有问题
    
五、测试wpa_supplicant。
  因为现在的无线wifi网络大多是wpa加密。 所以需要用到wpa_supplicant。
    android系统中也自带有wpa_supplicant。
    wifi的大致架构是这样的,app-->java framework-->|jni|-->c++ framework -->wifi.c -->wpa_supplicant-->sdio8686.
    所以我们要先测试低层先通了,因此,就得先测试确保wpa_supplicant和sdio8686通了的。
    我们编译android系统时,就已经把自带的wpa_supplicant和wpa_cli编译好并放到system/bin目录下了. 
    1.配置wpa_supplicant。
    在在目录下有个wpa_supplicant.conf。
    将wpa_supplicant.conf放到out/target/product/fs100/system/etc 目录下。如果是烧到板子上的话,需要重烧板子。
    要是以网络挂载的方式,则放到相应的目录。
    #vim /rootfs/filesystem/etc/wpa_supplicant.conf           
                                                              
    修改内容如下:                                            
                                                              
    # WPA-PSK/TKIP                                            
                                                              
     ctrl_interface=/var/run/wpa_supplicant                    
                                                              
     network={                                                 
                     ssid="fs100"     //填写无线网络的的用户名
                      key_mgmt=WPA-PSK                          
                      proto=WPA                                 
                     pairwise=TKIP                             
                     group=TKIP                                
                     psk="1234567890" //填写密码               
     }                                                         
    
    2.#mkdir –p /var/run/wpa_supplicant
    3.确认是否加载驱动。
     用ifconfig -a查看是否已经创建wlan0这个节点。
      没有就把驱动加载进内核。
    4.创建连接暗文密码(PSK密码),通过明码转换                                                                    
      #cd /etc (该目录下要有 wpa_supplicant.conf)                                                                                                  
      #wpa_passphrase fs100 1234567890 >> wpa_supplicant.conf                                                   
      此时将在wpa_supplicant.conf文件中生成:                                                                    
      network={                                                                                                  
            ssid="fs100"                                                                                        
            #psk="1234567890" //此物为路由器中设定的人类能读得懂的密码,供我们使用                               
            psk=d290464a66df8541cee4f100627446177ee7ce5d9eb36981e4dff412730b2d5a //这个又1234567890转换后的psk密码
      }                                                                                                          
       -----------------------------------------------------------------------------------------------------<----
       ---->遇到问题:在测试过程中要用到wpa_passphrase这个命令程序。但是android并没提供wpa_passphrase       <----
       ---->这命令,我只通过wpa_supplicant-0.7.3这个源码包编译出这个命令。但是这个命令在开发板上执行        <----
       ---->不了(编译链不对)。我又找到android源码中有wpa_passphrase.c,而编译出来的路径下通过find         <----
       ---->命令去找这个wpa_passphrase又找不到。                                                            <----
       ---->解决办法:同过分析wpa_passphrase.c才发现,里面就短短的一个mian函数。有个提示                    <----
       ---->usage: wpa_passphrase [passphrase]\n""\nIf passphrase is left out, it will be read from "<----
     ---->"stdin\n                                                                                        <----
       ---->现在不用我说都知道了吧!所以这个在哪运行都可以,这命令就是为了通过ssid和passphrase得到暗文。    <----
       ---->所以我在Ubuntu底下直接执行:wpa_passphrase fs100 1234567890                                     <----
       ---->同样输出信息:network= {                                                                         <----                             
       ---->                          ssid="fs100"                                                          <----                             
       ---->                          #psk="1234567890"                                                     <----
       ---->                          psk=d290464a66df8541cee4f100627446177ee7ce5d9eb36981e4dff412730b2d5a  <----
       ---->                    }                                                                           <----                             
       -----------------------------------------------------------------------------------------------------<----- 
     5.然后修改wpa_supplicant.conf配置文件,把 psk="1234567890" 这一行明文密码改成生成的暗文密码,即:
               psk=d290464a66df8541cee4f100627446177ee7ce5d9eb36981e4dff412730b2d5a                  
                                                                                                     
       其它生成的多余信息删除,最后wpa_supplicant.conf文件如下:                                     
       # WPA-PSK/TKIP                                                                                
                                                                                              
       ctrl_interface=/var/run/wpa_supplicant                                                        
                                                                                              
       network={                                                                                     
                  ssid="FS2410"     //填写无线网络的的用户名                                         
                  key_mgmt=WPA-PSK                                                                   
                  proto=WPA                                                                          
                  pairwise=TKIP                                                                      
                  group=TKIP                                                                         
                 psk=d290464a66df8541cee4f100627446177ee7ce5d9eb36981e4dff412730b2d5a               
       } 
       
     6.链接AP:
             在开发板终端输入wpa_supplicant回车,会显示帮助信息,最后有个:                         
             example:                                                                             
                   wpa_supplicant -Dwext -iwlan0 -c/etc/wpa_supplicant.conf                       
             拷贝example用法,执行:                                                              
             # wpa_supplicant -Dwext -iwlan0 -c/etc/wpa_supplicant.conf &(需要一直放在后台运行)   
             打印信息有:                                                                         
             Trying to associate with 00:26:f2:0d:5a:c4 (SSID='fs100' freq=2412 MHz)             
             Associated with 00:23:68:28:4e:a8                                                    
             CTRL-EVENT-DISCONNECTED bssid=00:23:68:28:4e:a8 reason=0                             
             Associated with 00:26:f2:0d:5a:c4                                                    
             WPA: Key negotiation completed with 00:26:f2:0d:5a:c4 [PTK=TKIP GTK=TKIP]            
             CTRL-EVENT-CONNECTED - Connection to 00:26:f2:0d:5a:c4 completed (auth) [id=0 id_str=]
                                                                                           
             分配IP地址:                                                                         
             #ifconfig wlan0 192.168.1.5  //也可以动态分配 dhclient wlan0                         
             再ping下网关,是否连接成功:                                                         
             #ping 192.168.1.1                                                                    
                                                                                            
             3 packets transmitted, 3 packets received, 0% packet loss                            
             round-trip min/avg/max = 17.627/20.023/24.631 ms                                     
             ……                                                                                 
                                                                                            
             无线网卡连接成功!      
         

六、android的wifi移植。
     现在底层的都通了,接下来就把wifi移植到android系统中。
     1.提供固件firmware给驱动。
        WIFI需要的firmware要复制到/etc/firmware。   或者复制到WIFI驱动指定的位置,然后WIFI驱动会自动加载。
        在此把wifi模块提供的sd8686.bin sd8686_helper.bin放到out/target/product/fs100/system/etc/firmware目录下。
        其实,和上面放目录是一样的。
     
     2.修改自己定制的BoardConfig.mk,使能wpa_supplicant。
       修改vendor/farsight/fs100/BoardConfig.mk:                                                       
          把BOARD_WPA_SUPPLICANT_DRIVER :=true                                                          
          改为BOARD_WPA_SUPPLICANT_DRIVER := WEXT                                                                                                                                                
          目的是:把driver_wext.c作为wpa_supplicant的driver。wpa_supplicant通过它去与内核的wifi驱动打交道。
       修改external/wpa_supplicant/Android.mk                                                           
           把WPA_BUILD_SUPPLICANT :=false                                                               
           改为WPA_BUILD_SUPPLICANT := true                                                         
          默认使用驱动driver_wext.c。                                                                   
          如果使用定制的wpa_supplicant驱动(例如 madwifi),可以设置:                                     
          BOARD_WPA_SUPPLICANT_DRIVER := MADWIFI 
           
                                                                   
      3.使wpa_supplicant打印更多的调试信息
    wpa_supplicant默认信息显示的等级为SG_INFO,为了输出更多信息,可修改:  
        修改external/wpa_supplicant/common.c                                   
            把int wpa_debug_level = MSG_INFO;                                  
            改为:int wpa_debug_level = MSG_DEBUG;                             
        修改external/wpa_supplicant/common.h                                   
            把宏定义#define wpa_printf(level, ...)中的if ((level) >= MSG_INFO) 
            改为if ((level) >= MSG_DEBUG) 
      4.提供一个合适的wpa_supplicant.conf 
        修改 wpa_supplicant.conf:                                                                                                                 
           把external/wpa_supplicant/wpa_supplicant.conf拷贝到out/target/product/fs100/system/etc/wifi/目录下,
           并把:ctrl_interface=DIR=/data/misc/wifi/wpa_supplicant GROUP=wifi                                 
           改为:                                                                                             
           ctrl_interface=wlan0                                                                               
      5.修改init.rc配置路径和权限
        A)配置init.rc文件修改out/target/product/fs100/root/init.rc, 让wifi用户拥有相关的权限, 
          在 #give system access to wpa_supplicant.conf for backup and restore后面增加:
             #add by liyulei 2011-11-19                                                                                                                       
             mkdir /data/misc/wifi/sockets 0777 wifi wifi                                                                                                           
        chown wifi wifi /data/misc/wifi                                                                                                                   
             chown wifi wifi /data/misc/wifi/wpa_supplicant.conf                                                                                                           
             #for dhcp                                                                                                                                      
             mkdir /data/misc/dhcp 0777 dhcp dhcp                                                                                                    
             chmod 0770 /data/misc/dhcp                                                                                                             
             #end add                                                                                                                                                      
        B)注释原有环境变量#export PATH 
          改 为:                                                                                                                             
         export PATH /sbin:/system/sbin:/system/bin:/system/xbin                                                                                       
          #above modified by liyulei 2011-11-19                                                                                                                            
      6. 确保wpa_supplicant和dhcpcd是通过init.c起来的。
         配置init.rc文件修改out/target/product/fs100/root/init.rc, 在末尾添加:                                       
             #add by liyulei 2011-11-19
             for wifi Android private socket                                             
             service wpa_supplicant /system/bin/wpa_supplicant -dd -Dwext -iwlan0 -c/system/etc/wifi/wpa_supplicant.conf
                     socket wpa_wlan0 dgram 660 wifi wifi                              
                     group system wifi inet                                                   
                     disabled                                                              
                     oneshot                                                                
             #for dhcp                                                                          
             service dhcpcd /system/bin/dhcpcd wlan0                                          
                     group system dhcp                                                           
                     disabled                                                                           
                     oneshot                                                                       
             #end add。                                                                                                
   7.设置驱动以模块方式加载
   拷贝内核生成的drivers/net/wireless/libertas/libertas_sdio驱动模块
   到out/target/product/fs100/system/lib/modules/目录下。
   然后修改hardware/libhardware_legacy/wifi/wifi.c                                                                       
          1)修改wifi.c中的宏:
             #define WIFI_DRIVER_MODULE_PATH            "/system/lib/modules/libertas_sdio.ko"                                                            
             #define WIFI_DRIVER_MODULE_NAME            "libertas_sdio"                                                                                       
             #define WIFI_TEST_INTERFACE              "wlan0"  
          或者,也可以这样改:
            修改vendor/farsight/fs100/BoardConfig.mk:
              WIFI_DRIVER_MODULE_PATH  ="/system/lib/modules/libertas_sdio.ko"                                                            
              WIFI_DRIVER_MODULE_NAME  ="libertas_sdio"                                                                                       
              WIFI_TEST_INTERFACE     ="wlan0"  
            和上面一样的效果。
                                                                                                           
      8.添加休眠唤醒代码:
      1)在frameworks\base\core\Jni\android_net_wifi_Wifi.cpp中添加相应的休眠唤醒函数:
      在static JNINativeMethod gWifiMethods[]结构体添加方法: 
      {"moduleWakeupStatus","()Z",(void*)android_net_wifi_moduleWakeupStatus},
      {"WaitTime","(I)Z",(void*)android_net_wifi_WaitTime}, 
      并实现这些方法/*add by liyulei 2011-11-20*/                                                      
          static jboolean android_net_wifi_moduleWakeupStatus(JNIEnv* env, jobject clazz)  
        {                                                                                  
         return (jboolean)(::wifi_module_wakeup_status() == 0);                            
        }                                                                                  
        static jboolean android_net_wifi_WaitTime(JNIEnv* env, jobject clazz, jint waittime)
        {                                                                                  
         return (jboolean)(::wifi_module_wait_time(waittime) == 0);                       
        }                                                                                  
        /*end add by liyulei*/  
        
        2)而相应的细体实现在wifi.c中:
         在wifi.c添加:                                                            
           /*add by caoyi 2011-10-20*/               
           int wifi_module_wakeup_status()           
              {                                      
                    return 1;                        
              }                                      
              int wifi_module_wait_time(int watiTime)     
             {                                            
                sleep(watiTime);                         
                   return  0;                                                                                                                                                    
              }                                      
           /*end add*/                               
      
      9.配置dhcpcd.conf
        修改源码目录下external/dhcpcd下的Android.mk文件
    取消注释
       -->26  include $(CLEAR_VARS)
       -->27 LOCAL_MODULE := dhcpcd.conf
       -->28 LOCAL_MODULE_TAGS := user
       -->29 LOCAL_MODULE_CLASS := ETC
       -->30 LOCAL_MODULE_PATH := $(etc_dir)
       -->31 LOCAL_SRC_FILES := android.conf
       -->32 include $(BUILD_PREBUILT)   
       然后重新编译mm,将编译产生的dhcpcd.conf放置到文件系统的目录system/etc/dhcpcd/dhcpcd.conf
        最后确定dhcpcd.conf内容有:
       interface wlan0
       option subnet_mask, routers, domain_name_servers 
       option ntp_servers  
       没有的话,就修改。
          
      10.修改WifiStateTracker.java
       将frameworks/base/wifi/java/android/net/wifi/WifiStateTracker.java 里的mInterfaceName = SystemProperties.get("wifi.interface", "eth0");
       改为:
       mInterfaceName = SystemProperties.get("wifi.interface", "wlan0");
       最后重新编译mm
       编译镜像文件后烧写到开发板上就可以使用WiFi模块连接网络
阅读(5239) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~