|
@@ -39,6 +39,7 @@
|
|
#include "sta.h"
|
|
#include "sta.h"
|
|
#include "iwl-op-mode.h"
|
|
#include "iwl-op-mode.h"
|
|
#include "mvm.h"
|
|
#include "mvm.h"
|
|
|
|
+#include "debugfs.h"
|
|
|
|
|
|
#define RS_NAME "iwl-mvm-rs"
|
|
#define RS_NAME "iwl-mvm-rs"
|
|
|
|
|
|
@@ -3057,6 +3058,20 @@ static void rs_set_lq_ss_params(struct iwl_mvm *mvm,
|
|
if (!iwl_mvm_bt_coex_is_mimo_allowed(mvm, sta))
|
|
if (!iwl_mvm_bt_coex_is_mimo_allowed(mvm, sta))
|
|
goto out;
|
|
goto out;
|
|
|
|
|
|
|
|
+ /* Check if forcing the decision is configured.
|
|
|
|
+ * Note that SISO is forced by not allowing STBC or BFER
|
|
|
|
+ */
|
|
|
|
+ if (lq_sta->ss_force == RS_SS_FORCE_STBC)
|
|
|
|
+ ss_params |= (LQ_SS_STBC_1SS_ALLOWED | LQ_SS_FORCE);
|
|
|
|
+ else if (lq_sta->ss_force == RS_SS_FORCE_BFER)
|
|
|
|
+ ss_params |= (LQ_SS_BFER_ALLOWED | LQ_SS_FORCE);
|
|
|
|
+
|
|
|
|
+ if (lq_sta->ss_force != RS_SS_FORCE_NONE) {
|
|
|
|
+ IWL_DEBUG_RATE(mvm, "Forcing single stream Tx decision %d\n",
|
|
|
|
+ lq_sta->ss_force);
|
|
|
|
+ goto out;
|
|
|
|
+ }
|
|
|
|
+
|
|
if (lq_sta->stbc_capable)
|
|
if (lq_sta->stbc_capable)
|
|
ss_params |= LQ_SS_STBC_1SS_ALLOWED;
|
|
ss_params |= LQ_SS_STBC_1SS_ALLOWED;
|
|
|
|
|
|
@@ -3502,9 +3517,73 @@ static const struct file_operations rs_sta_dbgfs_drv_tx_stats_ops = {
|
|
.llseek = default_llseek,
|
|
.llseek = default_llseek,
|
|
};
|
|
};
|
|
|
|
|
|
|
|
+static ssize_t iwl_dbgfs_ss_force_read(struct file *file,
|
|
|
|
+ char __user *user_buf,
|
|
|
|
+ size_t count, loff_t *ppos)
|
|
|
|
+{
|
|
|
|
+ struct iwl_lq_sta *lq_sta = file->private_data;
|
|
|
|
+ char buf[12];
|
|
|
|
+ int bufsz = sizeof(buf);
|
|
|
|
+ int pos = 0;
|
|
|
|
+ static const char * const ss_force_name[] = {
|
|
|
|
+ [RS_SS_FORCE_NONE] = "none",
|
|
|
|
+ [RS_SS_FORCE_STBC] = "stbc",
|
|
|
|
+ [RS_SS_FORCE_BFER] = "bfer",
|
|
|
|
+ [RS_SS_FORCE_SISO] = "siso",
|
|
|
|
+ };
|
|
|
|
+
|
|
|
|
+ pos += scnprintf(buf+pos, bufsz-pos, "%s\n",
|
|
|
|
+ ss_force_name[lq_sta->ss_force]);
|
|
|
|
+ return simple_read_from_buffer(user_buf, count, ppos, buf, pos);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static ssize_t iwl_dbgfs_ss_force_write(struct iwl_lq_sta *lq_sta, char *buf,
|
|
|
|
+ size_t count, loff_t *ppos)
|
|
|
|
+{
|
|
|
|
+ struct iwl_mvm *mvm = lq_sta->pers.drv;
|
|
|
|
+ int ret = 0;
|
|
|
|
+
|
|
|
|
+ if (!strncmp("none", buf, 4)) {
|
|
|
|
+ lq_sta->ss_force = RS_SS_FORCE_NONE;
|
|
|
|
+ } else if (!strncmp("siso", buf, 4)) {
|
|
|
|
+ lq_sta->ss_force = RS_SS_FORCE_SISO;
|
|
|
|
+ } else if (!strncmp("stbc", buf, 4)) {
|
|
|
|
+ if (lq_sta->stbc_capable) {
|
|
|
|
+ lq_sta->ss_force = RS_SS_FORCE_STBC;
|
|
|
|
+ } else {
|
|
|
|
+ IWL_ERR(mvm,
|
|
|
|
+ "can't force STBC. peer doesn't support\n");
|
|
|
|
+ ret = -EINVAL;
|
|
|
|
+ }
|
|
|
|
+ } else if (!strncmp("bfer", buf, 4)) {
|
|
|
|
+ if (lq_sta->bfer_capable) {
|
|
|
|
+ lq_sta->ss_force = RS_SS_FORCE_BFER;
|
|
|
|
+ } else {
|
|
|
|
+ IWL_ERR(mvm,
|
|
|
|
+ "can't force BFER. peer doesn't support\n");
|
|
|
|
+ ret = -EINVAL;
|
|
|
|
+ }
|
|
|
|
+ } else {
|
|
|
|
+ IWL_ERR(mvm, "valid values none|siso|stbc|bfer\n");
|
|
|
|
+ ret = -EINVAL;
|
|
|
|
+ }
|
|
|
|
+ return ret ?: count;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+#define MVM_DEBUGFS_READ_WRITE_FILE_OPS(name, bufsz) \
|
|
|
|
+ _MVM_DEBUGFS_READ_WRITE_FILE_OPS(name, bufsz, struct iwl_lq_sta)
|
|
|
|
+#define MVM_DEBUGFS_ADD_FILE_RS(name, parent, mode) do { \
|
|
|
|
+ if (!debugfs_create_file(#name, mode, parent, lq_sta, \
|
|
|
|
+ &iwl_dbgfs_##name##_ops)) \
|
|
|
|
+ goto err; \
|
|
|
|
+ } while (0)
|
|
|
|
+
|
|
|
|
+MVM_DEBUGFS_READ_WRITE_FILE_OPS(ss_force, 32);
|
|
|
|
+
|
|
static void rs_add_debugfs(void *mvm, void *mvm_sta, struct dentry *dir)
|
|
static void rs_add_debugfs(void *mvm, void *mvm_sta, struct dentry *dir)
|
|
{
|
|
{
|
|
struct iwl_lq_sta *lq_sta = mvm_sta;
|
|
struct iwl_lq_sta *lq_sta = mvm_sta;
|
|
|
|
+
|
|
debugfs_create_file("rate_scale_table", S_IRUSR | S_IWUSR, dir,
|
|
debugfs_create_file("rate_scale_table", S_IRUSR | S_IWUSR, dir,
|
|
lq_sta, &rs_sta_dbgfs_scale_table_ops);
|
|
lq_sta, &rs_sta_dbgfs_scale_table_ops);
|
|
debugfs_create_file("rate_stats_table", S_IRUSR, dir,
|
|
debugfs_create_file("rate_stats_table", S_IRUSR, dir,
|
|
@@ -3515,6 +3594,11 @@ static void rs_add_debugfs(void *mvm, void *mvm_sta, struct dentry *dir)
|
|
&lq_sta->tx_agg_tid_en);
|
|
&lq_sta->tx_agg_tid_en);
|
|
debugfs_create_u8("reduced_tpc", S_IRUSR | S_IWUSR, dir,
|
|
debugfs_create_u8("reduced_tpc", S_IRUSR | S_IWUSR, dir,
|
|
&lq_sta->pers.dbg_fixed_txp_reduction);
|
|
&lq_sta->pers.dbg_fixed_txp_reduction);
|
|
|
|
+
|
|
|
|
+ MVM_DEBUGFS_ADD_FILE_RS(ss_force, dir, S_IRUSR | S_IWUSR);
|
|
|
|
+ return;
|
|
|
|
+err:
|
|
|
|
+ IWL_ERR((struct iwl_mvm *)mvm, "Can't create debugfs entity\n");
|
|
}
|
|
}
|
|
|
|
|
|
static void rs_remove_debugfs(void *mvm, void *mvm_sta)
|
|
static void rs_remove_debugfs(void *mvm, void *mvm_sta)
|