1e5b75505Sopenharmony_ci/* 2e5b75505Sopenharmony_ci * FST module - FST group object implementation 3e5b75505Sopenharmony_ci * Copyright (c) 2014, Qualcomm Atheros, Inc. 4e5b75505Sopenharmony_ci * 5e5b75505Sopenharmony_ci * This software may be distributed under the terms of the BSD license. 6e5b75505Sopenharmony_ci * See README for more details. 7e5b75505Sopenharmony_ci */ 8e5b75505Sopenharmony_ci 9e5b75505Sopenharmony_ci#include "utils/includes.h" 10e5b75505Sopenharmony_ci#include "utils/common.h" 11e5b75505Sopenharmony_ci#include "common/defs.h" 12e5b75505Sopenharmony_ci#include "common/ieee802_11_defs.h" 13e5b75505Sopenharmony_ci#include "common/ieee802_11_common.h" 14e5b75505Sopenharmony_ci#include "drivers/driver.h" 15e5b75505Sopenharmony_ci#include "fst/fst_internal.h" 16e5b75505Sopenharmony_ci#include "fst/fst_defs.h" 17e5b75505Sopenharmony_ci 18e5b75505Sopenharmony_ci 19e5b75505Sopenharmony_cistruct dl_list fst_global_groups_list; 20e5b75505Sopenharmony_ci 21e5b75505Sopenharmony_ci 22e5b75505Sopenharmony_cistatic void fst_dump_mb_ies(const char *group_id, const char *ifname, 23e5b75505Sopenharmony_ci struct wpabuf *mbies) 24e5b75505Sopenharmony_ci{ 25e5b75505Sopenharmony_ci const u8 *p = wpabuf_head(mbies); 26e5b75505Sopenharmony_ci size_t s = wpabuf_len(mbies); 27e5b75505Sopenharmony_ci 28e5b75505Sopenharmony_ci while (s >= 2) { 29e5b75505Sopenharmony_ci const struct multi_band_ie *mbie = 30e5b75505Sopenharmony_ci (const struct multi_band_ie *) p; 31e5b75505Sopenharmony_ci WPA_ASSERT(mbie->eid == WLAN_EID_MULTI_BAND); 32e5b75505Sopenharmony_ci WPA_ASSERT(2U + mbie->len >= sizeof(*mbie)); 33e5b75505Sopenharmony_ci 34e5b75505Sopenharmony_ci fst_printf(MSG_WARNING, 35e5b75505Sopenharmony_ci "%s: %s: mb_ctrl=%u band_id=%u op_class=%u chan=%u bssid=" 36e5b75505Sopenharmony_ci MACSTR 37e5b75505Sopenharmony_ci " beacon_int=%u tsf_offs=[%u %u %u %u %u %u %u %u] mb_cc=0x%02x tmout=%u", 38e5b75505Sopenharmony_ci group_id, ifname, 39e5b75505Sopenharmony_ci mbie->mb_ctrl, mbie->band_id, mbie->op_class, 40e5b75505Sopenharmony_ci mbie->chan, MAC2STR(mbie->bssid), mbie->beacon_int, 41e5b75505Sopenharmony_ci mbie->tsf_offs[0], mbie->tsf_offs[1], 42e5b75505Sopenharmony_ci mbie->tsf_offs[2], mbie->tsf_offs[3], 43e5b75505Sopenharmony_ci mbie->tsf_offs[4], mbie->tsf_offs[5], 44e5b75505Sopenharmony_ci mbie->tsf_offs[6], mbie->tsf_offs[7], 45e5b75505Sopenharmony_ci mbie->mb_connection_capability, 46e5b75505Sopenharmony_ci mbie->fst_session_tmout); 47e5b75505Sopenharmony_ci 48e5b75505Sopenharmony_ci p += 2 + mbie->len; 49e5b75505Sopenharmony_ci s -= 2 + mbie->len; 50e5b75505Sopenharmony_ci } 51e5b75505Sopenharmony_ci} 52e5b75505Sopenharmony_ci 53e5b75505Sopenharmony_ci 54e5b75505Sopenharmony_cistatic void fst_fill_mb_ie(struct wpabuf *buf, const u8 *bssid, 55e5b75505Sopenharmony_ci const u8 *own_addr, enum mb_band_id band, u8 channel) 56e5b75505Sopenharmony_ci{ 57e5b75505Sopenharmony_ci struct multi_band_ie *mbie; 58e5b75505Sopenharmony_ci size_t len = sizeof(*mbie); 59e5b75505Sopenharmony_ci 60e5b75505Sopenharmony_ci if (own_addr) 61e5b75505Sopenharmony_ci len += ETH_ALEN; 62e5b75505Sopenharmony_ci 63e5b75505Sopenharmony_ci mbie = wpabuf_put(buf, len); 64e5b75505Sopenharmony_ci 65e5b75505Sopenharmony_ci os_memset(mbie, 0, len); 66e5b75505Sopenharmony_ci 67e5b75505Sopenharmony_ci mbie->eid = WLAN_EID_MULTI_BAND; 68e5b75505Sopenharmony_ci mbie->len = len - 2; 69e5b75505Sopenharmony_ci#ifdef HOSTAPD 70e5b75505Sopenharmony_ci mbie->mb_ctrl = MB_STA_ROLE_AP; 71e5b75505Sopenharmony_ci mbie->mb_connection_capability = MB_CONNECTION_CAPABILITY_AP; 72e5b75505Sopenharmony_ci#else /* HOSTAPD */ 73e5b75505Sopenharmony_ci mbie->mb_ctrl = MB_STA_ROLE_NON_PCP_NON_AP; 74e5b75505Sopenharmony_ci mbie->mb_connection_capability = 0; 75e5b75505Sopenharmony_ci#endif /* HOSTAPD */ 76e5b75505Sopenharmony_ci if (bssid) 77e5b75505Sopenharmony_ci os_memcpy(mbie->bssid, bssid, ETH_ALEN); 78e5b75505Sopenharmony_ci mbie->band_id = band; 79e5b75505Sopenharmony_ci mbie->op_class = 0; /* means all */ 80e5b75505Sopenharmony_ci mbie->chan = channel; 81e5b75505Sopenharmony_ci mbie->fst_session_tmout = FST_DEFAULT_SESSION_TIMEOUT_TU; 82e5b75505Sopenharmony_ci 83e5b75505Sopenharmony_ci if (own_addr) { 84e5b75505Sopenharmony_ci mbie->mb_ctrl |= MB_CTRL_STA_MAC_PRESENT; 85e5b75505Sopenharmony_ci os_memcpy(&mbie[1], own_addr, ETH_ALEN); 86e5b75505Sopenharmony_ci } 87e5b75505Sopenharmony_ci} 88e5b75505Sopenharmony_ci 89e5b75505Sopenharmony_ci 90e5b75505Sopenharmony_cistatic unsigned fst_fill_iface_mb_ies(struct fst_iface *f, struct wpabuf *buf) 91e5b75505Sopenharmony_ci{ 92e5b75505Sopenharmony_ci const u8 *bssid; 93e5b75505Sopenharmony_ci 94e5b75505Sopenharmony_ci bssid = fst_iface_get_bssid(f); 95e5b75505Sopenharmony_ci if (bssid) { 96e5b75505Sopenharmony_ci enum hostapd_hw_mode hw_mode; 97e5b75505Sopenharmony_ci u8 channel; 98e5b75505Sopenharmony_ci 99e5b75505Sopenharmony_ci if (buf) { 100e5b75505Sopenharmony_ci fst_iface_get_channel_info(f, &hw_mode, &channel); 101e5b75505Sopenharmony_ci fst_fill_mb_ie(buf, bssid, fst_iface_get_addr(f), 102e5b75505Sopenharmony_ci fst_hw_mode_to_band(hw_mode), channel); 103e5b75505Sopenharmony_ci } 104e5b75505Sopenharmony_ci return 1; 105e5b75505Sopenharmony_ci } else { 106e5b75505Sopenharmony_ci unsigned bands[MB_BAND_ID_WIFI_60GHZ + 1] = {}; 107e5b75505Sopenharmony_ci struct hostapd_hw_modes *modes; 108e5b75505Sopenharmony_ci enum mb_band_id b; 109e5b75505Sopenharmony_ci int num_modes = fst_iface_get_hw_modes(f, &modes); 110e5b75505Sopenharmony_ci int ret = 0; 111e5b75505Sopenharmony_ci 112e5b75505Sopenharmony_ci while (num_modes--) { 113e5b75505Sopenharmony_ci b = fst_hw_mode_to_band(modes->mode); 114e5b75505Sopenharmony_ci modes++; 115e5b75505Sopenharmony_ci if (b >= ARRAY_SIZE(bands) || bands[b]++) 116e5b75505Sopenharmony_ci continue; 117e5b75505Sopenharmony_ci ret++; 118e5b75505Sopenharmony_ci if (buf) 119e5b75505Sopenharmony_ci fst_fill_mb_ie(buf, NULL, fst_iface_get_addr(f), 120e5b75505Sopenharmony_ci b, MB_STA_CHANNEL_ALL); 121e5b75505Sopenharmony_ci } 122e5b75505Sopenharmony_ci return ret; 123e5b75505Sopenharmony_ci } 124e5b75505Sopenharmony_ci} 125e5b75505Sopenharmony_ci 126e5b75505Sopenharmony_ci 127e5b75505Sopenharmony_cistatic struct wpabuf * fst_group_create_mb_ie(struct fst_group *g, 128e5b75505Sopenharmony_ci struct fst_iface *i) 129e5b75505Sopenharmony_ci{ 130e5b75505Sopenharmony_ci struct wpabuf *buf; 131e5b75505Sopenharmony_ci struct fst_iface *f; 132e5b75505Sopenharmony_ci unsigned int nof_mbies = 0; 133e5b75505Sopenharmony_ci unsigned int nof_ifaces_added = 0; 134e5b75505Sopenharmony_ci 135e5b75505Sopenharmony_ci foreach_fst_group_iface(g, f) { 136e5b75505Sopenharmony_ci if (f == i) 137e5b75505Sopenharmony_ci continue; 138e5b75505Sopenharmony_ci nof_mbies += fst_fill_iface_mb_ies(f, NULL); 139e5b75505Sopenharmony_ci } 140e5b75505Sopenharmony_ci 141e5b75505Sopenharmony_ci buf = wpabuf_alloc(nof_mbies * 142e5b75505Sopenharmony_ci (sizeof(struct multi_band_ie) + ETH_ALEN)); 143e5b75505Sopenharmony_ci if (!buf) { 144e5b75505Sopenharmony_ci fst_printf_iface(i, MSG_ERROR, 145e5b75505Sopenharmony_ci "cannot allocate mem for %u MB IEs", 146e5b75505Sopenharmony_ci nof_mbies); 147e5b75505Sopenharmony_ci return NULL; 148e5b75505Sopenharmony_ci } 149e5b75505Sopenharmony_ci 150e5b75505Sopenharmony_ci /* The list is sorted in descending order by priorities, so MB IEs will 151e5b75505Sopenharmony_ci * be arranged in the same order, as required by spec (see corresponding 152e5b75505Sopenharmony_ci * comment in.fst_attach(). 153e5b75505Sopenharmony_ci */ 154e5b75505Sopenharmony_ci foreach_fst_group_iface(g, f) { 155e5b75505Sopenharmony_ci if (f == i) 156e5b75505Sopenharmony_ci continue; 157e5b75505Sopenharmony_ci 158e5b75505Sopenharmony_ci fst_fill_iface_mb_ies(f, buf); 159e5b75505Sopenharmony_ci ++nof_ifaces_added; 160e5b75505Sopenharmony_ci 161e5b75505Sopenharmony_ci fst_printf_iface(i, MSG_DEBUG, "added to MB IE"); 162e5b75505Sopenharmony_ci } 163e5b75505Sopenharmony_ci 164e5b75505Sopenharmony_ci if (!nof_ifaces_added) { 165e5b75505Sopenharmony_ci wpabuf_free(buf); 166e5b75505Sopenharmony_ci buf = NULL; 167e5b75505Sopenharmony_ci fst_printf_iface(i, MSG_INFO, 168e5b75505Sopenharmony_ci "cannot add MB IE: no backup ifaces"); 169e5b75505Sopenharmony_ci } else { 170e5b75505Sopenharmony_ci fst_dump_mb_ies(fst_group_get_id(g), fst_iface_get_name(i), 171e5b75505Sopenharmony_ci buf); 172e5b75505Sopenharmony_ci } 173e5b75505Sopenharmony_ci 174e5b75505Sopenharmony_ci return buf; 175e5b75505Sopenharmony_ci} 176e5b75505Sopenharmony_ci 177e5b75505Sopenharmony_ci 178e5b75505Sopenharmony_cistatic const u8 * fst_mbie_get_peer_addr(const struct multi_band_ie *mbie) 179e5b75505Sopenharmony_ci{ 180e5b75505Sopenharmony_ci const u8 *peer_addr = NULL; 181e5b75505Sopenharmony_ci 182e5b75505Sopenharmony_ci switch (MB_CTRL_ROLE(mbie->mb_ctrl)) { 183e5b75505Sopenharmony_ci case MB_STA_ROLE_AP: 184e5b75505Sopenharmony_ci peer_addr = mbie->bssid; 185e5b75505Sopenharmony_ci break; 186e5b75505Sopenharmony_ci case MB_STA_ROLE_NON_PCP_NON_AP: 187e5b75505Sopenharmony_ci if (mbie->mb_ctrl & MB_CTRL_STA_MAC_PRESENT && 188e5b75505Sopenharmony_ci (size_t) 2 + mbie->len >= sizeof(*mbie) + ETH_ALEN) 189e5b75505Sopenharmony_ci peer_addr = (const u8 *) &mbie[1]; 190e5b75505Sopenharmony_ci break; 191e5b75505Sopenharmony_ci default: 192e5b75505Sopenharmony_ci break; 193e5b75505Sopenharmony_ci } 194e5b75505Sopenharmony_ci 195e5b75505Sopenharmony_ci return peer_addr; 196e5b75505Sopenharmony_ci} 197e5b75505Sopenharmony_ci 198e5b75505Sopenharmony_ci 199e5b75505Sopenharmony_cistatic const u8 * fst_mbie_get_peer_addr_for_band(const struct wpabuf *mbies, 200e5b75505Sopenharmony_ci u8 band_id) 201e5b75505Sopenharmony_ci{ 202e5b75505Sopenharmony_ci const u8 *p = wpabuf_head(mbies); 203e5b75505Sopenharmony_ci size_t s = wpabuf_len(mbies); 204e5b75505Sopenharmony_ci 205e5b75505Sopenharmony_ci while (s >= 2) { 206e5b75505Sopenharmony_ci const struct multi_band_ie *mbie = 207e5b75505Sopenharmony_ci (const struct multi_band_ie *) p; 208e5b75505Sopenharmony_ci 209e5b75505Sopenharmony_ci if (mbie->eid != WLAN_EID_MULTI_BAND) { 210e5b75505Sopenharmony_ci fst_printf(MSG_INFO, "unexpected eid %d", mbie->eid); 211e5b75505Sopenharmony_ci return NULL; 212e5b75505Sopenharmony_ci } 213e5b75505Sopenharmony_ci 214e5b75505Sopenharmony_ci if (mbie->len < sizeof(*mbie) - 2 || mbie->len > s - 2) { 215e5b75505Sopenharmony_ci fst_printf(MSG_INFO, "invalid mbie len %d", 216e5b75505Sopenharmony_ci mbie->len); 217e5b75505Sopenharmony_ci return NULL; 218e5b75505Sopenharmony_ci } 219e5b75505Sopenharmony_ci 220e5b75505Sopenharmony_ci if (mbie->band_id == band_id) 221e5b75505Sopenharmony_ci return fst_mbie_get_peer_addr(mbie); 222e5b75505Sopenharmony_ci 223e5b75505Sopenharmony_ci p += 2 + mbie->len; 224e5b75505Sopenharmony_ci s -= 2 + mbie->len; 225e5b75505Sopenharmony_ci } 226e5b75505Sopenharmony_ci 227e5b75505Sopenharmony_ci fst_printf(MSG_INFO, "mbie doesn't contain band %d", band_id); 228e5b75505Sopenharmony_ci return NULL; 229e5b75505Sopenharmony_ci} 230e5b75505Sopenharmony_ci 231e5b75505Sopenharmony_ci 232e5b75505Sopenharmony_cistruct fst_iface * fst_group_get_iface_by_name(struct fst_group *g, 233e5b75505Sopenharmony_ci const char *ifname) 234e5b75505Sopenharmony_ci{ 235e5b75505Sopenharmony_ci struct fst_iface *f; 236e5b75505Sopenharmony_ci 237e5b75505Sopenharmony_ci foreach_fst_group_iface(g, f) { 238e5b75505Sopenharmony_ci const char *in = fst_iface_get_name(f); 239e5b75505Sopenharmony_ci 240e5b75505Sopenharmony_ci if (os_strncmp(in, ifname, os_strlen(in)) == 0) 241e5b75505Sopenharmony_ci return f; 242e5b75505Sopenharmony_ci } 243e5b75505Sopenharmony_ci 244e5b75505Sopenharmony_ci return NULL; 245e5b75505Sopenharmony_ci} 246e5b75505Sopenharmony_ci 247e5b75505Sopenharmony_ci 248e5b75505Sopenharmony_ciu8 fst_group_assign_dialog_token(struct fst_group *g) 249e5b75505Sopenharmony_ci{ 250e5b75505Sopenharmony_ci g->dialog_token++; 251e5b75505Sopenharmony_ci if (g->dialog_token == 0) 252e5b75505Sopenharmony_ci g->dialog_token++; 253e5b75505Sopenharmony_ci return g->dialog_token; 254e5b75505Sopenharmony_ci} 255e5b75505Sopenharmony_ci 256e5b75505Sopenharmony_ci 257e5b75505Sopenharmony_ciu32 fst_group_assign_fsts_id(struct fst_group *g) 258e5b75505Sopenharmony_ci{ 259e5b75505Sopenharmony_ci g->fsts_id++; 260e5b75505Sopenharmony_ci return g->fsts_id; 261e5b75505Sopenharmony_ci} 262e5b75505Sopenharmony_ci 263e5b75505Sopenharmony_ci 264e5b75505Sopenharmony_ci/** 265e5b75505Sopenharmony_ci * fst_group_get_peer_other_connection_1 - Find peer's "other" connection 266e5b75505Sopenharmony_ci * (iface, MAC tuple) by using peer's MB IE on iface. 267e5b75505Sopenharmony_ci * 268e5b75505Sopenharmony_ci * @iface: iface on which FST Setup Request was received 269e5b75505Sopenharmony_ci * @peer_addr: Peer address on iface 270e5b75505Sopenharmony_ci * @band_id: "other" connection band id 271e5b75505Sopenharmony_ci * @other_peer_addr (out): Peer's MAC address on the "other" connection (on the 272e5b75505Sopenharmony_ci * "other" iface) 273e5b75505Sopenharmony_ci * 274e5b75505Sopenharmony_ci * This function parses peer's MB IE on iface. It looks for peer's MAC address 275e5b75505Sopenharmony_ci * on band_id (tmp_peer_addr). Next all interfaces are iterated to find an 276e5b75505Sopenharmony_ci * interface which correlates with band_id. If such interface is found, peer 277e5b75505Sopenharmony_ci * database is iterated to see if tmp_peer_addr is connected over it. 278e5b75505Sopenharmony_ci */ 279e5b75505Sopenharmony_cistatic struct fst_iface * 280e5b75505Sopenharmony_cifst_group_get_peer_other_connection_1(struct fst_iface *iface, 281e5b75505Sopenharmony_ci const u8 *peer_addr, u8 band_id, 282e5b75505Sopenharmony_ci u8 *other_peer_addr) 283e5b75505Sopenharmony_ci{ 284e5b75505Sopenharmony_ci const struct wpabuf *mbies; 285e5b75505Sopenharmony_ci struct fst_iface *other_iface; 286e5b75505Sopenharmony_ci const u8 *tmp_peer_addr; 287e5b75505Sopenharmony_ci 288e5b75505Sopenharmony_ci /* Get peer's MB IEs on iface */ 289e5b75505Sopenharmony_ci mbies = fst_iface_get_peer_mb_ie(iface, peer_addr); 290e5b75505Sopenharmony_ci if (!mbies) 291e5b75505Sopenharmony_ci return NULL; 292e5b75505Sopenharmony_ci 293e5b75505Sopenharmony_ci /* Get peer's MAC address on the "other" interface */ 294e5b75505Sopenharmony_ci tmp_peer_addr = fst_mbie_get_peer_addr_for_band(mbies, band_id); 295e5b75505Sopenharmony_ci if (!tmp_peer_addr) { 296e5b75505Sopenharmony_ci fst_printf(MSG_INFO, 297e5b75505Sopenharmony_ci "couldn't extract other peer addr from mbies"); 298e5b75505Sopenharmony_ci return NULL; 299e5b75505Sopenharmony_ci } 300e5b75505Sopenharmony_ci 301e5b75505Sopenharmony_ci fst_printf(MSG_DEBUG, "found other peer addr from mbies: " MACSTR, 302e5b75505Sopenharmony_ci MAC2STR(tmp_peer_addr)); 303e5b75505Sopenharmony_ci 304e5b75505Sopenharmony_ci foreach_fst_group_iface(fst_iface_get_group(iface), other_iface) { 305e5b75505Sopenharmony_ci if (other_iface == iface || 306e5b75505Sopenharmony_ci band_id != fst_iface_get_band_id(other_iface)) 307e5b75505Sopenharmony_ci continue; 308e5b75505Sopenharmony_ci if (fst_iface_is_connected(other_iface, tmp_peer_addr, FALSE)) { 309e5b75505Sopenharmony_ci os_memcpy(other_peer_addr, tmp_peer_addr, ETH_ALEN); 310e5b75505Sopenharmony_ci return other_iface; 311e5b75505Sopenharmony_ci } 312e5b75505Sopenharmony_ci } 313e5b75505Sopenharmony_ci 314e5b75505Sopenharmony_ci return NULL; 315e5b75505Sopenharmony_ci} 316e5b75505Sopenharmony_ci 317e5b75505Sopenharmony_ci 318e5b75505Sopenharmony_ci/** 319e5b75505Sopenharmony_ci * fst_group_get_peer_other_connection_2 - Find peer's "other" connection 320e5b75505Sopenharmony_ci * (iface, MAC tuple) by using MB IEs of other peers. 321e5b75505Sopenharmony_ci * 322e5b75505Sopenharmony_ci * @iface: iface on which FST Setup Request was received 323e5b75505Sopenharmony_ci * @peer_addr: Peer address on iface 324e5b75505Sopenharmony_ci * @band_id: "other" connection band id 325e5b75505Sopenharmony_ci * @other_peer_addr (out): Peer's MAC address on the "other" connection (on the 326e5b75505Sopenharmony_ci * "other" iface) 327e5b75505Sopenharmony_ci * 328e5b75505Sopenharmony_ci * This function iterates all connection (other_iface, cur_peer_addr tuples). 329e5b75505Sopenharmony_ci * For each connection, MB IE (of cur_peer_addr on other_iface) is parsed and 330e5b75505Sopenharmony_ci * MAC address on iface's band_id is extracted (this_peer_addr). 331e5b75505Sopenharmony_ci * this_peer_addr is then compared to peer_addr. A match indicates we have 332e5b75505Sopenharmony_ci * found the "other" connection. 333e5b75505Sopenharmony_ci */ 334e5b75505Sopenharmony_cistatic struct fst_iface * 335e5b75505Sopenharmony_cifst_group_get_peer_other_connection_2(struct fst_iface *iface, 336e5b75505Sopenharmony_ci const u8 *peer_addr, u8 band_id, 337e5b75505Sopenharmony_ci u8 *other_peer_addr) 338e5b75505Sopenharmony_ci{ 339e5b75505Sopenharmony_ci u8 this_band_id = fst_iface_get_band_id(iface); 340e5b75505Sopenharmony_ci const u8 *cur_peer_addr, *this_peer_addr; 341e5b75505Sopenharmony_ci struct fst_get_peer_ctx *ctx; 342e5b75505Sopenharmony_ci struct fst_iface *other_iface; 343e5b75505Sopenharmony_ci const struct wpabuf *cur_mbie; 344e5b75505Sopenharmony_ci 345e5b75505Sopenharmony_ci foreach_fst_group_iface(fst_iface_get_group(iface), other_iface) { 346e5b75505Sopenharmony_ci if (other_iface == iface || 347e5b75505Sopenharmony_ci band_id != fst_iface_get_band_id(other_iface)) 348e5b75505Sopenharmony_ci continue; 349e5b75505Sopenharmony_ci cur_peer_addr = fst_iface_get_peer_first(other_iface, &ctx, 350e5b75505Sopenharmony_ci TRUE); 351e5b75505Sopenharmony_ci for (; cur_peer_addr; 352e5b75505Sopenharmony_ci cur_peer_addr = fst_iface_get_peer_next(other_iface, &ctx, 353e5b75505Sopenharmony_ci TRUE)) { 354e5b75505Sopenharmony_ci cur_mbie = fst_iface_get_peer_mb_ie(other_iface, 355e5b75505Sopenharmony_ci cur_peer_addr); 356e5b75505Sopenharmony_ci if (!cur_mbie) 357e5b75505Sopenharmony_ci continue; 358e5b75505Sopenharmony_ci this_peer_addr = fst_mbie_get_peer_addr_for_band( 359e5b75505Sopenharmony_ci cur_mbie, this_band_id); 360e5b75505Sopenharmony_ci if (!this_peer_addr) 361e5b75505Sopenharmony_ci continue; 362e5b75505Sopenharmony_ci if (os_memcmp(this_peer_addr, peer_addr, ETH_ALEN) == 363e5b75505Sopenharmony_ci 0) { 364e5b75505Sopenharmony_ci os_memcpy(other_peer_addr, cur_peer_addr, 365e5b75505Sopenharmony_ci ETH_ALEN); 366e5b75505Sopenharmony_ci return other_iface; 367e5b75505Sopenharmony_ci } 368e5b75505Sopenharmony_ci } 369e5b75505Sopenharmony_ci } 370e5b75505Sopenharmony_ci 371e5b75505Sopenharmony_ci return NULL; 372e5b75505Sopenharmony_ci} 373e5b75505Sopenharmony_ci 374e5b75505Sopenharmony_ci 375e5b75505Sopenharmony_ci/** 376e5b75505Sopenharmony_ci * fst_group_get_peer_other_connection - Find peer's "other" connection (iface, 377e5b75505Sopenharmony_ci * MAC tuple). 378e5b75505Sopenharmony_ci * 379e5b75505Sopenharmony_ci * @iface: iface on which FST Setup Request was received 380e5b75505Sopenharmony_ci * @peer_addr: Peer address on iface 381e5b75505Sopenharmony_ci * @band_id: "other" connection band id 382e5b75505Sopenharmony_ci * @other_peer_addr (out): Peer's MAC address on the "other" connection (on the 383e5b75505Sopenharmony_ci * "other" iface) 384e5b75505Sopenharmony_ci * 385e5b75505Sopenharmony_ci * This function is called upon receiving FST Setup Request from some peer who 386e5b75505Sopenharmony_ci * has peer_addr on iface. It searches for another connection of the same peer 387e5b75505Sopenharmony_ci * on different interface which correlates with band_id. MB IEs received from 388e5b75505Sopenharmony_ci * peer (on the two different interfaces) are used to identify same peer. 389e5b75505Sopenharmony_ci */ 390e5b75505Sopenharmony_cistruct fst_iface * 391e5b75505Sopenharmony_cifst_group_get_peer_other_connection(struct fst_iface *iface, 392e5b75505Sopenharmony_ci const u8 *peer_addr, u8 band_id, 393e5b75505Sopenharmony_ci u8 *other_peer_addr) 394e5b75505Sopenharmony_ci{ 395e5b75505Sopenharmony_ci struct fst_iface *other_iface; 396e5b75505Sopenharmony_ci 397e5b75505Sopenharmony_ci fst_printf(MSG_DEBUG, "%s: %s:" MACSTR ", %d", __func__, 398e5b75505Sopenharmony_ci fst_iface_get_name(iface), MAC2STR(peer_addr), band_id); 399e5b75505Sopenharmony_ci 400e5b75505Sopenharmony_ci /* 401e5b75505Sopenharmony_ci * Two search methods are used: 402e5b75505Sopenharmony_ci * 1. Use peer's MB IE on iface to extract peer's MAC address on 403e5b75505Sopenharmony_ci * "other" connection. Then check if such "other" connection exists. 404e5b75505Sopenharmony_ci * 2. Iterate peer database, examine each MB IE to see if it points to 405e5b75505Sopenharmony_ci * (iface, peer_addr) tuple 406e5b75505Sopenharmony_ci */ 407e5b75505Sopenharmony_ci 408e5b75505Sopenharmony_ci other_iface = fst_group_get_peer_other_connection_1(iface, peer_addr, 409e5b75505Sopenharmony_ci band_id, 410e5b75505Sopenharmony_ci other_peer_addr); 411e5b75505Sopenharmony_ci if (other_iface) { 412e5b75505Sopenharmony_ci fst_printf(MSG_DEBUG, "found by method #1. %s:" MACSTR, 413e5b75505Sopenharmony_ci fst_iface_get_name(other_iface), 414e5b75505Sopenharmony_ci MAC2STR(other_peer_addr)); 415e5b75505Sopenharmony_ci return other_iface; 416e5b75505Sopenharmony_ci } 417e5b75505Sopenharmony_ci 418e5b75505Sopenharmony_ci other_iface = fst_group_get_peer_other_connection_2(iface, peer_addr, 419e5b75505Sopenharmony_ci band_id, 420e5b75505Sopenharmony_ci other_peer_addr); 421e5b75505Sopenharmony_ci if (other_iface) { 422e5b75505Sopenharmony_ci fst_printf(MSG_DEBUG, "found by method #2. %s:" MACSTR, 423e5b75505Sopenharmony_ci fst_iface_get_name(other_iface), 424e5b75505Sopenharmony_ci MAC2STR(other_peer_addr)); 425e5b75505Sopenharmony_ci return other_iface; 426e5b75505Sopenharmony_ci } 427e5b75505Sopenharmony_ci 428e5b75505Sopenharmony_ci fst_printf(MSG_INFO, "%s: other connection not found", __func__); 429e5b75505Sopenharmony_ci return NULL; 430e5b75505Sopenharmony_ci} 431e5b75505Sopenharmony_ci 432e5b75505Sopenharmony_ci 433e5b75505Sopenharmony_cistruct fst_group * fst_group_create(const char *group_id) 434e5b75505Sopenharmony_ci{ 435e5b75505Sopenharmony_ci struct fst_group *g; 436e5b75505Sopenharmony_ci 437e5b75505Sopenharmony_ci g = os_zalloc(sizeof(*g)); 438e5b75505Sopenharmony_ci if (g == NULL) { 439e5b75505Sopenharmony_ci fst_printf(MSG_ERROR, "%s: Cannot alloc group", group_id); 440e5b75505Sopenharmony_ci return NULL; 441e5b75505Sopenharmony_ci } 442e5b75505Sopenharmony_ci 443e5b75505Sopenharmony_ci dl_list_init(&g->ifaces); 444e5b75505Sopenharmony_ci os_strlcpy(g->group_id, group_id, sizeof(g->group_id)); 445e5b75505Sopenharmony_ci 446e5b75505Sopenharmony_ci dl_list_add_tail(&fst_global_groups_list, &g->global_groups_lentry); 447e5b75505Sopenharmony_ci fst_printf_group(g, MSG_DEBUG, "instance created"); 448e5b75505Sopenharmony_ci 449e5b75505Sopenharmony_ci foreach_fst_ctrl_call(on_group_created, g); 450e5b75505Sopenharmony_ci 451e5b75505Sopenharmony_ci return g; 452e5b75505Sopenharmony_ci} 453e5b75505Sopenharmony_ci 454e5b75505Sopenharmony_ci 455e5b75505Sopenharmony_civoid fst_group_attach_iface(struct fst_group *g, struct fst_iface *i) 456e5b75505Sopenharmony_ci{ 457e5b75505Sopenharmony_ci struct dl_list *list = &g->ifaces; 458e5b75505Sopenharmony_ci struct fst_iface *f; 459e5b75505Sopenharmony_ci 460e5b75505Sopenharmony_ci /* 461e5b75505Sopenharmony_ci * Add new interface to the list. 462e5b75505Sopenharmony_ci * The list is sorted in descending order by priority to allow 463e5b75505Sopenharmony_ci * multiple MB IEs creation according to the spec (see 10.32 Multi-band 464e5b75505Sopenharmony_ci * operation, 10.32.1 General), as they should be ordered according to 465e5b75505Sopenharmony_ci * priorities. 466e5b75505Sopenharmony_ci */ 467e5b75505Sopenharmony_ci foreach_fst_group_iface(g, f) { 468e5b75505Sopenharmony_ci if (fst_iface_get_priority(f) < fst_iface_get_priority(i)) 469e5b75505Sopenharmony_ci break; 470e5b75505Sopenharmony_ci list = &f->group_lentry; 471e5b75505Sopenharmony_ci } 472e5b75505Sopenharmony_ci dl_list_add(list, &i->group_lentry); 473e5b75505Sopenharmony_ci} 474e5b75505Sopenharmony_ci 475e5b75505Sopenharmony_ci 476e5b75505Sopenharmony_civoid fst_group_detach_iface(struct fst_group *g, struct fst_iface *i) 477e5b75505Sopenharmony_ci{ 478e5b75505Sopenharmony_ci dl_list_del(&i->group_lentry); 479e5b75505Sopenharmony_ci} 480e5b75505Sopenharmony_ci 481e5b75505Sopenharmony_ci 482e5b75505Sopenharmony_civoid fst_group_delete(struct fst_group *group) 483e5b75505Sopenharmony_ci{ 484e5b75505Sopenharmony_ci struct fst_session *s; 485e5b75505Sopenharmony_ci 486e5b75505Sopenharmony_ci dl_list_del(&group->global_groups_lentry); 487e5b75505Sopenharmony_ci WPA_ASSERT(dl_list_empty(&group->ifaces)); 488e5b75505Sopenharmony_ci foreach_fst_ctrl_call(on_group_deleted, group); 489e5b75505Sopenharmony_ci fst_printf_group(group, MSG_DEBUG, "instance deleted"); 490e5b75505Sopenharmony_ci while ((s = fst_session_global_get_first_by_group(group)) != NULL) 491e5b75505Sopenharmony_ci fst_session_delete(s); 492e5b75505Sopenharmony_ci os_free(group); 493e5b75505Sopenharmony_ci} 494e5b75505Sopenharmony_ci 495e5b75505Sopenharmony_ci 496e5b75505Sopenharmony_ciBoolean fst_group_delete_if_empty(struct fst_group *group) 497e5b75505Sopenharmony_ci{ 498e5b75505Sopenharmony_ci Boolean is_empty = !fst_group_has_ifaces(group) && 499e5b75505Sopenharmony_ci !fst_session_global_get_first_by_group(group); 500e5b75505Sopenharmony_ci 501e5b75505Sopenharmony_ci if (is_empty) 502e5b75505Sopenharmony_ci fst_group_delete(group); 503e5b75505Sopenharmony_ci 504e5b75505Sopenharmony_ci return is_empty; 505e5b75505Sopenharmony_ci} 506e5b75505Sopenharmony_ci 507e5b75505Sopenharmony_ci 508e5b75505Sopenharmony_civoid fst_group_update_ie(struct fst_group *g) 509e5b75505Sopenharmony_ci{ 510e5b75505Sopenharmony_ci struct fst_iface *i; 511e5b75505Sopenharmony_ci 512e5b75505Sopenharmony_ci foreach_fst_group_iface(g, i) { 513e5b75505Sopenharmony_ci struct wpabuf *mbie = fst_group_create_mb_ie(g, i); 514e5b75505Sopenharmony_ci 515e5b75505Sopenharmony_ci if (!mbie) 516e5b75505Sopenharmony_ci fst_printf_iface(i, MSG_WARNING, "cannot create MB IE"); 517e5b75505Sopenharmony_ci 518e5b75505Sopenharmony_ci fst_iface_attach_mbie(i, mbie); 519e5b75505Sopenharmony_ci fst_iface_set_ies(i, mbie); 520e5b75505Sopenharmony_ci fst_printf_iface(i, MSG_DEBUG, "multi-band IE set to %p", mbie); 521e5b75505Sopenharmony_ci } 522e5b75505Sopenharmony_ci} 523