ce_diag.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457
  1. /*
  2. * Copyright (c) 2015-2016 The Linux Foundation. All rights reserved.
  3. *
  4. * Previously licensed under the ISC license by Qualcomm Atheros, Inc.
  5. *
  6. *
  7. * Permission to use, copy, modify, and/or distribute this software for
  8. * any purpose with or without fee is hereby granted, provided that the
  9. * above copyright notice and this permission notice appear in all
  10. * copies.
  11. *
  12. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
  13. * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
  14. * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
  15. * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  16. * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
  17. * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
  18. * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  19. * PERFORMANCE OF THIS SOFTWARE.
  20. */
  21. /*
  22. * This file was originally distributed by Qualcomm Atheros, Inc.
  23. * under proprietary terms before Copyright ownership was assigned
  24. * to the Linux Foundation.
  25. */
  26. #include <osdep.h>
  27. #include "a_types.h"
  28. #include "athdefs.h"
  29. #include "osapi_linux.h"
  30. #include "targcfg.h"
  31. #include "qdf_lock.h"
  32. #include "qdf_status.h"
  33. #include "qdf_status.h"
  34. #include <qdf_atomic.h> /* qdf_atomic_read */
  35. #include <targaddrs.h>
  36. #include <bmi_msg.h>
  37. #include "hif_io32.h"
  38. #include <hif.h>
  39. #include <htc_services.h>
  40. #include "regtable.h"
  41. #include <a_debug.h>
  42. #include "hif_main.h"
  43. #include "ce_api.h"
  44. #include "qdf_trace.h"
  45. #ifdef CONFIG_CNSS
  46. #include <net/cnss.h>
  47. #endif
  48. #include "hif_debug.h"
  49. #include "epping_main.h"
  50. void
  51. hif_dump_target_memory(struct hif_opaque_softc *hif_ctx, void *ramdump_base,
  52. uint32_t address, uint32_t size)
  53. {
  54. struct hif_softc *scn = HIF_GET_SOFTC(hif_ctx);
  55. uint32_t loc = address;
  56. uint32_t val = 0;
  57. uint32_t j = 0;
  58. u8 *temp = ramdump_base;
  59. if (Q_TARGET_ACCESS_BEGIN(scn) < 0)
  60. return;
  61. while (j < size) {
  62. val = hif_read32_mb(scn->mem + loc + j);
  63. qdf_mem_copy(temp, &val, 4);
  64. j += 4;
  65. temp += 4;
  66. }
  67. Q_TARGET_ACCESS_END(scn);
  68. }
  69. /*
  70. * TBDXXX: Should be a function call specific to each Target-type.
  71. * This convoluted macro converts from Target CPU Virtual Address
  72. * Space to CE Address Space. As part of this process, we
  73. * conservatively fetch the current PCIE_BAR. MOST of the time,
  74. * this should match the upper bits of PCI space for this device;
  75. * but that's not guaranteed.
  76. */
  77. #ifdef QCA_WIFI_3_0
  78. #define TARG_CPU_SPACE_TO_CE_SPACE(pci_addr, addr) \
  79. (scn->mem_pa + addr)
  80. #else
  81. #define TARG_CPU_SPACE_TO_CE_SPACE(pci_addr, addr) \
  82. (((hif_read32_mb((pci_addr) + \
  83. (SOC_CORE_BASE_ADDRESS|CORE_CTRL_ADDRESS)) & 0x7ff) << 21) \
  84. | 0x100000 | ((addr) & 0xfffff))
  85. #endif
  86. /* Wait up to this many Ms for a Diagnostic Access CE operation to complete */
  87. #define DIAG_ACCESS_CE_TIMEOUT_MS 10
  88. /*
  89. * Diagnostic read/write access is provided for startup/config/debug usage.
  90. * Caller must guarantee proper alignment, when applicable, and single user
  91. * at any moment.
  92. */
  93. QDF_STATUS
  94. hif_diag_read_mem(struct hif_opaque_softc *hif_ctx, uint32_t address,
  95. uint8_t *data, int nbytes)
  96. {
  97. struct hif_softc *scn = HIF_GET_SOFTC(hif_ctx);
  98. struct HIF_CE_state *hif_state = HIF_GET_CE_STATE(scn);
  99. QDF_STATUS status = QDF_STATUS_SUCCESS;
  100. qdf_dma_addr_t buf;
  101. unsigned int completed_nbytes, orig_nbytes, remaining_bytes;
  102. unsigned int id;
  103. unsigned int flags;
  104. struct CE_handle *ce_diag;
  105. qdf_dma_addr_t CE_data; /* Host buffer address in CE space */
  106. qdf_dma_addr_t CE_data_base = 0;
  107. void *data_buf = NULL;
  108. int i;
  109. unsigned int mux_id = 0;
  110. unsigned int transaction_id = 0xffff;
  111. qdf_dma_addr_t ce_phy_addr = address;
  112. unsigned int toeplitz_hash_result;
  113. unsigned int user_flags = 0;
  114. transaction_id = (mux_id & MUX_ID_MASK) |
  115. (transaction_id & TRANSACTION_ID_MASK);
  116. #ifdef QCA_WIFI_3_0
  117. user_flags &= DESC_DATA_FLAG_MASK;
  118. #endif
  119. /* This code cannot handle reads to non-memory space. Redirect to the
  120. * register read fn but preserve the multi word read capability of
  121. * this fn
  122. */
  123. if (address < DRAM_BASE_ADDRESS) {
  124. if ((address & 0x3) || ((uintptr_t) data & 0x3))
  125. return QDF_STATUS_E_INVAL;
  126. while ((nbytes >= 4) &&
  127. (QDF_STATUS_SUCCESS == (status =
  128. hif_diag_read_access(hif_ctx, address,
  129. (uint32_t *)data)))) {
  130. nbytes -= sizeof(uint32_t);
  131. address += sizeof(uint32_t);
  132. data += sizeof(uint32_t);
  133. }
  134. return status;
  135. }
  136. ce_diag = hif_state->ce_diag;
  137. A_TARGET_ACCESS_LIKELY(scn);
  138. /*
  139. * Allocate a temporary bounce buffer to hold caller's data
  140. * to be DMA'ed from Target. This guarantees
  141. * 1) 4-byte alignment
  142. * 2) Buffer in DMA-able space
  143. */
  144. orig_nbytes = nbytes;
  145. data_buf = qdf_mem_alloc_consistent(scn->qdf_dev, scn->qdf_dev->dev,
  146. orig_nbytes, &CE_data_base);
  147. if (!data_buf) {
  148. status = QDF_STATUS_E_NOMEM;
  149. goto done;
  150. }
  151. qdf_mem_set(data_buf, orig_nbytes, 0);
  152. qdf_mem_dma_sync_single_for_device(scn->qdf_dev, CE_data_base,
  153. orig_nbytes, DMA_FROM_DEVICE);
  154. remaining_bytes = orig_nbytes;
  155. CE_data = CE_data_base;
  156. while (remaining_bytes) {
  157. nbytes = min(remaining_bytes, DIAG_TRANSFER_LIMIT);
  158. {
  159. status = ce_recv_buf_enqueue(ce_diag, NULL, CE_data);
  160. if (status != QDF_STATUS_SUCCESS)
  161. goto done;
  162. }
  163. { /* Request CE to send from Target(!)
  164. * address to Host buffer */
  165. /*
  166. * The address supplied by the caller is in the
  167. * Target CPU virtual address space.
  168. *
  169. * In order to use this address with the diagnostic CE,
  170. * convert it from
  171. * Target CPU virtual address space
  172. * to
  173. * CE address space
  174. */
  175. A_TARGET_ACCESS_BEGIN_RET(scn);
  176. ce_phy_addr =
  177. TARG_CPU_SPACE_TO_CE_SPACE(scn->mem, address);
  178. A_TARGET_ACCESS_END_RET(scn);
  179. status =
  180. ce_send(ce_diag, NULL, ce_phy_addr, nbytes,
  181. transaction_id, 0, user_flags);
  182. if (status != QDF_STATUS_SUCCESS)
  183. goto done;
  184. }
  185. i = 0;
  186. while (ce_completed_send_next(ce_diag, NULL, NULL, &buf,
  187. &completed_nbytes, &id, NULL, NULL,
  188. &toeplitz_hash_result) != QDF_STATUS_SUCCESS) {
  189. qdf_mdelay(1);
  190. if (i++ > DIAG_ACCESS_CE_TIMEOUT_MS) {
  191. status = QDF_STATUS_E_BUSY;
  192. goto done;
  193. }
  194. }
  195. if (nbytes != completed_nbytes) {
  196. status = QDF_STATUS_E_FAILURE;
  197. goto done;
  198. }
  199. if (buf != ce_phy_addr) {
  200. status = QDF_STATUS_E_FAILURE;
  201. goto done;
  202. }
  203. i = 0;
  204. while (ce_completed_recv_next
  205. (ce_diag, NULL, NULL, &buf,
  206. &completed_nbytes, &id,
  207. &flags) != QDF_STATUS_SUCCESS) {
  208. qdf_mdelay(1);
  209. if (i++ > DIAG_ACCESS_CE_TIMEOUT_MS) {
  210. status = QDF_STATUS_E_BUSY;
  211. goto done;
  212. }
  213. }
  214. if (nbytes != completed_nbytes) {
  215. status = QDF_STATUS_E_FAILURE;
  216. goto done;
  217. }
  218. if (buf != CE_data) {
  219. status = QDF_STATUS_E_FAILURE;
  220. goto done;
  221. }
  222. remaining_bytes -= nbytes;
  223. address += nbytes;
  224. CE_data += nbytes;
  225. }
  226. done:
  227. A_TARGET_ACCESS_UNLIKELY(scn);
  228. if (status == QDF_STATUS_SUCCESS)
  229. qdf_mem_copy(data, data_buf, orig_nbytes);
  230. else
  231. HIF_ERROR("%s failure (0x%x)", __func__, address);
  232. if (data_buf)
  233. qdf_mem_free_consistent(scn->qdf_dev, scn->qdf_dev->dev,
  234. orig_nbytes, data_buf, CE_data_base, 0);
  235. return status;
  236. }
  237. /* Read 4-byte aligned data from Target memory or register */
  238. QDF_STATUS hif_diag_read_access(struct hif_opaque_softc *hif_ctx,
  239. uint32_t address, uint32_t *data)
  240. {
  241. struct hif_softc *scn = HIF_GET_SOFTC(hif_ctx);
  242. if (address >= DRAM_BASE_ADDRESS) {
  243. /* Assume range doesn't cross this boundary */
  244. return hif_diag_read_mem(hif_ctx, address, (uint8_t *) data,
  245. sizeof(uint32_t));
  246. } else {
  247. A_TARGET_ACCESS_BEGIN_RET(scn);
  248. *data = A_TARGET_READ(scn, address);
  249. A_TARGET_ACCESS_END_RET(scn);
  250. return QDF_STATUS_SUCCESS;
  251. }
  252. }
  253. QDF_STATUS hif_diag_write_mem(struct hif_opaque_softc *hif_ctx,
  254. uint32_t address, uint8_t *data, int nbytes)
  255. {
  256. struct hif_softc *scn = HIF_GET_SOFTC(hif_ctx);
  257. struct HIF_CE_state *hif_state = HIF_GET_CE_STATE(hif_ctx);
  258. QDF_STATUS status = QDF_STATUS_SUCCESS;
  259. qdf_dma_addr_t buf;
  260. unsigned int completed_nbytes, orig_nbytes, remaining_bytes;
  261. unsigned int id;
  262. unsigned int flags;
  263. struct CE_handle *ce_diag;
  264. void *data_buf = NULL;
  265. qdf_dma_addr_t CE_data; /* Host buffer address in CE space */
  266. qdf_dma_addr_t CE_data_base = 0;
  267. int i;
  268. unsigned int mux_id = 0;
  269. unsigned int transaction_id = 0xffff;
  270. qdf_dma_addr_t ce_phy_addr = address;
  271. unsigned int toeplitz_hash_result;
  272. unsigned int user_flags = 0;
  273. ce_diag = hif_state->ce_diag;
  274. transaction_id = (mux_id & MUX_ID_MASK) |
  275. (transaction_id & TRANSACTION_ID_MASK);
  276. #ifdef QCA_WIFI_3_0
  277. user_flags &= DESC_DATA_FLAG_MASK;
  278. #endif
  279. A_TARGET_ACCESS_LIKELY(scn);
  280. /*
  281. * Allocate a temporary bounce buffer to hold caller's data
  282. * to be DMA'ed to Target. This guarantees
  283. * 1) 4-byte alignment
  284. * 2) Buffer in DMA-able space
  285. */
  286. orig_nbytes = nbytes;
  287. data_buf = qdf_mem_alloc_consistent(scn->qdf_dev, scn->qdf_dev->dev,
  288. orig_nbytes, &CE_data_base);
  289. if (!data_buf) {
  290. status = A_NO_MEMORY;
  291. goto done;
  292. }
  293. /* Copy caller's data to allocated DMA buf */
  294. qdf_mem_copy(data_buf, data, orig_nbytes);
  295. qdf_mem_dma_sync_single_for_device(scn->qdf_dev, CE_data_base,
  296. orig_nbytes, DMA_TO_DEVICE);
  297. /*
  298. * The address supplied by the caller is in the
  299. * Target CPU virtual address space.
  300. *
  301. * In order to use this address with the diagnostic CE,
  302. * convert it from
  303. * Target CPU virtual address space
  304. * to
  305. * CE address space
  306. */
  307. A_TARGET_ACCESS_BEGIN_RET(scn);
  308. ce_phy_addr = TARG_CPU_SPACE_TO_CE_SPACE(scn->mem, address);
  309. A_TARGET_ACCESS_END_RET(scn);
  310. remaining_bytes = orig_nbytes;
  311. CE_data = CE_data_base;
  312. while (remaining_bytes) {
  313. nbytes = min(remaining_bytes, DIAG_TRANSFER_LIMIT);
  314. { /* Set up to receive directly into Target(!) address */
  315. status = ce_recv_buf_enqueue(ce_diag,
  316. NULL, ce_phy_addr);
  317. if (status != QDF_STATUS_SUCCESS)
  318. goto done;
  319. }
  320. {
  321. /*
  322. * Request CE to send caller-supplied data that
  323. * was copied to bounce buffer to Target(!) address.
  324. */
  325. status =
  326. ce_send(ce_diag, NULL,
  327. (qdf_dma_addr_t) CE_data, nbytes,
  328. transaction_id, 0, user_flags);
  329. if (status != QDF_STATUS_SUCCESS)
  330. goto done;
  331. }
  332. i = 0;
  333. while (ce_completed_send_next(ce_diag, NULL, NULL, &buf,
  334. &completed_nbytes, &id,
  335. NULL, NULL, &toeplitz_hash_result) !=
  336. QDF_STATUS_SUCCESS) {
  337. qdf_mdelay(1);
  338. if (i++ > DIAG_ACCESS_CE_TIMEOUT_MS) {
  339. status = QDF_STATUS_E_BUSY;
  340. goto done;
  341. }
  342. }
  343. if (nbytes != completed_nbytes) {
  344. status = QDF_STATUS_E_FAILURE;
  345. goto done;
  346. }
  347. if (buf != CE_data) {
  348. status = QDF_STATUS_E_FAILURE;
  349. goto done;
  350. }
  351. i = 0;
  352. while (ce_completed_recv_next
  353. (ce_diag, NULL, NULL, &buf,
  354. &completed_nbytes, &id,
  355. &flags) != QDF_STATUS_SUCCESS) {
  356. qdf_mdelay(1);
  357. if (i++ > DIAG_ACCESS_CE_TIMEOUT_MS) {
  358. status = QDF_STATUS_E_BUSY;
  359. goto done;
  360. }
  361. }
  362. if (nbytes != completed_nbytes) {
  363. status = QDF_STATUS_E_FAILURE;
  364. goto done;
  365. }
  366. if (buf != ce_phy_addr) {
  367. status = QDF_STATUS_E_FAILURE;
  368. goto done;
  369. }
  370. remaining_bytes -= nbytes;
  371. address += nbytes;
  372. CE_data += nbytes;
  373. }
  374. done:
  375. A_TARGET_ACCESS_UNLIKELY(scn);
  376. if (data_buf) {
  377. qdf_mem_free_consistent(scn->qdf_dev, scn->qdf_dev->dev,
  378. orig_nbytes, data_buf, CE_data_base, 0);
  379. }
  380. if (status != QDF_STATUS_SUCCESS) {
  381. HIF_ERROR("%s failure (0x%llu)", __func__,
  382. (uint64_t)ce_phy_addr);
  383. }
  384. return status;
  385. }
  386. /* Write 4B data to Target memory or register */
  387. QDF_STATUS hif_diag_write_access(struct hif_opaque_softc *hif_ctx,
  388. uint32_t address, uint32_t data)
  389. {
  390. struct hif_softc *scn = HIF_GET_SOFTC(hif_ctx);
  391. if (address >= DRAM_BASE_ADDRESS) {
  392. /* Assume range doesn't cross this boundary */
  393. uint32_t data_buf = data;
  394. return hif_diag_write_mem(hif_ctx, address,
  395. (uint8_t *) &data_buf,
  396. sizeof(uint32_t));
  397. } else {
  398. A_TARGET_ACCESS_BEGIN_RET(scn);
  399. A_TARGET_WRITE(scn, address, data);
  400. A_TARGET_ACCESS_END_RET(scn);
  401. return QDF_STATUS_SUCCESS;
  402. }
  403. }