xref: /freebsd/usr.sbin/bsdconfig/networking/share/device.subr (revision 4ef0123eb70fb892d6b6b511c67e94abb3301c96)
1if [ ! "$_NETWORKING_DEVICE_SUBR" ]; then _NETWORKING_DEVICE_SUBR=1
2#
3# Copyright (c) 2006-2013 Devin Teske
4# All Rights Reserved.
5#
6# Redistribution and use in source and binary forms, with or without
7# modification, are permitted provided that the following conditions
8# are met:
9# 1. Redistributions of source code must retain the above copyright
10#    notice, this list of conditions and the following disclaimer.
11# 2. Redistributions in binary form must reproduce the above copyright
12#    notice, this list of conditions and the following disclaimer in the
13#    documentation and/or other materials provided with the distribution.
14#
15# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING BUT NOT LIMITED TO, THE
17# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18# ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20# DAMAGES (INLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25# SUCH DAMAGE.
26#
27# $FreeBSD$
28#
29############################################################ INCLUDES
30
31BSDCFG_SHARE="/usr/share/bsdconfig"
32. $BSDCFG_SHARE/common.subr || exit 1
33f_dprintf "%s: loading includes..." networking/device.subr
34f_include $BSDCFG_SHARE/device.subr
35f_include $BSDCFG_SHARE/dialog.subr
36f_include $BSDCFG_SHARE/media/tcpip.subr
37f_include $BSDCFG_SHARE/networking/common.subr
38f_include $BSDCFG_SHARE/networking/ipaddr.subr
39f_include $BSDCFG_SHARE/networking/media.subr
40f_include $BSDCFG_SHARE/networking/netmask.subr
41f_include $BSDCFG_SHARE/networking/resolv.subr
42f_include $BSDCFG_SHARE/networking/routing.subr
43f_include $BSDCFG_SHARE/sysrc.subr
44
45BSDCFG_LIBE="/usr/libexec/bsdconfig" APP_DIR="120.networking"
46f_include_lang $BSDCFG_LIBE/$APP_DIR/include/messages.subr
47
48############################################################ GLOBALS
49
50#
51# Settings used while interacting with various dialog(1) menus
52#
53: ${DIALOG_MENU_NETDEV_KICK_INTERFACES=1}
54: ${DIALOG_MENU_NETDEV_SLEEP_AFTER_KICK=3}
55
56############################################################ FUNCTIONS
57
58# f_dialog_menu_netdev [$default]
59#
60# Display a list of network devices with descriptions. Optionally, if present
61# and non-NULL, initially highlight $default interface.
62#
63f_dialog_menu_netdev()
64{
65	local defaultitem="${1%\*}" # Tim trailing asterisk if present
66
67	#
68	# Display a message to let the user know we're working...
69	# (message will remain until we throw up the next dialog)
70	#
71	f_dialog_info "$msg_probing_network_interfaces"
72
73	#
74	# Get list of usable network interfaces
75	#
76	local d='[[:digit:]]+:'
77	local iflist="`echo "$(ifconfig -l):" | sed -E -e "
78		# Convert all spaces to colons
79		y/ /:/
80
81		# Prune unsavory interfaces
82		s/lo$d//g
83		s/ppp$d//g
84		s/sl$d//g
85		s/faith$d//g
86
87		# Convert all colons back into spaces
88		y/:/ /
89	"`"
90
91	#
92	# Optionally kick interfaces in the head to get them to accurately
93	# track the carrier status in realtime (required on FreeBSD).
94	#
95	if [ "$DIALOG_MENU_NETDEV_KICK_INTERFACES" ]; then
96		DIALOG_MENU_NETDEV_KICK_INTERFACES=
97
98		local ifn
99		for ifn in $iflist; do
100			f_quietly ifconfig $ifn up
101		done
102
103		if [ "$DIALOG_MENU_NETDEV_SLEEP_AFTER_KICK" ]; then
104			# interfaces need time to update carrier status
105			sleep $DIALOG_MENU_NETDEV_SLEEP_AFTER_KICK
106		fi
107	fi
108
109	#
110	# Mark any "active" interfaces with an asterisk (*)
111	# to the right of the device name.
112	#
113	interfaces=$(
114		for ifn in $iflist; do
115			active=$( ifconfig $ifn | awk \
116			'
117				( $1 == "status:" ) \
118				{
119					if ( $2 == "active" ) { print 1; exit }
120				}
121			' )
122			printf "'%s%s' '%s'\n" \
123				$ifn "${active:+*}" "$( f_device_desc $ifn )"
124		done
125	)
126	if [ ! "$interfaces" ]; then
127		f_dialog_msgbox "$msg_no_network_interfaces"
128		return $FAILURE
129	fi
130
131	#
132	# Maybe the default item was marked as active
133	#
134	if [ "$defaultitem" ]; then
135		ifconfig "$defaultitem" 2> /dev/null | awk \
136			'( $1 == "status:" && $2 != "active" ) { exit 0 }' ||
137			defaultitem="$defaultitem*"
138	fi
139
140	local hline="$hline_arrows_tab_enter"
141
142	#
143	# Ask user to select an interface
144	#
145	local prompt="$msg_select_network_interface"
146	local height width rows
147	eval f_dialog_menu_size height width rows \
148	                        \"\$DIALOG_TITLE\"     \
149	                        \"\$DIALOG_BACKTITLE\" \
150	                        \"\$prompt\"           \
151	                        \"\$hline\"            \
152	                        $interfaces
153	local menu_choice
154	menu_choice=$( eval $DIALOG \
155		--title \"\$DIALOG_TITLE\"         \
156		--backtitle \"\$DIALOG_BACKTITLE\" \
157		--hline \"\$hline\"                \
158		--ok-label \"\$msg_ok\"            \
159		--cancel-label \"\$msg_cancel\"    \
160		--default-item \"\$defaultitem\"   \
161		--menu \"\$prompt\"                \
162		$height $width $rows               \
163		$interfaces                        \
164		2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
165	)
166	local retval=$?
167	f_dialog_menutag_store -s "$menu_choice"
168	return $retval
169}
170
171# f_dialog_menu_netdev_edit $interface $ipaddr $netmask $options $dhcp
172#
173# Allow a user to edit network interface settings. Current values are not
174# probed but rather taken from the positional arguments.
175#
176f_dialog_menu_netdev_edit()
177{
178	local interface="$1" ipaddr="$2" netmask="$3" options="$4" dhcp="$5"
179	local prompt menu_list height width rows
180
181	#
182	# Create a duplicate set of variables for change-tracking...
183	#
184	local ipaddr_orig="$2"  \
185	      netmask_orig="$3" \
186	      options_orig="$4" \
187	      dhcp_orig="$5"
188
189	local hline="$hline_arrows_tab_enter"
190	prompt=$( printf "$msg_network_configuration" "$interface" )
191
192	#
193	# Loop forever until the user has finished configuring the different
194	# components of the network interface.
195	#
196	# To apply the settings, we need to know each of the following:
197	# 	- IP Address
198	# 	- Network subnet mask
199	# 	- Additional ifconfig(8) options
200	#
201	# It is only when we have all of the above values that we can make the
202	# changes effective because all three options must be specified at-once
203	# to ifconfig(8).
204	#
205	local defaultitem=
206	while :; do
207		local dhcp_status="$msg_disabled"
208		[ "$dhcp" ] && dhcp_status="$msg_enabled"
209
210		#
211		# Display configuration-edit menu
212		#
213		menu_list="
214			'X $msg_save_exit' '$msg_return_to_previous_menu'
215			'2 $msg_dhcp'      '$dhcp_status'
216			'3 $msg_ipaddr4'   '$ipaddr'
217			'4 $msg_netmask'   '$netmask'
218			'5 $msg_options'   '$options'
219		"
220		eval f_dialog_menu_size height width rows \
221		                        \"\$DIALOG_TITLE\"     \
222		                        \"\$DIALOG_BACKTITLE\" \
223		                        \"\$prompt\"           \
224		                        \"\$hline\"            \
225		                        $menu_list
226		local tag
227		tag=$( eval $DIALOG \
228			--title \"\$DIALOG_TITLE\"         \
229			--backtitle \"\$DIALOG_BACKTITLE\" \
230			--hline \"\$hline\"                \
231			--ok-label \"\$msg_ok\"            \
232			--cancel-label \"\$msg_cancel\"    \
233			--help-button                      \
234			--help-label \"\$msg_help\"        \
235			${USE_XDIALOG:+--help \"\"}        \
236			--default-item \"\$defaultitem\"   \
237			--menu \"\$prompt\"                \
238			$height $width $rows               \
239			$menu_list                         \
240			2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
241		)
242		local retval=$?
243		f_dialog_data_sanitize tag
244
245		if [ $retval -eq 2 ]; then
246			# The Help button was pressed
247			f_show_help "$TCP_HELPFILE"
248			continue
249		elif [ $retval -ne $SUCCESS ]; then
250			# "Cancel" was chosen (-1) or ESC was pressed (255)
251			return $retval
252		else
253			# Only update default-item on success
254			defaultitem="$tag"
255		fi
256
257		#
258		# Call the below ``modifier functions'' whose job it is to take
259		# input from the user and assign the newly-acquired values back
260		# to the ipaddr, netmask, and options variables for us to re-
261		# read and display in the summary dialog.
262		#
263		case "$tag" in
264		X\ *) break;;
265		2\ *) #
266		      # Do not proceed if/when there are NFS-mounts currently
267		      # active. If the network is changed while NFS-exported
268		      # directories are mounted, the system may hang (if any
269		      # NFS mounts are using that interface).
270		      #
271		      if f_nfs_mounted && ! f_jailed; then
272		      	local setting="$( printf "$msg_current_dhcp_status" \
273		      	                         "$interface" "$dhcp_status" )"
274			f_show_msg "$msg_nfs_mounts_may_cause_hang" "$setting"
275		      	continue
276		      fi
277
278		      #
279		      # Toggle DHCP status
280		      #
281		      if [ "$dhcp_status" = "$msg_enabled" ]; then
282		      	dhcp=
283		      else
284		      	trap - SIGINT
285		      	( # Execute within sub-shell to allow/catch Ctrl-C
286		      	  trap 'exit $FAILURE' SIGINT
287		      	  msg=$( printf "$msg_scanning_for_dhcp" "$interface" )
288		      	  if [ "$USE_XDIALOG" ]; then
289		      	  	(
290		      	  	  f_quietly ifconfig $interface delete
291		      	  	  f_quietly dhclient $interface
292		      	  	) |
293		      	  	  f_xdialog_info "$msg"
294		      	  else
295		      	  	f_dialog_info "$msg"
296		      	  	f_quietly ifconfig $interface delete
297		      	  	f_quietly dhclient $interface
298		      	  fi
299		      	)
300		      	retval=$?
301		      	trap 'interrupt' SIGINT
302		      	if [ $retval -eq $SUCCESS ]; then
303		      		dhcp=1
304		      		ipaddr=$( f_ifconfig_inet $interface )
305		      		netmask=$( f_ifconfig_netmask $interface )
306		      		options=
307
308		      		# Fixup search/domain in resolv.conf(5)
309		      		hostname=$( f_sysrc_get \
310				            	'hostname:-$(hostname)' )
311		      		f_dialog_resolv_conf_update "$hostname"
312		      	fi
313		      fi
314		      ;;
315		3\ *) f_dialog_input_ipaddr "$interface" "$ipaddr"
316		      [ $? -eq $SUCCESS ] && dhcp=;;
317		4\ *) f_dialog_input_netmask "$interface" "$netmask"
318		      [ $? -eq $SUCCESS -a "$_netmask" ] && dhcp=;;
319		5\ *) f_dialog_menu_media_options "$interface" "$options"
320		      [ $? -eq $SUCCESS ] && dhcp=;;
321		esac
322	done
323
324	#
325	# Save only if the user changed at least one feature of the interface
326	#
327	if [ "$ipaddr"  != "$ipaddr_orig"  -o \
328	     "$netmask" != "$netmask_orig" -o \
329	     "$options" != "$options_orig" -o \
330	     "$dhcp"    != "$dhcp_orig" ]
331	then
332		f_show_info "$msg_saving_network_interface" "$interface"
333
334		local value=
335		if [ "$dhcp" ]; then
336			f_sysrc_delete defaultrouter
337			value=DHCP
338		else
339			value="inet $ipaddr netmask $netmask"
340			value="$value${options:+ }$options"
341		fi
342
343		f_sysrc_set ifconfig_$interface "$value"
344	fi
345
346	#
347	# Re/Apply the settings if desired
348	#
349	if [ ! "$dhcp" ]; then
350		if f_yesno "$msg_bring_interface_up" "$interface"
351		then
352			f_show_info "$msg_bring_interface_up" "$interface"
353
354			local dr="$( f_sysrc_get defaultrouter )" err
355			if [ "$dr" = "NO" -o ! "$dr" ]; then
356				dr=$( f_route_get_default )
357				[ "$dr" ] && f_sysrc_set defaultrouter "$dr"
358			fi
359			#
360			# Make a backup of resolv.conf(5) before using
361			# ifconfig(8) and then restore it afterward. This
362			# allows preservation of nameservers acquired via
363			# DHCP on FreeBSD-8.x (normally lost as ifconfig(8)
364			# usage causes dhclient(8) to exit which scrubs
365			# resolv.conf(5) by-default upon termination).
366			#
367			f_quietly cp -fp "$RESOLV_CONF" "$RESOLV_CONF.$$"
368			err=$( ifconfig $interface inet $ipaddr \
369			       	netmask $netmask $options 2>&1 )
370			if [ $? -eq $SUCCESS ]; then
371				if [ "$dr" -a "$dr" != "NO" ]; then
372					err=$( route add default "$dr" 2>&1 )
373					[ $? -eq $SUCCESS ] || \
374						dialog_msgbox "$err"
375				fi
376			else
377				dialog_msgbox "$err"
378			fi
379			if cmp -s "$RESOLV_CONF" "$RESOLV_CONF.$$"; then
380				f_quietly rm -f "$RESOLV_CONF.$$"
381			else
382				f_quietly mv -f "$RESOLV_CONF.$$" "$RESOLV_CONF"
383			fi
384		fi
385	fi
386
387	return $SUCCESS
388}
389
390############################################################ MAIN
391
392f_dprintf "%s: Successfully loaded." networking/device.subr
393
394fi # ! $_NETWORKING_DEVICE_SUBR
395