|
@@ -0,0 +1,439 @@
|
|
|
+NETIFD_MAIN_DIR="${NETIFD_MAIN_DIR:-/lib/netifd}"
|
|
|
+
|
|
|
+. /usr/share/libubox/jshn.sh
|
|
|
+. $NETIFD_MAIN_DIR/utils.sh
|
|
|
+
|
|
|
+CMD_UP=0
|
|
|
+CMD_SET_DATA=1
|
|
|
+CMD_PROCESS_ADD=2
|
|
|
+CMD_PROCESS_KILL_ALL=3
|
|
|
+CMD_SET_RETRY=4
|
|
|
+
|
|
|
+add_driver() {
|
|
|
+ return
|
|
|
+}
|
|
|
+
|
|
|
+wireless_setup_vif_failed() {
|
|
|
+ local error="$1"
|
|
|
+ echo "Interface $_w_iface setup failed: $error"
|
|
|
+}
|
|
|
+
|
|
|
+wireless_setup_failed() {
|
|
|
+ local error="$1"
|
|
|
+
|
|
|
+ echo "Device setup failed: $error"
|
|
|
+ wireless_set_retry 0
|
|
|
+}
|
|
|
+
|
|
|
+prepare_key_wep() {
|
|
|
+ local key="$1"
|
|
|
+ local hex=1
|
|
|
+
|
|
|
+ echo -n "$key" | grep -qE "[^a-fA-F0-9]" && hex=0
|
|
|
+ [ "${#key}" -eq 10 -a $hex -eq 1 ] || \
|
|
|
+ [ "${#key}" -eq 26 -a $hex -eq 1 ] || {
|
|
|
+ [ "${key:0:2}" = "s:" ] && key="${key#s:}"
|
|
|
+ key="$(echo -n "$key" | hexdump -ve '1/1 "%02x" ""')"
|
|
|
+ }
|
|
|
+ echo "$key"
|
|
|
+}
|
|
|
+
|
|
|
+_wdev_prepare_channel() {
|
|
|
+ json_get_vars channel band hwmode
|
|
|
+
|
|
|
+ auto_channel=0
|
|
|
+ enable_ht=0
|
|
|
+ htmode=
|
|
|
+ hwmode="${hwmode##11}"
|
|
|
+
|
|
|
+ case "$channel" in
|
|
|
+ ""|0|auto)
|
|
|
+ channel=0
|
|
|
+ auto_channel=1
|
|
|
+ ;;
|
|
|
+ [0-9]*) ;;
|
|
|
+ *)
|
|
|
+ wireless_setup_failed "INVALID_CHANNEL"
|
|
|
+ ;;
|
|
|
+ esac
|
|
|
+
|
|
|
+ case "$hwmode" in
|
|
|
+ a|b|g|ad) ;;
|
|
|
+ *)
|
|
|
+ if [ "$channel" -gt 14 ]; then
|
|
|
+ hwmode=a
|
|
|
+ else
|
|
|
+ hwmode=g
|
|
|
+ fi
|
|
|
+ ;;
|
|
|
+ esac
|
|
|
+
|
|
|
+ case "$band" in
|
|
|
+ 2g) hwmode=g;;
|
|
|
+ 5g|6g) hwmode=a;;
|
|
|
+ 60g) hwmode=ad;;
|
|
|
+ *)
|
|
|
+ case "$hwmode" in
|
|
|
+ *a) band=5g;;
|
|
|
+ *ad) band=60g;;
|
|
|
+ *b|*g) band=2g;;
|
|
|
+ esac
|
|
|
+ ;;
|
|
|
+ esac
|
|
|
+}
|
|
|
+
|
|
|
+_wdev_handler() {
|
|
|
+ json_load "$data"
|
|
|
+
|
|
|
+ json_select config
|
|
|
+ _wdev_prepare_channel
|
|
|
+ json_select ..
|
|
|
+
|
|
|
+ eval "drv_$1_$2 \"$interface\""
|
|
|
+}
|
|
|
+
|
|
|
+_wdev_msg_call() {
|
|
|
+ local old_cb
|
|
|
+
|
|
|
+ json_set_namespace wdev old_cb
|
|
|
+ "$@"
|
|
|
+ json_set_namespace $old_cb
|
|
|
+}
|
|
|
+
|
|
|
+_wdev_wrapper() {
|
|
|
+ while [ -n "$1" ]; do
|
|
|
+ eval "$1() { _wdev_msg_call _$1 \"\$@\"; }"
|
|
|
+ shift
|
|
|
+ done
|
|
|
+}
|
|
|
+
|
|
|
+_wdev_notify_init() {
|
|
|
+ local command="$1"; shift;
|
|
|
+
|
|
|
+ json_init
|
|
|
+ json_add_int "command" "$command"
|
|
|
+ json_add_string "device" "$__netifd_device"
|
|
|
+ while [ -n "$1" ]; do
|
|
|
+ local name="$1"; shift
|
|
|
+ local value="$1"; shift
|
|
|
+ json_add_string "$name" "$value"
|
|
|
+ done
|
|
|
+ json_add_object "data"
|
|
|
+}
|
|
|
+
|
|
|
+_wdev_notify() {
|
|
|
+ local options="$1"
|
|
|
+
|
|
|
+ json_close_object
|
|
|
+ ubus $options call network.wireless notify "$(json_dump)"
|
|
|
+}
|
|
|
+
|
|
|
+_wdev_add_variables() {
|
|
|
+ while [ -n "$1" ]; do
|
|
|
+ local var="${1%%=*}"
|
|
|
+ local val="$1"
|
|
|
+ shift
|
|
|
+ [[ "$var" = "$val" ]] && continue
|
|
|
+ val="${val#*=}"
|
|
|
+ json_add_string "$var" "$val"
|
|
|
+ done
|
|
|
+}
|
|
|
+
|
|
|
+_wireless_add_vif() {
|
|
|
+ local name="$1"; shift
|
|
|
+ local ifname="$1"; shift
|
|
|
+
|
|
|
+ _wdev_notify_init $CMD_SET_DATA "interface" "$name"
|
|
|
+ json_add_string "ifname" "$ifname"
|
|
|
+ _wdev_add_variables "$@"
|
|
|
+ _wdev_notify
|
|
|
+}
|
|
|
+
|
|
|
+_wireless_add_vlan() {
|
|
|
+ local name="$1"; shift
|
|
|
+ local ifname="$1"; shift
|
|
|
+
|
|
|
+ _wdev_notify_init $CMD_SET_DATA interface "$__cur_interface" "vlan" "$name"
|
|
|
+ json_add_string "ifname" "$ifname"
|
|
|
+ _wdev_add_variables "$@"
|
|
|
+ _wdev_notify
|
|
|
+}
|
|
|
+
|
|
|
+_wireless_set_up() {
|
|
|
+ _wdev_notify_init $CMD_UP
|
|
|
+ _wdev_notify
|
|
|
+}
|
|
|
+
|
|
|
+_wireless_set_data() {
|
|
|
+ _wdev_notify_init $CMD_SET_DATA
|
|
|
+ _wdev_add_variables "$@"
|
|
|
+ _wdev_notify
|
|
|
+}
|
|
|
+
|
|
|
+_wireless_add_process() {
|
|
|
+ _wdev_notify_init $CMD_PROCESS_ADD
|
|
|
+ local exe="$2"
|
|
|
+ [ -L "$exe" ] && exe="$(readlink -f "$exe")"
|
|
|
+ json_add_int pid "$1"
|
|
|
+ json_add_string exe "$exe"
|
|
|
+ [ -n "$3" ] && json_add_boolean required 1
|
|
|
+ [ -n "$4" ] && json_add_boolean keep 1
|
|
|
+ exe2="$(readlink -f /proc/$1/exe)"
|
|
|
+ [ "$exe" != "$exe2" ] && echo "WARNING (wireless_add_process): executable path $exe does not match process $1 path ($exe2)"
|
|
|
+ _wdev_notify
|
|
|
+}
|
|
|
+
|
|
|
+_wireless_process_kill_all() {
|
|
|
+ _wdev_notify_init $CMD_PROCESS_KILL_ALL
|
|
|
+ [ -n "$1" ] && json_add_int signal "$1"
|
|
|
+ _wdev_notify
|
|
|
+}
|
|
|
+
|
|
|
+_wireless_set_retry() {
|
|
|
+ _wdev_notify_init $CMD_SET_RETRY
|
|
|
+ json_add_int retry "$1"
|
|
|
+ _wdev_notify
|
|
|
+}
|
|
|
+
|
|
|
+_wdev_wrapper \
|
|
|
+ wireless_add_vif \
|
|
|
+ wireless_add_vlan \
|
|
|
+ wireless_set_up \
|
|
|
+ wireless_set_data \
|
|
|
+ wireless_add_process \
|
|
|
+ wireless_process_kill_all \
|
|
|
+ wireless_set_retry \
|
|
|
+
|
|
|
+wireless_vif_parse_encryption() {
|
|
|
+ json_get_vars encryption
|
|
|
+ set_default encryption none
|
|
|
+
|
|
|
+ auth_mode_open=1
|
|
|
+ auth_mode_shared=0
|
|
|
+ auth_type=none
|
|
|
+
|
|
|
+ if [ "$hwmode" = "ad" ]; then
|
|
|
+ wpa_cipher="GCMP"
|
|
|
+ else
|
|
|
+ wpa_cipher="CCMP"
|
|
|
+ fi
|
|
|
+
|
|
|
+ case "$encryption" in
|
|
|
+ *tkip+aes|*tkip+ccmp|*aes+tkip|*ccmp+tkip) wpa_cipher="CCMP TKIP";;
|
|
|
+ *ccmp256) wpa_cipher="CCMP-256";;
|
|
|
+ *aes|*ccmp) wpa_cipher="CCMP";;
|
|
|
+ *tkip) wpa_cipher="TKIP";;
|
|
|
+ *gcmp256) wpa_cipher="GCMP-256";;
|
|
|
+ *gcmp) wpa_cipher="GCMP";;
|
|
|
+ wpa3-192*) wpa_cipher="GCMP-256";;
|
|
|
+ esac
|
|
|
+
|
|
|
+ # 802.11n requires CCMP for WPA
|
|
|
+ [ "$enable_ht:$wpa_cipher" = "1:TKIP" ] && wpa_cipher="CCMP TKIP"
|
|
|
+
|
|
|
+ # Examples:
|
|
|
+ # psk-mixed/tkip => WPA1+2 PSK, TKIP
|
|
|
+ # wpa-psk2/tkip+aes => WPA2 PSK, CCMP+TKIP
|
|
|
+ # wpa2/tkip+aes => WPA2 RADIUS, CCMP+TKIP
|
|
|
+
|
|
|
+ case "$encryption" in
|
|
|
+ wpa2*|wpa3*|*psk2*|psk3*|sae*|owe*)
|
|
|
+ wpa=2
|
|
|
+ ;;
|
|
|
+ wpa*mixed*|*psk*mixed*)
|
|
|
+ wpa=3
|
|
|
+ ;;
|
|
|
+ wpa*|*psk*)
|
|
|
+ wpa=1
|
|
|
+ ;;
|
|
|
+ *)
|
|
|
+ wpa=0
|
|
|
+ wpa_cipher=
|
|
|
+ ;;
|
|
|
+ esac
|
|
|
+ wpa_pairwise="$wpa_cipher"
|
|
|
+
|
|
|
+ case "$encryption" in
|
|
|
+ owe*)
|
|
|
+ auth_type=owe
|
|
|
+ ;;
|
|
|
+ wpa3-192*)
|
|
|
+ auth_type=eap192
|
|
|
+ ;;
|
|
|
+ wpa3-mixed*)
|
|
|
+ auth_type=eap-eap2
|
|
|
+ ;;
|
|
|
+ wpa3*)
|
|
|
+ auth_type=eap2
|
|
|
+ ;;
|
|
|
+ psk3-mixed*|sae-mixed*)
|
|
|
+ auth_type=psk-sae
|
|
|
+ ;;
|
|
|
+ psk3*|sae*)
|
|
|
+ auth_type=sae
|
|
|
+ ;;
|
|
|
+ *psk*)
|
|
|
+ auth_type=psk
|
|
|
+ ;;
|
|
|
+ *wpa*|*8021x*)
|
|
|
+ auth_type=eap
|
|
|
+ ;;
|
|
|
+ *wep*)
|
|
|
+ auth_type=wep
|
|
|
+ case "$encryption" in
|
|
|
+ *shared*)
|
|
|
+ auth_mode_open=0
|
|
|
+ auth_mode_shared=1
|
|
|
+ ;;
|
|
|
+ *mixed*)
|
|
|
+ auth_mode_shared=1
|
|
|
+ ;;
|
|
|
+ esac
|
|
|
+ ;;
|
|
|
+ esac
|
|
|
+
|
|
|
+ case "$encryption" in
|
|
|
+ *osen*)
|
|
|
+ auth_osen=1
|
|
|
+ ;;
|
|
|
+ esac
|
|
|
+}
|
|
|
+
|
|
|
+_wireless_set_brsnoop_isolation() {
|
|
|
+ local multicast_to_unicast="$1"
|
|
|
+ local isolate
|
|
|
+
|
|
|
+ json_get_vars isolate proxy_arp
|
|
|
+
|
|
|
+ [ ${isolate:-0} -gt 0 -o -z "$network_bridge" ] && return
|
|
|
+ [ ${multicast_to_unicast:-1} -gt 0 -o ${proxy_arp:-0} -gt 0 ] && json_add_boolean isolate 1
|
|
|
+}
|
|
|
+
|
|
|
+for_each_interface() {
|
|
|
+ local _w_types="$1"; shift
|
|
|
+ local _w_ifaces _w_iface
|
|
|
+ local _w_type
|
|
|
+ local _w_found
|
|
|
+
|
|
|
+ local multicast_to_unicast
|
|
|
+
|
|
|
+ json_get_keys _w_ifaces interfaces
|
|
|
+ json_select interfaces
|
|
|
+ for _w_iface in $_w_ifaces; do
|
|
|
+ json_select "$_w_iface"
|
|
|
+ if [ -n "$_w_types" ]; then
|
|
|
+ json_get_var network_bridge bridge
|
|
|
+ json_get_var network_ifname bridge-ifname
|
|
|
+ json_get_var multicast_to_unicast multicast_to_unicast
|
|
|
+ json_select config
|
|
|
+ _wireless_set_brsnoop_isolation "$multicast_to_unicast"
|
|
|
+ json_get_var _w_type mode
|
|
|
+ json_select ..
|
|
|
+ _w_types=" $_w_types "
|
|
|
+ [[ "${_w_types%$_w_type*}" = "$_w_types" ]] && {
|
|
|
+ json_select ..
|
|
|
+ continue
|
|
|
+ }
|
|
|
+ fi
|
|
|
+ __cur_interface="$_w_iface"
|
|
|
+ "$@" "$_w_iface"
|
|
|
+ json_select ..
|
|
|
+ done
|
|
|
+ json_select ..
|
|
|
+}
|
|
|
+
|
|
|
+for_each_vlan() {
|
|
|
+ local _w_vlans _w_vlan
|
|
|
+
|
|
|
+ json_get_keys _w_vlans vlans
|
|
|
+ json_select vlans
|
|
|
+ for _w_vlan in $_w_vlans; do
|
|
|
+ json_select "$_w_vlan"
|
|
|
+ json_select config
|
|
|
+ "$@" "$_w_vlan"
|
|
|
+ json_select ..
|
|
|
+ json_select ..
|
|
|
+ done
|
|
|
+ json_select ..
|
|
|
+}
|
|
|
+
|
|
|
+for_each_station() {
|
|
|
+ local _w_stas _w_sta
|
|
|
+
|
|
|
+ json_get_keys _w_stas stas
|
|
|
+ json_select stas
|
|
|
+ for _w_sta in $_w_stas; do
|
|
|
+ json_select "$_w_sta"
|
|
|
+ json_select config
|
|
|
+ "$@" "$_w_sta"
|
|
|
+ json_select ..
|
|
|
+ json_select ..
|
|
|
+ done
|
|
|
+ json_select ..
|
|
|
+}
|
|
|
+
|
|
|
+_wdev_common_device_config() {
|
|
|
+ config_add_string channel hwmode band htmode noscan
|
|
|
+}
|
|
|
+
|
|
|
+_wdev_common_iface_config() {
|
|
|
+ config_add_string mode ssid encryption 'key:wpakey'
|
|
|
+ config_add_boolean bridge_isolate
|
|
|
+}
|
|
|
+
|
|
|
+_wdev_common_vlan_config() {
|
|
|
+ config_add_string name vid iface
|
|
|
+ config_add_boolean bridge_isolate
|
|
|
+}
|
|
|
+
|
|
|
+_wdev_common_station_config() {
|
|
|
+ config_add_string mac key vid iface
|
|
|
+}
|
|
|
+
|
|
|
+init_wireless_driver() {
|
|
|
+ name="$1"; shift
|
|
|
+ cmd="$1"; shift
|
|
|
+
|
|
|
+ case "$cmd" in
|
|
|
+ dump)
|
|
|
+ add_driver() {
|
|
|
+ eval "drv_$1_cleanup"
|
|
|
+
|
|
|
+ json_init
|
|
|
+ json_add_string name "$1"
|
|
|
+
|
|
|
+ json_add_array device
|
|
|
+ _wdev_common_device_config
|
|
|
+ eval "drv_$1_init_device_config"
|
|
|
+ json_close_array
|
|
|
+
|
|
|
+ json_add_array iface
|
|
|
+ _wdev_common_iface_config
|
|
|
+ eval "drv_$1_init_iface_config"
|
|
|
+ json_close_array
|
|
|
+
|
|
|
+ json_add_array vlan
|
|
|
+ _wdev_common_vlan_config
|
|
|
+ eval "drv_$1_init_vlan_config"
|
|
|
+ json_close_array
|
|
|
+
|
|
|
+ json_add_array station
|
|
|
+ _wdev_common_station_config
|
|
|
+ eval "drv_$1_init_station_config"
|
|
|
+ json_close_array
|
|
|
+
|
|
|
+ json_dump
|
|
|
+ }
|
|
|
+ ;;
|
|
|
+ setup|teardown)
|
|
|
+ interface="$1"; shift
|
|
|
+ data="$1"; shift
|
|
|
+ export __netifd_device="$interface"
|
|
|
+
|
|
|
+ add_driver() {
|
|
|
+ [[ "$name" == "$1" ]] || return 0
|
|
|
+ _wdev_handler "$1" "$cmd"
|
|
|
+ }
|
|
|
+ ;;
|
|
|
+ esac
|
|
|
+}
|