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