|
@@ -0,0 +1,277 @@
|
|
|
+/*
|
|
|
+ * Copyright (c) 2016-2018 The Linux Foundation. All rights reserved.
|
|
|
+ *
|
|
|
+ * Permission to use, copy, modify, and/or distribute this software for
|
|
|
+ * any purpose with or without fee is hereby granted, provided that the
|
|
|
+ * above copyright notice and this permission notice appear in all
|
|
|
+ * copies.
|
|
|
+ *
|
|
|
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
|
|
|
+ * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
|
|
|
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
|
|
|
+ * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
|
|
|
+ * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
|
|
|
+ * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
|
|
+ * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
|
+ * PERFORMANCE OF THIS SOFTWARE.
|
|
|
+ */
|
|
|
+
|
|
|
+#include "wmi_unified_action_oui_tlv.h"
|
|
|
+
|
|
|
+bool wmi_get_action_oui_id(enum action_oui_id action_id,
|
|
|
+ wmi_vendor_oui_action_id *id)
|
|
|
+{
|
|
|
+ switch (action_id) {
|
|
|
+
|
|
|
+ case ACTION_OUI_CONNECT_1X1:
|
|
|
+ *id = WMI_VENDOR_OUI_ACTION_CONNECTION_1X1;
|
|
|
+ return true;
|
|
|
+
|
|
|
+ case ACTION_OUI_ITO_EXTENSION:
|
|
|
+ *id = WMI_VENDOR_OUI_ACTION_ITO_EXTENSION;
|
|
|
+ return true;
|
|
|
+
|
|
|
+ case ACTION_OUI_CCKM_1X1:
|
|
|
+ *id = WMI_VENDOR_OUI_ACTION_CCKM_1X1;
|
|
|
+ return true;
|
|
|
+
|
|
|
+ case ACTION_OUI_ITO_ALTERNATE:
|
|
|
+ *id = WMI_VENDOR_OUI_ACTION_ALT_ITO;
|
|
|
+ return true;
|
|
|
+
|
|
|
+ case ACTION_OUI_SWITCH_TO_11N_MODE:
|
|
|
+ *id = WMI_VENDOR_OUI_ACTION_SWITCH_TO_11N_MODE;
|
|
|
+ return true;
|
|
|
+
|
|
|
+ default:
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+uint32_t wmi_get_action_oui_info_mask(uint32_t info_mask)
|
|
|
+{
|
|
|
+ uint32_t info_presence = WMI_BEACON_INFO_PRESENCE_OUI_EXT;
|
|
|
+
|
|
|
+ if (info_mask & ACTION_OUI_INFO_MAC_ADDRESS)
|
|
|
+ info_presence |= WMI_BEACON_INFO_PRESENCE_MAC_ADDRESS;
|
|
|
+
|
|
|
+ if (info_mask & ACTION_OUI_INFO_AP_CAPABILITY_NSS)
|
|
|
+ info_presence |= WMI_BEACON_INFO_PRESENCE_AP_CAPABILITY_NSS;
|
|
|
+
|
|
|
+ if (info_mask & ACTION_OUI_INFO_AP_CAPABILITY_HT)
|
|
|
+ info_presence |= WMI_BEACON_INFO_PRESENCE_AP_CAPABILITY_HT;
|
|
|
+
|
|
|
+ if (info_mask & ACTION_OUI_INFO_AP_CAPABILITY_VHT)
|
|
|
+ info_presence |= WMI_BEACON_INFO_PRESENCE_AP_CAPABILITY_VHT;
|
|
|
+
|
|
|
+ if (info_mask & ACTION_OUI_INFO_AP_CAPABILITY_BAND)
|
|
|
+ info_presence |= WMI_BEACON_INFO_PRESENCE_AP_CAPABILITY_BAND;
|
|
|
+
|
|
|
+ return info_presence;
|
|
|
+}
|
|
|
+
|
|
|
+void wmi_fill_oui_extensions(struct action_oui_extension *extension,
|
|
|
+ uint32_t no_oui_extns,
|
|
|
+ wmi_vendor_oui_ext *cmd_ext)
|
|
|
+{
|
|
|
+ uint32_t i;
|
|
|
+ uint32_t buffer_length;
|
|
|
+
|
|
|
+ for (i = 0; i < no_oui_extns; i++) {
|
|
|
+ WMITLV_SET_HDR(&cmd_ext->tlv_header,
|
|
|
+ WMITLV_TAG_STRUC_wmi_vendor_oui_ext,
|
|
|
+ WMITLV_GET_STRUCT_TLVLEN(wmi_vendor_oui_ext));
|
|
|
+ cmd_ext->info_presence_bit_mask =
|
|
|
+ wmi_get_action_oui_info_mask(extension->info_mask);
|
|
|
+
|
|
|
+ cmd_ext->oui_header_length = extension->oui_length;
|
|
|
+ cmd_ext->oui_data_length = extension->data_length;
|
|
|
+ cmd_ext->mac_address_length = extension->mac_addr_length;
|
|
|
+ cmd_ext->capability_data_length =
|
|
|
+ extension->capability_length;
|
|
|
+
|
|
|
+ buffer_length = extension->oui_length +
|
|
|
+ extension->data_length +
|
|
|
+ extension->data_mask_length +
|
|
|
+ extension->mac_addr_length +
|
|
|
+ extension->mac_mask_length +
|
|
|
+ extension->capability_length;
|
|
|
+
|
|
|
+ cmd_ext->buf_data_length = buffer_length + 1;
|
|
|
+
|
|
|
+ cmd_ext++;
|
|
|
+ extension++;
|
|
|
+ }
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+QDF_STATUS
|
|
|
+wmi_fill_oui_extensions_buffer(struct action_oui_extension *extension,
|
|
|
+ wmi_vendor_oui_ext *cmd_ext,
|
|
|
+ uint32_t no_oui_extns, uint32_t rem_var_buf_len,
|
|
|
+ uint8_t *var_buf)
|
|
|
+{
|
|
|
+ uint8_t i;
|
|
|
+
|
|
|
+ for (i = 0; i < (uint8_t)no_oui_extns; i++) {
|
|
|
+ if ((rem_var_buf_len - cmd_ext->buf_data_length) < 0) {
|
|
|
+ WMI_LOGE(FL("Invalid action oui command length"));
|
|
|
+ return QDF_STATUS_E_INVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ var_buf[0] = i;
|
|
|
+ var_buf++;
|
|
|
+
|
|
|
+ qdf_mem_copy(var_buf, extension->oui, extension->oui_length);
|
|
|
+ var_buf += extension->oui_length;
|
|
|
+
|
|
|
+ if (extension->data_length) {
|
|
|
+ qdf_mem_copy(var_buf, extension->data,
|
|
|
+ extension->data_length);
|
|
|
+ var_buf += extension->data_length;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (extension->data_mask_length) {
|
|
|
+ qdf_mem_copy(var_buf, extension->data_mask,
|
|
|
+ extension->data_mask_length);
|
|
|
+ var_buf += extension->data_mask_length;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (extension->mac_addr_length) {
|
|
|
+ qdf_mem_copy(var_buf, extension->mac_addr,
|
|
|
+ extension->mac_addr_length);
|
|
|
+ var_buf += extension->mac_addr_length;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (extension->mac_mask_length) {
|
|
|
+ qdf_mem_copy(var_buf, extension->mac_mask,
|
|
|
+ extension->mac_mask_length);
|
|
|
+ var_buf += extension->mac_mask_length;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (extension->capability_length) {
|
|
|
+ qdf_mem_copy(var_buf, extension->capability,
|
|
|
+ extension->capability_length);
|
|
|
+ var_buf += extension->capability_length;
|
|
|
+ }
|
|
|
+
|
|
|
+ rem_var_buf_len -= cmd_ext->buf_data_length;
|
|
|
+ cmd_ext++;
|
|
|
+ extension++;
|
|
|
+ }
|
|
|
+
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
+}
|
|
|
+
|
|
|
+QDF_STATUS
|
|
|
+send_action_oui_cmd_tlv(wmi_unified_t wmi_handle,
|
|
|
+ struct action_oui_request *req)
|
|
|
+{
|
|
|
+ wmi_pdev_config_vendor_oui_action_fixed_param *cmd;
|
|
|
+ wmi_vendor_oui_ext *cmd_ext;
|
|
|
+ wmi_buf_t wmi_buf;
|
|
|
+ struct action_oui_extension *extension;
|
|
|
+ uint32_t len;
|
|
|
+ uint32_t i;
|
|
|
+ uint8_t *buf_ptr;
|
|
|
+ uint32_t no_oui_extns;
|
|
|
+ uint32_t total_no_oui_extns;
|
|
|
+ uint32_t var_buf_len = 0;
|
|
|
+ wmi_vendor_oui_action_id action_id;
|
|
|
+ bool valid;
|
|
|
+ uint32_t rem_var_buf_len;
|
|
|
+ QDF_STATUS status;
|
|
|
+
|
|
|
+ if (!req) {
|
|
|
+ WMI_LOGE(FL("action oui is empty"));
|
|
|
+ return QDF_STATUS_E_INVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ no_oui_extns = req->no_oui_extensions;
|
|
|
+ total_no_oui_extns = req->total_no_oui_extensions;
|
|
|
+
|
|
|
+ len = sizeof(*cmd);
|
|
|
+ len += WMI_TLV_HDR_SIZE; /* Array of wmi_vendor_oui_ext structures */
|
|
|
+
|
|
|
+ if (!no_oui_extns ||
|
|
|
+ no_oui_extns > WMI_MAX_VENDOR_OUI_ACTION_SUPPORTED_PER_ACTION ||
|
|
|
+ (total_no_oui_extns > WMI_VENDOR_OUI_ACTION_MAX_ACTION_ID *
|
|
|
+ WMI_MAX_VENDOR_OUI_ACTION_SUPPORTED_PER_ACTION)) {
|
|
|
+ WMI_LOGE(FL("Invalid number of action oui extensions"));
|
|
|
+ return QDF_STATUS_E_INVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ valid = wmi_get_action_oui_id(req->action_id, &action_id);
|
|
|
+ if (!valid) {
|
|
|
+ WMI_LOGE(FL("Invalid action id"));
|
|
|
+ return QDF_STATUS_E_INVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ len += no_oui_extns * sizeof(*cmd_ext);
|
|
|
+ len += WMI_TLV_HDR_SIZE; /* Variable length buffer */
|
|
|
+
|
|
|
+ extension = req->extension;
|
|
|
+ for (i = 0; i < no_oui_extns; i++) {
|
|
|
+ var_buf_len += extension->oui_length +
|
|
|
+ extension->data_length +
|
|
|
+ extension->data_mask_length +
|
|
|
+ extension->mac_addr_length +
|
|
|
+ extension->mac_mask_length +
|
|
|
+ extension->capability_length;
|
|
|
+ extension++;
|
|
|
+ }
|
|
|
+
|
|
|
+ var_buf_len += no_oui_extns; /* to store indexes */
|
|
|
+ rem_var_buf_len = var_buf_len;
|
|
|
+ var_buf_len = (var_buf_len + 3) & ~0x3;
|
|
|
+ len += var_buf_len;
|
|
|
+
|
|
|
+ wmi_buf = wmi_buf_alloc(wmi_handle, len);
|
|
|
+ if (!wmi_buf) {
|
|
|
+ WMI_LOGE(FL("Failed to allocate wmi buffer"));
|
|
|
+ return QDF_STATUS_E_FAILURE;
|
|
|
+ }
|
|
|
+
|
|
|
+ buf_ptr = (uint8_t *)wmi_buf_data(wmi_buf);
|
|
|
+ cmd = (wmi_pdev_config_vendor_oui_action_fixed_param *)buf_ptr;
|
|
|
+
|
|
|
+ WMITLV_SET_HDR(&cmd->tlv_header,
|
|
|
+ WMITLV_TAG_STRUC_wmi_pdev_config_vendor_oui_action_fixed_param,
|
|
|
+ WMITLV_GET_STRUCT_TLVLEN(
|
|
|
+ wmi_pdev_config_vendor_oui_action_fixed_param));
|
|
|
+
|
|
|
+ cmd->action_id = action_id;
|
|
|
+ cmd->total_num_vendor_oui = total_no_oui_extns;
|
|
|
+ cmd->num_vendor_oui_ext = no_oui_extns;
|
|
|
+
|
|
|
+ buf_ptr += sizeof(*cmd);
|
|
|
+ WMITLV_SET_HDR(buf_ptr, WMITLV_TAG_ARRAY_STRUC,
|
|
|
+ no_oui_extns * sizeof(*cmd_ext));
|
|
|
+ buf_ptr += WMI_TLV_HDR_SIZE;
|
|
|
+ cmd_ext = (wmi_vendor_oui_ext *)buf_ptr;
|
|
|
+ wmi_fill_oui_extensions(req->extension, no_oui_extns, cmd_ext);
|
|
|
+
|
|
|
+ buf_ptr += no_oui_extns * sizeof(*cmd_ext);
|
|
|
+ WMITLV_SET_HDR(buf_ptr, WMITLV_TAG_ARRAY_BYTE, var_buf_len);
|
|
|
+ buf_ptr += WMI_TLV_HDR_SIZE;
|
|
|
+ status = wmi_fill_oui_extensions_buffer(req->extension,
|
|
|
+ cmd_ext, no_oui_extns,
|
|
|
+ rem_var_buf_len, buf_ptr);
|
|
|
+ if (!QDF_IS_STATUS_SUCCESS(status)) {
|
|
|
+ wmi_buf_free(wmi_buf);
|
|
|
+ wmi_buf = NULL;
|
|
|
+ return QDF_STATUS_E_INVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ buf_ptr += var_buf_len;
|
|
|
+
|
|
|
+ if (wmi_unified_cmd_send(wmi_handle, wmi_buf, len,
|
|
|
+ WMI_PDEV_CONFIG_VENDOR_OUI_ACTION_CMDID)) {
|
|
|
+ WMI_LOGE(FL("WMI_PDEV_CONFIG_VENDOR_OUI_ACTION send fail"));
|
|
|
+ wmi_buf_free(wmi_buf);
|
|
|
+ wmi_buf = NULL;
|
|
|
+ return QDF_STATUS_E_FAILURE;
|
|
|
+ }
|
|
|
+
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
+}
|