设为首页 收藏本站
查看: 1477|回复: 0

[经验分享] Android WIFI 移植

[复制链接]

尚未签到

发表于 2015-10-2 09:15:25 | 显示全部楼层 |阅读模式
在 Android 2.2上移植了2个wifi模块,vt6656和rt2070,总结一下要点。
首先,将wifi linux驱动编译成模块,并将驱动(vntwusb.ko或rt3070sta.ko放到/system/lib/modules/中。然后,做如下修改:
1。修改 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  #放置wifi interface的地方
           mkdir /data/misc/wifi 0771 wifi wifi
          chmod 0771 /data/misc/wifi
          chmod 0660 /data/misc/wifi/wpa_supplicant.conf  #wifi的配置文件,将由wpa_supplicant根据实际配置写入该文件
          mkdir /data/misc/wifi/sockets 0777 wifi wifi  #与上层通过socket通信的路径
           # Prepare for wifi
           setprop wifi.interface ra0  #intreface名称设置,这在framework/base/wifi/java/android/net/wifi/WifiStateTracker.java中会用到,以处理dhcp。rt2070用ra0,而vt6656使用eth1。
这里0771对目录权限的处理是为了所有用户能对下一级进行搜索,而红字特别提醒的权限配置,是因为/data/misc/wifi/sockets目录不仅为wifi拥有者服务,还因为通信的原因要和其他用户联系,要不然,将会出现Unable to open connection to supplicant on "/data/system/wpa_supplicant/ra0": Connection refused,或permission denied的错误。很多人干脆将上述所有的权限都设为0777,当然也行,但总觉得有些粗糙。
service的修改:
service wpa_supplicant /system/bin/logwrapper /system/bin/wpa_supplicant \
    -Dwext -ira0 -c/data/misc/wifi/wpa_supplicant.conf  #也可以用/system/etc/wifi/wpa_supplicant.conf代替
    user root
    group system wifi inet
#    socket wpa_wlan0 dgram 660 wifi wifi   #屏蔽该项是因为这项是用于UDP连接的
    disable
    oneshot
service dhcpcd /system/bin/logwrapper /system/bin/dhcpcd -d -B ra0
    group system dhcp wifi
    disabled
    oneshot
2。修改system/etc/wifi/wpa_supplicant.conf (在源码中是修改external/wpa_supplicant/wpa_supplicant.conf)
将ctrl_interface=wlan0改成ctrl_interface=DIR=/data/system/wpa_supplicant GROUP=wifi  #这个路径在wifi.c中用到。
3。修改system/etc/dhcpcd/dhcpcd.conf
将其中的interface名称改成ra0
4。修改芯片厂商的配置 BoardConfig.mk。
例如,Freeescale 是在device/fsl/imx51_bbg/,加入:
HAVE_CUSTOM_WIFI_DRIVER_2 := true
BOARD_WPA_SUPPLICANT_DRIVER := WEXT
而下面已经定义的wifi驱动的路径我感到还是屏蔽为好,直接在wifi.c中修改不是更直观些。
5。修改hardware/libhardware_legacy/wifi/wifi.c
ifndef WIFI_DRIVER_MODULE_PATH
#define WIFI_DRIVER_MODULE_PATH         "/system/lib/modules/rt3070sta.ko"
#endif
#ifndef WIFI_DRIVER_MODULE_NAME
#define WIFI_DRIVER_MODULE_NAME         "rt3070sta"
#endif
在文件中还可以看到其他一些信息,如IFACE_DIR[]           = "/data/system/wpa_supplicant", 就是interface的安放的路径。MODULE_FILE[]         = "/proc/modules",这是insmod安放module的路径。在调试时可以查询是否已经安装了模块,接口是否启动。
6。源码修改
修改external/wpa_supplicant/wpa_ctrl.c: 找到chmod那行,将chmod(ctrl->local.sun_path, S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP);改成chmod(ctrl->local.sun_path, S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH);  #增加权限
这是为了防止出现ioctl 写message的错误。
修改external/wpa_supplicant/driver_wext.c:
这是为了避免wpa_supplicant与下层驱动通讯时出现ioctl[SIOCSIWPRIV]错误,因为现在大部分wifi模块对SIOCSIWPRIV命令不处理,而这个命令要用于侦测wifi强度RSSI的,比较简单的方法是在wifi驱动中增加个空函数。例如,对于rt2070,有一个sta_ioctl.c,找到SIOCSIWPRIV,将其对应个空函数
static int handler_SIOCSIWPRIV(struct net_device *dev, struct iw_request_info *info,
                         union iwreq_data *wrqu, char *extra)
{
       return 0;
}
不过,这样做就有些信息,如RSSI,MAC地址等就没法在上层显示了。比较好的方法如下:
在struct wpa_driver_wext_data 中增加
    u8 ssid[32];
    unsigned int ssid_len;
2个变量。将原来的wpa_driver_priv_driver_cmd函数改成如下函数(我从Porting wifi driver to Android抄来稍作修改):
static int wpa_driver_priv_driver_cmd(void *priv, char *cmd, char *buf, size_t buf_len)
{
    struct wpa_driver_wext_data *drv = priv;
    int ret = -1;
    wpa_printf(MSG_DEBUG, "AWEXT: %s %s", __func__, cmd);
    if (os_strcasecmp(cmd, "start") == 0) {
        wpa_printf(MSG_DEBUG,"Start command");
        return (ret);
    }
    if (os_strcasecmp(cmd, "stop") == 0) {
        wpa_printf(MSG_DEBUG,"Stop command");
    }
    else if (os_strcasecmp(cmd, "macaddr") == 0) {
        struct ifreq ifr;
        os_memset(&ifr, 0, sizeof(ifr));
        os_strncpy(ifr.ifr_name, drv->ifname, IFNAMSIZ);
        if (ioctl(drv->ioctl_sock, SIOCGIFHWADDR, &ifr) < 0) {
            perror(&quot;ioctl[SIOCGIFHWADDR]&quot;);
            ret = -1;
        } else {
            u8 *macaddr = (u8 *) ifr.ifr_hwaddr.sa_data;
            ret = snprintf(buf, buf_len, &quot;Macaddr = &quot; MACSTR &quot;\n&quot;,
                           MAC2STR(macaddr));
        }
    }
    else if (os_strcasecmp(cmd, &quot;scan-passive&quot;) == 0) {
        wpa_printf(MSG_DEBUG,&quot;Scan Passive command&quot;);
    }
    else if (os_strcasecmp(cmd, &quot;scan-active&quot;) == 0) {
        wpa_printf(MSG_DEBUG,&quot;Scan Active command&quot;);
    }
    else if (os_strcasecmp(cmd, &quot;linkspeed&quot;) == 0) {
        struct iwreq wrq;
        unsigned int linkspeed;
        os_strncpy(wrq.ifr_name, drv->ifname, IFNAMSIZ);
        wpa_printf(MSG_DEBUG,&quot;Link Speed command&quot;);
        if (ioctl(drv->ioctl_sock, SIOCGIWRATE, &wrq) < 0) {
            perror(&quot;ioctl[SIOCGIWRATE]&quot;);
            ret = -1;
        } else {
            linkspeed = wrq.u.bitrate.value / 1000000;
            ret = snprintf(buf, buf_len, &quot;LinkSpeed %d\n&quot;, linkspeed);
        }
    }
    else if (os_strncasecmp(cmd, &quot;scan-channels&quot;, 13) == 0) {
    }
    else if ((os_strcasecmp(cmd, &quot;rssi&quot;) == 0) || (os_strcasecmp(cmd, &quot;rssi-approx&quot;) == 0)) {
        struct iwreq wrq;
        struct iw_statistics stats;
        signed int rssi;
        wpa_printf(MSG_DEBUG, &quot;>>>. DRIVER AWEXT RSSI &quot;);
        wrq.u.data.pointer = (caddr_t) &stats;
        wrq.u.data.length = sizeof(stats);
        wrq.u.data.flags = 1; /* Clear updated flag */
        strncpy(wrq.ifr_name, drv->ifname, IFNAMSIZ);
        if (ioctl(drv->ioctl_sock, SIOCGIWSTATS, &wrq) < 0) {
            perror(&quot;ioctl[SIOCGIWSTATS]&quot;);
            ret = -1;
        } else {
            if (stats.qual.updated & IW_QUAL_DBM) {
                /* Values in dBm, stored in u8 with range 63 : -192 */
                rssi = ( stats.qual.level > 63 ) ?
                    stats.qual.level - 0x100 :
                    stats.qual.level;
            } else {
                rssi = stats.qual.level;
            }
            if (drv->ssid_len != 0 && drv->ssid_len < buf_len) {
                os_memcpy((void *) buf, (void *) (drv->ssid),
                        drv->ssid_len );
                ret = drv->ssid_len;
                ret += snprintf(&buf[ret], buf_len-ret,
                        &quot; rssi %d\n&quot;, rssi);
                if (ret < (int)buf_len) {
                    return( ret );
                }
                ret = -1;
            }
        }
    }
    else if (os_strncasecmp(cmd, &quot;powermode&quot;, 9) == 0) {
    }
    else if (os_strncasecmp(cmd, &quot;getpower&quot;, 8) == 0) {
    }
    else if (os_strncasecmp(cmd, &quot;get-rts-threshold&quot;, 17) == 0) {
        struct iwreq wrq;
        unsigned int rtsThreshold;
        strncpy(wrq.ifr_name, drv->ifname, IFNAMSIZ);
        if (ioctl(drv->ioctl_sock, SIOCGIWRTS, &wrq) < 0) {
            perror(&quot;ioctl[SIOCGIWRTS]&quot;);
            ret = -1;
        } else {
            rtsThreshold = wrq.u.rts.value;
            wpa_printf(MSG_DEBUG,&quot;Get RTS Threshold command = %d&quot;,
                rtsThreshold);
            ret = snprintf(buf, buf_len, &quot;rts-threshold = %u\n&quot;,
                rtsThreshold);
            if (ret < (int)buf_len) {
                return( ret );
            }
        }
    }
    else if (os_strncasecmp(cmd, &quot;set-rts-threshold&quot;, 17) == 0) {
        struct iwreq wrq;
        unsigned int rtsThreshold;
        char *cp = cmd + 17;
        char *endp;
        strncpy(wrq.ifr_name, drv->ifname, IFNAMSIZ);
        if (*cp != '\0') {
            rtsThreshold = (unsigned int)strtol(cp, &endp, 0);
            if (endp != cp) {
                wrq.u.rts.value = rtsThreshold;
                wrq.u.rts.fixed = 1;
                wrq.u.rts.disabled = 0;
                if (ioctl(drv->ioctl_sock, SIOCSIWRTS, &wrq) < 0) {
                    perror(&quot;ioctl[SIOCGIWRTS]&quot;);
                    ret = -1;
                } else {
                    rtsThreshold = wrq.u.rts.value;
                    wpa_printf(MSG_DEBUG,&quot;Set RTS Threshold command = %d&quot;, rtsThreshold);
                    ret = 0;
                }
            }
        }
    }
    else if (os_strcasecmp(cmd, &quot;btcoexscan-start&quot;) == 0) {
    }
    else if (os_strcasecmp(cmd, &quot;btcoexscan-stop&quot;) == 0) {
    }
    else if (os_strcasecmp(cmd, &quot;rxfilter-start&quot;) == 0) {
        wpa_printf(MSG_DEBUG,&quot;Rx Data Filter Start command&quot;);
    }
    else if (os_strcasecmp(cmd, &quot;rxfilter-stop&quot;) == 0) {
        wpa_printf(MSG_DEBUG,&quot;Rx Data Filter Stop command&quot;);
    }
    else if (os_strcasecmp(cmd, &quot;rxfilter-statistics&quot;) == 0) {
    }
    else if (os_strncasecmp(cmd, &quot;rxfilter-add&quot;, 12) == 0 ) {
    }
    else if (os_strncasecmp(cmd, &quot;rxfilter-remove&quot;,15) == 0) {
    }
    else if (os_strcasecmp(cmd, &quot;snr&quot;) == 0) {
        struct iwreq wrq;
        struct iw_statistics stats;
        int snr, rssi, noise;
        wrq.u.data.pointer = (caddr_t) &stats;
        wrq.u.data.length = sizeof(stats);
        wrq.u.data.flags = 1; /* Clear updated flag */
        strncpy(wrq.ifr_name, drv->ifname, IFNAMSIZ);
        if (ioctl(drv->ioctl_sock, SIOCGIWSTATS, &wrq) < 0) {
            perror(&quot;ioctl[SIOCGIWSTATS]&quot;);
            ret = -1;
        } else {
            if (stats.qual.updated & IW_QUAL_DBM) {
                /* Values in dBm, stored in u8 with range 63 : -192 */
                rssi = ( stats.qual.level > 63 ) ?
                    stats.qual.level - 0x100 :
                    stats.qual.level;
                noise = ( stats.qual.noise > 63 ) ?
                    stats.qual.noise - 0x100 :
                    stats.qual.noise;
            } else {
                rssi = stats.qual.level;
                noise = stats.qual.noise;
            }
            snr = rssi - noise;
            ret = snprintf(buf, buf_len, &quot;snr = %u\n&quot;, (unsigned int)snr);
            if (ret < (int)buf_len) {
                return( ret );
            }
        }
    }
    else if (os_strncasecmp(cmd, &quot;btcoexmode&quot;, 10) == 0) {
    }
    else if( os_strcasecmp(cmd, &quot;btcoexstat&quot;) == 0 ) {
    }
    else {
        wpa_printf(MSG_DEBUG,&quot;Unsupported command&quot;);
    }
    return (ret);
}
经过这些修改后,wifi应该可以上去了。
问题留存:
1。这次修改的是5.x版本的wpa_supplicant,android已经出了6.x版本,我在那里没有发现driver_wext.c!,那我上面的修改如何实现?看来还要研究。
2。修改好的driver_wext.c可以显示MAC地址,在上层的设置界面也能显示信号强度,但在主页面上却没有显示强度,估计还要在java程序中找找原因。
3。在设置界面,如果将wifi关闭后再打开,wifi连不上,需要手工在调试终端打ifconfig ra0 up才能触发连接上。估计是java程序在关闭wifi时没有将wifi设置进行初始化操作,要去看看设置的数据库修改情况。
,,

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-121585-1-1.html 上篇帖子: 360随身wifi怎样购买?360随身wifi怎样预约? 下篇帖子: 免费WiFi,仅仅为好久没联系的你们
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表