From: Zerun Fu <zerun...@corigine.com>

The original logic try to get the NSP resource every time the application
call 'rte_eth_dev_fw_version_get()' interface, there are chances the NSP
resource busy and fail to get it. And because the local string variables
not initialized, there will be out of range problem when it fail to get
the NSP resource.

Fix this by initializing the local string variables and storing the
firmware version string once we get it.

Fixes: 128c8ad951bf ("net/nfp: support getting firmware version")
Cc: sta...@dpdk.org

Signed-off-by: Zerun Fu <zerun...@corigine.com>
Reviewed-by: Chaoyong He <chaoyong...@corigine.com>
Reviewed-by: Long Wu <long...@corigine.com>
Reviewed-by: Peng Zhang <peng.zh...@corigine.com>
---
 drivers/net/nfp/nfp_net_common.c | 26 ++++++++++++++++++--------
 drivers/net/nfp/nfp_net_common.h |  6 ++++++
 2 files changed, 24 insertions(+), 8 deletions(-)

diff --git a/drivers/net/nfp/nfp_net_common.c b/drivers/net/nfp/nfp_net_common.c
index a6573ed45c..84893e2d73 100644
--- a/drivers/net/nfp/nfp_net_common.c
+++ b/drivers/net/nfp/nfp_net_common.c
@@ -193,9 +193,6 @@ nfp_net_notify_port_speed(struct nfp_net_hw *hw,
                        nfp_net_link_speed_rte2nfp(link->link_speed));
 }
 
-/* The length of firmware version string */
-#define FW_VER_LEN        32
-
 /**
  * Reconfigure the firmware of VF configure
  *
@@ -2274,11 +2271,11 @@ nfp_net_firmware_version_get(struct rte_eth_dev *dev,
                size_t fw_size)
 {
        struct nfp_net_hw *hw;
-       char mip_name[FW_VER_LEN];
-       char app_name[FW_VER_LEN];
-       char nsp_version[FW_VER_LEN];
-       char vnic_version[FW_VER_LEN];
        struct nfp_net_hw_priv *hw_priv;
+       char app_name[FW_VER_LEN] = {0};
+       char mip_name[FW_VER_LEN] = {0};
+       char nsp_version[FW_VER_LEN] = {0};
+       char vnic_version[FW_VER_LEN] = {0};
 
        if (fw_size < FW_VER_LEN)
                return FW_VER_LEN;
@@ -2286,6 +2283,11 @@ nfp_net_firmware_version_get(struct rte_eth_dev *dev,
        hw = nfp_net_get_hw(dev);
        hw_priv = dev->process_private;
 
+       if (hw->fw_version[0] != 0) {
+               snprintf(fw_version, FW_VER_LEN, "%s", hw->fw_version);
+               return 0;
+       }
+
        if (!rte_eth_dev_is_repr(dev)) {
                snprintf(vnic_version, FW_VER_LEN, "%d.%d.%d.%d",
                        hw->ver.extend, hw->ver.class,
@@ -2298,8 +2300,16 @@ nfp_net_firmware_version_get(struct rte_eth_dev *dev,
        nfp_net_get_mip_name(hw_priv, mip_name);
        nfp_net_get_app_name(hw_priv, app_name);
 
-       snprintf(fw_version, FW_VER_LEN, "%s %s %s %s",
+       if (nsp_version[0] == 0 || mip_name[0] == 0) {
+               snprintf(fw_version, FW_VER_LEN, "%s %s %s %s",
                        vnic_version, nsp_version, mip_name, app_name);
+               return 0;
+       }
+
+       snprintf(hw->fw_version, FW_VER_LEN, "%s %s %s %s",
+                       vnic_version, nsp_version, mip_name, app_name);
+
+       snprintf(fw_version, FW_VER_LEN, "%s", hw->fw_version);
 
        return 0;
 }
diff --git a/drivers/net/nfp/nfp_net_common.h b/drivers/net/nfp/nfp_net_common.h
index 6c739891be..3c4d305b01 100644
--- a/drivers/net/nfp/nfp_net_common.h
+++ b/drivers/net/nfp/nfp_net_common.h
@@ -50,6 +50,9 @@
                                RTE_ETH_RSS_NONFRAG_IPV6_UDP  | \
                                RTE_ETH_RSS_NONFRAG_IPV6_SCTP)
 
+/* The length of firmware version string */
+#define FW_VER_LEN        32
+
 /*
  * Each PF has corresponding word to beat:
  * Offset | Usage
@@ -255,6 +258,9 @@ struct nfp_net_hw {
 
        /** Used for rte_flow of CoreNIC firmware */
        struct nfp_net_priv *priv;
+
+       /** Used for firmware version */
+       char fw_version[FW_VER_LEN];
 };
 
 static inline uint32_t
-- 
2.39.1

Reply via email to