hal_rx.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. /*
  2. * Copyright (c) 2016-2017 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 "hal_api.h"
  19. /* TODO: See if the following definition is available in HW headers */
  20. #define HAL_REO_OWNED 4
  21. #define HAL_REO_QUEUE_DESC 8
  22. #define HAL_REO_QUEUE_EXT_DESC 9
  23. /* TODO: Using associated link desc counter 1 for Rx. Check with FW on
  24. * how these counters are assigned
  25. */
  26. #define HAL_RX_LINK_DESC_CNTR 1
  27. /* TODO: Following definition should be from HW headers */
  28. #define HAL_DESC_REO_OWNED 4
  29. /* TODO: Move this to common header file */
  30. static inline void hal_uniform_desc_hdr_setup(uint32_t *desc, uint32_t owner,
  31. uint32_t buffer_type)
  32. {
  33. HAL_DESC_SET_FIELD(desc, UNIFORM_DESCRIPTOR_HEADER_0, OWNER,
  34. owner);
  35. HAL_DESC_SET_FIELD(desc, UNIFORM_DESCRIPTOR_HEADER_0, BUFFER_TYPE,
  36. buffer_type);
  37. }
  38. #ifndef TID_TO_WME_AC
  39. #define WME_AC_BE 0 /* best effort */
  40. #define WME_AC_BK 1 /* background */
  41. #define WME_AC_VI 2 /* video */
  42. #define WME_AC_VO 3 /* voice */
  43. #define TID_TO_WME_AC(_tid) ( \
  44. (((_tid) == 0) || ((_tid) == 3)) ? WME_AC_BE : \
  45. (((_tid) == 1) || ((_tid) == 2)) ? WME_AC_BK : \
  46. (((_tid) == 4) || ((_tid) == 5)) ? WME_AC_VI : \
  47. WME_AC_VO)
  48. #endif
  49. #define HAL_NON_QOS_TID 16
  50. /**
  51. * When hash based routing is enabled, routing of the rx packet
  52. * is done based on the following value: 1 _ _ _ _ The last 4
  53. * bits are based on hash[3:0]. This means the possible values
  54. * are 0x10 to 0x1f. This value is used to look-up the
  55. * ring ID configured in Destination_Ring_Ctrl_IX_* register.
  56. * The Destination_Ring_Ctrl_IX_2 and Destination_Ring_Ctrl_IX_3
  57. * registers need to be configured to set-up the 16 entries to
  58. * map the hash values to a ring number. There are 3 bits per
  59. * hash entry – which are mapped as follows:
  60. * 0: TCL, 1:SW1, 2:SW2, * 3:SW3, 4:SW4, 5:Release, 6:FW(WIFI),
  61. * 7: NOT_USED.
  62. */
  63. #ifdef IPA_OFFLOAD
  64. /**
  65. * When IPA is enabled, there will be 3 available rings.
  66. * Otherwise there will be 4.
  67. */
  68. #define REO_REMAP_REGISTER_2 ( \
  69. ((0x1 << 0) | (0x2 << 3) | (0x3 << 6) | (0x1 << 9) | \
  70. (0x2 << 12) | (0x3 << 15) | (0x1 << 18) | (0x2 << 21)) << 8)
  71. #define REO_REMAP_REGISTER_3 ( \
  72. ((0x3 << 0) | (0x1 << 3) | (0x2 << 6) | (0x3 << 9) | \
  73. (0x1 << 12) | (0x2 << 15) | (0x3 << 18) | (0x1 << 21)) << 8)
  74. #else
  75. #define REO_REMAP_REGISTER_2 ( \
  76. ((0x1 << 0) | (0x2 << 3) | (0x3 << 6) | (0x4 << 9) | \
  77. (0x1 << 12) | (0x2 << 15) | (0x3 << 18) | (0x4 << 21)) << 8)
  78. #define REO_REMAP_REGISTER_3 ( \
  79. ((0x1 << 0) | (0x2 << 3) | (0x3 << 6) | (0x4 << 9) | \
  80. (0x1 << 12) | (0x2 << 15) | (0x3 << 18) | (0x4 << 21)) << 8)
  81. #endif
  82. /**
  83. * hal_reo_qdesc_setup - Setup HW REO queue descriptor
  84. *
  85. * @hal_soc: Opaque HAL SOC handle
  86. * @ba_window_size: BlockAck window size
  87. * @start_seq: Starting sequence number
  88. * @hw_qdesc_vaddr: Virtual address of REO queue descriptor memory
  89. * @hw_qdesc_paddr: Physical address of REO queue descriptor memory
  90. * @tid: TID
  91. *
  92. */
  93. void hal_reo_qdesc_setup(void *hal_soc, int tid, uint32_t ba_window_size,
  94. uint32_t start_seq, void *hw_qdesc_vaddr, qdf_dma_addr_t hw_qdesc_paddr,
  95. int pn_type)
  96. {
  97. uint32_t *reo_queue_desc = (uint32_t *)hw_qdesc_vaddr;
  98. uint32_t *reo_queue_ext_desc;
  99. uint32_t reg_val;
  100. uint32_t pn_enable, pn_size;
  101. qdf_mem_zero(hw_qdesc_vaddr, sizeof(struct rx_reo_queue));
  102. hal_uniform_desc_hdr_setup(reo_queue_desc, HAL_DESC_REO_OWNED,
  103. HAL_REO_QUEUE_DESC);
  104. /* This a just a SW meta data and will be copied to REO destination
  105. * descriptors indicated by hardware.
  106. * TODO: Setting TID in this field. See if we should set something else.
  107. */
  108. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_1,
  109. RECEIVE_QUEUE_NUMBER, tid);
  110. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2,
  111. VLD, 1);
  112. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2,
  113. ASSOCIATED_LINK_DESCRIPTOR_COUNTER, HAL_RX_LINK_DESC_CNTR);
  114. /*
  115. * Fields DISABLE_DUPLICATE_DETECTION and SOFT_REORDER_ENABLE will be 0
  116. */
  117. reg_val = TID_TO_WME_AC(tid);
  118. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, AC, reg_val);
  119. if (ba_window_size < 1)
  120. ba_window_size = 1;
  121. /* Set RTY bit for non-BA case. Duplicate detection is currently not
  122. * done by HW in non-BA case if RTY bit is not set.
  123. * TODO: This is a temporary War and should be removed once HW fix is
  124. * made to check and discard duplicates even if RTY bit is not set.
  125. */
  126. if (ba_window_size == 1)
  127. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, RTY, 1);
  128. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, BA_WINDOW_SIZE,
  129. ba_window_size - 1);
  130. switch (pn_type) {
  131. case HAL_PN_WPA:
  132. pn_enable = 1;
  133. pn_size = PN_SIZE_48;
  134. case HAL_PN_WAPI_EVEN:
  135. case HAL_PN_WAPI_UNEVEN:
  136. pn_enable = 1;
  137. pn_size = PN_SIZE_128;
  138. default:
  139. pn_enable = 0;
  140. }
  141. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, PN_CHECK_NEEDED,
  142. pn_enable);
  143. if (pn_type == HAL_PN_WAPI_EVEN)
  144. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2,
  145. PN_SHALL_BE_EVEN, 1);
  146. else if (pn_type == HAL_PN_WAPI_UNEVEN)
  147. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2,
  148. PN_SHALL_BE_UNEVEN, 1);
  149. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, PN_HANDLING_ENABLE,
  150. pn_enable);
  151. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2, PN_SIZE,
  152. pn_size);
  153. /* TODO: Check if RX_REO_QUEUE_2_IGNORE_AMPDU_FLAG need to be set
  154. * based on BA window size and/or AMPDU capabilities
  155. */
  156. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_2,
  157. IGNORE_AMPDU_FLAG, 1);
  158. if (start_seq <= 0xfff)
  159. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_3, SSN,
  160. start_seq);
  161. /* TODO: SVLD should be set to 1 if a valid SSN is received in ADDBA,
  162. * but REO is not delivering packets if we set it to 1. Need to enable
  163. * this once the issue is resolved */
  164. HAL_DESC_SET_FIELD(reo_queue_desc, RX_REO_QUEUE_3, SVLD, 0);
  165. /* TODO: Check if we should set start PN for WAPI */
  166. #ifdef notyet
  167. /* Setup first queue extension if BA window size is more than 1 */
  168. if (ba_window_size > 1) {
  169. reo_queue_ext_desc =
  170. (uint32_t *)(((struct rx_reo_queue *)reo_queue_desc) +
  171. 1);
  172. qdf_mem_zero(reo_queue_ext_desc,
  173. sizeof(struct rx_reo_queue_ext));
  174. hal_uniform_desc_hdr_setup(reo_queue_ext_desc,
  175. HAL_DESC_REO_OWNED, HAL_REO_QUEUE_EXT_DESC);
  176. }
  177. /* Setup second queue extension if BA window size is more than 105 */
  178. if (ba_window_size > 105) {
  179. reo_queue_ext_desc = (uint32_t *)
  180. (((struct rx_reo_queue_ext *)reo_queue_ext_desc) + 1);
  181. qdf_mem_zero(reo_queue_ext_desc,
  182. sizeof(struct rx_reo_queue_ext));
  183. hal_uniform_desc_hdr_setup(reo_queue_ext_desc,
  184. HAL_DESC_REO_OWNED, HAL_REO_QUEUE_EXT_DESC);
  185. }
  186. /* Setup third queue extension if BA window size is more than 210 */
  187. if (ba_window_size > 210) {
  188. reo_queue_ext_desc = (uint32_t *)
  189. (((struct rx_reo_queue_ext *)reo_queue_ext_desc) + 1);
  190. qdf_mem_zero(reo_queue_ext_desc,
  191. sizeof(struct rx_reo_queue_ext));
  192. hal_uniform_desc_hdr_setup(reo_queue_ext_desc,
  193. HAL_DESC_REO_OWNED, HAL_REO_QUEUE_EXT_DESC);
  194. }
  195. #else
  196. /* TODO: HW queue descriptors are currently allocated for max BA
  197. * window size for all QOS TIDs so that same descriptor can be used
  198. * later when ADDBA request is recevied. This should be changed to
  199. * allocate HW queue descriptors based on BA window size being
  200. * negotiated (0 for non BA cases), and reallocate when BA window
  201. * size changes and also send WMI message to FW to change the REO
  202. * queue descriptor in Rx peer entry as part of dp_rx_tid_update.
  203. */
  204. if (tid != HAL_NON_QOS_TID) {
  205. reo_queue_ext_desc = (uint32_t *)
  206. (((struct rx_reo_queue *)reo_queue_desc) + 1);
  207. qdf_mem_zero(reo_queue_ext_desc, 3 *
  208. sizeof(struct rx_reo_queue_ext));
  209. /* Initialize first reo queue extension descriptor */
  210. hal_uniform_desc_hdr_setup(reo_queue_ext_desc,
  211. HAL_DESC_REO_OWNED, HAL_REO_QUEUE_EXT_DESC);
  212. /* Initialize second reo queue extension descriptor */
  213. reo_queue_ext_desc = (uint32_t *)
  214. (((struct rx_reo_queue_ext *)reo_queue_ext_desc) + 1);
  215. hal_uniform_desc_hdr_setup(reo_queue_ext_desc,
  216. HAL_DESC_REO_OWNED, HAL_REO_QUEUE_EXT_DESC);
  217. /* Initialize third reo queue extension descriptor */
  218. reo_queue_ext_desc = (uint32_t *)
  219. (((struct rx_reo_queue_ext *)reo_queue_ext_desc) + 1);
  220. hal_uniform_desc_hdr_setup(reo_queue_ext_desc,
  221. HAL_DESC_REO_OWNED, HAL_REO_QUEUE_EXT_DESC);
  222. }
  223. #endif
  224. }
  225. /**
  226. * hal_reo_setup - Initialize HW REO block
  227. *
  228. * @hal_soc: Opaque HAL SOC handle
  229. * @reo_params: parameters needed by HAL for REO config
  230. */
  231. void hal_reo_setup(void *hal_soc,
  232. struct hal_reo_params *reo_params)
  233. {
  234. struct hal_soc *soc = (struct hal_soc *)hal_soc;
  235. HAL_REG_WRITE(soc, HWIO_REO_R0_GENERAL_ENABLE_ADDR(
  236. SEQ_WCSS_UMAC_REO_REG_OFFSET),
  237. HAL_SM(HWIO_REO_R0_GENERAL_ENABLE,
  238. FRAGMENT_DEST_RING, HAL_SRNG_REO_EXCEPTION) |
  239. HAL_SM(HWIO_REO_R0_GENERAL_ENABLE, AGING_LIST_ENABLE, 1) |
  240. HAL_SM(HWIO_REO_R0_GENERAL_ENABLE, AGING_FLUSH_ENABLE, 1));
  241. /* Other ring enable bits and REO_ENABLE will be set by FW */
  242. /* TODO: Setup destination ring mapping if enabled */
  243. /* TODO: Error destination ring setting is left to default.
  244. * Default setting is to send all errors to release ring.
  245. */
  246. HAL_REG_WRITE(soc,
  247. HWIO_REO_R0_AGING_THRESHOLD_IX_0_ADDR(
  248. SEQ_WCSS_UMAC_REO_REG_OFFSET),
  249. HAL_DEFAULT_REO_TIMEOUT_MS * 1000);
  250. HAL_REG_WRITE(soc,
  251. HWIO_REO_R0_AGING_THRESHOLD_IX_1_ADDR(
  252. SEQ_WCSS_UMAC_REO_REG_OFFSET),
  253. (HAL_DEFAULT_REO_TIMEOUT_MS * 1000));
  254. HAL_REG_WRITE(soc,
  255. HWIO_REO_R0_AGING_THRESHOLD_IX_2_ADDR(
  256. SEQ_WCSS_UMAC_REO_REG_OFFSET),
  257. (HAL_DEFAULT_REO_TIMEOUT_MS * 1000));
  258. HAL_REG_WRITE(soc,
  259. HWIO_REO_R0_AGING_THRESHOLD_IX_3_ADDR(
  260. SEQ_WCSS_UMAC_REO_REG_OFFSET),
  261. (HAL_DEFAULT_REO_TIMEOUT_MS * 1000));
  262. if (reo_params->rx_hash_enabled) {
  263. HAL_REG_WRITE(soc,
  264. HWIO_REO_R0_DESTINATION_RING_CTRL_IX_2_ADDR(
  265. SEQ_WCSS_UMAC_REO_REG_OFFSET),
  266. REO_REMAP_REGISTER_2);
  267. QDF_TRACE(QDF_MODULE_ID_DP, QDF_TRACE_LEVEL_ERROR,
  268. FL("HWIO_REO_R0_DESTINATION_RING_CTRL_IX_2_ADDR 0x%x\n"),
  269. HAL_REG_READ(soc,
  270. HWIO_REO_R0_DESTINATION_RING_CTRL_IX_2_ADDR(
  271. SEQ_WCSS_UMAC_REO_REG_OFFSET)));
  272. HAL_REG_WRITE(soc,
  273. HWIO_REO_R0_DESTINATION_RING_CTRL_IX_3_ADDR(
  274. SEQ_WCSS_UMAC_REO_REG_OFFSET),
  275. REO_REMAP_REGISTER_3);
  276. QDF_TRACE(QDF_MODULE_ID_DP, QDF_TRACE_LEVEL_ERROR,
  277. FL("HWIO_REO_R0_DESTINATION_RING_CTRL_IX_3_ADDR 0x%x\n"),
  278. HAL_REG_READ(soc,
  279. HWIO_REO_R0_DESTINATION_RING_CTRL_IX_3_ADDR(
  280. SEQ_WCSS_UMAC_REO_REG_OFFSET)));
  281. }
  282. /* TODO: Check if the following registers shoould be setup by host:
  283. * AGING_CONTROL
  284. * HIGH_MEMORY_THRESHOLD
  285. * GLOBAL_LINK_DESC_COUNT_THRESH_IX_0[1,2]
  286. * GLOBAL_LINK_DESC_COUNT_CTRL
  287. */
  288. }