mt76: use mac80211 txq scheduling
Performance improvement and preparation for adding airtime fairness support Signed-off-by: Felix Fietkau <nbd@nbd.name>
This commit is contained in:
parent
d908d4ec4d
commit
90fdc1717b
5 changed files with 68 additions and 62 deletions
|
@ -184,9 +184,7 @@ mt76_dma_tx_cleanup(struct mt76_dev *dev, enum mt76_txq_id qid, bool flush)
|
||||||
last = readl(&q->regs->dma_idx);
|
last = readl(&q->regs->dma_idx);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!flush)
|
if (flush)
|
||||||
mt76_txq_schedule(dev, sq);
|
|
||||||
else
|
|
||||||
mt76_dma_sync_idx(dev, q);
|
mt76_dma_sync_idx(dev, q);
|
||||||
|
|
||||||
wake = wake && q->stopped &&
|
wake = wake && q->stopped &&
|
||||||
|
@ -199,6 +197,8 @@ mt76_dma_tx_cleanup(struct mt76_dev *dev, enum mt76_txq_id qid, bool flush)
|
||||||
|
|
||||||
spin_unlock_bh(&q->lock);
|
spin_unlock_bh(&q->lock);
|
||||||
|
|
||||||
|
if (!flush)
|
||||||
|
mt76_txq_schedule(dev, qid);
|
||||||
if (wake)
|
if (wake)
|
||||||
ieee80211_wake_queue(dev->hw, qid);
|
ieee80211_wake_queue(dev->hw, qid);
|
||||||
}
|
}
|
||||||
|
|
|
@ -568,6 +568,7 @@ mt76_check_sta(struct mt76_dev *dev, struct sk_buff *skb)
|
||||||
struct ieee80211_sta *sta;
|
struct ieee80211_sta *sta;
|
||||||
struct mt76_wcid *wcid = status->wcid;
|
struct mt76_wcid *wcid = status->wcid;
|
||||||
bool ps;
|
bool ps;
|
||||||
|
int i;
|
||||||
|
|
||||||
if (ieee80211_is_pspoll(hdr->frame_control) && !wcid) {
|
if (ieee80211_is_pspoll(hdr->frame_control) && !wcid) {
|
||||||
sta = ieee80211_find_sta_by_ifaddr(dev->hw, hdr->addr2, NULL);
|
sta = ieee80211_find_sta_by_ifaddr(dev->hw, hdr->addr2, NULL);
|
||||||
|
@ -614,6 +615,20 @@ mt76_check_sta(struct mt76_dev *dev, struct sk_buff *skb)
|
||||||
|
|
||||||
dev->drv->sta_ps(dev, sta, ps);
|
dev->drv->sta_ps(dev, sta, ps);
|
||||||
ieee80211_sta_ps_transition(sta, ps);
|
ieee80211_sta_ps_transition(sta, ps);
|
||||||
|
|
||||||
|
if (ps)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (i = 0; i < ARRAY_SIZE(sta->txq); i++) {
|
||||||
|
struct mt76_txq *mtxq;
|
||||||
|
|
||||||
|
if (!sta->txq[i])
|
||||||
|
continue;
|
||||||
|
|
||||||
|
mtxq = (struct mt76_txq *) sta->txq[i]->drv_priv;
|
||||||
|
if (!skb_queue_empty(&mtxq->retry_q))
|
||||||
|
ieee80211_schedule_txq(dev->hw, sta->txq[i]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void mt76_rx_complete(struct mt76_dev *dev, struct sk_buff_head *frames,
|
void mt76_rx_complete(struct mt76_dev *dev, struct sk_buff_head *frames,
|
||||||
|
|
|
@ -216,7 +216,6 @@ struct mt76_wcid {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mt76_txq {
|
struct mt76_txq {
|
||||||
struct list_head list;
|
|
||||||
struct mt76_sw_queue *swq;
|
struct mt76_sw_queue *swq;
|
||||||
struct mt76_wcid *wcid;
|
struct mt76_wcid *wcid;
|
||||||
|
|
||||||
|
@ -676,7 +675,7 @@ void mt76_txq_remove(struct mt76_dev *dev, struct ieee80211_txq *txq);
|
||||||
void mt76_wake_tx_queue(struct ieee80211_hw *hw, struct ieee80211_txq *txq);
|
void mt76_wake_tx_queue(struct ieee80211_hw *hw, struct ieee80211_txq *txq);
|
||||||
void mt76_stop_tx_queues(struct mt76_dev *dev, struct ieee80211_sta *sta,
|
void mt76_stop_tx_queues(struct mt76_dev *dev, struct ieee80211_sta *sta,
|
||||||
bool send_bar);
|
bool send_bar);
|
||||||
void mt76_txq_schedule(struct mt76_dev *dev, struct mt76_sw_queue *sq);
|
void mt76_txq_schedule(struct mt76_dev *dev, enum mt76_txq_id qid);
|
||||||
void mt76_txq_schedule_all(struct mt76_dev *dev);
|
void mt76_txq_schedule_all(struct mt76_dev *dev);
|
||||||
void mt76_release_buffered_frames(struct ieee80211_hw *hw,
|
void mt76_release_buffered_frames(struct ieee80211_hw *hw,
|
||||||
struct ieee80211_sta *sta,
|
struct ieee80211_sta *sta,
|
||||||
|
|
|
@ -479,23 +479,37 @@ mt76_txq_send_burst(struct mt76_dev *dev, struct mt76_sw_queue *sq,
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
mt76_txq_schedule_list(struct mt76_dev *dev, struct mt76_sw_queue *sq)
|
mt76_txq_schedule_list(struct mt76_dev *dev, enum mt76_txq_id qid)
|
||||||
{
|
{
|
||||||
|
struct mt76_sw_queue *sq = &dev->q_tx[qid];
|
||||||
struct mt76_queue *hwq = sq->q;
|
struct mt76_queue *hwq = sq->q;
|
||||||
struct mt76_txq *mtxq, *mtxq_last;
|
struct ieee80211_txq *txq;
|
||||||
int len = 0;
|
struct mt76_txq *mtxq;
|
||||||
|
struct mt76_wcid *wcid;
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
restart:
|
spin_lock_bh(&hwq->lock);
|
||||||
mtxq_last = list_last_entry(&sq->swq, struct mt76_txq, list);
|
while (1) {
|
||||||
while (!list_empty(&sq->swq)) {
|
|
||||||
bool empty = false;
|
bool empty = false;
|
||||||
int cur;
|
|
||||||
|
if (sq->swq_queued >= 4)
|
||||||
|
break;
|
||||||
|
|
||||||
if (test_bit(MT76_OFFCHANNEL, &dev->state) ||
|
if (test_bit(MT76_OFFCHANNEL, &dev->state) ||
|
||||||
test_bit(MT76_RESET, &dev->state))
|
test_bit(MT76_RESET, &dev->state)) {
|
||||||
return -EBUSY;
|
ret = -EBUSY;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
txq = ieee80211_next_txq(dev->hw, qid);
|
||||||
|
if (!txq)
|
||||||
|
break;
|
||||||
|
|
||||||
|
mtxq = (struct mt76_txq *)txq->drv_priv;
|
||||||
|
wcid = mtxq->wcid;
|
||||||
|
if (wcid && test_bit(MT_WCID_FLAG_PS, &wcid->flags))
|
||||||
|
continue;
|
||||||
|
|
||||||
mtxq = list_first_entry(&sq->swq, struct mt76_txq, list);
|
|
||||||
if (mtxq->send_bar && mtxq->aggr) {
|
if (mtxq->send_bar && mtxq->aggr) {
|
||||||
struct ieee80211_txq *txq = mtxq_to_txq(mtxq);
|
struct ieee80211_txq *txq = mtxq_to_txq(mtxq);
|
||||||
struct ieee80211_sta *sta = txq->sta;
|
struct ieee80211_sta *sta = txq->sta;
|
||||||
|
@ -507,38 +521,37 @@ restart:
|
||||||
spin_unlock_bh(&hwq->lock);
|
spin_unlock_bh(&hwq->lock);
|
||||||
ieee80211_send_bar(vif, sta->addr, tid, agg_ssn);
|
ieee80211_send_bar(vif, sta->addr, tid, agg_ssn);
|
||||||
spin_lock_bh(&hwq->lock);
|
spin_lock_bh(&hwq->lock);
|
||||||
goto restart;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
list_del_init(&mtxq->list);
|
ret += mt76_txq_send_burst(dev, sq, mtxq, &empty);
|
||||||
|
if (skb_queue_empty(&mtxq->retry_q))
|
||||||
cur = mt76_txq_send_burst(dev, sq, mtxq, &empty);
|
empty = true;
|
||||||
if (!empty)
|
ieee80211_return_txq(dev->hw, txq, !empty);
|
||||||
list_add_tail(&mtxq->list, &sq->swq);
|
|
||||||
|
|
||||||
if (cur < 0)
|
|
||||||
return cur;
|
|
||||||
|
|
||||||
len += cur;
|
|
||||||
|
|
||||||
if (mtxq == mtxq_last)
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
spin_unlock_bh(&hwq->lock);
|
||||||
|
|
||||||
return len;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
void mt76_txq_schedule(struct mt76_dev *dev, struct mt76_sw_queue *sq)
|
void mt76_txq_schedule(struct mt76_dev *dev, enum mt76_txq_id qid)
|
||||||
{
|
{
|
||||||
|
struct mt76_sw_queue *sq = &dev->q_tx[qid];
|
||||||
int len;
|
int len;
|
||||||
|
|
||||||
rcu_read_lock();
|
if (qid >= 4)
|
||||||
do {
|
return;
|
||||||
if (sq->swq_queued >= 4 || list_empty(&sq->swq))
|
|
||||||
break;
|
|
||||||
|
|
||||||
len = mt76_txq_schedule_list(dev, sq);
|
if (sq->swq_queued >= 4)
|
||||||
|
return;
|
||||||
|
|
||||||
|
rcu_read_lock();
|
||||||
|
|
||||||
|
do {
|
||||||
|
ieee80211_txq_schedule_start(dev->hw, qid);
|
||||||
|
len = mt76_txq_schedule_list(dev, qid);
|
||||||
|
ieee80211_txq_schedule_end(dev->hw, qid);
|
||||||
} while (len > 0);
|
} while (len > 0);
|
||||||
|
|
||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(mt76_txq_schedule);
|
EXPORT_SYMBOL_GPL(mt76_txq_schedule);
|
||||||
|
@ -547,13 +560,8 @@ void mt76_txq_schedule_all(struct mt76_dev *dev)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
for (i = 0; i <= MT_TXQ_BK; i++) {
|
for (i = 0; i <= MT_TXQ_BK; i++)
|
||||||
struct mt76_queue *q = dev->q_tx[i].q;
|
mt76_txq_schedule(dev, i);
|
||||||
|
|
||||||
spin_lock_bh(&q->lock);
|
|
||||||
mt76_txq_schedule(dev, &dev->q_tx[i]);
|
|
||||||
spin_unlock_bh(&q->lock);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(mt76_txq_schedule_all);
|
EXPORT_SYMBOL_GPL(mt76_txq_schedule_all);
|
||||||
|
|
||||||
|
@ -575,8 +583,6 @@ void mt76_stop_tx_queues(struct mt76_dev *dev, struct ieee80211_sta *sta,
|
||||||
|
|
||||||
spin_lock_bh(&hwq->lock);
|
spin_lock_bh(&hwq->lock);
|
||||||
mtxq->send_bar = mtxq->aggr && send_bar;
|
mtxq->send_bar = mtxq->aggr && send_bar;
|
||||||
if (!list_empty(&mtxq->list))
|
|
||||||
list_del_init(&mtxq->list);
|
|
||||||
spin_unlock_bh(&hwq->lock);
|
spin_unlock_bh(&hwq->lock);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -585,24 +591,16 @@ EXPORT_SYMBOL_GPL(mt76_stop_tx_queues);
|
||||||
void mt76_wake_tx_queue(struct ieee80211_hw *hw, struct ieee80211_txq *txq)
|
void mt76_wake_tx_queue(struct ieee80211_hw *hw, struct ieee80211_txq *txq)
|
||||||
{
|
{
|
||||||
struct mt76_dev *dev = hw->priv;
|
struct mt76_dev *dev = hw->priv;
|
||||||
struct mt76_txq *mtxq = (struct mt76_txq *)txq->drv_priv;
|
|
||||||
struct mt76_sw_queue *sq = mtxq->swq;
|
|
||||||
struct mt76_queue *hwq = sq->q;
|
|
||||||
|
|
||||||
if (!test_bit(MT76_STATE_RUNNING, &dev->state))
|
if (!test_bit(MT76_STATE_RUNNING, &dev->state))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
spin_lock_bh(&hwq->lock);
|
mt76_txq_schedule(dev, txq->ac);
|
||||||
if (list_empty(&mtxq->list))
|
|
||||||
list_add_tail(&mtxq->list, &sq->swq);
|
|
||||||
mt76_txq_schedule(dev, sq);
|
|
||||||
spin_unlock_bh(&hwq->lock);
|
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(mt76_wake_tx_queue);
|
EXPORT_SYMBOL_GPL(mt76_wake_tx_queue);
|
||||||
|
|
||||||
void mt76_txq_remove(struct mt76_dev *dev, struct ieee80211_txq *txq)
|
void mt76_txq_remove(struct mt76_dev *dev, struct ieee80211_txq *txq)
|
||||||
{
|
{
|
||||||
struct mt76_queue *hwq;
|
|
||||||
struct mt76_txq *mtxq;
|
struct mt76_txq *mtxq;
|
||||||
struct sk_buff *skb;
|
struct sk_buff *skb;
|
||||||
|
|
||||||
|
@ -610,12 +608,6 @@ void mt76_txq_remove(struct mt76_dev *dev, struct ieee80211_txq *txq)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
mtxq = (struct mt76_txq *) txq->drv_priv;
|
mtxq = (struct mt76_txq *) txq->drv_priv;
|
||||||
hwq = mtxq->swq->q;
|
|
||||||
|
|
||||||
spin_lock_bh(&hwq->lock);
|
|
||||||
if (!list_empty(&mtxq->list))
|
|
||||||
list_del_init(&mtxq->list);
|
|
||||||
spin_unlock_bh(&hwq->lock);
|
|
||||||
|
|
||||||
while ((skb = skb_dequeue(&mtxq->retry_q)) != NULL)
|
while ((skb = skb_dequeue(&mtxq->retry_q)) != NULL)
|
||||||
ieee80211_free_txskb(dev->hw, skb);
|
ieee80211_free_txskb(dev->hw, skb);
|
||||||
|
@ -626,7 +618,6 @@ void mt76_txq_init(struct mt76_dev *dev, struct ieee80211_txq *txq)
|
||||||
{
|
{
|
||||||
struct mt76_txq *mtxq = (struct mt76_txq *) txq->drv_priv;
|
struct mt76_txq *mtxq = (struct mt76_txq *) txq->drv_priv;
|
||||||
|
|
||||||
INIT_LIST_HEAD(&mtxq->list);
|
|
||||||
skb_queue_head_init(&mtxq->retry_q);
|
skb_queue_head_init(&mtxq->retry_q);
|
||||||
|
|
||||||
mtxq->swq = &dev->q_tx[mt76_txq_get_qid(txq)];
|
mtxq->swq = &dev->q_tx[mt76_txq_get_qid(txq)];
|
||||||
|
|
|
@ -645,7 +645,6 @@ static void mt76u_tx_tasklet(unsigned long data)
|
||||||
dev->drv->tx_complete_skb(dev, i, &entry);
|
dev->drv->tx_complete_skb(dev, i, &entry);
|
||||||
spin_lock_bh(&q->lock);
|
spin_lock_bh(&q->lock);
|
||||||
}
|
}
|
||||||
mt76_txq_schedule(dev, sq);
|
|
||||||
|
|
||||||
wake = q->stopped && q->queued < q->ndesc - 8;
|
wake = q->stopped && q->queued < q->ndesc - 8;
|
||||||
if (wake)
|
if (wake)
|
||||||
|
@ -656,6 +655,8 @@ static void mt76u_tx_tasklet(unsigned long data)
|
||||||
|
|
||||||
spin_unlock_bh(&q->lock);
|
spin_unlock_bh(&q->lock);
|
||||||
|
|
||||||
|
mt76_txq_schedule(dev, i);
|
||||||
|
|
||||||
if (!test_and_set_bit(MT76_READING_STATS, &dev->state))
|
if (!test_and_set_bit(MT76_READING_STATS, &dev->state))
|
||||||
ieee80211_queue_delayed_work(dev->hw,
|
ieee80211_queue_delayed_work(dev->hw,
|
||||||
&dev->usb.stat_work,
|
&dev->usb.stat_work,
|
||||||
|
|
Loading…
Add table
Reference in a new issue