xref: /freebsd/libexec/rc/network.subr (revision 184c1b943937986c81e1996d999d21626ec7a4ff)
1#
2# Copyright (c) 2003 The FreeBSD Project. All rights reserved.
3#
4# Redistribution and use in source and binary forms, with or without
5# modification, are permitted provided that the following conditions
6# are met:
7# 1. Redistributions of source code must retain the above copyright
8#    notice, this list of conditions and the following disclaimer.
9# 2. Redistributions in binary form must reproduce the above copyright
10#    notice, this list of conditions and the following disclaimer in the
11#    documentation and/or other materials provided with the distribution.
12#
13# THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``AS IS'' AND
14# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16# ARE DISCLAIMED.  IN NO EVENT SHALL THE PROJECT OR CONTRIBUTORS BE LIABLE
17# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
23# SUCH DAMAGE.
24#
25# $FreeBSD$
26#
27IFCONFIG_CMD="/sbin/ifconfig"
28: ${netif_ipexpand_max:=2048}
29
30#
31# Subroutines commonly used from network startup scripts.
32# Requires that rc.conf be loaded first.
33#
34
35# ifn_start ifn
36#	Bring up and configure an interface.  If some configuration is
37#	applied, print the interface configuration.
38#
39ifn_start()
40{
41	local ifn cfg
42	ifn="$1"
43	cfg=1
44
45	[ -z "$ifn" ] && err 1 "ifn_start called without an interface"
46
47	ifscript_up ${ifn} && cfg=0
48	ifconfig_up ${ifn} && cfg=0
49	if ! noafif $ifn; then
50		afexists inet && ipv4_up ${ifn} && cfg=0
51		afexists inet6 && ipv6_up ${ifn} && cfg=0
52	fi
53	childif_create ${ifn} && cfg=0
54
55	return $cfg
56}
57
58# ifn_stop ifn
59#	Shutdown and de-configure an interface.  If action is taken,
60#	print the interface name.
61#
62ifn_stop()
63{
64	local ifn cfg
65	ifn="$1"
66	cfg=1
67
68	[ -z "$ifn" ] && err 1 "ifn_stop called without an interface"
69
70	if ! noafif $ifn; then
71		afexists inet6 && ipv6_down ${ifn} && cfg=0
72		afexists inet && ipv4_down ${ifn} && cfg=0
73	fi
74	ifconfig_down ${ifn} && cfg=0
75	ifscript_down ${ifn} && cfg=0
76	childif_destroy ${ifn} && cfg=0
77
78	return $cfg
79}
80
81# ifn_vnetup ifn
82#	Move ifn to the specified vnet jail.
83#
84ifn_vnetup()
85{
86
87	ifn_vnet0 $1 vnet
88}
89
90# ifn_vnetdown ifn
91#	Reclaim ifn from the specified vnet jail.
92#
93ifn_vnetdown()
94{
95
96	ifn_vnet0 $1 -vnet
97}
98
99# ifn_vnet0 ifn action
100#	Helper function for ifn_vnetup and ifn_vnetdown.
101#
102ifn_vnet0()
103{
104	local _ifn _cfg _action _vnet
105	_ifn="$1"
106	_action="$2"
107	_cfg=1
108
109	if _vnet=$(vnetif $_ifn); then
110		${IFCONFIG_CMD} $_ifn $_action $_vnet && _cfg=0
111	fi
112
113	return $_cfg
114}
115
116# ifconfig_up if
117#	Evaluate ifconfig(8) arguments for interface $if and
118#	run ifconfig(8) with those arguments. It returns 0 if
119#	arguments were found and executed or 1 if the interface
120#	had no arguments.  Pseudo arguments DHCP and WPA are handled
121#	here.
122#
123ifconfig_up()
124{
125	local _cfg _ifconfig_descr _ipv6_opts ifconfig_args
126	_cfg=1
127
128	# Make sure lo0 always comes up.
129	if [ "$1" = "lo0" ]; then
130		_cfg=0
131	fi
132
133	# inet6 specific
134	if ! noafif $1 && afexists inet6; then
135		if checkyesno ipv6_activate_all_interfaces; then
136			_ipv6_opts="-ifdisabled"
137		fi
138
139		# backward compatibility: $ipv6_enable
140		case $ipv6_enable in
141		[Yy][Ee][Ss]|[Tt][Rr][Uu][Ee]|[Oo][Nn]|1)
142			case $1 in
143			bridge[0-9]*)
144				# No accept_rtadv by default on if_bridge(4)
145				# to avoid a conflict with the member
146				# interfaces.
147			;;
148			*)
149				if ! checkyesno ipv6_gateway_enable; then
150					_ipv6_opts="${_ipv6_opts} accept_rtadv"
151				fi
152			;;
153			esac
154		;;
155		esac
156
157		case $ipv6_cpe_wanif in
158		$1)
159			_ipv6_opts="${_ipv6_opts} -no_radr accept_rtadv"
160		;;
161		esac
162
163		if [ -n "${_ipv6_opts}" ]; then
164			${IFCONFIG_CMD} $1 inet6 ${_ipv6_opts}
165		fi
166	fi
167
168	# ifconfig_IF
169	ifconfig_args=`ifconfig_getargs $1`
170	if [ -n "${ifconfig_args}" ]; then
171		eval ${IFCONFIG_CMD} $1 ${ifconfig_args}
172		_cfg=0
173	fi
174
175	# inet6 specific
176	if ! noafif $1 && afexists inet6; then
177		# ifconfig_IF_ipv6
178		ifconfig_args=`ifconfig_getargs $1 ipv6`
179		if [ -n "${ifconfig_args}" ]; then
180			# backward compatibility: inet6 keyword
181			case "${ifconfig_args}" in
182			:*|[0-9a-fA-F]*:*)
183				warn "\$ifconfig_$1_ipv6 needs leading" \
184				    "\"inet6\" keyword for an IPv6 address."
185				ifconfig_args="inet6 ${ifconfig_args}"
186			;;
187			esac
188			${IFCONFIG_CMD} $1 inet6 -ifdisabled
189			eval ${IFCONFIG_CMD} $1 ${ifconfig_args}
190			_cfg=0
191		fi
192
193		# $ipv6_prefix_IF will be handled in
194		# ipv6_prefix_hostid_addr_common().
195		ifconfig_args=`get_if_var $1 ipv6_prefix_IF`
196		if [ -n "${ifconfig_args}" ]; then
197			${IFCONFIG_CMD} $1 inet6 -ifdisabled
198			_cfg=0
199		fi
200
201		# backward compatibility: $ipv6_ifconfig_IF
202		ifconfig_args=`get_if_var $1 ipv6_ifconfig_IF`
203		if [ -n "${ifconfig_args}" ]; then
204			warn "\$ipv6_ifconfig_$1 is obsolete." \
205			    "  Use ifconfig_$1_ipv6 instead."
206			${IFCONFIG_CMD} $1 inet6 -ifdisabled
207			eval ${IFCONFIG_CMD} $1 inet6 ${ifconfig_args}
208			_cfg=0
209		fi
210	fi
211
212	ifalias $1 link alias
213	ifalias $1 ether alias
214
215	_ifconfig_descr=`get_if_var $1 ifconfig_IF_descr`
216	if [ -n "${_ifconfig_descr}" ]; then
217		${IFCONFIG_CMD} $1 description "${_ifconfig_descr}"
218	fi
219
220	if wpaif $1; then
221		/etc/rc.d/wpa_supplicant start $1
222		_cfg=0		# XXX: not sure this should count
223	elif hostapif $1; then
224		/etc/rc.d/hostapd start $1
225		_cfg=0
226	elif [ ${_cfg} -eq 0 ]; then
227		${IFCONFIG_CMD} $1 up
228	fi
229
230	if ! noafif $1 && afexists inet6; then
231		ipv6_accept_rtadv_up $1 && _cfg=0
232	fi
233
234	if dhcpif $1; then
235		if [ $_cfg -ne 0 ] ; then
236			${IFCONFIG_CMD} $1 up
237		fi
238		if syncdhcpif $1; then
239			/etc/rc.d/dhclient start $1
240		fi
241		_cfg=0
242	fi
243
244	return $_cfg
245}
246
247# ifconfig_down if
248#	returns 1 if wpa_supplicant or dhclient was stopped or
249#	the interface exists.
250#
251ifconfig_down()
252{
253	local _cfg
254	_cfg=1
255
256	if wpaif $1; then
257		/etc/rc.d/wpa_supplicant stop $1
258		_cfg=0
259	elif hostapif $1; then
260		/etc/rc.d/hostapd stop $1
261		_cfg=0
262	elif dhcpif $1; then
263		/etc/rc.d/dhclient stop $1
264		_cfg=0
265	fi
266
267	if ifexists $1; then
268		${IFCONFIG_CMD} $1 down
269		_cfg=0
270	fi
271
272	return $_cfg
273}
274
275# get_if_var if var [default]
276#	Return the value of the pseudo-hash corresponding to $if where
277#	$var is a string containg the sub-string "IF" which will be
278#	replaced with $if after the characters defined in _punct are
279#	replaced with '_'. If the variable is unset, replace it with
280#	$default if given.
281get_if_var()
282{
283	local _if _punct _punct_c _var _default prefix suffix
284
285	if [ $# -ne 2 -a $# -ne 3 ]; then
286		err 3 'USAGE: get_if_var name var [default]'
287	fi
288
289	_if=$1
290	_punct=".-/+"
291	ltr ${_if} "${_punct}" '_' _if
292	_var=$2
293	_default=$3
294
295	prefix=${_var%%IF*}
296	suffix=${_var##*IF}
297	eval echo \${${prefix}${_if}${suffix}-${_default}}
298}
299
300# _ifconfig_getargs if [af]
301#	Prints the arguments for the supplied interface to stdout.
302#	Returns 1 if empty.  In general, ifconfig_getargs should be used
303#	outside this file.
304_ifconfig_getargs()
305{
306	local _ifn _af
307	_ifn=$1
308	_af=${2+_$2}
309
310	if [ -z "$_ifn" ]; then
311		return 1
312	fi
313
314	get_if_var $_ifn ifconfig_IF$_af "$ifconfig_DEFAULT"
315}
316
317# ifconfig_getargs if [af]
318#	Takes the result from _ifconfig_getargs and removes pseudo
319#	args such as DHCP and WPA.
320ifconfig_getargs()
321{
322	local _tmpargs _arg _args _vnet
323	_tmpargs=`_ifconfig_getargs $1 $2`
324	if [ $? -eq 1 ]; then
325		return 1
326	fi
327	_args=
328	_vnet=0
329
330	for _arg in $_tmpargs; do
331		case $_arg:$_vnet in
332		[Dd][Hh][Cc][Pp]:0) ;;
333		[Nn][Oo][Aa][Uu][Tt][Oo]:0) ;;
334		[Nn][Oo][Ss][Yy][Nn][Cc][Dd][Hh][Cc][Pp]:0) ;;
335		[Ss][Yy][Nn][Cc][Dd][Hh][Cc][Pp]:0) ;;
336		[Ww][Pp][Aa]:0) ;;
337		[Hh][Oo][Ss][Tt][Aa][Pp]:0) ;;
338		vnet:0)	_vnet=1 ;;
339		*:1)	_vnet=0 ;;
340		*:0)
341			_args="$_args $_arg"
342		;;
343		esac
344	done
345
346	echo $_args
347}
348
349# autoif
350#	Returns 0 if the interface should be automatically configured at
351#	boot time and 1 otherwise.
352autoif()
353{
354	local _tmpargs _arg
355	_tmpargs=`_ifconfig_getargs $1`
356
357	for _arg in $_tmpargs; do
358		case $_arg in
359		[Nn][Oo][Aa][Uu][Tt][Oo])
360			return 1
361			;;
362		esac
363	done
364
365	return 0
366}
367
368# dhcpif if
369#	Returns 0 if the interface is a DHCP interface and 1 otherwise.
370dhcpif()
371{
372	local _tmpargs _arg
373	_tmpargs=`_ifconfig_getargs $1`
374
375	case $1 in
376	lo[0-9]*|\
377	stf[0-9]*|\
378	lp[0-9]*|\
379	sl[0-9]*)
380		return 1
381		;;
382	esac
383	if noafif $1; then
384		return 1
385	fi
386
387	for _arg in $_tmpargs; do
388		case $_arg in
389		[Dd][Hh][Cc][Pp])
390			return 0
391			;;
392		[Nn][Oo][Ss][Yy][Nn][Cc][Dd][Hh][Cc][Pp])
393			return 0
394			;;
395		[Ss][Yy][Nn][Cc][Dd][Hh][Cc][Pp])
396			return 0
397			;;
398		esac
399	done
400
401	return 1
402}
403
404# syncdhcpif
405#	Returns 0 if the interface should be configured synchronously and
406#	1 otherwise.
407syncdhcpif()
408{
409	local _tmpargs _arg
410	_tmpargs=`_ifconfig_getargs $1`
411
412	if noafif $1; then
413		return 1
414	fi
415
416	for _arg in $_tmpargs; do
417		case $_arg in
418		[Nn][Oo][Ss][Yy][Nn][Cc][Dd][Hh][Cc][Pp])
419			return 1
420			;;
421		[Ss][Yy][Nn][Cc][Dd][Hh][Cc][Pp])
422			return 0
423			;;
424		esac
425	done
426
427	checkyesno synchronous_dhclient
428}
429
430# wpaif if
431#	Returns 0 if the interface is a WPA interface and 1 otherwise.
432wpaif()
433{
434	local _tmpargs _arg
435	_tmpargs=`_ifconfig_getargs $1`
436
437	for _arg in $_tmpargs; do
438		case $_arg in
439		[Ww][Pp][Aa])
440			return 0
441			;;
442		esac
443	done
444
445	return 1
446}
447
448# hostapif if
449#	Returns 0 if the interface is a HOSTAP interface and 1 otherwise.
450hostapif()
451{
452	local _tmpargs _arg
453	_tmpargs=`_ifconfig_getargs $1`
454
455	for _arg in $_tmpargs; do
456		case $_arg in
457		[Hh][Oo][Ss][Tt][Aa][Pp])
458			return 0
459			;;
460		esac
461	done
462
463	return 1
464}
465
466# vnetif if
467#	Returns 0 and echo jail if "vnet" keyword is specified on the
468#	interface, and 1 otherwise.
469vnetif()
470{
471	local _tmpargs _arg _vnet
472	_tmpargs=`_ifconfig_getargs $1`
473
474	_vnet=0
475	for _arg in $_tmpargs; do
476		case $_arg:$_vnet in
477		vnet:0)	_vnet=1 ;;
478		*:1)	echo $_arg; return 0 ;;
479		esac
480	done
481
482	return 1
483}
484
485# afexists af
486#	Returns 0 if the address family is enabled in the kernel
487#	1 otherwise.
488afexists()
489{
490	local _af
491	_af=$1
492
493	case ${_af} in
494	inet|inet6)
495		check_kern_features ${_af}
496		;;
497	link|ether)
498		return 0
499		;;
500	*)
501		err 1 "afexists(): Unsupported address family: $_af"
502		;;
503	esac
504}
505
506# noafif if
507#	Returns 0 if the interface has no af configuration and 1 otherwise.
508noafif()
509{
510	local _if
511	_if=$1
512
513	case $_if in
514	pflog[0-9]*|\
515	pfsync[0-9]*|\
516	usbus[0-9]*|\
517	an[0-9]*|\
518	ath[0-9]*|\
519	ipw[0-9]*|\
520	ipfw[0-9]*|\
521	iwi[0-9]*|\
522	iwn[0-9]*|\
523	ral[0-9]*|\
524	wi[0-9]*|\
525	wl[0-9]*|\
526	wpi[0-9]*)
527		return 0
528		;;
529	esac
530
531	return 1
532}
533
534# ipv6if if
535#	Returns 0 if the interface should be configured for IPv6 and
536#	1 otherwise.
537ipv6if()
538{
539	local _if _tmpargs i
540	_if=$1
541
542	if ! afexists inet6; then
543		return 1
544	fi
545
546	# lo0 is always IPv6-enabled
547	case $_if in
548	lo0)
549		return 0
550		;;
551	esac
552
553	case "${ipv6_network_interfaces}" in
554	$_if|"$_if "*|*" $_if"|*" $_if "*|[Aa][Uu][Tt][Oo])
555		# True if $ifconfig_IF_ipv6 is defined.
556		_tmpargs=`_ifconfig_getargs $_if ipv6`
557		if [ -n "${_tmpargs}" ]; then
558			return 0
559		fi
560
561		# True if $ipv6_prefix_IF is defined.
562		_tmpargs=`get_if_var $_if ipv6_prefix_IF`
563		if [ -n "${_tmpargs}" ]; then
564			return 0
565		fi
566
567		# backward compatibility: True if $ipv6_ifconfig_IF is defined.
568		_tmpargs=`get_if_var $_if ipv6_ifconfig_IF`
569		if [ -n "${_tmpargs}" ]; then
570			return 0
571		fi
572		;;
573	esac
574
575	return 1
576}
577
578# ipv6_autoconfif if
579#	Returns 0 if the interface should be configured for IPv6 with
580#	Stateless Address Configuration; 1 otherwise.
581ipv6_autoconfif()
582{
583	local _if _tmpargs _arg
584	_if=$1
585
586	case $_if in
587	lo[0-9]*|\
588	stf[0-9]*|\
589	lp[0-9]*|\
590	sl[0-9]*)
591		return 1
592		;;
593	esac
594	if noafif $_if; then
595		return 1
596	fi
597	if ! ipv6if $_if; then
598		return 1
599	fi
600	if checkyesno ipv6_gateway_enable; then
601		return 1
602	fi
603	_tmpargs=`get_if_var $_if ipv6_prefix_IF`
604	if [ -n "${_tmpargs}" ]; then
605		return 1
606	fi
607	# backward compatibility: $ipv6_enable
608	case $ipv6_enable in
609	[Yy][Ee][Ss]|[Tt][Rr][Uu][Ee]|[Oo][Nn]|1)
610		if checkyesno ipv6_gateway_enable; then
611			return 1
612		fi
613		case $1 in
614		bridge[0-9]*)
615			# No accept_rtadv by default on if_bridge(4)
616			# to avoid a conflict with the member
617			# interfaces.
618			return 1
619		;;
620		*)
621			return 0
622		;;
623		esac
624	;;
625	esac
626
627	_tmpargs=`_ifconfig_getargs $_if ipv6`
628	for _arg in $_tmpargs; do
629		case $_arg in
630		accept_rtadv)
631			return 0
632			;;
633		esac
634	done
635
636	# backward compatibility: $ipv6_ifconfig_IF
637	_tmpargs=`get_if_var $_if ipv6_ifconfig_IF`
638	for _arg in $_tmpargs; do
639		case $_arg in
640		accept_rtadv)
641			return 0
642			;;
643		esac
644	done
645
646	return 1
647}
648
649# ifexists if
650#	Returns 0 if the interface exists and 1 otherwise.
651ifexists()
652{
653	[ -z "$1" ] && return 1
654	${IFCONFIG_CMD} -n $1 > /dev/null 2>&1
655}
656
657# ipv4_up if
658#	add IPv4 addresses to the interface $if
659ipv4_up()
660{
661	local _if _ret
662	_if=$1
663	_ret=1
664
665	# Add 127.0.0.1/8 to lo0 unless otherwise specified.
666	if [ "${_if}" = "lo0" ]; then
667		ifconfig_args=`get_if_var ${_if} ifconfig_IF`
668		if [ -z "${ifconfig_args}" ]; then
669			${IFCONFIG_CMD} ${_if} inet 127.0.0.1/8 alias
670		fi
671	fi
672	ifalias ${_if} inet alias && _ret=0
673
674	return $_ret
675}
676
677# ipv6_up if
678#	add IPv6 addresses to the interface $if
679ipv6_up()
680{
681	local _if _ret
682	_if=$1
683	_ret=1
684
685	if ! ipv6if $_if; then
686		return 0
687	fi
688
689	ifalias ${_if} inet6 alias && _ret=0
690	ipv6_prefix_hostid_addr_common ${_if} alias && _ret=0
691
692	return $_ret
693}
694
695# ipv4_down if
696#	remove IPv4 addresses from the interface $if
697ipv4_down()
698{
699	local _if _ifs _ret inetList oldifs _inet
700	_if=$1
701	_ifs="^"
702	_ret=1
703
704	ifalias ${_if} inet -alias && _ret=0
705
706	inetList="`${IFCONFIG_CMD} ${_if} | grep 'inet ' | tr "\n\t" "$_ifs"`"
707
708	oldifs="$IFS"
709	IFS="$_ifs"
710	for _inet in $inetList ; do
711		# get rid of extraneous line
712		case $_inet in
713		inet\ *)	;;
714		*)		continue ;;
715		esac
716
717		_inet=`expr "$_inet" : '.*\(inet \([0-9]\{1,3\}\.\)\{3\}[0-9]\{1,3\}\).*'`
718
719		IFS="$oldifs"
720		${IFCONFIG_CMD} ${_if} ${_inet} delete
721		IFS="$_ifs"
722		_ret=0
723	done
724	IFS="$oldifs"
725
726	return $_ret
727}
728
729# ipv6_down if
730#	remove IPv6 addresses from the interface $if
731ipv6_down()
732{
733	local _if _ifs _ret inetList oldifs _inet6
734	_if=$1
735	_ifs="^"
736	_ret=1
737
738	if ! ipv6if $_if; then
739		return 0
740	fi
741
742	ipv6_accept_rtadv_down ${_if} && _ret=0
743	ipv6_prefix_hostid_addr_common ${_if} -alias && _ret=0
744	ifalias ${_if} inet6 -alias && _ret=0
745
746	inetList="`${IFCONFIG_CMD} ${_if} | grep 'inet6 ' | tr "\n\t" "$_ifs"`"
747
748	oldifs="$IFS"
749	IFS="$_ifs"
750	for _inet6 in $inetList ; do
751		# get rid of extraneous line
752		case $_inet6 in
753		inet6\ *)	;;
754		*)		continue ;;
755		esac
756
757		_inet6=`expr "$_inet6" : '.*\(inet6 \([0-9a-f:]*\)\).*'`
758
759		IFS="$oldifs"
760		${IFCONFIG_CMD} ${_if} ${_inet6} -alias
761		IFS="$_ifs"
762		_ret=0
763	done
764	IFS="$oldifs"
765
766	return $_ret
767}
768
769# ifalias if af action
770#	Configure or remove aliases for network interface $if.
771#	It returns 0 if at least one alias was configured or
772#	removed, or 1 if there were none.
773#
774ifalias()
775{
776	local _ret
777	_ret=1
778
779	afexists $2 || return $_ret
780
781	case "$2" in
782	inet|inet6|link|ether)
783		ifalias_af_common $1 $2 $3 && _ret=0
784		;;
785	esac
786
787	return $_ret
788}
789
790# ifalias_expand_addr af action addr
791#	Expand address range ("N-M") specification in addr.
792#	"addr" must not include an address-family keyword.
793#	The results will include an address-family keyword.
794#
795ifalias_expand_addr()
796{
797	local _af _action
798
799	_af=$1
800	_action=$2
801	shift 2
802
803	afexists $_af || return
804	ifalias_expand_addr_$_af $_action $*
805}
806
807# ifalias_expand_addr_inet action addr
808#	Helper function for ifalias_expand_addr().  Handles IPv4.
809#
810ifalias_expand_addr_inet()
811{
812	local _action _arg _cidr _cidr_addr _exargs
813	local _ipaddr _plen _range _iphead _iptail _iplow _iphigh _ipcount
814	local _retstr _c
815	_action=$1
816	_arg=$2
817	shift 2
818	_exargs=$*
819	_retstr=
820
821	case $_action:$_arg:$_exargs in
822	*:*--*)		return ;;	# invalid
823	tmp:*[0-9]-[0-9]*:*)		# to be expanded
824		_action="alias"
825	;;
826	*:*[0-9]-[0-9]*:*)		# to be expanded
827	;;
828	tmp:*:*netmask*)		# already expanded w/ netmask option
829		echo ${_arg%/[0-9]*} $_exargs && return
830	;;
831	tmp:*:*)			# already expanded w/o netmask option
832		echo $_arg $_exargs && return
833	;;
834	*:*:*netmask*)			# already expanded w/ netmask option
835		echo inet ${_arg%/[0-9]*} $_exargs && return
836	;;
837	*:*:*)				# already expanded w/o netmask option
838		echo inet $_arg $_exargs && return
839	;;
840	esac
841
842	for _cidr in $_arg; do
843		_ipaddr=${_cidr%%/*}
844		_plen=${_cidr##*/}
845		# When subnet prefix length is not specified, use /32.
846		case $_plen in
847		$_ipaddr)	_plen=32 ;;	# "/" character not found
848		esac
849
850		OIFS=$IFS
851		IFS=. set -- $_ipaddr
852		_range=
853		_iphead=
854		_iptail=
855		for _c in $@; do
856			case $_range:$_c in
857			:[0-9]*-[0-9]*)
858				_range=$_c
859			;;
860			:*)
861				_iphead="${_iphead}${_iphead:+.}${_c}"
862			;;
863			*:*)
864				_iptail="${_iptail}${_iptail:+.}${_c}"
865			;;
866			esac
867		done
868		IFS=$OIFS
869		_iplow=${_range%-*}
870		_iphigh=${_range#*-}
871
872		# clear netmask when removing aliases
873		if [ "$_action" = "-alias" ]; then
874			_plen=""
875		fi
876
877		_ipcount=$_iplow
878		while [ "$_ipcount" -le "$_iphigh" ]; do
879			_retstr="${_retstr} ${_iphead}${_iphead:+.}${_ipcount}${_iptail:+.}${_iptail}${_plen:+/}${_plen}"
880			if [ $_ipcount -gt $(($_iplow + $netif_ipexpand_max)) ]; then
881				warn "Range specification is too large (${_iphead}${_iphead:+.}${_iplow}${_iptail:+.}${_iptail}-${_iphead}${_iphead:+.}${_iphigh}${_iptail:+.}${_iptail}).  ${_iphead}${_iphead:+.}${_iplow}${_iptail:+.}${_iptail}-${_iphead}${_iphead:+.}${_ipcount}${_iptail:+.}${_iptail} was processed.  Increase \$netif_ipexpand_max in rc.conf."
882				break
883			else
884				_ipcount=$(($_ipcount + 1))
885			fi
886			# Forcibly set /32 for remaining aliases.
887			_plen=32
888		done
889	done
890
891	for _c in $_retstr; do
892		ifalias_expand_addr_inet $_action $_c $_exargs
893	done
894}
895
896# ifalias_expand_addr_inet6 action addr
897#	Helper function for ifalias_expand_addr().  Handles IPv6.
898#
899ifalias_expand_addr_inet6()
900{
901	local _action _arg _cidr _cidr_addr _exargs
902	local _ipaddr _plen _ipleft _ipright _iplow _iphigh _ipcount
903	local _ipv4part
904	local _retstr _c
905	_action=$1
906	_arg=$2
907	shift 2
908	_exargs=$*
909	_retstr=
910
911	case $_action:$_arg:$_exargs in
912	*:*--*:*)	return ;;	# invalid
913	tmp:*[0-9a-zA-Z]-[0-9a-zA-Z]*:*)# to be expanded
914		_action="alias"
915	;;
916	*:*[0-9a-zA-Z]-[0-9a-zA-Z]*:*)	# to be expanded
917	;;
918	tmp:*:*prefixlen*)	# already expanded w/ prefixlen option
919		echo ${_arg%/[0-9]*} $_exargs && return
920	;;
921	tmp:*:*)		# already expanded w/o prefixlen option
922		echo $_arg $_exargs && return
923	;;
924	*:*:*prefixlen*)	# already expanded w/ prefixlen option
925		echo inet6 ${_arg%/[0-9]*} $_exargs && return
926	;;
927	*:*:*)			# already expanded w/o prefixlen option
928		echo inet6 $_arg $_exargs && return
929	;;
930	esac
931
932	for _cidr in $_arg; do
933		_ipaddr="${_cidr%%/*}"
934		_plen="${_cidr##*/}"
935
936		case $_action:$_ipaddr:$_cidr in
937		-alias:*:*)		unset _plen ;;
938		*:$_cidr:$_ipaddr)	unset _plen ;;
939		esac
940
941		if [ "${_ipaddr%:*.*.*.*}" = "$_ipaddr" ]; then
942			# Handle !v4mapped && !v4compat addresses.
943
944			# The default prefix length is 64.
945			case $_ipaddr:$_cidr in
946			$_cidr:$_ipaddr)	_plen="64" ;;
947			esac
948			_ipleft=${_ipaddr%-*}
949			_ipright=${_ipaddr#*-}
950			_iplow=${_ipleft##*:}
951			_iphigh=${_ipright%%:*}
952			_ipleft=${_ipleft%:*}
953			_ipright=${_ipright#*:}
954
955			if [ "$_iphigh" = "$_ipright" ]; then
956				unset _ipright
957			else
958				_ipright=:$_ipright
959			fi
960
961			if [ -n "$_iplow" -a -n "$_iphigh" ]; then
962				_iplow=$((0x$_iplow))
963				_iphigh=$((0x$_iphigh))
964				_ipcount=$_iplow
965				while [ $_ipcount -le $_iphigh ]; do
966					_r=`printf "%s:%04x%s%s" \
967					    $_ipleft $_ipcount $_ipright \
968					    ${_plen:+/}$_plen`
969					_retstr="$_retstr $_r"
970					if [ $_ipcount -gt $(($_iplow + $netif_ipexpand_max)) ]
971					then
972						warn "Range specification is too large $(printf '(%s:%x%s-%s:%x%s)' "$_ipleft" "$_iplow" "$_ipright" "$_ipleft" "$_iphigh" "$_ipright"). $(printf '%s:%x%s-%s:%x%s' "$_ipleft" "$_iplow" "$_ipright" "$_ipleft" "$_ipcount" "$_ipright") was processed.  Increase \$netif_ipexpand_max in rc.conf."
973						break
974					else
975						_ipcount=$(($_ipcount + 1))
976					fi
977				done
978			else
979				_retstr="${_ipaddr}${_plen:+/}${_plen}"
980			fi
981
982			for _c in $_retstr; do
983				ifalias_expand_addr_inet6 $_action $_c $_exargs
984			done
985		else
986			# v4mapped/v4compat should handle as an IPv4 alias
987			_ipv4part=${_ipaddr##*:}
988
989			# Adjust prefix length if any.  If not, set the
990			# default prefix length as 32.
991			case $_ipaddr:$_cidr in
992			$_cidr:$_ipaddr)	_plen=32 ;;
993			*)			_plen=$(($_plen - 96)) ;;
994			esac
995
996			_retstr=`ifalias_expand_addr_inet \
997			    tmp ${_ipv4part}${_plen:+/}${_plen}`
998			for _c in $_retstr; do
999				ifalias_expand_addr_inet $_action $_c $_exargs
1000			done
1001		fi
1002	done
1003}
1004
1005# ifalias_af_common_handler if af action args
1006#	Helper function for ifalias_af_common().
1007#
1008ifalias_af_common_handler()
1009{
1010	local _ret _if _af _action _args _c _tmpargs
1011
1012	_ret=1
1013	_if=$1
1014	_af=$2
1015	_action=$3
1016	shift 3
1017	_args=$*
1018
1019	case $_args in
1020	${_af}\ *)	;;
1021	*)	return	;;
1022	esac
1023
1024	# link(ether) does not support address removal.
1025	case $_af:$_action in
1026	link:-alias|ether:-alias)	return ;;
1027	esac
1028
1029	_tmpargs=
1030	for _c in $_args; do
1031		case $_c in
1032		${_af})
1033			case $_tmpargs in
1034			${_af}\ *[0-9a-fA-F]-*)
1035				ifalias_af_common_handler $_if $_af $_action \
1036				`ifalias_expand_addr $_af $_action ${_tmpargs#${_af}\ }`
1037			;;
1038			${_af}\ *)
1039				${IFCONFIG_CMD} $_if $_tmpargs $_action && _ret=0
1040			;;
1041			esac
1042			_tmpargs=$_af
1043		;;
1044		*)
1045			_tmpargs="$_tmpargs $_c"
1046		;;
1047		esac
1048	done
1049	# Process the last component if any.
1050	if [ -n "$_tmpargs}" ]; then
1051		case $_tmpargs in
1052		${_af}\ *[0-9a-fA-F]-*)
1053			ifalias_af_common_handler $_if $_af $_action \
1054			`ifalias_expand_addr $_af $_action ${_tmpargs#${_af}\ }`
1055		;;
1056		${_af}\ *)
1057			${IFCONFIG_CMD} $_if $_tmpargs $_action && _ret=0
1058		;;
1059		esac
1060	fi
1061
1062	return $_ret
1063}
1064
1065# ifalias_af_common if af action
1066#	Helper function for ifalias().
1067#
1068ifalias_af_common()
1069{
1070	local _ret _if _af _action alias ifconfig_args _aliasn _c _tmpargs _iaf
1071	local _vif _punct=".-/+"
1072
1073	_ret=1
1074	_aliasn=
1075	_if=$1
1076	_af=$2
1077	_action=$3
1078
1079	# Normalize $_if before using it in a pattern to list_vars()
1080	ltr "$_if" "$_punct" "_" _vif
1081
1082	# ifconfig_IF_aliasN which starts with $_af
1083	for alias in `list_vars ifconfig_${_vif}_alias[0-9]\* |
1084		sort_lite -nk1.$((9+${#_vif}+7))`
1085	do
1086		eval ifconfig_args=\"\$$alias\"
1087		_iaf=
1088		case $ifconfig_args in
1089		inet\ *)	_iaf=inet ;;
1090		inet6\ *)	_iaf=inet6 ;;
1091		link\ *)	_iaf=link ;;
1092		ether\ *)	_iaf=ether ;;
1093		esac
1094
1095		case ${_af}:${_action}:${_iaf}:"${ifconfig_args}" in
1096		${_af}:*:${_af}:*)
1097			_aliasn="$_aliasn $ifconfig_args"
1098			;;
1099		${_af}:*:"":"")
1100			break
1101			;;
1102		inet:alias:"":*)
1103			_aliasn="$_aliasn inet $ifconfig_args"
1104			warn "\$${alias} needs leading" \
1105			    "\"inet\" keyword for an IPv4 address."
1106		esac
1107	done
1108
1109	# backward compatibility: ipv6_ifconfig_IF_aliasN.
1110	case $_af in
1111	inet6)
1112		for alias in `list_vars ipv6_ifconfig_${_vif}_alias[0-9]\* |
1113			sort_lite -nk1.$((14+${#_vif}+7))`
1114		do
1115			eval ifconfig_args=\"\$$alias\"
1116			case ${_action}:"${ifconfig_args}" in
1117			*:"")
1118				break
1119			;;
1120			alias:*)
1121				_aliasn="${_aliasn} inet6 ${ifconfig_args}"
1122				warn "\$${alias} is obsolete. " \
1123				    "Use ifconfig_${_vif}_aliasN instead."
1124			;;
1125			esac
1126		done
1127	esac
1128
1129	# backward compatibility: ipv4_addrs_IF.
1130	for _tmpargs in `get_if_var $_if ipv4_addrs_IF`; do
1131		_aliasn="$_aliasn inet $_tmpargs"
1132	done
1133
1134	# Handle ifconfig_IF_aliases, ifconfig_IF_aliasN, and the others.
1135	_tmpargs=
1136	for _c in `get_if_var $_if ifconfig_IF_aliases` $_aliasn; do
1137		case $_c in
1138		inet|inet6|link|ether)
1139			case $_tmpargs in
1140			${_af}\ *)
1141				eval ifalias_af_common_handler $_if $_af $_action $_tmpargs && _ret=0
1142			;;
1143			esac
1144			_tmpargs=$_c
1145		;;
1146		*)
1147			_tmpargs="$_tmpargs $_c"
1148		esac
1149	done
1150	# Process the last component
1151	case $_tmpargs in
1152	${_af}\ *)
1153		ifalias_af_common_handler $_if $_af $_action $_tmpargs && _ret=0
1154	;;
1155	esac
1156
1157	return $_ret
1158}
1159
1160# ipv6_prefix_hostid_addr_common if action
1161#	Add or remove IPv6 prefix + hostid addr on the interface $if
1162#
1163ipv6_prefix_hostid_addr_common()
1164{
1165	local _if _action prefix j
1166	_if=$1
1167	_action=$2
1168	prefix=`get_if_var ${_if} ipv6_prefix_IF`
1169
1170	if [ -n "${prefix}" ]; then
1171		for j in ${prefix}; do
1172			# The default prefixlen is 64.
1173			plen=${j#*/}
1174			case $j:$plen in
1175			$plen:$j)	plen=64 ;;
1176			*)		j=${j%/*} ;;
1177			esac
1178
1179			# Normalize the last part by removing ":"
1180			j=${j%::*}
1181			j=${j%:}
1182			${IFCONFIG_CMD} ${_if} inet6 $j:: \
1183				prefixlen $plen eui64 ${_action}
1184
1185			# if I am a router, add subnet router
1186			# anycast address (RFC 2373).
1187			if checkyesno ipv6_gateway_enable; then
1188				${IFCONFIG_CMD} ${_if} inet6 $j:: \
1189					prefixlen $plen ${_action} anycast
1190			fi
1191		done
1192	fi
1193}
1194
1195# ipv6_accept_rtadv_up if
1196#	Enable accepting Router Advertisement and send Router
1197#	Solicitation message
1198ipv6_accept_rtadv_up()
1199{
1200	if ipv6_autoconfif $1; then
1201		${IFCONFIG_CMD} $1 inet6 accept_rtadv up
1202		if [ -x /sbin/rtsol ]; then
1203			/sbin/rtsol ${rtsol_flags} $1
1204		fi
1205		return 0
1206	fi
1207	return 1
1208}
1209
1210# ipv6_accept_rtadv_down if
1211#	Disable accepting Router Advertisement
1212ipv6_accept_rtadv_down()
1213{
1214	if ipv6_autoconfif $1; then
1215		${IFCONFIG_CMD} $1 inet6 -accept_rtadv
1216	fi
1217}
1218
1219# ifscript_up if
1220#	Evaluate a startup script for the $if interface.
1221#	It returns 0 if a script was found and processed or
1222#	1 if no script was found.
1223#
1224ifscript_up()
1225{
1226	if [ -r /etc/start_if.$1 ]; then
1227		. /etc/start_if.$1
1228		return 0
1229	else
1230		return 1
1231	fi
1232}
1233
1234# ifscript_down if
1235#	Evaluate a shutdown script for the $if interface.
1236#	It returns 0 if a script was found and processed or
1237#	1 if no script was found.
1238#
1239ifscript_down()
1240{
1241	if [ -r /etc/stop_if.$1 ]; then
1242		. /etc/stop_if.$1
1243		return 0
1244	else
1245		return 1
1246	fi
1247}
1248
1249# wlan_up
1250#	Create IEEE802.11 interfaces.
1251#
1252wlan_up()
1253{
1254	local _list _iflist parent child_wlans child create_args debug_flags
1255	_list=
1256	_iflist=$*
1257
1258	# Parse wlans_$parent="$child ..."
1259	for parent in `set | sed -nE 's/wlans_([a-z]+[0-9]+)=.*/\1/p'`; do
1260		child_wlans=`get_if_var $parent wlans_IF`
1261		for child in ${child_wlans}; do
1262			create_args="wlandev $parent `get_if_var $child create_args_IF`"
1263			debug_flags="`get_if_var $child wlandebug_IF`"
1264			case $_iflist in
1265			""|$child|$child\ *|*\ $child\ *|*\ $child)	;;
1266			*)	continue ;;
1267			esac
1268			# Skip if ${child} already exists.
1269			if ${IFCONFIG_CMD} $child > /dev/null 2>&1; then
1270				continue
1271			fi
1272			if expr $child : 'wlan[0-9][0-9]*$' >/dev/null 2>&1; then
1273				${IFCONFIG_CMD} $child create ${create_args} && cfg=0
1274			else
1275				${IFCONFIG_CMD} wlan create ${create_args} name $child && cfg=0
1276			fi
1277			if [ $? -eq 0 ]; then
1278				_list="$_list $child"
1279			fi
1280			if [ -n "${debug_flags}" ]; then
1281				wlandebug -i $child ${debug_flags}
1282			fi
1283		done
1284	done
1285	if [ -n "${_list# }" ]; then
1286		echo "Created wlan(4) interfaces: ${_list# }."
1287	fi
1288	debug "Created wlan(4)s: ${_list# }"
1289}
1290
1291# wlan_down
1292#	Destroy IEEE802.11 interfaces.
1293#
1294wlan_down()
1295{
1296	local _list _iflist parent child_wlans child
1297	_list=
1298	_iflist=$*
1299
1300	# Parse wlans_$parent="$child ..."
1301	for parent in `set | sed -nE 's/wlans_([a-z]+[0-9]+)=.*/\1/p'`; do
1302		child_wlans=`get_if_var $parent wlans_IF`
1303		for child in ${child_wlans}; do
1304			case $_iflist in
1305			""|$child|$child\ *|*\ $child\ *|*\ $child)	;;
1306			*)	continue ;;
1307			esac
1308			# Skip if ${child} doesn't exists.
1309			if ! ${IFCONFIG_CMD} $child > /dev/null 2>&1; then
1310				continue
1311			fi
1312			${IFCONFIG_CMD} -n ${child} destroy
1313			if [ $? -eq 0 ]; then
1314				_list="$_list $child"
1315			fi
1316		done
1317	done
1318	if [ -n "${_list# }" ]; then
1319		echo "Destroyed wlan(4) interfaces: ${_list# }."
1320	fi
1321	debug "Destroyed wlan(4)s: ${_list# }"
1322}
1323
1324# clone_up
1325#	Create cloneable interfaces.
1326#
1327clone_up()
1328{
1329	local _list ifn ifopt _iflist _inet6 _n tmpargs
1330	_list=
1331	_iflist=$*
1332
1333	# create_args_IF
1334	for ifn in ${cloned_interfaces}; do
1335		# Parse ifn:ifopt.
1336		OIFS=$IFS; IFS=:; set -- $ifn; ifn=$1; ifopt=$2; IFS=$OIFS
1337		case $_iflist in
1338		""|$ifn|$ifn\ *|*\ $ifn\ *|*\ $ifn)	;;
1339		*)	continue ;;
1340		esac
1341		case $ifn in
1342		epair[0-9]*)
1343			# epair(4) uses epair[0-9] for creation and
1344			# epair[0-9][ab] for configuration.
1345			#
1346			# Skip if ${ifn}a or ${ifn}b already exist.
1347			if ${IFCONFIG_CMD} ${ifn}a > /dev/null 2>&1; then
1348				continue
1349			elif ${IFCONFIG_CMD} ${ifn}b > /dev/null 2>&1; then
1350				continue
1351			fi
1352			${IFCONFIG_CMD} ${ifn} create \
1353			    `get_if_var ${ifn} create_args_IF`
1354			if [ $? -eq 0 ]; then
1355				_list="$_list ${ifn}a ${ifn}b"
1356			fi
1357		;;
1358		*)
1359			# Skip if ${ifn} already exists.
1360			if ${IFCONFIG_CMD} $ifn > /dev/null 2>&1; then
1361				continue
1362			fi
1363			${IFCONFIG_CMD} ${ifn} create \
1364			    `get_if_var ${ifn} create_args_IF`
1365			if [ $? -eq 0 ]; then
1366				_list="$_list $ifn"
1367			fi
1368		esac
1369	done
1370	for ifn in ${gif_interfaces}; do
1371		# Parse ifn:ifopt.
1372		OIFS=$IFS; IFS=:; set -- $ifn; ifn=$1; ifopt=$2; IFS=$OIFS
1373		case $_iflist in
1374		""|$ifn|$ifn\ *|*\ $ifn\ *|*\ $ifn)	;;
1375		*)	continue ;;
1376		esac
1377		# Skip if ifn already exists.
1378		if ${IFCONFIG_CMD} $ifn > /dev/null 2>&1; then
1379			continue
1380		fi
1381		case $ifn in
1382		gif[0-9]*)
1383			${IFCONFIG_CMD} $ifn create
1384		;;
1385		*)
1386			_n=$(${IFCONFIG_CMD} gif create)
1387			${IFCONFIG_CMD} $_n name $ifn
1388		;;
1389		esac
1390		if [ $? -eq 0 ]; then
1391			_list="$_list $ifn"
1392			tmpargs=$(get_if_var $ifn gifconfig_IF)
1393			_inet6=''
1394			case "$tmpargs" in
1395			'')
1396			;;
1397			inet6[[:space:]]*)
1398				tmpargs=${tmpargs#inet6}
1399				_inet6=inet6
1400				# FALLTHROUGH
1401			;&
1402			*)
1403				${IFCONFIG_CMD} $ifn $_inet6 tunnel $tmpargs
1404			;;
1405			esac
1406		fi
1407	done
1408	if [ -n "${_list# }" ]; then
1409		echo "Created clone interfaces: ${_list# }."
1410	fi
1411	debug "Cloned: ${_list# }"
1412}
1413
1414# clone_down
1415#	Destroy cloned interfaces. Destroyed interfaces are echoed to
1416#	standard output.
1417#
1418clone_down()
1419{
1420	local _list ifn _difn ifopt _iflist _sticky
1421	_list=
1422	_iflist=$*
1423
1424	: ${cloned_interfaces_sticky:=NO}
1425	if checkyesno cloned_interfaces_sticky; then
1426		_sticky=1
1427	else
1428		_sticky=0
1429	fi
1430	for ifn in ${cloned_interfaces} ${gif_interfaces}; do
1431		# Parse ifn:ifopt.
1432		OIFS=$IFS; IFS=:; set -- $ifn; ifn=$1; ifopt=$2; IFS=$OIFS
1433		case $ifopt:$_sticky in
1434		sticky:*)	continue ;;	# :sticky => not destroy
1435		nosticky:*)	;;		# :nosticky => destroy
1436		*:1)		continue ;;	# global sticky knob == 1
1437		esac
1438		case $_iflist in
1439		""|$ifn|$ifn\ *|*\ $ifn\ *|*\ $ifn)	;;
1440		*)	continue ;;
1441		esac
1442		case $ifn in
1443		epair[0-9]*)
1444			# Note: epair(4) uses epair[0-9] for removal and
1445			# epair[0-9][ab] for configuration.
1446			#
1447			# Skip if both of ${ifn}a and ${ifn}b do not exist.
1448			if ${IFCONFIG_CMD} ${ifn}a > /dev/null 2>&1; then
1449				_difn=${ifn}a
1450			elif ${IFCONFIG_CMD} ${ifn}b > /dev/null 2>&1; then
1451				_difn=${ifn}b
1452			else
1453				continue
1454			fi
1455			${IFCONFIG_CMD} -n $_difn destroy
1456			if [ $? -eq 0 ]; then
1457				_list="$_list ${ifn}a ${ifn}b"
1458			fi
1459		;;
1460		*)
1461			# Skip if ifn does not exist.
1462			if ! ${IFCONFIG_CMD} $ifn > /dev/null 2>&1; then
1463				continue
1464			fi
1465			${IFCONFIG_CMD} -n ${ifn} destroy
1466			if [ $? -eq 0 ]; then
1467				_list="$_list $ifn"
1468			fi
1469		;;
1470		esac
1471	done
1472	if [ -n "${_list# }" ]; then
1473		echo "Destroyed clone interfaces: ${_list# }."
1474	fi
1475	debug "Destroyed clones: ${_list# }"
1476}
1477
1478# childif_create
1479#	Create and configure child interfaces.  Return 0 if child
1480#	interfaces are created.
1481#
1482childif_create()
1483{
1484	local cfg child child_vlans create_args debug_flags ifn i
1485	cfg=1
1486	ifn=$1
1487
1488	# Create vlan interfaces
1489	child_vlans=`get_if_var $ifn vlans_IF`
1490
1491	if [ -n "${child_vlans}" ]; then
1492		load_kld if_vlan
1493	fi
1494
1495	for child in ${child_vlans}; do
1496		if expr $child : '[1-9][0-9]*$' >/dev/null 2>&1; then
1497			child="${ifn}.${child}"
1498			create_args=`get_if_var $child create_args_IF`
1499			${IFCONFIG_CMD} $child create ${create_args} && cfg=0
1500		else
1501			create_args="vlandev $ifn `get_if_var $child create_args_IF`"
1502			if expr $child : 'vlan[0-9][0-9]*$' >/dev/null 2>&1; then
1503				${IFCONFIG_CMD} $child create ${create_args} && cfg=0
1504			else
1505				i=`${IFCONFIG_CMD} vlan create ${create_args}`
1506				${IFCONFIG_CMD} $i name $child && cfg=0
1507			fi
1508		fi
1509		if autoif $child; then
1510			ifn_start $child
1511		fi
1512	done
1513
1514	return ${cfg}
1515}
1516
1517# childif_destroy
1518#	Destroy child interfaces.
1519#
1520childif_destroy()
1521{
1522	local cfg child child_vlans ifn
1523	cfg=1
1524
1525	child_vlans=`get_if_var $ifn vlans_IF`
1526	for child in ${child_vlans}; do
1527		if expr $child : '[1-9][0-9]*$' >/dev/null 2>&1; then
1528			child="${ifn}.${child}"
1529		fi
1530		if ! ifexists $child; then
1531			continue
1532		fi
1533		${IFCONFIG_CMD} -n $child destroy && cfg=0
1534	done
1535
1536	return ${cfg}
1537}
1538
1539# ng_mkpeer
1540#	Create netgraph nodes.
1541#
1542ng_mkpeer()
1543{
1544	ngctl -f - 2> /dev/null <<EOF
1545mkpeer $*
1546msg dummy nodeinfo
1547EOF
1548}
1549
1550# ng_create_one
1551#	Create netgraph nodes.
1552#
1553ng_create_one()
1554{
1555	local t
1556
1557	ng_mkpeer $* | while read line; do
1558		t=`expr "${line}" : '.* name="\([a-z]*[0-9]*\)" .*'`
1559		if [ -n "${t}" ]; then
1560			echo ${t}
1561			return
1562		fi
1563	done
1564}
1565
1566# ifnet_rename [ifname]
1567#	Rename interfaces if ifconfig_IF_name is defined.
1568#
1569ifnet_rename()
1570{
1571	local _if _ifname
1572
1573	# ifconfig_IF_name
1574	for _if in ${*:-$(${IFCONFIG_CMD} -l)}; do
1575		_ifname=`get_if_var $_if ifconfig_IF_name`
1576		if [ ! -z "$_ifname" ]; then
1577			${IFCONFIG_CMD} $_if name $_ifname
1578		fi
1579	done
1580
1581	return 0
1582}
1583
1584# list_net_interfaces type
1585#	List all network interfaces. The type of interface returned
1586#	can be controlled by the type argument. The type
1587#	argument can be any of the following:
1588#		nodhcp	- all interfaces, excluding DHCP configured interfaces
1589#		dhcp	- list only DHCP configured interfaces
1590#		noautoconf	- all interfaces, excluding IPv6 Stateless
1591#				  Address Autoconf configured interfaces
1592#		autoconf	- list only IPv6 Stateless Address Autoconf
1593#				  configured interfaces
1594#	If no argument is specified all network interfaces are output.
1595#	Note that the list will include cloned interfaces if applicable.
1596#	Cloned interfaces must already exist to have a chance to appear
1597#	in the list if ${network_interfaces} is set to `auto'.
1598#
1599list_net_interfaces()
1600{
1601	local type _tmplist _list _autolist _lo _if
1602	type=$1
1603
1604	# Get a list of ALL the interfaces and make lo0 first if it's there.
1605	#
1606	_tmplist=
1607	case ${network_interfaces} in
1608	[Aa][Uu][Tt][Oo])
1609		_autolist="`${IFCONFIG_CMD} -l`"
1610		_lo=
1611		for _if in ${_autolist} ; do
1612			if autoif $_if; then
1613				if [ "$_if" = "lo0" ]; then
1614					_lo="lo0 "
1615				else
1616					_tmplist="${_tmplist} ${_if}"
1617				fi
1618			fi
1619		done
1620		_tmplist="${_lo}${_tmplist# }"
1621	;;
1622	*)
1623		for _if in ${network_interfaces} ${cloned_interfaces}; do
1624			# epair(4) uses epair[0-9] for creation and
1625			# epair[0-9][ab] for configuration.
1626			case $_if in
1627			epair[0-9]*)
1628				_tmplist="$_tmplist ${_if}a ${_if}b"
1629			;;
1630			*)
1631				_tmplist="$_tmplist $_if"
1632			;;
1633			esac
1634		done
1635		#
1636		# lo0 is effectively mandatory, so help prevent foot-shooting
1637		#
1638		case "$_tmplist" in
1639		lo0|'lo0 '*|*' lo0'|*' lo0 '*)
1640			# This is fine, do nothing
1641			_tmplist="${_tmplist# }"
1642		;;
1643		*)
1644			_tmplist="lo0 ${_tmplist# }"
1645		;;
1646		esac
1647	;;
1648	esac
1649
1650	_list=
1651	case "$type" in
1652	nodhcp)
1653		for _if in ${_tmplist} ; do
1654			if ! dhcpif $_if && \
1655			   [ -n "`_ifconfig_getargs $_if`" ]; then
1656				_list="${_list# } ${_if}"
1657			fi
1658		done
1659	;;
1660	dhcp)
1661		for _if in ${_tmplist} ; do
1662			if dhcpif $_if; then
1663				_list="${_list# } ${_if}"
1664			fi
1665		done
1666	;;
1667	noautoconf)
1668		for _if in ${_tmplist} ; do
1669			if ! ipv6_autoconfif $_if && \
1670			   [ -n "`_ifconfig_getargs $_if ipv6`" ]; then
1671				_list="${_list# } ${_if}"
1672			fi
1673		done
1674	;;
1675	autoconf)
1676		for _if in ${_tmplist} ; do
1677			if ipv6_autoconfif $_if; then
1678				_list="${_list# } ${_if}"
1679			fi
1680		done
1681	;;
1682	*)
1683		_list=${_tmplist}
1684	;;
1685	esac
1686
1687	echo $_list
1688
1689	return 0
1690}
1691
1692# get_default_if -address_family
1693#	Get the interface of the default route for the given address family.
1694#	The -address_family argument must be suitable passing to route(8).
1695#
1696get_default_if()
1697{
1698	local routeget oldifs defif line
1699	defif=
1700	oldifs="$IFS"
1701	IFS="
1702"
1703	for line in `route -n get $1 default 2>/dev/null`; do
1704		case $line in
1705		*interface:*)
1706			defif=${line##*: }
1707			;;
1708		esac
1709	done
1710	IFS=${oldifs}
1711
1712	echo $defif
1713}
1714
1715# hexdigit arg
1716#	Echo decimal number $arg (single digit) in hexadecimal format.
1717hexdigit()
1718{
1719	printf '%x\n' "$1"
1720}
1721
1722# hexprint arg
1723#	Echo decimal number $arg (multiple digits) in hexadecimal format.
1724hexprint()
1725{
1726	printf '%x\n' "$1"
1727}
1728
1729is_wired_interface()
1730{
1731	local media
1732
1733	case `${IFCONFIG_CMD} $1 2>/dev/null` in
1734	*media:?Ethernet*) media=Ethernet ;;
1735	esac
1736
1737	test "$media" = "Ethernet"
1738}
1739
1740# network6_getladdr if [flag]
1741#	Echo link-local address from $if if any.
1742#	If flag is defined, tentative ones will be excluded.
1743network6_getladdr()
1744{
1745	local _if _flag proto addr rest
1746	_if=$1
1747	_flag=$2
1748
1749	${IFCONFIG_CMD} $_if 2>/dev/null | while read proto addr rest; do
1750		case "${proto}/${addr}/${_flag}/${rest}" in
1751		inet6/fe80::*//*)
1752			echo ${addr}
1753		;;
1754		inet6/fe80:://*tentative*)	# w/o flag
1755			sleep `${SYSCTL_N} net.inet6.ip6.dad_count`
1756			network6_getladdr $_if $_flags
1757		;;
1758		inet6/fe80::/*/*tentative*)	# w/ flag
1759			echo ${addr}
1760		;;
1761		*)
1762			continue
1763		;;
1764		esac
1765
1766		return
1767	done
1768}
1769