Commit 4f34dace authored by Thomas Pedersen's avatar Thomas Pedersen Committed by Kalle Valo

ath6kl: send TCMD response through testmode events

ath6kl no longer knows what it is transmitting through
cfg80211_testmode, and simply passes opaque buffers between userspace
and the firmware. Leave the CONT_RX enum for backwards compatibility.

kvalo: change ATH6KL_TM_CMD_RX_REPORT to return -EOPNOTSUPP
Signed-off-by: default avatarThomas Pedersen <twpedersen@qca.qualcomm.com>
Signed-off-by: default avatarKalle Valo <kvalo@qca.qualcomm.com>
parent 792ecb33
...@@ -15,6 +15,7 @@ ...@@ -15,6 +15,7 @@
*/ */
#include "testmode.h" #include "testmode.h"
#include "debug.h"
#include <net/netlink.h> #include <net/netlink.h>
...@@ -30,7 +31,7 @@ enum ath6kl_tm_attr { ...@@ -30,7 +31,7 @@ enum ath6kl_tm_attr {
enum ath6kl_tm_cmd { enum ath6kl_tm_cmd {
ATH6KL_TM_CMD_TCMD = 0, ATH6KL_TM_CMD_TCMD = 0,
ATH6KL_TM_CMD_RX_REPORT = 1, ATH6KL_TM_CMD_RX_REPORT = 1, /* not used anymore */
}; };
#define ATH6KL_TM_DATA_MAX_LEN 5000 #define ATH6KL_TM_DATA_MAX_LEN 5000
...@@ -41,84 +42,33 @@ static const struct nla_policy ath6kl_tm_policy[ATH6KL_TM_ATTR_MAX + 1] = { ...@@ -41,84 +42,33 @@ static const struct nla_policy ath6kl_tm_policy[ATH6KL_TM_ATTR_MAX + 1] = {
.len = ATH6KL_TM_DATA_MAX_LEN }, .len = ATH6KL_TM_DATA_MAX_LEN },
}; };
void ath6kl_tm_rx_report_event(struct ath6kl *ar, void *buf, size_t buf_len) void ath6kl_tm_rx_event(struct ath6kl *ar, void *buf, size_t buf_len)
{ {
if (down_interruptible(&ar->sem)) struct sk_buff *skb;
return;
kfree(ar->tm.rx_report);
ar->tm.rx_report = kmemdup(buf, buf_len, GFP_KERNEL);
ar->tm.rx_report_len = buf_len;
up(&ar->sem);
wake_up(&ar->event_wq);
}
static int ath6kl_tm_rx_report(struct ath6kl *ar, void *buf, size_t buf_len,
struct sk_buff *skb)
{
int ret = 0;
long left;
if (down_interruptible(&ar->sem))
return -ERESTARTSYS;
if (!test_bit(WMI_READY, &ar->flag)) {
ret = -EIO;
goto out;
}
if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
ret = -EBUSY;
goto out;
}
if (ath6kl_wmi_test_cmd(ar->wmi, buf, buf_len) < 0) {
up(&ar->sem);
return -EIO;
}
left = wait_event_interruptible_timeout(ar->event_wq,
ar->tm.rx_report != NULL,
WMI_TIMEOUT);
if (left == 0) { if (!buf || buf_len == 0)
ret = -ETIMEDOUT; return;
goto out;
} else if (left < 0) {
ret = left;
goto out;
}
if (ar->tm.rx_report == NULL || ar->tm.rx_report_len == 0) { skb = cfg80211_testmode_alloc_event_skb(ar->wiphy, buf_len, GFP_KERNEL);
ret = -EINVAL; if (!skb) {
goto out; ath6kl_warn("failed to allocate testmode rx skb!\n");
return;
} }
NLA_PUT_U32(skb, ATH6KL_TM_ATTR_CMD, ATH6KL_TM_CMD_TCMD);
NLA_PUT(skb, ATH6KL_TM_ATTR_DATA, ar->tm.rx_report_len, NLA_PUT(skb, ATH6KL_TM_ATTR_DATA, buf_len, buf);
ar->tm.rx_report); cfg80211_testmode_event(skb, GFP_KERNEL);
return;
kfree(ar->tm.rx_report);
ar->tm.rx_report = NULL;
out:
up(&ar->sem);
return ret;
nla_put_failure: nla_put_failure:
ret = -ENOBUFS; kfree_skb(skb);
goto out; ath6kl_warn("nla_put failed on testmode rx skb!\n");
} }
int ath6kl_tm_cmd(struct wiphy *wiphy, void *data, int len) int ath6kl_tm_cmd(struct wiphy *wiphy, void *data, int len)
{ {
struct ath6kl *ar = wiphy_priv(wiphy); struct ath6kl *ar = wiphy_priv(wiphy);
struct nlattr *tb[ATH6KL_TM_ATTR_MAX + 1]; struct nlattr *tb[ATH6KL_TM_ATTR_MAX + 1];
int err, buf_len, reply_len; int err, buf_len;
struct sk_buff *skb;
void *buf; void *buf;
err = nla_parse(tb, ATH6KL_TM_ATTR_MAX, data, len, err = nla_parse(tb, ATH6KL_TM_ATTR_MAX, data, len,
...@@ -143,24 +93,6 @@ int ath6kl_tm_cmd(struct wiphy *wiphy, void *data, int len) ...@@ -143,24 +93,6 @@ int ath6kl_tm_cmd(struct wiphy *wiphy, void *data, int len)
break; break;
case ATH6KL_TM_CMD_RX_REPORT: case ATH6KL_TM_CMD_RX_REPORT:
if (!tb[ATH6KL_TM_ATTR_DATA])
return -EINVAL;
buf = nla_data(tb[ATH6KL_TM_ATTR_DATA]);
buf_len = nla_len(tb[ATH6KL_TM_ATTR_DATA]);
reply_len = nla_total_size(ATH6KL_TM_DATA_MAX_LEN);
skb = cfg80211_testmode_alloc_reply_skb(wiphy, reply_len);
if (!skb)
return -ENOMEM;
err = ath6kl_tm_rx_report(ar, buf, buf_len, skb);
if (err < 0) {
kfree_skb(skb);
return err;
}
return cfg80211_testmode_reply(skb);
default: default:
return -EOPNOTSUPP; return -EOPNOTSUPP;
} }
......
...@@ -18,13 +18,13 @@ ...@@ -18,13 +18,13 @@
#ifdef CONFIG_NL80211_TESTMODE #ifdef CONFIG_NL80211_TESTMODE
void ath6kl_tm_rx_report_event(struct ath6kl *ar, void *buf, size_t buf_len); void ath6kl_tm_rx_event(struct ath6kl *ar, void *buf, size_t buf_len);
int ath6kl_tm_cmd(struct wiphy *wiphy, void *data, int len); int ath6kl_tm_cmd(struct wiphy *wiphy, void *data, int len);
#else #else
static inline void ath6kl_tm_rx_report_event(struct ath6kl *ar, void *buf, static inline void ath6kl_tm_rx_event(struct ath6kl *ar, void *buf,
size_t buf_len) size_t buf_len)
{ {
} }
......
...@@ -1145,9 +1145,9 @@ static int ath6kl_wmi_bitrate_reply_rx(struct wmi *wmi, u8 *datap, int len) ...@@ -1145,9 +1145,9 @@ static int ath6kl_wmi_bitrate_reply_rx(struct wmi *wmi, u8 *datap, int len)
return 0; return 0;
} }
static int ath6kl_wmi_tcmd_test_report_rx(struct wmi *wmi, u8 *datap, int len) static int ath6kl_wmi_test_rx(struct wmi *wmi, u8 *datap, int len)
{ {
ath6kl_tm_rx_report_event(wmi->parent_dev, datap, len); ath6kl_tm_rx_event(wmi->parent_dev, datap, len);
return 0; return 0;
} }
...@@ -3402,7 +3402,7 @@ int ath6kl_wmi_control_rx(struct wmi *wmi, struct sk_buff *skb) ...@@ -3402,7 +3402,7 @@ int ath6kl_wmi_control_rx(struct wmi *wmi, struct sk_buff *skb)
break; break;
case WMI_TEST_EVENTID: case WMI_TEST_EVENTID:
ath6kl_dbg(ATH6KL_DBG_WMI, "WMI_TEST_EVENTID\n"); ath6kl_dbg(ATH6KL_DBG_WMI, "WMI_TEST_EVENTID\n");
ret = ath6kl_wmi_tcmd_test_report_rx(wmi, datap, len); ret = ath6kl_wmi_test_rx(wmi, datap, len);
break; break;
case WMI_GET_FIXRATES_CMDID: case WMI_GET_FIXRATES_CMDID:
ath6kl_dbg(ATH6KL_DBG_WMI, "WMI_GET_FIXRATES_CMDID\n"); ath6kl_dbg(ATH6KL_DBG_WMI, "WMI_GET_FIXRATES_CMDID\n");
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment