mt76: mt7663s: fix possible quota leak in mt7663s_refill_sched_quota
authorLorenzo Bianconi <lorenzo@kernel.org>
Sat, 5 Sep 2020 09:26:05 +0000 (11:26 +0200)
committerFelix Fietkau <nbd@nbd.name>
Thu, 24 Sep 2020 16:10:19 +0000 (18:10 +0200)
Look just at reported quota since the hw sporadically reports mcu tx
quota without setting WHIER_TX_DONE_INT_EN bit

Tested-by: Sean Wang <sean.wang@mediatek.com>
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
Signed-off-by: Felix Fietkau <nbd@nbd.name>
drivers/net/wireless/mediatek/mt76/mt7615/sdio_txrx.c

index 4754b3a..4033fe4 100644 (file)
 #include "sdio.h"
 #include "mac.h"
 
-static void mt7663s_refill_sched_quota(struct mt76_dev *dev, u32 *data)
+static int mt7663s_refill_sched_quota(struct mt76_dev *dev, u32 *data)
 {
+       u32 ple_ac_data_quota[] = {
+               FIELD_GET(TXQ_CNT_L, data[4]), /* VO */
+               FIELD_GET(TXQ_CNT_H, data[3]), /* VI */
+               FIELD_GET(TXQ_CNT_L, data[3]), /* BE */
+               FIELD_GET(TXQ_CNT_H, data[2]), /* BK */
+       };
+       u32 pse_ac_data_quota[] = {
+               FIELD_GET(TXQ_CNT_H, data[1]), /* VO */
+               FIELD_GET(TXQ_CNT_L, data[1]), /* VI */
+               FIELD_GET(TXQ_CNT_H, data[0]), /* BE */
+               FIELD_GET(TXQ_CNT_L, data[0]), /* BK */
+       };
+       u32 pse_mcu_quota = FIELD_GET(TXQ_CNT_L, data[2]);
+       u32 pse_data_quota = 0, ple_data_quota = 0;
        struct mt76_sdio *sdio = &dev->sdio;
+       int i;
+
+       for (i = 0; i < ARRAY_SIZE(pse_ac_data_quota); i++) {
+               pse_data_quota += pse_ac_data_quota[i];
+               ple_data_quota += ple_ac_data_quota[i];
+       }
+
+       if (!pse_data_quota && !ple_data_quota && !pse_mcu_quota)
+               return 0;
 
        mutex_lock(&sdio->sched.lock);
-       sdio->sched.pse_data_quota += FIELD_GET(TXQ_CNT_L, data[0]) + /* BK */
-                                     FIELD_GET(TXQ_CNT_H, data[0]) + /* BE */
-                                     FIELD_GET(TXQ_CNT_L, data[1]) + /* VI */
-                                     FIELD_GET(TXQ_CNT_H, data[1]);  /* VO */
-       sdio->sched.ple_data_quota += FIELD_GET(TXQ_CNT_H, data[2]) + /* BK */
-                                     FIELD_GET(TXQ_CNT_L, data[3]) + /* BE */
-                                     FIELD_GET(TXQ_CNT_H, data[3]) + /* VI */
-                                     FIELD_GET(TXQ_CNT_L, data[4]);  /* VO */
-       sdio->sched.pse_mcu_quota += FIELD_GET(TXQ_CNT_L, data[2]);
+       sdio->sched.pse_mcu_quota += pse_mcu_quota;
+       sdio->sched.pse_data_quota += pse_data_quota;
+       sdio->sched.ple_data_quota += ple_data_quota;
        mutex_unlock(&sdio->sched.lock);
+
+       return pse_data_quota + ple_data_quota + pse_mcu_quota;
 }
 
 static struct sk_buff *mt7663s_build_rx_skb(void *data, int data_len,
@@ -259,10 +278,8 @@ void mt7663s_rx_work(struct work_struct *work)
                }
        }
 
-       if (intr->isr & WHIER_TX_DONE_INT_EN) {
-               mt7663s_refill_sched_quota(dev, intr->tx.wtqcr);
+       if (mt7663s_refill_sched_quota(dev, intr->tx.wtqcr))
                queue_work(sdio->txrx_wq, &sdio->tx.xmit_work);
-       }
 
        if (nframes) {
                queue_work(sdio->txrx_wq, &sdio->rx.recv_work);