wmi_unified_apf_tlv.c 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. /*
  2. * Copyright (c) 2016-2018 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 "wmi_unified_apf_tlv.h"
  19. QDF_STATUS wmi_send_set_active_apf_mode_cmd_tlv(wmi_unified_t wmi_handle,
  20. uint8_t vdev_id,
  21. enum wmi_host_active_apf_mode
  22. ucast_mode,
  23. enum wmi_host_active_apf_mode
  24. mcast_bcast_mode)
  25. {
  26. const WMITLV_TAG_ID tag_id =
  27. WMITLV_TAG_STRUC_wmi_bpf_set_vdev_active_mode_cmd_fixed_param;
  28. const uint32_t tlv_len = WMITLV_GET_STRUCT_TLVLEN(
  29. wmi_bpf_set_vdev_active_mode_cmd_fixed_param);
  30. QDF_STATUS status;
  31. wmi_bpf_set_vdev_active_mode_cmd_fixed_param *cmd;
  32. wmi_buf_t buf;
  33. WMI_LOGD("Sending WMI_BPF_SET_VDEV_ACTIVE_MODE_CMDID(%u, %d, %d)",
  34. vdev_id, ucast_mode, mcast_bcast_mode);
  35. /* allocate command buffer */
  36. buf = wmi_buf_alloc(wmi_handle, sizeof(*cmd));
  37. if (!buf) {
  38. WMI_LOGE("%s: wmi_buf_alloc failed", __func__);
  39. return QDF_STATUS_E_NOMEM;
  40. }
  41. /* set TLV header */
  42. cmd = (wmi_bpf_set_vdev_active_mode_cmd_fixed_param *)wmi_buf_data(buf);
  43. WMITLV_SET_HDR(&cmd->tlv_header, tag_id, tlv_len);
  44. /* populate data */
  45. cmd->vdev_id = vdev_id;
  46. cmd->uc_mode = ucast_mode;
  47. cmd->mcbc_mode = mcast_bcast_mode;
  48. /* send to FW */
  49. status = wmi_unified_cmd_send(wmi_handle, buf, sizeof(*cmd),
  50. WMI_BPF_SET_VDEV_ACTIVE_MODE_CMDID);
  51. if (QDF_IS_STATUS_ERROR(status)) {
  52. WMI_LOGE("Failed to send WMI_BPF_SET_VDEV_ACTIVE_MODE_CMDID:%d",
  53. status);
  54. wmi_buf_free(buf);
  55. return status;
  56. }
  57. return QDF_STATUS_SUCCESS;
  58. }
  59. QDF_STATUS wmi_send_apf_enable_cmd_tlv(wmi_unified_t wmi_handle,
  60. uint32_t vdev_id,
  61. bool enable)
  62. {
  63. wmi_bpf_set_vdev_enable_cmd_fixed_param *cmd;
  64. wmi_buf_t buf;
  65. buf = wmi_buf_alloc(wmi_handle, sizeof(*cmd));
  66. if (!buf) {
  67. WMI_LOGP("%s: wmi_buf_alloc failed", __func__);
  68. return QDF_STATUS_E_NOMEM;
  69. }
  70. cmd = (wmi_bpf_set_vdev_enable_cmd_fixed_param *) wmi_buf_data(buf);
  71. WMITLV_SET_HDR(&cmd->tlv_header,
  72. WMITLV_TAG_STRUC_wmi_bpf_set_vdev_enable_cmd_fixed_param,
  73. WMITLV_GET_STRUCT_TLVLEN(
  74. wmi_bpf_set_vdev_enable_cmd_fixed_param));
  75. cmd->vdev_id = vdev_id;
  76. cmd->is_enabled = enable;
  77. if (wmi_unified_cmd_send(wmi_handle, buf, sizeof(*cmd),
  78. WMI_BPF_SET_VDEV_ENABLE_CMDID)) {
  79. WMI_LOGE("%s: Failed to enable/disable APF interpreter",
  80. __func__);
  81. wmi_buf_free(buf);
  82. return QDF_STATUS_E_FAILURE;
  83. }
  84. return QDF_STATUS_SUCCESS;
  85. }
  86. QDF_STATUS
  87. wmi_send_apf_write_work_memory_cmd_tlv(wmi_unified_t wmi_handle,
  88. struct wmi_apf_write_memory_params
  89. *apf_write_params)
  90. {
  91. wmi_bpf_set_vdev_work_memory_cmd_fixed_param *cmd;
  92. uint32_t wmi_buf_len;
  93. wmi_buf_t buf;
  94. uint8_t *buf_ptr;
  95. uint32_t aligned_len = 0;
  96. wmi_buf_len = sizeof(*cmd);
  97. if (apf_write_params->length) {
  98. aligned_len = roundup(apf_write_params->length,
  99. sizeof(A_UINT32));
  100. wmi_buf_len += WMI_TLV_HDR_SIZE + aligned_len;
  101. }
  102. buf = wmi_buf_alloc(wmi_handle, wmi_buf_len);
  103. if (!buf) {
  104. WMI_LOGP("%s: wmi_buf_alloc failed", __func__);
  105. return QDF_STATUS_E_NOMEM;
  106. }
  107. buf_ptr = wmi_buf_data(buf);
  108. cmd = (wmi_bpf_set_vdev_work_memory_cmd_fixed_param *)buf_ptr;
  109. WMITLV_SET_HDR(&cmd->tlv_header,
  110. WMITLV_TAG_STRUC_wmi_bpf_set_vdev_work_memory_cmd_fixed_param,
  111. WMITLV_GET_STRUCT_TLVLEN(
  112. wmi_bpf_set_vdev_work_memory_cmd_fixed_param));
  113. cmd->vdev_id = apf_write_params->vdev_id;
  114. cmd->bpf_version = apf_write_params->apf_version;
  115. cmd->program_len = apf_write_params->program_len;
  116. cmd->addr_offset = apf_write_params->addr_offset;
  117. cmd->length = apf_write_params->length;
  118. if (apf_write_params->length) {
  119. buf_ptr += sizeof(*cmd);
  120. WMITLV_SET_HDR(buf_ptr, WMITLV_TAG_ARRAY_BYTE,
  121. aligned_len);
  122. buf_ptr += WMI_TLV_HDR_SIZE;
  123. qdf_mem_copy(buf_ptr, apf_write_params->buf,
  124. apf_write_params->length);
  125. }
  126. if (wmi_unified_cmd_send(wmi_handle, buf, wmi_buf_len,
  127. WMI_BPF_SET_VDEV_WORK_MEMORY_CMDID)) {
  128. WMI_LOGE("%s: Failed to write APF work memory", __func__);
  129. wmi_buf_free(buf);
  130. return QDF_STATUS_E_FAILURE;
  131. }
  132. return QDF_STATUS_SUCCESS;
  133. }
  134. QDF_STATUS
  135. wmi_send_apf_read_work_memory_cmd_tlv(wmi_unified_t wmi_handle,
  136. struct wmi_apf_read_memory_params
  137. *apf_read_params)
  138. {
  139. wmi_bpf_get_vdev_work_memory_cmd_fixed_param *cmd;
  140. wmi_buf_t buf;
  141. buf = wmi_buf_alloc(wmi_handle, sizeof(*cmd));
  142. if (!buf) {
  143. WMI_LOGP("%s: wmi_buf_alloc failed", __func__);
  144. return QDF_STATUS_E_NOMEM;
  145. }
  146. cmd = (wmi_bpf_get_vdev_work_memory_cmd_fixed_param *)
  147. wmi_buf_data(buf);
  148. WMITLV_SET_HDR(&cmd->tlv_header,
  149. WMITLV_TAG_STRUC_wmi_bpf_get_vdev_work_memory_cmd_fixed_param,
  150. WMITLV_GET_STRUCT_TLVLEN(
  151. wmi_bpf_get_vdev_work_memory_cmd_fixed_param));
  152. cmd->vdev_id = apf_read_params->vdev_id;
  153. cmd->addr_offset = apf_read_params->addr_offset;
  154. cmd->length = apf_read_params->length;
  155. if (wmi_unified_cmd_send(wmi_handle, buf, sizeof(*cmd),
  156. WMI_BPF_GET_VDEV_WORK_MEMORY_CMDID)) {
  157. WMI_LOGE("%s: Failed to get APF work memory", __func__);
  158. wmi_buf_free(buf);
  159. return QDF_STATUS_E_FAILURE;
  160. }
  161. return QDF_STATUS_SUCCESS;
  162. }
  163. QDF_STATUS
  164. wmi_extract_apf_read_memory_resp_event_tlv(wmi_unified_t wmi_handle,
  165. void *evt_buf,
  166. struct wmi_apf_read_memory_resp_event_params
  167. *resp)
  168. {
  169. WMI_BPF_GET_VDEV_WORK_MEMORY_RESP_EVENTID_param_tlvs *param_buf;
  170. wmi_bpf_get_vdev_work_memory_resp_evt_fixed_param *data_event;
  171. param_buf = evt_buf;
  172. if (!param_buf) {
  173. WMI_LOGE("encrypt decrypt resp evt_buf is NULL");
  174. return QDF_STATUS_E_INVAL;
  175. }
  176. data_event = param_buf->fixed_param;
  177. resp->vdev_id = data_event->vdev_id;
  178. resp->offset = data_event->offset;
  179. resp->more_data = data_event->fragment;
  180. if (data_event->length > param_buf->num_data) {
  181. WMI_LOGE("FW msg data_len %d more than TLV hdr %d",
  182. data_event->length,
  183. param_buf->num_data);
  184. return QDF_STATUS_E_INVAL;
  185. }
  186. resp->length = data_event->length;
  187. if (resp->length)
  188. resp->data = (uint8_t *)param_buf->data;
  189. return QDF_STATUS_SUCCESS;
  190. }