为什么xor()不适用于openwrt

时间:2017-06-14 02:38:35

标签: awk xor openwrt dnsmasq

我们正在将openwrt移植到我们的设备上,这是一个mips平台。但我们找到了服务" dnsmasq"无法在我们的设备上正确启动。

然后我们找到了根本原因," /etc/init.d/dnsmasq"使用脚本" /bin/ipcalc.sh"为" dnsmasq"生成错误的配置和" dnsmasq"无法解析配置。

以下是" /bin/ipcalc.sh"的原始代码的链接;在github https://github.com/respeaker/openwrt/blob/master/package/base-files/files/bin/ipcalc.sh

然后我们用compl()替换compl32()的所有调用,然后它就可以了。当我们调查脚本时,发现compl32()是一个非常简单的函数,它只调用xor()。但看起来像xor()无法产生我们期待的东西。

这是我们正在使用的/bin/ipcalc.sh的代码,我在其中添加了一些调试:

#!/bin/sh

awk -f - $* <<EOF
function bitcount(c) {
    c=and(rshift(c, 1),0x55555555)+and(c,0x55555555)
    c=and(rshift(c, 2),0x33333333)+and(c,0x33333333)
    c=and(rshift(c, 4),0x0f0f0f0f)+and(c,0x0f0f0f0f)
    c=and(rshift(c, 8),0x00ff00ff)+and(c,0x00ff00ff)
    c=and(rshift(c,16),0x0000ffff)+and(c,0x0000ffff)
    return c
}

function ip2int(ip) {
    for (ret=0,n=split(ip,a,"\."),x=1;x<=n;x++)    ret=or(lshift(ret,8),a[x])
    return ret
}

function int2ip(ip,ret,x) {
    ret=and(ip,255)
    ip=rshift(ip,8)
    for(;x<3;ret=and(ip,255)"."ret,ip=rshift(ip,8),x++);
    return ret
}

function compl32(v) {
    ret=xor(v, 0xffffffff)
    return ret
}

BEGIN {
    print "var1="compl32(255)
    print "var2="compl(0xff)
    print "var3="xor(0xff,0xffffffff)
    print "var4="xor(255,0xffffffff)
    slpos=index(ARGV[1],"/")
    if (slpos == 0) {
            ipaddr=ip2int(ARGV[1])
            dotpos=index(ARGV[2],".")
            if (dotpos == 0)
                    netmask=compl(2**(32-int(ARGV[2]))-1)
            else
                    netmask=ip2int(ARGV[2])
    } else {
            ipaddr=ip2int(substr(ARGV[1],0,slpos-1))
            netmask=compl(2**(32-int(substr(ARGV[1],slpos+1)))-1)
            ARGV[4]=ARGV[3]
            ARGV[3]=ARGV[2]
    }

    network=and(ipaddr,netmask)
    broadcast=or(network,compl(netmask))

    start=or(network,and(ip2int(ARGV[3]),compl(netmask)))
    limit=network+1
    if (start<limit) start=limit

    end=start+ARGV[4]
    limit=or(network,compl(netmask))-1
    if (end>limit) end=limit

    print "IP="int2ip(ipaddr)
    print "NETMASK="int2ip(netmask)
    print "BROADCAST="int2ip(broadcast)
    print "NETWORK="int2ip(network)
    print "PREFIX="32-bitcount(compl(netmask))

    # range calculations:
    # ipcalc <ip> <netmask> <start> <num>

    if (ARGC > 3) {
            print "START="int2ip(start)
            print "END="int2ip(end)
    }
}
EOF

以下是运行结果:

root@OpenWrt:/tmp# ipcalc.sh 192.168.1.1/24 100 50
var1=255
var2=4294967295
var3=0
var4=255
IP=192.168.1.1
NETMASK=255.255.255.0
BROADCAST=192.168.1.255
NETWORK=192.168.1.0
PREFIX=32
START=192.168.1.100
END=192.168.1.150
root@OpenWrt:/tmp#

这很奇怪。为什么xor()不起作用?

2 个答案:

答案 0 :(得分:0)

您在BEGIN for var3中的表达式在我的系统上产生var3 = 4294967040(GNU Awk 4.1.4)。看起来你正在使用一个破碎的Awk-请尝试不同的版本。

答案 1 :(得分:0)

$ man awk

标准 awk实用程序符合IEEE Std 1003.1-2008(&#39; POSIX.1&#39;&#39;) 规范,除了awk不支持{n,m}模式匹配。

标志[-dV]和[-safe],以及命令 fflush compl xor lshift rshift ,是该规范的扩展

根据我的经验,FreeBSD中的某些实现并没有完全实现这些扩展。