3 # Copyright (c) 2011 Nathan Whitehorn
4 # Copyright (c) 2013-2016 Devin Teske
7 # Redistribution and use in source and binary forms, with or without
8 # modification, are permitted provided that the following conditions
10 # 1. Redistributions of source code must retain the above copyright
11 # notice, this list of conditions and the following disclaimer.
12 # 2. Redistributions in binary form must reproduce the above copyright
13 # notice, this list of conditions and the following disclaimer in the
14 # documentation and/or other materials provided with the distribution.
16 # THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17 # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 # ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20 # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22 # OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23 # HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24 # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25 # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30 ############################################################ INCLUDES
32 BSDCFG_SHARE="/usr/share/bsdconfig"
33 . $BSDCFG_SHARE/common.subr || exit 1
34 f_include $BSDCFG_SHARE/dialog.subr
35 f_dialog_backtitle "FreeBSD Installer"
37 ############################################################ FUNCTIONS
41 local error_str iface_up ifconfig_args=
44 # Setup what was selected
45 # NB: Do not change order of arguments (or regdomain will be ignored)
47 [ "$2" ] && ifconfig_args="$ifconfig_args country $2"
48 [ "$1" ] && ifconfig_args="$ifconfig_args regdomain $1"
49 [ "$ifconfig_args" ] || return $SUCCESS # Nothing to do
50 ifconfig_args="${ifconfig_args# }"
52 # Regdomain/country cannot be applied while interface is running
53 iface_up=$( ifconfig -lu | grep -w "$WLAN_IFACE" )
54 [ "$iface_up" ] && ifconfig "$WLAN_IFACE" down
55 error_str=$( ifconfig "$WLAN_IFACE" $ifconfig_args 2>&1 |
56 sed -e 's/ifconfig: //' )
57 if [ "$iface_up" ]; then
58 # Restart wpa_supplicant(8) (should not fail).
59 wpa_supplicant -B -i "$WLAN_IFACE" -c \
60 "$BSDINSTALL_TMPETC/wpa_supplicant.conf"
62 if [ "$error_str" ]; then
64 --title "$msg_error" \
65 --backtitle "$DIALOG_BACKTITLE" \
69 "Error while applying chosen settings ($error_str)" \
70 0 0 || return $SUCCESS # Skip
71 return $FAILURE # Restart
73 awk 'sub(/^\t\t/,"")||1' \
74 > "$BSDINSTALL_TMPETC/rc.conf.net.wlan" <<-EOF
75 create_args_$WLAN_IFACE="$ifconfig_args"
82 dialog_country_select()
84 local input regdomains countries regdomain country prompt
85 local no_default="<not selected>"
86 local default_regdomain="${1:-$no_default}"
87 local default_country="${2:-$no_default}"
90 # Parse available countries/regdomains
92 input=$( ifconfig "$WLAN_IFACE" list countries | sed -e 's/DEBUG//gi' )
93 regdomains=$( echo "$input" | awk '
94 sub(/.*domains:/, ""), /[^[:alnum:][[:space:]]/ {
95 n = split($0, domains)
96 for (i = 1; i <= n; i++)
97 printf "'\''%s'\'' '\'\''", domains[i]
100 countries=$( echo "$input" | awk '
101 sub(/Country codes:/, ""), sub(/Regulatory.*/, "") {
102 while (match($0, /[[:upper:]][[:upper:][:digit:]] /)) {
103 country = substr($0, RSTART)
104 sub(/ [[:upper:]][[:upper:][:digit:]].*/, "", country)
105 code = substr(country, 1, 2)
106 desc = substr(country, 4)
107 sub(/[[:space:]]*$/, "", desc)
108 printf "'\''%s'\'' '\''%s'\''\n", code, desc
109 $0 = substr($0, RSTART + RLENGTH)
114 f_dialog_title "Regdomain selection"
115 prompt="Select your regdomain."
116 eval f_dialog_menu_size height width rows \
117 \"\$DIALOG_TITLE\" \"\$DIALOG_BACKTITLE\" \
118 \"\$prompt\" \"\" $regdomains
119 regdomain=$( eval $DIALOG \
120 --title \"\$DIALOG_TITLE\" \
121 --backtitle \"\$DIALOG_BACKTITLE\" \
122 --cancel-label \"\$msg_skip\" \
123 --default-item \"\$default_regdomain\" \
124 --menu \"\$prompt\" \
125 $height $width $rows \
127 2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
129 f_dialog_data_sanitize regdomain
131 f_dialog_title "Country selection"
132 prompt="Select your country."
133 eval f_dialog_menu_size height width rows \
134 \"\$DIALOG_TITLE\" \"\$DIALOG_BACKTITLE\" \
135 \"\$prompt\" \"\" $countries
136 country=$( eval $DIALOG \
137 --title \"\$DIALOG_TITLE\" \
138 --backtitle \"\$DIALOG_BACKTITLE\" \
139 --cancel-label \"\$msg_skip\" \
140 --default-item \"\$default_country\" \
141 --menu \"\$prompt\" \
142 $height $width $rows \
144 2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
146 f_dialog_data_sanitize country
148 country_set "$regdomain" "$country"
151 ############################################################ MAIN
153 : > "$BSDINSTALL_TMPETC/wpa_supplicant.conf"
154 chmod 0600 "$BSDINSTALL_TMPETC/wpa_supplicant.conf"
156 cat >> "$BSDINSTALL_TMPETC/wpa_supplicant.conf" << EOF
157 ctrl_interface=/var/run/wpa_supplicant
165 # Try to reach wpa_supplicant. If it isn't running and we can modify the
166 # existing system, start it. Otherwise, fail.
168 if ! wpa_cli ping > /dev/null 2>&1; then
169 if [ ! "$BSDINSTALL_CONFIGCURRENT" ]; then
170 $DIALOG --backtitle "$DIALOG_BACKTITLE" --title "$msg_error" --msgbox \
171 "Wireless cannot be configured without making changes to the local system!" 0 0
174 wpa_supplicant -B -i $1 -c "$BSDINSTALL_TMPETC/wpa_supplicant.conf") || exit 1
176 if ! wpa_cli ping > /dev/null 2>&1 && [ ! "$BSDINSTALL_CONFIGCURRENT" ]; then
178 --title "$msg_error" \
179 --backtitle "$DIALOG_BACKTITLE" \
180 --msgbox "Could not start wpa_supplicant!" \
187 # There is no way to check country/regdomain without (possible)
188 # interface state modification
190 if [ "$BSDINSTALL_CONFIGCURRENT" ]; then
191 # Get current country/regdomain for selected interface
192 WLAN_IFACE=$( wpa_cli ifname | tail -1 )
193 INPUT=$( ifconfig "$WLAN_IFACE" list regdomain | head -1 )
194 DEF_REGDOMAIN=$( echo "$INPUT" | cut -w -f 2 )
195 DEF_COUNTRY=$( echo "$INPUT" | cut -w -f 4 )
196 [ "$DEF_REGDOMAIN" = 0 ] && DEF_REGDOMAIN="<not selected>"
197 [ "$DEF_COUNTRY" = 0 ] && DEF_COUNTRY="<not selected>"
198 f_dialog_title "Regdomain/country"
200 --title "$DIALOG_TITLE" \
201 --backtitle "$DIALOG_BACKTITLE" \
202 --yesno "Change regdomain/country (now $DEF_REGDOMAIN/$DEF_COUNTRY)?" \
204 if [ $? -eq 0 ]; then
206 dialog_country_select "$DEF_REGDOMAIN" "$DEF_COUNTRY"
207 if [ $? -eq $SUCCESS ]; then
216 output=$( wpa_cli scan 2>&1 )
217 f_dprintf "%s" "$output"
218 f_dialog_title "Scanning"
220 --title "$DIALOG_TITLE" \
221 --backtitle "$DIALOG_BACKTITLE" \
222 --ok-label "$msg_skip" \
223 --pause "Waiting 5 seconds to scan for wireless networks..." \
226 SCAN_RESULTS=$( wpa_cli scan_results )
227 NETWORKS=$( echo "$SCAN_RESULTS" | awk -F '\t' '
228 /..:..:..:..:..:../ && $5 { printf "\"%s\"\t%s\n", $5, $4 }
231 if [ ! "$NETWORKS" ]; then
232 f_dialog_title "$msg_error"
234 --title "$DIALOG_TITLE" \
235 --backtitle "$DIALOG_BACKTITLE" \
236 --yesno "No wireless networks were found. Rescan?" \
241 f_dialog_title "Network Selection"
242 prompt="Select a wireless network to connect to."
243 menu_list=$( echo $NETWORKS | tr '\n' ' ' )
244 NETWORK=$( eval $DIALOG \
245 --title \"\$DIALOG_TITLE\" \
246 --backtitle \"\$DIALOG_BACKTITLE\" \
248 --extra-label \"Rescan\" \
249 --menu \"\$prompt\" \
250 $height $width $rows \
252 2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
255 f_dialog_data_sanitize NETWORK
259 # here we ask if the user wants to select the network manually
260 f_dialog_title "Network Selection"
261 f_yesno "Do you want to select the network manually?" || exit 1
262 f_dialog_input NETWORK "Enter SSID" || exit 1
263 prompt="Select encryption type"
270 eval f_dialog_menu_size height width rows \"\$DIALOG_TITLE\" \
271 \"\$DIALOG_BACKTITLE\" \"\$prompt\" \"\" $menu_list
272 ENCRYPTION=$( eval $DIALOG \
273 --title \"\$DIALOG_TITLE\" \
274 --backtitle \"\$DIALOG_BACKTITLE\" \
275 --menu \"\$prompt\" \
276 $height $width $rows \
278 2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
283 $DIALOG_EXTRA) # Rescan
288 [ "$ENCRYPTION" ] || ENCRYPTION=$( echo "$NETWORKS" |
289 awk -F '\t' "/^\"$NETWORK\"\t/ { print \$2 }" )
291 if echo "$ENCRYPTION" | grep -q PSK; then
293 --title "WPA Setup" \
294 --backtitle "$DIALOG_BACKTITLE" \
298 "SSID" 1 0 "$NETWORK" 1 12 0 0 2 \
299 "Password" 2 0 "" 2 12 15 63 1 \
300 2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
302 awk 'sub(/^\t/,"")||1' \
303 >> "$BSDINSTALL_TMPETC/wpa_supplicant.conf" <<-EOF
311 elif echo "$ENCRYPTION" | grep -q EAP; then
312 USERPASS=$( $DIALOG \
313 --title "WPA-Enterprise Setup" \
314 --backtitle "$DIALOG_BACKTITLE" \
318 "SSID" 1 0 "$NETWORK" 1 12 0 0 2 \
319 "Username" 2 0 "" 2 12 25 63 0 \
320 "Password" 3 0 "" 3 12 25 63 1 \
321 2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
323 awk 'sub(/^\t/,"")||1' \
324 >> "$BSDINSTALL_TMPETC/wpa_supplicant.conf" <<-EOF
329 echo "$USERPASS" | awk '
330 NR == 1 { printf "\n\t\tidentity=\"%s\"", $1 }
331 NR == 2 { printf "\n\t\tpassword=\"%s\"", $1 }
336 elif echo "$ENCRYPTION" | grep -q WEP; then
338 --title "WEP Setup" \
339 --backtitle "$DIALOG_BACKTITLE" \
343 "SSID" 1 0 "$NETWORK" 1 12 0 0 2 \
344 "WEP Key 0" 2 0 "" 2 12 15 0 1 \
345 2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
347 awk 'sub(/^\t/,"")||1' \
348 >> "$BSDINSTALL_TMPETC/wpa_supplicant.conf" <<-EOF
359 awk 'sub(/^\t/,"")||1' \
360 >> "$BSDINSTALL_TMPETC/wpa_supplicant.conf" <<-EOF
370 # Connect to any open networks policy
371 cat >> "$BSDINSTALL_TMPETC/wpa_supplicant.conf" << EOF
378 # Bring up new network
379 if [ "$BSDINSTALL_CONFIGCURRENT" ]; then
380 output=$( wpa_cli reconfigure 2>&1 )
381 f_dprintf "%s" "$output"
386 ################################################################################
388 ################################################################################