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(&params);
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