|
@@ -109,6 +109,8 @@ static void pmksa_cache_set_expiration(struct rsn_pmksa_cache *pmksa)
|
|
|
* @pmksa: Pointer to PMKSA cache data from pmksa_cache_init()
|
|
|
* @pmk: The new pairwise master key
|
|
|
* @pmk_len: PMK length in bytes, usually PMK_LEN (32)
|
|
|
+ * @kck: Key confirmation key or %NULL if not yet derived
|
|
|
+ * @kck_len: KCK length in bytes
|
|
|
* @aa: Authenticator address
|
|
|
* @spa: Supplicant address
|
|
|
* @network_ctx: Network configuration context for this PMK
|
|
@@ -122,6 +124,7 @@ static void pmksa_cache_set_expiration(struct rsn_pmksa_cache *pmksa)
|
|
|
*/
|
|
|
struct rsn_pmksa_cache_entry *
|
|
|
pmksa_cache_add(struct rsn_pmksa_cache *pmksa, const u8 *pmk, size_t pmk_len,
|
|
|
+ const u8 *kck, size_t kck_len,
|
|
|
const u8 *aa, const u8 *spa, void *network_ctx, int akmp)
|
|
|
{
|
|
|
struct rsn_pmksa_cache_entry *entry, *pos, *prev;
|
|
@@ -130,13 +133,19 @@ pmksa_cache_add(struct rsn_pmksa_cache *pmksa, const u8 *pmk, size_t pmk_len,
|
|
|
if (pmk_len > PMK_LEN)
|
|
|
return NULL;
|
|
|
|
|
|
+ if (wpa_key_mgmt_suite_b(akmp) && !kck)
|
|
|
+ return NULL;
|
|
|
+
|
|
|
entry = os_zalloc(sizeof(*entry));
|
|
|
if (entry == NULL)
|
|
|
return NULL;
|
|
|
os_memcpy(entry->pmk, pmk, pmk_len);
|
|
|
entry->pmk_len = pmk_len;
|
|
|
- rsn_pmkid(pmk, pmk_len, aa, spa, entry->pmkid,
|
|
|
- wpa_key_mgmt_sha256(akmp));
|
|
|
+ if (wpa_key_mgmt_suite_b(akmp))
|
|
|
+ rsn_pmkid_suite_b(kck, kck_len, aa, spa, entry->pmkid);
|
|
|
+ else
|
|
|
+ rsn_pmkid(pmk, pmk_len, aa, spa, entry->pmkid,
|
|
|
+ wpa_key_mgmt_sha256(akmp));
|
|
|
os_get_reltime(&now);
|
|
|
entry->expiration = now.sec + pmksa->sm->dot11RSNAConfigPMKLifetime;
|
|
|
entry->reauth_time = now.sec + pmksa->sm->dot11RSNAConfigPMKLifetime *
|
|
@@ -333,6 +342,7 @@ pmksa_cache_clone_entry(struct rsn_pmksa_cache *pmksa,
|
|
|
struct rsn_pmksa_cache_entry *new_entry;
|
|
|
|
|
|
new_entry = pmksa_cache_add(pmksa, old_entry->pmk, old_entry->pmk_len,
|
|
|
+ NULL, 0,
|
|
|
aa, pmksa->sm->own_addr,
|
|
|
old_entry->network_ctx, old_entry->akmp);
|
|
|
if (new_entry == NULL)
|