0%

[OpenWrt]在MT7628上实现wan、wwan与4G动态切换

​ 本人使用基于MT7628的开发板运行openwrt lede17.01系统,单独使用有线wan、无线路由wwan和4G功能时都很正常。但是当同时使用以上3种网络接口时,发现接口之间并不能动态切换,当拔掉网线时,即使wifi和4g都正常连接,但是路由器不能ping通外网。当wifi连接断了,也会影响剩余网络接口的通信。

1478258-20200829231826228-276073933.png

经过一番调试,发现当网线热插拔时,系统并没有触发链路断开事件通知netifd进程来关闭网络,导致其接口的配置残留在路由表中,从而影响其他接口搜索默认网关。

1478258-20200829231845621-1743179636.png

根本原因是MT7628系列设备具有内置交换机功能,由于交换机和CPU端口之间始终连接,因此本地以太网接口永远不会看到链路断开事件,这意味着netifd将不会关闭并取消配置关联的网络,导致热插拔无效。

解决方法

既然是硬件原因导致无法使用热插拔事件,那只好通过轮询方式来检测网络。定期对外网进行ping测试,如果测试正常,则无需切换网络;如果测试失败,则自动切换到另一个网络接口,以此循环。通过以下脚本,能实现网络接口的动态切换。

1.网络测试及切换脚本

通过以下脚本,可以实现周期ping测试和网络接口切换功能。

脚本名称:/etc/pingtest

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
#!/bin/sh

state=0
IP="www.baidu.com"

ping_status() {
for var in 1 2 3; do
if ping -c 1 $IP > /dev/null; then
#echo "$IP Ping is successful."
return 0
else
#echo "$IP Ping is failure"
if [ $var -eq 3 ]; then
return 1
fi
fi
done
}

while true
do
ping_status

if [ $? -ne 0 ]; then
case $state in
0) #echo "wan to wwan"
if uci get network.wwan > /dev/null; then
uci set network.wwan.metric=0
fi

if uci get network.wan > /dev/null; then
uci set network.wan.metric=20
fi

if uci get network.4G > /dev/null; then
uci set network.4G.metric=10
fi
;;
1) #echo "wwan to 4G"
if uci get network.wwan > /dev/null; then
uci set network.wwan.metric=20
fi

if uci get network.wan > /dev/null; then
uci set network.wan.metric=10
fi

if uci get network.4G > /dev/null; then
uci set network.4G.metric=0
fi
;;
2) #echo "4G to wan"
if uci get network.wwan > /dev/null; then
uci set network.wwan.metric=10
fi

if uci get network.wan > /dev/null; then
uci set network.wan.metric=0
fi

if uci get network.4G > /dev/null; then
uci set network.4G.metric=20
fi
;;
esac

uci commit network;
/etc/init.d/network reload;
let state++
if [ ${state} -ge 3 ]; then
state=0
fi
fi

sleep 10

done
  • 进行对外网进行一轮ping测试,最多测试3次,当有一次测试通过则代表成功,不作网络处理;如果3次测试失败,则表示网络已断。
  • 如果网络已断,通过UCI命令修改network文件中各接口的默认网关优先级参数metric(值越小优先级越高),使用uci commit network命令保存修改,并重新加载配置文件:/etc/init.d/network reload。
  • 完成一轮检测后,10秒后再次检测,以此循环。如果当前使用的网络接口断了,脚本就会自动切换到适合的下一个网络接口。

2.开机自启脚本

在/etc/init.d/目录下编写开机启动的shell脚本,该脚本运行时,在后台启动/etc/pingtest脚本。

脚本名称:/etc/init.d/interface_check

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
#!/bin/sh /etc/rc.common
START=98
STOP=99

USE_PROCD=1

start_service()
{
procd_open_instance
procd_set_param respawn
. /etc/pingtest &
procd_close_instance
}

stop_service()
{
return;
}
  • START=98,STOP=99分别是启动和关闭和优先级
  • USE_PROCD=1表示使用进程,配合procd_open_instance和
    procd_close_instance命令启动新进程,让/etc/pingtest脚本在进程中运行。

3.开启启动脚本软链接

在/etc/rc.d目录下新建开机启动脚本/etc/init.d/interface_check的软链接,下次开机脚本就会自动运行。

1
ln -s /etc/init.d/interface_check  /etc/rc.d/S98interface_check

S98是脚本启动的优先级

链接:https://www.jianshu.com/p/f058229a42d8