blob: 596ffa6ff560a983d455845e088b44c04ce76749 [file] [log] [blame]
/*
* Copyright (c) 2023 Nordic Semiconductor ASA
*
* SPDX-License-Identifier: Apache-2.0
*/
#include "supp_events.h"
#include "includes.h"
#include "common.h"
#include "common/ieee802_11_defs.h"
#include "wpa_supplicant_i.h"
#ifdef CONFIG_AP
#include "ap/sta_info.h"
#include "ap/ieee802_11.h"
#include "ap/hostapd.h"
#endif /* CONFIG_AP */
#include <zephyr/net/wifi_mgmt.h>
/* Re-defines MAC2STR with address of the element */
#define MACADDR2STR(a) &(a)[0], &(a)[1], &(a)[2], &(a)[3], &(a)[4], &(a)[5]
static const struct wpa_supp_event_info {
const char *event_str;
enum supplicant_event_num event;
} wpa_supp_event_info[] = {
{ "CTRL-EVENT-CONNECTED", SUPPLICANT_EVENT_CONNECTED },
{ "CTRL-EVENT-DISCONNECTED", SUPPLICANT_EVENT_DISCONNECTED },
{ "CTRL-EVENT-ASSOC-REJECT", SUPPLICANT_EVENT_ASSOC_REJECT },
{ "CTRL-EVENT-AUTH-REJECT", SUPPLICANT_EVENT_AUTH_REJECT },
{ "CTRL-EVENT-SSID-TEMP-DISABLED", SUPPLICANT_EVENT_SSID_TEMP_DISABLED },
{ "CTRL-EVENT-SSID-REENABLED", SUPPLICANT_EVENT_SSID_REENABLED },
{ "CTRL-EVENT-BSS-ADDED", SUPPLICANT_EVENT_BSS_ADDED },
{ "CTRL-EVENT-BSS-REMOVED", SUPPLICANT_EVENT_BSS_REMOVED },
{ "CTRL-EVENT-TERMINATING", SUPPLICANT_EVENT_TERMINATING },
{ "CTRL-EVENT-SCAN-STARTED", SUPPLICANT_EVENT_SCAN_STARTED },
{ "CTRL-EVENT-SCAN-RESULTS", SUPPLICANT_EVENT_SCAN_RESULTS },
{ "CTRL-EVENT-SCAN-FAILED", SUPPLICANT_EVENT_SCAN_FAILED },
{ "CTRL-EVENT-NETWORK-NOT-FOUND", SUPPLICANT_EVENT_NETWORK_NOT_FOUND },
{ "CTRL-EVENT-NETWORK-ADDED", SUPPLICANT_EVENT_NETWORK_ADDED },
{ "CTRL-EVENT-NETWORK-REMOVED", SUPPLICANT_EVENT_NETWORK_REMOVED },
{ "CTRL-EVENT-DSCP-POLICY", SUPPLICANT_EVENT_DSCP_POLICY },
{ "CTRL-EVENT-REGDOM-CHANGE", SUPPLICANT_EVENT_REGDOM_CHANGE },
};
static void copy_mac_addr(const unsigned int *src, uint8_t *dst)
{
int i;
for (i = 0; i < ETH_ALEN; i++) {
dst[i] = src[i];
}
}
static enum wifi_conn_status wpas_to_wifi_mgmt_conn_status(int status)
{
switch (status) {
case WLAN_STATUS_SUCCESS:
return WIFI_STATUS_CONN_SUCCESS;
case WLAN_REASON_4WAY_HANDSHAKE_TIMEOUT:
return WIFI_STATUS_CONN_WRONG_PASSWORD;
/* Handle non-supplicant errors */
case -ETIMEDOUT:
return WIFI_STATUS_CONN_TIMEOUT;
default:
return WIFI_STATUS_CONN_FAIL;
}
}
static enum wifi_disconn_reason wpas_to_wifi_mgmt_disconn_status(int status)
{
switch (status) {
case WIFI_REASON_DISCONN_SUCCESS:
return WIFI_REASON_DISCONN_SUCCESS;
case WLAN_REASON_DEAUTH_LEAVING:
return WIFI_REASON_DISCONN_AP_LEAVING;
case WLAN_REASON_DISASSOC_DUE_TO_INACTIVITY:
return WIFI_REASON_DISCONN_INACTIVITY;
case WLAN_REASON_UNSPECIFIED:
/* fall through */
default:
return WIFI_REASON_DISCONN_UNSPECIFIED;
}
}
static int supplicant_process_status(struct supplicant_int_event_data *event_data,
char *supplicant_status)
{
int ret = 1; /* For cases where parsing is not being done*/
int i;
union supplicant_event_data *data;
unsigned int tmp_mac_addr[ETH_ALEN];
unsigned int event_prefix_len;
char *event_no_prefix;
struct wpa_supp_event_info event_info;
data = (union supplicant_event_data *)event_data->data;
for (i = 0; i < ARRAY_SIZE(wpa_supp_event_info); i++) {
if (strncmp(supplicant_status, wpa_supp_event_info[i].event_str,
strlen(wpa_supp_event_info[i].event_str)) == 0) {
event_info = wpa_supp_event_info[i];
break;
}
}
if (i >= ARRAY_SIZE(wpa_supp_event_info)) {
/* This is not a bug but rather implementation gap (intentional or not) */
wpa_printf(MSG_DEBUG, "Event not supported: %s", supplicant_status);
return -ENOTSUP;
}
/* Skip the event prefix and a space */
event_prefix_len = strlen(event_info.event_str) + 1;
event_no_prefix = supplicant_status + event_prefix_len;
event_data->event = event_info.event;
switch (event_data->event) {
case SUPPLICANT_EVENT_CONNECTED:
ret = sscanf(event_no_prefix, "- Connection to"
MACSTR, MACADDR2STR(tmp_mac_addr));
event_data->data_len = sizeof(data->connected);
copy_mac_addr(tmp_mac_addr, data->connected.bssid);
break;
case SUPPLICANT_EVENT_DISCONNECTED:
ret = sscanf(event_no_prefix,
MACSTR" reason=%d", MACADDR2STR(tmp_mac_addr),
&data->disconnected.reason_code);
event_data->data_len = sizeof(data->disconnected);
copy_mac_addr(tmp_mac_addr, data->disconnected.bssid);
break;
case SUPPLICANT_EVENT_ASSOC_REJECT:
/* TODO */
break;
case SUPPLICANT_EVENT_AUTH_REJECT:
ret = sscanf(event_no_prefix, MACSTR
" auth_type=%u auth_transaction=%u status_code=%u",
MACADDR2STR(tmp_mac_addr),
&data->auth_reject.auth_type,
&data->auth_reject.auth_transaction,
&data->auth_reject.status_code);
event_data->data_len = sizeof(data->auth_reject);
copy_mac_addr(tmp_mac_addr, data->auth_reject.bssid);
break;
case SUPPLICANT_EVENT_SSID_TEMP_DISABLED:
ret = sscanf(event_no_prefix,
"id=%d ssid=%s auth_failures=%u duration=%d reason=%s",
&data->temp_disabled.id, data->temp_disabled.ssid,
&data->temp_disabled.auth_failures,
&data->temp_disabled.duration,
data->temp_disabled.reason_code);
event_data->data_len = sizeof(data->temp_disabled);
break;
case SUPPLICANT_EVENT_SSID_REENABLED:
ret = sscanf(event_no_prefix,
"id=%d ssid=%s", &data->reenabled.id,
data->reenabled.ssid);
event_data->data_len = sizeof(data->reenabled);
break;
case SUPPLICANT_EVENT_BSS_ADDED:
ret = sscanf(event_no_prefix, "%u "MACSTR,
&data->bss_added.id,
MACADDR2STR(tmp_mac_addr));
copy_mac_addr(tmp_mac_addr, data->bss_added.bssid);
event_data->data_len = sizeof(data->bss_added);
break;
case SUPPLICANT_EVENT_BSS_REMOVED:
ret = sscanf(event_no_prefix,
"%u "MACSTR,
&data->bss_removed.id,
MACADDR2STR(tmp_mac_addr));
event_data->data_len = sizeof(data->bss_removed);
copy_mac_addr(tmp_mac_addr, data->bss_removed.bssid);
break;
case SUPPLICANT_EVENT_TERMINATING:
case SUPPLICANT_EVENT_SCAN_STARTED:
case SUPPLICANT_EVENT_SCAN_RESULTS:
case SUPPLICANT_EVENT_SCAN_FAILED:
case SUPPLICANT_EVENT_NETWORK_NOT_FOUND:
case SUPPLICANT_EVENT_NETWORK_ADDED:
case SUPPLICANT_EVENT_NETWORK_REMOVED:
strncpy(data->supplicant_event_str, event_info.event_str,
sizeof(data->supplicant_event_str) - 1);
event_data->data_len = strlen(data->supplicant_event_str) + 1;
case SUPPLICANT_EVENT_DSCP_POLICY:
/* TODO */
break;
default:
break;
}
if (ret <= 0) {
wpa_printf(MSG_ERROR, "%s Parse failed: %s",
event_info.event_str, strerror(errno));
}
return ret;
}
int supplicant_send_wifi_mgmt_conn_event(void *ctx, int status_code)
{
struct wpa_supplicant *wpa_s = ctx;
int status = wpas_to_wifi_mgmt_conn_status(status_code);
enum net_event_wifi_cmd event;
if (!wpa_s || !wpa_s->current_ssid) {
return -EINVAL;
}
if (wpa_s->current_ssid->mode == WPAS_MODE_AP) {
event = NET_EVENT_WIFI_CMD_AP_ENABLE_RESULT;
} else {
event = NET_EVENT_WIFI_CMD_CONNECT_RESULT;
}
return supplicant_send_wifi_mgmt_event(wpa_s->ifname,
event,
(void *)&status,
sizeof(int));
}
int supplicant_send_wifi_mgmt_disc_event(void *ctx, int reason_code)
{
struct wpa_supplicant *wpa_s = ctx;
int status = wpas_to_wifi_mgmt_disconn_status(reason_code);
enum net_event_wifi_cmd event;
if (!wpa_s || !wpa_s->current_ssid) {
return -EINVAL;
}
if (wpa_s->wpa_state >= WPA_COMPLETED) {
if (wpa_s->current_ssid->mode == WPAS_MODE_AP) {
event = NET_EVENT_WIFI_CMD_AP_DISABLE_RESULT;
} else {
event = NET_EVENT_WIFI_CMD_DISCONNECT_RESULT;
}
} else {
if (wpa_s->current_ssid->mode == WPAS_MODE_AP) {
event = NET_EVENT_WIFI_CMD_AP_ENABLE_RESULT;
} else {
event = NET_EVENT_WIFI_CMD_CONNECT_RESULT;
}
}
return supplicant_send_wifi_mgmt_event(wpa_s->ifname,
event,
(void *)&status,
sizeof(int));
}
#ifdef CONFIG_AP
static enum wifi_link_mode get_sta_link_mode(struct wpa_supplicant *wpa_s, struct sta_info *sta)
{
if (sta->flags & WLAN_STA_HE) {
return WIFI_6;
} else if (sta->flags & WLAN_STA_VHT) {
return WIFI_5;
} else if (sta->flags & WLAN_STA_HT) {
return WIFI_4;
} else if (sta->flags & WLAN_STA_NONERP) {
return WIFI_1;
} else if (wpa_s->assoc_freq > 4000) {
return WIFI_2;
} else if (wpa_s->assoc_freq > 2000) {
return WIFI_3;
} else {
return WIFI_LINK_MODE_UNKNOWN;
}
}
static bool is_twt_capable(struct wpa_supplicant *wpa_s, struct sta_info *sta)
{
#ifdef CONFIG_WIFI_NM_WPA_SUPPLICANT_11AX
return hostapd_get_he_twt_responder(wpa_s->ap_iface->bss[0], IEEE80211_MODE_AP);
#else
return false;
#endif
}
int supplicant_send_wifi_mgmt_ap_status(void *ctx,
enum net_event_wifi_cmd event,
enum wifi_ap_status ap_status)
{
struct wpa_supplicant *wpa_s = ctx;
char *ifname = wpa_s->ifname;
int status = ap_status;
return supplicant_send_wifi_mgmt_event(ifname,
event,
(void *)&status,
sizeof(int));
}
int supplicant_send_wifi_mgmt_ap_sta_event(void *ctx,
enum net_event_wifi_cmd event,
void *data)
{
struct sta_info *sta = data;
struct wpa_supplicant *ap_ctx = ctx;
char *ifname = ap_ctx->ifname;
struct wifi_ap_sta_info sta_info = { 0 };
if (!ap_ctx || !sta) {
return -EINVAL;
}
memcpy(sta_info.mac, sta->addr, sizeof(sta_info.mac));
if (event == NET_EVENT_WIFI_CMD_AP_STA_CONNECTED) {
sta_info.link_mode = get_sta_link_mode(ap_ctx, sta);
sta_info.twt_capable = is_twt_capable(ap_ctx, sta);
}
return supplicant_send_wifi_mgmt_event(ifname,
event,
(void *)&sta_info,
sizeof(sta_info));
}
#endif /* CONFIG_AP */
int supplicant_send_wifi_mgmt_event(const char *ifname, enum net_event_wifi_cmd event,
void *supplicant_status, size_t len)
{
struct net_if *iface = net_if_get_by_index(net_if_get_by_name(ifname));
union supplicant_event_data data;
struct supplicant_int_event_data event_data;
if (!iface) {
wpa_printf(MSG_ERROR, "Could not find iface for %s", ifname);
return -ENODEV;
}
switch (event) {
case NET_EVENT_WIFI_CMD_CONNECT_RESULT:
wifi_mgmt_raise_connect_result_event(
iface,
*(int *)supplicant_status);
break;
case NET_EVENT_WIFI_CMD_DISCONNECT_RESULT:
wifi_mgmt_raise_disconnect_result_event(
iface,
*(int *)supplicant_status);
break;
#ifdef CONFIG_WIFI_NM_WPA_SUPPLICANT_ROAMING
case NET_EVENT_WIFI_CMD_SIGNAL_CHANGE:
net_mgmt_event_notify_with_info(NET_EVENT_WIFI_SIGNAL_CHANGE,
iface, NULL, 0);
break;
case NET_EVENT_WIFI_CMD_NEIGHBOR_REP_RECEIVED:
wifi_mgmt_raise_neighbor_rep_recv_event(
iface,
(char *)supplicant_status, len);
break;
case NET_EVENT_WIFI_CMD_NEIGHBOR_REP_COMPLETE:
net_mgmt_event_notify_with_info(NET_EVENT_WIFI_NEIGHBOR_REP_COMP,
iface, NULL, 0);
break;
#endif
#ifdef CONFIG_AP
case NET_EVENT_WIFI_CMD_AP_ENABLE_RESULT:
wifi_mgmt_raise_ap_enable_result_event(iface,
*(int *)supplicant_status);
break;
case NET_EVENT_WIFI_CMD_AP_DISABLE_RESULT:
wifi_mgmt_raise_ap_disable_result_event(iface,
*(int *)supplicant_status);
break;
case NET_EVENT_WIFI_CMD_AP_STA_CONNECTED:
wifi_mgmt_raise_ap_sta_connected_event(iface,
(struct wifi_ap_sta_info *)supplicant_status);
break;
case NET_EVENT_WIFI_CMD_AP_STA_DISCONNECTED:
wifi_mgmt_raise_ap_sta_disconnected_event(iface,
(struct wifi_ap_sta_info *)supplicant_status);
break;
#endif /* CONFIG_AP */
case NET_EVENT_WIFI_CMD_SUPPLICANT:
event_data.data = &data;
if (supplicant_process_status(&event_data, (char *)supplicant_status) > 0) {
net_mgmt_event_notify_with_info(NET_EVENT_SUPPLICANT_INT_EVENT,
iface, &event_data, sizeof(event_data));
}
break;
default:
wpa_printf(MSG_ERROR, "Unsupported event %d", event);
return -EINVAL;
}
return 0;
}
int supplicant_generate_state_event(const char *ifname,
enum net_event_supplicant_cmd event,
int status)
{
struct net_if *iface;
iface = net_if_get_by_index(net_if_get_by_name(ifname));
if (!iface) {
wpa_printf(MSG_ERROR, "Could not find iface for %s", ifname);
return -ENODEV;
}
switch (event) {
case NET_EVENT_SUPPLICANT_CMD_READY:
net_mgmt_event_notify(NET_EVENT_SUPPLICANT_READY, iface);
break;
case NET_EVENT_SUPPLICANT_CMD_NOT_READY:
net_mgmt_event_notify(NET_EVENT_SUPPLICANT_NOT_READY, iface);
break;
case NET_EVENT_SUPPLICANT_CMD_IFACE_ADDED:
net_mgmt_event_notify(NET_EVENT_SUPPLICANT_IFACE_ADDED, iface);
break;
case NET_EVENT_SUPPLICANT_CMD_IFACE_REMOVING:
net_mgmt_event_notify(NET_EVENT_SUPPLICANT_IFACE_REMOVING, iface);
break;
case NET_EVENT_SUPPLICANT_CMD_IFACE_REMOVED:
net_mgmt_event_notify_with_info(NET_EVENT_SUPPLICANT_IFACE_REMOVED,
iface, &status, sizeof(status));
break;
default:
wpa_printf(MSG_ERROR, "Unsupported event %d", event);
return -EINVAL;
}
return 0;
}