hal_rx.c 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  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. #include "wcss_seq_hwiobase.h"
  19. #include "wcss_seq_hwioreg.h"
  20. #include "sw_xml_headers.h"
  21. #include "tlv_hdr.h"
  22. #include "hal_api.h"
  23. /* TODO: See if the following definition is available in HW headers */
  24. #define HAL_REO_OWNED 4
  25. #define HAL_REO_QUEUE_DESC 8
  26. #define HAL_REO_QUEUE_EXT_DESC 9
  27. #define PN_SIZE_24 0
  28. #define PN_SIZE_48 1
  29. #define PN_SIZE_128 2
  30. /* TODO: Using associated link desc counter 1 for Rx. Check with FW on
  31. * how these counters are assigned
  32. */
  33. #define HAL_RX_LINK_DESC_CNTR 1
  34. /* TODO: Following definition should be from HW headers */
  35. #define HAL_DESC_REO_OWNED 4
  36. /* TODO: Move this to common header file */
  37. static inline void hal_uniform_desc_hdr_setup(uint32_t *desc, uint32_t owner,
  38. uint32_t buffer_type)
  39. {
  40. HAL_DESC_SET_FIELD(desc, UNIFORM_DESCRIPTOR_HEADER_0, OWNER,
  41. owner);
  42. HAL_DESC_SET_FIELD(desc, UNIFORM_DESCRIPTOR_HEADER_0, BUFFER_TYPE,
  43. buffer_type);
  44. }
  45. #ifndef TID_TO_WME_AC
  46. #define WME_AC_BE 0 /* best effort */
  47. #define WME_AC_BK 1 /* background */
  48. #define WME_AC_VI 2 /* video */
  49. #define WME_AC_VO 3 /* voice */
  50. #define TID_TO_WME_AC(_tid) ( \
  51. (((_tid) == 0) || ((_tid) == 3)) ? WME_AC_BE : \
  52. (((_tid) == 1) || ((_tid) == 2)) ? WME_AC_BK : \
  53. (((_tid) == 4) || ((_tid) == 5)) ? WME_AC_VI : \
  54. WME_AC_VO)
  55. #endif
  56. #define HAL_NON_QOS_TID 16
  57. /**
  58. * hal_reo_qdesc_setup - Setup HW REO queue descriptor
  59. *
  60. * @hal_soc: Opaque HAL SOC handle
  61. * @ba_window_size: BlockAck window size
  62. * @start_seq: Starting sequence number
  63. * @hw_qdesc_vaddr: Virtual address of REO queue descriptor memory
  64. * @hw_qdesc_paddr: Physical address of REO queue descriptor memory
  65. * @tid: TID
  66. *
  67. */
  68. void hal_reo_qdesc_setup(void *hal_soc, int tid, uint32_t ba_window_size,
  69. uint32_t start_seq, void *hw_qdesc_vaddr, qdf_dma_addr_t hw_qdesc_paddr,
  70. int pn_type)
  71. {
  72. uint32_t *reo_queue_desc = (uint32_t *)hw_qdesc_vaddr;
  73. uint32_t *reo_queue_ext_desc;
  74. uint32_t reg_val;
  75. uint32_t pn_enable, pn_size;
  76. qdf_mem_zero(hw_qdesc_vaddr, sizeof(struct rx_reo_queue));
  77. hal_uniform_desc_hdr_setup(reo_queue_desc, HAL_DESC_REO_OWNED,
  78. HAL_REO_QUEUE_DESC);
  79. /* This a just a SW meta data and will be copied to REO destination
  80. * descriptors indicated by hardware.
  81. * TODO: Setting TID in this field. See if we should set something else.
  82. */
  83. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_1,
  84. RECEIVE_QUEUE_NUMBER, tid);
  85. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2,
  86. VLD, 1);
  87. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2,
  88. ASSOCIATED_LINK_DESCRIPTOR_COUNTER, HAL_RX_LINK_DESC_CNTR);
  89. /*
  90. * Fields DISABLE_DUPLICATE_DETECTION and SOFT_REORDER_ENABLE will be 0
  91. */
  92. reg_val = TID_TO_WME_AC(tid);
  93. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, AC, reg_val);
  94. /* Check the purpose of RTY field.
  95. * HW documentation says "Retry bit is checked if this bit is set"
  96. */
  97. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, CHK_2K_MODE, 1);
  98. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, OOR_MODE, 1);
  99. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, BA_WINDOW_SIZE,
  100. ba_window_size - 1);
  101. switch (pn_type) {
  102. case HAL_PN_WPA:
  103. pn_enable = 1;
  104. pn_size = PN_SIZE_48;
  105. case HAL_PN_WAPI_EVEN:
  106. case HAL_PN_WAPI_UNEVEN:
  107. pn_enable = 1;
  108. pn_size = PN_SIZE_128;
  109. default:
  110. pn_enable = 0;
  111. }
  112. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, PN_CHECK_NEEDED,
  113. pn_enable);
  114. if (pn_type == HAL_PN_WAPI_EVEN)
  115. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2,
  116. PN_SHALL_BE_EVEN, 1);
  117. else if (pn_type == HAL_PN_WAPI_UNEVEN)
  118. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2,
  119. PN_SHALL_BE_UNEVEN, 1);
  120. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, PN_HANDLING_ENABLE,
  121. pn_enable);
  122. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, PN_SIZE,
  123. pn_size);
  124. /* TODO: Check if RX_REO_QUEUE_2_IGNORE_AMPDU_FLAG need to be set
  125. * any other cases
  126. */
  127. if ((ba_window_size <= 1) || (tid == HAL_NON_QOS_TID)) {
  128. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2,
  129. IGNORE_AMPDU_FLAG, 1);
  130. }
  131. if (start_seq <= 0xfff) {
  132. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_3, SVLD, 1);
  133. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_3, SSN,
  134. start_seq);
  135. }
  136. /* TODO: Check if we should set start PN for WAPI */
  137. #ifdef notyet
  138. /* Setup first queue extension if BA window size is more than 1 */
  139. if (ba_window_size > 1) {
  140. reo_queue_ext_desc =
  141. (uint32_t *)(((struct rx_reo_queue *)reo_queue_desc) +
  142. 1);
  143. qdf_mem_zero(reo_queue_ext_desc,
  144. sizeof(struct rx_reo_queue_ext));
  145. hal_uniform_desc_hdr_setup(reo_queue_ext_desc,
  146. HAL_DESC_REO_OWNED, HAL_REO_QUEUE_EXT_DESC);
  147. }
  148. /* Setup second queue extension if BA window size is more than 105 */
  149. if (ba_window_size > 105) {
  150. reo_queue_ext_desc = (uint32_t *)
  151. (((struct rx_reo_queue_ext *)reo_queue_ext_desc) + 1);
  152. qdf_mem_zero(reo_queue_ext_desc,
  153. sizeof(struct rx_reo_queue_ext));
  154. hal_uniform_desc_hdr_setup(reo_queue_ext_desc,
  155. HAL_DESC_REO_OWNED, HAL_REO_QUEUE_EXT_DESC);
  156. }
  157. /* Setup third queue extension if BA window size is more than 210 */
  158. if (ba_window_size > 210) {
  159. reo_queue_ext_desc = (uint32_t *)
  160. (((struct rx_reo_queue_ext *)reo_queue_ext_desc) + 1);
  161. qdf_mem_zero(reo_queue_ext_desc,
  162. sizeof(struct rx_reo_queue_ext));
  163. hal_uniform_desc_hdr_setup(reo_queue_ext_desc,
  164. HAL_DESC_REO_OWNED, HAL_REO_QUEUE_EXT_DESC);
  165. }
  166. #else
  167. /* TODO: HW queue descriptors are currently allocated for max BA
  168. * window size for all QOS TIDs so that same descriptor can be used
  169. * later when ADDBA request is recevied. This should be changed to
  170. * allocate HW queue descriptors based on BA window size being
  171. * negotiated (0 for non BA cases), and reallocate when BA window
  172. * size changes and also send WMI message to FW to change the REO
  173. * queue descriptor in Rx peer entry as part of dp_rx_tid_update.
  174. */
  175. if (tid != HAL_NON_QOS_TID) {
  176. reo_queue_ext_desc = (uint32_t *)
  177. (((struct rx_reo_queue *)reo_queue_desc) + 1);
  178. qdf_mem_zero(reo_queue_ext_desc, 3 *
  179. sizeof(struct rx_reo_queue_ext));
  180. /* Initialize first reo queue extension descriptor */
  181. hal_uniform_desc_hdr_setup(reo_queue_ext_desc,
  182. HAL_DESC_REO_OWNED, HAL_REO_QUEUE_EXT_DESC);
  183. /* Initialize second reo queue extension descriptor */
  184. reo_queue_ext_desc = (uint32_t *)
  185. (((struct rx_reo_queue_ext *)reo_queue_ext_desc) + 1);
  186. hal_uniform_desc_hdr_setup(reo_queue_ext_desc,
  187. HAL_DESC_REO_OWNED, HAL_REO_QUEUE_EXT_DESC);
  188. /* Initialize third reo queue extension descriptor */
  189. reo_queue_ext_desc = (uint32_t *)
  190. (((struct rx_reo_queue_ext *)reo_queue_ext_desc) + 1);
  191. hal_uniform_desc_hdr_setup(reo_queue_ext_desc,
  192. HAL_DESC_REO_OWNED, HAL_REO_QUEUE_EXT_DESC);
  193. }
  194. #endif
  195. }
  196. /**
  197. * hal_reo_setup - Initialize HW REO block
  198. *
  199. * @hal_soc: Opaque HAL SOC handle
  200. */
  201. void hal_reo_setup(void *hal_soc)
  202. {
  203. struct hal_soc *soc = (struct hal_soc *)hal_soc;
  204. HAL_REG_WRITE(soc, HWIO_REO_R0_GENERAL_ENABLE_ADDR(
  205. SEQ_WCSS_UMAC_REO_REG_OFFSET),
  206. HAL_SM(HWIO_REO_R0_GENERAL_ENABLE,
  207. FRAGMENT_DEST_RING, HAL_SRNG_REO_EXCEPTION) |
  208. HAL_SM(HWIO_REO_R0_GENERAL_ENABLE, AGING_LIST_ENABLE, 1) |
  209. HAL_SM(HWIO_REO_R0_GENERAL_ENABLE, AGING_FLUSH_ENABLE, 1));
  210. /* Other ring enable bits and REO_ENABLE will be set by FW */
  211. /* TODO: Setup destination ring mapping if enabled */
  212. /* TODO: Error destination ring setting is left to default.
  213. * Default setting is to send all errors to release ring.
  214. */
  215. HAL_REG_WRITE(soc,
  216. HWIO_REO_R0_AGING_THRESHOLD_IX_0_ADDR(
  217. SEQ_WCSS_UMAC_REO_REG_OFFSET),
  218. HAL_DEFAULT_REO_TIMEOUT_MS * 1000);
  219. HAL_REG_WRITE(soc,
  220. HWIO_REO_R0_AGING_THRESHOLD_IX_1_ADDR(
  221. SEQ_WCSS_UMAC_REO_REG_OFFSET),
  222. (HAL_DEFAULT_REO_TIMEOUT_MS * 1000));
  223. HAL_REG_WRITE(soc,
  224. HWIO_REO_R0_AGING_THRESHOLD_IX_2_ADDR(
  225. SEQ_WCSS_UMAC_REO_REG_OFFSET),
  226. (HAL_DEFAULT_REO_TIMEOUT_MS * 1000));
  227. HAL_REG_WRITE(soc,
  228. HWIO_REO_R0_AGING_THRESHOLD_IX_3_ADDR(
  229. SEQ_WCSS_UMAC_REO_REG_OFFSET),
  230. (HAL_DEFAULT_REO_TIMEOUT_MS * 1000));
  231. /* TODO: Check if the following registers shoould be setup by host:
  232. * AGING_CONTROL
  233. * HIGH_MEMORY_THRESHOLD
  234. * GLOBAL_LINK_DESC_COUNT_THRESH_IX_0[1,2]
  235. * GLOBAL_LINK_DESC_COUNT_CTRL
  236. */
  237. }