xref: /freebsd/sys/contrib/dev/iwlwifi/fw/dump.c (revision a4128aad8503277614f2d214011ef60a19447b83)
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