Commit 5e616ad2 authored by Felix Fietkau's avatar Felix Fietkau

mt76: fix wcid allocation issues

mt76 core uses ffs() to find the next free bit. This works well for 32 bit
architectures where BITS_PER_LONG is 32. ffs only checks 32 bit values, so
allocation fails on 64 bit architectures.
Additionally, the wcid mask array was too small in cases where the array
was not a multiple of BITS_PER_LONG.
Fix this by making the wcid mask array u32 instead and use DIV_ROUND_UP
for the size, just in case we ever bump it to a value that's not a multiple
of 32.
Reported-by: default avatarRyder Lee <ryder.lee@mediatek.com>
Signed-off-by: default avatarFelix Fietkau <nbd@nbd.name>
parent ec2bb3a5
...@@ -537,8 +537,8 @@ struct mt76_dev { ...@@ -537,8 +537,8 @@ struct mt76_dev {
wait_queue_head_t tx_wait; wait_queue_head_t tx_wait;
struct sk_buff_head status_list; struct sk_buff_head status_list;
unsigned long wcid_mask[MT76_N_WCIDS / BITS_PER_LONG]; u32 wcid_mask[DIV_ROUND_UP(MT76_N_WCIDS, 32)];
unsigned long wcid_phy_mask[MT76_N_WCIDS / BITS_PER_LONG]; u32 wcid_phy_mask[DIV_ROUND_UP(MT76_N_WCIDS, 32)];
struct mt76_wcid global_wcid; struct mt76_wcid global_wcid;
struct mt76_wcid __rcu *wcid[MT76_N_WCIDS]; struct mt76_wcid __rcu *wcid[MT76_N_WCIDS];
......
...@@ -42,17 +42,17 @@ bool __mt76_poll_msec(struct mt76_dev *dev, u32 offset, u32 mask, u32 val, ...@@ -42,17 +42,17 @@ bool __mt76_poll_msec(struct mt76_dev *dev, u32 offset, u32 mask, u32 val,
} }
EXPORT_SYMBOL_GPL(__mt76_poll_msec); EXPORT_SYMBOL_GPL(__mt76_poll_msec);
int mt76_wcid_alloc(unsigned long *mask, int size) int mt76_wcid_alloc(u32 *mask, int size)
{ {
int i, idx = 0, cur; int i, idx = 0, cur;
for (i = 0; i < DIV_ROUND_UP(size, BITS_PER_LONG); i++) { for (i = 0; i < DIV_ROUND_UP(size, 32); i++) {
idx = ffs(~mask[i]); idx = ffs(~mask[i]);
if (!idx) if (!idx)
continue; continue;
idx--; idx--;
cur = i * BITS_PER_LONG + idx; cur = i * 32 + idx;
if (cur >= size) if (cur >= size)
break; break;
...@@ -74,13 +74,13 @@ int mt76_get_min_avg_rssi(struct mt76_dev *dev, bool ext_phy) ...@@ -74,13 +74,13 @@ int mt76_get_min_avg_rssi(struct mt76_dev *dev, bool ext_phy)
rcu_read_lock(); rcu_read_lock();
for (i = 0; i < ARRAY_SIZE(dev->wcid_mask); i++) { for (i = 0; i < ARRAY_SIZE(dev->wcid_mask); i++) {
unsigned long mask = dev->wcid_mask[i]; u32 mask = dev->wcid_mask[i];
unsigned long phy_mask = dev->wcid_phy_mask[i]; u32 phy_mask = dev->wcid_phy_mask[i];
if (!mask) if (!mask)
continue; continue;
for (j = i * BITS_PER_LONG; mask; j++, mask >>= 1, phy_mask >>= 1) { for (j = i * 32; mask; j++, mask >>= 1, phy_mask >>= 1) {
if (!(mask & 1)) if (!(mask & 1))
continue; continue;
......
...@@ -14,24 +14,24 @@ ...@@ -14,24 +14,24 @@
#define MT76_INCR(_var, _size) \ #define MT76_INCR(_var, _size) \
(_var = (((_var) + 1) % (_size))) (_var = (((_var) + 1) % (_size)))
int mt76_wcid_alloc(unsigned long *mask, int size); int mt76_wcid_alloc(u32 *mask, int size);
static inline bool static inline bool
mt76_wcid_mask_test(unsigned long *mask, int idx) mt76_wcid_mask_test(u32 *mask, int idx)
{ {
return mask[idx / BITS_PER_LONG] & BIT(idx % BITS_PER_LONG); return mask[idx / 32] & BIT(idx % 32);
} }
static inline void static inline void
mt76_wcid_mask_set(unsigned long *mask, int idx) mt76_wcid_mask_set(u32 *mask, int idx)
{ {
mask[idx / BITS_PER_LONG] |= BIT(idx % BITS_PER_LONG); mask[idx / 32] |= BIT(idx % 32);
} }
static inline void static inline void
mt76_wcid_mask_clear(unsigned long *mask, int idx) mt76_wcid_mask_clear(u32 *mask, int idx)
{ {
mask[idx / BITS_PER_LONG] &= ~BIT(idx % BITS_PER_LONG); mask[idx / 32] &= ~BIT(idx % 32);
} }
static inline void static inline void
......
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