| /* |
| * hostapd - MBO |
| * Copyright (c) 2016, Qualcomm Atheros, 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/ieee802_11_defs.h" |
| #include "common/ieee802_11_common.h" |
| #include "hostapd.h" |
| #include "sta_info.h" |
| #include "mbo_ap.h" |
| |
| |
| void mbo_ap_sta_free(struct sta_info *sta) |
| { |
| struct mbo_non_pref_chan_info *info, *prev; |
| |
| info = sta->non_pref_chan; |
| sta->non_pref_chan = NULL; |
| while (info) { |
| prev = info; |
| info = info->next; |
| os_free(prev); |
| } |
| } |
| |
| |
| static void mbo_ap_parse_non_pref_chan(struct sta_info *sta, |
| const u8 *buf, size_t len) |
| { |
| struct mbo_non_pref_chan_info *info, *tmp; |
| char channels[200], *pos, *end; |
| size_t num_chan, i; |
| int ret; |
| |
| if (len <= 3) |
| return; /* Not enough room for any channels */ |
| |
| num_chan = len - 3; |
| info = os_zalloc(sizeof(*info) + num_chan); |
| if (!info) |
| return; |
| info->op_class = buf[0]; |
| info->pref = buf[len - 2]; |
| info->reason_code = buf[len - 1]; |
| info->num_channels = num_chan; |
| buf++; |
| os_memcpy(info->channels, buf, num_chan); |
| if (!sta->non_pref_chan) { |
| sta->non_pref_chan = info; |
| } else { |
| tmp = sta->non_pref_chan; |
| while (tmp->next) |
| tmp = tmp->next; |
| tmp->next = info; |
| } |
| |
| pos = channels; |
| end = pos + sizeof(channels); |
| *pos = '\0'; |
| for (i = 0; i < num_chan; i++) { |
| ret = os_snprintf(pos, end - pos, "%s%u", |
| i == 0 ? "" : " ", buf[i]); |
| if (os_snprintf_error(end - pos, ret)) { |
| *pos = '\0'; |
| break; |
| } |
| pos += ret; |
| } |
| |
| wpa_printf(MSG_DEBUG, "MBO: STA " MACSTR |
| " non-preferred channel list (op class %u, pref %u, reason code %u, channels %s)", |
| MAC2STR(sta->addr), info->op_class, info->pref, |
| info->reason_code, channels); |
| } |
| |
| |
| void mbo_ap_check_sta_assoc(struct hostapd_data *hapd, struct sta_info *sta, |
| struct ieee802_11_elems *elems) |
| { |
| const u8 *pos, *attr, *end; |
| size_t len; |
| |
| if (!hapd->conf->mbo_enabled || !elems->mbo) |
| return; |
| |
| pos = elems->mbo + 4; |
| len = elems->mbo_len - 4; |
| wpa_hexdump(MSG_DEBUG, "MBO: Association Request attributes", pos, len); |
| |
| attr = get_ie(pos, len, MBO_ATTR_ID_CELL_DATA_CAPA); |
| if (attr && attr[1] >= 1) |
| sta->cell_capa = attr[2]; |
| |
| mbo_ap_sta_free(sta); |
| end = pos + len; |
| while (end - pos > 1) { |
| u8 ie_len = pos[1]; |
| |
| if (2 + ie_len > end - pos) |
| break; |
| |
| if (pos[0] == MBO_ATTR_ID_NON_PREF_CHAN_REPORT) |
| mbo_ap_parse_non_pref_chan(sta, pos + 2, ie_len); |
| pos += 2 + pos[1]; |
| } |
| } |
| |
| |
| int mbo_ap_get_info(struct sta_info *sta, char *buf, size_t buflen) |
| { |
| char *pos = buf, *end = buf + buflen; |
| int ret; |
| struct mbo_non_pref_chan_info *info; |
| u8 i; |
| unsigned int count = 0; |
| |
| if (!sta->cell_capa) |
| return 0; |
| |
| ret = os_snprintf(pos, end - pos, "mbo_cell_capa=%u\n", sta->cell_capa); |
| if (os_snprintf_error(end - pos, ret)) |
| return pos - buf; |
| pos += ret; |
| |
| for (info = sta->non_pref_chan; info; info = info->next) { |
| char *pos2 = pos; |
| |
| ret = os_snprintf(pos2, end - pos2, |
| "non_pref_chan[%u]=%u:%u:%u:", |
| count, info->op_class, info->pref, |
| info->reason_code); |
| count++; |
| if (os_snprintf_error(end - pos2, ret)) |
| break; |
| pos2 += ret; |
| |
| for (i = 0; i < info->num_channels; i++) { |
| ret = os_snprintf(pos2, end - pos2, "%u%s", |
| info->channels[i], |
| i + 1 < info->num_channels ? |
| "," : ""); |
| if (os_snprintf_error(end - pos2, ret)) { |
| pos2 = NULL; |
| break; |
| } |
| pos2 += ret; |
| } |
| |
| if (!pos2) |
| break; |
| ret = os_snprintf(pos2, end - pos2, "\n"); |
| if (os_snprintf_error(end - pos2, ret)) |
| break; |
| pos2 += ret; |
| pos = pos2; |
| } |
| |
| return pos - buf; |
| } |
| |
| |
| static void mbo_ap_wnm_notif_req_cell_capa(struct sta_info *sta, |
| const u8 *buf, size_t len) |
| { |
| if (len < 1) |
| return; |
| wpa_printf(MSG_DEBUG, "MBO: STA " MACSTR |
| " updated cellular data capability: %u", |
| MAC2STR(sta->addr), buf[0]); |
| sta->cell_capa = buf[0]; |
| } |
| |
| |
| static void mbo_ap_wnm_notif_req_elem(struct sta_info *sta, u8 type, |
| const u8 *buf, size_t len, |
| int *first_non_pref_chan) |
| { |
| switch (type) { |
| case WFA_WNM_NOTIF_SUBELEM_NON_PREF_CHAN_REPORT: |
| if (*first_non_pref_chan) { |
| /* |
| * Need to free the previously stored entries now to |
| * allow the update to replace all entries. |
| */ |
| *first_non_pref_chan = 0; |
| mbo_ap_sta_free(sta); |
| } |
| mbo_ap_parse_non_pref_chan(sta, buf, len); |
| break; |
| case WFA_WNM_NOTIF_SUBELEM_CELL_DATA_CAPA: |
| mbo_ap_wnm_notif_req_cell_capa(sta, buf, len); |
| break; |
| default: |
| wpa_printf(MSG_DEBUG, |
| "MBO: Ignore unknown WNM Notification WFA subelement %u", |
| type); |
| break; |
| } |
| } |
| |
| |
| void mbo_ap_wnm_notification_req(struct hostapd_data *hapd, const u8 *addr, |
| const u8 *buf, size_t len) |
| { |
| const u8 *pos, *end; |
| u8 ie_len; |
| struct sta_info *sta; |
| int first_non_pref_chan = 1; |
| |
| if (!hapd->conf->mbo_enabled) |
| return; |
| |
| sta = ap_get_sta(hapd, addr); |
| if (!sta) |
| return; |
| |
| pos = buf; |
| end = buf + len; |
| |
| while (end - pos > 1) { |
| ie_len = pos[1]; |
| |
| if (2 + ie_len > end - pos) |
| break; |
| |
| if (pos[0] == WLAN_EID_VENDOR_SPECIFIC && |
| ie_len >= 4 && WPA_GET_BE24(pos + 2) == OUI_WFA) |
| mbo_ap_wnm_notif_req_elem(sta, pos[5], |
| pos + 6, ie_len - 4, |
| &first_non_pref_chan); |
| else |
| wpa_printf(MSG_DEBUG, |
| "MBO: Ignore unknown WNM Notification element %u (len=%u)", |
| pos[0], pos[1]); |
| |
| pos += 2 + pos[1]; |
| } |
| } |