dp_tx_desc.h 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. /*
  2. * Copyright (c) 2016 The Linux Foundation. All rights reserved.
  3. *
  4. * Permission to use, copy, modify, and/or distribute this software for
  5. * any purpose with or without fee is hereby granted, provided that the
  6. * above copyright notice and this permission notice appear in all
  7. * copies.
  8. *
  9. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
  10. * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
  11. * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
  12. * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  13. * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
  14. * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
  15. * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  16. * PERFORMANCE OF THIS SOFTWARE.
  17. */
  18. #ifndef DP_TX_DESC_H
  19. #define DP_TX_DESC_H
  20. #include "dp_types.h"
  21. #include "dp_tx.h"
  22. /**
  23. * 21 bits cookie
  24. * 3 bits ring id 0 ~ 7, mask 0x1C0000, offset 18
  25. * 8 bits page id 0 ~ 255, mask 0x03C800, offset 10
  26. * 10 bits offset id 0 ~ 1023 mask 0x0003FF, offset 0
  27. */
  28. /* ???Ring ID needed??? */
  29. #define DP_TX_DESC_ID_POOL_MASK 0x1C0000
  30. #define DP_TX_DESC_ID_POOL_OS 18
  31. #define DP_TX_DESC_ID_PAGE_MASK 0x03FC00
  32. #define DP_TX_DESC_ID_PAGE_OS 10
  33. #define DP_TX_DESC_ID_OFFSET_MASK 0x0003FF
  34. #define DP_TX_DESC_ID_OFFSET_OS 0
  35. /**
  36. * In case of TX descriptor pool and CPU core is combined
  37. * TX context and TX comp context also should running on the same core
  38. * in this case, each TX desciptror pool operation will be serialized by core
  39. * TX and TX_COMP will not race. locking for protection is not requried
  40. * TX_DESC_POOL_PER_CORE : this is most likely for WIN
  41. * MCL, TX descriptor pool will be tied to VDEV instance.
  42. * Then locking protection is required
  43. */
  44. #ifdef TX_CORE_ALIGNED_SEND
  45. #define TX_DESC_LOCK_CREATE(lock) /* NOOP */
  46. #define TX_DESC_LOCK_DESTROY(lock) /* NOOP */
  47. #define TX_DESC_LOCK_LOCK(lock) /* NOOP */
  48. #define TX_DESC_LOCK_UNLOCK(lock) /* NOOP */
  49. #else
  50. #define TX_DESC_LOCK_CREATE(lock) qdf_spinlock_create(lock)
  51. #define TX_DESC_LOCK_DESTROY(lock) qdf_spinlock_destroy(lock)
  52. #define TX_DESC_LOCK_LOCK(lock) qdf_spin_lock(lock)
  53. #define TX_DESC_LOCK_UNLOCK(lock) qdf_spin_unlock(lock)
  54. #endif /* TX_CORE_ALIGNED_SEND */
  55. QDF_STATUS dp_tx_desc_pool_alloc(struct dp_soc *soc, uint8_t pool_id,
  56. uint16_t num_elem);
  57. QDF_STATUS dp_tx_desc_pool_free(struct dp_soc *soc, uint8_t pool_id);
  58. QDF_STATUS dp_tx_ext_desc_pool_alloc(struct dp_soc *soc, uint8_t pool_id,
  59. uint16_t num_elem);
  60. QDF_STATUS dp_tx_ext_desc_pool_free(struct dp_soc *soc, uint8_t pool_id);
  61. /**
  62. * dp_tx_desc_alloc() - Allocate a Software Tx Descriptor from given pool
  63. *
  64. * @param soc Handle to DP SoC structure
  65. * @param pool_id
  66. *
  67. * Return:
  68. */
  69. static inline struct dp_tx_desc_s *dp_tx_desc_alloc(struct dp_soc *soc,
  70. uint8_t desc_pool_id)
  71. {
  72. struct dp_tx_desc_s *tx_desc = NULL;
  73. TX_DESC_LOCK_LOCK(&soc->tx_desc[desc_pool_id].lock);
  74. tx_desc = soc->tx_desc[desc_pool_id].freelist;
  75. /* Pool is exhausted */
  76. if (!tx_desc) {
  77. TX_DESC_LOCK_UNLOCK(&soc->tx_desc[desc_pool_id].lock);
  78. return NULL;
  79. }
  80. if (soc->tx_desc[desc_pool_id].freelist) {
  81. soc->tx_desc[desc_pool_id].freelist =
  82. soc->tx_desc[desc_pool_id].freelist->next;
  83. soc->tx_desc[desc_pool_id].num_allocated++;
  84. }
  85. DP_STATS_ADD(pdev, pub.tx.desc_in_use, 1);
  86. tx_desc->flags |= DP_TX_DESC_FLAG_ALLOCATED;
  87. TX_DESC_LOCK_UNLOCK(&soc->tx_desc[desc_pool_id].lock);
  88. return tx_desc;
  89. }
  90. /**
  91. * dp_tx_desc_free() - Fee a tx descriptor and attach it to free list
  92. *
  93. * @soc Handle to DP SoC structure
  94. * @pool_id
  95. * @tx_desc
  96. */
  97. static inline void
  98. dp_tx_desc_free(struct dp_soc *soc, struct dp_tx_desc_s *tx_desc,
  99. uint8_t desc_pool_id)
  100. {
  101. TX_DESC_LOCK_LOCK(&soc->tx_desc[desc_pool_id].lock);
  102. tx_desc->flags &= ~DP_TX_DESC_FLAG_ALLOCATED;
  103. tx_desc->next = soc->tx_desc[desc_pool_id].freelist;
  104. soc->tx_desc[desc_pool_id].freelist = tx_desc;
  105. DP_STATS_SUB(pdev, pub.tx.desc_in_use, 1);
  106. TX_DESC_LOCK_UNLOCK(&soc->tx_desc[desc_pool_id].lock);
  107. }
  108. /**
  109. * dp_tx_desc_find() - find dp tx descriptor from cokie
  110. * @soc - handle for the device sending the data
  111. * @tx_desc_id - the ID of the descriptor in question
  112. * @return the descriptor object that has the specified ID
  113. *
  114. * Use a tx descriptor ID to find the corresponding descriptor object.
  115. *
  116. */
  117. static inline struct dp_tx_desc_s *dp_tx_desc_find(struct dp_soc *soc,
  118. uint8_t pool_id, uint16_t page_id, uint16_t offset)
  119. {
  120. return soc->tx_desc[pool_id].desc_pages.cacheable_pages[page_id] +
  121. soc->tx_desc[pool_id].elem_size * offset;
  122. }
  123. /**
  124. * dp_tx_ext_desc_alloc() - Get tx extension descriptor from pool
  125. * @soc: handle for the device sending the data
  126. * @pool_id: target pool id
  127. *
  128. * Return: None
  129. */
  130. static inline
  131. struct dp_tx_ext_desc_elem_s *dp_tx_ext_desc_alloc(struct dp_soc *soc,
  132. uint8_t desc_pool_id)
  133. {
  134. struct dp_tx_ext_desc_elem_s *c_elem;
  135. TX_DESC_LOCK_LOCK(&soc->tx_ext_desc[desc_pool_id].lock);
  136. c_elem = soc->tx_ext_desc[desc_pool_id].freelist;
  137. soc->tx_ext_desc[desc_pool_id].freelist =
  138. soc->tx_ext_desc[desc_pool_id].freelist->next;
  139. TX_DESC_LOCK_UNLOCK(&soc->tx_ext_desc[desc_pool_id].lock);
  140. return c_elem;
  141. }
  142. /**
  143. * dp_tx_ext_desc_free() - Release tx extension descriptor to the pool
  144. * @soc: handle for the device sending the data
  145. * @pool_id: target pool id
  146. * @elem: ext descriptor pointer should release
  147. *
  148. * Return: None
  149. */
  150. static inline void dp_tx_ext_desc_free(struct dp_soc *soc,
  151. struct dp_tx_ext_desc_elem_s *elem, uint8_t desc_pool_id)
  152. {
  153. TX_DESC_LOCK_LOCK(&soc->tx_ext_desc[desc_pool_id].lock);
  154. elem->next = soc->tx_ext_desc[desc_pool_id].freelist;
  155. soc->tx_ext_desc[desc_pool_id].freelist = elem;
  156. TX_DESC_LOCK_UNLOCK(&soc->tx_ext_desc[desc_pool_id].lock);
  157. return;
  158. }
  159. #endif /* DP_TX_DESC_H */