xref: /freebsd/sys/contrib/dev/iwlwifi/fw/dump.c (revision b214fcceacad6b842545150664bd2695c1c2b34f)
1 // SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
2 /*
3  * Copyright (C) 2012-2014, 2018-2021 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 
16 /*
17  * Note: This structure is read from the device with IO accesses,
18  * and the reading already does the endian conversion. As it is
19  * read with u32-sized accesses, any members with a different size
20  * need to be ordered correctly though!
21  */
22 struct iwl_error_event_table_v1 {
23 	u32 valid;		/* (nonzero) valid, (0) log is empty */
24 	u32 error_id;		/* type of error */
25 	u32 pc;			/* program counter */
26 	u32 blink1;		/* branch link */
27 	u32 blink2;		/* branch link */
28 	u32 ilink1;		/* interrupt link */
29 	u32 ilink2;		/* interrupt link */
30 	u32 data1;		/* error-specific data */
31 	u32 data2;		/* error-specific data */
32 	u32 data3;		/* error-specific data */
33 	u32 bcon_time;		/* beacon timer */
34 	u32 tsf_low;		/* network timestamp function timer */
35 	u32 tsf_hi;		/* network timestamp function timer */
36 	u32 gp1;		/* GP1 timer register */
37 	u32 gp2;		/* GP2 timer register */
38 	u32 gp3;		/* GP3 timer register */
39 	u32 ucode_ver;		/* uCode version */
40 	u32 hw_ver;		/* HW Silicon version */
41 	u32 brd_ver;		/* HW board version */
42 	u32 log_pc;		/* log program counter */
43 	u32 frame_ptr;		/* frame pointer */
44 	u32 stack_ptr;		/* stack pointer */
45 	u32 hcmd;		/* last host command header */
46 	u32 isr0;		/* isr status register LMPM_NIC_ISR0:
47 				 * rxtx_flag */
48 	u32 isr1;		/* isr status register LMPM_NIC_ISR1:
49 				 * host_flag */
50 	u32 isr2;		/* isr status register LMPM_NIC_ISR2:
51 				 * enc_flag */
52 	u32 isr3;		/* isr status register LMPM_NIC_ISR3:
53 				 * time_flag */
54 	u32 isr4;		/* isr status register LMPM_NIC_ISR4:
55 				 * wico interrupt */
56 	u32 isr_pref;		/* isr status register LMPM_NIC_PREF_STAT */
57 	u32 wait_event;		/* wait event() caller address */
58 	u32 l2p_control;	/* L2pControlField */
59 	u32 l2p_duration;	/* L2pDurationField */
60 	u32 l2p_mhvalid;	/* L2pMhValidBits */
61 	u32 l2p_addr_match;	/* L2pAddrMatchStat */
62 	u32 lmpm_pmg_sel;	/* indicate which clocks are turned on
63 				 * (LMPM_PMG_SEL) */
64 	u32 u_timestamp;	/* indicate when the date and time of the
65 				 * compilation */
66 	u32 flow_handler;	/* FH read/write pointers, RX credit */
67 } __packed /* LOG_ERROR_TABLE_API_S_VER_1 */;
68 
69 struct iwl_error_event_table {
70 	u32 valid;		/* (nonzero) valid, (0) log is empty */
71 	u32 error_id;		/* type of error */
72 	u32 trm_hw_status0;	/* TRM HW status */
73 	u32 trm_hw_status1;	/* TRM HW status */
74 	u32 blink2;		/* branch link */
75 	u32 ilink1;		/* interrupt link */
76 	u32 ilink2;		/* interrupt link */
77 	u32 data1;		/* error-specific data */
78 	u32 data2;		/* error-specific data */
79 	u32 data3;		/* error-specific data */
80 	u32 bcon_time;		/* beacon timer */
81 	u32 tsf_low;		/* network timestamp function timer */
82 	u32 tsf_hi;		/* network timestamp function timer */
83 	u32 gp1;		/* GP1 timer register */
84 	u32 gp2;		/* GP2 timer register */
85 	u32 fw_rev_type;	/* firmware revision type */
86 	u32 major;		/* uCode version major */
87 	u32 minor;		/* uCode version minor */
88 	u32 hw_ver;		/* HW Silicon version */
89 	u32 brd_ver;		/* HW board version */
90 	u32 log_pc;		/* log program counter */
91 	u32 frame_ptr;		/* frame pointer */
92 	u32 stack_ptr;		/* stack pointer */
93 	u32 hcmd;		/* last host command header */
94 	u32 isr0;		/* isr status register LMPM_NIC_ISR0:
95 				 * rxtx_flag */
96 	u32 isr1;		/* isr status register LMPM_NIC_ISR1:
97 				 * host_flag */
98 	u32 isr2;		/* isr status register LMPM_NIC_ISR2:
99 				 * enc_flag */
100 	u32 isr3;		/* isr status register LMPM_NIC_ISR3:
101 				 * time_flag */
102 	u32 isr4;		/* isr status register LMPM_NIC_ISR4:
103 				 * wico interrupt */
104 	u32 last_cmd_id;	/* last HCMD id handled by the firmware */
105 	u32 wait_event;		/* wait event() caller address */
106 	u32 l2p_control;	/* L2pControlField */
107 	u32 l2p_duration;	/* L2pDurationField */
108 	u32 l2p_mhvalid;	/* L2pMhValidBits */
109 	u32 l2p_addr_match;	/* L2pAddrMatchStat */
110 	u32 lmpm_pmg_sel;	/* indicate which clocks are turned on
111 				 * (LMPM_PMG_SEL) */
112 	u32 u_timestamp;	/* indicate when the date and time of the
113 				 * compilation */
114 	u32 flow_handler;	/* FH read/write pointers, RX credit */
115 } __packed /* LOG_ERROR_TABLE_API_S_VER_3 */;
116 
117 /*
118  * UMAC error struct - relevant starting from family 8000 chip.
119  * Note: This structure is read from the device with IO accesses,
120  * and the reading already does the endian conversion. As it is
121  * read with u32-sized accesses, any members with a different size
122  * need to be ordered correctly though!
123  */
124 struct iwl_umac_error_event_table {
125 	u32 valid;		/* (nonzero) valid, (0) log is empty */
126 	u32 error_id;		/* type of error */
127 	u32 blink1;		/* branch link */
128 	u32 blink2;		/* branch link */
129 	u32 ilink1;		/* interrupt link */
130 	u32 ilink2;		/* interrupt link */
131 	u32 data1;		/* error-specific data */
132 	u32 data2;		/* error-specific data */
133 	u32 data3;		/* error-specific data */
134 	u32 umac_major;
135 	u32 umac_minor;
136 	u32 frame_pointer;	/* core register 27*/
137 	u32 stack_pointer;	/* core register 28 */
138 	u32 cmd_header;		/* latest host cmd sent to UMAC */
139 	u32 nic_isr_pref;	/* ISR status register */
140 } __packed;
141 
142 #define ERROR_START_OFFSET  (1 * sizeof(u32))
143 #define ERROR_ELEM_SIZE     (7 * sizeof(u32))
144 
145 static void iwl_fwrt_dump_umac_error_log(struct iwl_fw_runtime *fwrt)
146 {
147 	struct iwl_trans *trans = fwrt->trans;
148 	struct iwl_umac_error_event_table table = {};
149 	u32 base = fwrt->trans->dbg.umac_error_event_table;
150 
151 	if (!base &&
152 	    !(fwrt->trans->dbg.error_event_table_tlv_status &
153 	      IWL_ERROR_EVENT_TABLE_UMAC))
154 		return;
155 
156 	iwl_trans_read_mem_bytes(trans, base, &table, sizeof(table));
157 
158 	if (table.valid)
159 		fwrt->dump.umac_err_id = table.error_id;
160 
161 	if (ERROR_START_OFFSET <= table.valid * ERROR_ELEM_SIZE) {
162 		IWL_ERR(trans, "Start IWL Error Log Dump:\n");
163 		IWL_ERR(trans, "Transport status: 0x%08lX, valid: %d\n",
164 			fwrt->trans->status, table.valid);
165 	}
166 
167 	IWL_ERR(fwrt, "0x%08X | %s\n", table.error_id,
168 		iwl_fw_lookup_assert_desc(table.error_id));
169 	IWL_ERR(fwrt, "0x%08X | umac branchlink1\n", table.blink1);
170 	IWL_ERR(fwrt, "0x%08X | umac branchlink2\n", table.blink2);
171 	IWL_ERR(fwrt, "0x%08X | umac interruptlink1\n", table.ilink1);
172 	IWL_ERR(fwrt, "0x%08X | umac interruptlink2\n", table.ilink2);
173 	IWL_ERR(fwrt, "0x%08X | umac data1\n", table.data1);
174 	IWL_ERR(fwrt, "0x%08X | umac data2\n", table.data2);
175 	IWL_ERR(fwrt, "0x%08X | umac data3\n", table.data3);
176 	IWL_ERR(fwrt, "0x%08X | umac major\n", table.umac_major);
177 	IWL_ERR(fwrt, "0x%08X | umac minor\n", table.umac_minor);
178 	IWL_ERR(fwrt, "0x%08X | frame pointer\n", table.frame_pointer);
179 	IWL_ERR(fwrt, "0x%08X | stack pointer\n", table.stack_pointer);
180 	IWL_ERR(fwrt, "0x%08X | last host cmd\n", table.cmd_header);
181 	IWL_ERR(fwrt, "0x%08X | isr status reg\n", table.nic_isr_pref);
182 }
183 
184 static void iwl_fwrt_dump_lmac_error_log(struct iwl_fw_runtime *fwrt, u8 lmac_num)
185 {
186 	struct iwl_trans *trans = fwrt->trans;
187 	struct iwl_error_event_table table = {};
188 	u32 val, base = fwrt->trans->dbg.lmac_error_event_table[lmac_num];
189 
190 	if (fwrt->cur_fw_img == IWL_UCODE_INIT) {
191 		if (!base)
192 			base = fwrt->fw->init_errlog_ptr;
193 	} else {
194 		if (!base)
195 			base = fwrt->fw->inst_errlog_ptr;
196 	}
197 
198 	if (base < 0x400000) {
199 		IWL_ERR(fwrt,
200 			"Not valid error log pointer 0x%08X for %s uCode\n",
201 			base,
202 			(fwrt->cur_fw_img == IWL_UCODE_INIT)
203 			? "Init" : "RT");
204 		return;
205 	}
206 
207 	/* check if there is a HW error */
208 	val = iwl_trans_read_mem32(trans, base);
209 	if (((val & ~0xf) == 0xa5a5a5a0) || ((val & ~0xf) == 0x5a5a5a50)) {
210 		int err;
211 
212 		IWL_ERR(trans, "HW error, resetting before reading\n");
213 
214 		/* reset the device */
215 		iwl_trans_sw_reset(trans);
216 
217 		err = iwl_finish_nic_init(trans);
218 		if (err)
219 			return;
220 	}
221 
222 	iwl_trans_read_mem_bytes(trans, base, &table, sizeof(table));
223 
224 	if (table.valid)
225 		fwrt->dump.lmac_err_id[lmac_num] = table.error_id;
226 
227 	if (ERROR_START_OFFSET <= table.valid * ERROR_ELEM_SIZE) {
228 		IWL_ERR(trans, "Start IWL Error Log Dump:\n");
229 		IWL_ERR(trans, "Transport status: 0x%08lX, valid: %d\n",
230 			fwrt->trans->status, table.valid);
231 	}
232 
233 	/* Do not change this output - scripts rely on it */
234 
235 	IWL_ERR(fwrt, "Loaded firmware version: %s\n", fwrt->fw->fw_version);
236 
237 	IWL_ERR(fwrt, "0x%08X | %-28s\n", table.error_id,
238 		iwl_fw_lookup_assert_desc(table.error_id));
239 	IWL_ERR(fwrt, "0x%08X | trm_hw_status0\n", table.trm_hw_status0);
240 	IWL_ERR(fwrt, "0x%08X | trm_hw_status1\n", table.trm_hw_status1);
241 	IWL_ERR(fwrt, "0x%08X | branchlink2\n", table.blink2);
242 	IWL_ERR(fwrt, "0x%08X | interruptlink1\n", table.ilink1);
243 	IWL_ERR(fwrt, "0x%08X | interruptlink2\n", table.ilink2);
244 	IWL_ERR(fwrt, "0x%08X | data1\n", table.data1);
245 	IWL_ERR(fwrt, "0x%08X | data2\n", table.data2);
246 	IWL_ERR(fwrt, "0x%08X | data3\n", table.data3);
247 	IWL_ERR(fwrt, "0x%08X | beacon time\n", table.bcon_time);
248 	IWL_ERR(fwrt, "0x%08X | tsf low\n", table.tsf_low);
249 	IWL_ERR(fwrt, "0x%08X | tsf hi\n", table.tsf_hi);
250 	IWL_ERR(fwrt, "0x%08X | time gp1\n", table.gp1);
251 	IWL_ERR(fwrt, "0x%08X | time gp2\n", table.gp2);
252 	IWL_ERR(fwrt, "0x%08X | uCode revision type\n", table.fw_rev_type);
253 	IWL_ERR(fwrt, "0x%08X | uCode version major\n", table.major);
254 	IWL_ERR(fwrt, "0x%08X | uCode version minor\n", table.minor);
255 	IWL_ERR(fwrt, "0x%08X | hw version\n", table.hw_ver);
256 	IWL_ERR(fwrt, "0x%08X | board version\n", table.brd_ver);
257 	IWL_ERR(fwrt, "0x%08X | hcmd\n", table.hcmd);
258 	IWL_ERR(fwrt, "0x%08X | isr0\n", table.isr0);
259 	IWL_ERR(fwrt, "0x%08X | isr1\n", table.isr1);
260 	IWL_ERR(fwrt, "0x%08X | isr2\n", table.isr2);
261 	IWL_ERR(fwrt, "0x%08X | isr3\n", table.isr3);
262 	IWL_ERR(fwrt, "0x%08X | isr4\n", table.isr4);
263 	IWL_ERR(fwrt, "0x%08X | last cmd Id\n", table.last_cmd_id);
264 	IWL_ERR(fwrt, "0x%08X | wait_event\n", table.wait_event);
265 	IWL_ERR(fwrt, "0x%08X | l2p_control\n", table.l2p_control);
266 	IWL_ERR(fwrt, "0x%08X | l2p_duration\n", table.l2p_duration);
267 	IWL_ERR(fwrt, "0x%08X | l2p_mhvalid\n", table.l2p_mhvalid);
268 	IWL_ERR(fwrt, "0x%08X | l2p_addr_match\n", table.l2p_addr_match);
269 	IWL_ERR(fwrt, "0x%08X | lmpm_pmg_sel\n", table.lmpm_pmg_sel);
270 	IWL_ERR(fwrt, "0x%08X | timestamp\n", table.u_timestamp);
271 	IWL_ERR(fwrt, "0x%08X | flow_handler\n", table.flow_handler);
272 }
273 
274 /*
275  * TCM error struct.
276  * Note: This structure is read from the device with IO accesses,
277  * and the reading already does the endian conversion. As it is
278  * read with u32-sized accesses, any members with a different size
279  * need to be ordered correctly though!
280  */
281 struct iwl_tcm_error_event_table {
282 	u32 valid;
283 	u32 error_id;
284 	u32 blink2;
285 	u32 ilink1;
286 	u32 ilink2;
287 	u32 data1, data2, data3;
288 	u32 logpc;
289 	u32 frame_pointer;
290 	u32 stack_pointer;
291 	u32 msgid;
292 	u32 isr;
293 	u32 hw_status[5];
294 	u32 sw_status[1];
295 	u32 reserved[4];
296 } __packed; /* TCM_LOG_ERROR_TABLE_API_S_VER_1 */
297 
298 static void iwl_fwrt_dump_tcm_error_log(struct iwl_fw_runtime *fwrt)
299 {
300 	struct iwl_trans *trans = fwrt->trans;
301 	struct iwl_tcm_error_event_table table = {};
302 	u32 base = fwrt->trans->dbg.tcm_error_event_table;
303 	int i;
304 
305 	if (!base ||
306 	    !(fwrt->trans->dbg.error_event_table_tlv_status &
307 	      IWL_ERROR_EVENT_TABLE_TCM))
308 		return;
309 
310 	iwl_trans_read_mem_bytes(trans, base, &table, sizeof(table));
311 
312 	IWL_ERR(fwrt, "TCM status:\n");
313 	IWL_ERR(fwrt, "0x%08X | error ID\n", table.error_id);
314 	IWL_ERR(fwrt, "0x%08X | tcm branchlink2\n", table.blink2);
315 	IWL_ERR(fwrt, "0x%08X | tcm interruptlink1\n", table.ilink1);
316 	IWL_ERR(fwrt, "0x%08X | tcm interruptlink2\n", table.ilink2);
317 	IWL_ERR(fwrt, "0x%08X | tcm data1\n", table.data1);
318 	IWL_ERR(fwrt, "0x%08X | tcm data2\n", table.data2);
319 	IWL_ERR(fwrt, "0x%08X | tcm data3\n", table.data3);
320 	IWL_ERR(fwrt, "0x%08X | tcm log PC\n", table.logpc);
321 	IWL_ERR(fwrt, "0x%08X | tcm frame pointer\n", table.frame_pointer);
322 	IWL_ERR(fwrt, "0x%08X | tcm stack pointer\n", table.stack_pointer);
323 	IWL_ERR(fwrt, "0x%08X | tcm msg ID\n", table.msgid);
324 	IWL_ERR(fwrt, "0x%08X | tcm ISR status\n", table.isr);
325 	for (i = 0; i < ARRAY_SIZE(table.hw_status); i++)
326 		IWL_ERR(fwrt, "0x%08X | tcm HW status[%d]\n",
327 			table.hw_status[i], i);
328 	for (i = 0; i < ARRAY_SIZE(table.sw_status); i++)
329 		IWL_ERR(fwrt, "0x%08X | tcm SW status[%d]\n",
330 			table.sw_status[i], i);
331 
332 	if (trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_BZ) {
333 		u32 scratch = iwl_read32(trans, CSR_FUNC_SCRATCH);
334 
335 		IWL_ERR(fwrt, "Function Scratch status:\n");
336 		IWL_ERR(fwrt, "0x%08X | Func Scratch\n", scratch);
337 	}
338 }
339 
340 static void iwl_fwrt_dump_iml_error_log(struct iwl_fw_runtime *fwrt)
341 {
342 	struct iwl_trans *trans = fwrt->trans;
343 	u32 error, data1;
344 
345 	if (fwrt->trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_22000) {
346 		error = UMAG_SB_CPU_2_STATUS;
347 		data1 = UMAG_SB_CPU_1_STATUS;
348 	} else if (fwrt->trans->trans_cfg->device_family >=
349 		   IWL_DEVICE_FAMILY_8000) {
350 		error = SB_CPU_2_STATUS;
351 		data1 = SB_CPU_1_STATUS;
352 	} else {
353 		return;
354 	}
355 
356 	error = iwl_read_umac_prph(trans, UMAG_SB_CPU_2_STATUS);
357 
358 	IWL_ERR(trans, "IML/ROM dump:\n");
359 
360 	if (error & 0xFFFF0000)
361 		IWL_ERR(trans, "0x%04X | IML/ROM SYSASSERT\n", error >> 16);
362 
363 	IWL_ERR(fwrt, "0x%08X | IML/ROM error/state\n", error);
364 	IWL_ERR(fwrt, "0x%08X | IML/ROM data1\n",
365 		iwl_read_umac_prph(trans, data1));
366 
367 	if (fwrt->trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_22000)
368 		IWL_ERR(fwrt, "0x%08X | IML/ROM WFPM_AUTH_KEY_0\n",
369 			iwl_read_umac_prph(trans, SB_MODIFY_CFG_FLAG));
370 }
371 
372 #define FSEQ_REG(x) { .addr = (x), .str = #x, }
373 
374 static void iwl_fwrt_dump_fseq_regs(struct iwl_fw_runtime *fwrt)
375 {
376 	struct iwl_trans *trans = fwrt->trans;
377 	int i;
378 	struct {
379 		u32 addr;
380 		const char *str;
381 	} fseq_regs[] = {
382 		FSEQ_REG(FSEQ_ERROR_CODE),
383 		FSEQ_REG(FSEQ_TOP_INIT_VERSION),
384 		FSEQ_REG(FSEQ_CNVIO_INIT_VERSION),
385 		FSEQ_REG(FSEQ_OTP_VERSION),
386 		FSEQ_REG(FSEQ_TOP_CONTENT_VERSION),
387 		FSEQ_REG(FSEQ_ALIVE_TOKEN),
388 		FSEQ_REG(FSEQ_CNVI_ID),
389 		FSEQ_REG(FSEQ_CNVR_ID),
390 		FSEQ_REG(CNVI_AUX_MISC_CHIP),
391 		FSEQ_REG(CNVR_AUX_MISC_CHIP),
392 		FSEQ_REG(CNVR_SCU_SD_REGS_SD_REG_DIG_DCDC_VTRIM),
393 		FSEQ_REG(CNVR_SCU_SD_REGS_SD_REG_ACTIVE_VDIG_MIRROR),
394 	};
395 
396 	if (!iwl_trans_grab_nic_access(trans))
397 		return;
398 
399 	IWL_ERR(fwrt, "Fseq Registers:\n");
400 
401 	for (i = 0; i < ARRAY_SIZE(fseq_regs); i++)
402 		IWL_ERR(fwrt, "0x%08X | %s\n",
403 			iwl_read_prph_no_grab(trans, fseq_regs[i].addr),
404 			fseq_regs[i].str);
405 
406 	iwl_trans_release_nic_access(trans);
407 }
408 
409 void iwl_fwrt_dump_error_logs(struct iwl_fw_runtime *fwrt)
410 {
411 	if (!test_bit(STATUS_DEVICE_ENABLED, &fwrt->trans->status)) {
412 		IWL_ERR(fwrt,
413 			"DEVICE_ENABLED bit is not set. Aborting dump.\n");
414 		return;
415 	}
416 
417 	iwl_fwrt_dump_lmac_error_log(fwrt, 0);
418 	if (fwrt->trans->dbg.lmac_error_event_table[1])
419 		iwl_fwrt_dump_lmac_error_log(fwrt, 1);
420 	iwl_fwrt_dump_umac_error_log(fwrt);
421 	iwl_fwrt_dump_tcm_error_log(fwrt);
422 	iwl_fwrt_dump_iml_error_log(fwrt);
423 	iwl_fwrt_dump_fseq_regs(fwrt);
424 }
425 IWL_EXPORT_SYMBOL(iwl_fwrt_dump_error_logs);
426