wifi: rtw89: support SER L1 simulation
authorZong-Zhe Yang <kevin_yang@realtek.com>
Wed, 14 Sep 2022 03:50:34 +0000 (11:50 +0800)
committerKalle Valo <kvalo@kernel.org>
Mon, 19 Sep 2022 10:03:38 +0000 (13:03 +0300)
SER (system error recovery) can deal with different crash types by
different levels of processes. Previous FW crash simulation triggers
a CPU exception which is one kind of SER L2 type. It can verify SER L2
flow which includes HW/FW restart.

Now, we want to increase crash simulation types. A debug function is added
to trigger control error in purpose for SER L1 simulation/verification.
And, debugfs fw_crash is extended to accept different parameters.

echo 1 > fw_crash:
simulate CPU exception as before
(keep 1 for compatibility with previous)

It will be catched and handled by SER L2.
(this requires HW/FW restart)

echo 2 > fw_crash:
simulate control error

It will be catched and handled by SER L1.
(driver and FW cooperate to recover this)

Besides, in order to apply to the above two cases,
rename RTW89_FLAG_RESTART_TRIGGER to RTW89_FLAG_CRASH_SIMULATING
and adjust where SER flow clears this bit for both L1 and L2.

Signed-off-by: Zong-Zhe Yang <kevin_yang@realtek.com>
Signed-off-by: Ping-Ke Shih <pkshih@realtek.com>
Signed-off-by: Kalle Valo <kvalo@kernel.org>
Link: https://lore.kernel.org/r/20220914035034.14521-5-pkshih@realtek.com
drivers/net/wireless/realtek/rtw89/core.h
drivers/net/wireless/realtek/rtw89/debug.c
drivers/net/wireless/realtek/rtw89/mac.c
drivers/net/wireless/realtek/rtw89/mac.h
drivers/net/wireless/realtek/rtw89/ser.c

index 70d5a10c1a9aa4761e5557ee621e86bcfdf2e760..6e54a00d55b68dbb0f0885b5c0dfd791b76ad3a4 100644 (file)
@@ -2894,7 +2894,7 @@ enum rtw89_flags {
        RTW89_FLAG_LEISURE_PS,
        RTW89_FLAG_LOW_POWER_MODE,
        RTW89_FLAG_INACTIVE_PS,
-       RTW89_FLAG_RESTART_TRIGGER,
+       RTW89_FLAG_CRASH_SIMULATING,
 
        NUM_OF_RTW89_FLAGS,
 };
index eed28c3b440f4aae529da367a83aef037c22e235..730e83d54257ff8c326c32c06da253bcd2e8c337 100644 (file)
@@ -2190,6 +2190,37 @@ out:
        return count;
 }
 
+static int rtw89_dbg_trigger_ctrl_error(struct rtw89_dev *rtwdev)
+{
+       struct rtw89_cpuio_ctrl ctrl_para = {0};
+       u16 pkt_id;
+
+       rtw89_leave_ps_mode(rtwdev);
+
+       pkt_id = rtw89_mac_dle_buf_req(rtwdev, 0x20, true);
+       switch (pkt_id) {
+       case 0xffff:
+               return -ETIMEDOUT;
+       case 0xfff:
+               return -ENOMEM;
+       default:
+               break;
+       }
+
+       /* intentionally, enqueue two pkt, but has only one pkt id */
+       ctrl_para.cmd_type = CPUIO_OP_CMD_ENQ_TO_HEAD;
+       ctrl_para.start_pktid = pkt_id;
+       ctrl_para.end_pktid = pkt_id;
+       ctrl_para.pkt_num = 1; /* start from 0 */
+       ctrl_para.dst_pid = WDE_DLE_PORT_ID_WDRLS;
+       ctrl_para.dst_qid = WDE_DLE_QUEID_NO_REPORT;
+
+       if (rtw89_mac_set_cpuio(rtwdev, &ctrl_para, true))
+               return -EFAULT;
+
+       return 0;
+}
+
 static int
 rtw89_debug_priv_fw_crash_get(struct seq_file *m, void *v)
 {
@@ -2197,10 +2228,15 @@ rtw89_debug_priv_fw_crash_get(struct seq_file *m, void *v)
        struct rtw89_dev *rtwdev = debugfs_priv->rtwdev;
 
        seq_printf(m, "%d\n",
-                  test_bit(RTW89_FLAG_RESTART_TRIGGER, rtwdev->flags));
+                  test_bit(RTW89_FLAG_CRASH_SIMULATING, rtwdev->flags));
        return 0;
 }
 
+enum rtw89_dbg_crash_simulation_type {
+       RTW89_DBG_SIM_CPU_EXCEPTION = 1,
+       RTW89_DBG_SIM_CTRL_ERROR = 2,
+};
+
 static ssize_t
 rtw89_debug_priv_fw_crash_set(struct file *filp, const char __user *user_buf,
                              size_t count, loff_t *loff)
@@ -2208,22 +2244,30 @@ rtw89_debug_priv_fw_crash_set(struct file *filp, const char __user *user_buf,
        struct seq_file *m = (struct seq_file *)filp->private_data;
        struct rtw89_debugfs_priv *debugfs_priv = m->private;
        struct rtw89_dev *rtwdev = debugfs_priv->rtwdev;
-       bool fw_crash;
+       int (*sim)(struct rtw89_dev *rtwdev);
+       u8 crash_type;
        int ret;
 
-       if (!RTW89_CHK_FW_FEATURE(CRASH_TRIGGER, &rtwdev->fw))
-               return -EOPNOTSUPP;
-
-       ret = kstrtobool_from_user(user_buf, count, &fw_crash);
+       ret = kstrtou8_from_user(user_buf, count, 0, &crash_type);
        if (ret)
                return -EINVAL;
 
-       if (!fw_crash)
+       switch (crash_type) {
+       case RTW89_DBG_SIM_CPU_EXCEPTION:
+               if (!RTW89_CHK_FW_FEATURE(CRASH_TRIGGER, &rtwdev->fw))
+                       return -EOPNOTSUPP;
+               sim = rtw89_fw_h2c_trigger_cpu_exception;
+               break;
+       case RTW89_DBG_SIM_CTRL_ERROR:
+               sim = rtw89_dbg_trigger_ctrl_error;
+               break;
+       default:
                return -EINVAL;
+       }
 
        mutex_lock(&rtwdev->mutex);
-       set_bit(RTW89_FLAG_RESTART_TRIGGER, rtwdev->flags);
-       ret = rtw89_fw_h2c_trigger_cpu_exception(rtwdev);
+       set_bit(RTW89_FLAG_CRASH_SIMULATING, rtwdev->flags);
+       ret = sim(rtwdev);
        mutex_unlock(&rtwdev->mutex);
 
        if (ret)
index d2d002d61e97d364cb76b00dd5923bca79cea21e..d126fa963e66b82ad7ee50e558f28f069aa78bd5 100644 (file)
@@ -2487,8 +2487,7 @@ int rtw89_mac_resume_sch_tx_v1(struct rtw89_dev *rtwdev, u8 mac_idx, u32 tx_en)
 }
 EXPORT_SYMBOL(rtw89_mac_resume_sch_tx_v1);
 
-static u16 rtw89_mac_dle_buf_req(struct rtw89_dev *rtwdev, u16 buf_len,
-                                bool wd)
+u16 rtw89_mac_dle_buf_req(struct rtw89_dev *rtwdev, u16 buf_len, bool wd)
 {
        u32 val, reg;
        int ret;
@@ -2508,9 +2507,8 @@ static u16 rtw89_mac_dle_buf_req(struct rtw89_dev *rtwdev, u16 buf_len,
        return FIELD_GET(B_AX_WD_BUF_STAT_PKTID_MASK, val);
 }
 
-static int rtw89_mac_set_cpuio(struct rtw89_dev *rtwdev,
-                              struct rtw89_cpuio_ctrl *ctrl_para,
-                              bool wd)
+int rtw89_mac_set_cpuio(struct rtw89_dev *rtwdev,
+                       struct rtw89_cpuio_ctrl *ctrl_para, bool wd)
 {
        u32 val, cmd_type, reg;
        int ret;
index 9d8574714b051dfae6708657073337b973b56c9f..22db80716b56077a3613507d1f5ca3b120da99b7 100644 (file)
@@ -1004,5 +1004,8 @@ enum rtw89_mac_xtal_si_offset {
 int rtw89_mac_write_xtal_si(struct rtw89_dev *rtwdev, u8 offset, u8 val, u8 mask);
 int rtw89_mac_read_xtal_si(struct rtw89_dev *rtwdev, u8 offset, u8 *val);
 void rtw89_mac_pkt_drop_vif(struct rtw89_dev *rtwdev, struct rtw89_vif *rtwvif);
+u16 rtw89_mac_dle_buf_req(struct rtw89_dev *rtwdev, u16 buf_len, bool wd);
+int rtw89_mac_set_cpuio(struct rtw89_dev *rtwdev,
+                       struct rtw89_cpuio_ctrl *ctrl_para, bool wd);
 
 #endif
index 1b01ff0402fb60dbf6e6d7c1eefd6c79aff7ebab..c1a4bc1c64d16fba26f642c0a43118a66182ab12 100644 (file)
@@ -396,6 +396,7 @@ static void ser_idle_st_hdl(struct rtw89_ser *ser, u8 evt)
        switch (evt) {
        case SER_EV_STATE_IN:
                rtw89_hci_recovery_complete(rtwdev);
+               clear_bit(RTW89_FLAG_CRASH_SIMULATING, rtwdev->flags);
                break;
        case SER_EV_L1_RESET:
                ser_state_goto(ser, SER_RESET_TRX_ST);
@@ -632,7 +633,6 @@ static void ser_l2_reset_st_hdl(struct rtw89_ser *ser, u8 evt)
                fallthrough;
        case SER_EV_L2_RECFG_DONE:
                ser_state_goto(ser, SER_IDLE_ST);
-               clear_bit(RTW89_FLAG_RESTART_TRIGGER, rtwdev->flags);
                break;
 
        case SER_EV_STATE_OUT: