Android下同时使用WIFI与3G网络
http://blog.csdn.net/roger__wong/article/details/8603275
编程时候发现当打开WIFI网络的时候系统默认就会把3G网络给关闭,估计是出于省电的原因才如此做的,但目前我需要同时使用这两个网络接口,经过一番研究找到了方法,故写成日志记录下来。
1、Android网络机制。
wifi的开启关闭首先由应用程序触发的事件开始(比如用户点击开启图标),然后传入framework层,framework层启动wpa_supplicant,同时建立socket连接与wpa_supplicant进行通信,包括对wpa_supplicant的控制与得到wpa_supplicant的回传结果。若连接上了一个无线网络,则调用dhcp服务申请ip地址(或者使用预先设置好的ip,不过大部分情况下是dhcp的)。若这一切正常则断开3g网络连接并关闭3g接口。
因此从这个角度分析,若要同时启动3g和wifi则必须绕过framework层,手动完成上述所有操作。
因此我找到的方法是在3g网络不断的情况下,手动启动wifi网络,连接指定ap并申请ip。
2、操作步骤。
(1) 准备工作
因为牵扯到底层操作,所以root必不可少,在root的情况下进入adb shell,若adb daemon没有root使用adb root命令进而使adb shell获得root权限。
原则上使用android里的终端软件也是可以的,但我没有测试。
(2)加载wifi驱动。
不加载驱动在netcfg里面根本看不到wlan0网卡,因此首先要加载驱动。
一般情况下,驱动放在/system/lib/modules下,我的驱动模块是wlan.ko。
insmod /system/lib/modules/wlan.ko。
加载前:
加载后:
发现多出来了wlan0也就是wifi网卡
(3)开启wifi网卡
使用netcfg wlan0 up开启wifi网卡,使用ifconfig wlan0 up也同样可以。
开启后:
(4)连接ap
启动wpa_supplicant并使之连接ap,在使用android系统时,如果开启wifi后范围内有一个之前连接过的wifi(而且这个wifi的ssid和密码还没改的话),那么我们的系统就会自动连接到这个wifi。因此手机里肯定是在某处记录者各个曾经连接过的wifi的ssid和密码。
我的手机(小米2,miui开发版3.xxx)这个默认位置是/data/misc/wifi/wpa_supplicant.conf。当把这个配置文件传给wpa_supplicant启动的时候,就会自动尝试连接其中记录的wifi网络。
使用命令wpa_supplicant -iwlan0 -c/data/misc/wifi/wpa_supplicant.conf -B 启动wpa_supplicant,-B代表后台运行,-i指定无线网卡。
此时若一切正常就会连接上无线ap,使用dmesg查看信息得到:
可以看到有连接成功的信息。
(5)使用dhcp获取ip信息
连接上了暂时还不能用,因为没有配置ip,使用dhcpcd wlan0配置ip信息,使用netcfg wlan0 dhcp也可以。
若出现类似以下信息则说明成功。
此时我们就在android下同时启用了wifi和3g。
此时使用netcfg可以看到两个接口都有ip,说明我们的方法有效。
当然最重要的是以上操作都可以在android用简单的程序来实现。
使用Runtime.getRuntime().exec(new String[] { "su","your command" });来执行命令,之后拿到相应输出流可以得到执行的结果,根据结果再执行下一步命令即可。
(6)更改路由表
到此为止手机已经连接上了无线AP,但本机数据包还无法通过WiFi接口走,原因是没有配置路由导致OS并不知道什么数据包该从WiFi走,什么数据包该从3G走,所以最后一步就是配置路由,告诉OS若IP包的源地址是3G接口的IP,则从3G端口走,否则从WiFi端口走。配置过程及命令如下,主要是使用了了ip命令:
ip route add table 5 via 192.168.1.1 dev wlan0 //添加一个路由表,若通过该路由表路由,则通过设备wlan0路由到地址192.168.1.1,也就是无线AP
ip route add from 192.168.1.140 table 5 prio 30000 //添加路由项,若源地址是192.168.1.140,则通过路由表5进行路由,优先级为30000
类似的,给3G接口也添加路由条目,优先级稍微低一点
然后设置dns,因为WiFi接口没法使用3G接口的默认DNS
setprop net.dns1 8.8.8.8
接着删除默认路由表项
ip route del default via 172.18.206.76 dev rmnet_usb0
重新添加指向wifi的默认路由表
ip route add default via 192.168.1.140 dev wlan0