Linux Audio

Check our new training course

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