|
Takashi Iwai |
b28a98 |
From fd17bf041b40e3dac705c4313854becbe07b7557 Mon Sep 17 00:00:00 2001
|
|
Takashi Iwai |
b28a98 |
From: Johannes Berg <johannes.berg@intel.com>
|
|
Takashi Iwai |
b28a98 |
Date: Tue, 28 Jun 2022 17:49:12 +0200
|
|
Takashi Iwai |
b28a98 |
Subject: [PATCH] wifi: mac80211: refactor elements parsing with parameter struct
|
|
Takashi Iwai |
b28a98 |
Git-commit: fd17bf041b40e3dac705c4313854becbe07b7557
|
|
Takashi Iwai |
b28a98 |
Patch-mainline: v6.0-rc1
|
|
Takashi Iwai |
b28a98 |
References: CVE-2022-42719 bsc#1204051
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
Refactor the element parsing into a version that has
|
|
Takashi Iwai |
b28a98 |
a parameter struct so we can add more parameters more
|
|
Takashi Iwai |
b28a98 |
easily in the future.
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
|
|
Takashi Iwai |
b28a98 |
Acked-by: Takashi Iwai <tiwai@suse.de>
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
---
|
|
Takashi Iwai |
b28a98 |
net/mac80211/ieee80211_i.h | 50 ++++++++++++++++++++++++++++----
|
|
Takashi Iwai |
b28a98 |
net/mac80211/util.c | 58 +++++++++++++++++++-------------------
|
|
Takashi Iwai |
b28a98 |
2 files changed, 74 insertions(+), 34 deletions(-)
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
diff --git a/net/mac80211/ieee80211_i.h b/net/mac80211/ieee80211_i.h
|
|
Takashi Iwai |
b28a98 |
index a0743c78d171..2cf13ea4c9f7 100644
|
|
Takashi Iwai |
b28a98 |
--- a/net/mac80211/ieee80211_i.h
|
|
Takashi Iwai |
b28a98 |
+++ b/net/mac80211/ieee80211_i.h
|
|
Takashi Iwai |
b28a98 |
@@ -2138,11 +2138,51 @@ static inline void ieee80211_tx_skb(struct ieee80211_sub_if_data *sdata,
|
|
Takashi Iwai |
b28a98 |
ieee80211_tx_skb_tid(sdata, skb, 7);
|
|
Takashi Iwai |
b28a98 |
}
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
-struct ieee802_11_elems *ieee802_11_parse_elems_crc(const u8 *start, size_t len,
|
|
Takashi Iwai |
b28a98 |
- bool action,
|
|
Takashi Iwai |
b28a98 |
- u64 filter, u32 crc,
|
|
Takashi Iwai |
b28a98 |
- const u8 *transmitter_bssid,
|
|
Takashi Iwai |
b28a98 |
- const u8 *bss_bssid);
|
|
Takashi Iwai |
b28a98 |
+/**
|
|
Takashi Iwai |
b28a98 |
+ * struct ieee80211_elems_parse_params - element parsing parameters
|
|
Takashi Iwai |
b28a98 |
+ * @start: pointer to the elements
|
|
Takashi Iwai |
b28a98 |
+ * @len: length of the elements
|
|
Takashi Iwai |
b28a98 |
+ * @action: %true if the elements came from an action frame
|
|
Takashi Iwai |
b28a98 |
+ * @filter: bitmap of element IDs to filter out while calculating
|
|
Takashi Iwai |
b28a98 |
+ * the element CRC
|
|
Takashi Iwai |
b28a98 |
+ * @crc: CRC starting value
|
|
Takashi Iwai |
b28a98 |
+ * @transmitter_bssid: transmitter BSSID to parse the multi-BSSID
|
|
Takashi Iwai |
b28a98 |
+ * element
|
|
Takashi Iwai |
b28a98 |
+ * @bss_bssid: BSSID of the BSS we want to obtain elements for
|
|
Takashi Iwai |
b28a98 |
+ * when parsing the multi-BSSID element
|
|
Takashi Iwai |
b28a98 |
+ */
|
|
Takashi Iwai |
b28a98 |
+struct ieee80211_elems_parse_params {
|
|
Takashi Iwai |
b28a98 |
+ const u8 *start;
|
|
Takashi Iwai |
b28a98 |
+ size_t len;
|
|
Takashi Iwai |
b28a98 |
+ bool action;
|
|
Takashi Iwai |
b28a98 |
+ u64 filter;
|
|
Takashi Iwai |
b28a98 |
+ u32 crc;
|
|
Takashi Iwai |
b28a98 |
+ const u8 *transmitter_bssid;
|
|
Takashi Iwai |
b28a98 |
+ const u8 *bss_bssid;
|
|
Takashi Iwai |
b28a98 |
+};
|
|
Takashi Iwai |
b28a98 |
+
|
|
Takashi Iwai |
b28a98 |
+struct ieee802_11_elems *
|
|
Takashi Iwai |
b28a98 |
+ieee802_11_parse_elems_full(struct ieee80211_elems_parse_params *params);
|
|
Takashi Iwai |
b28a98 |
+
|
|
Takashi Iwai |
b28a98 |
+static inline struct ieee802_11_elems *
|
|
Takashi Iwai |
b28a98 |
+ieee802_11_parse_elems_crc(const u8 *start, size_t len, bool action,
|
|
Takashi Iwai |
b28a98 |
+ u64 filter, u32 crc,
|
|
Takashi Iwai |
b28a98 |
+ const u8 *transmitter_bssid,
|
|
Takashi Iwai |
b28a98 |
+ const u8 *bss_bssid)
|
|
Takashi Iwai |
b28a98 |
+{
|
|
Takashi Iwai |
b28a98 |
+ struct ieee80211_elems_parse_params params = {
|
|
Takashi Iwai |
b28a98 |
+ .start = start,
|
|
Takashi Iwai |
b28a98 |
+ .len = len,
|
|
Takashi Iwai |
b28a98 |
+ .action = action,
|
|
Takashi Iwai |
b28a98 |
+ .filter = filter,
|
|
Takashi Iwai |
b28a98 |
+ .crc = crc,
|
|
Takashi Iwai |
b28a98 |
+ .transmitter_bssid = transmitter_bssid,
|
|
Takashi Iwai |
b28a98 |
+ .bss_bssid = bss_bssid,
|
|
Takashi Iwai |
b28a98 |
+ };
|
|
Takashi Iwai |
b28a98 |
+
|
|
Takashi Iwai |
b28a98 |
+ return ieee802_11_parse_elems_full(¶ms);
|
|
Takashi Iwai |
b28a98 |
+}
|
|
Takashi Iwai |
b28a98 |
+
|
|
Takashi Iwai |
b28a98 |
static inline struct ieee802_11_elems *
|
|
Takashi Iwai |
b28a98 |
ieee802_11_parse_elems(const u8 *start, size_t len, bool action,
|
|
Takashi Iwai |
b28a98 |
const u8 *transmitter_bssid,
|
|
Takashi Iwai |
b28a98 |
diff --git a/net/mac80211/util.c b/net/mac80211/util.c
|
|
Takashi Iwai |
b28a98 |
index 2b2c8e1472f3..0ff09c639c50 100644
|
|
Takashi Iwai |
b28a98 |
--- a/net/mac80211/util.c
|
|
Takashi Iwai |
b28a98 |
+++ b/net/mac80211/util.c
|
|
Takashi Iwai |
b28a98 |
@@ -1026,19 +1026,19 @@ static void ieee80211_parse_extension_element(u32 *crc,
|
|
Takashi Iwai |
b28a98 |
}
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
static u32
|
|
Takashi Iwai |
b28a98 |
-_ieee802_11_parse_elems_crc(const u8 *start, size_t len, bool action,
|
|
Takashi Iwai |
b28a98 |
- struct ieee802_11_elems *elems,
|
|
Takashi Iwai |
b28a98 |
- u64 filter, u32 crc,
|
|
Takashi Iwai |
b28a98 |
- const struct element *check_inherit)
|
|
Takashi Iwai |
b28a98 |
+_ieee802_11_parse_elems_full(struct ieee80211_elems_parse_params *params,
|
|
Takashi Iwai |
b28a98 |
+ struct ieee802_11_elems *elems,
|
|
Takashi Iwai |
b28a98 |
+ const struct element *check_inherit)
|
|
Takashi Iwai |
b28a98 |
{
|
|
Takashi Iwai |
b28a98 |
const struct element *elem;
|
|
Takashi Iwai |
b28a98 |
- bool calc_crc = filter != 0;
|
|
Takashi Iwai |
b28a98 |
+ bool calc_crc = params->filter != 0;
|
|
Takashi Iwai |
b28a98 |
DECLARE_BITMAP(seen_elems, 256);
|
|
Takashi Iwai |
b28a98 |
+ u32 crc = params->crc;
|
|
Takashi Iwai |
b28a98 |
const u8 *ie;
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
bitmap_zero(seen_elems, 256);
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
- for_each_element(elem, start, len) {
|
|
Takashi Iwai |
b28a98 |
+ for_each_element(elem, params->start, params->len) {
|
|
Takashi Iwai |
b28a98 |
bool elem_parse_failed;
|
|
Takashi Iwai |
b28a98 |
u8 id = elem->id;
|
|
Takashi Iwai |
b28a98 |
u8 elen = elem->datalen;
|
|
Takashi Iwai |
b28a98 |
@@ -1101,7 +1101,7 @@ _ieee802_11_parse_elems_crc(const u8 *start, size_t len, bool action,
|
|
Takashi Iwai |
b28a98 |
break;
|
|
Takashi Iwai |
b28a98 |
}
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
- if (calc_crc && id < 64 && (filter & (1ULL << id)))
|
|
Takashi Iwai |
b28a98 |
+ if (calc_crc && id < 64 && (params->filter & (1ULL << id)))
|
|
Takashi Iwai |
b28a98 |
crc = crc32_be(crc, pos - 2, elen + 2);
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
elem_parse_failed = false;
|
|
Takashi Iwai |
b28a98 |
@@ -1282,7 +1282,7 @@ _ieee802_11_parse_elems_crc(const u8 *start, size_t len, bool action,
|
|
Takashi Iwai |
b28a98 |
elems->mesh_chansw_params_ie = (void *)pos;
|
|
Takashi Iwai |
b28a98 |
break;
|
|
Takashi Iwai |
b28a98 |
case WLAN_EID_WIDE_BW_CHANNEL_SWITCH:
|
|
Takashi Iwai |
b28a98 |
- if (!action ||
|
|
Takashi Iwai |
b28a98 |
+ if (!params->action ||
|
|
Takashi Iwai |
b28a98 |
elen < sizeof(*elems->wide_bw_chansw_ie)) {
|
|
Takashi Iwai |
b28a98 |
elem_parse_failed = true;
|
|
Takashi Iwai |
b28a98 |
break;
|
|
Takashi Iwai |
b28a98 |
@@ -1290,7 +1290,7 @@ _ieee802_11_parse_elems_crc(const u8 *start, size_t len, bool action,
|
|
Takashi Iwai |
b28a98 |
elems->wide_bw_chansw_ie = (void *)pos;
|
|
Takashi Iwai |
b28a98 |
break;
|
|
Takashi Iwai |
b28a98 |
case WLAN_EID_CHANNEL_SWITCH_WRAPPER:
|
|
Takashi Iwai |
b28a98 |
- if (action) {
|
|
Takashi Iwai |
b28a98 |
+ if (params->action) {
|
|
Takashi Iwai |
b28a98 |
elem_parse_failed = true;
|
|
Takashi Iwai |
b28a98 |
break;
|
|
Takashi Iwai |
b28a98 |
}
|
|
Takashi Iwai |
b28a98 |
@@ -1417,7 +1417,7 @@ _ieee802_11_parse_elems_crc(const u8 *start, size_t len, bool action,
|
|
Takashi Iwai |
b28a98 |
__set_bit(id, seen_elems);
|
|
Takashi Iwai |
b28a98 |
}
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
- if (!for_each_element_completed(elem, start, len))
|
|
Takashi Iwai |
b28a98 |
+ if (!for_each_element_completed(elem, params->start, params->len))
|
|
Takashi Iwai |
b28a98 |
elems->parse_error = true;
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
return crc;
|
|
Takashi Iwai |
b28a98 |
@@ -1491,11 +1491,8 @@ static size_t ieee802_11_find_bssid_profile(const u8 *start, size_t len,
|
|
Takashi Iwai |
b28a98 |
return found ? profile_len : 0;
|
|
Takashi Iwai |
b28a98 |
}
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
-struct ieee802_11_elems *ieee802_11_parse_elems_crc(const u8 *start, size_t len,
|
|
Takashi Iwai |
b28a98 |
- bool action, u64 filter,
|
|
Takashi Iwai |
b28a98 |
- u32 crc,
|
|
Takashi Iwai |
b28a98 |
- const u8 *transmitter_bssid,
|
|
Takashi Iwai |
b28a98 |
- const u8 *bss_bssid)
|
|
Takashi Iwai |
b28a98 |
+struct ieee802_11_elems *
|
|
Takashi Iwai |
b28a98 |
+ieee802_11_parse_elems_full(struct ieee80211_elems_parse_params *params)
|
|
Takashi Iwai |
b28a98 |
{
|
|
Takashi Iwai |
b28a98 |
struct ieee802_11_elems *elems;
|
|
Takashi Iwai |
b28a98 |
const struct element *non_inherit = NULL;
|
|
Takashi Iwai |
b28a98 |
@@ -1505,15 +1502,16 @@ struct ieee802_11_elems *ieee802_11_parse_elems_crc(const u8 *start, size_t len,
|
|
Takashi Iwai |
b28a98 |
elems = kzalloc(sizeof(*elems), GFP_ATOMIC);
|
|
Takashi Iwai |
b28a98 |
if (!elems)
|
|
Takashi Iwai |
b28a98 |
return NULL;
|
|
Takashi Iwai |
b28a98 |
- elems->ie_start = start;
|
|
Takashi Iwai |
b28a98 |
- elems->total_len = len;
|
|
Takashi Iwai |
b28a98 |
+ elems->ie_start = params->start;
|
|
Takashi Iwai |
b28a98 |
+ elems->total_len = params->len;
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
- nontransmitted_profile = kmalloc(len, GFP_ATOMIC);
|
|
Takashi Iwai |
b28a98 |
+ nontransmitted_profile = kmalloc(params->len, GFP_ATOMIC);
|
|
Takashi Iwai |
b28a98 |
if (nontransmitted_profile) {
|
|
Takashi Iwai |
b28a98 |
nontransmitted_profile_len =
|
|
Takashi Iwai |
b28a98 |
- ieee802_11_find_bssid_profile(start, len, elems,
|
|
Takashi Iwai |
b28a98 |
- transmitter_bssid,
|
|
Takashi Iwai |
b28a98 |
- bss_bssid,
|
|
Takashi Iwai |
b28a98 |
+ ieee802_11_find_bssid_profile(params->start, params->len,
|
|
Takashi Iwai |
b28a98 |
+ elems,
|
|
Takashi Iwai |
b28a98 |
+ params->transmitter_bssid,
|
|
Takashi Iwai |
b28a98 |
+ params->bss_bssid,
|
|
Takashi Iwai |
b28a98 |
nontransmitted_profile);
|
|
Takashi Iwai |
b28a98 |
non_inherit =
|
|
Takashi Iwai |
b28a98 |
cfg80211_find_ext_elem(WLAN_EID_EXT_NON_INHERITANCE,
|
|
Takashi Iwai |
b28a98 |
@@ -1521,14 +1519,18 @@ struct ieee802_11_elems *ieee802_11_parse_elems_crc(const u8 *start, size_t len,
|
|
Takashi Iwai |
b28a98 |
nontransmitted_profile_len);
|
|
Takashi Iwai |
b28a98 |
}
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
- crc = _ieee802_11_parse_elems_crc(start, len, action, elems, filter,
|
|
Takashi Iwai |
b28a98 |
- crc, non_inherit);
|
|
Takashi Iwai |
b28a98 |
+ elems->crc = _ieee802_11_parse_elems_full(params, elems, non_inherit);
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
/* Override with nontransmitted profile, if found */
|
|
Takashi Iwai |
b28a98 |
- if (nontransmitted_profile_len)
|
|
Takashi Iwai |
b28a98 |
- _ieee802_11_parse_elems_crc(nontransmitted_profile,
|
|
Takashi Iwai |
b28a98 |
- nontransmitted_profile_len,
|
|
Takashi Iwai |
b28a98 |
- action, elems, 0, 0, NULL);
|
|
Takashi Iwai |
b28a98 |
+ if (nontransmitted_profile_len) {
|
|
Takashi Iwai |
b28a98 |
+ struct ieee80211_elems_parse_params sub = {
|
|
Takashi Iwai |
b28a98 |
+ .start = nontransmitted_profile,
|
|
Takashi Iwai |
b28a98 |
+ .len = nontransmitted_profile_len,
|
|
Takashi Iwai |
b28a98 |
+ .action = params->action,
|
|
Takashi Iwai |
b28a98 |
+ };
|
|
Takashi Iwai |
b28a98 |
+
|
|
Takashi Iwai |
b28a98 |
+ _ieee802_11_parse_elems_full(&sub, elems, NULL);
|
|
Takashi Iwai |
b28a98 |
+ }
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
if (elems->tim && !elems->parse_error) {
|
|
Takashi Iwai |
b28a98 |
const struct ieee80211_tim_ie *tim_ie = elems->tim;
|
|
Takashi Iwai |
b28a98 |
@@ -1550,8 +1552,6 @@ struct ieee802_11_elems *ieee802_11_parse_elems_crc(const u8 *start, size_t len,
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
kfree(nontransmitted_profile);
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
- elems->crc = crc;
|
|
Takashi Iwai |
b28a98 |
-
|
|
Takashi Iwai |
b28a98 |
return elems;
|
|
Takashi Iwai |
b28a98 |
}
|
|
Takashi Iwai |
b28a98 |
|
|
Takashi Iwai |
b28a98 |
--
|
|
Takashi Iwai |
b28a98 |
2.35.3
|
|
Takashi Iwai |
b28a98 |
|