Path: blob/main/sys/contrib/dev/iwlwifi/fw/dump.c
106879 views
// SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause1/*2* Copyright (C) 2012-2014, 2018-2025 Intel Corporation3* Copyright (C) 2013-2014 Intel Mobile Communications GmbH4* Copyright (C) 2015-2017 Intel Deutschland GmbH5*/6#include <linux/devcoredump.h>7#include "iwl-drv.h"8#include "runtime.h"9#include "dbg.h"10#include "debugfs.h"11#include "iwl-io.h"12#include "iwl-prph.h"13#include "iwl-csr.h"14#include "pnvm.h"1516#define FW_ASSERT_LMAC_FATAL 0x7017#define FW_ASSERT_LMAC2_FATAL 0x7218#define FW_ASSERT_UMAC_FATAL 0x7119#define UMAC_RT_NMI_LMAC2_FATAL 0x7220#define RT_NMI_INTERRUPT_OTHER_LMAC_FATAL 0x7321#define FW_ASSERT_NMI_UNKNOWN 0x842223/*24* Note: This structure is read from the device with IO accesses,25* and the reading already does the endian conversion. As it is26* read with u32-sized accesses, any members with a different size27* need to be ordered correctly though!28*/29struct iwl_error_event_table {30u32 valid; /* (nonzero) valid, (0) log is empty */31u32 error_id; /* type of error */32u32 trm_hw_status0; /* TRM HW status */33u32 trm_hw_status1; /* TRM HW status */34u32 blink2; /* branch link */35u32 ilink1; /* interrupt link */36u32 ilink2; /* interrupt link */37u32 data1; /* error-specific data */38u32 data2; /* error-specific data */39u32 data3; /* error-specific data */40u32 bcon_time; /* beacon timer */41u32 tsf_low; /* network timestamp function timer */42u32 tsf_hi; /* network timestamp function timer */43u32 gp1; /* GP1 timer register */44u32 gp2; /* GP2 timer register */45u32 fw_rev_type; /* firmware revision type */46u32 major; /* uCode version major */47u32 minor; /* uCode version minor */48u32 hw_ver; /* HW Silicon version */49u32 brd_ver; /* HW board version */50u32 log_pc; /* log program counter */51u32 frame_ptr; /* frame pointer */52u32 stack_ptr; /* stack pointer */53u32 hcmd; /* last host command header */54u32 isr0; /* isr status register LMPM_NIC_ISR0:55* rxtx_flag */56u32 isr1; /* isr status register LMPM_NIC_ISR1:57* host_flag */58u32 isr2; /* isr status register LMPM_NIC_ISR2:59* enc_flag */60u32 isr3; /* isr status register LMPM_NIC_ISR3:61* time_flag */62u32 isr4; /* isr status register LMPM_NIC_ISR4:63* wico interrupt */64u32 last_cmd_id; /* last HCMD id handled by the firmware */65u32 wait_event; /* wait event() caller address */66u32 l2p_control; /* L2pControlField */67u32 l2p_duration; /* L2pDurationField */68u32 l2p_mhvalid; /* L2pMhValidBits */69u32 l2p_addr_match; /* L2pAddrMatchStat */70u32 lmpm_pmg_sel; /* indicate which clocks are turned on71* (LMPM_PMG_SEL) */72u32 u_timestamp; /* indicate when the date and time of the73* compilation */74u32 flow_handler; /* FH read/write pointers, RX credit */75} __packed /* LOG_ERROR_TABLE_API_S_VER_3 */;7677/*78* UMAC error struct - relevant starting from family 8000 chip.79* Note: This structure is read from the device with IO accesses,80* and the reading already does the endian conversion. As it is81* read with u32-sized accesses, any members with a different size82* need to be ordered correctly though!83*/84struct iwl_umac_error_event_table {85u32 valid; /* (nonzero) valid, (0) log is empty */86u32 error_id; /* type of error */87u32 blink1; /* branch link */88u32 blink2; /* branch link */89u32 ilink1; /* interrupt link */90u32 ilink2; /* interrupt link */91u32 data1; /* error-specific data */92u32 data2; /* error-specific data */93u32 data3; /* error-specific data */94u32 umac_major;95u32 umac_minor;96u32 frame_pointer; /* core register 27*/97u32 stack_pointer; /* core register 28 */98u32 cmd_header; /* latest host cmd sent to UMAC */99u32 nic_isr_pref; /* ISR status register */100} __packed;101102#define ERROR_START_OFFSET (1 * sizeof(u32))103#define ERROR_ELEM_SIZE (7 * sizeof(u32))104105static bool iwl_fwrt_if_errorid_other_cpu(u32 err_id)106{107err_id &= 0xFF;108109if ((err_id >= FW_ASSERT_LMAC_FATAL &&110err_id <= RT_NMI_INTERRUPT_OTHER_LMAC_FATAL) ||111err_id == FW_ASSERT_NMI_UNKNOWN)112return true;113return false;114}115116static void iwl_fwrt_dump_umac_error_log(struct iwl_fw_runtime *fwrt)117{118struct iwl_trans *trans = fwrt->trans;119struct iwl_umac_error_event_table table = {};120u32 base = fwrt->trans->dbg.umac_error_event_table;121char pnvm_name[MAX_PNVM_NAME];122123if (!base &&124!(fwrt->trans->dbg.error_event_table_tlv_status &125IWL_ERROR_EVENT_TABLE_UMAC))126return;127128iwl_trans_read_mem_bytes(trans, base, &table, sizeof(table));129130if (table.valid)131fwrt->dump.umac_err_id = table.error_id;132133if (!iwl_fwrt_if_errorid_other_cpu(fwrt->dump.umac_err_id) &&134!fwrt->trans->dbg.dump_file_name_ext_valid) {135fwrt->trans->dbg.dump_file_name_ext_valid = true;136snprintf(fwrt->trans->dbg.dump_file_name_ext, IWL_FW_INI_MAX_NAME,137"0x%x", fwrt->dump.umac_err_id);138}139140if (ERROR_START_OFFSET <= table.valid * ERROR_ELEM_SIZE) {141IWL_ERR(trans, "Start IWL Error Log Dump:\n");142IWL_ERR(trans, "Transport status: 0x%08lX, valid: %d\n",143fwrt->trans->status, table.valid);144}145146if ((table.error_id & ~FW_SYSASSERT_CPU_MASK) ==147FW_SYSASSERT_PNVM_MISSING) {148iwl_pnvm_get_fs_name(trans, pnvm_name, sizeof(pnvm_name));149IWL_ERR(fwrt, "PNVM data is missing, please install %s\n",150pnvm_name);151}152153IWL_ERR(fwrt, "0x%08X | %s\n", table.error_id,154iwl_fw_lookup_assert_desc(table.error_id));155IWL_ERR(fwrt, "0x%08X | umac branchlink1\n", table.blink1);156IWL_ERR(fwrt, "0x%08X | umac branchlink2\n", table.blink2);157IWL_ERR(fwrt, "0x%08X | umac interruptlink1\n", table.ilink1);158IWL_ERR(fwrt, "0x%08X | umac interruptlink2\n", table.ilink2);159IWL_ERR(fwrt, "0x%08X | umac data1\n", table.data1);160IWL_ERR(fwrt, "0x%08X | umac data2\n", table.data2);161IWL_ERR(fwrt, "0x%08X | umac data3\n", table.data3);162IWL_ERR(fwrt, "0x%08X | umac major\n", table.umac_major);163IWL_ERR(fwrt, "0x%08X | umac minor\n", table.umac_minor);164IWL_ERR(fwrt, "0x%08X | frame pointer\n", table.frame_pointer);165IWL_ERR(fwrt, "0x%08X | stack pointer\n", table.stack_pointer);166IWL_ERR(fwrt, "0x%08X | last host cmd\n", table.cmd_header);167IWL_ERR(fwrt, "0x%08X | isr status reg\n", table.nic_isr_pref);168}169170static void iwl_fwrt_dump_lmac_error_log(struct iwl_fw_runtime *fwrt, u8 lmac_num)171{172struct iwl_trans *trans = fwrt->trans;173struct iwl_error_event_table table = {};174u32 val, base = fwrt->trans->dbg.lmac_error_event_table[lmac_num];175176if (fwrt->cur_fw_img == IWL_UCODE_INIT) {177if (!base)178base = fwrt->fw->init_errlog_ptr;179} else {180if (!base)181base = fwrt->fw->inst_errlog_ptr;182}183184if (!base) {185IWL_ERR(fwrt,186"Not valid error log pointer 0x%08X for %s uCode\n",187base,188(fwrt->cur_fw_img == IWL_UCODE_INIT)189? "Init" : "RT");190return;191}192193/* check if there is a HW error */194val = iwl_trans_read_mem32(trans, base);195if (iwl_trans_is_hw_error_value(val)) {196int err;197198IWL_ERR(trans, "HW error, resetting before reading\n");199200/* reset the device */201err = iwl_trans_sw_reset(trans);202if (err)203return;204205err = iwl_finish_nic_init(trans);206if (err)207return;208}209210iwl_trans_read_mem_bytes(trans, base, &table, sizeof(table));211212if (table.valid)213fwrt->dump.lmac_err_id[lmac_num] = table.error_id;214215if (!iwl_fwrt_if_errorid_other_cpu(fwrt->dump.lmac_err_id[lmac_num]) &&216!fwrt->trans->dbg.dump_file_name_ext_valid) {217fwrt->trans->dbg.dump_file_name_ext_valid = true;218snprintf(fwrt->trans->dbg.dump_file_name_ext, IWL_FW_INI_MAX_NAME,219"0x%x", fwrt->dump.lmac_err_id[lmac_num]);220}221222if (ERROR_START_OFFSET <= table.valid * ERROR_ELEM_SIZE) {223IWL_ERR(trans, "Start IWL Error Log Dump:\n");224IWL_ERR(trans, "Transport status: 0x%08lX, valid: %d\n",225fwrt->trans->status, table.valid);226}227228/* Do not change this output - scripts rely on it */229230IWL_ERR(fwrt, "Loaded firmware version: %s\n", fwrt->fw->fw_version);231232IWL_ERR(fwrt, "0x%08X | %-28s\n", table.error_id,233iwl_fw_lookup_assert_desc(table.error_id));234IWL_ERR(fwrt, "0x%08X | trm_hw_status0\n", table.trm_hw_status0);235IWL_ERR(fwrt, "0x%08X | trm_hw_status1\n", table.trm_hw_status1);236IWL_ERR(fwrt, "0x%08X | branchlink2\n", table.blink2);237IWL_ERR(fwrt, "0x%08X | interruptlink1\n", table.ilink1);238IWL_ERR(fwrt, "0x%08X | interruptlink2\n", table.ilink2);239IWL_ERR(fwrt, "0x%08X | data1\n", table.data1);240IWL_ERR(fwrt, "0x%08X | data2\n", table.data2);241IWL_ERR(fwrt, "0x%08X | data3\n", table.data3);242IWL_ERR(fwrt, "0x%08X | beacon time\n", table.bcon_time);243IWL_ERR(fwrt, "0x%08X | tsf low\n", table.tsf_low);244IWL_ERR(fwrt, "0x%08X | tsf hi\n", table.tsf_hi);245IWL_ERR(fwrt, "0x%08X | time gp1\n", table.gp1);246IWL_ERR(fwrt, "0x%08X | time gp2\n", table.gp2);247IWL_ERR(fwrt, "0x%08X | uCode revision type\n", table.fw_rev_type);248IWL_ERR(fwrt, "0x%08X | uCode version major\n", table.major);249IWL_ERR(fwrt, "0x%08X | uCode version minor\n", table.minor);250IWL_ERR(fwrt, "0x%08X | hw version\n", table.hw_ver);251IWL_ERR(fwrt, "0x%08X | board version\n", table.brd_ver);252IWL_ERR(fwrt, "0x%08X | hcmd\n", table.hcmd);253IWL_ERR(fwrt, "0x%08X | isr0\n", table.isr0);254IWL_ERR(fwrt, "0x%08X | isr1\n", table.isr1);255IWL_ERR(fwrt, "0x%08X | isr2\n", table.isr2);256IWL_ERR(fwrt, "0x%08X | isr3\n", table.isr3);257IWL_ERR(fwrt, "0x%08X | isr4\n", table.isr4);258IWL_ERR(fwrt, "0x%08X | last cmd Id\n", table.last_cmd_id);259IWL_ERR(fwrt, "0x%08X | wait_event\n", table.wait_event);260IWL_ERR(fwrt, "0x%08X | l2p_control\n", table.l2p_control);261IWL_ERR(fwrt, "0x%08X | l2p_duration\n", table.l2p_duration);262IWL_ERR(fwrt, "0x%08X | l2p_mhvalid\n", table.l2p_mhvalid);263IWL_ERR(fwrt, "0x%08X | l2p_addr_match\n", table.l2p_addr_match);264IWL_ERR(fwrt, "0x%08X | lmpm_pmg_sel\n", table.lmpm_pmg_sel);265IWL_ERR(fwrt, "0x%08X | timestamp\n", table.u_timestamp);266IWL_ERR(fwrt, "0x%08X | flow_handler\n", table.flow_handler);267}268269/*270* TCM error struct.271* Note: This structure is read from the device with IO accesses,272* and the reading already does the endian conversion. As it is273* read with u32-sized accesses, any members with a different size274* need to be ordered correctly though!275*/276struct iwl_tcm_error_event_table {277u32 valid;278u32 error_id;279u32 blink2;280u32 ilink1;281u32 ilink2;282u32 data1, data2, data3;283u32 logpc;284u32 frame_pointer;285u32 stack_pointer;286u32 msgid;287u32 isr;288u32 hw_status[5];289u32 sw_status[1];290u32 reserved[4];291} __packed; /* TCM_LOG_ERROR_TABLE_API_S_VER_1 */292293static void iwl_fwrt_dump_tcm_error_log(struct iwl_fw_runtime *fwrt, int idx)294{295struct iwl_trans *trans = fwrt->trans;296struct iwl_tcm_error_event_table table = {};297u32 base = fwrt->trans->dbg.tcm_error_event_table[idx];298int i;299u32 flag = idx ? IWL_ERROR_EVENT_TABLE_TCM2 :300IWL_ERROR_EVENT_TABLE_TCM1;301302if (!base || !(fwrt->trans->dbg.error_event_table_tlv_status & flag))303return;304305iwl_trans_read_mem_bytes(trans, base, &table, sizeof(table));306307if (table.valid)308fwrt->dump.tcm_err_id[idx] = table.error_id;309310if (!iwl_fwrt_if_errorid_other_cpu(fwrt->dump.tcm_err_id[idx]) &&311!fwrt->trans->dbg.dump_file_name_ext_valid) {312fwrt->trans->dbg.dump_file_name_ext_valid = true;313snprintf(fwrt->trans->dbg.dump_file_name_ext, IWL_FW_INI_MAX_NAME,314"0x%x", fwrt->dump.tcm_err_id[idx]);315}316317IWL_ERR(fwrt, "TCM%d status:\n", idx + 1);318IWL_ERR(fwrt, "0x%08X | error ID\n", table.error_id);319IWL_ERR(fwrt, "0x%08X | tcm branchlink2\n", table.blink2);320IWL_ERR(fwrt, "0x%08X | tcm interruptlink1\n", table.ilink1);321IWL_ERR(fwrt, "0x%08X | tcm interruptlink2\n", table.ilink2);322IWL_ERR(fwrt, "0x%08X | tcm data1\n", table.data1);323IWL_ERR(fwrt, "0x%08X | tcm data2\n", table.data2);324IWL_ERR(fwrt, "0x%08X | tcm data3\n", table.data3);325IWL_ERR(fwrt, "0x%08X | tcm log PC\n", table.logpc);326IWL_ERR(fwrt, "0x%08X | tcm frame pointer\n", table.frame_pointer);327IWL_ERR(fwrt, "0x%08X | tcm stack pointer\n", table.stack_pointer);328IWL_ERR(fwrt, "0x%08X | tcm msg ID\n", table.msgid);329IWL_ERR(fwrt, "0x%08X | tcm ISR status\n", table.isr);330for (i = 0; i < ARRAY_SIZE(table.hw_status); i++)331IWL_ERR(fwrt, "0x%08X | tcm HW status[%d]\n",332table.hw_status[i], i);333for (i = 0; i < ARRAY_SIZE(table.sw_status); i++)334IWL_ERR(fwrt, "0x%08X | tcm SW status[%d]\n",335table.sw_status[i], i);336}337338/*339* RCM error struct.340* Note: This structure is read from the device with IO accesses,341* and the reading already does the endian conversion. As it is342* read with u32-sized accesses, any members with a different size343* need to be ordered correctly though!344*/345struct iwl_rcm_error_event_table {346u32 valid;347u32 error_id;348u32 blink2;349u32 ilink1;350u32 ilink2;351u32 data1, data2, data3;352u32 logpc;353u32 frame_pointer;354u32 stack_pointer;355u32 msgid;356u32 isr;357u32 frame_hw_status;358u32 mbx_lmac_to_rcm_req;359u32 mbx_rcm_to_lmac_req;360u32 mh_ctl;361u32 mh_addr1_lo;362u32 mh_info;363u32 mh_err;364u32 reserved[3];365} __packed; /* RCM_LOG_ERROR_TABLE_API_S_VER_1 */366367static void iwl_fwrt_dump_rcm_error_log(struct iwl_fw_runtime *fwrt, int idx)368{369struct iwl_trans *trans = fwrt->trans;370struct iwl_rcm_error_event_table table = {};371u32 base = fwrt->trans->dbg.rcm_error_event_table[idx];372u32 flag = idx ? IWL_ERROR_EVENT_TABLE_RCM2 :373IWL_ERROR_EVENT_TABLE_RCM1;374375if (!base || !(fwrt->trans->dbg.error_event_table_tlv_status & flag))376return;377378iwl_trans_read_mem_bytes(trans, base, &table, sizeof(table));379380if (table.valid)381fwrt->dump.rcm_err_id[idx] = table.error_id;382383if (!iwl_fwrt_if_errorid_other_cpu(fwrt->dump.rcm_err_id[idx]) &&384!fwrt->trans->dbg.dump_file_name_ext_valid) {385fwrt->trans->dbg.dump_file_name_ext_valid = true;386snprintf(fwrt->trans->dbg.dump_file_name_ext, IWL_FW_INI_MAX_NAME,387"0x%x", fwrt->dump.rcm_err_id[idx]);388}389390IWL_ERR(fwrt, "RCM%d status:\n", idx + 1);391IWL_ERR(fwrt, "0x%08X | error ID\n", table.error_id);392IWL_ERR(fwrt, "0x%08X | rcm branchlink2\n", table.blink2);393IWL_ERR(fwrt, "0x%08X | rcm interruptlink1\n", table.ilink1);394IWL_ERR(fwrt, "0x%08X | rcm interruptlink2\n", table.ilink2);395IWL_ERR(fwrt, "0x%08X | rcm data1\n", table.data1);396IWL_ERR(fwrt, "0x%08X | rcm data2\n", table.data2);397IWL_ERR(fwrt, "0x%08X | rcm data3\n", table.data3);398IWL_ERR(fwrt, "0x%08X | rcm log PC\n", table.logpc);399IWL_ERR(fwrt, "0x%08X | rcm frame pointer\n", table.frame_pointer);400IWL_ERR(fwrt, "0x%08X | rcm stack pointer\n", table.stack_pointer);401IWL_ERR(fwrt, "0x%08X | rcm msg ID\n", table.msgid);402IWL_ERR(fwrt, "0x%08X | rcm ISR status\n", table.isr);403IWL_ERR(fwrt, "0x%08X | frame HW status\n", table.frame_hw_status);404IWL_ERR(fwrt, "0x%08X | LMAC-to-RCM request mbox\n",405table.mbx_lmac_to_rcm_req);406IWL_ERR(fwrt, "0x%08X | RCM-to-LMAC request mbox\n",407table.mbx_rcm_to_lmac_req);408IWL_ERR(fwrt, "0x%08X | MAC header control\n", table.mh_ctl);409IWL_ERR(fwrt, "0x%08X | MAC header addr1 low\n", table.mh_addr1_lo);410IWL_ERR(fwrt, "0x%08X | MAC header info\n", table.mh_info);411IWL_ERR(fwrt, "0x%08X | MAC header error\n", table.mh_err);412}413414static void iwl_fwrt_dump_iml_error_log(struct iwl_fw_runtime *fwrt)415{416struct iwl_trans *trans = fwrt->trans;417u32 error, data1;418419if (fwrt->trans->mac_cfg->device_family >= IWL_DEVICE_FAMILY_22000) {420error = UMAG_SB_CPU_2_STATUS;421data1 = UMAG_SB_CPU_1_STATUS;422} else if (fwrt->trans->mac_cfg->device_family >=423IWL_DEVICE_FAMILY_8000) {424error = SB_CPU_2_STATUS;425data1 = SB_CPU_1_STATUS;426} else {427return;428}429430error = iwl_read_umac_prph(trans, error);431432IWL_ERR(trans, "IML/ROM dump:\n");433434if (error & 0xFFFF0000)435IWL_ERR(trans, "0x%04X | IML/ROM SYSASSERT\n", error >> 16);436437IWL_ERR(fwrt, "0x%08X | IML/ROM error/state\n", error);438IWL_ERR(fwrt, "0x%08X | IML/ROM data1\n",439iwl_read_umac_prph(trans, data1));440441if (fwrt->trans->mac_cfg->device_family >= IWL_DEVICE_FAMILY_22000)442IWL_ERR(fwrt, "0x%08X | IML/ROM WFPM_AUTH_KEY_0\n",443iwl_read_umac_prph(trans, SB_MODIFY_CFG_FLAG));444}445446#define FSEQ_REG(x) { .addr = (x), .str = #x, }447448static void iwl_fwrt_dump_fseq_regs(struct iwl_fw_runtime *fwrt)449{450struct iwl_trans *trans = fwrt->trans;451int i;452struct {453u32 addr;454const char *str;455} fseq_regs[] = {456FSEQ_REG(FSEQ_ERROR_CODE),457FSEQ_REG(FSEQ_TOP_INIT_VERSION),458FSEQ_REG(FSEQ_CNVIO_INIT_VERSION),459FSEQ_REG(FSEQ_OTP_VERSION),460FSEQ_REG(FSEQ_TOP_CONTENT_VERSION),461FSEQ_REG(FSEQ_ALIVE_TOKEN),462FSEQ_REG(FSEQ_CNVI_ID),463FSEQ_REG(FSEQ_CNVR_ID),464FSEQ_REG(CNVI_AUX_MISC_CHIP),465FSEQ_REG(CNVR_AUX_MISC_CHIP),466FSEQ_REG(CNVR_SCU_SD_REGS_SD_REG_DIG_DCDC_VTRIM),467FSEQ_REG(CNVR_SCU_SD_REGS_SD_REG_ACTIVE_VDIG_MIRROR),468FSEQ_REG(FSEQ_PREV_CNVIO_INIT_VERSION),469FSEQ_REG(FSEQ_WIFI_FSEQ_VERSION),470FSEQ_REG(FSEQ_BT_FSEQ_VERSION),471FSEQ_REG(FSEQ_CLASS_TP_VERSION),472};473474if (!iwl_trans_grab_nic_access(trans))475return;476477IWL_ERR(fwrt, "Fseq Registers:\n");478479for (i = 0; i < ARRAY_SIZE(fseq_regs); i++)480IWL_ERR(fwrt, "0x%08X | %s\n",481iwl_read_prph_no_grab(trans, fseq_regs[i].addr),482fseq_regs[i].str);483484iwl_trans_release_nic_access(trans);485}486487void iwl_fwrt_dump_error_logs(struct iwl_fw_runtime *fwrt)488{489struct iwl_pc_data *pc_data;490u8 count;491492if (!iwl_trans_device_enabled(fwrt->trans)) {493IWL_ERR(fwrt,494"DEVICE_ENABLED bit is not set. Aborting dump.\n");495return;496}497498iwl_fwrt_dump_lmac_error_log(fwrt, 0);499if (fwrt->trans->dbg.lmac_error_event_table[1])500iwl_fwrt_dump_lmac_error_log(fwrt, 1);501iwl_fwrt_dump_umac_error_log(fwrt);502iwl_fwrt_dump_tcm_error_log(fwrt, 0);503iwl_fwrt_dump_rcm_error_log(fwrt, 0);504if (fwrt->trans->dbg.tcm_error_event_table[1])505iwl_fwrt_dump_tcm_error_log(fwrt, 1);506if (fwrt->trans->dbg.rcm_error_event_table[1])507iwl_fwrt_dump_rcm_error_log(fwrt, 1);508iwl_fwrt_dump_iml_error_log(fwrt);509iwl_fwrt_dump_fseq_regs(fwrt);510if (fwrt->trans->mac_cfg->device_family >= IWL_DEVICE_FAMILY_22000) {511pc_data = fwrt->trans->dbg.pc_data;512513if (!iwl_trans_grab_nic_access(fwrt->trans))514return;515for (count = 0; count < fwrt->trans->dbg.num_pc;516count++, pc_data++)517IWL_ERR(fwrt, "%s: 0x%x\n",518pc_data->pc_name,519iwl_read_prph_no_grab(fwrt->trans,520pc_data->pc_address));521iwl_trans_release_nic_access(fwrt->trans);522}523524if (fwrt->trans->mac_cfg->device_family >= IWL_DEVICE_FAMILY_BZ) {525u32 scratch = iwl_read32(fwrt->trans, CSR_FUNC_SCRATCH);526527IWL_ERR(fwrt, "Function Scratch status:\n");528IWL_ERR(fwrt, "0x%08X | Func Scratch\n", scratch);529}530}531IWL_EXPORT_SYMBOL(iwl_fwrt_dump_error_logs);532533bool iwl_fwrt_read_err_table(struct iwl_trans *trans, u32 base, u32 *err_id)534{535struct error_table_start {536/* cf. struct iwl_error_event_table */537u32 valid;538__le32 err_id;539} err_info = {};540int ret;541542if (err_id)543*err_id = 0;544545if (!base)546return false;547548ret = iwl_trans_read_mem_bytes(trans, base,549&err_info, sizeof(err_info));550551if (ret)552return true;553554if (err_info.valid && err_id)555*err_id = le32_to_cpu(err_info.err_id);556557return !!err_info.valid;558}559IWL_EXPORT_SYMBOL(iwl_fwrt_read_err_table);560561562