htc.c 28 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081
  1. /*
  2. * Copyright (c) 2013-2017 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 "htc_debug.h"
  27. #include "htc_internal.h"
  28. #include <hif.h>
  29. #include <qdf_nbuf.h> /* qdf_nbuf_t */
  30. #include <qdf_types.h> /* qdf_print */
  31. #define MAX_HTC_RX_BUNDLE 2
  32. #if defined(WLAN_DEBUG) || defined(DEBUG)
  33. static ATH_DEBUG_MASK_DESCRIPTION g_htc_debug_description[] = {
  34. {ATH_DEBUG_SEND, "Send"},
  35. {ATH_DEBUG_RECV, "Recv"},
  36. {ATH_DEBUG_SYNC, "Sync"},
  37. {ATH_DEBUG_DUMP, "Dump Data (RX or TX)"},
  38. {ATH_DEBUG_SETUP, "Setup"},
  39. };
  40. ATH_DEBUG_INSTANTIATE_MODULE_VAR(htc,
  41. "htc",
  42. "Host Target Communications",
  43. ATH_DEBUG_MASK_DEFAULTS | ATH_DEBUG_INFO |
  44. ATH_DEBUG_SETUP,
  45. ATH_DEBUG_DESCRIPTION_COUNT
  46. (g_htc_debug_description),
  47. g_htc_debug_description);
  48. #endif
  49. extern unsigned int htc_credit_flow;
  50. static void reset_endpoint_states(HTC_TARGET *target);
  51. static void destroy_htc_tx_ctrl_packet(HTC_PACKET *pPacket)
  52. {
  53. qdf_nbuf_t netbuf;
  54. netbuf = (qdf_nbuf_t) GET_HTC_PACKET_NET_BUF_CONTEXT(pPacket);
  55. AR_DEBUG_PRINTF(ATH_DEBUG_TRC, ("free ctrl netbuf :0x%p\n", netbuf));
  56. if (netbuf != NULL)
  57. qdf_nbuf_free(netbuf);
  58. qdf_mem_free(pPacket);
  59. }
  60. static HTC_PACKET *build_htc_tx_ctrl_packet(qdf_device_t osdev)
  61. {
  62. HTC_PACKET *pPacket = NULL;
  63. qdf_nbuf_t netbuf;
  64. do {
  65. pPacket = (HTC_PACKET *) qdf_mem_malloc(sizeof(HTC_PACKET));
  66. if (pPacket == NULL)
  67. break;
  68. netbuf = qdf_nbuf_alloc(osdev, HTC_CONTROL_BUFFER_SIZE,
  69. 20, 4, true);
  70. if (NULL == netbuf) {
  71. qdf_mem_free(pPacket);
  72. pPacket = NULL;
  73. qdf_print("%s: nbuf alloc failed\n", __func__);
  74. break;
  75. }
  76. AR_DEBUG_PRINTF(ATH_DEBUG_TRC,
  77. ("alloc ctrl netbuf :0x%p\n", netbuf));
  78. SET_HTC_PACKET_NET_BUF_CONTEXT(pPacket, netbuf);
  79. } while (false);
  80. return pPacket;
  81. }
  82. void htc_free_control_tx_packet(HTC_TARGET *target, HTC_PACKET *pPacket)
  83. {
  84. #ifdef TODO_FIXME
  85. LOCK_HTC(target);
  86. HTC_PACKET_ENQUEUE(&target->ControlBufferTXFreeList, pPacket);
  87. UNLOCK_HTC(target);
  88. /* TODO_FIXME netbufs cannot be RESET! */
  89. #else
  90. destroy_htc_tx_ctrl_packet(pPacket);
  91. #endif
  92. }
  93. HTC_PACKET *htc_alloc_control_tx_packet(HTC_TARGET *target)
  94. {
  95. #ifdef TODO_FIXME
  96. HTC_PACKET *pPacket;
  97. LOCK_HTC(target);
  98. pPacket = htc_packet_dequeue(&target->ControlBufferTXFreeList);
  99. UNLOCK_HTC(target);
  100. return pPacket;
  101. #else
  102. return build_htc_tx_ctrl_packet(target->osdev);
  103. #endif
  104. }
  105. /* Set the target failure handling callback */
  106. void htc_set_target_failure_callback(HTC_HANDLE HTCHandle,
  107. HTC_TARGET_FAILURE Callback)
  108. {
  109. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  110. target->HTCInitInfo.TargetFailure = Callback;
  111. }
  112. void htc_dump(HTC_HANDLE HTCHandle, uint8_t CmdId, bool start)
  113. {
  114. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  115. hif_dump(target->hif_dev, CmdId, start);
  116. }
  117. /* cleanup the HTC instance */
  118. static void htc_cleanup(HTC_TARGET *target)
  119. {
  120. HTC_PACKET *pPacket;
  121. /* qdf_nbuf_t netbuf; */
  122. if (target->hif_dev != NULL) {
  123. hif_detach_htc(target->hif_dev);
  124. hif_mask_interrupt_call(target->hif_dev);
  125. target->hif_dev = NULL;
  126. }
  127. while (true) {
  128. pPacket = allocate_htc_packet_container(target);
  129. if (pPacket == NULL)
  130. break;
  131. qdf_mem_free(pPacket);
  132. }
  133. pPacket = target->pBundleFreeList;
  134. while (pPacket) {
  135. HTC_PACKET *pPacketTmp = (HTC_PACKET *) pPacket->ListLink.pNext;
  136. qdf_mem_free(pPacket);
  137. pPacket = pPacketTmp;
  138. }
  139. #ifdef TODO_FIXME
  140. while (true) {
  141. pPacket = htc_alloc_control_tx_packet(target);
  142. if (pPacket == NULL)
  143. break;
  144. netbuf = (qdf_nbuf_t) GET_HTC_PACKET_NET_BUF_CONTEXT(pPacket);
  145. if (netbuf != NULL)
  146. qdf_nbuf_free(netbuf);
  147. qdf_mem_free(pPacket);
  148. }
  149. #endif
  150. qdf_spinlock_destroy(&target->HTCLock);
  151. qdf_spinlock_destroy(&target->HTCRxLock);
  152. qdf_spinlock_destroy(&target->HTCTxLock);
  153. qdf_spinlock_destroy(&target->HTCCreditLock);
  154. /* free our instance */
  155. qdf_mem_free(target);
  156. }
  157. #ifdef FEATURE_RUNTIME_PM
  158. /**
  159. * htc_runtime_pm_init(): runtime pm related intialization
  160. *
  161. * need to initialize a work item.
  162. */
  163. static void htc_runtime_pm_init(HTC_TARGET *target)
  164. {
  165. qdf_create_work(0, &target->queue_kicker, htc_kick_queues, target);
  166. }
  167. /**
  168. * htc_runtime_suspend() - runtime suspend HTC
  169. *
  170. * @htc_ctx: HTC context pointer
  171. *
  172. * This is a dummy function for symmetry.
  173. *
  174. * Return: 0 for success
  175. */
  176. int htc_runtime_suspend(HTC_HANDLE htc_ctx)
  177. {
  178. return 0;
  179. }
  180. /**
  181. * htc_runtime_resume(): resume htc
  182. *
  183. * The htc message queue needs to be kicked off after
  184. * a runtime resume. Otherwise messages would get stuck.
  185. *
  186. * @htc_ctx: HTC context pointer
  187. *
  188. * Return: 0 for success;
  189. */
  190. int htc_runtime_resume(HTC_HANDLE htc_ctx)
  191. {
  192. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(htc_ctx);
  193. if (target == NULL)
  194. return 0;
  195. qdf_sched_work(0, &target->queue_kicker);
  196. return 0;
  197. }
  198. #else
  199. static inline void htc_runtime_pm_init(HTC_TARGET *target) { }
  200. #endif
  201. /* registered target arrival callback from the HIF layer */
  202. HTC_HANDLE htc_create(void *ol_sc, struct htc_init_info *pInfo,
  203. qdf_device_t osdev, uint32_t con_mode)
  204. {
  205. struct hif_msg_callbacks htcCallbacks;
  206. HTC_ENDPOINT *pEndpoint = NULL;
  207. HTC_TARGET *target = NULL;
  208. int i;
  209. if (ol_sc == NULL) {
  210. HTC_ERROR("%s: ol_sc = NULL", __func__);
  211. return NULL;
  212. }
  213. HTC_TRACE("+htc_create .. HIF :%p", ol_sc);
  214. A_REGISTER_MODULE_DEBUG_INFO(htc);
  215. target = (HTC_TARGET *) qdf_mem_malloc(sizeof(HTC_TARGET));
  216. if (target == NULL) {
  217. HTC_ERROR("%s: Unable to allocate memory", __func__);
  218. return NULL;
  219. }
  220. htc_runtime_pm_init(target);
  221. qdf_spinlock_create(&target->HTCLock);
  222. qdf_spinlock_create(&target->HTCRxLock);
  223. qdf_spinlock_create(&target->HTCTxLock);
  224. qdf_spinlock_create(&target->HTCCreditLock);
  225. target->is_nodrop_pkt = false;
  226. target->wmi_ep_count = 1;
  227. do {
  228. qdf_mem_copy(&target->HTCInitInfo, pInfo,
  229. sizeof(struct htc_init_info));
  230. target->host_handle = pInfo->pContext;
  231. target->osdev = osdev;
  232. target->con_mode = con_mode;
  233. reset_endpoint_states(target);
  234. INIT_HTC_PACKET_QUEUE(&target->ControlBufferTXFreeList);
  235. for (i = 0; i < HTC_PACKET_CONTAINER_ALLOCATION; i++) {
  236. HTC_PACKET *pPacket = (HTC_PACKET *)
  237. qdf_mem_malloc(sizeof(HTC_PACKET));
  238. if (pPacket != NULL)
  239. free_htc_packet_container(target, pPacket);
  240. }
  241. #ifdef TODO_FIXME
  242. for (i = 0; i < NUM_CONTROL_TX_BUFFERS; i++) {
  243. pPacket = build_htc_tx_ctrl_packet();
  244. if (pPacket == NULL)
  245. break;
  246. htc_free_control_tx_packet(target, pPacket);
  247. }
  248. #endif
  249. /* setup HIF layer callbacks */
  250. qdf_mem_zero(&htcCallbacks, sizeof(struct hif_msg_callbacks));
  251. htcCallbacks.Context = target;
  252. htcCallbacks.rxCompletionHandler = htc_rx_completion_handler;
  253. htcCallbacks.txCompletionHandler = htc_tx_completion_handler;
  254. htcCallbacks.txResourceAvailHandler =
  255. htc_tx_resource_avail_handler;
  256. htcCallbacks.fwEventHandler = htc_fw_event_handler;
  257. target->hif_dev = ol_sc;
  258. /* Get HIF default pipe for HTC message exchange */
  259. pEndpoint = &target->endpoint[ENDPOINT_0];
  260. hif_post_init(target->hif_dev, target, &htcCallbacks);
  261. hif_get_default_pipe(target->hif_dev, &pEndpoint->UL_PipeID,
  262. &pEndpoint->DL_PipeID);
  263. } while (false);
  264. htc_recv_init(target);
  265. HTC_TRACE("-htc_create: (0x%p)", target);
  266. return (HTC_HANDLE) target;
  267. }
  268. void htc_destroy(HTC_HANDLE HTCHandle)
  269. {
  270. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  271. AR_DEBUG_PRINTF(ATH_DEBUG_TRC,
  272. ("+htc_destroy .. Destroying :0x%p\n", target));
  273. hif_stop(htc_get_hif_device(HTCHandle));
  274. if (target)
  275. htc_cleanup(target);
  276. AR_DEBUG_PRINTF(ATH_DEBUG_TRC, ("-htc_destroy\n"));
  277. }
  278. /* get the low level HIF device for the caller , the caller may wish to do low
  279. * level HIF requests
  280. */
  281. void *htc_get_hif_device(HTC_HANDLE HTCHandle)
  282. {
  283. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  284. return target->hif_dev;
  285. }
  286. static void htc_control_tx_complete(void *Context, HTC_PACKET *pPacket)
  287. {
  288. HTC_TARGET *target = (HTC_TARGET *) Context;
  289. AR_DEBUG_PRINTF(ATH_DEBUG_TRC,
  290. ("+-htc_control_tx_complete 0x%p (l:%d)\n", pPacket,
  291. pPacket->ActualLength));
  292. htc_free_control_tx_packet(target, pPacket);
  293. }
  294. /* TODO, this is just a temporary max packet size */
  295. #define MAX_MESSAGE_SIZE 1536
  296. /**
  297. * htc_setup_epping_credit_allocation() - allocate credits/HTC buffers to WMI
  298. * @scn: pointer to hif_opaque_softc
  299. * @pEntry: pointer to tx credit allocation entry
  300. * @credits: number of credits
  301. *
  302. * Return: None
  303. */
  304. static void
  305. htc_setup_epping_credit_allocation(struct hif_opaque_softc *scn,
  306. struct htc_service_tx_credit_allocation *pEntry,
  307. int credits)
  308. {
  309. switch (hif_get_bus_type(scn)) {
  310. case QDF_BUS_TYPE_PCI:
  311. pEntry++;
  312. pEntry->service_id = WMI_DATA_BE_SVC;
  313. pEntry->CreditAllocation = (credits >> 1);
  314. pEntry++;
  315. pEntry->service_id = WMI_DATA_BK_SVC;
  316. pEntry->CreditAllocation = (credits >> 1);
  317. break;
  318. case QDF_BUS_TYPE_SDIO:
  319. pEntry++;
  320. pEntry->service_id = WMI_DATA_BE_SVC;
  321. pEntry->CreditAllocation = credits;
  322. break;
  323. default:
  324. break;
  325. }
  326. }
  327. /**
  328. * htc_setup_target_buffer_assignments() - setup target buffer assignments
  329. * @target: HTC Target Pointer
  330. *
  331. * Return: A_STATUS
  332. */
  333. static
  334. A_STATUS htc_setup_target_buffer_assignments(HTC_TARGET *target)
  335. {
  336. struct htc_service_tx_credit_allocation *pEntry;
  337. A_STATUS status;
  338. int credits;
  339. int creditsPerMaxMsg;
  340. creditsPerMaxMsg = MAX_MESSAGE_SIZE / target->TargetCreditSize;
  341. if (MAX_MESSAGE_SIZE % target->TargetCreditSize)
  342. creditsPerMaxMsg++;
  343. /* TODO, this should be configured by the caller! */
  344. credits = target->TotalTransmitCredits;
  345. pEntry = &target->ServiceTxAllocTable[0];
  346. status = A_OK;
  347. /*
  348. * Allocate all credists/HTC buffers to WMI.
  349. * no buffers are used/required for data. data always
  350. * remains on host.
  351. */
  352. if (HTC_IS_EPPING_ENABLED(target->con_mode)) {
  353. pEntry++;
  354. pEntry->service_id = WMI_CONTROL_SVC;
  355. pEntry->CreditAllocation = credits;
  356. /* endpoint ping is a testing tool directly on top of HTC in
  357. * both target and host sides.
  358. * In target side, the endppint ping fw has no wlan stack and
  359. * FW mboxping app directly sits on HTC and it simply drops
  360. * or loops back TX packets. For rx perf, FW mboxping app
  361. * generates packets and passes packets to HTC to send to host.
  362. * There is no WMI mesage exchanges between host and target
  363. * in endpoint ping case.
  364. * In host side, the endpoint ping driver is a Ethernet driver
  365. * and it directly sits on HTC. Only HIF, HTC, QDF, ADF are
  366. * used by the endpoint ping driver. There is no wifi stack
  367. * at all in host side also. For tx perf use case,
  368. * the user space mboxping app sends the raw packets to endpoint
  369. * ping driver and it directly forwards to HTC for transmission
  370. * to stress the bus. For the rx perf, HTC passes the received
  371. * packets to endpoint ping driver and it is passed to the user
  372. * space through the Ethernet interface.
  373. * For credit allocation, in SDIO bus case, only BE service is
  374. * used for tx/rx perf testing so that all credits are given
  375. * to BE service. In PCIe and USB bus case, endpoint ping uses
  376. * both BE and BK services to stress the bus so that the total
  377. * credits are equally distributed to BE and BK services.
  378. */
  379. htc_setup_epping_credit_allocation(target->hif_dev,
  380. pEntry, credits);
  381. } else {
  382. int i;
  383. uint32_t svc_id[] = {WMI_CONTROL_SVC, WMI_CONTROL_SVC_WMAC1,
  384. WMI_CONTROL_SVC_WMAC2};
  385. uint32_t max_wmi_svc = (sizeof(svc_id) / sizeof(uint32_t));
  386. if ((target->wmi_ep_count == 0) ||
  387. (target->wmi_ep_count > max_wmi_svc))
  388. return A_ERROR;
  389. /*
  390. * Divide credit among number of endpoints for WMI
  391. */
  392. credits = credits / target->wmi_ep_count;
  393. for (i = 0; i < target->wmi_ep_count; i++) {
  394. status = A_OK;
  395. pEntry++;
  396. pEntry->service_id = svc_id[i];
  397. pEntry->CreditAllocation = credits;
  398. }
  399. }
  400. if (A_SUCCESS(status)) {
  401. int i;
  402. for (i = 0; i < HTC_MAX_SERVICE_ALLOC_ENTRIES; i++) {
  403. if (target->ServiceTxAllocTable[i].service_id != 0) {
  404. AR_DEBUG_PRINTF(ATH_DEBUG_INIT,
  405. ("HTC Service Index : %d TX : 0x%2.2X : alloc:%d\n",
  406. i,
  407. target->ServiceTxAllocTable[i].
  408. service_id,
  409. target->ServiceTxAllocTable[i].
  410. CreditAllocation));
  411. }
  412. }
  413. }
  414. return status;
  415. }
  416. uint8_t htc_get_credit_allocation(HTC_TARGET *target, uint16_t service_id)
  417. {
  418. uint8_t allocation = 0;
  419. int i;
  420. for (i = 0; i < HTC_MAX_SERVICE_ALLOC_ENTRIES; i++) {
  421. if (target->ServiceTxAllocTable[i].service_id == service_id) {
  422. allocation =
  423. target->ServiceTxAllocTable[i].CreditAllocation;
  424. }
  425. }
  426. if (0 == allocation) {
  427. AR_DEBUG_PRINTF(ATH_DEBUG_RSVD1,
  428. ("HTC Service TX : 0x%2.2X : allocation is zero!\n",
  429. service_id));
  430. }
  431. return allocation;
  432. }
  433. QDF_STATUS htc_wait_target(HTC_HANDLE HTCHandle)
  434. {
  435. QDF_STATUS status = QDF_STATUS_SUCCESS;
  436. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  437. HTC_READY_EX_MSG *pReadyMsg;
  438. struct htc_service_connect_req connect;
  439. struct htc_service_connect_resp resp;
  440. HTC_READY_MSG *rdy_msg;
  441. uint16_t htc_rdy_msg_id;
  442. uint8_t i = 0;
  443. HTC_PACKET *rx_bundle_packet, *temp_bundle_packet;
  444. AR_DEBUG_PRINTF(ATH_DEBUG_TRC,
  445. ("htc_wait_target - Enter (target:0x%p)\n", HTCHandle));
  446. AR_DEBUG_PRINTF(ATH_DEBUG_RSVD1, ("+HWT\n"));
  447. do {
  448. status = hif_start(target->hif_dev);
  449. if (QDF_IS_STATUS_ERROR(status)) {
  450. AR_DEBUG_PRINTF(ATH_DEBUG_ERROR,
  451. ("hif_start failed\n"));
  452. break;
  453. }
  454. status = htc_wait_recv_ctrl_message(target);
  455. if (QDF_IS_STATUS_ERROR(status))
  456. break;
  457. if (target->CtrlResponseLength < (sizeof(HTC_READY_EX_MSG))) {
  458. AR_DEBUG_PRINTF(ATH_DEBUG_ERR,
  459. ("Invalid HTC Ready Msg Len:%d!\n",
  460. target->CtrlResponseLength));
  461. status = QDF_STATUS_E_BADMSG;
  462. break;
  463. }
  464. pReadyMsg = (HTC_READY_EX_MSG *) target->CtrlResponseBuffer;
  465. rdy_msg = &pReadyMsg->Version2_0_Info;
  466. htc_rdy_msg_id =
  467. HTC_GET_FIELD(rdy_msg, HTC_READY_MSG, MESSAGEID);
  468. if (htc_rdy_msg_id != HTC_MSG_READY_ID) {
  469. AR_DEBUG_PRINTF(ATH_DEBUG_ERR,
  470. ("Invalid HTC Ready Msg : 0x%X!\n",
  471. htc_rdy_msg_id));
  472. status = QDF_STATUS_E_BADMSG;
  473. break;
  474. }
  475. target->TotalTransmitCredits =
  476. HTC_GET_FIELD(rdy_msg, HTC_READY_MSG, CREDITCOUNT);
  477. target->TargetCreditSize =
  478. (int)HTC_GET_FIELD(rdy_msg, HTC_READY_MSG, CREDITSIZE);
  479. target->MaxMsgsPerHTCBundle =
  480. (uint8_t) pReadyMsg->MaxMsgsPerHTCBundle;
  481. /* for old fw this value is set to 0. But the minimum value
  482. * should be 1, i.e., no bundling
  483. */
  484. if (target->MaxMsgsPerHTCBundle < 1)
  485. target->MaxMsgsPerHTCBundle = 1;
  486. AR_DEBUG_PRINTF(ATH_DEBUG_INIT,
  487. ("Target Ready! : transmit resources : %d size:%d, MaxMsgsPerHTCBundle = %d\n",
  488. target->TotalTransmitCredits,
  489. target->TargetCreditSize,
  490. target->MaxMsgsPerHTCBundle));
  491. if ((0 == target->TotalTransmitCredits)
  492. || (0 == target->TargetCreditSize)) {
  493. status = QDF_STATUS_E_ABORTED;
  494. break;
  495. }
  496. /* Allocate expected number of RX bundle buffer allocation */
  497. if (HTC_RX_BUNDLE_ENABLED(target)) {
  498. temp_bundle_packet = NULL;
  499. for (i = 0; i < MAX_HTC_RX_BUNDLE; i++) {
  500. rx_bundle_packet =
  501. allocate_htc_bundle_packet(target);
  502. if (rx_bundle_packet != NULL)
  503. rx_bundle_packet->ListLink.pNext =
  504. (DL_LIST *)temp_bundle_packet;
  505. else
  506. break;
  507. temp_bundle_packet = rx_bundle_packet;
  508. }
  509. target->pBundleFreeList = temp_bundle_packet;
  510. }
  511. /* done processing */
  512. target->CtrlResponseProcessing = false;
  513. htc_setup_target_buffer_assignments(target);
  514. /* setup our pseudo HTC control endpoint connection */
  515. qdf_mem_zero(&connect, sizeof(connect));
  516. qdf_mem_zero(&resp, sizeof(resp));
  517. connect.EpCallbacks.pContext = target;
  518. connect.EpCallbacks.EpTxComplete = htc_control_tx_complete;
  519. connect.EpCallbacks.EpRecv = htc_control_rx_complete;
  520. connect.MaxSendQueueDepth = NUM_CONTROL_TX_BUFFERS;
  521. connect.service_id = HTC_CTRL_RSVD_SVC;
  522. /* connect fake service */
  523. status = htc_connect_service((HTC_HANDLE) target,
  524. &connect, &resp);
  525. } while (false);
  526. AR_DEBUG_PRINTF(ATH_DEBUG_TRC, ("htc_wait_target - Exit (%d)\n",
  527. status));
  528. AR_DEBUG_PRINTF(ATH_DEBUG_RSVD1, ("-HWT\n"));
  529. return status;
  530. }
  531. /* start HTC, this is called after all services are connected */
  532. static A_STATUS htc_config_target_hif_pipe(HTC_TARGET *target)
  533. {
  534. return A_OK;
  535. }
  536. static void reset_endpoint_states(HTC_TARGET *target)
  537. {
  538. HTC_ENDPOINT *pEndpoint;
  539. int i;
  540. for (i = ENDPOINT_0; i < ENDPOINT_MAX; i++) {
  541. pEndpoint = &target->endpoint[i];
  542. pEndpoint->service_id = 0;
  543. pEndpoint->MaxMsgLength = 0;
  544. pEndpoint->MaxTxQueueDepth = 0;
  545. pEndpoint->Id = i;
  546. INIT_HTC_PACKET_QUEUE(&pEndpoint->TxQueue);
  547. INIT_HTC_PACKET_QUEUE(&pEndpoint->TxLookupQueue);
  548. INIT_HTC_PACKET_QUEUE(&pEndpoint->RxBufferHoldQueue);
  549. pEndpoint->target = target;
  550. pEndpoint->TxCreditFlowEnabled = (bool)htc_credit_flow;
  551. qdf_atomic_init(&pEndpoint->TxProcessCount);
  552. }
  553. }
  554. /**
  555. * htc_start() - Main HTC function to trigger HTC start
  556. * @HTCHandle: pointer to HTC handle
  557. *
  558. * Return: QDF_STATUS_SUCCESS for success or an appropriate QDF_STATUS error
  559. */
  560. QDF_STATUS htc_start(HTC_HANDLE HTCHandle)
  561. {
  562. qdf_nbuf_t netbuf;
  563. QDF_STATUS status = QDF_STATUS_SUCCESS;
  564. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  565. HTC_SETUP_COMPLETE_EX_MSG *pSetupComp;
  566. HTC_PACKET *pSendPacket;
  567. AR_DEBUG_PRINTF(ATH_DEBUG_TRC, ("htc_start Enter\n"));
  568. do {
  569. htc_config_target_hif_pipe(target);
  570. /* allocate a buffer to send */
  571. pSendPacket = htc_alloc_control_tx_packet(target);
  572. if (NULL == pSendPacket) {
  573. AR_DEBUG_ASSERT(false);
  574. qdf_print("%s: allocControlTxPacket failed\n",
  575. __func__);
  576. status = QDF_STATUS_E_NOMEM;
  577. break;
  578. }
  579. netbuf =
  580. (qdf_nbuf_t) GET_HTC_PACKET_NET_BUF_CONTEXT(pSendPacket);
  581. /* assemble setup complete message */
  582. qdf_nbuf_put_tail(netbuf, sizeof(HTC_SETUP_COMPLETE_EX_MSG));
  583. pSetupComp =
  584. (HTC_SETUP_COMPLETE_EX_MSG *) qdf_nbuf_data(netbuf);
  585. qdf_mem_zero(pSetupComp, sizeof(HTC_SETUP_COMPLETE_EX_MSG));
  586. HTC_SET_FIELD(pSetupComp, HTC_SETUP_COMPLETE_EX_MSG,
  587. MESSAGEID, HTC_MSG_SETUP_COMPLETE_EX_ID);
  588. if (!htc_credit_flow) {
  589. AR_DEBUG_PRINTF(ATH_DEBUG_INIT,
  590. ("HTC will not use TX credit flow control\n"));
  591. pSetupComp->SetupFlags |=
  592. HTC_SETUP_COMPLETE_FLAGS_DISABLE_TX_CREDIT_FLOW;
  593. } else {
  594. AR_DEBUG_PRINTF(ATH_DEBUG_INIT,
  595. ("HTC using TX credit flow control\n"));
  596. }
  597. if ((hif_get_bus_type(target->hif_dev) == QDF_BUS_TYPE_SDIO) ||
  598. (hif_get_bus_type(target->hif_dev) ==
  599. QDF_BUS_TYPE_USB)) {
  600. if (HTC_RX_BUNDLE_ENABLED(target))
  601. pSetupComp->SetupFlags |=
  602. HTC_SETUP_COMPLETE_FLAGS_ENABLE_BUNDLE_RECV;
  603. hif_set_bundle_mode(target->hif_dev, true,
  604. HTC_MAX_MSG_PER_BUNDLE_RX);
  605. }
  606. SET_HTC_PACKET_INFO_TX(pSendPacket,
  607. NULL,
  608. (uint8_t *) pSetupComp,
  609. sizeof(HTC_SETUP_COMPLETE_EX_MSG),
  610. ENDPOINT_0, HTC_SERVICE_TX_PACKET_TAG);
  611. status = htc_send_pkt((HTC_HANDLE) target, pSendPacket);
  612. if (QDF_IS_STATUS_ERROR(status))
  613. break;
  614. } while (false);
  615. AR_DEBUG_PRINTF(ATH_DEBUG_TRC, ("htc_start Exit\n"));
  616. return status;
  617. }
  618. /*flush all queued buffers for surpriseremove case*/
  619. void htc_flush_surprise_remove(HTC_HANDLE HTCHandle)
  620. {
  621. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  622. int i;
  623. HTC_ENDPOINT *pEndpoint;
  624. #ifdef RX_SG_SUPPORT
  625. qdf_nbuf_t netbuf;
  626. qdf_nbuf_queue_t *rx_sg_queue = &target->RxSgQueue;
  627. #endif
  628. AR_DEBUG_PRINTF(ATH_DEBUG_TRC, ("+htc_flush_surprise_remove\n"));
  629. /* cleanup endpoints */
  630. for (i = 0; i < ENDPOINT_MAX; i++) {
  631. pEndpoint = &target->endpoint[i];
  632. htc_flush_rx_hold_queue(target, pEndpoint);
  633. htc_flush_endpoint_tx(target, pEndpoint, HTC_TX_PACKET_TAG_ALL);
  634. }
  635. hif_flush_surprise_remove(target->hif_dev);
  636. #ifdef RX_SG_SUPPORT
  637. LOCK_HTC_RX(target);
  638. while ((netbuf = qdf_nbuf_queue_remove(rx_sg_queue)) != NULL)
  639. qdf_nbuf_free(netbuf);
  640. RESET_RX_SG_CONFIG(target);
  641. UNLOCK_HTC_RX(target);
  642. #endif
  643. reset_endpoint_states(target);
  644. AR_DEBUG_PRINTF(ATH_DEBUG_TRC, ("-htc_flush_surprise_remove\n"));
  645. }
  646. /* stop HTC communications, i.e. stop interrupt reception, and flush all queued
  647. * buffers
  648. */
  649. void htc_stop(HTC_HANDLE HTCHandle)
  650. {
  651. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  652. int i;
  653. HTC_ENDPOINT *pEndpoint;
  654. #ifdef RX_SG_SUPPORT
  655. qdf_nbuf_t netbuf;
  656. qdf_nbuf_queue_t *rx_sg_queue = &target->RxSgQueue;
  657. #endif
  658. AR_DEBUG_PRINTF(ATH_DEBUG_TRC, ("+htc_stop\n"));
  659. /* cleanup endpoints */
  660. for (i = 0; i < ENDPOINT_MAX; i++) {
  661. pEndpoint = &target->endpoint[i];
  662. htc_flush_rx_hold_queue(target, pEndpoint);
  663. htc_flush_endpoint_tx(target, pEndpoint, HTC_TX_PACKET_TAG_ALL);
  664. if (pEndpoint->ul_is_polled) {
  665. qdf_timer_stop(&pEndpoint->ul_poll_timer);
  666. qdf_timer_free(&pEndpoint->ul_poll_timer);
  667. }
  668. }
  669. /* Note: htc_flush_endpoint_tx for all endpoints should be called before
  670. * hif_stop - otherwise htc_tx_completion_handler called from
  671. * hif_send_buffer_cleanup_on_pipe for residual tx frames in HIF layer,
  672. * might queue the packet again to HIF Layer - which could cause tx
  673. * buffer leak
  674. */
  675. hif_stop(target->hif_dev);
  676. #ifdef RX_SG_SUPPORT
  677. LOCK_HTC_RX(target);
  678. while ((netbuf = qdf_nbuf_queue_remove(rx_sg_queue)) != NULL)
  679. qdf_nbuf_free(netbuf);
  680. RESET_RX_SG_CONFIG(target);
  681. UNLOCK_HTC_RX(target);
  682. #endif
  683. reset_endpoint_states(target);
  684. AR_DEBUG_PRINTF(ATH_DEBUG_TRC, ("-htc_stop\n"));
  685. }
  686. void htc_dump_credit_states(HTC_HANDLE HTCHandle)
  687. {
  688. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  689. HTC_ENDPOINT *pEndpoint;
  690. int i;
  691. for (i = 0; i < ENDPOINT_MAX; i++) {
  692. pEndpoint = &target->endpoint[i];
  693. if (0 == pEndpoint->service_id)
  694. continue;
  695. AR_DEBUG_PRINTF(ATH_DEBUG_ANY,
  696. ("--- EP : %d service_id: 0x%X --------------\n",
  697. pEndpoint->Id, pEndpoint->service_id));
  698. AR_DEBUG_PRINTF(ATH_DEBUG_ANY,
  699. (" TxCredits : %d\n",
  700. pEndpoint->TxCredits));
  701. AR_DEBUG_PRINTF(ATH_DEBUG_ANY,
  702. (" TxCreditSize : %d\n",
  703. pEndpoint->TxCreditSize));
  704. AR_DEBUG_PRINTF(ATH_DEBUG_ANY,
  705. (" TxCreditsPerMaxMsg : %d\n",
  706. pEndpoint->TxCreditsPerMaxMsg));
  707. AR_DEBUG_PRINTF(ATH_DEBUG_ANY,
  708. (" TxQueueDepth : %d\n",
  709. HTC_PACKET_QUEUE_DEPTH(&pEndpoint->TxQueue)));
  710. AR_DEBUG_PRINTF(ATH_DEBUG_ANY,
  711. ("----------------------------------------\n"));
  712. }
  713. }
  714. bool htc_get_endpoint_statistics(HTC_HANDLE HTCHandle,
  715. HTC_ENDPOINT_ID Endpoint,
  716. enum htc_endpoint_stat_action Action,
  717. struct htc_endpoint_stats *pStats)
  718. {
  719. #ifdef HTC_EP_STAT_PROFILING
  720. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  721. bool clearStats = false;
  722. bool sample = false;
  723. switch (Action) {
  724. case HTC_EP_STAT_SAMPLE:
  725. sample = true;
  726. break;
  727. case HTC_EP_STAT_SAMPLE_AND_CLEAR:
  728. sample = true;
  729. clearStats = true;
  730. break;
  731. case HTC_EP_STAT_CLEAR:
  732. clearStats = true;
  733. break;
  734. default:
  735. break;
  736. }
  737. A_ASSERT(Endpoint < ENDPOINT_MAX);
  738. /* lock out TX and RX while we sample and/or clear */
  739. LOCK_HTC_TX(target);
  740. LOCK_HTC_RX(target);
  741. if (sample) {
  742. A_ASSERT(pStats != NULL);
  743. /* return the stats to the caller */
  744. qdf_mem_copy(pStats, &target->endpoint[Endpoint].endpoint_stats,
  745. sizeof(struct htc_endpoint_stats));
  746. }
  747. if (clearStats) {
  748. /* reset stats */
  749. qdf_mem_zero(&target->endpoint[Endpoint].endpoint_stats,
  750. sizeof(struct htc_endpoint_stats));
  751. }
  752. UNLOCK_HTC_RX(target);
  753. UNLOCK_HTC_TX(target);
  754. return true;
  755. #else
  756. return false;
  757. #endif
  758. }
  759. void *htc_get_targetdef(HTC_HANDLE htc_handle)
  760. {
  761. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(htc_handle);
  762. return hif_get_targetdef(target->hif_dev);
  763. }
  764. #ifdef IPA_OFFLOAD
  765. /**
  766. * htc_ipa_get_ce_resource() - get uc resource on lower layer
  767. * @htc_handle: htc context
  768. * @ce_sr_base_paddr: copyengine source ring base physical address
  769. * @ce_sr_ring_size: copyengine source ring size
  770. * @ce_reg_paddr: copyengine register physical address
  771. *
  772. * Return: None
  773. */
  774. void htc_ipa_get_ce_resource(HTC_HANDLE htc_handle,
  775. qdf_dma_addr_t *ce_sr_base_paddr,
  776. uint32_t *ce_sr_ring_size,
  777. qdf_dma_addr_t *ce_reg_paddr)
  778. {
  779. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(htc_handle);
  780. if (target->hif_dev != NULL) {
  781. hif_ipa_get_ce_resource(target->hif_dev,
  782. ce_sr_base_paddr,
  783. ce_sr_ring_size, ce_reg_paddr);
  784. }
  785. }
  786. #endif /* IPA_OFFLOAD */
  787. #if defined(DEBUG_HL_LOGGING) && defined(CONFIG_HL_SUPPORT)
  788. void htc_dump_bundle_stats(HTC_HANDLE HTCHandle)
  789. {
  790. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  791. int total, i;
  792. total = 0;
  793. for (i = 0; i < HTC_MAX_MSG_PER_BUNDLE_RX; i++)
  794. total += target->rx_bundle_stats[i];
  795. if (total) {
  796. AR_DEBUG_PRINTF(ATH_DEBUG_ANY, ("RX Bundle stats:\n"));
  797. AR_DEBUG_PRINTF(ATH_DEBUG_ANY, ("Total RX packets: %d\n",
  798. total));
  799. AR_DEBUG_PRINTF(ATH_DEBUG_ANY, (
  800. "Number of bundle: Number of packets\n"));
  801. for (i = 0; i < HTC_MAX_MSG_PER_BUNDLE_RX; i++)
  802. AR_DEBUG_PRINTF(ATH_DEBUG_ANY,
  803. ("%10d:%10d(%2d%s)\n", (i+1),
  804. target->rx_bundle_stats[i],
  805. ((target->rx_bundle_stats[i]*100)/
  806. total), "%"));
  807. }
  808. total = 0;
  809. for (i = 0; i < HTC_MAX_MSG_PER_BUNDLE_TX; i++)
  810. total += target->tx_bundle_stats[i];
  811. if (total) {
  812. AR_DEBUG_PRINTF(ATH_DEBUG_ANY, ("TX Bundle stats:\n"));
  813. AR_DEBUG_PRINTF(ATH_DEBUG_ANY, ("Total TX packets: %d\n",
  814. total));
  815. AR_DEBUG_PRINTF(ATH_DEBUG_ANY,
  816. ("Number of bundle: Number of packets\n"));
  817. for (i = 0; i < HTC_MAX_MSG_PER_BUNDLE_TX; i++)
  818. AR_DEBUG_PRINTF(ATH_DEBUG_ANY,
  819. ("%10d:%10d(%2d%s)\n", (i+1),
  820. target->tx_bundle_stats[i],
  821. ((target->tx_bundle_stats[i]*100)/
  822. total), "%"));
  823. }
  824. }
  825. void htc_clear_bundle_stats(HTC_HANDLE HTCHandle)
  826. {
  827. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  828. qdf_mem_zero(&target->rx_bundle_stats, sizeof(target->rx_bundle_stats));
  829. qdf_mem_zero(&target->tx_bundle_stats, sizeof(target->tx_bundle_stats));
  830. }
  831. #endif
  832. /**
  833. * htc_vote_link_down - API to vote for link down
  834. * @htc_handle: HTC handle
  835. *
  836. * API for upper layers to call HIF to vote for link down
  837. *
  838. * Return: void
  839. */
  840. void htc_vote_link_down(HTC_HANDLE htc_handle)
  841. {
  842. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(htc_handle);
  843. if (!target->hif_dev)
  844. return;
  845. hif_vote_link_down(target->hif_dev);
  846. }
  847. /**
  848. * htc_vote_link_up - API to vote for link up
  849. * @htc_handle: HTC Handle
  850. *
  851. * API for upper layers to call HIF to vote for link up
  852. *
  853. * Return: void
  854. */
  855. void htc_vote_link_up(HTC_HANDLE htc_handle)
  856. {
  857. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(htc_handle);
  858. if (!target->hif_dev)
  859. return;
  860. hif_vote_link_up(target->hif_dev);
  861. }
  862. /**
  863. * htc_can_suspend_link - API to query HIF for link status
  864. * @htc_handle: HTC Handle
  865. *
  866. * API for upper layers to call HIF to query if the link can suspend
  867. *
  868. * Return: void
  869. */
  870. bool htc_can_suspend_link(HTC_HANDLE htc_handle)
  871. {
  872. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(htc_handle);
  873. if (!target->hif_dev)
  874. return false;
  875. return hif_can_suspend_link(target->hif_dev);
  876. }
  877. #ifdef FEATURE_RUNTIME_PM
  878. int htc_pm_runtime_get(HTC_HANDLE htc_handle)
  879. {
  880. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(htc_handle);
  881. HTC_INFO("%s: %pS\n", __func__, (void *)_RET_IP_);
  882. return hif_pm_runtime_get(target->hif_dev);
  883. }
  884. int htc_pm_runtime_put(HTC_HANDLE htc_handle)
  885. {
  886. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(htc_handle);
  887. HTC_INFO("%s: %pS\n", __func__, (void *)_RET_IP_);
  888. return hif_pm_runtime_put(target->hif_dev);
  889. }
  890. #endif
  891. /**
  892. * htc_set_wmi_endpoint_count: Set number of WMI endpoint
  893. * @htc_handle: HTC handle
  894. * @wmi_ep_count: WMI enpoint count
  895. *
  896. * return: None
  897. */
  898. void htc_set_wmi_endpoint_count(HTC_HANDLE htc_handle, uint8_t wmi_ep_count)
  899. {
  900. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(htc_handle);
  901. target->wmi_ep_count = wmi_ep_count;
  902. }
  903. /**
  904. * htc_get_wmi_endpoint_count: Get number of WMI endpoint
  905. * @htc_handle: HTC handle
  906. *
  907. * return: WMI enpoint count
  908. */
  909. uint8_t htc_get_wmi_endpoint_count(HTC_HANDLE htc_handle)
  910. {
  911. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(htc_handle);
  912. return target->wmi_ep_count;
  913. }