qed*: enhance tx timeout debug info
authorManish Chopra <manishc@marvell.com>
Thu, 2 Dec 2021 21:01:56 +0000 (13:01 -0800)
committerJakub Kicinski <kuba@kernel.org>
Sat, 4 Dec 2021 02:24:20 +0000 (18:24 -0800)
This patch add some new qed APIs to query status block
info and report various data to MFW on tx timeout event

Along with that it enhances qede to dump more debug logs
(not just specific to the queue which was reported by stack)
on tx timeout which includes various other basic metadata about
all tx queues and other info (like status block etc.)

Signed-off-by: Manish Chopra <manishc@marvell.com>
Signed-off-by: Prabhakar Kushwaha <pkushwaha@marvell.com>
Signed-off-by: Alok Prasad <palok@marvell.com>
Signed-off-by: Ariel Elior <aelior@marvell.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
drivers/net/ethernet/qlogic/qed/qed_int.c
drivers/net/ethernet/qlogic/qed/qed_int.h
drivers/net/ethernet/qlogic/qed/qed_main.c
drivers/net/ethernet/qlogic/qed/qed_mcp.h
drivers/net/ethernet/qlogic/qed/qed_reg_addr.h
drivers/net/ethernet/qlogic/qede/qede_main.c
include/linux/qed/qed_if.h

index 6958adeca86d0e6c413115be436e1f0720307b1c..82e74f62b67773d94551cbdd64b7c8404e23c226 100644 (file)
@@ -2399,3 +2399,25 @@ int qed_int_set_timer_res(struct qed_hwfn *p_hwfn, struct qed_ptt *p_ptt,
 
        return rc;
 }
+
+int qed_int_get_sb_dbg(struct qed_hwfn *p_hwfn, struct qed_ptt *p_ptt,
+                      struct qed_sb_info *p_sb, struct qed_sb_info_dbg *p_info)
+{
+       u16 sbid = p_sb->igu_sb_id;
+       u32 i;
+
+       if (IS_VF(p_hwfn->cdev))
+               return -EINVAL;
+
+       if (sbid >= NUM_OF_SBS(p_hwfn->cdev))
+               return -EINVAL;
+
+       p_info->igu_prod = qed_rd(p_hwfn, p_ptt, IGU_REG_PRODUCER_MEMORY + sbid * 4);
+       p_info->igu_cons = qed_rd(p_hwfn, p_ptt, IGU_REG_CONSUMER_MEM + sbid * 4);
+
+       for (i = 0; i < PIS_PER_SB; i++)
+               p_info->pi[i] = (u16)qed_rd(p_hwfn, p_ptt,
+                                           CAU_REG_PI_MEMORY + sbid * 4 * PIS_PER_SB + i * 4);
+
+       return 0;
+}
index 84c17e97f5692cffd11c7890cf7a4bbcddabdf70..7e5127f617444afa4f9284d63ac03790308aafe0 100644 (file)
@@ -185,6 +185,19 @@ void qed_int_disable_post_isr_release(struct qed_dev *cdev);
  */
 void qed_int_attn_clr_enable(struct qed_dev *cdev, bool clr_enable);
 
+/**
+ * qed_int_get_sb_dbg: Read debug information regarding a given SB
+ *
+ * @p_hwfn: hw function pointer
+ * @p_ptt: ptt resource
+ * @p_sb: pointer to status block for which we want to get info
+ * @p_info: pointer to struct to fill with information regarding SB
+ *
+ * Return: 0 with status block info filled on success, otherwise return error
+ */
+int qed_int_get_sb_dbg(struct qed_hwfn *p_hwfn, struct qed_ptt *p_ptt,
+                      struct qed_sb_info *p_sb, struct qed_sb_info_dbg *p_info);
+
 /**
  * qed_db_rec_handler(): Doorbell Recovery handler.
  *          Run doorbell recovery in case of PF overflow (and flush DORQ if
index 7673b3e07736ceac48c21b960453e94d3ea299aa..a18d2ea96b26b7cbb113b886d91a47ee981ad82e 100644 (file)
@@ -2936,6 +2936,30 @@ out:
        return status;
 }
 
+static int
+qed_get_sb_info(struct qed_dev *cdev, struct qed_sb_info *sb,
+               u16 qid, struct qed_sb_info_dbg *sb_dbg)
+{
+       struct qed_hwfn *hwfn = &cdev->hwfns[qid % cdev->num_hwfns];
+       struct qed_ptt *ptt;
+       int rc;
+
+       if (IS_VF(cdev))
+               return -EINVAL;
+
+       ptt = qed_ptt_acquire(hwfn);
+       if (!ptt) {
+               DP_NOTICE(hwfn, "Can't acquire PTT\n");
+               return -EAGAIN;
+       }
+
+       memset(sb_dbg, 0, sizeof(*sb_dbg));
+       rc = qed_int_get_sb_dbg(hwfn, ptt, sb, sb_dbg);
+
+       qed_ptt_release(hwfn, ptt);
+       return rc;
+}
+
 static int qed_read_module_eeprom(struct qed_dev *cdev, char *buf,
                                  u8 dev_addr, u32 offset, u32 len)
 {
@@ -2978,6 +3002,27 @@ static int qed_set_grc_config(struct qed_dev *cdev, u32 cfg_id, u32 val)
        return rc;
 }
 
+static __printf(2, 3) void qed_mfw_report(struct qed_dev *cdev, char *fmt, ...)
+{
+       char buf[QED_MFW_REPORT_STR_SIZE];
+       struct qed_hwfn *p_hwfn;
+       struct qed_ptt *p_ptt;
+       va_list vl;
+
+       va_start(vl, fmt);
+       vsnprintf(buf, QED_MFW_REPORT_STR_SIZE, fmt, vl);
+       va_end(vl);
+
+       if (IS_PF(cdev)) {
+               p_hwfn = QED_LEADING_HWFN(cdev);
+               p_ptt = qed_ptt_acquire(p_hwfn);
+               if (p_ptt) {
+                       qed_mcp_send_raw_debug_data(p_hwfn, p_ptt, buf, strlen(buf));
+                       qed_ptt_release(p_hwfn, p_ptt);
+               }
+       }
+}
+
 static u8 qed_get_affin_hwfn_idx(struct qed_dev *cdev)
 {
        return QED_AFFIN_HWFN_IDX(cdev);
@@ -3038,6 +3083,8 @@ const struct qed_common_ops qed_common_ops_pass = {
        .read_nvm_cfg = &qed_nvm_flash_cfg_read,
        .read_nvm_cfg_len = &qed_nvm_flash_cfg_len,
        .set_grc_config = &qed_set_grc_config,
+       .mfw_report = &qed_mfw_report,
+       .get_sb_info = &qed_get_sb_info,
 };
 
 void qed_get_protocol_stats(struct qed_dev *cdev,
index 564723800d1594bae2d5c8a9623bebce73de0806..2c28d5f86497dd105343d9fad70b8d22f416cf13 100644 (file)
@@ -15,6 +15,8 @@
 #include "qed_hsi.h"
 #include "qed_dev_api.h"
 
+#define QED_MFW_REPORT_STR_SIZE        256
+
 struct qed_mcp_link_speed_params {
        bool                                    autoneg;
 
index 6f1a52e6beb2f8a5645708b563c77a79f1a92452..b5e35f433a204c665a4b2878b8f95bc899d1c8f7 100644 (file)
                0x1 << 1)
 #define  IGU_REG_BLOCK_CONFIGURATION_PXP_TPH_INTERFACE_EN      ( \
                0x1 << 0)
+#define IGU_REG_PRODUCER_MEMORY 0x182000UL
+#define IGU_REG_CONSUMER_MEM 0x183000UL
 #define  IGU_REG_MAPPING_MEMORY \
        0x184000UL
 #define IGU_REG_STATISTIC_NUM_VF_MSG_SENT \
index 06c6a58136063964abd91c99b4017dc29825f36e..b4e5a15e308b28b91fcaf6c419f091e3e748b2e7 100644 (file)
@@ -509,34 +509,95 @@ static int qede_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
        return 0;
 }
 
-static void qede_tx_log_print(struct qede_dev *edev, struct qede_tx_queue *txq)
+static void qede_fp_sb_dump(struct qede_dev *edev, struct qede_fastpath *fp)
 {
+       char *p_sb = (char *)fp->sb_info->sb_virt;
+       u32 sb_size, i;
+
+       sb_size = sizeof(struct status_block);
+
+       for (i = 0; i < sb_size; i += 8)
+               DP_NOTICE(edev,
+                         "%02hhX %02hhX %02hhX %02hhX  %02hhX %02hhX %02hhX %02hhX\n",
+                         p_sb[i], p_sb[i + 1], p_sb[i + 2], p_sb[i + 3],
+                         p_sb[i + 4], p_sb[i + 5], p_sb[i + 6], p_sb[i + 7]);
+}
+
+static void
+qede_txq_fp_log_metadata(struct qede_dev *edev,
+                        struct qede_fastpath *fp, struct qede_tx_queue *txq)
+{
+       struct qed_chain *p_chain = &txq->tx_pbl;
+
+       /* Dump txq/fp/sb ids etc. other metadata */
        DP_NOTICE(edev,
-                 "Txq[%d]: FW cons [host] %04x, SW cons %04x, SW prod %04x [Jiffies %lu]\n",
-                 txq->index, le16_to_cpu(*txq->hw_cons_ptr),
-                 qed_chain_get_cons_idx(&txq->tx_pbl),
-                 qed_chain_get_prod_idx(&txq->tx_pbl),
-                 jiffies);
+                 "fpid 0x%x sbid 0x%x txqid [0x%x] ndev_qid [0x%x] cos [0x%x] p_chain %p cap %d size %d jiffies %lu HZ 0x%x\n",
+                 fp->id, fp->sb_info->igu_sb_id, txq->index, txq->ndev_txq_id, txq->cos,
+                 p_chain, p_chain->capacity, p_chain->size, jiffies, HZ);
+
+       /* Dump all the relevant prod/cons indexes */
+       DP_NOTICE(edev,
+                 "hw cons %04x sw_tx_prod=0x%x, sw_tx_cons=0x%x, bd_prod 0x%x bd_cons 0x%x\n",
+                 le16_to_cpu(*txq->hw_cons_ptr), txq->sw_tx_prod, txq->sw_tx_cons,
+                 qed_chain_get_prod_idx(p_chain), qed_chain_get_cons_idx(p_chain));
+}
+
+static void
+qede_tx_log_print(struct qede_dev *edev, struct qede_fastpath *fp, struct qede_tx_queue *txq)
+{
+       struct qed_sb_info_dbg sb_dbg;
+       int rc;
+
+       /* sb info */
+       qede_fp_sb_dump(edev, fp);
+
+       memset(&sb_dbg, 0, sizeof(sb_dbg));
+       rc = edev->ops->common->get_sb_info(edev->cdev, fp->sb_info, (u16)fp->id, &sb_dbg);
+
+       DP_NOTICE(edev, "IGU: prod %08x cons %08x CAU Tx %04x\n",
+                 sb_dbg.igu_prod, sb_dbg.igu_cons, sb_dbg.pi[TX_PI(txq->cos)]);
+
+       /* report to mfw */
+       edev->ops->common->mfw_report(edev->cdev,
+                                     "Txq[%d]: FW cons [host] %04x, SW cons %04x, SW prod %04x [Jiffies %lu]\n",
+                                     txq->index, le16_to_cpu(*txq->hw_cons_ptr),
+                                     qed_chain_get_cons_idx(&txq->tx_pbl),
+                                     qed_chain_get_prod_idx(&txq->tx_pbl), jiffies);
+       if (!rc)
+               edev->ops->common->mfw_report(edev->cdev,
+                                             "Txq[%d]: SB[0x%04x] - IGU: prod %08x cons %08x CAU Tx %04x\n",
+                                             txq->index, fp->sb_info->igu_sb_id,
+                                             sb_dbg.igu_prod, sb_dbg.igu_cons,
+                                             sb_dbg.pi[TX_PI(txq->cos)]);
 }
 
 static void qede_tx_timeout(struct net_device *dev, unsigned int txqueue)
 {
        struct qede_dev *edev = netdev_priv(dev);
-       struct qede_tx_queue *txq;
-       int cos;
+       int i;
 
        netif_carrier_off(dev);
        DP_NOTICE(edev, "TX timeout on queue %u!\n", txqueue);
 
-       if (!(edev->fp_array[txqueue].type & QEDE_FASTPATH_TX))
-               return;
+       for_each_queue(i) {
+               struct qede_tx_queue *txq;
+               struct qede_fastpath *fp;
+               int cos;
 
-       for_each_cos_in_txq(edev, cos) {
-               txq = &edev->fp_array[txqueue].txq[cos];
+               fp = &edev->fp_array[i];
+               if (!(fp->type & QEDE_FASTPATH_TX))
+                       continue;
 
-               if (qed_chain_get_cons_idx(&txq->tx_pbl) !=
-                   qed_chain_get_prod_idx(&txq->tx_pbl))
-                       qede_tx_log_print(edev, txq);
+               for_each_cos_in_txq(edev, cos) {
+                       txq = &fp->txq[cos];
+
+                       /* Dump basic metadata for all queues */
+                       qede_txq_fp_log_metadata(edev, fp, txq);
+
+                       if (qed_chain_get_cons_idx(&txq->tx_pbl) !=
+                           qed_chain_get_prod_idx(&txq->tx_pbl))
+                               qede_tx_log_print(edev, fp, txq);
+               }
        }
 
        if (IS_VF(edev))
index 0dae7fcc5ef21ed081f4fc15d5159badfb010b14..9f4bfa2a4829782c6eb173e209a7cdcd3e072671 100644 (file)
@@ -807,6 +807,12 @@ struct qed_devlink {
        struct devlink_health_reporter *fw_reporter;
 };
 
+struct qed_sb_info_dbg {
+       u32 igu_prod;
+       u32 igu_cons;
+       u16 pi[PIS_PER_SB];
+};
+
 struct qed_common_cb_ops {
        void (*arfs_filter_op)(void *dev, void *fltr, u8 fw_rc);
        void (*link_update)(void *dev, struct qed_link_output *link);
@@ -1194,6 +1200,11 @@ struct qed_common_ops {
        struct devlink* (*devlink_register)(struct qed_dev *cdev);
 
        void (*devlink_unregister)(struct devlink *devlink);
+
+       __printf(2, 3) void (*mfw_report)(struct qed_dev *cdev, char *fmt, ...);
+
+       int (*get_sb_info)(struct qed_dev *cdev, struct qed_sb_info *sb,
+                          u16 qid, struct qed_sb_info_dbg *sb_dbg);
 };
 
 #define MASK_FIELD(_name, _value) \