Commit 838c7b8f authored by Kees Cook's avatar Kees Cook Committed by Johannes Berg

wifi: nl80211: Avoid address calculations via out of bounds array indexing

Before request->channels[] can be used, request->n_channels must be set.
Additionally, address calculations for memory after the "channels" array
need to be calculated from the allocation base ("request") rather than
via the first "out of bounds" index of "channels", otherwise run-time
bounds checking will throw a warning.
Reported-by: default avatarNathan Chancellor <nathan@kernel.org>
Fixes: e3eac9f3 ("wifi: cfg80211: Annotate struct cfg80211_scan_request with __counted_by")
Signed-off-by: default avatarKees Cook <keescook@chromium.org>
Tested-by: default avatarNathan Chancellor <nathan@kernel.org>
Link: https://msgid.link/20240424220057.work.819-kees@kernel.orgSigned-off-by: default avatarJohannes Berg <johannes.berg@intel.com>
parent 3d913719
...@@ -9162,6 +9162,7 @@ static int nl80211_trigger_scan(struct sk_buff *skb, struct genl_info *info) ...@@ -9162,6 +9162,7 @@ static int nl80211_trigger_scan(struct sk_buff *skb, struct genl_info *info)
struct wiphy *wiphy; struct wiphy *wiphy;
int err, tmp, n_ssids = 0, n_channels, i; int err, tmp, n_ssids = 0, n_channels, i;
size_t ie_len, size; size_t ie_len, size;
size_t ssids_offset, ie_offset;
wiphy = &rdev->wiphy; wiphy = &rdev->wiphy;
...@@ -9207,21 +9208,20 @@ static int nl80211_trigger_scan(struct sk_buff *skb, struct genl_info *info) ...@@ -9207,21 +9208,20 @@ static int nl80211_trigger_scan(struct sk_buff *skb, struct genl_info *info)
return -EINVAL; return -EINVAL;
size = struct_size(request, channels, n_channels); size = struct_size(request, channels, n_channels);
ssids_offset = size;
size = size_add(size, array_size(sizeof(*request->ssids), n_ssids)); size = size_add(size, array_size(sizeof(*request->ssids), n_ssids));
ie_offset = size;
size = size_add(size, ie_len); size = size_add(size, ie_len);
request = kzalloc(size, GFP_KERNEL); request = kzalloc(size, GFP_KERNEL);
if (!request) if (!request)
return -ENOMEM; return -ENOMEM;
request->n_channels = n_channels;
if (n_ssids) if (n_ssids)
request->ssids = (void *)&request->channels[n_channels]; request->ssids = (void *)request + ssids_offset;
request->n_ssids = n_ssids; request->n_ssids = n_ssids;
if (ie_len) { if (ie_len)
if (n_ssids) request->ie = (void *)request + ie_offset;
request->ie = (void *)(request->ssids + n_ssids);
else
request->ie = (void *)(request->channels + n_channels);
}
i = 0; i = 0;
if (scan_freqs) { if (scan_freqs) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment