iwlwifi: pnvm: Fix a memory leak in 'iwl_pnvm_get_from_fs()'
[linux-2.6-microblaze.git] / drivers / net / wireless / intel / iwlwifi / fw / dump.c
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, trans->trans_cfg);
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
333 static void iwl_fwrt_dump_iml_error_log(struct iwl_fw_runtime *fwrt)
334 {
335         struct iwl_trans *trans = fwrt->trans;
336         u32 error, data1;
337
338         if (fwrt->trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_22000) {
339                 error = UMAG_SB_CPU_2_STATUS;
340                 data1 = UMAG_SB_CPU_1_STATUS;
341         } else if (fwrt->trans->trans_cfg->device_family >=
342                    IWL_DEVICE_FAMILY_8000) {
343                 error = SB_CPU_2_STATUS;
344                 data1 = SB_CPU_1_STATUS;
345         } else {
346                 return;
347         }
348
349         error = iwl_read_umac_prph(trans, UMAG_SB_CPU_2_STATUS);
350
351         IWL_ERR(trans, "IML/ROM dump:\n");
352
353         if (error & 0xFFFF0000)
354                 IWL_ERR(trans, "0x%04X | IML/ROM SYSASSERT\n", error >> 16);
355
356         IWL_ERR(fwrt, "0x%08X | IML/ROM error/state\n", error);
357         IWL_ERR(fwrt, "0x%08X | IML/ROM data1\n",
358                 iwl_read_umac_prph(trans, data1));
359
360         if (fwrt->trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_22000)
361                 IWL_ERR(fwrt, "0x%08X | IML/ROM WFPM_AUTH_KEY_0\n",
362                         iwl_read_umac_prph(trans, SB_MODIFY_CFG_FLAG));
363 }
364
365 #define FSEQ_REG(x) { .addr = (x), .str = #x, }
366
367 static void iwl_fwrt_dump_fseq_regs(struct iwl_fw_runtime *fwrt)
368 {
369         struct iwl_trans *trans = fwrt->trans;
370         int i;
371         struct {
372                 u32 addr;
373                 const char *str;
374         } fseq_regs[] = {
375                 FSEQ_REG(FSEQ_ERROR_CODE),
376                 FSEQ_REG(FSEQ_TOP_INIT_VERSION),
377                 FSEQ_REG(FSEQ_CNVIO_INIT_VERSION),
378                 FSEQ_REG(FSEQ_OTP_VERSION),
379                 FSEQ_REG(FSEQ_TOP_CONTENT_VERSION),
380                 FSEQ_REG(FSEQ_ALIVE_TOKEN),
381                 FSEQ_REG(FSEQ_CNVI_ID),
382                 FSEQ_REG(FSEQ_CNVR_ID),
383                 FSEQ_REG(CNVI_AUX_MISC_CHIP),
384                 FSEQ_REG(CNVR_AUX_MISC_CHIP),
385                 FSEQ_REG(CNVR_SCU_SD_REGS_SD_REG_DIG_DCDC_VTRIM),
386                 FSEQ_REG(CNVR_SCU_SD_REGS_SD_REG_ACTIVE_VDIG_MIRROR),
387         };
388
389         if (!iwl_trans_grab_nic_access(trans))
390                 return;
391
392         IWL_ERR(fwrt, "Fseq Registers:\n");
393
394         for (i = 0; i < ARRAY_SIZE(fseq_regs); i++)
395                 IWL_ERR(fwrt, "0x%08X | %s\n",
396                         iwl_read_prph_no_grab(trans, fseq_regs[i].addr),
397                         fseq_regs[i].str);
398
399         iwl_trans_release_nic_access(trans);
400 }
401
402 void iwl_fwrt_dump_error_logs(struct iwl_fw_runtime *fwrt)
403 {
404         if (!test_bit(STATUS_DEVICE_ENABLED, &fwrt->trans->status)) {
405                 IWL_ERR(fwrt,
406                         "DEVICE_ENABLED bit is not set. Aborting dump.\n");
407                 return;
408         }
409
410         iwl_fwrt_dump_lmac_error_log(fwrt, 0);
411         if (fwrt->trans->dbg.lmac_error_event_table[1])
412                 iwl_fwrt_dump_lmac_error_log(fwrt, 1);
413         iwl_fwrt_dump_umac_error_log(fwrt);
414         iwl_fwrt_dump_tcm_error_log(fwrt);
415         iwl_fwrt_dump_iml_error_log(fwrt);
416         iwl_fwrt_dump_fseq_regs(fwrt);
417 }
418 IWL_EXPORT_SYMBOL(iwl_fwrt_dump_error_logs);