OpenWrt获取网关IP

OpenWrt获取网关IP

一、概述

OpenWrt也是Linux,题目其实也可以叫做“Linux获取网关IP”。一般想得知网关IP,都是因为本地接口设置了DHCP,网关IP,大多数也即是DHCP Server的IP(DHCP Relay除外)。

二、一般方法汇总

1、猜

没错,是猜,因为网关IP一般为XXX.XXX.XXX.1,所以ifconfig出来的接口IP,就可以推测出网关的IP。

2、查路由表

当连接建立之后,本地一般会生成默认的路由,因此查路由表,就可以直接得知网关IP。
此类方法有很多,比如一些命令,以及以这些命令为基础制作的脚本
1)route -n
2)ip route show
3)netstat -r
。。。
还有一些其他命令,如traceroute等。

3、写代码获取内核消息

是方法2的进阶,除了利用命令写脚本,也可以通过与内核通信,获取一些信息
如果这个博文所转载的http://blog.csdn.net/liangyamin/article/details/7242048,通过NLMSG从内核获取路由信息,本质还是在查路由表,

4、查看resolv.conf

针对dhcp,一般如果使用了dnsmasq,会记录一些信息如网关、dns等,这些信息一般会在resolv.conf/resolv.conf.auto等文件中,而这个文件大部会在/var/resolv.conf、/etc/resolv.conf等目录,具体可以查看dhcp或dnsmasq的配置文件。

三、解析DHCP消息

这里想着重提一下从dhcp消息中,获取dhcp server ip(即网关IP)。DHCP的详细原理这里就不在冗述了,直接看图

DHCP原理图

本端作为dhcp client,从收到dhcp offer开始,就已经知道dhcp server的IP,只要解析这个消息即可。
dhcp offer报文格式中的Siaddr字段内容,即dhcp server IP.

Siaddr: IP address of next server to use in bootstrap.

这个才是最根本的源头。
那么问题来了,dhcp程序有很多,比如常用的有dhclient或busybox中的udhcpc等,如何从进程中获得dhcp Packet的内容呢?
一种方式是打补丁,这些代码都是开源的,如果开发的系统允许自行修改,可以干脆修改dhcp源码增加一些输出。
另外发现dhcp程序的参数支持script运行,比如udhcpc的-s参数
-s,--script PROG Run PROG at DHCP events (default /usr/share/udhcpc/default.script)
以OpenWrt为例

root@OpenWrt:/# ps |grep dhcp
 1384 root      1516 S    udhcpc -S -p /var/run/udhcpc-ath01.pid -s /lib/netifd/dhcp.script -f -t 0 -i ath01 -C
 3212 root      1500 S    grep dhcp
root@OpenWrt:/# 

/lib/netifd/dhcp.script的内容:

root@OpenWrt:/# cat /lib/netifd/dhcp.script
#!/bin/sh
[ -z "$1" ] && echo "Error: should be run by udhcpc" && exit 1

. /lib/functions.sh
. /lib/netifd/netifd-proto.sh

set_classless_routes() {
    local max=128
    local type
    while [ -n "$1" -a -n "$2" -a $max -gt 0 ]; do
        proto_add_ipv4_route "${1%%/*}" "${1##*/}" "$2"
        max=$(($max-1))
        shift 2
    done
}

setup_interface () {
    proto_init_update "$IFNAME" 1
    proto_add_ipv4_address "$ip" "${subnet:-255.255.255.0}"
    # TODO: apply $broadcast

    for i in $router; do
        echo "i=$i" > /dev/console
        proto_add_ipv4_route 0.0.0.0 0 "$i"
    done

    # CIDR STATIC ROUTES (rfc3442)
    [ -n "$staticroutes" ] && set_classless_routes $staticroutes
    [ -n "$msstaticroutes" ] && set_classless_routes $msstaticroutes

    for dns in $dns; do
        proto_add_dns_server "$dns"
    done
    for domain in $domain; do
        proto_add_dns_search "$domain"
    done
    proto_send_update "$INTERFACE"

    # TODO
    # [ -n "$ntpsrv" ] && 	change_state network "$ifc" lease_ntpsrv "$ntpsrv"
    # [ -n "$timesvr" ] && 	change_state network "$ifc" lease_timesrv "$timesvr"
    # [ -n "$hostname" ] &&	change_state network "$ifc" lease_hostname "$hostname"
    # [ -n "$timezone" ] && 	change_state network "$ifc" lease_timezone "$timezone"
}

deconfig_interface() {
    proto_init_update "*" 0
    proto_send_update "$INTERFACE"
}

case "$1" in
    deconfig)
        deconfig_interface
    ;;
    renew|bound)
        setup_interface
    ;;
esac

echo "4=$4" > /dev/console

# user rules
[ -f /etc/udhcpc.user ] && . /etc/udhcpc.user

exit 0
root@OpenWrt:/# 

经测试$router的值就是Siaddr的值。

你可能感兴趣的:(openwrt,linux,dhcp,openwrt,dhcp,linux)