|
@@ -0,0 +1,1222 @@
|
|
|
+/*
|
|
|
+ * Copyright (c) 2012-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.
|
|
|
+ */
|
|
|
+
|
|
|
+/**
|
|
|
+ * DOC: wlan_hdd_station_info.c
|
|
|
+ *
|
|
|
+ * WLAN station info functions
|
|
|
+ *
|
|
|
+ */
|
|
|
+
|
|
|
+#include <wlan_hdd_includes.h>
|
|
|
+#include <linux/netdevice.h>
|
|
|
+#include <linux/skbuff.h>
|
|
|
+#include <linux/etherdevice.h>
|
|
|
+#include <linux/if_ether.h>
|
|
|
+#include <wlan_cp_stats_mc_ucfg_api.h>
|
|
|
+#include <wlan_hdd_stats.h>
|
|
|
+#include <wlan_hdd_hostapd.h>
|
|
|
+#include <wlan_hdd_station_info.h>
|
|
|
+
|
|
|
+/*
|
|
|
+ * define short names for the global vendor params
|
|
|
+ * used by __wlan_hdd_cfg80211_get_station_cmd()
|
|
|
+ */
|
|
|
+#define STATION_INVALID \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INVALID
|
|
|
+#define STATION_INFO \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO
|
|
|
+#define STATION_ASSOC_FAIL_REASON \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_ASSOC_FAIL_REASON
|
|
|
+#define STATION_REMOTE \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_REMOTE
|
|
|
+#define STATION_MAX \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_MAX
|
|
|
+
|
|
|
+/* define short names for get station info attributes */
|
|
|
+#define LINK_INFO_STANDARD_NL80211_ATTR \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_LINK_STANDARD_NL80211_ATTR
|
|
|
+#define AP_INFO_STANDARD_NL80211_ATTR \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_AP_STANDARD_NL80211_ATTR
|
|
|
+#define INFO_ROAM_COUNT \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_ROAM_COUNT
|
|
|
+#define INFO_AKM \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_AKM
|
|
|
+#define WLAN802_11_MODE \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_802_11_MODE
|
|
|
+#define AP_INFO_HS20_INDICATION \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_AP_HS20_INDICATION
|
|
|
+#define HT_OPERATION \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_HT_OPERATION
|
|
|
+#define VHT_OPERATION \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_VHT_OPERATION
|
|
|
+#define INFO_ASSOC_FAIL_REASON \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_ASSOC_FAIL_REASON
|
|
|
+#define REMOTE_MAX_PHY_RATE \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_MAX_PHY_RATE
|
|
|
+#define REMOTE_TX_PACKETS \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_TX_PACKETS
|
|
|
+#define REMOTE_TX_BYTES \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_TX_BYTES
|
|
|
+#define REMOTE_RX_PACKETS \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_RX_PACKETS
|
|
|
+#define REMOTE_RX_BYTES \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_RX_BYTES
|
|
|
+#define REMOTE_LAST_TX_RATE \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_LAST_TX_RATE
|
|
|
+#define REMOTE_LAST_RX_RATE \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_LAST_RX_RATE
|
|
|
+#define REMOTE_WMM \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_WMM
|
|
|
+#define REMOTE_SUPPORTED_MODE \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_SUPPORTED_MODE
|
|
|
+#define REMOTE_AMPDU \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_AMPDU
|
|
|
+#define REMOTE_TX_STBC \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_TX_STBC
|
|
|
+#define REMOTE_RX_STBC \
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_RX_STBC
|
|
|
+#define REMOTE_CH_WIDTH\
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_CH_WIDTH
|
|
|
+#define REMOTE_SGI_ENABLE\
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_REMOTE_SGI_ENABLE
|
|
|
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(4, 7, 0))
|
|
|
+ #define REMOTE_PAD\
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_INFO_PAD
|
|
|
+#endif
|
|
|
+
|
|
|
+static const struct nla_policy
|
|
|
+hdd_get_station_policy[STATION_MAX + 1] = {
|
|
|
+ [STATION_INFO] = {.type = NLA_FLAG},
|
|
|
+ [STATION_ASSOC_FAIL_REASON] = {.type = NLA_FLAG},
|
|
|
+ [STATION_REMOTE] = {.type = NLA_BINARY, .len = QDF_MAC_ADDR_SIZE},
|
|
|
+};
|
|
|
+
|
|
|
+#ifdef QCA_SUPPORT_CP_STATS
|
|
|
+static int hdd_get_sta_congestion(struct hdd_adapter *adapter,
|
|
|
+ uint32_t *congestion)
|
|
|
+{
|
|
|
+ QDF_STATUS status;
|
|
|
+ struct cca_stats cca_stats;
|
|
|
+
|
|
|
+ status = ucfg_mc_cp_stats_cca_stats_get(adapter->hdd_vdev, &cca_stats);
|
|
|
+ if (QDF_IS_STATUS_ERROR(status))
|
|
|
+ return -EINVAL;
|
|
|
+
|
|
|
+ *congestion = cca_stats.congestion;
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+#else
|
|
|
+static int hdd_get_sta_congestion(struct hdd_adapter *adapter,
|
|
|
+ uint32_t *congestion)
|
|
|
+{
|
|
|
+ struct hdd_station_ctx *hdd_sta_ctx;
|
|
|
+
|
|
|
+ hdd_sta_ctx = WLAN_HDD_GET_STATION_CTX_PTR(adapter);
|
|
|
+ *congestion = hdd_sta_ctx->conn_info.cca;
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_get_station_assoc_fail() - Handle get station assoc fail
|
|
|
+ * @hdd_ctx: HDD context within host driver
|
|
|
+ * @wdev: wireless device
|
|
|
+ *
|
|
|
+ * Handles QCA_NL80211_VENDOR_SUBCMD_GET_STATION_ASSOC_FAIL.
|
|
|
+ * Validate cmd attributes and send the station info to upper layers.
|
|
|
+ *
|
|
|
+ * Return: Success(0) or reason code for failure
|
|
|
+ */
|
|
|
+static int hdd_get_station_assoc_fail(struct hdd_context *hdd_ctx,
|
|
|
+ struct hdd_adapter *adapter)
|
|
|
+{
|
|
|
+ struct sk_buff *skb = NULL;
|
|
|
+ uint32_t nl_buf_len;
|
|
|
+ struct hdd_station_ctx *hdd_sta_ctx;
|
|
|
+ uint32_t congestion;
|
|
|
+
|
|
|
+ nl_buf_len = NLMSG_HDRLEN;
|
|
|
+ nl_buf_len += sizeof(uint32_t);
|
|
|
+ skb = cfg80211_vendor_cmd_alloc_reply_skb(hdd_ctx->wiphy, nl_buf_len);
|
|
|
+
|
|
|
+ if (!skb) {
|
|
|
+ hdd_err("cfg80211_vendor_cmd_alloc_reply_skb failed");
|
|
|
+ return -ENOMEM;
|
|
|
+ }
|
|
|
+
|
|
|
+ hdd_sta_ctx = WLAN_HDD_GET_STATION_CTX_PTR(adapter);
|
|
|
+
|
|
|
+ if (nla_put_u32(skb, INFO_ASSOC_FAIL_REASON,
|
|
|
+ hdd_sta_ctx->conn_info.assoc_status_code)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (hdd_get_sta_congestion(adapter, &congestion))
|
|
|
+ congestion = 0;
|
|
|
+
|
|
|
+ hdd_info("congestion:%d", congestion);
|
|
|
+ if (nla_put_u32(skb, NL80211_SURVEY_INFO_CHANNEL_TIME_BUSY,
|
|
|
+ congestion)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ return cfg80211_vendor_cmd_reply(skb);
|
|
|
+fail:
|
|
|
+ if (skb)
|
|
|
+ kfree_skb(skb);
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_map_auth_type() - transform auth type specific to
|
|
|
+ * vendor command
|
|
|
+ * @auth_type: csr auth type
|
|
|
+ *
|
|
|
+ * Return: Success(0) or reason code for failure
|
|
|
+ */
|
|
|
+static int hdd_convert_auth_type(uint32_t auth_type)
|
|
|
+{
|
|
|
+ uint32_t ret_val;
|
|
|
+
|
|
|
+ switch (auth_type) {
|
|
|
+ case eCSR_AUTH_TYPE_OPEN_SYSTEM:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_OPEN;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_SHARED_KEY:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_SHARED;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_WPA:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_WPA;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_WPA_PSK:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_WPA_PSK;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_AUTOSWITCH:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_AUTOSWITCH;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_WPA_NONE:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_WPA_NONE;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_RSN:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_RSN;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_RSN_PSK:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_RSN_PSK;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_FT_RSN:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_FT;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_FT_RSN_PSK:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_FT_PSK;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_WAPI_WAI_CERTIFICATE:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_WAI;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_WAPI_WAI_PSK:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_WAI_PSK;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_CCKM_WPA:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_CCKM_WPA;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_CCKM_RSN:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_CCKM_RSN;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_RSN_PSK_SHA256:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_SHA256_PSK;
|
|
|
+ break;
|
|
|
+ case eCSR_AUTH_TYPE_RSN_8021X_SHA256:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_SHA256;
|
|
|
+ break;
|
|
|
+ case eCSR_NUM_OF_SUPPORT_AUTH_TYPE:
|
|
|
+ case eCSR_AUTH_TYPE_FAILED:
|
|
|
+ case eCSR_AUTH_TYPE_NONE:
|
|
|
+ default:
|
|
|
+ ret_val = QCA_WLAN_AUTH_TYPE_INVALID;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ return ret_val;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_map_dot_11_mode() - transform dot11mode type specific to
|
|
|
+ * vendor command
|
|
|
+ * @dot11mode: dot11mode
|
|
|
+ *
|
|
|
+ * Return: Success(0) or reason code for failure
|
|
|
+ */
|
|
|
+static int hdd_convert_dot11mode(uint32_t dot11mode)
|
|
|
+{
|
|
|
+ uint32_t ret_val;
|
|
|
+
|
|
|
+ switch (dot11mode) {
|
|
|
+ case eCSR_CFG_DOT11_MODE_11A:
|
|
|
+ ret_val = QCA_WLAN_802_11_MODE_11A;
|
|
|
+ break;
|
|
|
+ case eCSR_CFG_DOT11_MODE_11B:
|
|
|
+ ret_val = QCA_WLAN_802_11_MODE_11B;
|
|
|
+ break;
|
|
|
+ case eCSR_CFG_DOT11_MODE_11G:
|
|
|
+ ret_val = QCA_WLAN_802_11_MODE_11G;
|
|
|
+ break;
|
|
|
+ case eCSR_CFG_DOT11_MODE_11N:
|
|
|
+ ret_val = QCA_WLAN_802_11_MODE_11N;
|
|
|
+ break;
|
|
|
+ case eCSR_CFG_DOT11_MODE_11AC:
|
|
|
+ ret_val = QCA_WLAN_802_11_MODE_11AC;
|
|
|
+ break;
|
|
|
+ case eCSR_CFG_DOT11_MODE_AUTO:
|
|
|
+ case eCSR_CFG_DOT11_MODE_ABG:
|
|
|
+ default:
|
|
|
+ ret_val = QCA_WLAN_802_11_MODE_INVALID;
|
|
|
+ }
|
|
|
+ return ret_val;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_tx_bitrate() - add tx bitrate attribute
|
|
|
+ * @skb: pointer to sk buff
|
|
|
+ * @hdd_sta_ctx: pointer to hdd station context
|
|
|
+ * @idx: attribute index
|
|
|
+ *
|
|
|
+ * Return: Success(0) or reason code for failure
|
|
|
+ */
|
|
|
+static int32_t hdd_add_tx_bitrate(struct sk_buff *skb,
|
|
|
+ struct hdd_station_ctx *hdd_sta_ctx,
|
|
|
+ int idx)
|
|
|
+{
|
|
|
+ struct nlattr *nla_attr;
|
|
|
+ uint32_t bitrate, bitrate_compat;
|
|
|
+
|
|
|
+ nla_attr = nla_nest_start(skb, idx);
|
|
|
+ if (!nla_attr) {
|
|
|
+ hdd_err("nla_nest_start failed");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* cfg80211_calculate_bitrate will return 0 for mcs >= 32 */
|
|
|
+ bitrate = cfg80211_calculate_bitrate(&hdd_sta_ctx->
|
|
|
+ cache_conn_info.txrate);
|
|
|
+
|
|
|
+ /* report 16-bit bitrate only if we can */
|
|
|
+ bitrate_compat = bitrate < (1UL << 16) ? bitrate : 0;
|
|
|
+
|
|
|
+ if (bitrate > 0) {
|
|
|
+ if (nla_put_u32(skb, NL80211_RATE_INFO_BITRATE32, bitrate)) {
|
|
|
+ hdd_err("put fail bitrate: %u", bitrate);
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ hdd_err("Invalid bitrate: %u", bitrate);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (bitrate_compat > 0) {
|
|
|
+ if (nla_put_u16(skb, NL80211_RATE_INFO_BITRATE,
|
|
|
+ bitrate_compat)) {
|
|
|
+ hdd_err("put fail bitrate_compat: %u", bitrate_compat);
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ hdd_err("Invalid bitrate_compat: %u", bitrate_compat);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (nla_put_u8(skb, NL80211_RATE_INFO_VHT_NSS,
|
|
|
+ hdd_sta_ctx->cache_conn_info.txrate.nss)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ nla_nest_end(skb, nla_attr);
|
|
|
+ return 0;
|
|
|
+fail:
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_sta_info() - add station info attribute
|
|
|
+ * @skb: pointer to sk buff
|
|
|
+ * @hdd_sta_ctx: pointer to hdd station context
|
|
|
+ * @idx: attribute index
|
|
|
+ *
|
|
|
+ * Return: Success(0) or reason code for failure
|
|
|
+ */
|
|
|
+static int32_t hdd_add_sta_info(struct sk_buff *skb,
|
|
|
+ struct hdd_station_ctx *hdd_sta_ctx,
|
|
|
+ int idx)
|
|
|
+{
|
|
|
+ struct nlattr *nla_attr;
|
|
|
+
|
|
|
+ nla_attr = nla_nest_start(skb, idx);
|
|
|
+ if (!nla_attr) {
|
|
|
+ hdd_err("nla_nest_start failed");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (nla_put_u8(skb, NL80211_STA_INFO_SIGNAL,
|
|
|
+ (hdd_sta_ctx->cache_conn_info.signal + 100))) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ if (hdd_add_tx_bitrate(skb, hdd_sta_ctx, NL80211_STA_INFO_TX_BITRATE)) {
|
|
|
+ hdd_err("hdd_add_tx_bitrate failed");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ nla_nest_end(skb, nla_attr);
|
|
|
+ return 0;
|
|
|
+fail:
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_survey_info() - add survey info attribute
|
|
|
+ * @skb: pointer to sk buff
|
|
|
+ * @hdd_sta_ctx: pointer to hdd station context
|
|
|
+ * @idx: attribute index
|
|
|
+ *
|
|
|
+ * Return: Success(0) or reason code for failure
|
|
|
+ */
|
|
|
+static int32_t hdd_add_survey_info(struct sk_buff *skb,
|
|
|
+ struct hdd_station_ctx *hdd_sta_ctx,
|
|
|
+ int idx)
|
|
|
+{
|
|
|
+ struct nlattr *nla_attr;
|
|
|
+
|
|
|
+ nla_attr = nla_nest_start(skb, idx);
|
|
|
+ if (!nla_attr)
|
|
|
+ goto fail;
|
|
|
+ if (nla_put_u32(skb, NL80211_SURVEY_INFO_FREQUENCY,
|
|
|
+ hdd_sta_ctx->cache_conn_info.freq) ||
|
|
|
+ nla_put_u8(skb, NL80211_SURVEY_INFO_NOISE,
|
|
|
+ (hdd_sta_ctx->cache_conn_info.noise + 100))) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ nla_nest_end(skb, nla_attr);
|
|
|
+ return 0;
|
|
|
+fail:
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_link_standard_info() - add link info attribute
|
|
|
+ * @skb: pointer to sk buff
|
|
|
+ * @hdd_sta_ctx: pointer to hdd station context
|
|
|
+ * @idx: attribute index
|
|
|
+ *
|
|
|
+ * Return: Success(0) or reason code for failure
|
|
|
+ */
|
|
|
+static int32_t
|
|
|
+hdd_add_link_standard_info(struct sk_buff *skb,
|
|
|
+ struct hdd_station_ctx *hdd_sta_ctx, int idx)
|
|
|
+{
|
|
|
+ struct nlattr *nla_attr;
|
|
|
+
|
|
|
+ nla_attr = nla_nest_start(skb, idx);
|
|
|
+ if (!nla_attr) {
|
|
|
+ hdd_err("nla_nest_start failed");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (nla_put(skb,
|
|
|
+ NL80211_ATTR_SSID,
|
|
|
+ hdd_sta_ctx->cache_conn_info.last_ssid.SSID.length,
|
|
|
+ hdd_sta_ctx->cache_conn_info.last_ssid.SSID.ssId)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ if (nla_put(skb, NL80211_ATTR_MAC, QDF_MAC_ADDR_SIZE,
|
|
|
+ hdd_sta_ctx->cache_conn_info.bssId.bytes)) {
|
|
|
+ hdd_err("put bssid failed");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ if (hdd_add_survey_info(skb, hdd_sta_ctx, NL80211_ATTR_SURVEY_INFO)) {
|
|
|
+ hdd_err("hdd_add_survey_info failed");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (hdd_add_sta_info(skb, hdd_sta_ctx, NL80211_ATTR_STA_INFO)) {
|
|
|
+ hdd_err("hdd_add_sta_info failed");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ nla_nest_end(skb, nla_attr);
|
|
|
+ return 0;
|
|
|
+fail:
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_ap_standard_info() - add ap info attribute
|
|
|
+ * @skb: pointer to sk buff
|
|
|
+ * @hdd_sta_ctx: pointer to hdd station context
|
|
|
+ * @idx: attribute index
|
|
|
+ *
|
|
|
+ * Return: Success(0) or reason code for failure
|
|
|
+ */
|
|
|
+static int32_t
|
|
|
+hdd_add_ap_standard_info(struct sk_buff *skb,
|
|
|
+ struct hdd_station_ctx *hdd_sta_ctx, int idx)
|
|
|
+{
|
|
|
+ struct nlattr *nla_attr;
|
|
|
+
|
|
|
+ nla_attr = nla_nest_start(skb, idx);
|
|
|
+ if (!nla_attr)
|
|
|
+ goto fail;
|
|
|
+ if (hdd_sta_ctx->cache_conn_info.conn_flag.vht_present)
|
|
|
+ if (nla_put(skb, NL80211_ATTR_VHT_CAPABILITY,
|
|
|
+ sizeof(hdd_sta_ctx->cache_conn_info.vht_caps),
|
|
|
+ &hdd_sta_ctx->cache_conn_info.vht_caps)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ if (hdd_sta_ctx->cache_conn_info.conn_flag.ht_present)
|
|
|
+ if (nla_put(skb, NL80211_ATTR_HT_CAPABILITY,
|
|
|
+ sizeof(hdd_sta_ctx->cache_conn_info.ht_caps),
|
|
|
+ &hdd_sta_ctx->cache_conn_info.ht_caps)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ nla_nest_end(skb, nla_attr);
|
|
|
+ return 0;
|
|
|
+fail:
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_get_station_info() - send BSS information to supplicant
|
|
|
+ * @hdd_ctx: pointer to hdd context
|
|
|
+ * @adapter: pointer to adapter
|
|
|
+ *
|
|
|
+ * Return: 0 if success else error status
|
|
|
+ */
|
|
|
+static int hdd_get_station_info(struct hdd_context *hdd_ctx,
|
|
|
+ struct hdd_adapter *adapter)
|
|
|
+{
|
|
|
+ struct sk_buff *skb = NULL;
|
|
|
+ uint8_t *tmp_hs20 = NULL;
|
|
|
+ uint32_t nl_buf_len;
|
|
|
+ struct hdd_station_ctx *hdd_sta_ctx;
|
|
|
+
|
|
|
+ hdd_sta_ctx = WLAN_HDD_GET_STATION_CTX_PTR(adapter);
|
|
|
+
|
|
|
+ nl_buf_len = NLMSG_HDRLEN;
|
|
|
+ nl_buf_len += sizeof(hdd_sta_ctx->
|
|
|
+ cache_conn_info.last_ssid.SSID.length) +
|
|
|
+ QDF_MAC_ADDR_SIZE +
|
|
|
+ sizeof(hdd_sta_ctx->cache_conn_info.freq) +
|
|
|
+ sizeof(hdd_sta_ctx->cache_conn_info.noise) +
|
|
|
+ sizeof(hdd_sta_ctx->cache_conn_info.signal) +
|
|
|
+ (sizeof(uint32_t) * 2) +
|
|
|
+ sizeof(hdd_sta_ctx->cache_conn_info.txrate.nss) +
|
|
|
+ sizeof(hdd_sta_ctx->cache_conn_info.roam_count) +
|
|
|
+ sizeof(hdd_sta_ctx->cache_conn_info.last_auth_type) +
|
|
|
+ sizeof(hdd_sta_ctx->cache_conn_info.dot11Mode);
|
|
|
+ if (hdd_sta_ctx->cache_conn_info.conn_flag.vht_present)
|
|
|
+ nl_buf_len += sizeof(hdd_sta_ctx->cache_conn_info.vht_caps);
|
|
|
+ if (hdd_sta_ctx->cache_conn_info.conn_flag.ht_present)
|
|
|
+ nl_buf_len += sizeof(hdd_sta_ctx->cache_conn_info.ht_caps);
|
|
|
+ if (hdd_sta_ctx->cache_conn_info.conn_flag.hs20_present) {
|
|
|
+ tmp_hs20 = (uint8_t *)&(hdd_sta_ctx->
|
|
|
+ cache_conn_info.hs20vendor_ie);
|
|
|
+ nl_buf_len += (sizeof(hdd_sta_ctx->
|
|
|
+ cache_conn_info.hs20vendor_ie) - 1);
|
|
|
+ }
|
|
|
+ if (hdd_sta_ctx->cache_conn_info.conn_flag.ht_op_present)
|
|
|
+ nl_buf_len += sizeof(hdd_sta_ctx->
|
|
|
+ cache_conn_info.ht_operation);
|
|
|
+ if (hdd_sta_ctx->cache_conn_info.conn_flag.vht_op_present)
|
|
|
+ nl_buf_len += sizeof(hdd_sta_ctx->
|
|
|
+ cache_conn_info.vht_operation);
|
|
|
+
|
|
|
+ skb = cfg80211_vendor_cmd_alloc_reply_skb(hdd_ctx->wiphy, nl_buf_len);
|
|
|
+ if (!skb) {
|
|
|
+ hdd_err("cfg80211_vendor_cmd_alloc_reply_skb failed");
|
|
|
+ return -ENOMEM;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (hdd_add_link_standard_info(skb, hdd_sta_ctx,
|
|
|
+ LINK_INFO_STANDARD_NL80211_ATTR)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ if (hdd_add_ap_standard_info(skb, hdd_sta_ctx,
|
|
|
+ AP_INFO_STANDARD_NL80211_ATTR)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ if (nla_put_u32(skb, INFO_ROAM_COUNT,
|
|
|
+ hdd_sta_ctx->cache_conn_info.roam_count) ||
|
|
|
+ nla_put_u32(skb, INFO_AKM,
|
|
|
+ hdd_convert_auth_type(
|
|
|
+ hdd_sta_ctx->cache_conn_info.last_auth_type)) ||
|
|
|
+ nla_put_u32(skb, WLAN802_11_MODE,
|
|
|
+ hdd_convert_dot11mode(
|
|
|
+ hdd_sta_ctx->cache_conn_info.dot11Mode))) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ if (hdd_sta_ctx->cache_conn_info.conn_flag.ht_op_present)
|
|
|
+ if (nla_put(skb, HT_OPERATION,
|
|
|
+ (sizeof(hdd_sta_ctx->cache_conn_info.ht_operation)),
|
|
|
+ &hdd_sta_ctx->cache_conn_info.ht_operation)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ if (hdd_sta_ctx->cache_conn_info.conn_flag.vht_op_present)
|
|
|
+ if (nla_put(skb, VHT_OPERATION,
|
|
|
+ (sizeof(hdd_sta_ctx->
|
|
|
+ cache_conn_info.vht_operation)),
|
|
|
+ &hdd_sta_ctx->cache_conn_info.vht_operation)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ if (hdd_sta_ctx->cache_conn_info.conn_flag.hs20_present)
|
|
|
+ if (nla_put(skb, AP_INFO_HS20_INDICATION,
|
|
|
+ (sizeof(hdd_sta_ctx->cache_conn_info.hs20vendor_ie)
|
|
|
+ - 1),
|
|
|
+ tmp_hs20 + 1)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ return cfg80211_vendor_cmd_reply(skb);
|
|
|
+fail:
|
|
|
+ if (skb)
|
|
|
+ kfree_skb(skb);
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(4, 7, 0))
|
|
|
+static inline int32_t remote_station_put_u64(struct sk_buff *skb,
|
|
|
+ int32_t attrtype,
|
|
|
+ uint64_t value)
|
|
|
+{
|
|
|
+ return nla_put_u64_64bit(skb, attrtype, value, REMOTE_PAD);
|
|
|
+}
|
|
|
+#else
|
|
|
+static inline int32_t remote_station_put_u64(struct sk_buff *skb,
|
|
|
+ int32_t attrtype,
|
|
|
+ uint64_t value)
|
|
|
+{
|
|
|
+ return nla_put_u64(skb, attrtype, value);
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_survey_info_sap_get_len - get data length used in
|
|
|
+ * hdd_add_survey_info_sap()
|
|
|
+ *
|
|
|
+ * This function calculates the data length used in hdd_add_survey_info_sap()
|
|
|
+ *
|
|
|
+ * Return: total data length used in hdd_add_survey_info_sap()
|
|
|
+ */
|
|
|
+static uint32_t hdd_add_survey_info_sap_get_len(void)
|
|
|
+{
|
|
|
+ return ((NLA_HDRLEN) + (sizeof(uint32_t) + NLA_HDRLEN));
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_survey_info - add survey info attribute
|
|
|
+ * @skb: pointer to response skb buffer
|
|
|
+ * @stainfo: station information
|
|
|
+ * @idx: attribute type index for nla_next_start()
|
|
|
+ *
|
|
|
+ * This function adds survey info attribute to response skb buffer
|
|
|
+ *
|
|
|
+ * Return : 0 on success and errno on failure
|
|
|
+ */
|
|
|
+static int32_t hdd_add_survey_info_sap(struct sk_buff *skb,
|
|
|
+ struct hdd_station_info *stainfo,
|
|
|
+ int idx)
|
|
|
+{
|
|
|
+ struct nlattr *nla_attr;
|
|
|
+
|
|
|
+ nla_attr = nla_nest_start(skb, idx);
|
|
|
+ if (!nla_attr)
|
|
|
+ goto fail;
|
|
|
+ if (nla_put_u32(skb, NL80211_SURVEY_INFO_FREQUENCY,
|
|
|
+ stainfo->freq)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ nla_nest_end(skb, nla_attr);
|
|
|
+ return 0;
|
|
|
+fail:
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_tx_bitrate_sap_get_len - get data length used in
|
|
|
+ * hdd_add_tx_bitrate_sap()
|
|
|
+ *
|
|
|
+ * This function calculates the data length used in hdd_add_tx_bitrate_sap()
|
|
|
+ *
|
|
|
+ * Return: total data length used in hdd_add_tx_bitrate_sap()
|
|
|
+ */
|
|
|
+static uint32_t hdd_add_tx_bitrate_sap_get_len(void)
|
|
|
+{
|
|
|
+ return ((NLA_HDRLEN) + (sizeof(uint8_t) + NLA_HDRLEN));
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_tx_bitrate_sap - add vhs nss info attribute
|
|
|
+ * @skb: pointer to response skb buffer
|
|
|
+ * @stainfo: station information
|
|
|
+ * @idx: attribute type index for nla_next_start()
|
|
|
+ *
|
|
|
+ * This function adds vht nss attribute to response skb buffer
|
|
|
+ *
|
|
|
+ * Return : 0 on success and errno on failure
|
|
|
+ */
|
|
|
+static int hdd_add_tx_bitrate_sap(struct sk_buff *skb,
|
|
|
+ struct hdd_station_info *stainfo,
|
|
|
+ int idx)
|
|
|
+{
|
|
|
+ struct nlattr *nla_attr;
|
|
|
+
|
|
|
+ nla_attr = nla_nest_start(skb, idx);
|
|
|
+ if (!nla_attr)
|
|
|
+ goto fail;
|
|
|
+
|
|
|
+ if (nla_put_u8(skb, NL80211_RATE_INFO_VHT_NSS,
|
|
|
+ stainfo->nss)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ nla_nest_end(skb, nla_attr);
|
|
|
+ return 0;
|
|
|
+fail:
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_sta_info_sap_get_len - get data length used in
|
|
|
+ * hdd_add_sta_info_sap()
|
|
|
+ *
|
|
|
+ * This function calculates the data length used in hdd_add_sta_info_sap()
|
|
|
+ *
|
|
|
+ * Return: total data length used in hdd_add_sta_info_sap()
|
|
|
+ */
|
|
|
+static uint32_t hdd_add_sta_info_sap_get_len(void)
|
|
|
+{
|
|
|
+ return ((NLA_HDRLEN) + (sizeof(uint8_t) + NLA_HDRLEN) +
|
|
|
+ hdd_add_tx_bitrate_sap_get_len());
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_sta_info_sap - add sta signal info attribute
|
|
|
+ * @skb: pointer to response skb buffer
|
|
|
+ * @stainfo: station information
|
|
|
+ * @idx: attribute type index for nla_next_start()
|
|
|
+ *
|
|
|
+ * This function adds sta signal attribute to response skb buffer
|
|
|
+ *
|
|
|
+ * Return : 0 on success and errno on failure
|
|
|
+ */
|
|
|
+static int32_t hdd_add_sta_info_sap(struct sk_buff *skb, int8_t rssi,
|
|
|
+ struct hdd_station_info *stainfo, int idx)
|
|
|
+{
|
|
|
+ struct nlattr *nla_attr;
|
|
|
+
|
|
|
+ nla_attr = nla_nest_start(skb, idx);
|
|
|
+ if (!nla_attr)
|
|
|
+ goto fail;
|
|
|
+
|
|
|
+ if (nla_put_u8(skb, NL80211_STA_INFO_SIGNAL, rssi)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ if (hdd_add_tx_bitrate_sap(skb, stainfo, NL80211_STA_INFO_TX_BITRATE))
|
|
|
+ goto fail;
|
|
|
+
|
|
|
+ nla_nest_end(skb, nla_attr);
|
|
|
+ return 0;
|
|
|
+fail:
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_link_standard_info_sap_get_len - get data length used in
|
|
|
+ * hdd_add_link_standard_info_sap()
|
|
|
+ *
|
|
|
+ * This function calculates the data length used in
|
|
|
+ * hdd_add_link_standard_info_sap()
|
|
|
+ *
|
|
|
+ * Return: total data length used in hdd_add_link_standard_info_sap()
|
|
|
+ */
|
|
|
+static uint32_t hdd_add_link_standard_info_sap_get_len(void)
|
|
|
+{
|
|
|
+ return ((NLA_HDRLEN) +
|
|
|
+ hdd_add_survey_info_sap_get_len() +
|
|
|
+ hdd_add_sta_info_sap_get_len() +
|
|
|
+ (sizeof(uint32_t) + NLA_HDRLEN));
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_link_standard_info_sap - add add link info attribut
|
|
|
+ * @skb: pointer to response skb buffer
|
|
|
+ * @stainfo: station information
|
|
|
+ * @idx: attribute type index for nla_next_start()
|
|
|
+ *
|
|
|
+ * This function adds link info attribut to response skb buffer
|
|
|
+ *
|
|
|
+ * Return : 0 on success and errno on failure
|
|
|
+ */
|
|
|
+static int hdd_add_link_standard_info_sap(struct sk_buff *skb, int8_t rssi,
|
|
|
+ struct hdd_station_info *stainfo,
|
|
|
+ int idx)
|
|
|
+{
|
|
|
+ struct nlattr *nla_attr;
|
|
|
+
|
|
|
+ nla_attr = nla_nest_start(skb, idx);
|
|
|
+ if (!nla_attr)
|
|
|
+ goto fail;
|
|
|
+ if (hdd_add_survey_info_sap(skb, stainfo, NL80211_ATTR_SURVEY_INFO))
|
|
|
+ goto fail;
|
|
|
+ if (hdd_add_sta_info_sap(skb, rssi, stainfo, NL80211_ATTR_STA_INFO))
|
|
|
+ goto fail;
|
|
|
+
|
|
|
+ if (nla_put_u32(skb, NL80211_ATTR_REASON_CODE, stainfo->reason_code)) {
|
|
|
+ hdd_err("Reason code put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ nla_nest_end(skb, nla_attr);
|
|
|
+ return 0;
|
|
|
+fail:
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_ap_standard_info_sap_get_len - get data length used in
|
|
|
+ * hdd_add_ap_standard_info_sap()
|
|
|
+ * @stainfo: station information
|
|
|
+ *
|
|
|
+ * This function calculates the data length used in
|
|
|
+ * hdd_add_ap_standard_info_sap()
|
|
|
+ *
|
|
|
+ * Return: total data length used in hdd_add_ap_standard_info_sap()
|
|
|
+ */
|
|
|
+static uint32_t hdd_add_ap_standard_info_sap_get_len(
|
|
|
+ struct hdd_station_info *stainfo)
|
|
|
+{
|
|
|
+ uint32_t len;
|
|
|
+
|
|
|
+ len = NLA_HDRLEN;
|
|
|
+ if (stainfo->vht_present)
|
|
|
+ len += (sizeof(stainfo->vht_caps) + NLA_HDRLEN);
|
|
|
+ if (stainfo->ht_present)
|
|
|
+ len += (sizeof(stainfo->ht_caps) + NLA_HDRLEN);
|
|
|
+
|
|
|
+ return len;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_add_ap_standard_info_sap - add HT and VHT info attributes
|
|
|
+ * @skb: pointer to response skb buffer
|
|
|
+ * @stainfo: station information
|
|
|
+ * @idx: attribute type index for nla_next_start()
|
|
|
+ *
|
|
|
+ * This function adds HT and VHT info attributes to response skb buffer
|
|
|
+ *
|
|
|
+ * Return : 0 on success and errno on failure
|
|
|
+ */
|
|
|
+static int hdd_add_ap_standard_info_sap(struct sk_buff *skb,
|
|
|
+ struct hdd_station_info *stainfo,
|
|
|
+ int idx)
|
|
|
+{
|
|
|
+ struct nlattr *nla_attr;
|
|
|
+
|
|
|
+ nla_attr = nla_nest_start(skb, idx);
|
|
|
+ if (!nla_attr)
|
|
|
+ goto fail;
|
|
|
+
|
|
|
+ if (stainfo->vht_present) {
|
|
|
+ if (nla_put(skb, NL80211_ATTR_VHT_CAPABILITY,
|
|
|
+ sizeof(stainfo->vht_caps),
|
|
|
+ &stainfo->vht_caps)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if (stainfo->ht_present) {
|
|
|
+ if (nla_put(skb, NL80211_ATTR_HT_CAPABILITY,
|
|
|
+ sizeof(stainfo->ht_caps),
|
|
|
+ &stainfo->ht_caps)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ nla_nest_end(skb, nla_attr);
|
|
|
+ return 0;
|
|
|
+fail:
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_decode_ch_width - decode channel band width based
|
|
|
+ * @ch_width: encoded enum value holding channel band width
|
|
|
+ *
|
|
|
+ * This function decodes channel band width from the given encoded enum value.
|
|
|
+ *
|
|
|
+ * Returns: decoded channel band width.
|
|
|
+ */
|
|
|
+static uint8_t hdd_decode_ch_width(tSirMacHTChannelWidth ch_width)
|
|
|
+{
|
|
|
+ switch (ch_width) {
|
|
|
+ case 0:
|
|
|
+ return 20;
|
|
|
+ case 1:
|
|
|
+ return 40;
|
|
|
+ case 2:
|
|
|
+ return 80;
|
|
|
+ case 3:
|
|
|
+ case 4:
|
|
|
+ return 160;
|
|
|
+ default:
|
|
|
+ hdd_debug("invalid enum: %d", ch_width);
|
|
|
+ return 20;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_get_cached_station_remote() - get cached(deleted) peer's info
|
|
|
+ * @hdd_ctx: hdd context
|
|
|
+ * @adapter: hostapd interface
|
|
|
+ * @mac_addr: mac address of requested peer
|
|
|
+ *
|
|
|
+ * This function collect and indicate the cached(deleted) peer's info
|
|
|
+ *
|
|
|
+ * Return: 0 on success, otherwise error value
|
|
|
+ */
|
|
|
+
|
|
|
+static int hdd_get_cached_station_remote(struct hdd_context *hdd_ctx,
|
|
|
+ struct hdd_adapter *adapter,
|
|
|
+ struct qdf_mac_addr mac_addr)
|
|
|
+{
|
|
|
+ struct hdd_station_info *stainfo = hdd_get_stainfo(
|
|
|
+ adapter->cache_sta_info,
|
|
|
+ mac_addr);
|
|
|
+ struct sk_buff *skb = NULL;
|
|
|
+ uint32_t nl_buf_len = NLMSG_HDRLEN;
|
|
|
+ uint8_t channel_width;
|
|
|
+
|
|
|
+ if (!stainfo) {
|
|
|
+ hdd_err("peer " MAC_ADDRESS_STR " not found",
|
|
|
+ MAC_ADDR_ARRAY(mac_addr.bytes));
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ nl_buf_len += hdd_add_link_standard_info_sap_get_len() +
|
|
|
+ hdd_add_ap_standard_info_sap_get_len(stainfo) +
|
|
|
+ (sizeof(stainfo->dot11_mode) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->ch_width) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->tx_rate) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->rx_rate) + NLA_HDRLEN);
|
|
|
+
|
|
|
+ skb = cfg80211_vendor_cmd_alloc_reply_skb(hdd_ctx->wiphy, nl_buf_len);
|
|
|
+ if (!skb) {
|
|
|
+ hdd_err("cfg80211_vendor_cmd_alloc_reply_skb failed");
|
|
|
+ return -ENOMEM;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (hdd_add_link_standard_info_sap(skb, stainfo->rssi, stainfo,
|
|
|
+ LINK_INFO_STANDARD_NL80211_ATTR)) {
|
|
|
+ hdd_err("link standard put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (hdd_add_ap_standard_info_sap(skb, stainfo,
|
|
|
+ AP_INFO_STANDARD_NL80211_ATTR)) {
|
|
|
+ hdd_err("ap standard put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* upper layer expects decoded channel BW */
|
|
|
+ channel_width = hdd_decode_ch_width(stainfo->ch_width);
|
|
|
+
|
|
|
+ if (nla_put_u32(skb, REMOTE_SUPPORTED_MODE,
|
|
|
+ hdd_convert_dot11mode(
|
|
|
+ stainfo->mode)) ||
|
|
|
+ nla_put_u8(skb, REMOTE_CH_WIDTH, channel_width)) {
|
|
|
+ hdd_err("remote ch put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ if (nla_put_u32(skb, REMOTE_LAST_TX_RATE, stainfo->tx_rate)) {
|
|
|
+ hdd_err("tx rate put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ if (nla_put_u32(skb, REMOTE_LAST_RX_RATE, stainfo->rx_rate)) {
|
|
|
+ hdd_err("rx rate put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ qdf_mem_zero(stainfo, sizeof(*stainfo));
|
|
|
+
|
|
|
+ return cfg80211_vendor_cmd_reply(skb);
|
|
|
+fail:
|
|
|
+ if (skb)
|
|
|
+ kfree_skb(skb);
|
|
|
+
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_get_cached_station_remote() - get connected peer's info
|
|
|
+ * @hdd_ctx: hdd context
|
|
|
+ * @adapter: hostapd interface
|
|
|
+ * @mac_addr: mac address of requested peer
|
|
|
+ *
|
|
|
+ * This function collect and indicate the connected peer's info
|
|
|
+ *
|
|
|
+ * Return: 0 on success, otherwise error value
|
|
|
+ */
|
|
|
+static int hdd_get_connected_station_info(struct hdd_context *hdd_ctx,
|
|
|
+ struct hdd_adapter *adapter,
|
|
|
+ struct qdf_mac_addr mac_addr,
|
|
|
+ struct hdd_station_info *stainfo)
|
|
|
+{
|
|
|
+ struct sk_buff *skb = NULL;
|
|
|
+ uint32_t nl_buf_len;
|
|
|
+ struct sir_peer_info_ext peer_info;
|
|
|
+ bool txrx_rate = true;
|
|
|
+
|
|
|
+ nl_buf_len = NLMSG_HDRLEN;
|
|
|
+ nl_buf_len += (sizeof(stainfo->max_phy_rate) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->tx_packets) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->tx_bytes) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->rx_packets) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->rx_bytes) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->is_qos_enabled) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->mode) + NLA_HDRLEN);
|
|
|
+
|
|
|
+ if (!hdd_ctx->config->sap_get_peer_info ||
|
|
|
+ wlan_hdd_get_peer_info(adapter, mac_addr, &peer_info)) {
|
|
|
+ hdd_err("fail to get tx/rx rate");
|
|
|
+ txrx_rate = false;
|
|
|
+ } else {
|
|
|
+ stainfo->tx_rate = peer_info.tx_rate;
|
|
|
+ stainfo->rx_rate = peer_info.rx_rate;
|
|
|
+ nl_buf_len += (sizeof(stainfo->tx_rate) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->rx_rate) + NLA_HDRLEN);
|
|
|
+ }
|
|
|
+
|
|
|
+ /* below info is only valid for HT/VHT mode */
|
|
|
+ if (stainfo->mode > SIR_SME_PHY_MODE_LEGACY)
|
|
|
+ nl_buf_len += (sizeof(stainfo->ampdu) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->tx_stbc) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->rx_stbc) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->ch_width) + NLA_HDRLEN) +
|
|
|
+ (sizeof(stainfo->sgi_enable) + NLA_HDRLEN);
|
|
|
+
|
|
|
+ hdd_info("buflen %d hdrlen %d", nl_buf_len, NLMSG_HDRLEN);
|
|
|
+
|
|
|
+ skb = cfg80211_vendor_cmd_alloc_reply_skb(hdd_ctx->wiphy,
|
|
|
+ nl_buf_len);
|
|
|
+ if (!skb) {
|
|
|
+ hdd_err("cfg80211_vendor_cmd_alloc_reply_skb failed");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ hdd_info("stainfo");
|
|
|
+ hdd_info("maxrate %x tx_pkts %x tx_bytes %llx",
|
|
|
+ stainfo->max_phy_rate, stainfo->tx_packets,
|
|
|
+ stainfo->tx_bytes);
|
|
|
+ hdd_info("rx_pkts %x rx_bytes %llx mode %x",
|
|
|
+ stainfo->rx_packets, stainfo->rx_bytes,
|
|
|
+ stainfo->mode);
|
|
|
+ if (stainfo->mode > SIR_SME_PHY_MODE_LEGACY) {
|
|
|
+ hdd_info("ampdu %d tx_stbc %d rx_stbc %d",
|
|
|
+ stainfo->ampdu, stainfo->tx_stbc,
|
|
|
+ stainfo->rx_stbc);
|
|
|
+ hdd_info("wmm %d chwidth %d sgi %d",
|
|
|
+ stainfo->is_qos_enabled,
|
|
|
+ stainfo->ch_width,
|
|
|
+ stainfo->sgi_enable);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (nla_put_u32(skb, REMOTE_MAX_PHY_RATE, stainfo->max_phy_rate) ||
|
|
|
+ nla_put_u32(skb, REMOTE_TX_PACKETS, stainfo->tx_packets) ||
|
|
|
+ remote_station_put_u64(skb, REMOTE_TX_BYTES, stainfo->tx_bytes) ||
|
|
|
+ nla_put_u32(skb, REMOTE_RX_PACKETS, stainfo->rx_packets) ||
|
|
|
+ remote_station_put_u64(skb, REMOTE_RX_BYTES, stainfo->rx_bytes) ||
|
|
|
+ nla_put_u8(skb, REMOTE_WMM, stainfo->is_qos_enabled) ||
|
|
|
+ nla_put_u8(skb, REMOTE_SUPPORTED_MODE, stainfo->mode)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (txrx_rate) {
|
|
|
+ if (nla_put_u32(skb, REMOTE_LAST_TX_RATE, stainfo->tx_rate) ||
|
|
|
+ nla_put_u32(skb, REMOTE_LAST_RX_RATE, stainfo->rx_rate)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ } else {
|
|
|
+ hdd_info("tx_rate %x rx_rate %x",
|
|
|
+ stainfo->tx_rate, stainfo->rx_rate);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if (stainfo->mode > SIR_SME_PHY_MODE_LEGACY) {
|
|
|
+ if (nla_put_u8(skb, REMOTE_AMPDU, stainfo->ampdu) ||
|
|
|
+ nla_put_u8(skb, REMOTE_TX_STBC, stainfo->tx_stbc) ||
|
|
|
+ nla_put_u8(skb, REMOTE_RX_STBC, stainfo->rx_stbc) ||
|
|
|
+ nla_put_u8(skb, REMOTE_CH_WIDTH, stainfo->ch_width) ||
|
|
|
+ nla_put_u8(skb, REMOTE_SGI_ENABLE, stainfo->sgi_enable)) {
|
|
|
+ hdd_err("put fail");
|
|
|
+ goto fail;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ return cfg80211_vendor_cmd_reply(skb);
|
|
|
+
|
|
|
+fail:
|
|
|
+ if (skb)
|
|
|
+ kfree_skb(skb);
|
|
|
+
|
|
|
+ return -EINVAL;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * hdd_get_station_remote() - get remote peer's info
|
|
|
+ * @hdd_ctx: hdd context
|
|
|
+ * @adapter: hostapd interface
|
|
|
+ * @mac_addr: mac address of requested peer
|
|
|
+ *
|
|
|
+ * This function collect and indicate the remote peer's info
|
|
|
+ *
|
|
|
+ * Return: 0 on success, otherwise error value
|
|
|
+ */
|
|
|
+static int hdd_get_station_remote(struct hdd_context *hdd_ctx,
|
|
|
+ struct hdd_adapter *adapter,
|
|
|
+ struct qdf_mac_addr mac_addr)
|
|
|
+{
|
|
|
+ struct hdd_station_info *stainfo = hdd_get_stainfo(adapter->sta_info,
|
|
|
+ mac_addr);
|
|
|
+ int status = 0;
|
|
|
+ bool is_associated = false;
|
|
|
+
|
|
|
+ if (!stainfo) {
|
|
|
+ status = hdd_get_cached_station_remote(hdd_ctx, adapter,
|
|
|
+ mac_addr);
|
|
|
+ return status;
|
|
|
+ }
|
|
|
+
|
|
|
+ is_associated = hdd_is_peer_associated(adapter, &mac_addr);
|
|
|
+ if (!is_associated) {
|
|
|
+ status = hdd_get_cached_station_remote(hdd_ctx, adapter,
|
|
|
+ mac_addr);
|
|
|
+ return status;
|
|
|
+ }
|
|
|
+
|
|
|
+ status = hdd_get_connected_station_info(hdd_ctx, adapter,
|
|
|
+ mac_addr, stainfo);
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * __hdd_cfg80211_get_station_cmd() - Handle get station vendor cmd
|
|
|
+ * @wiphy: corestack handler
|
|
|
+ * @wdev: wireless device
|
|
|
+ * @data: data
|
|
|
+ * @data_len: data length
|
|
|
+ *
|
|
|
+ * Handles QCA_NL80211_VENDOR_SUBCMD_GET_STATION.
|
|
|
+ * Validate cmd attributes and send the station info to upper layers.
|
|
|
+ *
|
|
|
+ * Return: Success(0) or reason code for failure
|
|
|
+ */
|
|
|
+static int
|
|
|
+__hdd_cfg80211_get_station_cmd(struct wiphy *wiphy,
|
|
|
+ struct wireless_dev *wdev,
|
|
|
+ const void *data,
|
|
|
+ int data_len)
|
|
|
+{
|
|
|
+ struct hdd_context *hdd_ctx = wiphy_priv(wiphy);
|
|
|
+ struct net_device *dev = wdev->netdev;
|
|
|
+ struct hdd_adapter *adapter = WLAN_HDD_GET_PRIV_PTR(dev);
|
|
|
+ struct nlattr *tb[QCA_WLAN_VENDOR_ATTR_GET_STATION_MAX + 1];
|
|
|
+ int32_t status;
|
|
|
+
|
|
|
+ hdd_enter_dev(dev);
|
|
|
+ if (hdd_get_conparam() == QDF_GLOBAL_FTM_MODE) {
|
|
|
+ hdd_err("Command not allowed in FTM mode");
|
|
|
+ status = -EPERM;
|
|
|
+ goto out;
|
|
|
+ }
|
|
|
+
|
|
|
+ status = wlan_hdd_validate_context(hdd_ctx);
|
|
|
+ if (status != 0)
|
|
|
+ goto out;
|
|
|
+
|
|
|
+ status = wlan_cfg80211_nla_parse(tb,
|
|
|
+ QCA_WLAN_VENDOR_ATTR_GET_STATION_MAX,
|
|
|
+ data, data_len,
|
|
|
+ hdd_get_station_policy);
|
|
|
+ if (status) {
|
|
|
+ hdd_err("Invalid ATTR");
|
|
|
+ goto out;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Parse and fetch Command Type*/
|
|
|
+ if (tb[STATION_INFO]) {
|
|
|
+ status = hdd_get_station_info(hdd_ctx, adapter);
|
|
|
+ } else if (tb[STATION_ASSOC_FAIL_REASON]) {
|
|
|
+ status = hdd_get_station_assoc_fail(hdd_ctx, adapter);
|
|
|
+ } else if (tb[STATION_REMOTE]) {
|
|
|
+ struct qdf_mac_addr mac_addr;
|
|
|
+
|
|
|
+ if (adapter->device_mode != QDF_SAP_MODE &&
|
|
|
+ adapter->device_mode != QDF_P2P_GO_MODE) {
|
|
|
+ hdd_err("invalid device_mode:%d", adapter->device_mode);
|
|
|
+ status = -EINVAL;
|
|
|
+ goto out;
|
|
|
+ }
|
|
|
+
|
|
|
+ nla_memcpy(mac_addr.bytes, tb[STATION_REMOTE],
|
|
|
+ QDF_MAC_ADDR_SIZE);
|
|
|
+
|
|
|
+ hdd_debug("STATION_REMOTE " MAC_ADDRESS_STR,
|
|
|
+ MAC_ADDR_ARRAY(mac_addr.bytes));
|
|
|
+
|
|
|
+ status = hdd_get_station_remote(hdd_ctx, adapter, mac_addr);
|
|
|
+ } else {
|
|
|
+ hdd_err("get station info cmd type failed");
|
|
|
+ status = -EINVAL;
|
|
|
+ goto out;
|
|
|
+ }
|
|
|
+ hdd_exit();
|
|
|
+out:
|
|
|
+ return status;
|
|
|
+}
|
|
|
+
|
|
|
+int32_t hdd_cfg80211_get_station_cmd(struct wiphy *wiphy,
|
|
|
+ struct wireless_dev *wdev,
|
|
|
+ const void *data,
|
|
|
+ int data_len)
|
|
|
+{
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ cds_ssr_protect(__func__);
|
|
|
+ ret = __hdd_cfg80211_get_station_cmd(wiphy, wdev, data, data_len);
|
|
|
+ cds_ssr_unprotect(__func__);
|
|
|
+
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+
|