mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2025-01-19 20:36:14 +07:00
79f25b10c9
We can dump data from the firmware either when it crashes,
or when the firmware is alive.
Not all the data is available if the firmware is running
(like the Tx / Rx FIFOs which are available only when the
firmware is halted), so we first check that the firmware
is alive to compute the required size for the dump and then
fill the buffer with the data.
When we allocate the buffer, we test the STATUS_FW_ERROR
bit to check if the firmware is alive or not. This bit
can be changed during the course of the dump since it is
modified in the interrupt handler.
We hit a case where we allocate the buffer while the
firmware is sill working, and while we start to fill the
buffer, the firmware crashes. Then we test STATUS_FW_ERROR
again and decide to fill the buffer with data like the
FIFOs even if no room was allocated for this data in the
buffer. This means that we overflow the buffer that was
allocated leading to memory corruption.
To fix this, test the STATUS_FW_ERROR bit only once and
rely on local variables to check if we should dump fifos
or other firmware components.
Fixes: 04fd2c2822
("iwlwifi: mvm: add rxf and txf to dump data")
Signed-off-by: Emmanuel Grumbach <emmanuel.grumbach@intel.com>
Signed-off-by: Luca Coelho <luciano.coelho@intel.com>
1219 lines
38 KiB
C
1219 lines
38 KiB
C
/******************************************************************************
|
|
*
|
|
* This file is provided under a dual BSD/GPLv2 license. When using or
|
|
* redistributing this file, you may do so under either license.
|
|
*
|
|
* GPL LICENSE SUMMARY
|
|
*
|
|
* Copyright(c) 2008 - 2014 Intel Corporation. All rights reserved.
|
|
* Copyright(c) 2013 - 2015 Intel Mobile Communications GmbH
|
|
* Copyright(c) 2015 - 2017 Intel Deutschland GmbH
|
|
* Copyright(c) 2018 Intel Corporation
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of version 2 of the GNU General Public License as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*
|
|
* The full GNU General Public License is included in this distribution
|
|
* in the file called COPYING.
|
|
*
|
|
* Contact Information:
|
|
* Intel Linux Wireless <linuxwifi@intel.com>
|
|
* Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
|
|
*
|
|
* BSD LICENSE
|
|
*
|
|
* Copyright(c) 2005 - 2014 Intel Corporation. All rights reserved.
|
|
* Copyright(c) 2013 - 2015 Intel Mobile Communications GmbH
|
|
* Copyright(c) 2015 - 2017 Intel Deutschland GmbH
|
|
* Copyright(c) 2018 Intel Corporation
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* * Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* * Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in
|
|
* the documentation and/or other materials provided with the
|
|
* distribution.
|
|
* * Neither the name Intel Corporation nor the names of its
|
|
* contributors may be used to endorse or promote products derived
|
|
* from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
*****************************************************************************/
|
|
#include <linux/devcoredump.h>
|
|
#include "iwl-drv.h"
|
|
#include "runtime.h"
|
|
#include "dbg.h"
|
|
#include "debugfs.h"
|
|
#include "iwl-io.h"
|
|
#include "iwl-prph.h"
|
|
#include "iwl-csr.h"
|
|
|
|
/**
|
|
* struct iwl_fw_dump_ptrs - set of pointers needed for the fw-error-dump
|
|
*
|
|
* @fwrt_ptr: pointer to the buffer coming from fwrt
|
|
* @trans_ptr: pointer to struct %iwl_trans_dump_data which contains the
|
|
* transport's data.
|
|
* @trans_len: length of the valid data in trans_ptr
|
|
* @fwrt_len: length of the valid data in fwrt_ptr
|
|
*/
|
|
struct iwl_fw_dump_ptrs {
|
|
struct iwl_trans_dump_data *trans_ptr;
|
|
void *fwrt_ptr;
|
|
u32 fwrt_len;
|
|
};
|
|
|
|
#define RADIO_REG_MAX_READ 0x2ad
|
|
static void iwl_read_radio_regs(struct iwl_fw_runtime *fwrt,
|
|
struct iwl_fw_error_dump_data **dump_data)
|
|
{
|
|
u8 *pos = (void *)(*dump_data)->data;
|
|
unsigned long flags;
|
|
int i;
|
|
|
|
IWL_DEBUG_INFO(fwrt, "WRT radio registers dump\n");
|
|
|
|
if (!iwl_trans_grab_nic_access(fwrt->trans, &flags))
|
|
return;
|
|
|
|
(*dump_data)->type = cpu_to_le32(IWL_FW_ERROR_DUMP_RADIO_REG);
|
|
(*dump_data)->len = cpu_to_le32(RADIO_REG_MAX_READ);
|
|
|
|
for (i = 0; i < RADIO_REG_MAX_READ; i++) {
|
|
u32 rd_cmd = RADIO_RSP_RD_CMD;
|
|
|
|
rd_cmd |= i << RADIO_RSP_ADDR_POS;
|
|
iwl_write_prph_no_grab(fwrt->trans, RSP_RADIO_CMD, rd_cmd);
|
|
*pos = (u8)iwl_read_prph_no_grab(fwrt->trans, RSP_RADIO_RDDAT);
|
|
|
|
pos++;
|
|
}
|
|
|
|
*dump_data = iwl_fw_error_next_data(*dump_data);
|
|
|
|
iwl_trans_release_nic_access(fwrt->trans, &flags);
|
|
}
|
|
|
|
static void iwl_fwrt_dump_rxf(struct iwl_fw_runtime *fwrt,
|
|
struct iwl_fw_error_dump_data **dump_data,
|
|
int size, u32 offset, int fifo_num)
|
|
{
|
|
struct iwl_fw_error_dump_fifo *fifo_hdr;
|
|
u32 *fifo_data;
|
|
u32 fifo_len;
|
|
int i;
|
|
|
|
fifo_hdr = (void *)(*dump_data)->data;
|
|
fifo_data = (void *)fifo_hdr->data;
|
|
fifo_len = size;
|
|
|
|
/* No need to try to read the data if the length is 0 */
|
|
if (fifo_len == 0)
|
|
return;
|
|
|
|
/* Add a TLV for the RXF */
|
|
(*dump_data)->type = cpu_to_le32(IWL_FW_ERROR_DUMP_RXF);
|
|
(*dump_data)->len = cpu_to_le32(fifo_len + sizeof(*fifo_hdr));
|
|
|
|
fifo_hdr->fifo_num = cpu_to_le32(fifo_num);
|
|
fifo_hdr->available_bytes =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
RXF_RD_D_SPACE + offset));
|
|
fifo_hdr->wr_ptr =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
RXF_RD_WR_PTR + offset));
|
|
fifo_hdr->rd_ptr =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
RXF_RD_RD_PTR + offset));
|
|
fifo_hdr->fence_ptr =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
RXF_RD_FENCE_PTR + offset));
|
|
fifo_hdr->fence_mode =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
RXF_SET_FENCE_MODE + offset));
|
|
|
|
/* Lock fence */
|
|
iwl_trans_write_prph(fwrt->trans, RXF_SET_FENCE_MODE + offset, 0x1);
|
|
/* Set fence pointer to the same place like WR pointer */
|
|
iwl_trans_write_prph(fwrt->trans, RXF_LD_WR2FENCE + offset, 0x1);
|
|
/* Set fence offset */
|
|
iwl_trans_write_prph(fwrt->trans,
|
|
RXF_LD_FENCE_OFFSET_ADDR + offset, 0x0);
|
|
|
|
/* Read FIFO */
|
|
fifo_len /= sizeof(u32); /* Size in DWORDS */
|
|
for (i = 0; i < fifo_len; i++)
|
|
fifo_data[i] = iwl_trans_read_prph(fwrt->trans,
|
|
RXF_FIFO_RD_FENCE_INC +
|
|
offset);
|
|
*dump_data = iwl_fw_error_next_data(*dump_data);
|
|
}
|
|
|
|
static void iwl_fwrt_dump_txf(struct iwl_fw_runtime *fwrt,
|
|
struct iwl_fw_error_dump_data **dump_data,
|
|
int size, u32 offset, int fifo_num)
|
|
{
|
|
struct iwl_fw_error_dump_fifo *fifo_hdr;
|
|
u32 *fifo_data;
|
|
u32 fifo_len;
|
|
int i;
|
|
|
|
fifo_hdr = (void *)(*dump_data)->data;
|
|
fifo_data = (void *)fifo_hdr->data;
|
|
fifo_len = size;
|
|
|
|
/* No need to try to read the data if the length is 0 */
|
|
if (fifo_len == 0)
|
|
return;
|
|
|
|
/* Add a TLV for the FIFO */
|
|
(*dump_data)->type = cpu_to_le32(IWL_FW_ERROR_DUMP_TXF);
|
|
(*dump_data)->len = cpu_to_le32(fifo_len + sizeof(*fifo_hdr));
|
|
|
|
fifo_hdr->fifo_num = cpu_to_le32(fifo_num);
|
|
fifo_hdr->available_bytes =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
TXF_FIFO_ITEM_CNT + offset));
|
|
fifo_hdr->wr_ptr =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
TXF_WR_PTR + offset));
|
|
fifo_hdr->rd_ptr =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
TXF_RD_PTR + offset));
|
|
fifo_hdr->fence_ptr =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
TXF_FENCE_PTR + offset));
|
|
fifo_hdr->fence_mode =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
TXF_LOCK_FENCE + offset));
|
|
|
|
/* Set the TXF_READ_MODIFY_ADDR to TXF_WR_PTR */
|
|
iwl_trans_write_prph(fwrt->trans, TXF_READ_MODIFY_ADDR + offset,
|
|
TXF_WR_PTR + offset);
|
|
|
|
/* Dummy-read to advance the read pointer to the head */
|
|
iwl_trans_read_prph(fwrt->trans, TXF_READ_MODIFY_DATA + offset);
|
|
|
|
/* Read FIFO */
|
|
fifo_len /= sizeof(u32); /* Size in DWORDS */
|
|
for (i = 0; i < fifo_len; i++)
|
|
fifo_data[i] = iwl_trans_read_prph(fwrt->trans,
|
|
TXF_READ_MODIFY_DATA +
|
|
offset);
|
|
*dump_data = iwl_fw_error_next_data(*dump_data);
|
|
}
|
|
|
|
static void iwl_fw_dump_fifos(struct iwl_fw_runtime *fwrt,
|
|
struct iwl_fw_error_dump_data **dump_data)
|
|
{
|
|
struct iwl_fw_error_dump_fifo *fifo_hdr;
|
|
struct iwl_fwrt_shared_mem_cfg *cfg = &fwrt->smem_cfg;
|
|
u32 *fifo_data;
|
|
u32 fifo_len;
|
|
unsigned long flags;
|
|
int i, j;
|
|
|
|
IWL_DEBUG_INFO(fwrt, "WRT FIFO dump\n");
|
|
|
|
if (!iwl_trans_grab_nic_access(fwrt->trans, &flags))
|
|
return;
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_RXF)) {
|
|
/* Pull RXF1 */
|
|
iwl_fwrt_dump_rxf(fwrt, dump_data,
|
|
cfg->lmac[0].rxfifo1_size, 0, 0);
|
|
/* Pull RXF2 */
|
|
iwl_fwrt_dump_rxf(fwrt, dump_data, cfg->rxfifo2_size,
|
|
RXF_DIFF_FROM_PREV, 1);
|
|
/* Pull LMAC2 RXF1 */
|
|
if (fwrt->smem_cfg.num_lmacs > 1)
|
|
iwl_fwrt_dump_rxf(fwrt, dump_data,
|
|
cfg->lmac[1].rxfifo1_size,
|
|
LMAC2_PRPH_OFFSET, 2);
|
|
}
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_TXF)) {
|
|
/* Pull TXF data from LMAC1 */
|
|
for (i = 0; i < fwrt->smem_cfg.num_txfifo_entries; i++) {
|
|
/* Mark the number of TXF we're pulling now */
|
|
iwl_trans_write_prph(fwrt->trans, TXF_LARC_NUM, i);
|
|
iwl_fwrt_dump_txf(fwrt, dump_data,
|
|
cfg->lmac[0].txfifo_size[i], 0, i);
|
|
}
|
|
|
|
/* Pull TXF data from LMAC2 */
|
|
if (fwrt->smem_cfg.num_lmacs > 1) {
|
|
for (i = 0; i < fwrt->smem_cfg.num_txfifo_entries;
|
|
i++) {
|
|
/* Mark the number of TXF we're pulling now */
|
|
iwl_trans_write_prph(fwrt->trans,
|
|
TXF_LARC_NUM +
|
|
LMAC2_PRPH_OFFSET, i);
|
|
iwl_fwrt_dump_txf(fwrt, dump_data,
|
|
cfg->lmac[1].txfifo_size[i],
|
|
LMAC2_PRPH_OFFSET,
|
|
i + cfg->num_txfifo_entries);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_INTERNAL_TXF) &&
|
|
fw_has_capa(&fwrt->fw->ucode_capa,
|
|
IWL_UCODE_TLV_CAPA_EXTEND_SHARED_MEM_CFG)) {
|
|
/* Pull UMAC internal TXF data from all TXFs */
|
|
for (i = 0;
|
|
i < ARRAY_SIZE(fwrt->smem_cfg.internal_txfifo_size);
|
|
i++) {
|
|
fifo_hdr = (void *)(*dump_data)->data;
|
|
fifo_data = (void *)fifo_hdr->data;
|
|
fifo_len = fwrt->smem_cfg.internal_txfifo_size[i];
|
|
|
|
/* No need to try to read the data if the length is 0 */
|
|
if (fifo_len == 0)
|
|
continue;
|
|
|
|
/* Add a TLV for the internal FIFOs */
|
|
(*dump_data)->type =
|
|
cpu_to_le32(IWL_FW_ERROR_DUMP_INTERNAL_TXF);
|
|
(*dump_data)->len =
|
|
cpu_to_le32(fifo_len + sizeof(*fifo_hdr));
|
|
|
|
fifo_hdr->fifo_num = cpu_to_le32(i);
|
|
|
|
/* Mark the number of TXF we're pulling now */
|
|
iwl_trans_write_prph(fwrt->trans, TXF_CPU2_NUM, i +
|
|
fwrt->smem_cfg.num_txfifo_entries);
|
|
|
|
fifo_hdr->available_bytes =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
TXF_CPU2_FIFO_ITEM_CNT));
|
|
fifo_hdr->wr_ptr =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
TXF_CPU2_WR_PTR));
|
|
fifo_hdr->rd_ptr =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
TXF_CPU2_RD_PTR));
|
|
fifo_hdr->fence_ptr =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
TXF_CPU2_FENCE_PTR));
|
|
fifo_hdr->fence_mode =
|
|
cpu_to_le32(iwl_trans_read_prph(fwrt->trans,
|
|
TXF_CPU2_LOCK_FENCE));
|
|
|
|
/* Set TXF_CPU2_READ_MODIFY_ADDR to TXF_CPU2_WR_PTR */
|
|
iwl_trans_write_prph(fwrt->trans,
|
|
TXF_CPU2_READ_MODIFY_ADDR,
|
|
TXF_CPU2_WR_PTR);
|
|
|
|
/* Dummy-read to advance the read pointer to head */
|
|
iwl_trans_read_prph(fwrt->trans,
|
|
TXF_CPU2_READ_MODIFY_DATA);
|
|
|
|
/* Read FIFO */
|
|
fifo_len /= sizeof(u32); /* Size in DWORDS */
|
|
for (j = 0; j < fifo_len; j++)
|
|
fifo_data[j] =
|
|
iwl_trans_read_prph(fwrt->trans,
|
|
TXF_CPU2_READ_MODIFY_DATA);
|
|
*dump_data = iwl_fw_error_next_data(*dump_data);
|
|
}
|
|
}
|
|
|
|
iwl_trans_release_nic_access(fwrt->trans, &flags);
|
|
}
|
|
|
|
#define IWL8260_ICCM_OFFSET 0x44000 /* Only for B-step */
|
|
#define IWL8260_ICCM_LEN 0xC000 /* Only for B-step */
|
|
|
|
struct iwl_prph_range {
|
|
u32 start, end;
|
|
};
|
|
|
|
static const struct iwl_prph_range iwl_prph_dump_addr_comm[] = {
|
|
{ .start = 0x00a00000, .end = 0x00a00000 },
|
|
{ .start = 0x00a0000c, .end = 0x00a00024 },
|
|
{ .start = 0x00a0002c, .end = 0x00a0003c },
|
|
{ .start = 0x00a00410, .end = 0x00a00418 },
|
|
{ .start = 0x00a00420, .end = 0x00a00420 },
|
|
{ .start = 0x00a00428, .end = 0x00a00428 },
|
|
{ .start = 0x00a00430, .end = 0x00a0043c },
|
|
{ .start = 0x00a00444, .end = 0x00a00444 },
|
|
{ .start = 0x00a004c0, .end = 0x00a004cc },
|
|
{ .start = 0x00a004d8, .end = 0x00a004d8 },
|
|
{ .start = 0x00a004e0, .end = 0x00a004f0 },
|
|
{ .start = 0x00a00840, .end = 0x00a00840 },
|
|
{ .start = 0x00a00850, .end = 0x00a00858 },
|
|
{ .start = 0x00a01004, .end = 0x00a01008 },
|
|
{ .start = 0x00a01010, .end = 0x00a01010 },
|
|
{ .start = 0x00a01018, .end = 0x00a01018 },
|
|
{ .start = 0x00a01024, .end = 0x00a01024 },
|
|
{ .start = 0x00a0102c, .end = 0x00a01034 },
|
|
{ .start = 0x00a0103c, .end = 0x00a01040 },
|
|
{ .start = 0x00a01048, .end = 0x00a01094 },
|
|
{ .start = 0x00a01c00, .end = 0x00a01c20 },
|
|
{ .start = 0x00a01c58, .end = 0x00a01c58 },
|
|
{ .start = 0x00a01c7c, .end = 0x00a01c7c },
|
|
{ .start = 0x00a01c28, .end = 0x00a01c54 },
|
|
{ .start = 0x00a01c5c, .end = 0x00a01c5c },
|
|
{ .start = 0x00a01c60, .end = 0x00a01cdc },
|
|
{ .start = 0x00a01ce0, .end = 0x00a01d0c },
|
|
{ .start = 0x00a01d18, .end = 0x00a01d20 },
|
|
{ .start = 0x00a01d2c, .end = 0x00a01d30 },
|
|
{ .start = 0x00a01d40, .end = 0x00a01d5c },
|
|
{ .start = 0x00a01d80, .end = 0x00a01d80 },
|
|
{ .start = 0x00a01d98, .end = 0x00a01d9c },
|
|
{ .start = 0x00a01da8, .end = 0x00a01da8 },
|
|
{ .start = 0x00a01db8, .end = 0x00a01df4 },
|
|
{ .start = 0x00a01dc0, .end = 0x00a01dfc },
|
|
{ .start = 0x00a01e00, .end = 0x00a01e2c },
|
|
{ .start = 0x00a01e40, .end = 0x00a01e60 },
|
|
{ .start = 0x00a01e68, .end = 0x00a01e6c },
|
|
{ .start = 0x00a01e74, .end = 0x00a01e74 },
|
|
{ .start = 0x00a01e84, .end = 0x00a01e90 },
|
|
{ .start = 0x00a01e9c, .end = 0x00a01ec4 },
|
|
{ .start = 0x00a01ed0, .end = 0x00a01ee0 },
|
|
{ .start = 0x00a01f00, .end = 0x00a01f1c },
|
|
{ .start = 0x00a01f44, .end = 0x00a01ffc },
|
|
{ .start = 0x00a02000, .end = 0x00a02048 },
|
|
{ .start = 0x00a02068, .end = 0x00a020f0 },
|
|
{ .start = 0x00a02100, .end = 0x00a02118 },
|
|
{ .start = 0x00a02140, .end = 0x00a0214c },
|
|
{ .start = 0x00a02168, .end = 0x00a0218c },
|
|
{ .start = 0x00a021c0, .end = 0x00a021c0 },
|
|
{ .start = 0x00a02400, .end = 0x00a02410 },
|
|
{ .start = 0x00a02418, .end = 0x00a02420 },
|
|
{ .start = 0x00a02428, .end = 0x00a0242c },
|
|
{ .start = 0x00a02434, .end = 0x00a02434 },
|
|
{ .start = 0x00a02440, .end = 0x00a02460 },
|
|
{ .start = 0x00a02468, .end = 0x00a024b0 },
|
|
{ .start = 0x00a024c8, .end = 0x00a024cc },
|
|
{ .start = 0x00a02500, .end = 0x00a02504 },
|
|
{ .start = 0x00a0250c, .end = 0x00a02510 },
|
|
{ .start = 0x00a02540, .end = 0x00a02554 },
|
|
{ .start = 0x00a02580, .end = 0x00a025f4 },
|
|
{ .start = 0x00a02600, .end = 0x00a0260c },
|
|
{ .start = 0x00a02648, .end = 0x00a02650 },
|
|
{ .start = 0x00a02680, .end = 0x00a02680 },
|
|
{ .start = 0x00a026c0, .end = 0x00a026d0 },
|
|
{ .start = 0x00a02700, .end = 0x00a0270c },
|
|
{ .start = 0x00a02804, .end = 0x00a02804 },
|
|
{ .start = 0x00a02818, .end = 0x00a0281c },
|
|
{ .start = 0x00a02c00, .end = 0x00a02db4 },
|
|
{ .start = 0x00a02df4, .end = 0x00a02fb0 },
|
|
{ .start = 0x00a03000, .end = 0x00a03014 },
|
|
{ .start = 0x00a0301c, .end = 0x00a0302c },
|
|
{ .start = 0x00a03034, .end = 0x00a03038 },
|
|
{ .start = 0x00a03040, .end = 0x00a03048 },
|
|
{ .start = 0x00a03060, .end = 0x00a03068 },
|
|
{ .start = 0x00a03070, .end = 0x00a03074 },
|
|
{ .start = 0x00a0307c, .end = 0x00a0307c },
|
|
{ .start = 0x00a03080, .end = 0x00a03084 },
|
|
{ .start = 0x00a0308c, .end = 0x00a03090 },
|
|
{ .start = 0x00a03098, .end = 0x00a03098 },
|
|
{ .start = 0x00a030a0, .end = 0x00a030a0 },
|
|
{ .start = 0x00a030a8, .end = 0x00a030b4 },
|
|
{ .start = 0x00a030bc, .end = 0x00a030bc },
|
|
{ .start = 0x00a030c0, .end = 0x00a0312c },
|
|
{ .start = 0x00a03c00, .end = 0x00a03c5c },
|
|
{ .start = 0x00a04400, .end = 0x00a04454 },
|
|
{ .start = 0x00a04460, .end = 0x00a04474 },
|
|
{ .start = 0x00a044c0, .end = 0x00a044ec },
|
|
{ .start = 0x00a04500, .end = 0x00a04504 },
|
|
{ .start = 0x00a04510, .end = 0x00a04538 },
|
|
{ .start = 0x00a04540, .end = 0x00a04548 },
|
|
{ .start = 0x00a04560, .end = 0x00a0457c },
|
|
{ .start = 0x00a04590, .end = 0x00a04598 },
|
|
{ .start = 0x00a045c0, .end = 0x00a045f4 },
|
|
};
|
|
|
|
static const struct iwl_prph_range iwl_prph_dump_addr_9000[] = {
|
|
{ .start = 0x00a05c00, .end = 0x00a05c18 },
|
|
{ .start = 0x00a05400, .end = 0x00a056e8 },
|
|
{ .start = 0x00a08000, .end = 0x00a098bc },
|
|
{ .start = 0x00a02400, .end = 0x00a02758 },
|
|
};
|
|
|
|
static void iwl_read_prph_block(struct iwl_trans *trans, u32 start,
|
|
u32 len_bytes, __le32 *data)
|
|
{
|
|
u32 i;
|
|
|
|
for (i = 0; i < len_bytes; i += 4)
|
|
*data++ = cpu_to_le32(iwl_read_prph_no_grab(trans, start + i));
|
|
}
|
|
|
|
static void iwl_dump_prph(struct iwl_trans *trans,
|
|
struct iwl_fw_error_dump_data **data,
|
|
const struct iwl_prph_range *iwl_prph_dump_addr,
|
|
u32 range_len)
|
|
{
|
|
struct iwl_fw_error_dump_prph *prph;
|
|
unsigned long flags;
|
|
u32 i;
|
|
|
|
IWL_DEBUG_INFO(trans, "WRT PRPH dump\n");
|
|
|
|
if (!iwl_trans_grab_nic_access(trans, &flags))
|
|
return;
|
|
|
|
for (i = 0; i < range_len; i++) {
|
|
/* The range includes both boundaries */
|
|
int num_bytes_in_chunk = iwl_prph_dump_addr[i].end -
|
|
iwl_prph_dump_addr[i].start + 4;
|
|
|
|
(*data)->type = cpu_to_le32(IWL_FW_ERROR_DUMP_PRPH);
|
|
(*data)->len = cpu_to_le32(sizeof(*prph) +
|
|
num_bytes_in_chunk);
|
|
prph = (void *)(*data)->data;
|
|
prph->prph_start = cpu_to_le32(iwl_prph_dump_addr[i].start);
|
|
|
|
iwl_read_prph_block(trans, iwl_prph_dump_addr[i].start,
|
|
/* our range is inclusive, hence + 4 */
|
|
iwl_prph_dump_addr[i].end -
|
|
iwl_prph_dump_addr[i].start + 4,
|
|
(void *)prph->data);
|
|
|
|
*data = iwl_fw_error_next_data(*data);
|
|
}
|
|
|
|
iwl_trans_release_nic_access(trans, &flags);
|
|
}
|
|
|
|
/*
|
|
* alloc_sgtable - allocates scallerlist table in the given size,
|
|
* fills it with pages and returns it
|
|
* @size: the size (in bytes) of the table
|
|
*/
|
|
static struct scatterlist *alloc_sgtable(int size)
|
|
{
|
|
int alloc_size, nents, i;
|
|
struct page *new_page;
|
|
struct scatterlist *iter;
|
|
struct scatterlist *table;
|
|
|
|
nents = DIV_ROUND_UP(size, PAGE_SIZE);
|
|
table = kcalloc(nents, sizeof(*table), GFP_KERNEL);
|
|
if (!table)
|
|
return NULL;
|
|
sg_init_table(table, nents);
|
|
iter = table;
|
|
for_each_sg(table, iter, sg_nents(table), i) {
|
|
new_page = alloc_page(GFP_KERNEL);
|
|
if (!new_page) {
|
|
/* release all previous allocated pages in the table */
|
|
iter = table;
|
|
for_each_sg(table, iter, sg_nents(table), i) {
|
|
new_page = sg_page(iter);
|
|
if (new_page)
|
|
__free_page(new_page);
|
|
}
|
|
return NULL;
|
|
}
|
|
alloc_size = min_t(int, size, PAGE_SIZE);
|
|
size -= PAGE_SIZE;
|
|
sg_set_page(iter, new_page, alloc_size, 0);
|
|
}
|
|
return table;
|
|
}
|
|
|
|
static int iwl_fw_get_prph_len(struct iwl_fw_runtime *fwrt)
|
|
{
|
|
u32 prph_len = 0;
|
|
int i;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(iwl_prph_dump_addr_comm);
|
|
i++) {
|
|
/* The range includes both boundaries */
|
|
int num_bytes_in_chunk =
|
|
iwl_prph_dump_addr_comm[i].end -
|
|
iwl_prph_dump_addr_comm[i].start + 4;
|
|
|
|
prph_len += sizeof(struct iwl_fw_error_dump_data) +
|
|
sizeof(struct iwl_fw_error_dump_prph) +
|
|
num_bytes_in_chunk;
|
|
}
|
|
|
|
if (fwrt->trans->cfg->mq_rx_supported) {
|
|
for (i = 0; i <
|
|
ARRAY_SIZE(iwl_prph_dump_addr_9000); i++) {
|
|
/* The range includes both boundaries */
|
|
int num_bytes_in_chunk =
|
|
iwl_prph_dump_addr_9000[i].end -
|
|
iwl_prph_dump_addr_9000[i].start + 4;
|
|
|
|
prph_len += sizeof(struct iwl_fw_error_dump_data) +
|
|
sizeof(struct iwl_fw_error_dump_prph) +
|
|
num_bytes_in_chunk;
|
|
}
|
|
}
|
|
return prph_len;
|
|
}
|
|
|
|
static void iwl_fw_dump_mem(struct iwl_fw_runtime *fwrt,
|
|
struct iwl_fw_error_dump_data **dump_data,
|
|
u32 sram_len, u32 sram_ofs, u32 smem_len,
|
|
u32 sram2_len)
|
|
{
|
|
const struct iwl_fw_dbg_mem_seg_tlv *fw_dbg_mem = fwrt->fw->dbg_mem_tlv;
|
|
struct iwl_fw_error_dump_mem *dump_mem;
|
|
int i;
|
|
|
|
if (!fwrt->fw->n_dbg_mem_tlv) {
|
|
(*dump_data)->type = cpu_to_le32(IWL_FW_ERROR_DUMP_MEM);
|
|
(*dump_data)->len = cpu_to_le32(sram_len + sizeof(*dump_mem));
|
|
dump_mem = (void *)(*dump_data)->data;
|
|
dump_mem->type = cpu_to_le32(IWL_FW_ERROR_DUMP_MEM_SRAM);
|
|
dump_mem->offset = cpu_to_le32(sram_ofs);
|
|
iwl_trans_read_mem_bytes(fwrt->trans, sram_ofs, dump_mem->data,
|
|
sram_len);
|
|
*dump_data = iwl_fw_error_next_data(*dump_data);
|
|
}
|
|
|
|
for (i = 0; i < fwrt->fw->n_dbg_mem_tlv; i++) {
|
|
u32 len = le32_to_cpu(fw_dbg_mem[i].len);
|
|
u32 ofs = le32_to_cpu(fw_dbg_mem[i].ofs);
|
|
|
|
(*dump_data)->type = cpu_to_le32(IWL_FW_ERROR_DUMP_MEM);
|
|
(*dump_data)->len = cpu_to_le32(len + sizeof(*dump_mem));
|
|
dump_mem = (void *)(*dump_data)->data;
|
|
dump_mem->type = fw_dbg_mem[i].data_type;
|
|
dump_mem->offset = cpu_to_le32(ofs);
|
|
|
|
IWL_DEBUG_INFO(fwrt, "WRT memory dump. Type=%u\n",
|
|
dump_mem->type);
|
|
|
|
iwl_trans_read_mem_bytes(fwrt->trans, ofs, dump_mem->data, len);
|
|
*dump_data = iwl_fw_error_next_data(*dump_data);
|
|
}
|
|
|
|
if (smem_len) {
|
|
IWL_DEBUG_INFO(fwrt, "WRT SMEM dump\n");
|
|
(*dump_data)->type = cpu_to_le32(IWL_FW_ERROR_DUMP_MEM);
|
|
(*dump_data)->len = cpu_to_le32(smem_len + sizeof(*dump_mem));
|
|
dump_mem = (void *)(*dump_data)->data;
|
|
dump_mem->type = cpu_to_le32(IWL_FW_ERROR_DUMP_MEM_SMEM);
|
|
dump_mem->offset = cpu_to_le32(fwrt->trans->cfg->smem_offset);
|
|
iwl_trans_read_mem_bytes(fwrt->trans,
|
|
fwrt->trans->cfg->smem_offset,
|
|
dump_mem->data, smem_len);
|
|
*dump_data = iwl_fw_error_next_data(*dump_data);
|
|
}
|
|
|
|
if (sram2_len) {
|
|
IWL_DEBUG_INFO(fwrt, "WRT SRAM dump\n");
|
|
(*dump_data)->type = cpu_to_le32(IWL_FW_ERROR_DUMP_MEM);
|
|
(*dump_data)->len = cpu_to_le32(sram2_len + sizeof(*dump_mem));
|
|
dump_mem = (void *)(*dump_data)->data;
|
|
dump_mem->type = cpu_to_le32(IWL_FW_ERROR_DUMP_MEM_SRAM);
|
|
dump_mem->offset = cpu_to_le32(fwrt->trans->cfg->dccm2_offset);
|
|
iwl_trans_read_mem_bytes(fwrt->trans,
|
|
fwrt->trans->cfg->dccm2_offset,
|
|
dump_mem->data, sram2_len);
|
|
*dump_data = iwl_fw_error_next_data(*dump_data);
|
|
}
|
|
}
|
|
|
|
void iwl_fw_error_dump(struct iwl_fw_runtime *fwrt)
|
|
{
|
|
struct iwl_fw_error_dump_file *dump_file;
|
|
struct iwl_fw_error_dump_data *dump_data;
|
|
struct iwl_fw_error_dump_info *dump_info;
|
|
struct iwl_fw_error_dump_smem_cfg *dump_smem_cfg;
|
|
struct iwl_fw_error_dump_trigger_desc *dump_trig;
|
|
struct iwl_fw_dump_ptrs *fw_error_dump;
|
|
struct scatterlist *sg_dump_data;
|
|
u32 sram_len, sram_ofs;
|
|
const struct iwl_fw_dbg_mem_seg_tlv *fw_dbg_mem = fwrt->fw->dbg_mem_tlv;
|
|
struct iwl_fwrt_shared_mem_cfg *mem_cfg = &fwrt->smem_cfg;
|
|
u32 file_len, fifo_data_len = 0, prph_len = 0, radio_len = 0;
|
|
u32 smem_len = fwrt->fw->n_dbg_mem_tlv ? 0 : fwrt->trans->cfg->smem_len;
|
|
u32 sram2_len = fwrt->fw->n_dbg_mem_tlv ?
|
|
0 : fwrt->trans->cfg->dccm2_len;
|
|
bool monitor_dump_only = false;
|
|
int i;
|
|
|
|
IWL_DEBUG_INFO(fwrt, "WRT dump start\n");
|
|
|
|
/* there's no point in fw dump if the bus is dead */
|
|
if (test_bit(STATUS_TRANS_DEAD, &fwrt->trans->status)) {
|
|
IWL_ERR(fwrt, "Skip fw error dump since bus is dead\n");
|
|
goto out;
|
|
}
|
|
|
|
if (fwrt->dump.trig &&
|
|
fwrt->dump.trig->mode & IWL_FW_DBG_TRIGGER_MONITOR_ONLY)
|
|
monitor_dump_only = true;
|
|
|
|
fw_error_dump = kzalloc(sizeof(*fw_error_dump), GFP_KERNEL);
|
|
if (!fw_error_dump)
|
|
goto out;
|
|
|
|
/* SRAM - include stack CCM if driver knows the values for it */
|
|
if (!fwrt->trans->cfg->dccm_offset || !fwrt->trans->cfg->dccm_len) {
|
|
const struct fw_img *img;
|
|
|
|
img = &fwrt->fw->img[fwrt->cur_fw_img];
|
|
sram_ofs = img->sec[IWL_UCODE_SECTION_DATA].offset;
|
|
sram_len = img->sec[IWL_UCODE_SECTION_DATA].len;
|
|
} else {
|
|
sram_ofs = fwrt->trans->cfg->dccm_offset;
|
|
sram_len = fwrt->trans->cfg->dccm_len;
|
|
}
|
|
|
|
/* reading RXF/TXF sizes */
|
|
if (test_bit(STATUS_FW_ERROR, &fwrt->trans->status)) {
|
|
fifo_data_len = 0;
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_RXF)) {
|
|
|
|
/* Count RXF2 size */
|
|
if (mem_cfg->rxfifo2_size) {
|
|
/* Add header info */
|
|
fifo_data_len +=
|
|
mem_cfg->rxfifo2_size +
|
|
sizeof(*dump_data) +
|
|
sizeof(struct iwl_fw_error_dump_fifo);
|
|
}
|
|
|
|
/* Count RXF1 sizes */
|
|
for (i = 0; i < mem_cfg->num_lmacs; i++) {
|
|
if (!mem_cfg->lmac[i].rxfifo1_size)
|
|
continue;
|
|
|
|
/* Add header info */
|
|
fifo_data_len +=
|
|
mem_cfg->lmac[i].rxfifo1_size +
|
|
sizeof(*dump_data) +
|
|
sizeof(struct iwl_fw_error_dump_fifo);
|
|
}
|
|
}
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_TXF)) {
|
|
size_t fifo_const_len = sizeof(*dump_data) +
|
|
sizeof(struct iwl_fw_error_dump_fifo);
|
|
|
|
/* Count TXF sizes */
|
|
for (i = 0; i < mem_cfg->num_lmacs; i++) {
|
|
int j;
|
|
|
|
for (j = 0; j < mem_cfg->num_txfifo_entries;
|
|
j++) {
|
|
if (!mem_cfg->lmac[i].txfifo_size[j])
|
|
continue;
|
|
|
|
/* Add header info */
|
|
fifo_data_len +=
|
|
fifo_const_len +
|
|
mem_cfg->lmac[i].txfifo_size[j];
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((fwrt->fw->dbg_dump_mask &
|
|
BIT(IWL_FW_ERROR_DUMP_INTERNAL_TXF)) &&
|
|
fw_has_capa(&fwrt->fw->ucode_capa,
|
|
IWL_UCODE_TLV_CAPA_EXTEND_SHARED_MEM_CFG)) {
|
|
for (i = 0;
|
|
i < ARRAY_SIZE(mem_cfg->internal_txfifo_size);
|
|
i++) {
|
|
if (!mem_cfg->internal_txfifo_size[i])
|
|
continue;
|
|
|
|
/* Add header info */
|
|
fifo_data_len +=
|
|
mem_cfg->internal_txfifo_size[i] +
|
|
sizeof(*dump_data) +
|
|
sizeof(struct iwl_fw_error_dump_fifo);
|
|
}
|
|
}
|
|
|
|
/* Make room for PRPH registers */
|
|
if (!fwrt->trans->cfg->gen2 &&
|
|
fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_PRPH))
|
|
prph_len += iwl_fw_get_prph_len(fwrt);
|
|
|
|
if (fwrt->trans->cfg->device_family == IWL_DEVICE_FAMILY_7000 &&
|
|
fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_RADIO_REG))
|
|
radio_len = sizeof(*dump_data) + RADIO_REG_MAX_READ;
|
|
}
|
|
|
|
file_len = sizeof(*dump_file) +
|
|
fifo_data_len +
|
|
prph_len +
|
|
radio_len;
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_DEV_FW_INFO))
|
|
file_len += sizeof(*dump_data) + sizeof(*dump_info);
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_MEM_CFG))
|
|
file_len += sizeof(*dump_data) + sizeof(*dump_smem_cfg);
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_MEM)) {
|
|
/* Make room for the SMEM, if it exists */
|
|
if (smem_len)
|
|
file_len += sizeof(*dump_data) + smem_len +
|
|
sizeof(struct iwl_fw_error_dump_mem);
|
|
|
|
/* Make room for the secondary SRAM, if it exists */
|
|
if (sram2_len)
|
|
file_len += sizeof(*dump_data) + sram2_len +
|
|
sizeof(struct iwl_fw_error_dump_mem);
|
|
|
|
/* Make room for MEM segments */
|
|
for (i = 0; i < fwrt->fw->n_dbg_mem_tlv; i++) {
|
|
file_len += sizeof(*dump_data) +
|
|
le32_to_cpu(fw_dbg_mem[i].len) +
|
|
sizeof(struct iwl_fw_error_dump_mem);
|
|
}
|
|
}
|
|
|
|
/* Make room for fw's virtual image pages, if it exists */
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_PAGING) &&
|
|
!fwrt->trans->cfg->gen2 &&
|
|
fwrt->fw->img[fwrt->cur_fw_img].paging_mem_size &&
|
|
fwrt->fw_paging_db[0].fw_paging_block)
|
|
file_len += fwrt->num_of_paging_blk *
|
|
(sizeof(*dump_data) +
|
|
sizeof(struct iwl_fw_error_dump_paging) +
|
|
PAGING_BLOCK_SIZE);
|
|
|
|
if (iwl_fw_dbg_is_d3_debug_enabled(fwrt) && fwrt->dump.d3_debug_data) {
|
|
file_len += sizeof(*dump_data) +
|
|
fwrt->trans->cfg->d3_debug_data_length * 2;
|
|
}
|
|
|
|
/* If we only want a monitor dump, reset the file length */
|
|
if (monitor_dump_only) {
|
|
file_len = sizeof(*dump_file) + sizeof(*dump_data) * 2 +
|
|
sizeof(*dump_info) + sizeof(*dump_smem_cfg);
|
|
}
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_ERROR_INFO) &&
|
|
fwrt->dump.desc)
|
|
file_len += sizeof(*dump_data) + sizeof(*dump_trig) +
|
|
fwrt->dump.desc->len;
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_MEM) &&
|
|
!fwrt->fw->n_dbg_mem_tlv)
|
|
file_len += sizeof(*dump_data) + sram_len +
|
|
sizeof(struct iwl_fw_error_dump_mem);
|
|
|
|
dump_file = vzalloc(file_len);
|
|
if (!dump_file) {
|
|
kfree(fw_error_dump);
|
|
goto out;
|
|
}
|
|
|
|
fw_error_dump->fwrt_ptr = dump_file;
|
|
|
|
dump_file->barker = cpu_to_le32(IWL_FW_ERROR_DUMP_BARKER);
|
|
dump_data = (void *)dump_file->data;
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_DEV_FW_INFO)) {
|
|
dump_data->type = cpu_to_le32(IWL_FW_ERROR_DUMP_DEV_FW_INFO);
|
|
dump_data->len = cpu_to_le32(sizeof(*dump_info));
|
|
dump_info = (void *)dump_data->data;
|
|
dump_info->device_family =
|
|
fwrt->trans->cfg->device_family ==
|
|
IWL_DEVICE_FAMILY_7000 ?
|
|
cpu_to_le32(IWL_FW_ERROR_DUMP_FAMILY_7) :
|
|
cpu_to_le32(IWL_FW_ERROR_DUMP_FAMILY_8);
|
|
dump_info->hw_step =
|
|
cpu_to_le32(CSR_HW_REV_STEP(fwrt->trans->hw_rev));
|
|
memcpy(dump_info->fw_human_readable, fwrt->fw->human_readable,
|
|
sizeof(dump_info->fw_human_readable));
|
|
strncpy(dump_info->dev_human_readable, fwrt->trans->cfg->name,
|
|
sizeof(dump_info->dev_human_readable) - 1);
|
|
strncpy(dump_info->bus_human_readable, fwrt->dev->bus->name,
|
|
sizeof(dump_info->bus_human_readable) - 1);
|
|
|
|
dump_data = iwl_fw_error_next_data(dump_data);
|
|
}
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_MEM_CFG)) {
|
|
/* Dump shared memory configuration */
|
|
dump_data->type = cpu_to_le32(IWL_FW_ERROR_DUMP_MEM_CFG);
|
|
dump_data->len = cpu_to_le32(sizeof(*dump_smem_cfg));
|
|
dump_smem_cfg = (void *)dump_data->data;
|
|
dump_smem_cfg->num_lmacs = cpu_to_le32(mem_cfg->num_lmacs);
|
|
dump_smem_cfg->num_txfifo_entries =
|
|
cpu_to_le32(mem_cfg->num_txfifo_entries);
|
|
for (i = 0; i < MAX_NUM_LMAC; i++) {
|
|
int j;
|
|
u32 *txf_size = mem_cfg->lmac[i].txfifo_size;
|
|
|
|
for (j = 0; j < TX_FIFO_MAX_NUM; j++)
|
|
dump_smem_cfg->lmac[i].txfifo_size[j] =
|
|
cpu_to_le32(txf_size[j]);
|
|
dump_smem_cfg->lmac[i].rxfifo1_size =
|
|
cpu_to_le32(mem_cfg->lmac[i].rxfifo1_size);
|
|
}
|
|
dump_smem_cfg->rxfifo2_size =
|
|
cpu_to_le32(mem_cfg->rxfifo2_size);
|
|
dump_smem_cfg->internal_txfifo_addr =
|
|
cpu_to_le32(mem_cfg->internal_txfifo_addr);
|
|
for (i = 0; i < TX_FIFO_INTERNAL_MAX_NUM; i++) {
|
|
dump_smem_cfg->internal_txfifo_size[i] =
|
|
cpu_to_le32(mem_cfg->internal_txfifo_size[i]);
|
|
}
|
|
|
|
dump_data = iwl_fw_error_next_data(dump_data);
|
|
}
|
|
|
|
/* We only dump the FIFOs if the FW is in error state */
|
|
if (fifo_data_len) {
|
|
iwl_fw_dump_fifos(fwrt, &dump_data);
|
|
if (radio_len)
|
|
iwl_read_radio_regs(fwrt, &dump_data);
|
|
}
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_ERROR_INFO) &&
|
|
fwrt->dump.desc) {
|
|
dump_data->type = cpu_to_le32(IWL_FW_ERROR_DUMP_ERROR_INFO);
|
|
dump_data->len = cpu_to_le32(sizeof(*dump_trig) +
|
|
fwrt->dump.desc->len);
|
|
dump_trig = (void *)dump_data->data;
|
|
memcpy(dump_trig, &fwrt->dump.desc->trig_desc,
|
|
sizeof(*dump_trig) + fwrt->dump.desc->len);
|
|
|
|
dump_data = iwl_fw_error_next_data(dump_data);
|
|
}
|
|
|
|
/* In case we only want monitor dump, skip to dump trasport data */
|
|
if (monitor_dump_only)
|
|
goto dump_trans_data;
|
|
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_MEM))
|
|
iwl_fw_dump_mem(fwrt, &dump_data, sram_len, sram_ofs, smem_len,
|
|
sram2_len);
|
|
|
|
|
|
if (iwl_fw_dbg_is_d3_debug_enabled(fwrt) && fwrt->dump.d3_debug_data) {
|
|
u32 addr = fwrt->trans->cfg->d3_debug_data_base_addr;
|
|
size_t data_size = fwrt->trans->cfg->d3_debug_data_length;
|
|
|
|
dump_data->type = cpu_to_le32(IWL_FW_ERROR_DUMP_D3_DEBUG_DATA);
|
|
dump_data->len = cpu_to_le32(data_size * 2);
|
|
|
|
memcpy(dump_data->data, fwrt->dump.d3_debug_data, data_size);
|
|
|
|
kfree(fwrt->dump.d3_debug_data);
|
|
fwrt->dump.d3_debug_data = NULL;
|
|
|
|
iwl_trans_read_mem_bytes(fwrt->trans, addr,
|
|
dump_data->data + data_size,
|
|
data_size);
|
|
|
|
dump_data = iwl_fw_error_next_data(dump_data);
|
|
}
|
|
|
|
/* Dump fw's virtual image */
|
|
if (fwrt->fw->dbg_dump_mask & BIT(IWL_FW_ERROR_DUMP_PAGING) &&
|
|
!fwrt->trans->cfg->gen2 &&
|
|
fwrt->fw->img[fwrt->cur_fw_img].paging_mem_size &&
|
|
fwrt->fw_paging_db[0].fw_paging_block) {
|
|
IWL_DEBUG_INFO(fwrt, "WRT paging dump\n");
|
|
for (i = 1; i < fwrt->num_of_paging_blk + 1; i++) {
|
|
struct iwl_fw_error_dump_paging *paging;
|
|
struct page *pages =
|
|
fwrt->fw_paging_db[i].fw_paging_block;
|
|
dma_addr_t addr = fwrt->fw_paging_db[i].fw_paging_phys;
|
|
|
|
dump_data->type = cpu_to_le32(IWL_FW_ERROR_DUMP_PAGING);
|
|
dump_data->len = cpu_to_le32(sizeof(*paging) +
|
|
PAGING_BLOCK_SIZE);
|
|
paging = (void *)dump_data->data;
|
|
paging->index = cpu_to_le32(i);
|
|
dma_sync_single_for_cpu(fwrt->trans->dev, addr,
|
|
PAGING_BLOCK_SIZE,
|
|
DMA_BIDIRECTIONAL);
|
|
memcpy(paging->data, page_address(pages),
|
|
PAGING_BLOCK_SIZE);
|
|
dump_data = iwl_fw_error_next_data(dump_data);
|
|
}
|
|
}
|
|
|
|
if (prph_len) {
|
|
iwl_dump_prph(fwrt->trans, &dump_data,
|
|
iwl_prph_dump_addr_comm,
|
|
ARRAY_SIZE(iwl_prph_dump_addr_comm));
|
|
|
|
if (fwrt->trans->cfg->mq_rx_supported)
|
|
iwl_dump_prph(fwrt->trans, &dump_data,
|
|
iwl_prph_dump_addr_9000,
|
|
ARRAY_SIZE(iwl_prph_dump_addr_9000));
|
|
}
|
|
|
|
dump_trans_data:
|
|
fw_error_dump->trans_ptr = iwl_trans_dump_data(fwrt->trans,
|
|
fwrt->dump.trig);
|
|
fw_error_dump->fwrt_len = file_len;
|
|
if (fw_error_dump->trans_ptr)
|
|
file_len += fw_error_dump->trans_ptr->len;
|
|
dump_file->file_len = cpu_to_le32(file_len);
|
|
|
|
sg_dump_data = alloc_sgtable(file_len);
|
|
if (sg_dump_data) {
|
|
sg_pcopy_from_buffer(sg_dump_data,
|
|
sg_nents(sg_dump_data),
|
|
fw_error_dump->fwrt_ptr,
|
|
fw_error_dump->fwrt_len, 0);
|
|
if (fw_error_dump->trans_ptr)
|
|
sg_pcopy_from_buffer(sg_dump_data,
|
|
sg_nents(sg_dump_data),
|
|
fw_error_dump->trans_ptr->data,
|
|
fw_error_dump->trans_ptr->len,
|
|
fw_error_dump->fwrt_len);
|
|
dev_coredumpsg(fwrt->trans->dev, sg_dump_data, file_len,
|
|
GFP_KERNEL);
|
|
}
|
|
vfree(fw_error_dump->fwrt_ptr);
|
|
vfree(fw_error_dump->trans_ptr);
|
|
kfree(fw_error_dump);
|
|
|
|
out:
|
|
iwl_fw_free_dump_desc(fwrt);
|
|
clear_bit(IWL_FWRT_STATUS_DUMPING, &fwrt->status);
|
|
IWL_DEBUG_INFO(fwrt, "WRT dump done\n");
|
|
}
|
|
IWL_EXPORT_SYMBOL(iwl_fw_error_dump);
|
|
|
|
const struct iwl_fw_dump_desc iwl_dump_desc_assert = {
|
|
.trig_desc = {
|
|
.type = cpu_to_le32(FW_DBG_TRIGGER_FW_ASSERT),
|
|
},
|
|
};
|
|
IWL_EXPORT_SYMBOL(iwl_dump_desc_assert);
|
|
|
|
int iwl_fw_dbg_collect_desc(struct iwl_fw_runtime *fwrt,
|
|
const struct iwl_fw_dump_desc *desc,
|
|
const struct iwl_fw_dbg_trigger_tlv *trigger)
|
|
{
|
|
unsigned int delay = 0;
|
|
|
|
if (trigger)
|
|
delay = msecs_to_jiffies(le32_to_cpu(trigger->stop_delay));
|
|
|
|
/*
|
|
* If the loading of the FW completed successfully, the next step is to
|
|
* get the SMEM config data. Thus, if fwrt->smem_cfg.num_lmacs is non
|
|
* zero, the FW was already loaded successully. If the state is "NO_FW"
|
|
* in such a case - exit, since FW may be dead. Otherwise, we
|
|
* can try to collect the data, since FW might just not be fully
|
|
* loaded (no "ALIVE" yet), and the debug data is accessible.
|
|
*
|
|
* Corner case: got the FW alive but crashed before getting the SMEM
|
|
* config. In such a case, due to HW access problems, we might
|
|
* collect garbage.
|
|
*/
|
|
if (fwrt->trans->state == IWL_TRANS_NO_FW &&
|
|
fwrt->smem_cfg.num_lmacs)
|
|
return -EIO;
|
|
|
|
if (test_and_set_bit(IWL_FWRT_STATUS_DUMPING, &fwrt->status))
|
|
return -EBUSY;
|
|
|
|
if (WARN_ON(fwrt->dump.desc))
|
|
iwl_fw_free_dump_desc(fwrt);
|
|
|
|
IWL_WARN(fwrt, "Collecting data: trigger %d fired.\n",
|
|
le32_to_cpu(desc->trig_desc.type));
|
|
|
|
fwrt->dump.desc = desc;
|
|
fwrt->dump.trig = trigger;
|
|
|
|
schedule_delayed_work(&fwrt->dump.wk, delay);
|
|
|
|
return 0;
|
|
}
|
|
IWL_EXPORT_SYMBOL(iwl_fw_dbg_collect_desc);
|
|
|
|
int iwl_fw_dbg_collect(struct iwl_fw_runtime *fwrt,
|
|
enum iwl_fw_dbg_trigger trig,
|
|
const char *str, size_t len,
|
|
const struct iwl_fw_dbg_trigger_tlv *trigger)
|
|
{
|
|
struct iwl_fw_dump_desc *desc;
|
|
|
|
if (trigger && trigger->flags & IWL_FW_DBG_FORCE_RESTART) {
|
|
IWL_WARN(fwrt, "Force restart: trigger %d fired.\n", trig);
|
|
iwl_force_nmi(fwrt->trans);
|
|
return 0;
|
|
}
|
|
|
|
desc = kzalloc(sizeof(*desc) + len, GFP_ATOMIC);
|
|
if (!desc)
|
|
return -ENOMEM;
|
|
|
|
desc->len = len;
|
|
desc->trig_desc.type = cpu_to_le32(trig);
|
|
memcpy(desc->trig_desc.data, str, len);
|
|
|
|
return iwl_fw_dbg_collect_desc(fwrt, desc, trigger);
|
|
}
|
|
IWL_EXPORT_SYMBOL(iwl_fw_dbg_collect);
|
|
|
|
int iwl_fw_dbg_collect_trig(struct iwl_fw_runtime *fwrt,
|
|
struct iwl_fw_dbg_trigger_tlv *trigger,
|
|
const char *fmt, ...)
|
|
{
|
|
u16 occurrences = le16_to_cpu(trigger->occurrences);
|
|
int ret, len = 0;
|
|
char buf[64];
|
|
|
|
if (!occurrences)
|
|
return 0;
|
|
|
|
if (fmt) {
|
|
va_list ap;
|
|
|
|
buf[sizeof(buf) - 1] = '\0';
|
|
|
|
va_start(ap, fmt);
|
|
vsnprintf(buf, sizeof(buf), fmt, ap);
|
|
va_end(ap);
|
|
|
|
/* check for truncation */
|
|
if (WARN_ON_ONCE(buf[sizeof(buf) - 1]))
|
|
buf[sizeof(buf) - 1] = '\0';
|
|
|
|
len = strlen(buf) + 1;
|
|
}
|
|
|
|
ret = iwl_fw_dbg_collect(fwrt, le32_to_cpu(trigger->id), buf, len,
|
|
trigger);
|
|
|
|
if (ret)
|
|
return ret;
|
|
|
|
trigger->occurrences = cpu_to_le16(occurrences - 1);
|
|
return 0;
|
|
}
|
|
IWL_EXPORT_SYMBOL(iwl_fw_dbg_collect_trig);
|
|
|
|
int iwl_fw_start_dbg_conf(struct iwl_fw_runtime *fwrt, u8 conf_id)
|
|
{
|
|
u8 *ptr;
|
|
int ret;
|
|
int i;
|
|
|
|
if (WARN_ONCE(conf_id >= ARRAY_SIZE(fwrt->fw->dbg_conf_tlv),
|
|
"Invalid configuration %d\n", conf_id))
|
|
return -EINVAL;
|
|
|
|
/* EARLY START - firmware's configuration is hard coded */
|
|
if ((!fwrt->fw->dbg_conf_tlv[conf_id] ||
|
|
!fwrt->fw->dbg_conf_tlv[conf_id]->num_of_hcmds) &&
|
|
conf_id == FW_DBG_START_FROM_ALIVE)
|
|
return 0;
|
|
|
|
if (!fwrt->fw->dbg_conf_tlv[conf_id])
|
|
return -EINVAL;
|
|
|
|
if (fwrt->dump.conf != FW_DBG_INVALID)
|
|
IWL_WARN(fwrt, "FW already configured (%d) - re-configuring\n",
|
|
fwrt->dump.conf);
|
|
|
|
/* Send all HCMDs for configuring the FW debug */
|
|
ptr = (void *)&fwrt->fw->dbg_conf_tlv[conf_id]->hcmd;
|
|
for (i = 0; i < fwrt->fw->dbg_conf_tlv[conf_id]->num_of_hcmds; i++) {
|
|
struct iwl_fw_dbg_conf_hcmd *cmd = (void *)ptr;
|
|
struct iwl_host_cmd hcmd = {
|
|
.id = cmd->id,
|
|
.len = { le16_to_cpu(cmd->len), },
|
|
.data = { cmd->data, },
|
|
};
|
|
|
|
ret = iwl_trans_send_cmd(fwrt->trans, &hcmd);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ptr += sizeof(*cmd);
|
|
ptr += le16_to_cpu(cmd->len);
|
|
}
|
|
|
|
fwrt->dump.conf = conf_id;
|
|
|
|
return 0;
|
|
}
|
|
IWL_EXPORT_SYMBOL(iwl_fw_start_dbg_conf);
|
|
|
|
void iwl_fw_error_dump_wk(struct work_struct *work)
|
|
{
|
|
struct iwl_fw_runtime *fwrt =
|
|
container_of(work, struct iwl_fw_runtime, dump.wk.work);
|
|
struct iwl_fw_dbg_params params = {0};
|
|
|
|
if (fwrt->ops && fwrt->ops->dump_start &&
|
|
fwrt->ops->dump_start(fwrt->ops_ctx))
|
|
return;
|
|
|
|
if (fwrt->ops && fwrt->ops->fw_running &&
|
|
!fwrt->ops->fw_running(fwrt->ops_ctx)) {
|
|
IWL_ERR(fwrt, "Firmware not running - cannot dump error\n");
|
|
iwl_fw_free_dump_desc(fwrt);
|
|
clear_bit(IWL_FWRT_STATUS_DUMPING, &fwrt->status);
|
|
goto out;
|
|
}
|
|
|
|
iwl_fw_dbg_stop_recording(fwrt, ¶ms);
|
|
|
|
iwl_fw_error_dump(fwrt);
|
|
|
|
/* start recording again if the firmware is not crashed */
|
|
if (!test_bit(STATUS_FW_ERROR, &fwrt->trans->status) &&
|
|
fwrt->fw->dbg_dest_tlv) {
|
|
/* wait before we collect the data till the DBGC stop */
|
|
udelay(500);
|
|
iwl_fw_dbg_restart_recording(fwrt, ¶ms);
|
|
}
|
|
out:
|
|
if (fwrt->ops && fwrt->ops->dump_end)
|
|
fwrt->ops->dump_end(fwrt->ops_ctx);
|
|
}
|
|
|
|
void iwl_fw_dbg_read_d3_debug_data(struct iwl_fw_runtime *fwrt)
|
|
{
|
|
const struct iwl_cfg *cfg = fwrt->trans->cfg;
|
|
|
|
if (!iwl_fw_dbg_is_d3_debug_enabled(fwrt))
|
|
return;
|
|
|
|
if (!fwrt->dump.d3_debug_data) {
|
|
fwrt->dump.d3_debug_data = kmalloc(cfg->d3_debug_data_length,
|
|
GFP_KERNEL);
|
|
if (!fwrt->dump.d3_debug_data) {
|
|
IWL_ERR(fwrt,
|
|
"failed to allocate memory for D3 debug data\n");
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* if the buffer holds previous debug data it is overwritten */
|
|
iwl_trans_read_mem_bytes(fwrt->trans, cfg->d3_debug_data_base_addr,
|
|
fwrt->dump.d3_debug_data,
|
|
cfg->d3_debug_data_length);
|
|
}
|
|
IWL_EXPORT_SYMBOL(iwl_fw_dbg_read_d3_debug_data);
|