wmi_unified_action_oui_tlv.c 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  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_action_oui_tlv.h"
  19. bool wmi_get_action_oui_id(enum action_oui_id action_id,
  20. wmi_vendor_oui_action_id *id)
  21. {
  22. switch (action_id) {
  23. case ACTION_OUI_CONNECT_1X1:
  24. *id = WMI_VENDOR_OUI_ACTION_CONNECTION_1X1;
  25. return true;
  26. case ACTION_OUI_ITO_EXTENSION:
  27. *id = WMI_VENDOR_OUI_ACTION_ITO_EXTENSION;
  28. return true;
  29. case ACTION_OUI_CCKM_1X1:
  30. *id = WMI_VENDOR_OUI_ACTION_CCKM_1X1;
  31. return true;
  32. case ACTION_OUI_ITO_ALTERNATE:
  33. *id = WMI_VENDOR_OUI_ACTION_ALT_ITO;
  34. return true;
  35. case ACTION_OUI_SWITCH_TO_11N_MODE:
  36. *id = WMI_VENDOR_OUI_ACTION_SWITCH_TO_11N_MODE;
  37. return true;
  38. case ACTION_OUI_CONNECT_1X1_WITH_1_CHAIN:
  39. *id = WMI_VENDOR_OUI_ACTION_CONNECTION_1X1_NUM_TX_RX_CHAINS_1;
  40. return true;
  41. default:
  42. return false;
  43. }
  44. }
  45. uint32_t wmi_get_action_oui_info_mask(uint32_t info_mask)
  46. {
  47. uint32_t info_presence = WMI_BEACON_INFO_PRESENCE_OUI_EXT;
  48. if (info_mask & ACTION_OUI_INFO_MAC_ADDRESS)
  49. info_presence |= WMI_BEACON_INFO_PRESENCE_MAC_ADDRESS;
  50. if (info_mask & ACTION_OUI_INFO_AP_CAPABILITY_NSS)
  51. info_presence |= WMI_BEACON_INFO_PRESENCE_AP_CAPABILITY_NSS;
  52. if (info_mask & ACTION_OUI_INFO_AP_CAPABILITY_HT)
  53. info_presence |= WMI_BEACON_INFO_PRESENCE_AP_CAPABILITY_HT;
  54. if (info_mask & ACTION_OUI_INFO_AP_CAPABILITY_VHT)
  55. info_presence |= WMI_BEACON_INFO_PRESENCE_AP_CAPABILITY_VHT;
  56. if (info_mask & ACTION_OUI_INFO_AP_CAPABILITY_BAND)
  57. info_presence |= WMI_BEACON_INFO_PRESENCE_AP_CAPABILITY_BAND;
  58. return info_presence;
  59. }
  60. void wmi_fill_oui_extensions(struct action_oui_extension *extension,
  61. uint32_t no_oui_extns,
  62. wmi_vendor_oui_ext *cmd_ext)
  63. {
  64. uint32_t i;
  65. uint32_t buffer_length;
  66. for (i = 0; i < no_oui_extns; i++) {
  67. WMITLV_SET_HDR(&cmd_ext->tlv_header,
  68. WMITLV_TAG_STRUC_wmi_vendor_oui_ext,
  69. WMITLV_GET_STRUCT_TLVLEN(wmi_vendor_oui_ext));
  70. cmd_ext->info_presence_bit_mask =
  71. wmi_get_action_oui_info_mask(extension->info_mask);
  72. cmd_ext->oui_header_length = extension->oui_length;
  73. cmd_ext->oui_data_length = extension->data_length;
  74. cmd_ext->mac_address_length = extension->mac_addr_length;
  75. cmd_ext->capability_data_length =
  76. extension->capability_length;
  77. buffer_length = extension->oui_length +
  78. extension->data_length +
  79. extension->data_mask_length +
  80. extension->mac_addr_length +
  81. extension->mac_mask_length +
  82. extension->capability_length;
  83. cmd_ext->buf_data_length = buffer_length + 1;
  84. cmd_ext++;
  85. extension++;
  86. }
  87. }
  88. QDF_STATUS
  89. wmi_fill_oui_extensions_buffer(struct action_oui_extension *extension,
  90. wmi_vendor_oui_ext *cmd_ext,
  91. uint32_t no_oui_extns, uint32_t rem_var_buf_len,
  92. uint8_t *var_buf)
  93. {
  94. uint8_t i;
  95. for (i = 0; i < (uint8_t)no_oui_extns; i++) {
  96. if ((rem_var_buf_len - cmd_ext->buf_data_length) < 0) {
  97. WMI_LOGE(FL("Invalid action oui command length"));
  98. return QDF_STATUS_E_INVAL;
  99. }
  100. var_buf[0] = i;
  101. var_buf++;
  102. qdf_mem_copy(var_buf, extension->oui, extension->oui_length);
  103. var_buf += extension->oui_length;
  104. if (extension->data_length) {
  105. qdf_mem_copy(var_buf, extension->data,
  106. extension->data_length);
  107. var_buf += extension->data_length;
  108. }
  109. if (extension->data_mask_length) {
  110. qdf_mem_copy(var_buf, extension->data_mask,
  111. extension->data_mask_length);
  112. var_buf += extension->data_mask_length;
  113. }
  114. if (extension->mac_addr_length) {
  115. qdf_mem_copy(var_buf, extension->mac_addr,
  116. extension->mac_addr_length);
  117. var_buf += extension->mac_addr_length;
  118. }
  119. if (extension->mac_mask_length) {
  120. qdf_mem_copy(var_buf, extension->mac_mask,
  121. extension->mac_mask_length);
  122. var_buf += extension->mac_mask_length;
  123. }
  124. if (extension->capability_length) {
  125. qdf_mem_copy(var_buf, extension->capability,
  126. extension->capability_length);
  127. var_buf += extension->capability_length;
  128. }
  129. rem_var_buf_len -= cmd_ext->buf_data_length;
  130. cmd_ext++;
  131. extension++;
  132. }
  133. return QDF_STATUS_SUCCESS;
  134. }
  135. QDF_STATUS
  136. send_action_oui_cmd_tlv(wmi_unified_t wmi_handle,
  137. struct action_oui_request *req)
  138. {
  139. wmi_pdev_config_vendor_oui_action_fixed_param *cmd;
  140. wmi_vendor_oui_ext *cmd_ext;
  141. wmi_buf_t wmi_buf;
  142. struct action_oui_extension *extension;
  143. uint32_t len;
  144. uint32_t i;
  145. uint8_t *buf_ptr;
  146. uint32_t no_oui_extns;
  147. uint32_t total_no_oui_extns;
  148. uint32_t var_buf_len = 0;
  149. wmi_vendor_oui_action_id action_id;
  150. bool valid;
  151. uint32_t rem_var_buf_len;
  152. QDF_STATUS status;
  153. if (!req) {
  154. WMI_LOGE(FL("action oui is empty"));
  155. return QDF_STATUS_E_INVAL;
  156. }
  157. no_oui_extns = req->no_oui_extensions;
  158. total_no_oui_extns = req->total_no_oui_extensions;
  159. len = sizeof(*cmd);
  160. len += WMI_TLV_HDR_SIZE; /* Array of wmi_vendor_oui_ext structures */
  161. if (!no_oui_extns ||
  162. no_oui_extns > WMI_MAX_VENDOR_OUI_ACTION_SUPPORTED_PER_ACTION ||
  163. (total_no_oui_extns > WMI_VENDOR_OUI_ACTION_MAX_ACTION_ID *
  164. WMI_MAX_VENDOR_OUI_ACTION_SUPPORTED_PER_ACTION)) {
  165. WMI_LOGE(FL("Invalid number of action oui extensions"));
  166. return QDF_STATUS_E_INVAL;
  167. }
  168. valid = wmi_get_action_oui_id(req->action_id, &action_id);
  169. if (!valid) {
  170. WMI_LOGE(FL("Invalid action id"));
  171. return QDF_STATUS_E_INVAL;
  172. }
  173. len += no_oui_extns * sizeof(*cmd_ext);
  174. len += WMI_TLV_HDR_SIZE; /* Variable length buffer */
  175. extension = req->extension;
  176. for (i = 0; i < no_oui_extns; i++) {
  177. var_buf_len += extension->oui_length +
  178. extension->data_length +
  179. extension->data_mask_length +
  180. extension->mac_addr_length +
  181. extension->mac_mask_length +
  182. extension->capability_length;
  183. extension++;
  184. }
  185. var_buf_len += no_oui_extns; /* to store indexes */
  186. rem_var_buf_len = var_buf_len;
  187. var_buf_len = (var_buf_len + 3) & ~0x3;
  188. len += var_buf_len;
  189. wmi_buf = wmi_buf_alloc(wmi_handle, len);
  190. if (!wmi_buf) {
  191. WMI_LOGE(FL("Failed to allocate wmi buffer"));
  192. return QDF_STATUS_E_FAILURE;
  193. }
  194. buf_ptr = (uint8_t *)wmi_buf_data(wmi_buf);
  195. cmd = (wmi_pdev_config_vendor_oui_action_fixed_param *)buf_ptr;
  196. WMITLV_SET_HDR(&cmd->tlv_header,
  197. WMITLV_TAG_STRUC_wmi_pdev_config_vendor_oui_action_fixed_param,
  198. WMITLV_GET_STRUCT_TLVLEN(
  199. wmi_pdev_config_vendor_oui_action_fixed_param));
  200. cmd->action_id = action_id;
  201. cmd->total_num_vendor_oui = total_no_oui_extns;
  202. cmd->num_vendor_oui_ext = no_oui_extns;
  203. buf_ptr += sizeof(*cmd);
  204. WMITLV_SET_HDR(buf_ptr, WMITLV_TAG_ARRAY_STRUC,
  205. no_oui_extns * sizeof(*cmd_ext));
  206. buf_ptr += WMI_TLV_HDR_SIZE;
  207. cmd_ext = (wmi_vendor_oui_ext *)buf_ptr;
  208. wmi_fill_oui_extensions(req->extension, no_oui_extns, cmd_ext);
  209. buf_ptr += no_oui_extns * sizeof(*cmd_ext);
  210. WMITLV_SET_HDR(buf_ptr, WMITLV_TAG_ARRAY_BYTE, var_buf_len);
  211. buf_ptr += WMI_TLV_HDR_SIZE;
  212. status = wmi_fill_oui_extensions_buffer(req->extension,
  213. cmd_ext, no_oui_extns,
  214. rem_var_buf_len, buf_ptr);
  215. if (!QDF_IS_STATUS_SUCCESS(status)) {
  216. wmi_buf_free(wmi_buf);
  217. wmi_buf = NULL;
  218. return QDF_STATUS_E_INVAL;
  219. }
  220. buf_ptr += var_buf_len;
  221. if (wmi_unified_cmd_send(wmi_handle, wmi_buf, len,
  222. WMI_PDEV_CONFIG_VENDOR_OUI_ACTION_CMDID)) {
  223. WMI_LOGE(FL("WMI_PDEV_CONFIG_VENDOR_OUI_ACTION send fail"));
  224. wmi_buf_free(wmi_buf);
  225. wmi_buf = NULL;
  226. return QDF_STATUS_E_FAILURE;
  227. }
  228. return QDF_STATUS_SUCCESS;
  229. }