1bfcc09ddSBjoern A. Zeeb // SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause 2bfcc09ddSBjoern A. Zeeb /* 39af1bba4SBjoern A. Zeeb * Copyright (C) 2012-2014, 2018-2023 Intel Corporation 4bfcc09ddSBjoern A. Zeeb * Copyright (C) 2013-2014 Intel Mobile Communications GmbH 5bfcc09ddSBjoern A. Zeeb * Copyright (C) 2015-2017 Intel Deutschland GmbH 6bfcc09ddSBjoern A. Zeeb */ 7bfcc09ddSBjoern A. Zeeb #include <linux/devcoredump.h> 8bfcc09ddSBjoern A. Zeeb #include "iwl-drv.h" 9bfcc09ddSBjoern A. Zeeb #include "runtime.h" 10bfcc09ddSBjoern A. Zeeb #include "dbg.h" 11bfcc09ddSBjoern A. Zeeb #include "debugfs.h" 12bfcc09ddSBjoern A. Zeeb #include "iwl-io.h" 13bfcc09ddSBjoern A. Zeeb #include "iwl-prph.h" 14bfcc09ddSBjoern A. Zeeb #include "iwl-csr.h" 15d9836fb4SBjoern A. Zeeb #include "pnvm.h" 16bfcc09ddSBjoern A. Zeeb 179af1bba4SBjoern A. Zeeb #define FW_ASSERT_LMAC_FATAL 0x70 189af1bba4SBjoern A. Zeeb #define FW_ASSERT_LMAC2_FATAL 0x72 199af1bba4SBjoern A. Zeeb #define FW_ASSERT_UMAC_FATAL 0x71 209af1bba4SBjoern A. Zeeb #define UMAC_RT_NMI_LMAC2_FATAL 0x72 219af1bba4SBjoern A. Zeeb #define RT_NMI_INTERRUPT_OTHER_LMAC_FATAL 0x73 229af1bba4SBjoern A. Zeeb #define FW_ASSERT_NMI_UNKNOWN 0x84 239af1bba4SBjoern A. Zeeb 24bfcc09ddSBjoern A. Zeeb /* 25bfcc09ddSBjoern A. Zeeb * Note: This structure is read from the device with IO accesses, 26bfcc09ddSBjoern A. Zeeb * and the reading already does the endian conversion. As it is 27bfcc09ddSBjoern A. Zeeb * read with u32-sized accesses, any members with a different size 28bfcc09ddSBjoern A. Zeeb * need to be ordered correctly though! 29bfcc09ddSBjoern A. Zeeb */ 30bfcc09ddSBjoern A. Zeeb struct iwl_error_event_table { 31bfcc09ddSBjoern A. Zeeb u32 valid; /* (nonzero) valid, (0) log is empty */ 32bfcc09ddSBjoern A. Zeeb u32 error_id; /* type of error */ 33bfcc09ddSBjoern A. Zeeb u32 trm_hw_status0; /* TRM HW status */ 34bfcc09ddSBjoern A. Zeeb u32 trm_hw_status1; /* TRM HW status */ 35bfcc09ddSBjoern A. Zeeb u32 blink2; /* branch link */ 36bfcc09ddSBjoern A. Zeeb u32 ilink1; /* interrupt link */ 37bfcc09ddSBjoern A. Zeeb u32 ilink2; /* interrupt link */ 38bfcc09ddSBjoern A. Zeeb u32 data1; /* error-specific data */ 39bfcc09ddSBjoern A. Zeeb u32 data2; /* error-specific data */ 40bfcc09ddSBjoern A. Zeeb u32 data3; /* error-specific data */ 41bfcc09ddSBjoern A. Zeeb u32 bcon_time; /* beacon timer */ 42bfcc09ddSBjoern A. Zeeb u32 tsf_low; /* network timestamp function timer */ 43bfcc09ddSBjoern A. Zeeb u32 tsf_hi; /* network timestamp function timer */ 44bfcc09ddSBjoern A. Zeeb u32 gp1; /* GP1 timer register */ 45bfcc09ddSBjoern A. Zeeb u32 gp2; /* GP2 timer register */ 46bfcc09ddSBjoern A. Zeeb u32 fw_rev_type; /* firmware revision type */ 47bfcc09ddSBjoern A. Zeeb u32 major; /* uCode version major */ 48bfcc09ddSBjoern A. Zeeb u32 minor; /* uCode version minor */ 49bfcc09ddSBjoern A. Zeeb u32 hw_ver; /* HW Silicon version */ 50bfcc09ddSBjoern A. Zeeb u32 brd_ver; /* HW board version */ 51bfcc09ddSBjoern A. Zeeb u32 log_pc; /* log program counter */ 52bfcc09ddSBjoern A. Zeeb u32 frame_ptr; /* frame pointer */ 53bfcc09ddSBjoern A. Zeeb u32 stack_ptr; /* stack pointer */ 54bfcc09ddSBjoern A. Zeeb u32 hcmd; /* last host command header */ 55bfcc09ddSBjoern A. Zeeb u32 isr0; /* isr status register LMPM_NIC_ISR0: 56bfcc09ddSBjoern A. Zeeb * rxtx_flag */ 57bfcc09ddSBjoern A. Zeeb u32 isr1; /* isr status register LMPM_NIC_ISR1: 58bfcc09ddSBjoern A. Zeeb * host_flag */ 59bfcc09ddSBjoern A. Zeeb u32 isr2; /* isr status register LMPM_NIC_ISR2: 60bfcc09ddSBjoern A. Zeeb * enc_flag */ 61bfcc09ddSBjoern A. Zeeb u32 isr3; /* isr status register LMPM_NIC_ISR3: 62bfcc09ddSBjoern A. Zeeb * time_flag */ 63bfcc09ddSBjoern A. Zeeb u32 isr4; /* isr status register LMPM_NIC_ISR4: 64bfcc09ddSBjoern A. Zeeb * wico interrupt */ 65bfcc09ddSBjoern A. Zeeb u32 last_cmd_id; /* last HCMD id handled by the firmware */ 66bfcc09ddSBjoern A. Zeeb u32 wait_event; /* wait event() caller address */ 67bfcc09ddSBjoern A. Zeeb u32 l2p_control; /* L2pControlField */ 68bfcc09ddSBjoern A. Zeeb u32 l2p_duration; /* L2pDurationField */ 69bfcc09ddSBjoern A. Zeeb u32 l2p_mhvalid; /* L2pMhValidBits */ 70bfcc09ddSBjoern A. Zeeb u32 l2p_addr_match; /* L2pAddrMatchStat */ 71bfcc09ddSBjoern A. Zeeb u32 lmpm_pmg_sel; /* indicate which clocks are turned on 72bfcc09ddSBjoern A. Zeeb * (LMPM_PMG_SEL) */ 73bfcc09ddSBjoern A. Zeeb u32 u_timestamp; /* indicate when the date and time of the 74bfcc09ddSBjoern A. Zeeb * compilation */ 75bfcc09ddSBjoern A. Zeeb u32 flow_handler; /* FH read/write pointers, RX credit */ 76bfcc09ddSBjoern A. Zeeb } __packed /* LOG_ERROR_TABLE_API_S_VER_3 */; 77bfcc09ddSBjoern A. Zeeb 78bfcc09ddSBjoern A. Zeeb /* 79bfcc09ddSBjoern A. Zeeb * UMAC error struct - relevant starting from family 8000 chip. 80bfcc09ddSBjoern A. Zeeb * Note: This structure is read from the device with IO accesses, 81bfcc09ddSBjoern A. Zeeb * and the reading already does the endian conversion. As it is 82bfcc09ddSBjoern A. Zeeb * read with u32-sized accesses, any members with a different size 83bfcc09ddSBjoern A. Zeeb * need to be ordered correctly though! 84bfcc09ddSBjoern A. Zeeb */ 85bfcc09ddSBjoern A. Zeeb struct iwl_umac_error_event_table { 86bfcc09ddSBjoern A. Zeeb u32 valid; /* (nonzero) valid, (0) log is empty */ 87bfcc09ddSBjoern A. Zeeb u32 error_id; /* type of error */ 88bfcc09ddSBjoern A. Zeeb u32 blink1; /* branch link */ 89bfcc09ddSBjoern A. Zeeb u32 blink2; /* branch link */ 90bfcc09ddSBjoern A. Zeeb u32 ilink1; /* interrupt link */ 91bfcc09ddSBjoern A. Zeeb u32 ilink2; /* interrupt link */ 92bfcc09ddSBjoern A. Zeeb u32 data1; /* error-specific data */ 93bfcc09ddSBjoern A. Zeeb u32 data2; /* error-specific data */ 94bfcc09ddSBjoern A. Zeeb u32 data3; /* error-specific data */ 95bfcc09ddSBjoern A. Zeeb u32 umac_major; 96bfcc09ddSBjoern A. Zeeb u32 umac_minor; 97bfcc09ddSBjoern A. Zeeb u32 frame_pointer; /* core register 27*/ 98bfcc09ddSBjoern A. Zeeb u32 stack_pointer; /* core register 28 */ 99bfcc09ddSBjoern A. Zeeb u32 cmd_header; /* latest host cmd sent to UMAC */ 100bfcc09ddSBjoern A. Zeeb u32 nic_isr_pref; /* ISR status register */ 101bfcc09ddSBjoern A. Zeeb } __packed; 102bfcc09ddSBjoern A. Zeeb 103bfcc09ddSBjoern A. Zeeb #define ERROR_START_OFFSET (1 * sizeof(u32)) 104bfcc09ddSBjoern A. Zeeb #define ERROR_ELEM_SIZE (7 * sizeof(u32)) 105bfcc09ddSBjoern A. Zeeb 1069af1bba4SBjoern A. Zeeb static bool iwl_fwrt_if_errorid_other_cpu(u32 err_id) 1079af1bba4SBjoern A. Zeeb { 1089af1bba4SBjoern A. Zeeb err_id &= 0xFF; 1099af1bba4SBjoern A. Zeeb 1109af1bba4SBjoern A. Zeeb if ((err_id >= FW_ASSERT_LMAC_FATAL && 1119af1bba4SBjoern A. Zeeb err_id <= RT_NMI_INTERRUPT_OTHER_LMAC_FATAL) || 1129af1bba4SBjoern A. Zeeb err_id == FW_ASSERT_NMI_UNKNOWN) 1139af1bba4SBjoern A. Zeeb return true; 1149af1bba4SBjoern A. Zeeb return false; 1159af1bba4SBjoern A. Zeeb } 1169af1bba4SBjoern A. Zeeb 117bfcc09ddSBjoern A. Zeeb static void iwl_fwrt_dump_umac_error_log(struct iwl_fw_runtime *fwrt) 118bfcc09ddSBjoern A. Zeeb { 119bfcc09ddSBjoern A. Zeeb struct iwl_trans *trans = fwrt->trans; 120bfcc09ddSBjoern A. Zeeb struct iwl_umac_error_event_table table = {}; 121bfcc09ddSBjoern A. Zeeb u32 base = fwrt->trans->dbg.umac_error_event_table; 122d9836fb4SBjoern A. Zeeb char pnvm_name[MAX_PNVM_NAME]; 123bfcc09ddSBjoern A. Zeeb 124bfcc09ddSBjoern A. Zeeb if (!base && 125bfcc09ddSBjoern A. Zeeb !(fwrt->trans->dbg.error_event_table_tlv_status & 126bfcc09ddSBjoern A. Zeeb IWL_ERROR_EVENT_TABLE_UMAC)) 127bfcc09ddSBjoern A. Zeeb return; 128bfcc09ddSBjoern A. Zeeb 129bfcc09ddSBjoern A. Zeeb iwl_trans_read_mem_bytes(trans, base, &table, sizeof(table)); 130bfcc09ddSBjoern A. Zeeb 131bfcc09ddSBjoern A. Zeeb if (table.valid) 132bfcc09ddSBjoern A. Zeeb fwrt->dump.umac_err_id = table.error_id; 133bfcc09ddSBjoern A. Zeeb 1349af1bba4SBjoern A. Zeeb if (!iwl_fwrt_if_errorid_other_cpu(fwrt->dump.umac_err_id) && 1359af1bba4SBjoern A. Zeeb !fwrt->trans->dbg.dump_file_name_ext_valid) { 1369af1bba4SBjoern A. Zeeb fwrt->trans->dbg.dump_file_name_ext_valid = true; 1379af1bba4SBjoern A. Zeeb snprintf(fwrt->trans->dbg.dump_file_name_ext, IWL_FW_INI_MAX_NAME, 1389af1bba4SBjoern A. Zeeb "0x%x", fwrt->dump.umac_err_id); 1399af1bba4SBjoern A. Zeeb } 1409af1bba4SBjoern A. Zeeb 141bfcc09ddSBjoern A. Zeeb if (ERROR_START_OFFSET <= table.valid * ERROR_ELEM_SIZE) { 142bfcc09ddSBjoern A. Zeeb IWL_ERR(trans, "Start IWL Error Log Dump:\n"); 143bfcc09ddSBjoern A. Zeeb IWL_ERR(trans, "Transport status: 0x%08lX, valid: %d\n", 144bfcc09ddSBjoern A. Zeeb fwrt->trans->status, table.valid); 145bfcc09ddSBjoern A. Zeeb } 146bfcc09ddSBjoern A. Zeeb 147d9836fb4SBjoern A. Zeeb if ((table.error_id & ~FW_SYSASSERT_CPU_MASK) == 148d9836fb4SBjoern A. Zeeb FW_SYSASSERT_PNVM_MISSING) { 149d9836fb4SBjoern A. Zeeb iwl_pnvm_get_fs_name(trans, pnvm_name, sizeof(pnvm_name)); 150d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "PNVM data is missing, please install %s\n", 151d9836fb4SBjoern A. Zeeb pnvm_name); 152d9836fb4SBjoern A. Zeeb } 153d9836fb4SBjoern A. Zeeb 154bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | %s\n", table.error_id, 155bfcc09ddSBjoern A. Zeeb iwl_fw_lookup_assert_desc(table.error_id)); 156bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | umac branchlink1\n", table.blink1); 157bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | umac branchlink2\n", table.blink2); 158bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | umac interruptlink1\n", table.ilink1); 159bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | umac interruptlink2\n", table.ilink2); 160bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | umac data1\n", table.data1); 161bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | umac data2\n", table.data2); 162bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | umac data3\n", table.data3); 163bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | umac major\n", table.umac_major); 164bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | umac minor\n", table.umac_minor); 165bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | frame pointer\n", table.frame_pointer); 166bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | stack pointer\n", table.stack_pointer); 167bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | last host cmd\n", table.cmd_header); 168bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | isr status reg\n", table.nic_isr_pref); 169bfcc09ddSBjoern A. Zeeb } 170bfcc09ddSBjoern A. Zeeb 171bfcc09ddSBjoern A. Zeeb static void iwl_fwrt_dump_lmac_error_log(struct iwl_fw_runtime *fwrt, u8 lmac_num) 172bfcc09ddSBjoern A. Zeeb { 173bfcc09ddSBjoern A. Zeeb struct iwl_trans *trans = fwrt->trans; 174bfcc09ddSBjoern A. Zeeb struct iwl_error_event_table table = {}; 175bfcc09ddSBjoern A. Zeeb u32 val, base = fwrt->trans->dbg.lmac_error_event_table[lmac_num]; 176bfcc09ddSBjoern A. Zeeb 177bfcc09ddSBjoern A. Zeeb if (fwrt->cur_fw_img == IWL_UCODE_INIT) { 178bfcc09ddSBjoern A. Zeeb if (!base) 179bfcc09ddSBjoern A. Zeeb base = fwrt->fw->init_errlog_ptr; 180bfcc09ddSBjoern A. Zeeb } else { 181bfcc09ddSBjoern A. Zeeb if (!base) 182bfcc09ddSBjoern A. Zeeb base = fwrt->fw->inst_errlog_ptr; 183bfcc09ddSBjoern A. Zeeb } 184bfcc09ddSBjoern A. Zeeb 185*a4128aadSBjoern A. Zeeb if (!base) { 186bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, 187bfcc09ddSBjoern A. Zeeb "Not valid error log pointer 0x%08X for %s uCode\n", 188bfcc09ddSBjoern A. Zeeb base, 189bfcc09ddSBjoern A. Zeeb (fwrt->cur_fw_img == IWL_UCODE_INIT) 190bfcc09ddSBjoern A. Zeeb ? "Init" : "RT"); 191bfcc09ddSBjoern A. Zeeb return; 192bfcc09ddSBjoern A. Zeeb } 193bfcc09ddSBjoern A. Zeeb 194bfcc09ddSBjoern A. Zeeb /* check if there is a HW error */ 195bfcc09ddSBjoern A. Zeeb val = iwl_trans_read_mem32(trans, base); 1969af1bba4SBjoern A. Zeeb if (iwl_trans_is_hw_error_value(val)) { 197bfcc09ddSBjoern A. Zeeb int err; 198bfcc09ddSBjoern A. Zeeb 199bfcc09ddSBjoern A. Zeeb IWL_ERR(trans, "HW error, resetting before reading\n"); 200bfcc09ddSBjoern A. Zeeb 201bfcc09ddSBjoern A. Zeeb /* reset the device */ 202d9836fb4SBjoern A. Zeeb err = iwl_trans_sw_reset(trans, true); 203d9836fb4SBjoern A. Zeeb if (err) 204d9836fb4SBjoern A. Zeeb return; 205bfcc09ddSBjoern A. Zeeb 206bfcc09ddSBjoern A. Zeeb err = iwl_finish_nic_init(trans); 207bfcc09ddSBjoern A. Zeeb if (err) 208bfcc09ddSBjoern A. Zeeb return; 209bfcc09ddSBjoern A. Zeeb } 210bfcc09ddSBjoern A. Zeeb 211bfcc09ddSBjoern A. Zeeb iwl_trans_read_mem_bytes(trans, base, &table, sizeof(table)); 212bfcc09ddSBjoern A. Zeeb 213bfcc09ddSBjoern A. Zeeb if (table.valid) 214bfcc09ddSBjoern A. Zeeb fwrt->dump.lmac_err_id[lmac_num] = table.error_id; 215bfcc09ddSBjoern A. Zeeb 2169af1bba4SBjoern A. Zeeb if (!iwl_fwrt_if_errorid_other_cpu(fwrt->dump.lmac_err_id[lmac_num]) && 2179af1bba4SBjoern A. Zeeb !fwrt->trans->dbg.dump_file_name_ext_valid) { 2189af1bba4SBjoern A. Zeeb fwrt->trans->dbg.dump_file_name_ext_valid = true; 2199af1bba4SBjoern A. Zeeb snprintf(fwrt->trans->dbg.dump_file_name_ext, IWL_FW_INI_MAX_NAME, 2209af1bba4SBjoern A. Zeeb "0x%x", fwrt->dump.lmac_err_id[lmac_num]); 2219af1bba4SBjoern A. Zeeb } 2229af1bba4SBjoern A. Zeeb 223bfcc09ddSBjoern A. Zeeb if (ERROR_START_OFFSET <= table.valid * ERROR_ELEM_SIZE) { 224bfcc09ddSBjoern A. Zeeb IWL_ERR(trans, "Start IWL Error Log Dump:\n"); 225bfcc09ddSBjoern A. Zeeb IWL_ERR(trans, "Transport status: 0x%08lX, valid: %d\n", 226bfcc09ddSBjoern A. Zeeb fwrt->trans->status, table.valid); 227bfcc09ddSBjoern A. Zeeb } 228bfcc09ddSBjoern A. Zeeb 229bfcc09ddSBjoern A. Zeeb /* Do not change this output - scripts rely on it */ 230bfcc09ddSBjoern A. Zeeb 231bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "Loaded firmware version: %s\n", fwrt->fw->fw_version); 232bfcc09ddSBjoern A. Zeeb 233bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | %-28s\n", table.error_id, 234bfcc09ddSBjoern A. Zeeb iwl_fw_lookup_assert_desc(table.error_id)); 235bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | trm_hw_status0\n", table.trm_hw_status0); 236bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | trm_hw_status1\n", table.trm_hw_status1); 237bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | branchlink2\n", table.blink2); 238bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | interruptlink1\n", table.ilink1); 239bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | interruptlink2\n", table.ilink2); 240bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | data1\n", table.data1); 241bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | data2\n", table.data2); 242bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | data3\n", table.data3); 243bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | beacon time\n", table.bcon_time); 244bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tsf low\n", table.tsf_low); 245bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tsf hi\n", table.tsf_hi); 246bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | time gp1\n", table.gp1); 247bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | time gp2\n", table.gp2); 248bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | uCode revision type\n", table.fw_rev_type); 249bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | uCode version major\n", table.major); 250bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | uCode version minor\n", table.minor); 251bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | hw version\n", table.hw_ver); 252bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | board version\n", table.brd_ver); 253bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | hcmd\n", table.hcmd); 254bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | isr0\n", table.isr0); 255bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | isr1\n", table.isr1); 256bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | isr2\n", table.isr2); 257bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | isr3\n", table.isr3); 258bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | isr4\n", table.isr4); 259bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | last cmd Id\n", table.last_cmd_id); 260bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | wait_event\n", table.wait_event); 261bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | l2p_control\n", table.l2p_control); 262bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | l2p_duration\n", table.l2p_duration); 263bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | l2p_mhvalid\n", table.l2p_mhvalid); 264bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | l2p_addr_match\n", table.l2p_addr_match); 265bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | lmpm_pmg_sel\n", table.lmpm_pmg_sel); 266bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | timestamp\n", table.u_timestamp); 267bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | flow_handler\n", table.flow_handler); 268bfcc09ddSBjoern A. Zeeb } 269bfcc09ddSBjoern A. Zeeb 270bfcc09ddSBjoern A. Zeeb /* 271bfcc09ddSBjoern A. Zeeb * TCM error struct. 272bfcc09ddSBjoern A. Zeeb * Note: This structure is read from the device with IO accesses, 273bfcc09ddSBjoern A. Zeeb * and the reading already does the endian conversion. As it is 274bfcc09ddSBjoern A. Zeeb * read with u32-sized accesses, any members with a different size 275bfcc09ddSBjoern A. Zeeb * need to be ordered correctly though! 276bfcc09ddSBjoern A. Zeeb */ 277bfcc09ddSBjoern A. Zeeb struct iwl_tcm_error_event_table { 278bfcc09ddSBjoern A. Zeeb u32 valid; 279bfcc09ddSBjoern A. Zeeb u32 error_id; 280bfcc09ddSBjoern A. Zeeb u32 blink2; 281bfcc09ddSBjoern A. Zeeb u32 ilink1; 282bfcc09ddSBjoern A. Zeeb u32 ilink2; 283bfcc09ddSBjoern A. Zeeb u32 data1, data2, data3; 284bfcc09ddSBjoern A. Zeeb u32 logpc; 285bfcc09ddSBjoern A. Zeeb u32 frame_pointer; 286bfcc09ddSBjoern A. Zeeb u32 stack_pointer; 287bfcc09ddSBjoern A. Zeeb u32 msgid; 288bfcc09ddSBjoern A. Zeeb u32 isr; 289bfcc09ddSBjoern A. Zeeb u32 hw_status[5]; 290bfcc09ddSBjoern A. Zeeb u32 sw_status[1]; 291bfcc09ddSBjoern A. Zeeb u32 reserved[4]; 292bfcc09ddSBjoern A. Zeeb } __packed; /* TCM_LOG_ERROR_TABLE_API_S_VER_1 */ 293bfcc09ddSBjoern A. Zeeb 294d9836fb4SBjoern A. Zeeb static void iwl_fwrt_dump_tcm_error_log(struct iwl_fw_runtime *fwrt, int idx) 295bfcc09ddSBjoern A. Zeeb { 296bfcc09ddSBjoern A. Zeeb struct iwl_trans *trans = fwrt->trans; 297bfcc09ddSBjoern A. Zeeb struct iwl_tcm_error_event_table table = {}; 298d9836fb4SBjoern A. Zeeb u32 base = fwrt->trans->dbg.tcm_error_event_table[idx]; 299bfcc09ddSBjoern A. Zeeb int i; 300d9836fb4SBjoern A. Zeeb u32 flag = idx ? IWL_ERROR_EVENT_TABLE_TCM2 : 301d9836fb4SBjoern A. Zeeb IWL_ERROR_EVENT_TABLE_TCM1; 302bfcc09ddSBjoern A. Zeeb 303d9836fb4SBjoern A. Zeeb if (!base || !(fwrt->trans->dbg.error_event_table_tlv_status & flag)) 304bfcc09ddSBjoern A. Zeeb return; 305bfcc09ddSBjoern A. Zeeb 306bfcc09ddSBjoern A. Zeeb iwl_trans_read_mem_bytes(trans, base, &table, sizeof(table)); 307bfcc09ddSBjoern A. Zeeb 3089af1bba4SBjoern A. Zeeb if (table.valid) 3099af1bba4SBjoern A. Zeeb fwrt->dump.tcm_err_id[idx] = table.error_id; 3109af1bba4SBjoern A. Zeeb 3119af1bba4SBjoern A. Zeeb if (!iwl_fwrt_if_errorid_other_cpu(fwrt->dump.tcm_err_id[idx]) && 3129af1bba4SBjoern A. Zeeb !fwrt->trans->dbg.dump_file_name_ext_valid) { 3139af1bba4SBjoern A. Zeeb fwrt->trans->dbg.dump_file_name_ext_valid = true; 3149af1bba4SBjoern A. Zeeb snprintf(fwrt->trans->dbg.dump_file_name_ext, IWL_FW_INI_MAX_NAME, 3159af1bba4SBjoern A. Zeeb "0x%x", fwrt->dump.tcm_err_id[idx]); 3169af1bba4SBjoern A. Zeeb } 3179af1bba4SBjoern A. Zeeb 318d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "TCM%d status:\n", idx + 1); 319bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | error ID\n", table.error_id); 320bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm branchlink2\n", table.blink2); 321bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm interruptlink1\n", table.ilink1); 322bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm interruptlink2\n", table.ilink2); 323bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm data1\n", table.data1); 324bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm data2\n", table.data2); 325bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm data3\n", table.data3); 326bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm log PC\n", table.logpc); 327bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm frame pointer\n", table.frame_pointer); 328bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm stack pointer\n", table.stack_pointer); 329bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm msg ID\n", table.msgid); 330bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm ISR status\n", table.isr); 331bfcc09ddSBjoern A. Zeeb for (i = 0; i < ARRAY_SIZE(table.hw_status); i++) 332bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm HW status[%d]\n", 333bfcc09ddSBjoern A. Zeeb table.hw_status[i], i); 334bfcc09ddSBjoern A. Zeeb for (i = 0; i < ARRAY_SIZE(table.sw_status); i++) 335bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | tcm SW status[%d]\n", 336bfcc09ddSBjoern A. Zeeb table.sw_status[i], i); 337bfcc09ddSBjoern A. Zeeb } 338d9836fb4SBjoern A. Zeeb 339d9836fb4SBjoern A. Zeeb /* 340d9836fb4SBjoern A. Zeeb * RCM error struct. 341d9836fb4SBjoern A. Zeeb * Note: This structure is read from the device with IO accesses, 342d9836fb4SBjoern A. Zeeb * and the reading already does the endian conversion. As it is 343d9836fb4SBjoern A. Zeeb * read with u32-sized accesses, any members with a different size 344d9836fb4SBjoern A. Zeeb * need to be ordered correctly though! 345d9836fb4SBjoern A. Zeeb */ 346d9836fb4SBjoern A. Zeeb struct iwl_rcm_error_event_table { 347d9836fb4SBjoern A. Zeeb u32 valid; 348d9836fb4SBjoern A. Zeeb u32 error_id; 349d9836fb4SBjoern A. Zeeb u32 blink2; 350d9836fb4SBjoern A. Zeeb u32 ilink1; 351d9836fb4SBjoern A. Zeeb u32 ilink2; 352d9836fb4SBjoern A. Zeeb u32 data1, data2, data3; 353d9836fb4SBjoern A. Zeeb u32 logpc; 354d9836fb4SBjoern A. Zeeb u32 frame_pointer; 355d9836fb4SBjoern A. Zeeb u32 stack_pointer; 356d9836fb4SBjoern A. Zeeb u32 msgid; 357d9836fb4SBjoern A. Zeeb u32 isr; 358d9836fb4SBjoern A. Zeeb u32 frame_hw_status; 359d9836fb4SBjoern A. Zeeb u32 mbx_lmac_to_rcm_req; 360d9836fb4SBjoern A. Zeeb u32 mbx_rcm_to_lmac_req; 361d9836fb4SBjoern A. Zeeb u32 mh_ctl; 362d9836fb4SBjoern A. Zeeb u32 mh_addr1_lo; 363d9836fb4SBjoern A. Zeeb u32 mh_info; 364d9836fb4SBjoern A. Zeeb u32 mh_err; 365d9836fb4SBjoern A. Zeeb u32 reserved[3]; 366d9836fb4SBjoern A. Zeeb } __packed; /* RCM_LOG_ERROR_TABLE_API_S_VER_1 */ 367d9836fb4SBjoern A. Zeeb 368d9836fb4SBjoern A. Zeeb static void iwl_fwrt_dump_rcm_error_log(struct iwl_fw_runtime *fwrt, int idx) 369d9836fb4SBjoern A. Zeeb { 370d9836fb4SBjoern A. Zeeb struct iwl_trans *trans = fwrt->trans; 371d9836fb4SBjoern A. Zeeb struct iwl_rcm_error_event_table table = {}; 372d9836fb4SBjoern A. Zeeb u32 base = fwrt->trans->dbg.rcm_error_event_table[idx]; 373d9836fb4SBjoern A. Zeeb u32 flag = idx ? IWL_ERROR_EVENT_TABLE_RCM2 : 374d9836fb4SBjoern A. Zeeb IWL_ERROR_EVENT_TABLE_RCM1; 375d9836fb4SBjoern A. Zeeb 376d9836fb4SBjoern A. Zeeb if (!base || !(fwrt->trans->dbg.error_event_table_tlv_status & flag)) 377d9836fb4SBjoern A. Zeeb return; 378d9836fb4SBjoern A. Zeeb 379d9836fb4SBjoern A. Zeeb iwl_trans_read_mem_bytes(trans, base, &table, sizeof(table)); 380d9836fb4SBjoern A. Zeeb 3819af1bba4SBjoern A. Zeeb if (table.valid) 3829af1bba4SBjoern A. Zeeb fwrt->dump.rcm_err_id[idx] = table.error_id; 3839af1bba4SBjoern A. Zeeb 3849af1bba4SBjoern A. Zeeb if (!iwl_fwrt_if_errorid_other_cpu(fwrt->dump.rcm_err_id[idx]) && 3859af1bba4SBjoern A. Zeeb !fwrt->trans->dbg.dump_file_name_ext_valid) { 3869af1bba4SBjoern A. Zeeb fwrt->trans->dbg.dump_file_name_ext_valid = true; 3879af1bba4SBjoern A. Zeeb snprintf(fwrt->trans->dbg.dump_file_name_ext, IWL_FW_INI_MAX_NAME, 3889af1bba4SBjoern A. Zeeb "0x%x", fwrt->dump.rcm_err_id[idx]); 3899af1bba4SBjoern A. Zeeb } 3909af1bba4SBjoern A. Zeeb 391d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "RCM%d status:\n", idx + 1); 392d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | error ID\n", table.error_id); 393d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | rcm branchlink2\n", table.blink2); 394d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | rcm interruptlink1\n", table.ilink1); 395d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | rcm interruptlink2\n", table.ilink2); 396d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | rcm data1\n", table.data1); 397d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | rcm data2\n", table.data2); 398d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | rcm data3\n", table.data3); 399d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | rcm log PC\n", table.logpc); 400d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | rcm frame pointer\n", table.frame_pointer); 401d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | rcm stack pointer\n", table.stack_pointer); 402d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | rcm msg ID\n", table.msgid); 403d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | rcm ISR status\n", table.isr); 404d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | frame HW status\n", table.frame_hw_status); 405d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | LMAC-to-RCM request mbox\n", 406d9836fb4SBjoern A. Zeeb table.mbx_lmac_to_rcm_req); 407d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | RCM-to-LMAC request mbox\n", 408d9836fb4SBjoern A. Zeeb table.mbx_rcm_to_lmac_req); 409d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | MAC header control\n", table.mh_ctl); 410d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | MAC header addr1 low\n", table.mh_addr1_lo); 411d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | MAC header info\n", table.mh_info); 412d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | MAC header error\n", table.mh_err); 413bfcc09ddSBjoern A. Zeeb } 414bfcc09ddSBjoern A. Zeeb 415bfcc09ddSBjoern A. Zeeb static void iwl_fwrt_dump_iml_error_log(struct iwl_fw_runtime *fwrt) 416bfcc09ddSBjoern A. Zeeb { 417bfcc09ddSBjoern A. Zeeb struct iwl_trans *trans = fwrt->trans; 418bfcc09ddSBjoern A. Zeeb u32 error, data1; 419bfcc09ddSBjoern A. Zeeb 420bfcc09ddSBjoern A. Zeeb if (fwrt->trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_22000) { 421bfcc09ddSBjoern A. Zeeb error = UMAG_SB_CPU_2_STATUS; 422bfcc09ddSBjoern A. Zeeb data1 = UMAG_SB_CPU_1_STATUS; 423bfcc09ddSBjoern A. Zeeb } else if (fwrt->trans->trans_cfg->device_family >= 424bfcc09ddSBjoern A. Zeeb IWL_DEVICE_FAMILY_8000) { 425bfcc09ddSBjoern A. Zeeb error = SB_CPU_2_STATUS; 426bfcc09ddSBjoern A. Zeeb data1 = SB_CPU_1_STATUS; 427bfcc09ddSBjoern A. Zeeb } else { 428bfcc09ddSBjoern A. Zeeb return; 429bfcc09ddSBjoern A. Zeeb } 430bfcc09ddSBjoern A. Zeeb 4319af1bba4SBjoern A. Zeeb error = iwl_read_umac_prph(trans, error); 432bfcc09ddSBjoern A. Zeeb 433bfcc09ddSBjoern A. Zeeb IWL_ERR(trans, "IML/ROM dump:\n"); 434bfcc09ddSBjoern A. Zeeb 435bfcc09ddSBjoern A. Zeeb if (error & 0xFFFF0000) 436bfcc09ddSBjoern A. Zeeb IWL_ERR(trans, "0x%04X | IML/ROM SYSASSERT\n", error >> 16); 437bfcc09ddSBjoern A. Zeeb 438bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | IML/ROM error/state\n", error); 439bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | IML/ROM data1\n", 440bfcc09ddSBjoern A. Zeeb iwl_read_umac_prph(trans, data1)); 441bfcc09ddSBjoern A. Zeeb 442bfcc09ddSBjoern A. Zeeb if (fwrt->trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_22000) 443bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | IML/ROM WFPM_AUTH_KEY_0\n", 444bfcc09ddSBjoern A. Zeeb iwl_read_umac_prph(trans, SB_MODIFY_CFG_FLAG)); 445bfcc09ddSBjoern A. Zeeb } 446bfcc09ddSBjoern A. Zeeb 447bfcc09ddSBjoern A. Zeeb #define FSEQ_REG(x) { .addr = (x), .str = #x, } 448bfcc09ddSBjoern A. Zeeb 449bfcc09ddSBjoern A. Zeeb static void iwl_fwrt_dump_fseq_regs(struct iwl_fw_runtime *fwrt) 450bfcc09ddSBjoern A. Zeeb { 451bfcc09ddSBjoern A. Zeeb struct iwl_trans *trans = fwrt->trans; 452bfcc09ddSBjoern A. Zeeb int i; 453bfcc09ddSBjoern A. Zeeb struct { 454bfcc09ddSBjoern A. Zeeb u32 addr; 455bfcc09ddSBjoern A. Zeeb const char *str; 456bfcc09ddSBjoern A. Zeeb } fseq_regs[] = { 457bfcc09ddSBjoern A. Zeeb FSEQ_REG(FSEQ_ERROR_CODE), 458bfcc09ddSBjoern A. Zeeb FSEQ_REG(FSEQ_TOP_INIT_VERSION), 459bfcc09ddSBjoern A. Zeeb FSEQ_REG(FSEQ_CNVIO_INIT_VERSION), 460bfcc09ddSBjoern A. Zeeb FSEQ_REG(FSEQ_OTP_VERSION), 461bfcc09ddSBjoern A. Zeeb FSEQ_REG(FSEQ_TOP_CONTENT_VERSION), 462bfcc09ddSBjoern A. Zeeb FSEQ_REG(FSEQ_ALIVE_TOKEN), 463bfcc09ddSBjoern A. Zeeb FSEQ_REG(FSEQ_CNVI_ID), 464bfcc09ddSBjoern A. Zeeb FSEQ_REG(FSEQ_CNVR_ID), 465bfcc09ddSBjoern A. Zeeb FSEQ_REG(CNVI_AUX_MISC_CHIP), 466bfcc09ddSBjoern A. Zeeb FSEQ_REG(CNVR_AUX_MISC_CHIP), 467bfcc09ddSBjoern A. Zeeb FSEQ_REG(CNVR_SCU_SD_REGS_SD_REG_DIG_DCDC_VTRIM), 468bfcc09ddSBjoern A. Zeeb FSEQ_REG(CNVR_SCU_SD_REGS_SD_REG_ACTIVE_VDIG_MIRROR), 4699af1bba4SBjoern A. Zeeb FSEQ_REG(FSEQ_PREV_CNVIO_INIT_VERSION), 4709af1bba4SBjoern A. Zeeb FSEQ_REG(FSEQ_WIFI_FSEQ_VERSION), 4719af1bba4SBjoern A. Zeeb FSEQ_REG(FSEQ_BT_FSEQ_VERSION), 4729af1bba4SBjoern A. Zeeb FSEQ_REG(FSEQ_CLASS_TP_VERSION), 473bfcc09ddSBjoern A. Zeeb }; 474bfcc09ddSBjoern A. Zeeb 475bfcc09ddSBjoern A. Zeeb if (!iwl_trans_grab_nic_access(trans)) 476bfcc09ddSBjoern A. Zeeb return; 477bfcc09ddSBjoern A. Zeeb 478bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "Fseq Registers:\n"); 479bfcc09ddSBjoern A. Zeeb 480bfcc09ddSBjoern A. Zeeb for (i = 0; i < ARRAY_SIZE(fseq_regs); i++) 481bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | %s\n", 482bfcc09ddSBjoern A. Zeeb iwl_read_prph_no_grab(trans, fseq_regs[i].addr), 483bfcc09ddSBjoern A. Zeeb fseq_regs[i].str); 484bfcc09ddSBjoern A. Zeeb 485bfcc09ddSBjoern A. Zeeb iwl_trans_release_nic_access(trans); 486bfcc09ddSBjoern A. Zeeb } 487bfcc09ddSBjoern A. Zeeb 488bfcc09ddSBjoern A. Zeeb void iwl_fwrt_dump_error_logs(struct iwl_fw_runtime *fwrt) 489bfcc09ddSBjoern A. Zeeb { 4909af1bba4SBjoern A. Zeeb struct iwl_pc_data *pc_data; 4919af1bba4SBjoern A. Zeeb u8 count; 4929af1bba4SBjoern A. Zeeb 493bfcc09ddSBjoern A. Zeeb if (!test_bit(STATUS_DEVICE_ENABLED, &fwrt->trans->status)) { 494bfcc09ddSBjoern A. Zeeb IWL_ERR(fwrt, 495bfcc09ddSBjoern A. Zeeb "DEVICE_ENABLED bit is not set. Aborting dump.\n"); 496bfcc09ddSBjoern A. Zeeb return; 497bfcc09ddSBjoern A. Zeeb } 498bfcc09ddSBjoern A. Zeeb 499bfcc09ddSBjoern A. Zeeb iwl_fwrt_dump_lmac_error_log(fwrt, 0); 500bfcc09ddSBjoern A. Zeeb if (fwrt->trans->dbg.lmac_error_event_table[1]) 501bfcc09ddSBjoern A. Zeeb iwl_fwrt_dump_lmac_error_log(fwrt, 1); 502bfcc09ddSBjoern A. Zeeb iwl_fwrt_dump_umac_error_log(fwrt); 503d9836fb4SBjoern A. Zeeb iwl_fwrt_dump_tcm_error_log(fwrt, 0); 504d9836fb4SBjoern A. Zeeb iwl_fwrt_dump_rcm_error_log(fwrt, 0); 5059af1bba4SBjoern A. Zeeb if (fwrt->trans->dbg.tcm_error_event_table[1]) 506d9836fb4SBjoern A. Zeeb iwl_fwrt_dump_tcm_error_log(fwrt, 1); 5079af1bba4SBjoern A. Zeeb if (fwrt->trans->dbg.rcm_error_event_table[1]) 508d9836fb4SBjoern A. Zeeb iwl_fwrt_dump_rcm_error_log(fwrt, 1); 509bfcc09ddSBjoern A. Zeeb iwl_fwrt_dump_iml_error_log(fwrt); 510bfcc09ddSBjoern A. Zeeb iwl_fwrt_dump_fseq_regs(fwrt); 5119af1bba4SBjoern A. Zeeb if (fwrt->trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_22000) { 5129af1bba4SBjoern A. Zeeb pc_data = fwrt->trans->dbg.pc_data; 5139af1bba4SBjoern A. Zeeb 5149af1bba4SBjoern A. Zeeb if (!iwl_trans_grab_nic_access(fwrt->trans)) 5159af1bba4SBjoern A. Zeeb return; 5169af1bba4SBjoern A. Zeeb for (count = 0; count < fwrt->trans->dbg.num_pc; 5179af1bba4SBjoern A. Zeeb count++, pc_data++) 5189af1bba4SBjoern A. Zeeb IWL_ERR(fwrt, "%s: 0x%x\n", 5199af1bba4SBjoern A. Zeeb pc_data->pc_name, 5209af1bba4SBjoern A. Zeeb iwl_read_prph_no_grab(fwrt->trans, 5219af1bba4SBjoern A. Zeeb pc_data->pc_address)); 5229af1bba4SBjoern A. Zeeb iwl_trans_release_nic_access(fwrt->trans); 5239af1bba4SBjoern A. Zeeb } 524d9836fb4SBjoern A. Zeeb 525d9836fb4SBjoern A. Zeeb if (fwrt->trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_BZ) { 526d9836fb4SBjoern A. Zeeb u32 scratch = iwl_read32(fwrt->trans, CSR_FUNC_SCRATCH); 527d9836fb4SBjoern A. Zeeb 528d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "Function Scratch status:\n"); 529d9836fb4SBjoern A. Zeeb IWL_ERR(fwrt, "0x%08X | Func Scratch\n", scratch); 530d9836fb4SBjoern A. Zeeb } 531bfcc09ddSBjoern A. Zeeb } 532bfcc09ddSBjoern A. Zeeb IWL_EXPORT_SYMBOL(iwl_fwrt_dump_error_logs); 533