hostapd/wpa_supplicant/nan_usd.c

535 lines
13 KiB
C
Raw Permalink Normal View History

/*
* NAN unsynchronized service discovery (USD)
* Copyright (c) 2024, Qualcomm Innovation Center, Inc.
*
* This software may be distributed under the terms of the BSD license.
* See README for more details.
*/
#include "utils/includes.h"
#include "utils/common.h"
#include "common/nan_de.h"
#include "wpa_supplicant_i.h"
#include "offchannel.h"
#include "driver_i.h"
#include "notify.h"
#include "p2p_supplicant.h"
#include "nan_usd.h"
static const char *
tx_status_result_txt(enum offchannel_send_action_result result)
{
switch (result) {
case OFFCHANNEL_SEND_ACTION_SUCCESS:
return "success";
case OFFCHANNEL_SEND_ACTION_NO_ACK:
return "no-ack";
case OFFCHANNEL_SEND_ACTION_FAILED:
return "failed";
}
return "?";
}
static void wpas_nan_de_tx_status(struct wpa_supplicant *wpa_s,
unsigned int freq, const u8 *dst,
const u8 *src, const u8 *bssid,
const u8 *data, size_t data_len,
enum offchannel_send_action_result result)
{
if (!wpa_s->nan_de)
return;
wpa_printf(MSG_DEBUG, "NAN: TX status A1=" MACSTR " A2=" MACSTR
" A3=" MACSTR " freq=%d len=%zu result=%s",
MAC2STR(dst), MAC2STR(src), MAC2STR(bssid), freq,
data_len, tx_status_result_txt(result));
nan_de_tx_status(wpa_s->nan_de, freq, dst);
}
struct wpas_nan_usd_tx_work {
unsigned int freq;
unsigned int wait_time;
u8 dst[ETH_ALEN];
u8 src[ETH_ALEN];
u8 bssid[ETH_ALEN];
struct wpabuf *buf;
};
static void wpas_nan_usd_tx_work_free(struct wpas_nan_usd_tx_work *twork)
{
if (!twork)
return;
wpabuf_free(twork->buf);
os_free(twork);
}
static void wpas_nan_usd_tx_work_done(struct wpa_supplicant *wpa_s)
{
struct wpas_nan_usd_tx_work *twork;
if (!wpa_s->nan_usd_tx_work)
return;
twork = wpa_s->nan_usd_tx_work->ctx;
wpas_nan_usd_tx_work_free(twork);
radio_work_done(wpa_s->nan_usd_tx_work);
wpa_s->nan_usd_tx_work = NULL;
}
static int wpas_nan_de_tx_send(struct wpa_supplicant *wpa_s, unsigned int freq,
unsigned int wait_time, const u8 *dst,
const u8 *src, const u8 *bssid,
const struct wpabuf *buf)
{
wpa_printf(MSG_DEBUG, "NAN: TX NAN SDF A1=" MACSTR " A2=" MACSTR
" A3=" MACSTR " freq=%d len=%zu",
MAC2STR(dst), MAC2STR(src), MAC2STR(bssid), freq,
wpabuf_len(buf));
return offchannel_send_action(wpa_s, freq, dst, src, bssid,
wpabuf_head(buf), wpabuf_len(buf),
wait_time, wpas_nan_de_tx_status, 1);
}
static void wpas_nan_usd_start_tx_cb(struct wpa_radio_work *work, int deinit)
{
struct wpa_supplicant *wpa_s = work->wpa_s;
struct wpas_nan_usd_tx_work *twork = work->ctx;
if (deinit) {
if (work->started) {
wpa_s->nan_usd_tx_work = NULL;
offchannel_send_action_done(wpa_s);
}
wpas_nan_usd_tx_work_free(twork);
return;
}
wpa_s->nan_usd_tx_work = work;
if (wpas_nan_de_tx_send(wpa_s, twork->freq, twork->wait_time,
twork->dst, twork->src, twork->bssid,
twork->buf) < 0)
wpas_nan_usd_tx_work_done(wpa_s);
}
static int wpas_nan_de_tx(void *ctx, unsigned int freq, unsigned int wait_time,
const u8 *dst, const u8 *src, const u8 *bssid,
const struct wpabuf *buf)
{
struct wpa_supplicant *wpa_s = ctx;
struct wpas_nan_usd_tx_work *twork;
if (wpa_s->nan_usd_tx_work || wpa_s->nan_usd_listen_work) {
/* Reuse ongoing radio work */
return wpas_nan_de_tx_send(wpa_s, freq, wait_time, dst, src,
bssid, buf);
}
twork = os_zalloc(sizeof(*twork));
if (!twork)
return -1;
twork->freq = freq;
twork->wait_time = wait_time;
os_memcpy(twork->dst, dst, ETH_ALEN);
os_memcpy(twork->src, src, ETH_ALEN);
os_memcpy(twork->bssid, bssid, ETH_ALEN);
twork->buf = wpabuf_dup(buf);
if (!twork->buf) {
wpas_nan_usd_tx_work_free(twork);
return -1;
}
if (radio_add_work(wpa_s, freq, "nan-usd-tx", 0,
wpas_nan_usd_start_tx_cb, twork) < 0) {
wpas_nan_usd_tx_work_free(twork);
return -1;
}
return 0;
}
struct wpas_nan_usd_listen_work {
unsigned int freq;
unsigned int duration;
};
static void wpas_nan_usd_listen_work_done(struct wpa_supplicant *wpa_s)
{
struct wpas_nan_usd_listen_work *lwork;
if (!wpa_s->nan_usd_listen_work)
return;
lwork = wpa_s->nan_usd_listen_work->ctx;
os_free(lwork);
radio_work_done(wpa_s->nan_usd_listen_work);
wpa_s->nan_usd_listen_work = NULL;
}
static void wpas_nan_usd_start_listen_cb(struct wpa_radio_work *work,
int deinit)
{
struct wpa_supplicant *wpa_s = work->wpa_s;
struct wpas_nan_usd_listen_work *lwork = work->ctx;
unsigned int duration;
if (deinit) {
if (work->started) {
wpa_s->nan_usd_listen_work = NULL;
wpa_drv_cancel_remain_on_channel(wpa_s);
}
os_free(lwork);
return;
}
wpa_s->nan_usd_listen_work = work;
duration = lwork->duration;
if (duration > wpa_s->max_remain_on_chan)
duration = wpa_s->max_remain_on_chan;
wpa_printf(MSG_DEBUG, "NAN: Start listen on %u MHz for %u ms",
lwork->freq, duration);
if (wpa_drv_remain_on_channel(wpa_s, lwork->freq, duration) < 0) {
wpa_printf(MSG_DEBUG,
"NAN: Failed to request the driver to remain on channel (%u MHz) for listen",
lwork->freq);
wpas_nan_usd_listen_work_done(wpa_s);
return;
}
}
static int wpas_nan_de_listen(void *ctx, unsigned int freq,
unsigned int duration)
{
struct wpa_supplicant *wpa_s = ctx;
struct wpas_nan_usd_listen_work *lwork;
lwork = os_zalloc(sizeof(*lwork));
if (!lwork)
return -1;
lwork->freq = freq;
lwork->duration = duration;
if (radio_add_work(wpa_s, freq, "nan-usd-listen", 0,
wpas_nan_usd_start_listen_cb, lwork) < 0) {
os_free(lwork);
return -1;
}
return 0;
}
static void
wpas_nan_de_discovery_result(void *ctx, int subscribe_id,
enum nan_service_protocol_type srv_proto_type,
const u8 *ssi, size_t ssi_len, int peer_publish_id,
const u8 *peer_addr, bool fsd, bool fsd_gas)
{
struct wpa_supplicant *wpa_s = ctx;
wpas_notify_nan_discovery_result(wpa_s, srv_proto_type, subscribe_id,
peer_publish_id, peer_addr, fsd,
fsd_gas, ssi, ssi_len);
}
static void wpas_nan_de_replied(void *ctx, int publish_id, const u8 *peer_addr,
int peer_subscribe_id,
enum nan_service_protocol_type srv_proto_type,
const u8 *ssi, size_t ssi_len)
{
struct wpa_supplicant *wpa_s = ctx;
wpas_notify_nan_replied(wpa_s, srv_proto_type, publish_id,
peer_subscribe_id, peer_addr, ssi, ssi_len);
}
static void wpas_nan_de_publish_terminated(void *ctx, int publish_id,
enum nan_de_reason reason)
{
struct wpa_supplicant *wpa_s = ctx;
wpas_notify_nan_publish_terminated(wpa_s, publish_id, reason);
}
static void wpas_nan_de_subscribe_terminated(void *ctx, int subscribe_id,
enum nan_de_reason reason)
{
struct wpa_supplicant *wpa_s = ctx;
wpas_notify_nan_subscribe_terminated(wpa_s, subscribe_id, reason);
}
static void wpas_nan_de_receive(void *ctx, int id, int peer_instance_id,
const u8 *ssi, size_t ssi_len,
const u8 *peer_addr)
{
struct wpa_supplicant *wpa_s = ctx;
wpas_notify_nan_receive(wpa_s, id, peer_instance_id, peer_addr,
ssi, ssi_len);
}
#ifdef CONFIG_P2P
static void wpas_nan_process_p2p_usd_elems(void *ctx, const u8 *buf,
u16 buf_len, const u8 *peer_addr,
unsigned int freq)
{
struct wpa_supplicant *wpa_s = ctx;
wpas_p2p_process_usd_elems(wpa_s, buf, buf_len, peer_addr, freq);
}
#endif /* CONFIG_P2P */
int wpas_nan_usd_init(struct wpa_supplicant *wpa_s)
{
struct nan_callbacks cb;
bool offload = wpa_s->drv_flags2 & WPA_DRIVER_FLAGS2_NAN_OFFLOAD;
os_memset(&cb, 0, sizeof(cb));
cb.ctx = wpa_s;
cb.tx = wpas_nan_de_tx;
cb.listen = wpas_nan_de_listen;
cb.discovery_result = wpas_nan_de_discovery_result;
cb.replied = wpas_nan_de_replied;
cb.publish_terminated = wpas_nan_de_publish_terminated;
cb.subscribe_terminated = wpas_nan_de_subscribe_terminated;
cb.receive = wpas_nan_de_receive;
#ifdef CONFIG_P2P
cb.process_p2p_usd_elems = wpas_nan_process_p2p_usd_elems;
#endif /* CONFIG_P2P */
wpa_s->nan_de = nan_de_init(wpa_s->own_addr, offload, false, &cb);
if (!wpa_s->nan_de)
return -1;
return 0;
}
void wpas_nan_usd_deinit(struct wpa_supplicant *wpa_s)
{
nan_de_deinit(wpa_s->nan_de);
wpa_s->nan_de = NULL;
}
void wpas_nan_usd_rx_sdf(struct wpa_supplicant *wpa_s, const u8 *src,
unsigned int freq, const u8 *buf, size_t len)
{
if (!wpa_s->nan_de)
return;
nan_de_rx_sdf(wpa_s->nan_de, src, freq, buf, len);
}
void wpas_nan_usd_flush(struct wpa_supplicant *wpa_s)
{
if (!wpa_s->nan_de)
return;
nan_de_flush(wpa_s->nan_de);
if (wpa_s->drv_flags2 & WPA_DRIVER_FLAGS2_NAN_OFFLOAD)
wpas_drv_nan_flush(wpa_s);
}
int wpas_nan_usd_publish(struct wpa_supplicant *wpa_s, const char *service_name,
enum nan_service_protocol_type srv_proto_type,
const struct wpabuf *ssi,
struct nan_publish_params *params, bool p2p)
{
int publish_id;
struct wpabuf *elems = NULL;
const u8 *addr;
if (!wpa_s->nan_de)
return -1;
if (p2p) {
elems = wpas_p2p_usd_elems(wpa_s);
addr = wpa_s->global->p2p_dev_addr;
} else {
addr = wpa_s->own_addr;
}
publish_id = nan_de_publish(wpa_s->nan_de, service_name, srv_proto_type,
ssi, elems, params, p2p);
if (publish_id >= 1 &&
(wpa_s->drv_flags2 & WPA_DRIVER_FLAGS2_NAN_OFFLOAD) &&
wpas_drv_nan_publish(wpa_s, addr, publish_id, service_name,
nan_de_get_service_id(wpa_s->nan_de,
publish_id),
srv_proto_type, ssi, elems, params) < 0) {
nan_de_cancel_publish(wpa_s->nan_de, publish_id);
publish_id = -1;
}
wpabuf_free(elems);
return publish_id;
}
void wpas_nan_usd_cancel_publish(struct wpa_supplicant *wpa_s, int publish_id)
{
if (!wpa_s->nan_de)
return;
nan_de_cancel_publish(wpa_s->nan_de, publish_id);
if (wpa_s->drv_flags2 & WPA_DRIVER_FLAGS2_NAN_OFFLOAD)
wpas_drv_nan_cancel_publish(wpa_s, publish_id);
}
int wpas_nan_usd_update_publish(struct wpa_supplicant *wpa_s, int publish_id,
const struct wpabuf *ssi)
{
int ret;
if (!wpa_s->nan_de)
return -1;
ret = nan_de_update_publish(wpa_s->nan_de, publish_id, ssi);
if (ret == 0 && (wpa_s->drv_flags2 & WPA_DRIVER_FLAGS2_NAN_OFFLOAD) &&
wpas_drv_nan_cancel_publish(wpa_s, publish_id) < 0)
return -1;
return ret;
}
int wpas_nan_usd_subscribe(struct wpa_supplicant *wpa_s,
const char *service_name,
enum nan_service_protocol_type srv_proto_type,
const struct wpabuf *ssi,
struct nan_subscribe_params *params, bool p2p)
{
int subscribe_id;
struct wpabuf *elems = NULL;
const u8 *addr;
if (!wpa_s->nan_de)
return -1;
if (p2p) {
elems = wpas_p2p_usd_elems(wpa_s);
addr = wpa_s->global->p2p_dev_addr;
} else {
addr = wpa_s->own_addr;
}
subscribe_id = nan_de_subscribe(wpa_s->nan_de, service_name,
srv_proto_type, ssi, elems, params,
p2p);
if (subscribe_id >= 1 &&
(wpa_s->drv_flags2 & WPA_DRIVER_FLAGS2_NAN_OFFLOAD) &&
wpas_drv_nan_subscribe(wpa_s, addr, subscribe_id, service_name,
nan_de_get_service_id(wpa_s->nan_de,
subscribe_id),
srv_proto_type, ssi, elems, params) < 0) {
nan_de_cancel_subscribe(wpa_s->nan_de, subscribe_id);
subscribe_id = -1;
}
wpabuf_free(elems);
return subscribe_id;
}
void wpas_nan_usd_cancel_subscribe(struct wpa_supplicant *wpa_s,
int subscribe_id)
{
if (!wpa_s->nan_de)
return;
nan_de_cancel_subscribe(wpa_s->nan_de, subscribe_id);
if (wpa_s->drv_flags2 & WPA_DRIVER_FLAGS2_NAN_OFFLOAD)
wpas_drv_nan_cancel_subscribe(wpa_s, subscribe_id);
}
int wpas_nan_usd_transmit(struct wpa_supplicant *wpa_s, int handle,
const struct wpabuf *ssi, const struct wpabuf *elems,
const u8 *peer_addr, u8 req_instance_id)
{
if (!wpa_s->nan_de)
return -1;
return nan_de_transmit(wpa_s->nan_de, handle, ssi, elems, peer_addr,
req_instance_id);
}
void wpas_nan_usd_remain_on_channel_cb(struct wpa_supplicant *wpa_s,
unsigned int freq, unsigned int duration)
{
wpas_nan_usd_listen_work_done(wpa_s);
if (wpa_s->nan_de)
nan_de_listen_started(wpa_s->nan_de, freq, duration);
}
void wpas_nan_usd_cancel_remain_on_channel_cb(struct wpa_supplicant *wpa_s,
unsigned int freq)
{
if (wpa_s->nan_de)
nan_de_listen_ended(wpa_s->nan_de, freq);
}
void wpas_nan_usd_tx_wait_expire(struct wpa_supplicant *wpa_s)
{
wpas_nan_usd_tx_work_done(wpa_s);
if (wpa_s->nan_de)
nan_de_tx_wait_ended(wpa_s->nan_de);
}
int * wpas_nan_usd_all_freqs(struct wpa_supplicant *wpa_s)
{
int i, j;
int *freqs = NULL;
if (!wpa_s->hw.modes)
return NULL;
for (i = 0; i < wpa_s->hw.num_modes; i++) {
struct hostapd_hw_modes *mode = &wpa_s->hw.modes[i];
for (j = 0; j < mode->num_channels; j++) {
struct hostapd_channel_data *chan = &mode->channels[j];
/* All 20 MHz channels on 2.4 and 5 GHz band */
if (chan->freq < 2412 || chan->freq > 5900)
continue;
/* that allow frames to be transmitted */
if (chan->flag & (HOSTAPD_CHAN_DISABLED |
HOSTAPD_CHAN_NO_IR |
HOSTAPD_CHAN_RADAR))
continue;
int_array_add_unique(&freqs, chan->freq);
}
}
return freqs;
}