htc_services.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. /*
  2. * Copyright (c) 2013-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 "htc_debug.h"
  27. #include "htc_internal.h"
  28. #include <qdf_nbuf.h> /* qdf_nbuf_t */
  29. #include "hif.h"
  30. /* use credit flow control over HTC */
  31. unsigned int htc_credit_flow = 1;
  32. #ifndef DEBUG_CREDIT
  33. #define DEBUG_CREDIT 0
  34. #endif
  35. /**
  36. * htc_update_htc_htt_config - API to update HIF with the HTT endpoint info
  37. * @target: HTC handle
  38. * @endpoint: Endpoint on which HTT messages flow
  39. *
  40. * Return: void
  41. */
  42. #ifdef HIF_PCI
  43. void htc_update_htc_htt_config(HTC_TARGET *target, int endpoint)
  44. {
  45. hif_save_htc_htt_config_endpoint(target->hif_dev, endpoint);
  46. }
  47. #else
  48. void htc_update_htc_htt_config(HTC_TARGET *target, int endpoint)
  49. {
  50. }
  51. #endif
  52. A_STATUS htc_connect_service(HTC_HANDLE HTCHandle,
  53. HTC_SERVICE_CONNECT_REQ *pConnectReq,
  54. HTC_SERVICE_CONNECT_RESP *pConnectResp)
  55. {
  56. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  57. A_STATUS status = A_OK;
  58. HTC_PACKET *pSendPacket = NULL;
  59. HTC_CONNECT_SERVICE_RESPONSE_MSG *pResponseMsg;
  60. HTC_CONNECT_SERVICE_MSG *pConnectMsg;
  61. HTC_ENDPOINT_ID assignedEndpoint = ENDPOINT_MAX;
  62. HTC_ENDPOINT *pEndpoint;
  63. unsigned int maxMsgSize = 0;
  64. qdf_nbuf_t netbuf;
  65. A_UINT8 txAlloc;
  66. int length;
  67. A_BOOL disableCreditFlowCtrl = false;
  68. A_UINT16 conn_flags;
  69. A_UINT16 rsp_msg_id, rsp_msg_serv_id, rsp_msg_max_msg_size;
  70. A_UINT8 rsp_msg_status, rsp_msg_end_id, rsp_msg_serv_meta_len;
  71. AR_DEBUG_PRINTF(ATH_DEBUG_TRC,
  72. ("+htc_connect_service, target:%p SvcID:0x%X\n", target,
  73. pConnectReq->service_id));
  74. do {
  75. AR_DEBUG_ASSERT(pConnectReq->service_id != 0);
  76. if (HTC_CTRL_RSVD_SVC == pConnectReq->service_id) {
  77. /* special case for pseudo control service */
  78. assignedEndpoint = ENDPOINT_0;
  79. maxMsgSize = HTC_MAX_CONTROL_MESSAGE_LENGTH;
  80. txAlloc = 0;
  81. } else {
  82. txAlloc = htc_get_credit_allocation(target,
  83. pConnectReq->service_id);
  84. if (!txAlloc) {
  85. AR_DEBUG_PRINTF(ATH_DEBUG_TRC,
  86. ("Service %d does not allocate target credits!\n",
  87. pConnectReq->service_id));
  88. }
  89. /* allocate a packet to send to the target */
  90. pSendPacket = htc_alloc_control_tx_packet(target);
  91. if (NULL == pSendPacket) {
  92. AR_DEBUG_ASSERT(false);
  93. status = A_NO_MEMORY;
  94. break;
  95. }
  96. netbuf =
  97. (qdf_nbuf_t)
  98. GET_HTC_PACKET_NET_BUF_CONTEXT(pSendPacket);
  99. length =
  100. sizeof(HTC_CONNECT_SERVICE_MSG) +
  101. pConnectReq->MetaDataLength;
  102. /* assemble connect service message */
  103. qdf_nbuf_put_tail(netbuf, length);
  104. pConnectMsg =
  105. (HTC_CONNECT_SERVICE_MSG *) qdf_nbuf_data(netbuf);
  106. if (NULL == pConnectMsg) {
  107. AR_DEBUG_ASSERT(0);
  108. status = A_EFAULT;
  109. break;
  110. }
  111. A_MEMZERO(pConnectMsg, sizeof(HTC_CONNECT_SERVICE_MSG));
  112. conn_flags =
  113. (pConnectReq->
  114. ConnectionFlags & ~HTC_SET_RECV_ALLOC_MASK) |
  115. HTC_CONNECT_FLAGS_SET_RECV_ALLOCATION(txAlloc);
  116. HTC_SET_FIELD(pConnectMsg, HTC_CONNECT_SERVICE_MSG,
  117. MESSAGEID, HTC_MSG_CONNECT_SERVICE_ID);
  118. HTC_SET_FIELD(pConnectMsg, HTC_CONNECT_SERVICE_MSG,
  119. SERVICE_ID, pConnectReq->service_id);
  120. HTC_SET_FIELD(pConnectMsg, HTC_CONNECT_SERVICE_MSG,
  121. CONNECTIONFLAGS, conn_flags);
  122. if (pConnectReq->
  123. ConnectionFlags &
  124. HTC_CONNECT_FLAGS_DISABLE_CREDIT_FLOW_CTRL) {
  125. disableCreditFlowCtrl = true;
  126. }
  127. #if defined(HIF_USB)
  128. if (!htc_credit_flow) {
  129. disableCreditFlowCtrl = true;
  130. }
  131. #else
  132. /* Only enable credit for WMI service */
  133. if (!htc_credit_flow
  134. && pConnectReq->service_id != WMI_CONTROL_SVC) {
  135. disableCreditFlowCtrl = true;
  136. }
  137. #endif
  138. /* check caller if it wants to transfer meta data */
  139. if ((pConnectReq->pMetaData != NULL) &&
  140. (pConnectReq->MetaDataLength <=
  141. HTC_SERVICE_META_DATA_MAX_LENGTH)) {
  142. /* copy meta data into message buffer (after header ) */
  143. A_MEMCPY((A_UINT8 *) pConnectMsg +
  144. sizeof(HTC_CONNECT_SERVICE_MSG),
  145. pConnectReq->pMetaData,
  146. pConnectReq->MetaDataLength);
  147. HTC_SET_FIELD(pConnectMsg,
  148. HTC_CONNECT_SERVICE_MSG,
  149. SERVICEMETALENGTH,
  150. pConnectReq->MetaDataLength);
  151. }
  152. SET_HTC_PACKET_INFO_TX(pSendPacket,
  153. NULL,
  154. (A_UINT8 *) pConnectMsg,
  155. length,
  156. ENDPOINT_0,
  157. HTC_SERVICE_TX_PACKET_TAG);
  158. status = htc_send_pkt((HTC_HANDLE) target, pSendPacket);
  159. /* we don't own it anymore */
  160. pSendPacket = NULL;
  161. if (A_FAILED(status)) {
  162. break;
  163. }
  164. /* wait for response */
  165. status = htc_wait_recv_ctrl_message(target);
  166. if (A_FAILED(status)) {
  167. break;
  168. }
  169. /* we controlled the buffer creation so it has to be properly aligned */
  170. pResponseMsg =
  171. (HTC_CONNECT_SERVICE_RESPONSE_MSG *) target->
  172. CtrlResponseBuffer;
  173. rsp_msg_id = HTC_GET_FIELD(pResponseMsg,
  174. HTC_CONNECT_SERVICE_RESPONSE_MSG,
  175. MESSAGEID);
  176. rsp_msg_serv_id =
  177. HTC_GET_FIELD(pResponseMsg,
  178. HTC_CONNECT_SERVICE_RESPONSE_MSG,
  179. SERVICEID);
  180. rsp_msg_status =
  181. HTC_GET_FIELD(pResponseMsg,
  182. HTC_CONNECT_SERVICE_RESPONSE_MSG,
  183. STATUS);
  184. rsp_msg_end_id =
  185. HTC_GET_FIELD(pResponseMsg,
  186. HTC_CONNECT_SERVICE_RESPONSE_MSG,
  187. ENDPOINTID);
  188. rsp_msg_max_msg_size =
  189. HTC_GET_FIELD(pResponseMsg,
  190. HTC_CONNECT_SERVICE_RESPONSE_MSG,
  191. MAXMSGSIZE);
  192. rsp_msg_serv_meta_len =
  193. HTC_GET_FIELD(pResponseMsg,
  194. HTC_CONNECT_SERVICE_RESPONSE_MSG,
  195. SERVICEMETALENGTH);
  196. if ((rsp_msg_id != HTC_MSG_CONNECT_SERVICE_RESPONSE_ID)
  197. || (target->CtrlResponseLength <
  198. sizeof(HTC_CONNECT_SERVICE_RESPONSE_MSG))) {
  199. /* this message is not valid */
  200. AR_DEBUG_ASSERT(false);
  201. status = A_EPROTO;
  202. break;
  203. }
  204. AR_DEBUG_PRINTF(ATH_DEBUG_TRC,
  205. ("htc_connect_service, service 0x%X connect response from target status:%d, assigned ep: %d\n",
  206. rsp_msg_serv_id, rsp_msg_status,
  207. rsp_msg_end_id));
  208. pConnectResp->ConnectRespCode = rsp_msg_status;
  209. /* check response status */
  210. if (rsp_msg_status != HTC_SERVICE_SUCCESS) {
  211. AR_DEBUG_PRINTF(ATH_DEBUG_ERR,
  212. (" Target failed service 0x%X connect request (status:%d)\n",
  213. rsp_msg_serv_id,
  214. rsp_msg_status));
  215. status = A_EPROTO;
  216. #ifdef QCA_TX_HTT2_SUPPORT
  217. /* Keep work and not to block the control message. */
  218. target->CtrlResponseProcessing = false;
  219. #endif /* QCA_TX_HTT2_SUPPORT */
  220. break;
  221. }
  222. assignedEndpoint = (HTC_ENDPOINT_ID) rsp_msg_end_id;
  223. maxMsgSize = rsp_msg_max_msg_size;
  224. if ((pConnectResp->pMetaData != NULL) &&
  225. (rsp_msg_serv_meta_len > 0) &&
  226. (rsp_msg_serv_meta_len <=
  227. HTC_SERVICE_META_DATA_MAX_LENGTH)) {
  228. /* caller supplied a buffer and the target responded with data */
  229. int copyLength =
  230. min((int)pConnectResp->BufferLength,
  231. (int)rsp_msg_serv_meta_len);
  232. /* copy the meta data */
  233. A_MEMCPY(pConnectResp->pMetaData,
  234. ((A_UINT8 *) pResponseMsg) +
  235. sizeof
  236. (HTC_CONNECT_SERVICE_RESPONSE_MSG),
  237. copyLength);
  238. pConnectResp->ActualLength = copyLength;
  239. }
  240. /* done processing response buffer */
  241. target->CtrlResponseProcessing = false;
  242. }
  243. /* the rest of these are parameter checks so set the error status */
  244. status = A_EPROTO;
  245. if (assignedEndpoint >= ENDPOINT_MAX) {
  246. AR_DEBUG_ASSERT(false);
  247. break;
  248. }
  249. if (0 == maxMsgSize) {
  250. AR_DEBUG_ASSERT(false);
  251. break;
  252. }
  253. pEndpoint = &target->endpoint[assignedEndpoint];
  254. pEndpoint->Id = assignedEndpoint;
  255. if (pEndpoint->service_id != 0) {
  256. /* endpoint already in use! */
  257. AR_DEBUG_ASSERT(false);
  258. break;
  259. }
  260. /* return assigned endpoint to caller */
  261. pConnectResp->Endpoint = assignedEndpoint;
  262. pConnectResp->MaxMsgLength = maxMsgSize;
  263. /* setup the endpoint */
  264. /* service_id marks the endpoint in use */
  265. pEndpoint->service_id = pConnectReq->service_id;
  266. pEndpoint->MaxTxQueueDepth = pConnectReq->MaxSendQueueDepth;
  267. pEndpoint->MaxMsgLength = maxMsgSize;
  268. pEndpoint->TxCredits = txAlloc;
  269. pEndpoint->TxCreditSize = target->TargetCreditSize;
  270. pEndpoint->TxCreditsPerMaxMsg =
  271. maxMsgSize / target->TargetCreditSize;
  272. if (maxMsgSize % target->TargetCreditSize) {
  273. pEndpoint->TxCreditsPerMaxMsg++;
  274. }
  275. #if DEBUG_CREDIT
  276. qdf_print(" Endpoint%d initial credit:%d, size:%d.\n",
  277. pEndpoint->Id, pEndpoint->TxCredits,
  278. pEndpoint->TxCreditSize);
  279. #endif
  280. /* copy all the callbacks */
  281. pEndpoint->EpCallBacks = pConnectReq->EpCallbacks;
  282. status = hif_map_service_to_pipe(target->hif_dev,
  283. pEndpoint->service_id,
  284. &pEndpoint->UL_PipeID,
  285. &pEndpoint->DL_PipeID,
  286. &pEndpoint->ul_is_polled,
  287. &pEndpoint->dl_is_polled);
  288. if (A_FAILED(status)) {
  289. break;
  290. }
  291. qdf_assert(!pEndpoint->dl_is_polled); /* not currently supported */
  292. if (pEndpoint->ul_is_polled) {
  293. qdf_timer_init(target->osdev,
  294. &pEndpoint->ul_poll_timer,
  295. htc_send_complete_check_cleanup,
  296. pEndpoint,
  297. QDF_TIMER_TYPE_SW);
  298. }
  299. AR_DEBUG_PRINTF(ATH_DEBUG_SETUP,
  300. ("HTC Service:0x%4.4X, ULpipe:%d DLpipe:%d id:%d Ready\n",
  301. pEndpoint->service_id, pEndpoint->UL_PipeID,
  302. pEndpoint->DL_PipeID, pEndpoint->Id));
  303. if (disableCreditFlowCtrl && pEndpoint->TxCreditFlowEnabled) {
  304. pEndpoint->TxCreditFlowEnabled = false;
  305. AR_DEBUG_PRINTF(ATH_DEBUG_WARN,
  306. ("HTC Service:0x%4.4X ep:%d TX flow control disabled\n",
  307. pEndpoint->service_id,
  308. assignedEndpoint));
  309. }
  310. } while (false);
  311. if (HTT_SERVICE_GROUP == (pConnectReq->service_id >> 8))
  312. htc_update_htc_htt_config(target, assignedEndpoint);
  313. AR_DEBUG_PRINTF(ATH_DEBUG_TRC, ("-htc_connect_service\n"));
  314. return status;
  315. }
  316. void htc_set_credit_distribution(HTC_HANDLE HTCHandle,
  317. void *pCreditDistContext,
  318. HTC_CREDIT_DIST_CALLBACK CreditDistFunc,
  319. HTC_CREDIT_INIT_CALLBACK CreditInitFunc,
  320. HTC_SERVICE_ID ServicePriorityOrder[],
  321. int ListLength)
  322. {
  323. /* NOT Supported, this transport does not use a credit based flow control mechanism */
  324. }
  325. void htc_fw_event_handler(void *context, QDF_STATUS status)
  326. {
  327. HTC_TARGET *target = (HTC_TARGET *) context;
  328. HTC_INIT_INFO *initInfo = &target->HTCInitInfo;
  329. /* check if target failure handler exists and pass error code to it. */
  330. if (target->HTCInitInfo.TargetFailure != NULL) {
  331. initInfo->TargetFailure(initInfo->pContext, status);
  332. }
  333. }
  334. /* Disable ASPM : disable PCIe low power */
  335. void htc_disable_aspm(HTC_HANDLE HTCHandle)
  336. {
  337. HTC_TARGET *target = GET_HTC_TARGET_FROM_HANDLE(HTCHandle);
  338. if (!target->hif_dev)
  339. return;
  340. hif_disable_aspm(target->hif_dev);
  341. }