The branch main has been updated by bz:

URL: 
https://cgit.FreeBSD.org/src/commit/?id=eb15fdb1b72de02e7a4c454f7eeeb1cee5cb83df

commit eb15fdb1b72de02e7a4c454f7eeeb1cee5cb83df
Merge: 6b627f88584c c1d365f39e08
Author:     Bjoern A. Zeeb <[email protected]>
AuthorDate: 2025-10-17 20:22:54 +0000
Commit:     Bjoern A. Zeeb <[email protected]>
CommitDate: 2025-10-23 21:26:02 +0000

    rtw88: update Realtek's rtw88 driver
    
    This version is based on
    git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
    e5f0a698b34ed76002dc5cff3804a61c80233a7a ( tag: v6.17 ).
    
    MFC after:      3 days

 sys/contrib/dev/rtw88/Makefile         |     9 +
 sys/contrib/dev/rtw88/coex.c           |    24 +-
 sys/contrib/dev/rtw88/debug.c          |    57 +-
 sys/contrib/dev/rtw88/fw.c             |    31 +-
 sys/contrib/dev/rtw88/fw.h             |     1 +
 sys/contrib/dev/rtw88/hci.h            |     8 +
 sys/contrib/dev/rtw88/mac.c            |    50 +-
 sys/contrib/dev/rtw88/mac.h            |     3 +
 sys/contrib/dev/rtw88/mac80211.c       |    11 +-
 sys/contrib/dev/rtw88/main.c           |   118 +-
 sys/contrib/dev/rtw88/main.h           |    65 +-
 sys/contrib/dev/rtw88/pci.c            |    55 +-
 sys/contrib/dev/rtw88/pci.h            |     1 +
 sys/contrib/dev/rtw88/phy.c            |   215 +-
 sys/contrib/dev/rtw88/phy.h            |    20 +-
 sys/contrib/dev/rtw88/reg.h            |    69 +-
 sys/contrib/dev/rtw88/rtw8703b.c       |    64 +-
 sys/contrib/dev/rtw88/rtw8723cs.c      |     2 +-
 sys/contrib/dev/rtw88/rtw8723d.c       |     8 +-
 sys/contrib/dev/rtw88/rtw8723de.c      |     3 +-
 sys/contrib/dev/rtw88/rtw8723ds.c      |     2 +-
 sys/contrib/dev/rtw88/rtw8723du.c      |     2 +-
 sys/contrib/dev/rtw88/rtw8723x.c       |    68 +-
 sys/contrib/dev/rtw88/rtw8723x.h       |     6 +
 sys/contrib/dev/rtw88/rtw8812a.c       |     5 +-
 sys/contrib/dev/rtw88/rtw8812au.c      |     2 +-
 sys/contrib/dev/rtw88/rtw8814a.c       |  2281 +++
 sys/contrib/dev/rtw88/rtw8814a.h       |    62 +
 sys/contrib/dev/rtw88/rtw8814a_table.c | 23930 +++++++++++++++++++++++++++++++
 sys/contrib/dev/rtw88/rtw8814a_table.h |    40 +
 sys/contrib/dev/rtw88/rtw8814ae.c      |    34 +
 sys/contrib/dev/rtw88/rtw8814au.c      |    54 +
 sys/contrib/dev/rtw88/rtw8821a.c       |     5 +-
 sys/contrib/dev/rtw88/rtw8821au.c      |     2 +-
 sys/contrib/dev/rtw88/rtw8821c.c       |    21 +-
 sys/contrib/dev/rtw88/rtw8821ce.c      |     3 +-
 sys/contrib/dev/rtw88/rtw8821cs.c      |     2 +-
 sys/contrib/dev/rtw88/rtw8821cu.c      |     2 +-
 sys/contrib/dev/rtw88/rtw8822b.c       |    22 +-
 sys/contrib/dev/rtw88/rtw8822be.c      |     3 +-
 sys/contrib/dev/rtw88/rtw8822bs.c      |     2 +-
 sys/contrib/dev/rtw88/rtw8822bu.c      |     8 +-
 sys/contrib/dev/rtw88/rtw8822c.c       |    11 +-
 sys/contrib/dev/rtw88/rtw8822ce.c      |     3 +-
 sys/contrib/dev/rtw88/rtw8822cs.c      |     2 +-
 sys/contrib/dev/rtw88/rtw8822cu.c      |     2 +-
 sys/contrib/dev/rtw88/rtw88xxa.c       |     2 +-
 sys/contrib/dev/rtw88/rx.c             |     6 +
 sys/contrib/dev/rtw88/sar.c            |     2 +-
 sys/contrib/dev/rtw88/sdio.c           |    37 +-
 sys/contrib/dev/rtw88/tx.c             |     3 +-
 sys/contrib/dev/rtw88/usb.c            |    65 +-
 sys/contrib/dev/rtw88/util.c           |     3 +-
 sys/modules/rtw88/Makefile             |     5 +-
 54 files changed, 27180 insertions(+), 331 deletions(-)

diff --cc sys/contrib/dev/rtw88/Makefile
index 0cbbb366e393,000000000000..0b3da05a2938
mode 100644,000000..100644
--- a/sys/contrib/dev/rtw88/Makefile
+++ b/sys/contrib/dev/rtw88/Makefile
@@@ -1,104 -1,0 +1,113 @@@
 +# SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
 +
 +obj-$(CONFIG_RTW88_CORE)      += rtw88_core.o
 +rtw88_core-y += main.o \
 +         mac80211.o \
 +         util.o \
 +         debug.o \
 +         tx.o \
 +         rx.o \
 +         mac.o \
 +         phy.o \
 +         coex.o \
 +         efuse.o \
 +         fw.o \
 +         ps.o \
 +         sec.o \
 +         bf.o \
 +         sar.o \
 +         regd.o
 +
 +rtw88_core-$(CONFIG_PM) += wow.o
 +
 +rtw88_core-$(CONFIG_RTW88_LEDS) += led.o
 +
 +obj-$(CONFIG_RTW88_8822B)     += rtw88_8822b.o
 +rtw88_8822b-objs              := rtw8822b.o rtw8822b_table.o
 +
 +obj-$(CONFIG_RTW88_8822BE)    += rtw88_8822be.o
 +rtw88_8822be-objs             := rtw8822be.o
 +
 +obj-$(CONFIG_RTW88_8822BS)    += rtw88_8822bs.o
 +rtw88_8822bs-objs             := rtw8822bs.o
 +
 +obj-$(CONFIG_RTW88_8822BU)    += rtw88_8822bu.o
 +rtw88_8822bu-objs             := rtw8822bu.o
 +
 +obj-$(CONFIG_RTW88_8822C)     += rtw88_8822c.o
 +rtw88_8822c-objs              := rtw8822c.o rtw8822c_table.o
 +
 +obj-$(CONFIG_RTW88_8822CE)    += rtw88_8822ce.o
 +rtw88_8822ce-objs             := rtw8822ce.o
 +
 +obj-$(CONFIG_RTW88_8822CS)    += rtw88_8822cs.o
 +rtw88_8822cs-objs             := rtw8822cs.o
 +
 +obj-$(CONFIG_RTW88_8822CU)    += rtw88_8822cu.o
 +rtw88_8822cu-objs             := rtw8822cu.o
 +
 +obj-$(CONFIG_RTW88_8723X)     += rtw88_8723x.o
 +rtw88_8723x-objs              := rtw8723x.o
 +
 +obj-$(CONFIG_RTW88_8703B)     += rtw88_8703b.o
 +rtw88_8703b-objs              := rtw8703b.o rtw8703b_tables.o
 +
 +obj-$(CONFIG_RTW88_8723CS)    += rtw88_8723cs.o
 +rtw88_8723cs-objs             := rtw8723cs.o
 +
 +obj-$(CONFIG_RTW88_8723D)     += rtw88_8723d.o
 +rtw88_8723d-objs              := rtw8723d.o rtw8723d_table.o
 +
 +obj-$(CONFIG_RTW88_8723DE)    += rtw88_8723de.o
 +rtw88_8723de-objs             := rtw8723de.o
 +
 +obj-$(CONFIG_RTW88_8723DS)    += rtw88_8723ds.o
 +rtw88_8723ds-objs             := rtw8723ds.o
 +
 +obj-$(CONFIG_RTW88_8723DU)    += rtw88_8723du.o
 +rtw88_8723du-objs             := rtw8723du.o
 +
 +obj-$(CONFIG_RTW88_8821C)     += rtw88_8821c.o
 +rtw88_8821c-objs              := rtw8821c.o rtw8821c_table.o
 +
 +obj-$(CONFIG_RTW88_8821CE)    += rtw88_8821ce.o
 +rtw88_8821ce-objs             := rtw8821ce.o
 +
 +obj-$(CONFIG_RTW88_8821CS)    += rtw88_8821cs.o
 +rtw88_8821cs-objs             := rtw8821cs.o
 +
 +obj-$(CONFIG_RTW88_8821CU)    += rtw88_8821cu.o
 +rtw88_8821cu-objs             := rtw8821cu.o
 +
 +obj-$(CONFIG_RTW88_88XXA)     += rtw88_88xxa.o
 +rtw88_88xxa-objs              := rtw88xxa.o
 +
 +obj-$(CONFIG_RTW88_8821A)     += rtw88_8821a.o
 +rtw88_8821a-objs              := rtw8821a.o rtw8821a_table.o
 +
 +obj-$(CONFIG_RTW88_8812A)     += rtw88_8812a.o
 +rtw88_8812a-objs              := rtw8812a.o rtw8812a_table.o
 +
 +obj-$(CONFIG_RTW88_8821AU)    += rtw88_8821au.o
 +rtw88_8821au-objs             := rtw8821au.o
 +
 +obj-$(CONFIG_RTW88_8812AU)    += rtw88_8812au.o
 +rtw88_8812au-objs             := rtw8812au.o
 +
++obj-$(CONFIG_RTW88_8814A)     += rtw88_8814a.o
++rtw88_8814a-objs              := rtw8814a.o rtw8814a_table.o
++
++obj-$(CONFIG_RTW88_8814AE)    += rtw88_8814ae.o
++rtw88_8814ae-objs             := rtw8814ae.o
++
++obj-$(CONFIG_RTW88_8814AU)    += rtw88_8814au.o
++rtw88_8814au-objs             := rtw8814au.o
++
 +obj-$(CONFIG_RTW88_PCI)               += rtw88_pci.o
 +rtw88_pci-objs                        := pci.o
 +
 +obj-$(CONFIG_RTW88_SDIO)      += rtw88_sdio.o
 +rtw88_sdio-objs                       := sdio.o
 +
 +obj-$(CONFIG_RTW88_USB)               += rtw88_usb.o
 +rtw88_usb-objs                        := usb.o
diff --cc sys/contrib/dev/rtw88/debug.c
index f0ee8e62da3b,000000000000..53742a3220f2
mode 100644,000000..100644
--- a/sys/contrib/dev/rtw88/debug.c
+++ b/sys/contrib/dev/rtw88/debug.c
@@@ -1,1362 -1,0 +1,1383 @@@
 +// SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
 +/* Copyright(c) 2018-2019  Realtek Corporation
 + */
 +
 +#include <linux/debugfs.h>
 +#include <linux/seq_file.h>
 +#include "main.h"
 +#include "coex.h"
 +#include "sec.h"
 +#include "fw.h"
 +#include "debug.h"
 +#include "phy.h"
 +#include "reg.h"
 +#include "ps.h"
 +#include "regd.h"
 +
 +#ifdef CONFIG_RTW88_DEBUGFS
 +
 +struct rtw_debugfs_priv {
 +      struct rtw_dev *rtwdev;
 +      int (*cb_read)(struct seq_file *m, void *v);
 +      ssize_t (*cb_write)(struct file *filp, const char __user *buffer,
 +                          size_t count, loff_t *loff);
 +      union {
 +              u32 cb_data;
 +              u8 *buf;
 +              struct {
 +                      u32 page_offset;
 +                      u32 page_num;
 +              } rsvd_page;
 +              struct {
 +                      u8 rf_path;
 +                      u32 rf_addr;
 +                      u32 rf_mask;
 +              };
 +              struct {
 +                      u32 addr;
 +                      u32 len;
 +              } read_reg;
 +              struct {
 +                      u8 bit;
 +              } dm_cap;
 +      };
 +};
 +
 +struct rtw_debugfs {
 +      struct rtw_debugfs_priv mac_0;
 +      struct rtw_debugfs_priv mac_1;
 +      struct rtw_debugfs_priv mac_2;
 +      struct rtw_debugfs_priv mac_3;
 +      struct rtw_debugfs_priv mac_4;
 +      struct rtw_debugfs_priv mac_5;
 +      struct rtw_debugfs_priv mac_6;
 +      struct rtw_debugfs_priv mac_7;
 +      struct rtw_debugfs_priv mac_10;
 +      struct rtw_debugfs_priv mac_11;
 +      struct rtw_debugfs_priv mac_12;
 +      struct rtw_debugfs_priv mac_13;
 +      struct rtw_debugfs_priv mac_14;
 +      struct rtw_debugfs_priv mac_15;
 +      struct rtw_debugfs_priv mac_16;
 +      struct rtw_debugfs_priv mac_17;
 +      struct rtw_debugfs_priv bb_8;
 +      struct rtw_debugfs_priv bb_9;
 +      struct rtw_debugfs_priv bb_a;
 +      struct rtw_debugfs_priv bb_b;
 +      struct rtw_debugfs_priv bb_c;
 +      struct rtw_debugfs_priv bb_d;
 +      struct rtw_debugfs_priv bb_e;
 +      struct rtw_debugfs_priv bb_f;
 +      struct rtw_debugfs_priv bb_18;
 +      struct rtw_debugfs_priv bb_19;
 +      struct rtw_debugfs_priv bb_1a;
 +      struct rtw_debugfs_priv bb_1b;
 +      struct rtw_debugfs_priv bb_1c;
 +      struct rtw_debugfs_priv bb_1d;
 +      struct rtw_debugfs_priv bb_1e;
 +      struct rtw_debugfs_priv bb_1f;
 +      struct rtw_debugfs_priv bb_2c;
 +      struct rtw_debugfs_priv bb_2d;
 +      struct rtw_debugfs_priv bb_40;
 +      struct rtw_debugfs_priv bb_41;
 +      struct rtw_debugfs_priv rf_dump;
 +      struct rtw_debugfs_priv tx_pwr_tbl;
 +      struct rtw_debugfs_priv write_reg;
 +      struct rtw_debugfs_priv h2c;
 +      struct rtw_debugfs_priv rf_write;
 +      struct rtw_debugfs_priv rf_read;
 +      struct rtw_debugfs_priv read_reg;
 +      struct rtw_debugfs_priv fix_rate;
 +      struct rtw_debugfs_priv dump_cam;
 +      struct rtw_debugfs_priv rsvd_page;
 +      struct rtw_debugfs_priv phy_info;
 +      struct rtw_debugfs_priv coex_enable;
 +      struct rtw_debugfs_priv coex_info;
 +      struct rtw_debugfs_priv edcca_enable;
 +      struct rtw_debugfs_priv fw_crash;
 +      struct rtw_debugfs_priv force_lowest_basic_rate;
 +      struct rtw_debugfs_priv dm_cap;
 +};
 +
 +static const char * const rtw_dm_cap_strs[] = {
 +      [RTW_DM_CAP_NA] = "NA",
 +      [RTW_DM_CAP_TXGAPK] = "TXGAPK",
 +};
 +
 +static int rtw_debugfs_single_show(struct seq_file *m, void *v)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = m->private;
 +
 +      return debugfs_priv->cb_read(m, v);
 +}
 +
 +static ssize_t rtw_debugfs_common_write(struct file *filp,
 +                                      const char __user *buffer,
 +                                      size_t count, loff_t *loff)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = filp->private_data;
 +
 +      return debugfs_priv->cb_write(filp, buffer, count, loff);
 +}
 +
 +static ssize_t rtw_debugfs_single_write(struct file *filp,
 +                                      const char __user *buffer,
 +                                      size_t count, loff_t *loff)
 +{
 +      struct seq_file *seqpriv = (struct seq_file *)filp->private_data;
 +      struct rtw_debugfs_priv *debugfs_priv = seqpriv->private;
 +
 +      return debugfs_priv->cb_write(filp, buffer, count, loff);
 +}
 +
 +static int rtw_debugfs_single_open_rw(struct inode *inode, struct file *filp)
 +{
 +      return single_open(filp, rtw_debugfs_single_show, inode->i_private);
 +}
 +
 +static int rtw_debugfs_close(struct inode *inode, struct file *filp)
 +{
 +      return 0;
 +}
 +
 +static const struct file_operations file_ops_single_r = {
 +      .owner = THIS_MODULE,
 +      .open = rtw_debugfs_single_open_rw,
 +      .read = seq_read,
 +      .llseek = seq_lseek,
 +      .release = single_release,
 +};
 +
 +static const struct file_operations file_ops_single_rw = {
 +      .owner = THIS_MODULE,
 +      .open = rtw_debugfs_single_open_rw,
 +      .release = single_release,
 +      .read = seq_read,
 +      .llseek = seq_lseek,
 +      .write = rtw_debugfs_single_write,
 +};
 +
 +static const struct file_operations file_ops_common_write = {
 +      .owner = THIS_MODULE,
 +      .write = rtw_debugfs_common_write,
 +      .open = simple_open,
 +      .release = rtw_debugfs_close,
 +};
 +
 +static int rtw_debugfs_get_read_reg(struct seq_file *m, void *v)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = m->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      u32 val, len, addr;
 +
 +      len = debugfs_priv->read_reg.len;
 +      addr = debugfs_priv->read_reg.addr;
 +      switch (len) {
 +      case 1:
 +              val = rtw_read8(rtwdev, addr);
 +              seq_printf(m, "reg 0x%03x: 0x%02x\n", addr, val);
 +              break;
 +      case 2:
 +              val = rtw_read16(rtwdev, addr);
 +              seq_printf(m, "reg 0x%03x: 0x%04x\n", addr, val);
 +              break;
 +      case 4:
 +              val = rtw_read32(rtwdev, addr);
 +              seq_printf(m, "reg 0x%03x: 0x%08x\n", addr, val);
 +              break;
 +      }
 +      return 0;
 +}
 +
 +static int rtw_debugfs_get_rf_read(struct seq_file *m, void *v)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = m->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      u32 val, addr, mask;
 +      u8 path;
 +
 +      path = debugfs_priv->rf_path;
 +      addr = debugfs_priv->rf_addr;
 +      mask = debugfs_priv->rf_mask;
 +
 +      mutex_lock(&rtwdev->mutex);
 +      val = rtw_read_rf(rtwdev, path, addr, mask);
 +      mutex_unlock(&rtwdev->mutex);
 +
 +      seq_printf(m, "rf_read path:%d addr:0x%08x mask:0x%08x val=0x%08x\n",
 +                 path, addr, mask, val);
 +
 +      return 0;
 +}
 +
 +static int rtw_debugfs_get_fix_rate(struct seq_file *m, void *v)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = m->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      struct rtw_dm_info *dm_info = &rtwdev->dm_info;
 +      u8 fix_rate = dm_info->fix_rate;
 +
 +      if (fix_rate >= DESC_RATE_MAX) {
 +              seq_printf(m, "Fix rate disabled, fix_rate = %u\n", fix_rate);
 +              return 0;
 +      }
 +
 +      seq_printf(m, "Data frames fixed at desc rate %u\n", fix_rate);
 +      return 0;
 +}
 +
 +static int rtw_debugfs_copy_from_user(char tmp[], int size,
 +                                    const char __user *buffer, size_t count,
 +                                    int num)
 +{
 +      int tmp_len;
 +
 +      memset(tmp, 0, size);
 +
 +      if (count < num)
 +              return -EFAULT;
 +
 +      tmp_len = (count > size - 1 ? size - 1 : count);
 +
 +      if (copy_from_user(tmp, buffer, tmp_len))
 +              return -EFAULT;
 +
 +      tmp[tmp_len] = '\0';
 +
 +      return 0;
 +}
 +
 +static ssize_t rtw_debugfs_set_read_reg(struct file *filp,
 +                                      const char __user *buffer,
 +                                      size_t count, loff_t *loff)
 +{
 +      struct seq_file *seqpriv = (struct seq_file *)filp->private_data;
 +      struct rtw_debugfs_priv *debugfs_priv = seqpriv->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      char tmp[32 + 1];
 +      u32 addr, len;
 +      int num;
 +      int ret;
 +
 +      ret = rtw_debugfs_copy_from_user(tmp, sizeof(tmp), buffer, count, 2);
 +      if (ret)
 +              return ret;
 +
 +      num = sscanf(tmp, "%x %x", &addr, &len);
 +
 +      if (num !=  2)
 +              return -EINVAL;
 +
 +      if (len != 1 && len != 2 && len != 4) {
 +              rtw_warn(rtwdev, "read reg setting wrong len\n");
 +              return -EINVAL;
 +      }
 +      debugfs_priv->read_reg.addr = addr;
 +      debugfs_priv->read_reg.len = len;
 +
 +      return count;
 +}
 +
 +static int rtw_debugfs_get_dump_cam(struct seq_file *m, void *v)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = m->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      u32 val, command;
 +      u32 hw_key_idx = debugfs_priv->cb_data << RTW_SEC_CAM_ENTRY_SHIFT;
 +      u32 read_cmd = RTW_SEC_CMD_POLLING;
 +      int i;
 +
 +      seq_printf(m, "cam entry%d\n", debugfs_priv->cb_data);
 +      seq_puts(m, "0x0      0x1      0x2     0x3     ");
 +      seq_puts(m, "0x4     0x5\n");
 +      mutex_lock(&rtwdev->mutex);
 +      for (i = 0; i <= 5; i++) {
 +              command = read_cmd | (hw_key_idx + i);
 +              rtw_write32(rtwdev, RTW_SEC_CMD_REG, command);
 +              val = rtw_read32(rtwdev, RTW_SEC_READ_REG);
 +              seq_printf(m, "%8.8x", val);
 +              if (i < 2)
 +                      seq_puts(m, " ");
 +      }
 +      seq_puts(m, "\n");
 +      mutex_unlock(&rtwdev->mutex);
 +      return 0;
 +}
 +
 +static int rtw_debugfs_get_rsvd_page(struct seq_file *m, void *v)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = m->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      u16 page_size = rtwdev->chip->page_size;
 +      u32 buf_size = debugfs_priv->rsvd_page.page_num * page_size;
 +      u32 offset = debugfs_priv->rsvd_page.page_offset * page_size;
 +      u8 *buf;
 +      int i;
 +      int ret;
 +
 +      buf = vzalloc(buf_size);
 +      if (!buf)
 +              return -ENOMEM;
 +
 +      ret = rtw_fw_dump_fifo(rtwdev, RTW_FW_FIFO_SEL_RSVD_PAGE, offset,
 +                             buf_size, (u32 *)buf);
 +      if (ret) {
 +              rtw_err(rtwdev, "failed to dump rsvd page\n");
 +              vfree(buf);
 +              return ret;
 +      }
 +
 +      for (i = 0 ; i < buf_size ; i += 8) {
 +              if (i % page_size == 0)
 +                      seq_printf(m, "PAGE %d\n", (i + offset) / page_size);
 +              seq_printf(m, "%8ph\n", buf + i);
 +      }
 +      vfree(buf);
 +
 +      return 0;
 +}
 +
 +static ssize_t rtw_debugfs_set_rsvd_page(struct file *filp,
 +                                       const char __user *buffer,
 +                                       size_t count, loff_t *loff)
 +{
 +      struct seq_file *seqpriv = (struct seq_file *)filp->private_data;
 +      struct rtw_debugfs_priv *debugfs_priv = seqpriv->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      char tmp[32 + 1];
 +      u32 offset, page_num;
 +      int num;
 +      int ret;
 +
 +      ret = rtw_debugfs_copy_from_user(tmp, sizeof(tmp), buffer, count, 2);
 +      if (ret)
 +              return ret;
 +
 +      num = sscanf(tmp, "%d %d", &offset, &page_num);
 +
 +      if (num != 2) {
 +              rtw_warn(rtwdev, "invalid arguments\n");
 +              return -EINVAL;
 +      }
 +
 +      debugfs_priv->rsvd_page.page_offset = offset;
 +      debugfs_priv->rsvd_page.page_num = page_num;
 +
 +      return count;
 +}
 +
 +static ssize_t rtw_debugfs_set_single_input(struct file *filp,
 +                                          const char __user *buffer,
 +                                          size_t count, loff_t *loff)
 +{
 +      struct seq_file *seqpriv = (struct seq_file *)filp->private_data;
 +      struct rtw_debugfs_priv *debugfs_priv = seqpriv->private;
 +      u32 input;
 +      int ret;
 +
 +      ret = kstrtou32_from_user(buffer, count, 0, &input);
 +      if (ret)
 +              return ret;
 +
 +      debugfs_priv->cb_data = input;
 +
 +      return count;
 +}
 +
 +static ssize_t rtw_debugfs_set_write_reg(struct file *filp,
 +                                       const char __user *buffer,
 +                                       size_t count, loff_t *loff)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = filp->private_data;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      char tmp[32 + 1];
 +      u32 addr, val, len;
 +      int num;
 +      int ret;
 +
 +      ret = rtw_debugfs_copy_from_user(tmp, sizeof(tmp), buffer, count, 3);
 +      if (ret)
 +              return ret;
 +
 +      /* write BB/MAC register */
 +      num = sscanf(tmp, "%x %x %x", &addr, &val, &len);
 +
 +      if (num !=  3)
 +              return -EINVAL;
 +
 +      switch (len) {
 +      case 1:
 +              rtw_dbg(rtwdev, RTW_DBG_DEBUGFS,
 +                      "reg write8 0x%03x: 0x%08x\n", addr, val);
 +              rtw_write8(rtwdev, addr, (u8)val);
 +              break;
 +      case 2:
 +              rtw_dbg(rtwdev, RTW_DBG_DEBUGFS,
 +                      "reg write16 0x%03x: 0x%08x\n", addr, val);
 +              rtw_write16(rtwdev, addr, (u16)val);
 +              break;
 +      case 4:
 +              rtw_dbg(rtwdev, RTW_DBG_DEBUGFS,
 +                      "reg write32 0x%03x: 0x%08x\n", addr, val);
 +              rtw_write32(rtwdev, addr, (u32)val);
 +              break;
 +      default:
 +              rtw_dbg(rtwdev, RTW_DBG_DEBUGFS,
 +                      "error write length = %d\n", len);
 +              break;
 +      }
 +
 +      return count;
 +}
 +
 +static ssize_t rtw_debugfs_set_h2c(struct file *filp,
 +                                 const char __user *buffer,
 +                                 size_t count, loff_t *loff)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = filp->private_data;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      char tmp[32 + 1];
 +      u8 param[8];
 +      int num;
 +      int ret;
 +
 +      ret = rtw_debugfs_copy_from_user(tmp, sizeof(tmp), buffer, count, 3);
 +      if (ret)
 +              return ret;
 +
 +      num = sscanf(tmp, "%hhx,%hhx,%hhx,%hhx,%hhx,%hhx,%hhx,%hhx",
 +                   &param[0], &param[1], &param[2], &param[3],
 +                   &param[4], &param[5], &param[6], &param[7]);
 +      if (num != 8) {
 +              rtw_warn(rtwdev, "invalid H2C command format for debug\n");
 +              return -EINVAL;
 +      }
 +
 +      mutex_lock(&rtwdev->mutex);
 +      rtw_fw_h2c_cmd_dbg(rtwdev, param);
 +      mutex_unlock(&rtwdev->mutex);
 +
 +      return count;
 +}
 +
 +static ssize_t rtw_debugfs_set_rf_write(struct file *filp,
 +                                      const char __user *buffer,
 +                                      size_t count, loff_t *loff)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = filp->private_data;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      char tmp[32 + 1];
 +      u32 path, addr, mask, val;
 +      int num;
 +      int ret;
 +
 +      ret = rtw_debugfs_copy_from_user(tmp, sizeof(tmp), buffer, count, 4);
 +      if (ret)
 +              return ret;
 +
 +      num = sscanf(tmp, "%x %x %x %x", &path, &addr, &mask, &val);
 +
 +      if (num !=  4) {
 +              rtw_warn(rtwdev, "invalid args, [path] [addr] [mask] [val]\n");
 +              return -EINVAL;
 +      }
 +
 +      mutex_lock(&rtwdev->mutex);
 +      rtw_write_rf(rtwdev, path, addr, mask, val);
 +      mutex_unlock(&rtwdev->mutex);
 +      rtw_dbg(rtwdev, RTW_DBG_DEBUGFS,
 +              "write_rf path:%d addr:0x%08x mask:0x%08x, val:0x%08x\n",
 +              path, addr, mask, val);
 +
 +      return count;
 +}
 +
 +static ssize_t rtw_debugfs_set_rf_read(struct file *filp,
 +                                     const char __user *buffer,
 +                                     size_t count, loff_t *loff)
 +{
 +      struct seq_file *seqpriv = (struct seq_file *)filp->private_data;
 +      struct rtw_debugfs_priv *debugfs_priv = seqpriv->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      char tmp[32 + 1];
 +      u32 path, addr, mask;
 +      int num;
 +      int ret;
 +
 +      ret = rtw_debugfs_copy_from_user(tmp, sizeof(tmp), buffer, count, 3);
 +      if (ret)
 +              return ret;
 +
 +      num = sscanf(tmp, "%x %x %x", &path, &addr, &mask);
 +
 +      if (num !=  3) {
 +              rtw_warn(rtwdev, "invalid args, [path] [addr] [mask] [val]\n");
 +              return -EINVAL;
 +      }
 +
 +      debugfs_priv->rf_path = path;
 +      debugfs_priv->rf_addr = addr;
 +      debugfs_priv->rf_mask = mask;
 +
 +      return count;
 +}
 +
 +static ssize_t rtw_debugfs_set_fix_rate(struct file *filp,
 +                                      const char __user *buffer,
 +                                      size_t count, loff_t *loff)
 +{
 +      struct seq_file *seqpriv = (struct seq_file *)filp->private_data;
 +      struct rtw_debugfs_priv *debugfs_priv = seqpriv->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      struct rtw_dm_info *dm_info = &rtwdev->dm_info;
 +      u8 fix_rate;
 +      int ret;
 +
 +      ret = kstrtou8_from_user(buffer, count, 0, &fix_rate);
 +      if (ret)
 +              return ret;
 +
 +      dm_info->fix_rate = fix_rate;
 +
 +      return count;
 +}
 +
 +static int rtw_debug_get_mac_page(struct seq_file *m, void *v)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = m->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      u32 page = debugfs_priv->cb_data;
 +      int i, n;
 +      int max = 0xff;
 +
 +      rtw_read32(rtwdev, debugfs_priv->cb_data);
 +      for (n = 0; n <= max; ) {
 +              seq_printf(m, "\n%8.8x  ", n + page);
 +              for (i = 0; i < 4 && n <= max; i++, n += 4)
 +                      seq_printf(m, "%8.8x    ",
 +                                 rtw_read32(rtwdev, (page | n)));
 +      }
 +      seq_puts(m, "\n");
 +      return 0;
 +}
 +
 +static int rtw_debug_get_bb_page(struct seq_file *m, void *v)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = m->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      u32 page = debugfs_priv->cb_data;
 +      int i, n;
 +      int max = 0xff;
 +
 +      rtw_read32(rtwdev, debugfs_priv->cb_data);
 +      for (n = 0; n <= max; ) {
 +              seq_printf(m, "\n%8.8x  ", n + page);
 +              for (i = 0; i < 4 && n <= max; i++, n += 4)
 +                      seq_printf(m, "%8.8x    ",
 +                                 rtw_read32(rtwdev, (page | n)));
 +      }
 +      seq_puts(m, "\n");
 +      return 0;
 +}
 +
 +static int rtw_debugfs_get_rf_dump(struct seq_file *m, void *v)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = m->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      u32 addr, offset, data;
 +      u8 path;
 +
 +      mutex_lock(&rtwdev->mutex);
 +
 +      for (path = 0; path < rtwdev->hal.rf_path_num; path++) {
 +              seq_printf(m, "RF path:%d\n", path);
 +              for (addr = 0; addr < 0x100; addr += 4) {
 +                      seq_printf(m, "%8.8x  ", addr);
 +                      for (offset = 0; offset < 4; offset++) {
 +                              data = rtw_read_rf(rtwdev, path, addr + offset,
 +                                                 0xffffffff);
 +                              seq_printf(m, "%8.8x    ", data);
 +                      }
 +                      seq_puts(m, "\n");
 +              }
 +              seq_puts(m, "\n");
 +      }
 +
 +      mutex_unlock(&rtwdev->mutex);
 +
 +      return 0;
 +}
 +
 +static void rtw_print_cck_rate_txt(struct seq_file *m, u8 rate)
 +{
 +      static const char * const
 +      cck_rate[] = {"1M", "2M", "5.5M", "11M"};
 +      u8 idx = rate - DESC_RATE1M;
 +
 +      seq_printf(m, " CCK_%-5s", cck_rate[idx]);
 +}
 +
 +static void rtw_print_ofdm_rate_txt(struct seq_file *m, u8 rate)
 +{
 +      static const char * const
 +      ofdm_rate[] = {"6M", "9M", "12M", "18M", "24M", "36M", "48M", "54M"};
 +      u8 idx = rate - DESC_RATE6M;
 +
 +      seq_printf(m, " OFDM_%-4s", ofdm_rate[idx]);
 +}
 +
 +static void rtw_print_ht_rate_txt(struct seq_file *m, u8 rate)
 +{
 +      u8 mcs_n = rate - DESC_RATEMCS0;
 +
 +      seq_printf(m, " MCS%-6u", mcs_n);
 +}
 +
 +static void rtw_print_vht_rate_txt(struct seq_file *m, u8 rate)
 +{
 +      u8 idx = rate - DESC_RATEVHT1SS_MCS0;
 +      u8 n_ss, mcs_n;
 +
 +      /* n spatial stream */
 +      n_ss = 1 + idx / 10;
 +      /* MCS n */
 +      mcs_n = idx % 10;
 +      seq_printf(m, " VHT%uSMCS%u", n_ss, mcs_n);
 +}
 +
 +static void rtw_print_rate(struct seq_file *m, u8 rate)
 +{
 +      switch (rate) {
 +      case DESC_RATE1M...DESC_RATE11M:
 +              rtw_print_cck_rate_txt(m, rate);
 +              break;
 +      case DESC_RATE6M...DESC_RATE54M:
 +              rtw_print_ofdm_rate_txt(m, rate);
 +              break;
-       case DESC_RATEMCS0...DESC_RATEMCS15:
++      case DESC_RATEMCS0...DESC_RATEMCS31:
 +              rtw_print_ht_rate_txt(m, rate);
 +              break;
-       case DESC_RATEVHT1SS_MCS0...DESC_RATEVHT2SS_MCS9:
++      case DESC_RATEVHT1SS_MCS0...DESC_RATEVHT4SS_MCS9:
 +              rtw_print_vht_rate_txt(m, rate);
 +              break;
 +      default:
 +              seq_printf(m, " Unknown rate=0x%x\n", rate);
 +              break;
 +      }
 +}
 +
 +#define case_REGD(src) \
 +      case RTW_REGD_##src: return #src
 +
 +static const char *rtw_get_regd_string(u8 regd)
 +{
 +      switch (regd) {
 +      case_REGD(FCC);
 +      case_REGD(MKK);
 +      case_REGD(ETSI);
 +      case_REGD(IC);
 +      case_REGD(KCC);
 +      case_REGD(ACMA);
 +      case_REGD(CHILE);
 +      case_REGD(UKRAINE);
 +      case_REGD(MEXICO);
 +      case_REGD(CN);
 +      case_REGD(WW);
 +      default:
 +              return "Unknown";
 +      }
 +}
 +
 +static int rtw_debugfs_get_tx_pwr_tbl(struct seq_file *m, void *v)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = m->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
++      struct rtw_power_params pwr_param = {0};
 +      struct rtw_hal *hal = &rtwdev->hal;
++      u8 nss = rtwdev->efuse.hw_cap.nss;
 +      u8 path, rate, bw, ch, regd;
-       struct rtw_power_params pwr_param = {0};
++      u8 max_ht_rate, max_rate;
 +
 +      mutex_lock(&rtwdev->mutex);
 +      bw = hal->current_band_width;
 +      ch = hal->current_channel;
 +      regd = rtw_regd_get(rtwdev);
 +
 +      seq_printf(m, "channel: %u\n", ch);
 +      seq_printf(m, "bandwidth: %u\n", bw);
 +      seq_printf(m, "regulatory: %s\n", rtw_get_regd_string(regd));
 +      seq_printf(m, "%-4s %-10s %-9s %-9s (%-4s %-4s %-4s) %-4s\n",
 +                 "path", "rate", "pwr", "base", "byr", "lmt", "sar", "rem");
 +
++      max_ht_rate = DESC_RATEMCS0 + nss * 8 - 1;
++
++      if (rtwdev->chip->vht_supported)
++              max_rate = DESC_RATEVHT1SS_MCS0 + nss * 10 - 1;
++      else
++              max_rate = max_ht_rate;
++
 +      mutex_lock(&hal->tx_power_mutex);
-       for (path = RF_PATH_A; path <= RF_PATH_B; path++) {
++      for (path = RF_PATH_A; path < hal->rf_path_num; path++) {
 +              /* there is no CCK rates used in 5G */
 +              if (hal->current_band_type == RTW_BAND_5G)
 +                      rate = DESC_RATE6M;
 +              else
 +                      rate = DESC_RATE1M;
 +
-               /* now, not support vht 3ss and vht 4ss*/
-               for (; rate <= DESC_RATEVHT2SS_MCS9; rate++) {
-                       /* now, not support ht 3ss and ht 4ss*/
-                       if (rate > DESC_RATEMCS15 &&
-                           rate < DESC_RATEVHT1SS_MCS0)
++              for (; rate <= max_rate; rate++) {
++                      if (rate > max_ht_rate && rate <= DESC_RATEMCS31)
 +                              continue;
 +
 +                      rtw_get_tx_power_params(rtwdev, path, rate, bw,
 +                                              ch, regd, &pwr_param);
 +
 +                      seq_printf(m, "%4c ", path + 'A');
 +                      rtw_print_rate(m, rate);
 +                      seq_printf(m, " %3u(0x%02x) %4u %4d (%4d %4d %4d) 
%4d\n",
 +                                 hal->tx_pwr_tbl[path][rate],
 +                                 hal->tx_pwr_tbl[path][rate],
 +                                 pwr_param.pwr_base,
 +                                 min3(pwr_param.pwr_offset,
 +                                      pwr_param.pwr_limit,
 +                                      pwr_param.pwr_sar),
 +                                 pwr_param.pwr_offset, pwr_param.pwr_limit,
 +                                 pwr_param.pwr_sar,
 +                                 pwr_param.pwr_remnant);
 +              }
 +      }
 +
 +      mutex_unlock(&hal->tx_power_mutex);
 +      mutex_unlock(&rtwdev->mutex);
 +
 +      return 0;
 +}
 +
 +void rtw_debugfs_get_simple_phy_info(struct seq_file *m)
 +{
 +      struct rtw_debugfs_priv *debugfs_priv = m->private;
 +      struct rtw_dev *rtwdev = debugfs_priv->rtwdev;
 +      struct rtw_hal *hal = &rtwdev->hal;
 +      struct rtw_dm_info *dm_info = &rtwdev->dm_info;
 +      struct rtw_traffic_stats *stats = &rtwdev->stats;
 +
 +      seq_printf(m, "%-40s = %ddBm/ %d\n", "RSSI/ STA Channel",
 +                 dm_info->rssi[RF_PATH_A] - 100, hal->current_channel);
 +
 +      seq_printf(m, "TP {Tx, Rx} = {%u, %u}Mbps\n",
 +                 stats->tx_throughput, stats->rx_throughput);
 +
 +      seq_puts(m, "[Tx Rate] = ");
 +      rtw_print_rate(m, dm_info->tx_rate);
 +      seq_printf(m, "(0x%x)\n", dm_info->tx_rate);
 +
 +      seq_puts(m, "[Rx Rate] = ");
 +      rtw_print_rate(m, dm_info->curr_rx_rate);
 +      seq_printf(m, "(0x%x)\n", dm_info->curr_rx_rate);
 +}
 +
*** 16222 LINES SKIPPED ***

Reply via email to