2 # map.sh - IPv4-in-IPv6 tunnel backend
4 # Author: Steven Barth <cyrus@openwrt.org>
5 # Copyright (c) 2014 cisco Systems, Inc.
7 # This program is free software; you can redistribute it and/or modify
8 # it under the terms of the GNU General Public License version 2
9 # as published by the Free Software Foundation
11 # This program is distributed in the hope that it will be useful,
12 # but WITHOUT ANY WARRANTY; without even the implied warranty of
13 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 # GNU General Public License for more details.
16 [ -n "$INCLUDE_ONLY" ] ||
{
18 .
/lib
/functions
/network.sh
28 local type legacymap mtu ttl tunlink zone encaplimit
29 local rule ipaddr ip4prefixlen ip6prefix ip6prefixlen peeraddr ealen psidlen psid offset
30 json_get_vars
type legacymap mtu ttl tunlink zone encaplimit
31 json_get_vars rule ipaddr ip4prefixlen ip6prefix ip6prefixlen peeraddr ealen psidlen psid offset
33 [ "$zone" = "-" ] && zone
=""
34 [ -z "$type" ] && type="map-e"
35 [ -z "$ip4prefixlen" ] && ip4prefixlen
=32
37 ( proto_add_host_dependency
"$cfg" "::" "$tunlink" )
39 # fixme: handle RA/DHCPv6 address race for LW
40 [ "$type" = lw4o6
] && sleep 5
42 if [ -z "$rule" ]; then
43 rule
="type=$type,ipv6prefix=$ip6prefix,prefix6len=$ip6prefixlen,ipv4prefix=$ipaddr,prefix4len=$ip4prefixlen"
44 [ -n "$psid" ] && rule
="$rule,psid=$psid"
45 [ -n "$psidlen" ] && rule
="$rule,psidlen=$psidlen"
46 [ -n "$offset" ] && rule
="$rule,offset=$offset"
47 [ -n "$ealen" ] && rule
="$rule,ealen=$ealen"
48 if [ "$type" = "map-t" ]; then
49 rule
="$rule,dmr=$peeraddr"
51 rule
="$rule,br=$peeraddr"
55 echo "rule=$rule" > /tmp
/map-
$cfg.rules
56 RULE_DATA
=$
(LEGACY
="$legacymap" mapcalc
${tunlink:-\*} $rule)
57 if [ "$?" != 0 ]; then
58 proto_notify_error
"$cfg" "INVALID_MAP_RULE"
59 proto_block_restart
"$cfg"
63 echo "$RULE_DATA" >> /tmp
/map-
$cfg.rules
66 if [ -z "$RULE_BMR" ]; then
67 proto_notify_error
"$cfg" "NO_MATCHING_PD"
68 proto_block_restart
"$cfg"
73 if [ "$type" = "lw4o6" -o "$type" = "map-e" ]; then
74 proto_init_update
"$link" 1
75 proto_add_ipv4_address $
(eval "echo \$RULE_${k}_IPV4ADDR") "" "" ""
78 json_add_string mode ipip6
79 json_add_int mtu
"${mtu:-1280}"
80 json_add_int ttl
"${ttl:-64}"
81 json_add_string
local $
(eval "echo \$RULE_${k}_IPV6ADDR")
82 json_add_string remote $
(eval "echo \$RULE_${k}_BR")
83 json_add_string link $
(eval "echo \$RULE_${k}_PD6IFACE")
84 json_add_object
"data"
85 [ -n "$encaplimit" ] && json_add_string encaplimit
"$encaplimit"
86 if [ "$type" = "map-e" ]; then
88 for i
in $
(seq $RULE_COUNT); do
89 [ "$(eval "echo \
$RULE_${i}_FMR
")" != 1 ] && continue
91 json_add_string prefix6
"$(eval "echo \
$RULE_${i}_IPV6PREFIX
")/$(eval "echo \
$RULE_${i}_PREFIX6LEN
")"
92 json_add_string prefix4
"$(eval "echo \
$RULE_${i}_IPV4PREFIX
")/$(eval "echo \
$RULE_${i}_PREFIX4LEN
")"
93 json_add_int ealen $
(eval "echo \$RULE_${i}_EALEN")
94 json_add_int offset $
(eval "echo \$RULE_${i}_OFFSET")
103 elif [ "$type" = "map-t" -a -f "/proc/net/nat46/control" ]; then
104 proto_init_update
"$link" 1
106 [ "$legacymap" = 1 ] && style
="MAP0"
108 echo add
$link > /proc
/net
/nat46
/control
109 local cfgstr
="local.style $style local.v4 $(eval "echo \
$RULE_${k}_IPV4PREFIX
")/$(eval "echo \
$RULE_${k}_PREFIX4LEN
")"
110 cfgstr
="$cfgstr local.v6 $(eval "echo \
$RULE_${k}_IPV6PREFIX
")/$(eval "echo \
$RULE_${k}_PREFIX6LEN
")"
111 cfgstr
="$cfgstr local.ea-len $(eval "echo \
$RULE_${k}_EALEN
") local.psid-offset $(eval "echo \
$RULE_${k}_OFFSET
")"
112 cfgstr
="$cfgstr remote.v4 0.0.0.0/0 remote.v6 $(eval "echo \
$RULE_${k}_DMR
") remote.style RFC6052 remote.ea-len 0 remote.psid-offset 0"
113 echo config
$link $cfgstr > /proc
/net
/nat46
/control
115 for i
in $
(seq $RULE_COUNT); do
116 [ "$(eval "echo \
$RULE_${i}_FMR
")" != 1 ] && continue
117 local cfgstr
="remote.style $style remote.v4 $(eval "echo \
$RULE_${i}_IPV4PREFIX
")/$(eval "echo \
$RULE_${i}_PREFIX4LEN
")"
118 cfgstr
="$cfgstr remote.v6 $(eval "echo \
$RULE_${i}_IPV6PREFIX
")/$(eval "echo \
$RULE_${i}_PREFIX6LEN
")"
119 cfgstr
="$cfgstr remote.ea-len $(eval "echo \
$RULE_${i}_EALEN
") remote.psid-offset $(eval "echo \
$RULE_${i}_OFFSET
")"
120 echo insert
$link $cfgstr > /proc
/net
/nat46
/control
123 proto_notify_error
"$cfg" "UNSUPPORTED_TYPE"
124 proto_block_restart
"$cfg"
127 proto_add_ipv4_route
"0.0.0.0" 0
129 [ -n "$zone" ] && json_add_string zone
"$zone"
131 json_add_array firewall
132 if [ -z "$(eval "echo \
$RULE_${k}_PORTSETS
")" ]; then
134 json_add_string
type nat
135 json_add_string target SNAT
136 json_add_string family inet
137 json_add_string snat_ip $
(eval "echo \$RULE_${k}_IPV4ADDR")
140 for portset
in $
(eval "echo \$RULE_${k}_PORTSETS"); do
141 for proto
in icmp tcp udp
; do
143 json_add_string
type nat
144 json_add_string target SNAT
145 json_add_string family inet
146 json_add_string proto
"$proto"
147 json_add_boolean connlimit_ports
1
148 json_add_string snat_ip $
(eval "echo \$RULE_${k}_IPV4ADDR")
149 json_add_string snat_port
"$portset"
154 if [ "$type" = "map-t" ]; then
155 [ -z "$zone" ] && zone
=$
(fw3
-q network
$iface 2>/dev
/null
)
159 json_add_string
type rule
160 json_add_string family inet6
161 json_add_string proto all
162 json_add_string direction
in
163 json_add_string dest
"$zone"
164 json_add_string src
"$zone"
165 json_add_string src_ip $
(eval "echo \$RULE_${k}_IPV6ADDR")
166 json_add_string target ACCEPT
169 json_add_string
type rule
170 json_add_string family inet6
171 json_add_string proto all
172 json_add_string direction out
173 json_add_string dest
"$zone"
174 json_add_string src
"$zone"
175 json_add_string dest_ip $
(eval "echo \$RULE_${k}_IPV6ADDR")
176 json_add_string target ACCEPT
179 proto_add_ipv6_route $
(eval "echo \$RULE_${k}_IPV6ADDR") 128
184 proto_send_update
"$cfg"
186 if [ "$type" = "lw4o6" -o "$type" = "map-e" ]; then
188 json_add_string name
"${cfg}_"
189 json_add_string ifname
"@$(eval "echo \
$RULE_${k}_PD6IFACE
")"
190 json_add_string proto
"static"
191 json_add_array ip6addr
192 json_add_string
"" "$(eval "echo \
$RULE_${k}_IPV6ADDR
")"
195 ubus call network add_dynamic
"$(json_dump)"
199 proto_map_teardown
() {
201 local link
="map-$cfg"
203 json_get_var
type type
205 [ -z "$type" ] && type="map-e"
208 "map-e"|
"lw4o6") ifdown
"${cfg}_" ;;
209 "map-t") [ -f "/proc/net/nat46/control" ] && echo del
$link > /proc
/net
/nat46
/control
;;
212 rm -f /tmp
/map-
$cfg.rules
215 proto_map_init_config
() {
219 proto_config_add_string
"rule"
220 proto_config_add_string
"ipaddr"
221 proto_config_add_int
"ip4prefixlen"
222 proto_config_add_string
"ip6prefix"
223 proto_config_add_int
"ip6prefixlen"
224 proto_config_add_string
"peeraddr"
225 proto_config_add_int
"ealen"
226 proto_config_add_int
"psidlen"
227 proto_config_add_int
"psid"
228 proto_config_add_int
"offset"
229 proto_config_add_boolean
"legacymap"
231 proto_config_add_string
"tunlink"
232 proto_config_add_int
"mtu"
233 proto_config_add_int
"ttl"
234 proto_config_add_string
"zone"
235 proto_config_add_string
"encaplimit"
238 [ -n "$INCLUDE_ONLY" ] ||
{