wlantest: Select correct TDLS context if multiple exists

Some corner cases may result in both directions of TDLS tracking context
existing. If that is the case, the incorrect one may end up getting
picked when figuring out which TK to use for decryption or fix
statistics counter to increment. Fix this by preferring the context that
has TDLS link up.

Signed-hostap: Jouni Malinen <j@w1.fi>
This commit is contained in:
Jouni Malinen 2013-05-26 21:46:54 +03:00
parent 4ac800db82
commit ace4e460e5

View File

@ -202,7 +202,7 @@ static void rx_data_bss_prot(struct wlantest *wt,
size_t dlen; size_t dlen;
int tid; int tid;
u8 pn[6], *rsc; u8 pn[6], *rsc;
struct wlantest_tdls *tdls = NULL; struct wlantest_tdls *tdls = NULL, *found;
const u8 *tk = NULL; const u8 *tk = NULL;
if (hdr->addr1[0] & 0x01) { if (hdr->addr1[0] & 0x01) {
@ -230,17 +230,24 @@ static void rx_data_bss_prot(struct wlantest *wt,
sta2 = sta_find(bss, hdr->addr1); sta2 = sta_find(bss, hdr->addr1);
if (sta == NULL || sta2 == NULL) if (sta == NULL || sta2 == NULL)
return; return;
found = NULL;
dl_list_for_each(tdls, &bss->tdls, struct wlantest_tdls, list) dl_list_for_each(tdls, &bss->tdls, struct wlantest_tdls, list)
{ {
if ((tdls->init == sta && tdls->resp == sta2) || if ((tdls->init == sta && tdls->resp == sta2) ||
(tdls->init == sta2 && tdls->resp == sta)) { (tdls->init == sta2 && tdls->resp == sta)) {
if (!tdls->link_up) found = tdls;
wpa_printf(MSG_DEBUG, "TDLS: Link not " if (tdls->link_up)
"up, but Data frame seen"); break;
tk = tdls->tpk.tk;
break;
} }
} }
if (found) {
if (!found->link_up)
add_note(wt, MSG_DEBUG,
"TDLS: Link not up, but Data "
"frame seen");
tk = found->tpk.tk;
tdls = found;
}
} }
if ((sta == NULL || if ((sta == NULL ||
(!sta->ptk_set && sta->pairwise_cipher != WPA_CIPHER_WEP40)) && (!sta->ptk_set && sta->pairwise_cipher != WPA_CIPHER_WEP40)) &&
@ -401,7 +408,7 @@ static struct wlantest_tdls * get_tdls(struct wlantest *wt, const u8 *bssid,
{ {
struct wlantest_bss *bss; struct wlantest_bss *bss;
struct wlantest_sta *sta1, *sta2; struct wlantest_sta *sta1, *sta2;
struct wlantest_tdls *tdls; struct wlantest_tdls *tdls, *found = NULL;
bss = bss_find(wt, bssid); bss = bss_find(wt, bssid);
if (bss == NULL) if (bss == NULL)
@ -415,11 +422,14 @@ static struct wlantest_tdls * get_tdls(struct wlantest *wt, const u8 *bssid,
dl_list_for_each(tdls, &bss->tdls, struct wlantest_tdls, list) { dl_list_for_each(tdls, &bss->tdls, struct wlantest_tdls, list) {
if ((tdls->init == sta1 && tdls->resp == sta2) || if ((tdls->init == sta1 && tdls->resp == sta2) ||
(tdls->init == sta2 && tdls->resp == sta1)) (tdls->init == sta2 && tdls->resp == sta1)) {
return tdls; found = tdls;
if (tdls->link_up)
break;
}
} }
return NULL; return found;
} }