提交 26b5858a 编写于 作者: L Luciano Coelho

wlcore: add a debugfs entry to allow changing the sleep mode by hand

For FW debugging purposes, we may need to change the sleep mode
(aka. sleep_auth) by hand, and set it to the mode we want.  To allow
this, a debugfs entry is added.

Now we store the sleep_auth value that has been set and use that
instead of the quirk to decide whether we should enter ELP or not.
Signed-off-by: NLuciano Coelho <coelho@ti.com>
Signed-off-by: NArik Nemtsov <arik@wizery.com>
上级 c954910b
...@@ -81,7 +81,10 @@ int wl1271_acx_sleep_auth(struct wl1271 *wl, u8 sleep_auth) ...@@ -81,7 +81,10 @@ int wl1271_acx_sleep_auth(struct wl1271 *wl, u8 sleep_auth)
auth->sleep_auth = sleep_auth; auth->sleep_auth = sleep_auth;
ret = wl1271_cmd_configure(wl, ACX_SLEEP_AUTH, auth, sizeof(*auth)); ret = wl1271_cmd_configure(wl, ACX_SLEEP_AUTH, auth, sizeof(*auth));
if (ret < 0)
goto out;
wl->sleep_auth = sleep_auth;
out: out:
kfree(auth); kfree(auth);
return ret; return ret;
......
...@@ -118,6 +118,8 @@ enum wl1271_psm_mode { ...@@ -118,6 +118,8 @@ enum wl1271_psm_mode {
/* Extreme low power */ /* Extreme low power */
WL1271_PSM_ELP = 2, WL1271_PSM_ELP = 2,
WL1271_PSM_MAX = WL1271_PSM_ELP,
}; };
struct acx_sleep_auth { struct acx_sleep_auth {
......
...@@ -963,6 +963,63 @@ static const struct file_operations fw_stats_raw_ops = { ...@@ -963,6 +963,63 @@ static const struct file_operations fw_stats_raw_ops = {
.llseek = default_llseek, .llseek = default_llseek,
}; };
static ssize_t sleep_auth_read(struct file *file, char __user *user_buf,
size_t count, loff_t *ppos)
{
struct wl1271 *wl = file->private_data;
return wl1271_format_buffer(user_buf, count,
ppos, "%d\n",
wl->sleep_auth);
}
static ssize_t sleep_auth_write(struct file *file,
const char __user *user_buf,
size_t count, loff_t *ppos)
{
struct wl1271 *wl = file->private_data;
unsigned long value;
int ret;
ret = kstrtoul_from_user(user_buf, count, 0, &value);
if (ret < 0) {
wl1271_warning("illegal value in sleep_auth");
return -EINVAL;
}
if (value < 0 || value > WL1271_PSM_MAX) {
wl1271_warning("sleep_auth must be between 0 and %d",
WL1271_PSM_MAX);
return -ERANGE;
}
mutex_lock(&wl->mutex);
if (wl->state == WL1271_STATE_OFF)
goto out;
ret = wl1271_ps_elp_wakeup(wl);
if (ret < 0)
goto out;
ret = wl1271_acx_sleep_auth(wl, value);
if (ret < 0)
goto out_sleep;
out_sleep:
wl1271_ps_elp_sleep(wl);
out:
mutex_unlock(&wl->mutex);
return count;
}
static const struct file_operations sleep_auth_ops = {
.read = sleep_auth_read,
.write = sleep_auth_write,
.open = simple_open,
.llseek = default_llseek,
};
static int wl1271_debugfs_add_files(struct wl1271 *wl, static int wl1271_debugfs_add_files(struct wl1271 *wl,
struct dentry *rootdir) struct dentry *rootdir)
{ {
...@@ -988,6 +1045,7 @@ static int wl1271_debugfs_add_files(struct wl1271 *wl, ...@@ -988,6 +1045,7 @@ static int wl1271_debugfs_add_files(struct wl1271 *wl,
DEBUGFS_ADD(irq_blk_threshold, rootdir); DEBUGFS_ADD(irq_blk_threshold, rootdir);
DEBUGFS_ADD(irq_timeout, rootdir); DEBUGFS_ADD(irq_timeout, rootdir);
DEBUGFS_ADD(fw_stats_raw, rootdir); DEBUGFS_ADD(fw_stats_raw, rootdir);
DEBUGFS_ADD(sleep_auth, rootdir);
streaming = debugfs_create_dir("rx_streaming", rootdir); streaming = debugfs_create_dir("rx_streaming", rootdir);
if (!streaming || IS_ERR(streaming)) if (!streaming || IS_ERR(streaming))
......
...@@ -1082,6 +1082,7 @@ int wl1271_plt_stop(struct wl1271 *wl) ...@@ -1082,6 +1082,7 @@ int wl1271_plt_stop(struct wl1271 *wl)
mutex_lock(&wl->mutex); mutex_lock(&wl->mutex);
wl1271_power_off(wl); wl1271_power_off(wl);
wl->flags = 0; wl->flags = 0;
wl->sleep_auth = WL1271_PSM_CAM;
wl->state = WL1271_STATE_OFF; wl->state = WL1271_STATE_OFF;
wl->plt = false; wl->plt = false;
wl->rx_counter = 0; wl->rx_counter = 0;
...@@ -1740,6 +1741,7 @@ static void wl1271_op_stop(struct ieee80211_hw *hw) ...@@ -1740,6 +1741,7 @@ static void wl1271_op_stop(struct ieee80211_hw *hw)
wl->ap_fw_ps_map = 0; wl->ap_fw_ps_map = 0;
wl->ap_ps_map = 0; wl->ap_ps_map = 0;
wl->sched_scanning = false; wl->sched_scanning = false;
wl->sleep_auth = WL1271_PSM_CAM;
memset(wl->roles_map, 0, sizeof(wl->roles_map)); memset(wl->roles_map, 0, sizeof(wl->roles_map));
memset(wl->links_map, 0, sizeof(wl->links_map)); memset(wl->links_map, 0, sizeof(wl->links_map));
memset(wl->roc_map, 0, sizeof(wl->roc_map)); memset(wl->roc_map, 0, sizeof(wl->roc_map));
...@@ -5174,6 +5176,7 @@ struct ieee80211_hw *wlcore_alloc_hw(size_t priv_size) ...@@ -5174,6 +5176,7 @@ struct ieee80211_hw *wlcore_alloc_hw(size_t priv_size)
wl->channel_type = NL80211_CHAN_NO_HT; wl->channel_type = NL80211_CHAN_NO_HT;
wl->flags = 0; wl->flags = 0;
wl->sg_enabled = true; wl->sg_enabled = true;
wl->sleep_auth = WL1271_PSM_CAM;
wl->hw_pg_ver = -1; wl->hw_pg_ver = -1;
wl->ap_ps_map = 0; wl->ap_ps_map = 0;
wl->ap_fw_ps_map = 0; wl->ap_fw_ps_map = 0;
......
...@@ -76,7 +76,7 @@ void wl1271_ps_elp_sleep(struct wl1271 *wl) ...@@ -76,7 +76,7 @@ void wl1271_ps_elp_sleep(struct wl1271 *wl)
struct wl12xx_vif *wlvif; struct wl12xx_vif *wlvif;
u32 timeout; u32 timeout;
if (wl->quirks & WLCORE_QUIRK_NO_ELP) if (wl->sleep_auth != WL1271_PSM_ELP)
return; return;
/* we shouldn't get consecutive sleep requests */ /* we shouldn't get consecutive sleep requests */
......
...@@ -387,6 +387,9 @@ struct wl1271 { ...@@ -387,6 +387,9 @@ struct wl1271 {
/* mutex for protecting the tx_flush function */ /* mutex for protecting the tx_flush function */
struct mutex flush_mutex; struct mutex flush_mutex;
/* sleep auth value currently configured to FW */
int sleep_auth;
}; };
int __devinit wlcore_probe(struct wl1271 *wl, struct platform_device *pdev); int __devinit wlcore_probe(struct wl1271 *wl, struct platform_device *pdev);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册