freedombone-utils-wifi 17KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534
  1. #!/bin/bash
  2. #
  3. # .---. . .
  4. # | | |
  5. # |--- .--. .-. .-. .-.| .-. .--.--. |.-. .-. .--. .-.
  6. # | | (.-' (.-' ( | ( )| | | | )( )| | (.-'
  7. # ' ' --' --' -' - -' ' ' -' -' -' ' - --'
  8. #
  9. # Freedom in the Cloud
  10. #
  11. # Wifi functions
  12. # License
  13. # =======
  14. #
  15. # Copyright (C) 2015-2016 Bob Mottram <bob@robotics.uk.to>
  16. #
  17. # This program is free software: you can redistribute it and/or modify
  18. # it under the terms of the GNU Affero General Public License as published by
  19. # the Free Software Foundation, either version 3 of the License, or
  20. # (at your option) any later version.
  21. #
  22. # This program is distributed in the hope that it will be useful,
  23. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  24. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  25. # GNU Affero General Public License for more details.
  26. #
  27. # You should have received a copy of the GNU Affero General Public License
  28. # along with this program. If not, see <http://www.gnu.org/licenses/>.
  29. WIFI_CHANNEL=2
  30. WIFI_INTERFACE=wlan0
  31. WIFI_TYPE='wpa2-psk'
  32. WIFI_SSID=
  33. WIFI_PASSPHRASE=
  34. WIFI_HOTSPOT='no'
  35. WIFI_NETWORKS_FILE=~/${PROJECT_NAME}-wifi.cfg
  36. # repo for atheros AR9271 wifi driver
  37. ATHEROS_WIFI_REPO="https://github.com/qca/open-ath9k-htc-firmware.git"
  38. function setup_wifi_atheros {
  39. if [[ $(running_as_root) == "0" ]]; then
  40. return
  41. fi
  42. atheros_drivers_file=drivers/ath9k_htc_driver_bbb.tar.gz
  43. if [ ! -f $atheros_drivers_file ]; then
  44. if [ ! -f ~/freedombone/$atheros_drivers_file ]; then
  45. return
  46. else
  47. atheros_drivers_file=~/freedombone/$atheros_drivers_file
  48. fi
  49. else
  50. atheros_drivers_file=$(pwd)/$atheros_drivers_file
  51. fi
  52. if [ ! -d /lib/firmware ]; then
  53. mkdir /lib/firmware
  54. if [ ! -d /lib/firmware ]; then
  55. return
  56. fi
  57. fi
  58. if [ -f /lib/firmware/htc_9271.fw ]; then
  59. return
  60. fi
  61. curr_dir=$(pwd)
  62. cd /lib/firmware
  63. tar -xzvf $atheros_drivers_file
  64. reset_usb_devices
  65. cd $curr_dir
  66. update_wifi_adaptors
  67. if [ $IFACE ]; then
  68. wpa_action ${IFACE} stop
  69. wpa_cli -i ${IFACE} terminate
  70. ifconfig $IFACE up
  71. fi
  72. }
  73. function setup_wifi {
  74. if [[ $SYSTEM_TYPE == "mesh"* ]]; then
  75. return
  76. fi
  77. if [ ! $WIFI_SSID ]; then
  78. return
  79. fi
  80. if [ ${#WIFI_SSID} -lt 2 ]; then
  81. return
  82. fi
  83. if [[ $(is_completed $FUNCNAME) == "1" ]]; then
  84. return
  85. fi
  86. HOTSPOT='no'
  87. if [[ $WIFI_HOTSPOT != 'no' ]]; then
  88. HOTSPOT='yes'
  89. fi
  90. if [ -f $WIFI_NETWORKS_FILE ]; then
  91. ${PROJECT_NAME}-wifi --networks $WIFI_NETWORKS_FILE
  92. mark_completed $FUNCNAME
  93. return
  94. fi
  95. if [[ $WIFI_TYPE != 'none' ]]; then
  96. if [ ! $WIFI_PASSPHRASE ]; then
  97. echo $'No wifi passphrase was given'
  98. return
  99. fi
  100. if [ ${#WIFI_PASSPHRASE} -lt 2 ]; then
  101. echo $'Wifi passphrase was too short'
  102. return
  103. fi
  104. ${PROJECT_NAME}-wifi -s $WIFI_SSID -t $WIFI_TYPE -p $WIFI_PASSPHRASE --hotspot $HOTSPOT --networks $WIFI_NETWORKS_FILE
  105. else
  106. ${PROJECT_NAME}-wifi -s $WIFI_SSID -t $WIFI_TYPE --hotspot $HOTSPOT --networks $WIFI_NETWORKS_FILE
  107. fi
  108. mark_completed $FUNCNAME
  109. }
  110. # ath9k_htc driver
  111. function install_atheros_wifi {
  112. if [[ $(is_completed $FUNCNAME) == "1" ]]; then
  113. return
  114. fi
  115. if [ $INSTALLING_ON_BBB != "yes" ]; then
  116. return
  117. fi
  118. if [[ $ENABLE_BATMAN != "yes" ]]; then
  119. return
  120. fi
  121. if [ -d $INSTALL_DIR/open-ath9k-htc-firmware ]; then
  122. return
  123. fi
  124. # have drivers already been installed ?
  125. if [ -f /lib/firmware/htc_9271.fw ]; then
  126. return
  127. fi
  128. apt-get -y install build-essential cmake git m4 texinfo
  129. if [ ! -d $INSTALL_DIR ]; then
  130. mkdir -p $INSTALL_DIR
  131. fi
  132. cd $INSTALL_DIR
  133. if [ ! -d $INSTALL_DIR/open-ath9k-htc-firmware ]; then
  134. function_check git_clone
  135. git_clone $ATHEROS_WIFI_REPO $INSTALL_DIR/open-ath9k-htc-firmware
  136. if [ ! "$?" = "0" ]; then
  137. rm -rf $INSTALL_DIR/open-ath9k-htc-firmware
  138. exit 74283
  139. fi
  140. fi
  141. cd $INSTALL_DIR/open-ath9k-htc-firmware
  142. git checkout 1.4.0
  143. make toolchain
  144. if [ ! "$?" = "0" ]; then
  145. rm -rf $INSTALL_DIR/open-ath9k-htc-firmware
  146. exit 24820
  147. fi
  148. make firmware
  149. if [ ! "$?" = "0" ]; then
  150. rm -rf $INSTALL_DIR/open-ath9k-htc-firmware
  151. exit 63412
  152. fi
  153. cp target_firmware/*.fw /lib/firmware/
  154. if [ ! "$?" = "0" ]; then
  155. exit 74681
  156. fi
  157. mark_completed $FUNCNAME
  158. }
  159. function update_wifi_adaptors {
  160. IFACE=
  161. IFACE_SECONDARY=
  162. for i in $(seq 10 -1 0); do
  163. if grep -q "wlan${i}" /proc/net/dev; then
  164. if [ ! $IFACE ]; then
  165. IFACE="wlan${i}"
  166. else
  167. IFACE_SECONDARY="wlan${i}"
  168. return
  169. fi
  170. fi
  171. done
  172. }
  173. function wifi_get_psk {
  174. ssid=$1
  175. passphrase=$2
  176. psk=$(wpa_passphrase "$ssid" "$passphrase" | grep 'psk=' | sed -n 2p | awk -F '=' '{print $2}')
  177. echo $psk
  178. }
  179. function hotspot_off {
  180. if [ ! -f /etc/hostapd/hostapd.conf ]; then
  181. return
  182. fi
  183. systemctl stop hostapd
  184. rm /etc/hostapd/hostapd.conf
  185. if [ -f /etc/network/interfaces_original ]; then
  186. cp /etc/network/interfaces_original /etc/network/interfaces
  187. else
  188. echo '# interfaces(5) file used by ifup(8) and ifdown(8)' > /etc/network/interfaces
  189. echo '# Include files from /etc/network/interfaces.d:' >> /etc/network/interfaces
  190. echo 'source-directory /etc/network/interfaces.d' >> /etc/network/interfaces
  191. fi
  192. wpa_action ${WIFI_INTERFACE} stop
  193. wpa_cli -i ${WIFI_INTERFACE} terminate
  194. systemctl restart network-manager
  195. }
  196. function hotspot_on {
  197. if [ ! -f /etc/default/hostapd ]; then
  198. echo $'/etc/default/hostapd was not found'
  199. exit 67241
  200. fi
  201. if [ ${#WIFI_PASSPHRASE} -lt 8 ]; then
  202. echo $'Wifi hotspot passphrase is too short'
  203. exit 25719
  204. fi
  205. sed -i 's|#DAEMON_CONF=.*|DAEMON_CONF="/etc/hostapd/hostapd.conf"|g' /etc/default/hostapd
  206. echo '### Wireless network name ###' > /etc/hostapd/hostapd.conf
  207. echo "interface=$WIFI_INTERFACE" >> /etc/hostapd/hostapd.conf
  208. echo '' >> /etc/hostapd/hostapd.conf
  209. echo '### Set your bridge name ###' >> /etc/hostapd/hostapd.conf
  210. echo 'bridge=br0' >> /etc/hostapd/hostapd.conf
  211. echo '' >> /etc/hostapd/hostapd.conf
  212. echo 'driver=nl80211' >> /etc/hostapd/hostapd.conf
  213. echo "country_code=UK" >> /etc/hostapd/hostapd.conf
  214. echo "ssid=$WIFI_SSID" >> /etc/hostapd/hostapd.conf
  215. echo 'hw_mode=g' >> /etc/hostapd/hostapd.conf
  216. echo 'channel=6' >> /etc/hostapd/hostapd.conf
  217. echo 'wpa=2' >> /etc/hostapd/hostapd.conf
  218. echo "wpa_passphrase=$WIFI_PASSPHRASE" >> /etc/hostapd/hostapd.conf
  219. echo '' >> /etc/hostapd/hostapd.conf
  220. echo '## Key management algorithms ##' >> /etc/hostapd/hostapd.conf
  221. echo 'wpa_key_mgmt=WPA-PSK' >> /etc/hostapd/hostapd.conf
  222. echo '' >> /etc/hostapd/hostapd.conf
  223. echo '## Set cipher suites (encryption algorithms) ##' >> /etc/hostapd/hostapd.conf
  224. echo '## TKIP = Temporal Key Integrity Protocol' >> /etc/hostapd/hostapd.conf
  225. echo '## CCMP = AES in Counter mode with CBC-MAC' >> /etc/hostapd/hostapd.conf
  226. echo 'wpa_pairwise=TKIP' >> /etc/hostapd/hostapd.conf
  227. echo 'rsn_pairwise=CCMP' >> /etc/hostapd/hostapd.conf
  228. echo '' >> /etc/hostapd/hostapd.conf
  229. echo '## Shared Key Authentication ##'
  230. echo 'auth_algs=1' >> /etc/hostapd/hostapd.conf
  231. echo '' >> /etc/hostapd/hostapd.conf
  232. echo '## Accept all MAC address ###' >> /etc/hostapd/hostapd.conf
  233. echo 'macaddr_acl=0' >> /etc/hostapd/hostapd.conf
  234. if [ ! -f /etc/network/interfaces_original ]; then
  235. if ! grep -q "# wifi enabled" /etc/network/interfaces; then
  236. cp /etc/network/interfaces /etc/network/interfaces_original
  237. fi
  238. fi
  239. echo '# wifi enabled' > /etc/network/interfaces
  240. echo 'auto lo br0' >> /etc/network/interfaces
  241. echo 'iface lo inet loopback' >> /etc/network/interfaces
  242. echo '' >> /etc/network/interfaces
  243. echo "# wireless $WIFI_INTERFACE" >> /etc/network/interfaces
  244. echo "allow-hotplug $WIFI_INTERFACE" >> /etc/network/interfaces
  245. echo "iface $WIFI_INTERFACE inet manual" >> /etc/network/interfaces
  246. echo '' >> /etc/network/interfaces
  247. echo '# eth0 connected to the ISP router' >> /etc/network/interfaces
  248. echo 'allow-hotplug eth0' >> /etc/network/interfaces
  249. echo 'iface eth0 inet manual' >> /etc/network/interfaces
  250. echo '' >> /etc/network/interfaces
  251. echo '# Setup bridge' >> /etc/network/interfaces
  252. echo 'iface br0 inet static' >> /etc/network/interfaces
  253. echo " bridge_ports $WIFI_INTERFACE eth0" >> /etc/network/interfaces
  254. systemctl restart network-manager
  255. ifconfig $WIFI_INTERFACE up
  256. systemctl restart hostapd
  257. }
  258. function wifi_wpa2_psk {
  259. ssid=$1
  260. passphrase=$2
  261. if [ ! -f /etc/network/interfaces_original ]; then
  262. if ! grep -q "# wifi enabled" /etc/network/interfaces; then
  263. cp /etc/network/interfaces /etc/network/interfaces_original
  264. fi
  265. fi
  266. echo '# wifi enabled' > /etc/network/interfaces
  267. echo 'auto lo' >> /etc/network/interfaces
  268. echo 'iface lo inet loopback' >> /etc/network/interfaces
  269. echo '' >> /etc/network/interfaces
  270. echo 'allow-hotplug eth0' >> /etc/network/interfaces
  271. echo 'iface eth0 inet dhcp' >> /etc/network/interfaces
  272. echo '' >> /etc/network/interfaces
  273. echo "allow-hotplug ${WIFI_INTERFACE}" >> /etc/network/interfaces
  274. echo "iface ${WIFI_INTERFACE} inet manual" >> /etc/network/interfaces
  275. echo " wpa-roam $WIFI_CONFIG" >> /etc/network/interfaces
  276. wpa_passphrase "$ssid" "$passphrase" > $WIFI_CONFIG
  277. systemctl restart network-manager
  278. ifconfig ${WIFI_INTERFACE} up
  279. }
  280. function wifi_none {
  281. ssid=$1
  282. if [ ! -f /etc/network/interfaces_original ]; then
  283. if ! grep -q "# wifi enabled" /etc/network/interfaces; then
  284. cp /etc/network/interfaces /etc/network/interfaces_original
  285. fi
  286. fi
  287. echo '# wifi enabled' > /etc/network/interfaces
  288. echo 'auto lo' >> /etc/network/interfaces
  289. echo 'iface lo inet loopback' >> /etc/network/interfaces
  290. echo '' >> /etc/network/interfaces
  291. echo 'allow-hotplug eth0' >> /etc/network/interfaces
  292. echo 'iface eth0 inet dhcp' >> /etc/network/interfaces
  293. echo '' >> /etc/network/interfaces
  294. echo "allow-hotplug ${WIFI_INTERFACE}" >> /etc/network/interfaces
  295. echo "iface ${WIFI_INTERFACE} inet manual" >> /etc/network/interfaces
  296. echo " wpa-roam $WIFI_CONFIG" >> /etc/network/interfaces
  297. echo 'update_config=1' > $WIFI_CONFIG
  298. echo 'eapol_version=1' >> $WIFI_CONFIG
  299. echo '' >> $WIFI_CONFIG
  300. echo 'network={' >> $WIFI_CONFIG
  301. if [[ "${ssid}" != $'any' && "${ssid}" != $'all' && "${ssid}" != $'open' ]]; then
  302. echo " ssid=\"${ssid}\"" >> $WIFI_CONFIG
  303. fi
  304. echo ' key_mgmt=NONE' >> $WIFI_CONFIG
  305. echo '}' >> $WIFI_CONFIG
  306. systemctl restart network-manager
  307. ifconfig ${WIFI_INTERFACE} up
  308. }
  309. function networks_from_file {
  310. if [ ! -f $WIFI_NETWORKS_FILE ]; then
  311. exit 4
  312. fi
  313. if [[ $(config_param_exists "WIFI_INTERFACE") == "0" ]]; then
  314. exit 5
  315. fi
  316. read_config_param "WIFI_INTERFACE"
  317. if [ ! -f /etc/network/interfaces_original ]; then
  318. if ! grep -q "# wifi enabled" /etc/network/interfaces; then
  319. cp /etc/network/interfaces /etc/network/interfaces_original
  320. fi
  321. fi
  322. echo '# wifi enabled' > /etc/network/interfaces
  323. echo 'auto lo' >> /etc/network/interfaces
  324. echo 'iface lo inet loopback' >> /etc/network/interfaces
  325. echo '' >> /etc/network/interfaces
  326. echo 'allow-hotplug eth0' >> /etc/network/interfaces
  327. echo 'iface eth0 inet dhcp' >> /etc/network/interfaces
  328. echo '' >> /etc/network/interfaces
  329. echo "allow-hotplug ${WIFI_INTERFACE}" >> /etc/network/interfaces
  330. echo "iface ${WIFI_INTERFACE} inet manual" >> /etc/network/interfaces
  331. echo " wpa-roam $WIFI_CONFIG" >> /etc/network/interfaces
  332. # remove wpa_supplicant.conf if it exists
  333. if [ -f $WIFI_CONFIG ]; then
  334. rm -f $WIFI_CONFIG
  335. fi
  336. echo 'update_config=1' > $WIFI_CONFIG
  337. echo 'eapol_version=1' >> $WIFI_CONFIG
  338. echo '' >> $WIFI_CONFIG
  339. ctr=0
  340. while read -r line
  341. do
  342. if [ ${#line} -gt 1 ]; then
  343. if [[ "$line" != '#'* ]]; then
  344. if [ $ctr -eq 0 ]; then
  345. WIFI_SSID="$line"
  346. fi
  347. if [ $ctr -eq 1 ]; then
  348. WIFI_TYPE="$line"
  349. if [[ $WIFI_TYPE == $'none' || $WIFI_TYPE == $'open' ]]; then
  350. echo 'network={' >> $WIFI_CONFIG
  351. if [[ "${WIFI_SSID}" != $'any' && "${WIFI_SSID}" != $'all' && "${WIFI_SSID}" != $'open' ]]; then
  352. echo " ssid=\"${WIFI_SSID}\"" >> $WIFI_CONFIG
  353. fi
  354. echo ' key_mgmt=NONE' >> $WIFI_CONFIG
  355. echo '}' >> $WIFI_CONFIG
  356. ctr=0
  357. continue
  358. fi
  359. fi
  360. if [ $ctr -eq 2 ]; then
  361. WIFI_PASSPHRASE="$line"
  362. wpa_passphrase "$WIFI_SSID" "$WIFI_PASSPHRASE" >> $WIFI_CONFIG
  363. ctr=0
  364. continue
  365. fi
  366. ctr=$((ctr + 1))
  367. fi
  368. fi
  369. done < $WIFI_NETWORKS_FILE
  370. chattr -i /etc/resolv.conf
  371. systemctl restart network-manager
  372. ifconfig ${WIFI_INTERFACE} up
  373. }
  374. function create_networks_interactive {
  375. remove_config_param "WIFI_INTERFACE"
  376. update_wifi_adaptors
  377. if [ ! $IFACE ]; then
  378. # Don't try to configure wifi if there are no adaptors
  379. return
  380. fi
  381. if [ -f $WIFI_NETWORKS_FILE ]; then
  382. rm $WIFI_NETWORKS_FILE
  383. fi
  384. echo $'# Add wifi networks as follows:' > $WIFI_NETWORKS_FILE
  385. echo '#' >> $WIFI_NETWORKS_FILE
  386. echo $'# MySSID' >> $WIFI_NETWORKS_FILE
  387. echo $'# wpa2-psk' >> $WIFI_NETWORKS_FILE
  388. echo $'# myWifiPassphrase' >> $WIFI_NETWORKS_FILE
  389. echo '#' >> $WIFI_NETWORKS_FILE
  390. echo $'# AnotherSSID' >> $WIFI_NETWORKS_FILE
  391. echo $'# none' >> $WIFI_NETWORKS_FILE
  392. echo '#' >> $WIFI_NETWORKS_FILE
  393. # By default connect to any open wifi
  394. WIFI_SSID=$'any'
  395. WIFI_TYPE=$'open'
  396. wifi_ctr=0
  397. wifi_networks_done=
  398. while [ ! $wifi_networks_done ]
  399. do
  400. data=$(tempfile 2>/dev/null)
  401. trap "rm -f $data" 0 1 2 5 15
  402. dialog --backtitle $"Freedombone Configuration" \
  403. --title $"Wifi Settings ${wifi_ctr}" \
  404. --form $"\nTo use this system via wifi (eg. USB dongle) enter the details below, otherwise just select Ok:" 13 65 4 \
  405. $"SSID:" 1 1 "$WIFI_SSID" 1 24 30 30 \
  406. $"Type (open/wpa2-psk):" 2 1 "$WIFI_TYPE" 2 24 10 10 \
  407. $"Passphrase:" 3 1 "$WIFI_PASSPHRASE" 3 24 50 50 \
  408. 2> $data
  409. sel=$?
  410. case $sel in
  411. 1) return;;
  412. 255) return;;
  413. esac
  414. WIFI_SSID=$(cat $data | sed -n 1p)
  415. WIFI_TYPE=$(cat $data | sed -n 2p)
  416. WIFI_PASSPHRASE=$(cat $data | sed -n 3p)
  417. # if these fields are empty then there are no more wifi networks
  418. if [ ${#WIFI_SSID} -lt 2 ]; then
  419. wifi_networks_done='yes'
  420. continue
  421. fi
  422. if [ ${#WIFI_TYPE} -lt 2 ]; then
  423. wifi_networks_done='yes'
  424. continue
  425. fi
  426. # update the wifi networks file
  427. echo '' >> $WIFI_NETWORKS_FILE
  428. echo "$WIFI_SSID" >> $WIFI_NETWORKS_FILE
  429. echo "$WIFI_TYPE" >> $WIFI_NETWORKS_FILE
  430. if [ ${#WIFI_PASSPHRASE} -gt 1 ]; then
  431. echo "$WIFI_PASSPHRASE" >> $WIFI_NETWORKS_FILE
  432. fi
  433. if [ ${#WIFI_SSID} -gt 1 ]; then
  434. if [ ${#WIFI_TYPE} -gt 1 ]; then
  435. if [[ "${WIFI_TYPE}" == $'none' || "${WIFI_TYPE}" == $'open' ]]; then
  436. write_config_param "WIFI_INTERFACE" "$WIFI_INTERFACE"
  437. return
  438. else
  439. if [ ${#WIFI_PASSPHRASE} -gt 1 ]; then
  440. write_config_param "WIFI_INTERFACE" "$WIFI_INTERFACE"
  441. return
  442. fi
  443. fi
  444. fi
  445. fi
  446. # clear values
  447. WIFI_SSID=
  448. WIFI_PASSPHRASE=
  449. wifi_ctr=$((wifi_ctr + 1))
  450. done
  451. }
  452. function disable_wifi {
  453. if [[ ${1} == 'yes' || ${1} == 'y' ]]; then
  454. hotspot_off
  455. echo '# interfaces(5) file used by ifup(8) and ifdown(8)' > /etc/network/interfaces
  456. echo '# Include files from /etc/network/interfaces.d:' >> /etc/network/interfaces
  457. echo 'source-directory /etc/network/interfaces.d' >> /etc/network/interfaces
  458. systemctl restart network-manager
  459. wpa_action ${WIFI_INTERFACE} stop
  460. wpa_cli -i ${WIFI_INTERFACE} terminate
  461. else
  462. networks_from_file
  463. fi
  464. }
  465. function count_wlan {
  466. # counts the number of wlan devices
  467. ctr=0
  468. for i in $(seq 0 1 10); do
  469. if grep -q "wlan${i}" /proc/net/dev; then
  470. ctr=$((ctr + 1))
  471. fi
  472. done
  473. echo $ctr
  474. }
  475. # NOTE: deliberately no exit 0