OSDN Git Service

iwlwifi: dbg_ini: support FW response/notification region type
authorShahar S Matityahu <shahar.s.matityahu@intel.com>
Tue, 23 Jul 2019 12:22:25 +0000 (15:22 +0300)
committerLuca Coelho <luciano.coelho@intel.com>
Fri, 25 Oct 2019 07:09:51 +0000 (10:09 +0300)
Allow the driver to collect FW response/notification region type
during dump and allow triggering dump collection for a given FW
response/notification.

Signed-off-by: Shahar S Matityahu <shahar.s.matityahu@intel.com>
Signed-off-by: Luca Coelho <luciano.coelho@intel.com>
drivers/net/wireless/intel/iwlwifi/fw/dbg.c
drivers/net/wireless/intel/iwlwifi/fw/error-dump.h
drivers/net/wireless/intel/iwlwifi/iwl-dbg-tlv.c

index c4b79f4..ce77ed1 100644 (file)
@@ -1057,9 +1057,11 @@ out:
 /**
  * struct iwl_dump_ini_region_data - region data
  * @reg_tlv: region TLV
+ * @dump_data: dump data
  */
 struct iwl_dump_ini_region_data {
        struct iwl_ucode_tlv *reg_tlv;
+       struct iwl_fwrt_dump_data *dump_data;
 };
 
 static int iwl_dump_ini_prph_iter(struct iwl_fw_runtime *fwrt,
@@ -1449,6 +1451,27 @@ iwl_dump_ini_err_table_iter(struct iwl_fw_runtime *fwrt,
        return sizeof(*range) + le32_to_cpu(range->range_data_size);
 }
 
+static int iwl_dump_ini_fw_pkt_iter(struct iwl_fw_runtime *fwrt,
+                                   struct iwl_dump_ini_region_data *reg_data,
+                                   void *range_ptr, int idx)
+{
+       struct iwl_fw_ini_error_dump_range *range = range_ptr;
+       struct iwl_rx_packet *pkt = reg_data->dump_data->fw_pkt;
+       u32 pkt_len;
+
+       if (!pkt)
+               return -EIO;
+
+       pkt_len = iwl_rx_packet_payload_len(pkt);
+
+       memcpy(&range->fw_pkt_hdr, &pkt->hdr, sizeof(range->fw_pkt_hdr));
+       range->range_data_size = cpu_to_le32(pkt_len);
+
+       memcpy(range->data, pkt->data, pkt_len);
+
+       return sizeof(*range) + le32_to_cpu(range->range_data_size);
+}
+
 static void *
 iwl_dump_ini_mem_fill_header(struct iwl_fw_runtime *fwrt,
                             struct iwl_dump_ini_region_data *reg_data,
@@ -1753,6 +1776,23 @@ iwl_dump_ini_err_table_get_size(struct iwl_fw_runtime *fwrt,
        return size;
 }
 
+static u32
+iwl_dump_ini_fw_pkt_get_size(struct iwl_fw_runtime *fwrt,
+                            struct iwl_dump_ini_region_data *reg_data)
+{
+       u32 size = 0;
+
+       if (!reg_data->dump_data->fw_pkt)
+               return 0;
+
+       size += iwl_rx_packet_payload_len(reg_data->dump_data->fw_pkt);
+       if (size)
+               size += sizeof(struct iwl_fw_ini_error_dump) +
+                       sizeof(struct iwl_fw_ini_error_dump_range);
+
+       return size;
+}
+
 /**
  * struct iwl_dump_ini_mem_ops - ini memory dump operations
  * @get_num_of_ranges: returns the number of memory ranges in the region.
@@ -1976,7 +2016,12 @@ static const struct iwl_dump_ini_mem_ops iwl_dump_ini_region_ops[] = {
                .fill_mem_hdr = iwl_dump_ini_err_table_fill_header,
                .fill_range = iwl_dump_ini_err_table_iter,
        },
-       [IWL_FW_INI_REGION_RSP_OR_NOTIF] = {},
+       [IWL_FW_INI_REGION_RSP_OR_NOTIF] = {
+               .get_num_of_ranges = iwl_dump_ini_single_range,
+               .get_size = iwl_dump_ini_fw_pkt_get_size,
+               .fill_mem_hdr = iwl_dump_ini_mem_fill_header,
+               .fill_range = iwl_dump_ini_fw_pkt_iter,
+       },
        [IWL_FW_INI_REGION_DEVICE_MEMORY] = {
                .get_num_of_ranges = iwl_dump_ini_mem_ranges,
                .get_size = iwl_dump_ini_mem_get_size,
@@ -2012,7 +2057,9 @@ static u32 iwl_dump_ini_trigger(struct iwl_fw_runtime *fwrt,
                                struct list_head *list)
 {
        struct iwl_fw_ini_trigger_tlv *trigger = dump_data->trig;
-       struct iwl_dump_ini_region_data reg_data = {};
+       struct iwl_dump_ini_region_data reg_data = {
+               .dump_data = dump_data,
+       };
        int i;
        u32 size = 0;
        u64 regions_mask = le64_to_cpu(trigger->regions_mask);
@@ -2155,6 +2202,8 @@ static void iwl_dump_ini_list_free(struct list_head *list)
 static void iwl_fw_error_dump_data_free(struct iwl_fwrt_dump_data *dump_data)
 {
        dump_data->trig = NULL;
+       kfree(dump_data->fw_pkt);
+       dump_data->fw_pkt = NULL;
 }
 
 static void iwl_fw_error_ini_dump(struct iwl_fw_runtime *fwrt,
index a7bf17e..f008e1b 100644 (file)
@@ -65,6 +65,7 @@
 #define __fw_error_dump_h__
 
 #include <linux/types.h>
+#include "fw/api/cmdhdr.h"
 
 #define IWL_FW_ERROR_DUMP_BARKER       0x14789632
 #define IWL_FW_INI_ERROR_DUMP_BARKER   0x14789633
@@ -327,6 +328,7 @@ struct iwl_fw_ini_fifo_hdr {
  * @dram_base_addr: base address of dram monitor range
  * @page_num: page number of memory range
  * @fifo_hdr: fifo header of memory range
+ * @fw_pkt: FW packet header of memory range
  * @data: the actual memory
  */
 struct iwl_fw_ini_error_dump_range {
@@ -336,6 +338,7 @@ struct iwl_fw_ini_error_dump_range {
                __le64 dram_base_addr;
                __le32 page_num;
                struct iwl_fw_ini_fifo_hdr fifo_hdr;
+               struct iwl_cmd_header fw_pkt_hdr;
        };
        __le32 data[];
 } __packed;
index f813b23..4f2a4d8 100644 (file)
@@ -940,6 +940,33 @@ int iwl_dbg_tlv_gen_active_trigs(struct iwl_fw_runtime *fwrt, u32 new_domain)
        return 0;
 }
 
+static bool iwl_dbg_tlv_check_fw_pkt(struct iwl_fw_runtime *fwrt,
+                                    struct iwl_fwrt_dump_data *dump_data,
+                                    union iwl_dbg_tlv_tp_data *tp_data,
+                                    u32 trig_data)
+{
+       struct iwl_rx_packet *pkt = tp_data->fw_pkt;
+       struct iwl_cmd_header *wanted_hdr = (void *)&trig_data;
+
+       if (pkt && ((wanted_hdr->cmd == 0 && wanted_hdr->group_id == 0) ||
+                   (pkt->hdr.cmd == wanted_hdr->cmd &&
+                    pkt->hdr.group_id == wanted_hdr->group_id))) {
+               struct iwl_rx_packet *fw_pkt =
+                       kmemdup(pkt,
+                               sizeof(*pkt) + iwl_rx_packet_payload_len(pkt),
+                               GFP_ATOMIC);
+
+               if (!fw_pkt)
+                       return false;
+
+               dump_data->fw_pkt = fw_pkt;
+
+               return true;
+       }
+
+       return false;
+}
+
 static int
 iwl_dbg_tlv_tp_trigger(struct iwl_fw_runtime *fwrt,
                       struct list_head *active_trig_list,
@@ -1039,6 +1066,11 @@ void iwl_dbg_tlv_time_point(struct iwl_fw_runtime *fwrt,
                iwl_dbg_tlv_set_periodic_trigs(fwrt);
                iwl_dbg_tlv_send_hcmds(fwrt, hcmd_list);
                break;
+       case IWL_FW_INI_TIME_POINT_FW_RSP_OR_NOTIF:
+               iwl_dbg_tlv_send_hcmds(fwrt, hcmd_list);
+               iwl_dbg_tlv_tp_trigger(fwrt, trig_list, tp_data,
+                                      iwl_dbg_tlv_check_fw_pkt);
+               break;
        default:
                iwl_dbg_tlv_send_hcmds(fwrt, hcmd_list);
                iwl_dbg_tlv_tp_trigger(fwrt, trig_list, tp_data, NULL);