network.subr revision 134429
125184Sjkh#
2113674Smtm# Copyright (c) 2003 The FreeBSD Project. All rights reserved.
3113674Smtm#
4113674Smtm# Redistribution and use in source and binary forms, with or without
5113674Smtm# modification, are permitted provided that the following conditions
6113674Smtm# are met:
7113674Smtm# 1. Redistributions of source code must retain the above copyright
8113674Smtm#    notice, this list of conditions and the following disclaimer.
9113674Smtm# 2. Redistributions in binary form must reproduce the above copyright
10113674Smtm#    notice, this list of conditions and the following disclaimer in the
11113674Smtm#    documentation and/or other materials provided with the distribution.
12113674Smtm#
13113674Smtm# THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``AS IS'' AND
14113674Smtm# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15113674Smtm# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16113674Smtm# ARE DISCLAIMED.  IN NO EVENT SHALL THE PROJECT OR CONTRIBUTORS BE LIABLE
17113674Smtm# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18113674Smtm# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19113674Smtm# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20113674Smtm# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21113674Smtm# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22113674Smtm# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
23113674Smtm# SUCH DAMAGE.
24113674Smtm#
2550472Speter# $FreeBSD: head/etc/network.subr 134429 2004-08-28 07:58:02Z yar $
2666830Sobrien#
2725184Sjkh
28113674Smtm#
29113674Smtm# Subroutines commonly used from network startup scripts.
30113674Smtm# Requires that rc.conf be loaded first.
31113674Smtm#
3225184Sjkh
33113674Smtm# ifconfig_up if
34113674Smtm#	Evaluate ifconfig(8) arguments for interface $if and
35113674Smtm#	run ifconfig(8) with those arguments. It returns 0 if
36113674Smtm#	arguments were found and executed or 1 if the interface
37113674Smtm#	had no arguments.
38113674Smtm#
39113674Smtmifconfig_up()
40113674Smtm{
41113674Smtm	eval ifconfig_args=\$ifconfig_$1
42113674Smtm	if [ -n "${ifconfig_args}" ]; then
43113674Smtm		ifconfig $1 ${ifconfig_args}
44113674Smtm		return 0
45113674Smtm	fi
46113674Smtm	return 1
47113674Smtm}
4825184Sjkh
49116029Smtm# ifconfig_down if
50116029Smtm#	Remove all inet entries from the $if interface. It returns
51116029Smtm#	0 if inet entries were found and removed. It returns 1 if
52116100Smtm#	no entries were found or they could not be removed.
53116029Smtm#
54116029Smtmifconfig_down()
55116029Smtm{
56116029Smtm	[ -z "$1" ] && return 1
57116029Smtm	_ifs="^"
58116029Smtm	_ret=1
59116029Smtm
60116029Smtm	inetList="`ifconfig $1 | grep 'inet ' | tr "\n" "$_ifs"`"
61116029Smtm
62116029Smtm	oldifs="$IFS"
63116029Smtm	IFS="$_ifs"
64116029Smtm	for _inet in $inetList ; do
65116029Smtm		# get rid of extraneous line
66116029Smtm		[ -z "$_inet" ] && break
67116029Smtm
68116029Smtm		_inet=`expr "$_inet" : '.*\(inet \([0-9]\{1,3\}\.\)\{3\}[0-9]\{1,3\}\).*'`
69116029Smtm
70116032Smtm		IFS="$oldifs"
71116032Smtm		ifconfig $1 ${_inet} delete
72116032Smtm		IFS="$_ifs"
73116029Smtm		_ret=0
74116029Smtm	done
75116029Smtm	IFS="$oldifs"
76116029Smtm
77116029Smtm	return $_ret
78116029Smtm}
79116029Smtm
80113674Smtm# ifalias_up if
81113674Smtm#	Configure aliases for network interface $if.
82113674Smtm#	It returns 0 if at least one alias was configured or
83113674Smtm#	1 if there were none.
84113674Smtm#
85113674Smtmifalias_up()
86113674Smtm{
87113674Smtm	_ret=1
88113674Smtm	alias=0
89113674Smtm	while : ; do
90113674Smtm		eval ifconfig_args=\$ifconfig_$1_alias${alias}
91113674Smtm		if [ -n "${ifconfig_args}" ]; then
92113674Smtm			ifconfig $1 ${ifconfig_args} alias
93113674Smtm			alias=$((${alias} + 1))
94113674Smtm			_ret=0
95113674Smtm		else
96113674Smtm			break
97113674Smtm		fi
98113674Smtm	done
99113674Smtm	return $_ret
100113674Smtm}
101100280Sgordon
102116029Smtm#ifalias_down if
103116029Smtm#	Remove aliases for network interface $if.
104116029Smtm#	It returns 0 if at least one alias was removed or
105116029Smtm#	1 if there were none.
106116029Smtm#
107116029Smtmifalias_down()
108116029Smtm{
109116029Smtm	_ret=1
110116029Smtm	alias=0
111116029Smtm	while : ; do
112116029Smtm		eval ifconfig_args=\$ifconfig_$1_alias${alias}
113116029Smtm		if [ -n "${ifconfig_args}" ]; then
114116029Smtm			ifconfig $1 ${ifconfig_args} -alias
115116029Smtm			alias=$((${alias} + 1))
116116029Smtm			_ret=0
117116029Smtm		else
118116029Smtm			break
119116029Smtm		fi
120116029Smtm	done
121116029Smtm	return $_ret
122116029Smtm}
123116029Smtm
124113674Smtm# ifscript_up if
125113674Smtm#	Evaluate a startup script for the $if interface.
126113674Smtm#	It returns 0 if a script was found and processed or
127113674Smtm#	1 if no script was found.
128113674Smtm#
129113674Smtmifscript_up()
130100280Sgordon{
131113674Smtm	if [ -r /etc/start_if.$1 ]; then
132113674Smtm		. /etc/start_if.$1
133113674Smtm		return 0
134113674Smtm	fi
135113674Smtm	return 1
136100280Sgordon}
137100280Sgordon
138116029Smtm# ifscript_down if
139116029Smtm#	Evaluate a shutdown script for the $if interface.
140116029Smtm#	It returns 0 if a script was found and processed or
141116029Smtm#	1 if no script was found.
142116029Smtm#
143116029Smtmifscript_down()
144116029Smtm{
145116029Smtm	if [ -r /etc/stop_if.$1 ]; then
146116029Smtm		. /etc/stop_if.$1
147116029Smtm		return 0
148116029Smtm	fi
149116029Smtm	return 1
150116029Smtm}
151116029Smtm
152113674Smtm# Create cloneable interfaces.
153113674Smtm#
154113674Smtmclone_up()
155100280Sgordon{
156113674Smtm	_prefix=
157113674Smtm	_list=
158113674Smtm	for ifn in ${cloned_interfaces}; do
159113674Smtm		ifconfig ${ifn} create
160116774Skuriyama		if [ $? -eq 0 ]; then
161113674Smtm			_list="${_list}${_prefix}${ifn}"
162113674Smtm			[ -z "$_prefix" ] && _prefix=' '
163113674Smtm		fi
164113674Smtm	done
165113674Smtm	debug "Cloned: ${_list}"
166113674Smtm}
167100280Sgordon
168113674Smtm# Destroy cloned interfaces. Destroyed interfaces are echoed
169113674Smtm# to standard output.
170113674Smtm#
171113674Smtmclone_down()
172113674Smtm{
173113674Smtm	_prefix=
174113674Smtm	_list=
175113674Smtm	for ifn in ${cloned_interfaces}; do
176113674Smtm		ifconfig ${ifn} destroy
177116774Skuriyama		if [ $? -eq 0 ]; then
178113674Smtm			_list="${_list}${_prefix}${ifn}"
179113674Smtm			[ -z "$_prefix" ] && _prefix=' '
180113674Smtm		fi
181113674Smtm	done
182113674Smtm	debug "Destroyed clones: ${_list}"
183100280Sgordon}
184100280Sgordon
185113674Smtmgif_up() {
186100282Sdougb	case ${gif_interfaces} in
187100282Sdougb	[Nn][Oo] | '')
188100282Sdougb		;;
189100282Sdougb	*)
190100282Sdougb		for i in ${gif_interfaces}; do
191100282Sdougb			eval peers=\$gifconfig_$i
192100282Sdougb			case ${peers} in
193100282Sdougb			'')
194100282Sdougb				continue
195100282Sdougb				;;
196100282Sdougb			*)
197100282Sdougb				ifconfig $i create >/dev/null 2>&1
198100282Sdougb				ifconfig $i tunnel ${peers}
199103710Sume				ifconfig $i up
200100282Sdougb				;;
201100282Sdougb			esac
202100282Sdougb		done
203100282Sdougb		;;
204100282Sdougb	esac
205100282Sdougb}
206100282Sdougb
207113674Smtm#
208113674Smtm# ipx_up ifn
209113674Smtm# Configure any IPX addresses for interface $ifn. Returns 0 if IPX
210113674Smtm# arguments were found and configured; returns 1 otherwise.
211113674Smtm#
212113674Smtmipx_up()
213100280Sgordon{
214113674Smtm	ifn="$1"
215113674Smtm	eval ifconfig_args=\$ifconfig_${ifn}_ipx
216113674Smtm	if [ -n "${ifconfig_args}" ]; then
217113674Smtm		ifconfig ${ifn} ${ifconfig_args}
218113674Smtm		return 0
21985831Sdes	fi
220113674Smtm	return 1
221113674Smtm}
22285831Sdes
223116029Smtm# ipx_down ifn
224116029Smtm#	Remove IPX addresses for interface $ifn. Returns 0 if IPX
225116029Smtm#	addresses were found and unconfigured. It returns 1, otherwise.
226113674Smtm#
227116029Smtmipx_down()
228116029Smtm{
229116100Smtm	[ -z "$1" ] && return 1
230116100Smtm	_ifs="^"
231116100Smtm	_ret=1
232116100Smtm
233116100Smtm	ipxList="`ifconfig $1 | grep 'ipx ' | tr "\n" "$_ifs"`"
234116100Smtm
235116100Smtm	oldifs="$IFS"
236116100Smtm	IFS="$_ifs"
237116100Smtm	for _ipx in $ipxList ; do
238116100Smtm		# get rid of extraneous line
239116100Smtm		[ -z "$_ipx" ] && break
240116100Smtm
241116100Smtm		_ipx=`expr "$_ipx" : '.*\(ipx [0-9a-h]\{1,8\}H*\.[0-9a-h]\{1,12\}\).*'`
242116100Smtm
243116100Smtm		IFS="$oldifs"
244116100Smtm		ifconfig $1 ${_ipx} delete
245116100Smtm		IFS="$_ifs"
246116100Smtm		_ret=0
247116100Smtm	done
248116100Smtm	IFS="$oldifs"
249116100Smtm
250116100Smtm	return $_ret
251116029Smtm}
252116029Smtm
253116029Smtm#
254113674Smtm# list_net_interfaces type
255113674Smtm#	List all network interfaces. The type of interface returned
256113674Smtm#	can be controlled by the type argument. The type
257113674Smtm#	argument can be any of the following:
258113674Smtm#		nodhcp - all interfaces, excluding DHCP configured interfaces
259113674Smtm#		dhcp   - list only DHCP configured interfaces
260113674Smtm#	If no argument is specified all network interfaces are output.
261134429Syar#	Note that the list will include cloned interfaces if applicable.
262134429Syar#	Cloned interfaces must already exist to have a chance to appear
263134429Syar#	in the list if ${network_interfaces} is set to `auto'.
264113674Smtm#
265113674Smtmlist_net_interfaces()
266113674Smtm{
267113674Smtm	type=$1
26865532Snectar
269113674Smtm	# Get a list of ALL the interfaces
27051231Ssheldonh	#
27151231Ssheldonh	case ${network_interfaces} in
27251231Ssheldonh	[Aa][Uu][Tt][Oo])
273113674Smtm		_tmplist="`ifconfig -l`"
27451231Ssheldonh		;;
27583677Sbrooks	*)
276134429Syar		_tmplist="${network_interfaces} ${cloned_interfaces}"
27783677Sbrooks		;;
27851231Ssheldonh	esac
27949122Sbrian
280113674Smtm	if [ -z "$type" ]; then
281113674Smtm		echo $_tmplist
282113674Smtm		return 0
283113674Smtm	fi
28449122Sbrian
285113674Smtm	# Separate out dhcp and non-dhcp intefraces
286113674Smtm	#
287113674Smtm	_aprefix=
288134376Syar	_bprefix=
289113674Smtm	for _if in ${_tmplist} ; do
290113674Smtm		eval _ifarg="\$ifconfig_${_if}"
291113674Smtm		case "$_ifarg" in
29251231Ssheldonh		[Dd][Hh][Cc][Pp])
293113674Smtm			_dhcplist="${_dhcplist}${_aprefix}${_if}"
294113674Smtm			[ -z "$_aprefix" ] && _aprefix=' '
29551231Ssheldonh			;;
296113674Smtm		''|*)
297113674Smtm			_nodhcplist="${_nodhcplist}${_bprefix}${_if}"
298113674Smtm			[ -z "$_bprefix" ] && _bprefix=' '
29951231Ssheldonh			;;
30051231Ssheldonh		esac
30154458Sobrien	done
30251231Ssheldonh
303118797Smbr	case ${pccard_ifconfig} in
304118797Smbr	[Dd][Hh][Cc][Pp])
305118797Smbr		for _if in ${removable_interfaces} ; do
306118797Smbr			_test_if=`ifconfig ${_if} 2>&1`
307118797Smbr			case "$_test_if" in
308118797Smbr			"ifconfig: interface $_if does not exist")
309118797Smbr				;;
310118797Smbr			*)
311118797Smbr				_dhcplist="${_dhcplist}${_aprefix}${_if}"
312118797Smbr				[ -z "$_aprefix" ] && _aprefix=' '
313118797Smbr				;;
314118797Smbr			esac
315118797Smbr		done
316118797Smbr		;;
317118797Smbr	*)
318118797Smbr		;;
319118797Smbr	esac
320118797Smbr
321113674Smtm	case "$type" in
322113674Smtm	nodhcp)
323113674Smtm		echo $_nodhcplist
324113674Smtm		;;
325113674Smtm	dhcp)
326113674Smtm		echo $_dhcplist
327113674Smtm		;;
328113674Smtm	esac
329130151Sschweikh	return 0
33025184Sjkh}
331114942Sume
332114942Sumehexdigit()
333114942Sume{
334114942Sume	if [ $1 -lt 10 ]; then
335114942Sume		echo $1
336114942Sume	else
337114942Sume		case $1 in
338114942Sume		10)	echo a ;;
339114942Sume		11)	echo b ;;
340114942Sume		12)	echo c ;;
341114942Sume		13)	echo d ;;
342114942Sume		14)	echo e ;;
343114942Sume		15)	echo f ;;
344114942Sume		esac
345114942Sume	fi
346114942Sume}
347114942Sume
348114942Sumehexprint()
349114942Sume{
350114942Sume	val=$1
351114942Sume	str=''
352114942Sume
353114942Sume	dig=`hexdigit $((${val} & 15))`
354114942Sume	str=${dig}${str}
355114942Sume	val=$((${val} >> 4))
356114942Sume	while [ ${val} -gt 0 ]; do
357114942Sume		dig=`hexdigit $((${val} & 15))`
358114942Sume		str=${dig}${str}
359114942Sume		val=$((${val} >> 4))
360114942Sume	done
361114942Sume
362114942Sume	echo ${str}
363114942Sume}
364114942Sume
365114942Sume# Setup the interfaces for IPv6
366114942Sumenetwork6_interface_setup()
367114942Sume{
368114942Sume	interfaces=$*
369114942Sume	rtsol_interfaces=''
370114942Sume	case ${ipv6_gateway_enable} in
371114942Sume	[Yy][Ee][Ss])
372114942Sume		rtsol_available=no
373114942Sume		;;
374114942Sume	*)
375114942Sume		rtsol_available=yes
376114942Sume		;;
377114942Sume	esac
378114942Sume	for i in $interfaces; do
379114942Sume		rtsol_interface=yes
380114942Sume		eval prefix=\$ipv6_prefix_$i
381114942Sume		if [ -n "${prefix}" ]; then
382114942Sume			rtsol_available=no
383114942Sume			rtsol_interface=no
384114942Sume			laddr=`network6_getladdr $i`
385114942Sume			hostid=`expr "${laddr}" : 'fe80::\(.*\)%\(.*\)'`
386114942Sume			for j in ${prefix}; do
387114942Sume				address=$j\:${hostid}
388114942Sume				ifconfig $i inet6 ${address} prefixlen 64 alias
389114942Sume
390114942Sume				case ${ipv6_gateway_enable} in
391114942Sume				[Yy][Ee][Ss])
392114942Sume					# subnet-router anycast address
393114942Sume					# (rfc2373)
394114942Sume					ifconfig $i inet6 $j:: prefixlen 64 \
395114942Sume						alias anycast
396114942Sume					;;
397114942Sume				esac
398114942Sume			done
399114942Sume		fi
400114942Sume		eval ipv6_ifconfig=\$ipv6_ifconfig_$i
401114942Sume		if [ -n "${ipv6_ifconfig}" ]; then
402114942Sume			rtsol_available=no
403114942Sume			rtsol_interface=no
404114942Sume			ifconfig $i inet6 ${ipv6_ifconfig} alias
405114942Sume		fi
406114942Sume
407114942Sume		if [ ${rtsol_available} = yes -a ${rtsol_interface} = yes ]
408114942Sume		then
409114942Sume			case ${i} in
410114942Sume			lo0|gif[0-9]*|stf[0-9]*|faith[0-9]*|lp[0-9]*|sl[0-9]*|tun[0-9]*)
411114942Sume				;;
412114942Sume			*)
413114942Sume				rtsol_interfaces="${rtsol_interfaces} ${i}"
414114942Sume				;;
415114942Sume			esac
416114942Sume		else
417114942Sume			ifconfig $i inet6
418114942Sume		fi
419114942Sume	done
420114942Sume
421114942Sume	if [ ${rtsol_available} = yes -a -n "${rtsol_interfaces}" ]; then
422114942Sume		# Act as endhost - automatically configured.
423114942Sume		# You can configure only single interface, as
424114942Sume		# specification assumes that autoconfigured host has
425114942Sume		# single interface only.
426114942Sume		sysctl net.inet6.ip6.accept_rtadv=1
427114942Sume		set ${rtsol_interfaces}
428114942Sume		ifconfig $1 up
429118666Sume		rtsol ${rtsol_flags} $1
430114942Sume	fi
431114942Sume
432114942Sume	for i in $interfaces; do
433114942Sume		alias=0
434114942Sume		while : ; do
435114942Sume			eval ipv6_ifconfig=\$ipv6_ifconfig_${i}_alias${alias}
436114942Sume			if [ -z "${ipv6_ifconfig}" ]; then
437114942Sume				break;
438114942Sume			fi
439114942Sume			ifconfig $i inet6 ${ipv6_ifconfig} alias
440114942Sume			alias=$((${alias} + 1))
441114942Sume		done
442114942Sume	done
443114942Sume}
444114942Sume
445114942Sume# Setup IPv6 to IPv4 mapping
446114942Sumenetwork6_stf_setup()
447114942Sume{
448114942Sume	case ${stf_interface_ipv4addr} in
449114942Sume	[Nn][Oo] | '')
450114942Sume		;;
451114942Sume	*)
452114942Sume		# assign IPv6 addr and interface route for 6to4 interface
453114942Sume		stf_prefixlen=$((16+${stf_interface_ipv4plen:-0}))
454114942Sume		OIFS="$IFS"
455114942Sume		IFS=".$IFS"
456114942Sume		set ${stf_interface_ipv4addr}
457114942Sume		IFS="$OIFS"
458114942Sume		hexfrag1=`hexprint $(($1*256 + $2))`
459114942Sume		hexfrag2=`hexprint $(($3*256 + $4))`
460114942Sume		ipv4_in_hexformat="${hexfrag1}:${hexfrag2}"
461114942Sume		case ${stf_interface_ipv6_ifid} in
462114942Sume		[Aa][Uu][Tt][Oo] | '')
463114942Sume			for i in ${ipv6_network_interfaces}; do
464114942Sume				laddr=`network6_getladdr ${i}`
465114942Sume				case ${laddr} in
466114942Sume				'')
467114942Sume					;;
468114942Sume				*)
469114942Sume					break
470114942Sume					;;
471114942Sume				esac
472114942Sume			done
473114942Sume			stf_interface_ipv6_ifid=`expr "${laddr}" : \
474114942Sume						      'fe80::\(.*\)%\(.*\)'`
475114942Sume			case ${stf_interface_ipv6_ifid} in
476114942Sume			'')
477114942Sume				stf_interface_ipv6_ifid=0:0:0:1
478114942Sume				;;
479114942Sume			esac
480114942Sume			;;
481114942Sume		esac
482114942Sume		ifconfig stf0 create >/dev/null 2>&1
483114942Sume		ifconfig stf0 inet6 2002:${ipv4_in_hexformat}:${stf_interface_ipv6_slaid:-0}:${stf_interface_ipv6_ifid} \
484114942Sume			prefixlen ${stf_prefixlen}
485114942Sume		# disallow packets to malicious 6to4 prefix
486114942Sume		route add -inet6 2002:e000:: -prefixlen 20 ::1 -reject
487114942Sume		route add -inet6 2002:7f00:: -prefixlen 24 ::1 -reject
488114942Sume		route add -inet6 2002:0000:: -prefixlen 24 ::1 -reject
489114942Sume		route add -inet6 2002:ff00:: -prefixlen 24 ::1 -reject
490114942Sume		;;
491114942Sume	esac
492114942Sume}
493114942Sume
494114942Sume# Setup static routes
495114942Sumenetwork6_static_routes_setup()
496114942Sume{
497114942Sume	# Set up any static routes.
498114942Sume	case ${ipv6_defaultrouter} in
499114942Sume	[Nn][Oo] | '')
500114942Sume		;;
501114942Sume	*)
502114942Sume		ipv6_static_routes="default ${ipv6_static_routes}"
503114942Sume		ipv6_route_default="default ${ipv6_defaultrouter}"
504114942Sume		;;
505114942Sume	esac
506114942Sume	case ${ipv6_static_routes} in
507114942Sume	[Nn][Oo] | '')
508114942Sume		;;
509114942Sume	*)
510114942Sume		for i in ${ipv6_static_routes}; do
511114942Sume			eval ipv6_route_args=\$ipv6_route_${i}
512114942Sume			route add -inet6 ${ipv6_route_args}
513114942Sume		done
514114942Sume		;;
515114942Sume	esac
516114942Sume}
517114942Sume
518114942Sume# Setup faith
519114942Sumenetwork6_faith_setup()
520114942Sume{
521114942Sume	case ${ipv6_faith_prefix} in
522114942Sume	[Nn][Oo] | '')
523114942Sume		;;
524114942Sume	*)
525114942Sume		sysctl net.inet6.ip6.keepfaith=1
526114942Sume		ifconfig faith0 create >/dev/null 2>&1
527114942Sume		ifconfig faith0 up
528114942Sume		for prefix in ${ipv6_faith_prefix}; do
529114942Sume			prefixlen=`expr "${prefix}" : ".*/\(.*\)"`
530114942Sume			case ${prefixlen} in
531114942Sume			'')
532114942Sume				prefixlen=96
533114942Sume				;;
534114942Sume			*)
535114942Sume				prefix=`expr "${prefix}" : \
536114942Sume					     "\(.*\)/${prefixlen}"`
537114942Sume				;;
538114942Sume			esac
539114942Sume			route add -inet6 ${prefix} -prefixlen ${prefixlen} ::1
540114942Sume			route change -inet6 ${prefix} -prefixlen ${prefixlen} \
541114942Sume				-ifp faith0
542114942Sume		done
543114942Sume		;;
544114942Sume	esac
545114942Sume}
546114942Sume
547114942Sume# Install the "default interface" to kernel, which will be used
548114942Sume# as the default route when there's no router.
549114942Sumenetwork6_default_interface_setup()
550114942Sume{
551114942Sume	# Choose IPv6 default interface if it is not clearly specified.
552114942Sume	case ${ipv6_default_interface} in
553114942Sume	'')
554114942Sume		for i in ${ipv6_network_interfaces}; do
555114942Sume			case $i in
556114942Sume			lo0|faith[0-9]*)
557114942Sume				continue
558114942Sume				;;
559114942Sume			esac
560114942Sume			laddr=`network6_getladdr $i exclude_tentative`
561114942Sume			case ${laddr} in
562114942Sume			'')
563114942Sume				;;
564114942Sume			*)
565114942Sume				ipv6_default_interface=$i
566114942Sume				break
567114942Sume				;;
568114942Sume			esac
569114942Sume		done
570114942Sume		;;
571114942Sume	esac
572114942Sume
573114942Sume	# Disallow unicast packets without outgoing scope identifiers,
574114942Sume	# or route such packets to a "default" interface, if it is specified.
575114942Sume	route add -inet6 fe80:: -prefixlen 10 ::1 -reject
576114942Sume	case ${ipv6_default_interface} in
577114942Sume	[Nn][Oo] | '')
578114942Sume		route add -inet6 ff02:: -prefixlen 16 ::1 -reject
579114942Sume		;;
580114942Sume	*)
581114942Sume		laddr=`network6_getladdr ${ipv6_default_interface}`
582114942Sume		route add -inet6 ff02:: ${laddr} -prefixlen 16 -interface \
583114942Sume			-cloning
584114942Sume
585114942Sume		# Disable installing the default interface with the
586114942Sume		# case net.inet6.ip6.forwarding=0 and
587114942Sume		# net.inet6.ip6.accept_rtadv=0, due to avoid conflict
588114942Sume		# between the default router list and the manual
589114942Sume		# configured default route.
590114942Sume		case ${ipv6_gateway_enable} in
591114942Sume		[Yy][Ee][Ss])
592114942Sume			;;
593114942Sume		*)
594114942Sume			if [ `sysctl -n net.inet6.ip6.accept_rtadv` -eq 1 ]
595114942Sume			then
596114942Sume				ndp -I ${ipv6_default_interface}
597114942Sume			fi
598114942Sume			;;
599114942Sume		esac
600114942Sume		;;
601114942Sume	esac
602114942Sume}
603114942Sume
604114942Sumenetwork6_getladdr()
605114942Sume{
606114942Sume	ifconfig $1 2>/dev/null | while read proto addr rest; do
607114942Sume		case ${proto} in
608114942Sume		inet6)
609114942Sume			case ${addr} in
610114942Sume			fe80::*)
611114942Sume				if [ -z "$2" ]; then
612114942Sume					echo ${addr}
613114942Sume					return
614114942Sume				fi
615114942Sume				case ${rest} in
616114942Sume				*tentative*)
617114942Sume					continue
618114942Sume					;;
619114942Sume				*)
620114942Sume					echo ${addr}
621114942Sume					return
622114942Sume				esac
623114942Sume			esac
624114942Sume		esac
625114942Sume	done
626114942Sume}
627