mt76x2_dma.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. /*
  2. * Copyright (C) 2016 Felix Fietkau <nbd@nbd.name>
  3. *
  4. * Permission to use, copy, modify, and/or distribute this software for any
  5. * purpose with or without fee is hereby granted, provided that the above
  6. * copyright notice and this permission notice appear in all copies.
  7. *
  8. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  9. * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  10. * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
  11. * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  12. * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
  13. * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  14. * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  15. */
  16. #include "mt76x2.h"
  17. #include "mt76x2_dma.h"
  18. int
  19. mt76x2_tx_queue_mcu(struct mt76x2_dev *dev, enum mt76_txq_id qid,
  20. struct sk_buff *skb, int cmd, int seq)
  21. {
  22. struct mt76_queue *q = &dev->mt76.q_tx[qid];
  23. struct mt76_queue_buf buf;
  24. dma_addr_t addr;
  25. u32 tx_info;
  26. tx_info = MT_MCU_MSG_TYPE_CMD |
  27. FIELD_PREP(MT_MCU_MSG_CMD_TYPE, cmd) |
  28. FIELD_PREP(MT_MCU_MSG_CMD_SEQ, seq) |
  29. FIELD_PREP(MT_MCU_MSG_PORT, CPU_TX_PORT) |
  30. FIELD_PREP(MT_MCU_MSG_LEN, skb->len);
  31. addr = dma_map_single(dev->mt76.dev, skb->data, skb->len,
  32. DMA_TO_DEVICE);
  33. if (dma_mapping_error(dev->mt76.dev, addr))
  34. return -ENOMEM;
  35. buf.addr = addr;
  36. buf.len = skb->len;
  37. spin_lock_bh(&q->lock);
  38. mt76_queue_add_buf(dev, q, &buf, 1, tx_info, skb, NULL);
  39. mt76_queue_kick(dev, q);
  40. spin_unlock_bh(&q->lock);
  41. return 0;
  42. }
  43. static int
  44. mt76x2_init_tx_queue(struct mt76x2_dev *dev, struct mt76_queue *q,
  45. int idx, int n_desc)
  46. {
  47. int ret;
  48. q->regs = dev->mt76.regs + MT_TX_RING_BASE + idx * MT_RING_SIZE;
  49. q->ndesc = n_desc;
  50. q->hw_idx = idx;
  51. ret = mt76_queue_alloc(dev, q);
  52. if (ret)
  53. return ret;
  54. mt76x2_irq_enable(dev, MT_INT_TX_DONE(idx));
  55. return 0;
  56. }
  57. void mt76x2_queue_rx_skb(struct mt76_dev *mdev, enum mt76_rxq_id q,
  58. struct sk_buff *skb)
  59. {
  60. struct mt76x2_dev *dev = container_of(mdev, struct mt76x2_dev, mt76);
  61. void *rxwi = skb->data;
  62. if (q == MT_RXQ_MCU) {
  63. skb_queue_tail(&dev->mcu.res_q, skb);
  64. wake_up(&dev->mcu.wait);
  65. return;
  66. }
  67. skb_pull(skb, sizeof(struct mt76x2_rxwi));
  68. if (mt76x2_mac_process_rx(dev, skb, rxwi)) {
  69. dev_kfree_skb(skb);
  70. return;
  71. }
  72. mt76_rx(&dev->mt76, q, skb);
  73. }
  74. static int
  75. mt76x2_init_rx_queue(struct mt76x2_dev *dev, struct mt76_queue *q,
  76. int idx, int n_desc, int bufsize)
  77. {
  78. int ret;
  79. q->regs = dev->mt76.regs + MT_RX_RING_BASE + idx * MT_RING_SIZE;
  80. q->ndesc = n_desc;
  81. q->buf_size = bufsize;
  82. ret = mt76_queue_alloc(dev, q);
  83. if (ret)
  84. return ret;
  85. mt76x2_irq_enable(dev, MT_INT_RX_DONE(idx));
  86. return 0;
  87. }
  88. static void
  89. mt76x2_tx_tasklet(unsigned long data)
  90. {
  91. struct mt76x2_dev *dev = (struct mt76x2_dev *) data;
  92. int i;
  93. mt76x2_mac_process_tx_status_fifo(dev);
  94. for (i = MT_TXQ_MCU; i >= 0; i--)
  95. mt76_queue_tx_cleanup(dev, i, false);
  96. mt76x2_mac_poll_tx_status(dev, false);
  97. mt76x2_irq_enable(dev, MT_INT_TX_DONE_ALL);
  98. }
  99. int mt76x2_dma_init(struct mt76x2_dev *dev)
  100. {
  101. static const u8 wmm_queue_map[] = {
  102. [IEEE80211_AC_BE] = 0,
  103. [IEEE80211_AC_BK] = 1,
  104. [IEEE80211_AC_VI] = 2,
  105. [IEEE80211_AC_VO] = 3,
  106. };
  107. int ret;
  108. int i;
  109. struct mt76_txwi_cache __maybe_unused *t;
  110. struct mt76_queue *q;
  111. BUILD_BUG_ON(sizeof(t->txwi) < sizeof(struct mt76x2_txwi));
  112. BUILD_BUG_ON(sizeof(struct mt76x2_rxwi) > MT_RX_HEADROOM);
  113. mt76_dma_attach(&dev->mt76);
  114. init_waitqueue_head(&dev->mcu.wait);
  115. skb_queue_head_init(&dev->mcu.res_q);
  116. tasklet_init(&dev->tx_tasklet, mt76x2_tx_tasklet, (unsigned long) dev);
  117. mt76_wr(dev, MT_WPDMA_RST_IDX, ~0);
  118. for (i = 0; i < ARRAY_SIZE(wmm_queue_map); i++) {
  119. ret = mt76x2_init_tx_queue(dev, &dev->mt76.q_tx[i],
  120. wmm_queue_map[i], MT_TX_RING_SIZE);
  121. if (ret)
  122. return ret;
  123. }
  124. ret = mt76x2_init_tx_queue(dev, &dev->mt76.q_tx[MT_TXQ_PSD],
  125. MT_TX_HW_QUEUE_MGMT, MT_TX_RING_SIZE);
  126. if (ret)
  127. return ret;
  128. ret = mt76x2_init_tx_queue(dev, &dev->mt76.q_tx[MT_TXQ_MCU],
  129. MT_TX_HW_QUEUE_MCU, MT_MCU_RING_SIZE);
  130. if (ret)
  131. return ret;
  132. ret = mt76x2_init_rx_queue(dev, &dev->mt76.q_rx[MT_RXQ_MCU], 1,
  133. MT_MCU_RING_SIZE, MT_RX_BUF_SIZE);
  134. if (ret)
  135. return ret;
  136. q = &dev->mt76.q_rx[MT_RXQ_MAIN];
  137. q->buf_offset = MT_RX_HEADROOM - sizeof(struct mt76x2_rxwi);
  138. ret = mt76x2_init_rx_queue(dev, q, 0, MT76x2_RX_RING_SIZE, MT_RX_BUF_SIZE);
  139. if (ret)
  140. return ret;
  141. return mt76_init_queues(dev);
  142. }
  143. void mt76x2_dma_cleanup(struct mt76x2_dev *dev)
  144. {
  145. tasklet_kill(&dev->tx_tasklet);
  146. mt76_dma_cleanup(&dev->mt76);
  147. }