network.subr revision 130151
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 130151 2004-06-06 11:46:29Z schweikh $
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.
261113674Smtm#	Note that the list always includes cloned interfaces.
262113674Smtm#
263113674Smtmlist_net_interfaces()
264113674Smtm{
265113674Smtm	type=$1
26665532Snectar
267113674Smtm	# Get a list of ALL the interfaces
26851231Ssheldonh	#
26951231Ssheldonh	case ${network_interfaces} in
27051231Ssheldonh	[Aa][Uu][Tt][Oo])
271113674Smtm		_tmplist="`ifconfig -l`"
27251231Ssheldonh		;;
27383677Sbrooks	*)
274113674Smtm		_tmplist="${network_interfaces}"
27583677Sbrooks		;;
27651231Ssheldonh	esac
277113674Smtm	_tmplist="${_tmplist} ${cloned_interfaces}"
27849122Sbrian
279113674Smtm	if [ -z "$type" ]; then
280113674Smtm		echo $_tmplist
281113674Smtm		return 0
282113674Smtm	fi
28349122Sbrian
284113674Smtm	# Separate out dhcp and non-dhcp intefraces
285113674Smtm	#
286113674Smtm	_aprefix=
287113674Smtm	_brefix=
288113674Smtm	for _if in ${_tmplist} ; do
289113674Smtm		eval _ifarg="\$ifconfig_${_if}"
290113674Smtm		case "$_ifarg" in
29151231Ssheldonh		[Dd][Hh][Cc][Pp])
292113674Smtm			_dhcplist="${_dhcplist}${_aprefix}${_if}"
293113674Smtm			[ -z "$_aprefix" ] && _aprefix=' '
29451231Ssheldonh			;;
295113674Smtm		''|*)
296113674Smtm			_nodhcplist="${_nodhcplist}${_bprefix}${_if}"
297113674Smtm			[ -z "$_bprefix" ] && _bprefix=' '
29851231Ssheldonh			;;
29951231Ssheldonh		esac
30054458Sobrien	done
30151231Ssheldonh
302118797Smbr	case ${pccard_ifconfig} in
303118797Smbr	[Dd][Hh][Cc][Pp])
304118797Smbr		for _if in ${removable_interfaces} ; do
305118797Smbr			_test_if=`ifconfig ${_if} 2>&1`
306118797Smbr			case "$_test_if" in
307118797Smbr			"ifconfig: interface $_if does not exist")
308118797Smbr				;;
309118797Smbr			*)
310118797Smbr				_dhcplist="${_dhcplist}${_aprefix}${_if}"
311118797Smbr				[ -z "$_aprefix" ] && _aprefix=' '
312118797Smbr				;;
313118797Smbr			esac
314118797Smbr		done
315118797Smbr		;;
316118797Smbr	*)
317118797Smbr		;;
318118797Smbr	esac
319118797Smbr
320113674Smtm	case "$type" in
321113674Smtm	nodhcp)
322113674Smtm		echo $_nodhcplist
323113674Smtm		;;
324113674Smtm	dhcp)
325113674Smtm		echo $_dhcplist
326113674Smtm		;;
327113674Smtm	esac
328130151Sschweikh	return 0
32925184Sjkh}
330114942Sume
331114942Sumehexdigit()
332114942Sume{
333114942Sume	if [ $1 -lt 10 ]; then
334114942Sume		echo $1
335114942Sume	else
336114942Sume		case $1 in
337114942Sume		10)	echo a ;;
338114942Sume		11)	echo b ;;
339114942Sume		12)	echo c ;;
340114942Sume		13)	echo d ;;
341114942Sume		14)	echo e ;;
342114942Sume		15)	echo f ;;
343114942Sume		esac
344114942Sume	fi
345114942Sume}
346114942Sume
347114942Sumehexprint()
348114942Sume{
349114942Sume	val=$1
350114942Sume	str=''
351114942Sume
352114942Sume	dig=`hexdigit $((${val} & 15))`
353114942Sume	str=${dig}${str}
354114942Sume	val=$((${val} >> 4))
355114942Sume	while [ ${val} -gt 0 ]; do
356114942Sume		dig=`hexdigit $((${val} & 15))`
357114942Sume		str=${dig}${str}
358114942Sume		val=$((${val} >> 4))
359114942Sume	done
360114942Sume
361114942Sume	echo ${str}
362114942Sume}
363114942Sume
364114942Sume# Setup the interfaces for IPv6
365114942Sumenetwork6_interface_setup()
366114942Sume{
367114942Sume	interfaces=$*
368114942Sume	rtsol_interfaces=''
369114942Sume	case ${ipv6_gateway_enable} in
370114942Sume	[Yy][Ee][Ss])
371114942Sume		rtsol_available=no
372114942Sume		;;
373114942Sume	*)
374114942Sume		rtsol_available=yes
375114942Sume		;;
376114942Sume	esac
377114942Sume	for i in $interfaces; do
378114942Sume		rtsol_interface=yes
379114942Sume		eval prefix=\$ipv6_prefix_$i
380114942Sume		if [ -n "${prefix}" ]; then
381114942Sume			rtsol_available=no
382114942Sume			rtsol_interface=no
383114942Sume			laddr=`network6_getladdr $i`
384114942Sume			hostid=`expr "${laddr}" : 'fe80::\(.*\)%\(.*\)'`
385114942Sume			for j in ${prefix}; do
386114942Sume				address=$j\:${hostid}
387114942Sume				ifconfig $i inet6 ${address} prefixlen 64 alias
388114942Sume
389114942Sume				case ${ipv6_gateway_enable} in
390114942Sume				[Yy][Ee][Ss])
391114942Sume					# subnet-router anycast address
392114942Sume					# (rfc2373)
393114942Sume					ifconfig $i inet6 $j:: prefixlen 64 \
394114942Sume						alias anycast
395114942Sume					;;
396114942Sume				esac
397114942Sume			done
398114942Sume		fi
399114942Sume		eval ipv6_ifconfig=\$ipv6_ifconfig_$i
400114942Sume		if [ -n "${ipv6_ifconfig}" ]; then
401114942Sume			rtsol_available=no
402114942Sume			rtsol_interface=no
403114942Sume			ifconfig $i inet6 ${ipv6_ifconfig} alias
404114942Sume		fi
405114942Sume
406114942Sume		if [ ${rtsol_available} = yes -a ${rtsol_interface} = yes ]
407114942Sume		then
408114942Sume			case ${i} in
409114942Sume			lo0|gif[0-9]*|stf[0-9]*|faith[0-9]*|lp[0-9]*|sl[0-9]*|tun[0-9]*)
410114942Sume				;;
411114942Sume			*)
412114942Sume				rtsol_interfaces="${rtsol_interfaces} ${i}"
413114942Sume				;;
414114942Sume			esac
415114942Sume		else
416114942Sume			ifconfig $i inet6
417114942Sume		fi
418114942Sume	done
419114942Sume
420114942Sume	if [ ${rtsol_available} = yes -a -n "${rtsol_interfaces}" ]; then
421114942Sume		# Act as endhost - automatically configured.
422114942Sume		# You can configure only single interface, as
423114942Sume		# specification assumes that autoconfigured host has
424114942Sume		# single interface only.
425114942Sume		sysctl net.inet6.ip6.accept_rtadv=1
426114942Sume		set ${rtsol_interfaces}
427114942Sume		ifconfig $1 up
428118666Sume		rtsol ${rtsol_flags} $1
429114942Sume	fi
430114942Sume
431114942Sume	for i in $interfaces; do
432114942Sume		alias=0
433114942Sume		while : ; do
434114942Sume			eval ipv6_ifconfig=\$ipv6_ifconfig_${i}_alias${alias}
435114942Sume			if [ -z "${ipv6_ifconfig}" ]; then
436114942Sume				break;
437114942Sume			fi
438114942Sume			ifconfig $i inet6 ${ipv6_ifconfig} alias
439114942Sume			alias=$((${alias} + 1))
440114942Sume		done
441114942Sume	done
442114942Sume}
443114942Sume
444114942Sume# Setup IPv6 to IPv4 mapping
445114942Sumenetwork6_stf_setup()
446114942Sume{
447114942Sume	case ${stf_interface_ipv4addr} in
448114942Sume	[Nn][Oo] | '')
449114942Sume		;;
450114942Sume	*)
451114942Sume		# assign IPv6 addr and interface route for 6to4 interface
452114942Sume		stf_prefixlen=$((16+${stf_interface_ipv4plen:-0}))
453114942Sume		OIFS="$IFS"
454114942Sume		IFS=".$IFS"
455114942Sume		set ${stf_interface_ipv4addr}
456114942Sume		IFS="$OIFS"
457114942Sume		hexfrag1=`hexprint $(($1*256 + $2))`
458114942Sume		hexfrag2=`hexprint $(($3*256 + $4))`
459114942Sume		ipv4_in_hexformat="${hexfrag1}:${hexfrag2}"
460114942Sume		case ${stf_interface_ipv6_ifid} in
461114942Sume		[Aa][Uu][Tt][Oo] | '')
462114942Sume			for i in ${ipv6_network_interfaces}; do
463114942Sume				laddr=`network6_getladdr ${i}`
464114942Sume				case ${laddr} in
465114942Sume				'')
466114942Sume					;;
467114942Sume				*)
468114942Sume					break
469114942Sume					;;
470114942Sume				esac
471114942Sume			done
472114942Sume			stf_interface_ipv6_ifid=`expr "${laddr}" : \
473114942Sume						      'fe80::\(.*\)%\(.*\)'`
474114942Sume			case ${stf_interface_ipv6_ifid} in
475114942Sume			'')
476114942Sume				stf_interface_ipv6_ifid=0:0:0:1
477114942Sume				;;
478114942Sume			esac
479114942Sume			;;
480114942Sume		esac
481114942Sume		ifconfig stf0 create >/dev/null 2>&1
482114942Sume		ifconfig stf0 inet6 2002:${ipv4_in_hexformat}:${stf_interface_ipv6_slaid:-0}:${stf_interface_ipv6_ifid} \
483114942Sume			prefixlen ${stf_prefixlen}
484114942Sume		# disallow packets to malicious 6to4 prefix
485114942Sume		route add -inet6 2002:e000:: -prefixlen 20 ::1 -reject
486114942Sume		route add -inet6 2002:7f00:: -prefixlen 24 ::1 -reject
487114942Sume		route add -inet6 2002:0000:: -prefixlen 24 ::1 -reject
488114942Sume		route add -inet6 2002:ff00:: -prefixlen 24 ::1 -reject
489114942Sume		;;
490114942Sume	esac
491114942Sume}
492114942Sume
493114942Sume# Setup static routes
494114942Sumenetwork6_static_routes_setup()
495114942Sume{
496114942Sume	# Set up any static routes.
497114942Sume	case ${ipv6_defaultrouter} in
498114942Sume	[Nn][Oo] | '')
499114942Sume		;;
500114942Sume	*)
501114942Sume		ipv6_static_routes="default ${ipv6_static_routes}"
502114942Sume		ipv6_route_default="default ${ipv6_defaultrouter}"
503114942Sume		;;
504114942Sume	esac
505114942Sume	case ${ipv6_static_routes} in
506114942Sume	[Nn][Oo] | '')
507114942Sume		;;
508114942Sume	*)
509114942Sume		for i in ${ipv6_static_routes}; do
510114942Sume			eval ipv6_route_args=\$ipv6_route_${i}
511114942Sume			route add -inet6 ${ipv6_route_args}
512114942Sume		done
513114942Sume		;;
514114942Sume	esac
515114942Sume}
516114942Sume
517114942Sume# Setup faith
518114942Sumenetwork6_faith_setup()
519114942Sume{
520114942Sume	case ${ipv6_faith_prefix} in
521114942Sume	[Nn][Oo] | '')
522114942Sume		;;
523114942Sume	*)
524114942Sume		sysctl net.inet6.ip6.keepfaith=1
525114942Sume		ifconfig faith0 create >/dev/null 2>&1
526114942Sume		ifconfig faith0 up
527114942Sume		for prefix in ${ipv6_faith_prefix}; do
528114942Sume			prefixlen=`expr "${prefix}" : ".*/\(.*\)"`
529114942Sume			case ${prefixlen} in
530114942Sume			'')
531114942Sume				prefixlen=96
532114942Sume				;;
533114942Sume			*)
534114942Sume				prefix=`expr "${prefix}" : \
535114942Sume					     "\(.*\)/${prefixlen}"`
536114942Sume				;;
537114942Sume			esac
538114942Sume			route add -inet6 ${prefix} -prefixlen ${prefixlen} ::1
539114942Sume			route change -inet6 ${prefix} -prefixlen ${prefixlen} \
540114942Sume				-ifp faith0
541114942Sume		done
542114942Sume		;;
543114942Sume	esac
544114942Sume}
545114942Sume
546114942Sume# Install the "default interface" to kernel, which will be used
547114942Sume# as the default route when there's no router.
548114942Sumenetwork6_default_interface_setup()
549114942Sume{
550114942Sume	# Choose IPv6 default interface if it is not clearly specified.
551114942Sume	case ${ipv6_default_interface} in
552114942Sume	'')
553114942Sume		for i in ${ipv6_network_interfaces}; do
554114942Sume			case $i in
555114942Sume			lo0|faith[0-9]*)
556114942Sume				continue
557114942Sume				;;
558114942Sume			esac
559114942Sume			laddr=`network6_getladdr $i exclude_tentative`
560114942Sume			case ${laddr} in
561114942Sume			'')
562114942Sume				;;
563114942Sume			*)
564114942Sume				ipv6_default_interface=$i
565114942Sume				break
566114942Sume				;;
567114942Sume			esac
568114942Sume		done
569114942Sume		;;
570114942Sume	esac
571114942Sume
572114942Sume	# Disallow unicast packets without outgoing scope identifiers,
573114942Sume	# or route such packets to a "default" interface, if it is specified.
574114942Sume	route add -inet6 fe80:: -prefixlen 10 ::1 -reject
575114942Sume	case ${ipv6_default_interface} in
576114942Sume	[Nn][Oo] | '')
577114942Sume		route add -inet6 ff02:: -prefixlen 16 ::1 -reject
578114942Sume		;;
579114942Sume	*)
580114942Sume		laddr=`network6_getladdr ${ipv6_default_interface}`
581114942Sume		route add -inet6 ff02:: ${laddr} -prefixlen 16 -interface \
582114942Sume			-cloning
583114942Sume
584114942Sume		# Disable installing the default interface with the
585114942Sume		# case net.inet6.ip6.forwarding=0 and
586114942Sume		# net.inet6.ip6.accept_rtadv=0, due to avoid conflict
587114942Sume		# between the default router list and the manual
588114942Sume		# configured default route.
589114942Sume		case ${ipv6_gateway_enable} in
590114942Sume		[Yy][Ee][Ss])
591114942Sume			;;
592114942Sume		*)
593114942Sume			if [ `sysctl -n net.inet6.ip6.accept_rtadv` -eq 1 ]
594114942Sume			then
595114942Sume				ndp -I ${ipv6_default_interface}
596114942Sume			fi
597114942Sume			;;
598114942Sume		esac
599114942Sume		;;
600114942Sume	esac
601114942Sume}
602114942Sume
603114942Sumenetwork6_getladdr()
604114942Sume{
605114942Sume	ifconfig $1 2>/dev/null | while read proto addr rest; do
606114942Sume		case ${proto} in
607114942Sume		inet6)
608114942Sume			case ${addr} in
609114942Sume			fe80::*)
610114942Sume				if [ -z "$2" ]; then
611114942Sume					echo ${addr}
612114942Sume					return
613114942Sume				fi
614114942Sume				case ${rest} in
615114942Sume				*tentative*)
616114942Sume					continue
617114942Sume					;;
618114942Sume				*)
619114942Sume					echo ${addr}
620114942Sume					return
621114942Sume				esac
622114942Sume			esac
623114942Sume		esac
624114942Sume	done
625114942Sume}
626