]> www.infradead.org Git - linux.git/commitdiff
ixgbe: extend .info_get() with stored versions
authorJedrzej Jagielski <jedrzej.jagielski@intel.com>
Thu, 10 Apr 2025 13:00:03 +0000 (15:00 +0200)
committerTony Nguyen <anthony.l.nguyen@intel.com>
Tue, 15 Apr 2025 14:36:32 +0000 (07:36 -0700)
Add functions reading inactive versions from the inactive flash
bank.

Print stored versions for the content present in the inactive bank.
If there's pending update the versions reflect the ones which
are going to be loaded after reload. If there's no pending update
both running and stored are the same, which means there won't
be any NVM change on reload.

Co-developed-by: Slawomir Mrozowicz <slawomirx.mrozowicz@intel.com>
Signed-off-by: Slawomir Mrozowicz <slawomirx.mrozowicz@intel.com>
Co-developed-by: Piotr Kwapulinski <piotr.kwapulinski@intel.com>
Signed-off-by: Piotr Kwapulinski <piotr.kwapulinski@intel.com>
Signed-off-by: Jedrzej Jagielski <jedrzej.jagielski@intel.com>
Signed-off-by: Tony Nguyen <anthony.l.nguyen@intel.com>
drivers/net/ethernet/intel/ixgbe/devlink/devlink.c
drivers/net/ethernet/intel/ixgbe/ixgbe_e610.c
drivers/net/ethernet/intel/ixgbe/ixgbe_e610.h

index a7c39e951c7b749969ca4d9c64d251f76c0a7bd2..f7bafc1dad2bb5ce440d8b525c4e20ed3554dadb 100644 (file)
@@ -6,6 +6,15 @@
 
 struct ixgbe_info_ctx {
        char buf[128];
+       struct ixgbe_orom_info pending_orom;
+       struct ixgbe_nvm_info pending_nvm;
+       struct ixgbe_netlist_info pending_netlist;
+       struct ixgbe_hw_dev_caps dev_caps;
+};
+
+enum ixgbe_devlink_version_type {
+       IXGBE_DL_VERSION_RUNNING,
+       IXGBE_DL_VERSION_STORED
 };
 
 static void ixgbe_info_get_dsn(struct ixgbe_adapter *adapter,
@@ -20,7 +29,8 @@ static void ixgbe_info_get_dsn(struct ixgbe_adapter *adapter,
 }
 
 static void ixgbe_info_orom_ver(struct ixgbe_adapter *adapter,
-                               struct ixgbe_info_ctx *ctx)
+                               struct ixgbe_info_ctx *ctx,
+                               enum ixgbe_devlink_version_type type)
 {
        struct ixgbe_hw *hw = &adapter->hw;
        struct ixgbe_nvm_version nvm_ver;
@@ -30,6 +40,10 @@ static void ixgbe_info_orom_ver(struct ixgbe_adapter *adapter,
        if (hw->mac.type == ixgbe_mac_e610) {
                struct ixgbe_orom_info *orom = &adapter->hw.flash.orom;
 
+               if (type == IXGBE_DL_VERSION_STORED &&
+                   ctx->dev_caps.common_cap.nvm_update_pending_orom)
+                       orom = &ctx->pending_orom;
+
                snprintf(ctx->buf, sizeof(ctx->buf), "%u.%u.%u",
                         orom->major, orom->build, orom->patch);
                return;
@@ -51,14 +65,20 @@ static void ixgbe_info_orom_ver(struct ixgbe_adapter *adapter,
 }
 
 static void ixgbe_info_eetrack(struct ixgbe_adapter *adapter,
-                              struct ixgbe_info_ctx *ctx)
+                              struct ixgbe_info_ctx *ctx,
+                              enum ixgbe_devlink_version_type type)
 {
        struct ixgbe_hw *hw = &adapter->hw;
        struct ixgbe_nvm_version nvm_ver;
 
        if (hw->mac.type == ixgbe_mac_e610) {
-               snprintf(ctx->buf, sizeof(ctx->buf), "0x%08x",
-                        hw->flash.nvm.eetrack);
+               u32 eetrack = hw->flash.nvm.eetrack;
+
+               if (type == IXGBE_DL_VERSION_STORED &&
+                   ctx->dev_caps.common_cap.nvm_update_pending_nvm)
+                       eetrack = ctx->pending_nvm.eetrack;
+
+               snprintf(ctx->buf, sizeof(ctx->buf), "0x%08x", eetrack);
                return;
        }
 
@@ -92,34 +112,54 @@ static void ixgbe_info_fw_build(struct ixgbe_adapter *adapter,
 }
 
 static void ixgbe_info_fw_srev(struct ixgbe_adapter *adapter,
-                              struct ixgbe_info_ctx *ctx)
+                              struct ixgbe_info_ctx *ctx,
+                              enum ixgbe_devlink_version_type type)
 {
        struct ixgbe_nvm_info *nvm = &adapter->hw.flash.nvm;
 
+       if (type == IXGBE_DL_VERSION_STORED &&
+           ctx->dev_caps.common_cap.nvm_update_pending_nvm)
+               nvm = &ctx->pending_nvm;
+
        snprintf(ctx->buf, sizeof(ctx->buf), "%u", nvm->srev);
 }
 
 static void ixgbe_info_orom_srev(struct ixgbe_adapter *adapter,
-                                struct ixgbe_info_ctx *ctx)
+                                struct ixgbe_info_ctx *ctx,
+                                enum ixgbe_devlink_version_type type)
 {
        struct ixgbe_orom_info *orom = &adapter->hw.flash.orom;
 
+       if (type == IXGBE_DL_VERSION_STORED &&
+           ctx->dev_caps.common_cap.nvm_update_pending_orom)
+               orom = &ctx->pending_orom;
+
        snprintf(ctx->buf, sizeof(ctx->buf), "%u", orom->srev);
 }
 
 static void ixgbe_info_nvm_ver(struct ixgbe_adapter *adapter,
-                              struct ixgbe_info_ctx *ctx)
+                              struct ixgbe_info_ctx *ctx,
+                              enum ixgbe_devlink_version_type type)
 {
        struct ixgbe_nvm_info *nvm = &adapter->hw.flash.nvm;
 
+       if (type == IXGBE_DL_VERSION_STORED &&
+           ctx->dev_caps.common_cap.nvm_update_pending_nvm)
+               nvm = &ctx->pending_nvm;
+
        snprintf(ctx->buf, sizeof(ctx->buf), "%x.%02x", nvm->major, nvm->minor);
 }
 
 static void ixgbe_info_netlist_ver(struct ixgbe_adapter *adapter,
-                                  struct ixgbe_info_ctx *ctx)
+                                  struct ixgbe_info_ctx *ctx,
+                                  enum ixgbe_devlink_version_type type)
 {
        struct ixgbe_netlist_info *netlist = &adapter->hw.flash.netlist;
 
+       if (type == IXGBE_DL_VERSION_STORED &&
+           ctx->dev_caps.common_cap.nvm_update_pending_netlist)
+               netlist = &ctx->pending_netlist;
+
        /* The netlist version fields are BCD formatted */
        snprintf(ctx->buf, sizeof(ctx->buf), "%x.%x.%x-%x.%x.%x",
                 netlist->major, netlist->minor,
@@ -128,13 +168,57 @@ static void ixgbe_info_netlist_ver(struct ixgbe_adapter *adapter,
 }
 
 static void ixgbe_info_netlist_build(struct ixgbe_adapter *adapter,
-                                    struct ixgbe_info_ctx *ctx)
+                                    struct ixgbe_info_ctx *ctx,
+                                    enum ixgbe_devlink_version_type type)
 {
        struct ixgbe_netlist_info *netlist = &adapter->hw.flash.netlist;
 
+       if (type == IXGBE_DL_VERSION_STORED &&
+           ctx->dev_caps.common_cap.nvm_update_pending_netlist)
+               netlist = &ctx->pending_netlist;
+
        snprintf(ctx->buf, sizeof(ctx->buf), "0x%08x", netlist->hash);
 }
 
+static int ixgbe_set_ctx_dev_caps(struct ixgbe_hw *hw,
+                                 struct ixgbe_info_ctx *ctx,
+                                 struct netlink_ext_ack *extack)
+{
+       bool *pending_orom, *pending_nvm, *pending_netlist;
+       int err;
+
+       err = ixgbe_discover_dev_caps(hw, &ctx->dev_caps);
+       if (err) {
+               NL_SET_ERR_MSG_MOD(extack,
+                                  "Unable to discover device capabilities");
+               return err;
+       }
+
+       pending_orom = &ctx->dev_caps.common_cap.nvm_update_pending_orom;
+       pending_nvm = &ctx->dev_caps.common_cap.nvm_update_pending_nvm;
+       pending_netlist = &ctx->dev_caps.common_cap.nvm_update_pending_netlist;
+
+       if (*pending_orom) {
+               err = ixgbe_get_inactive_orom_ver(hw, &ctx->pending_orom);
+               if (err)
+                       *pending_orom = false;
+       }
+
+       if (*pending_nvm) {
+               err = ixgbe_get_inactive_nvm_ver(hw, &ctx->pending_nvm);
+               if (err)
+                       *pending_nvm = false;
+       }
+
+       if (*pending_netlist) {
+               err = ixgbe_get_inactive_netlist_ver(hw, &ctx->pending_netlist);
+               if (err)
+                       *pending_netlist = false;
+       }
+
+       return 0;
+}
+
 static int ixgbe_devlink_info_get_e610(struct ixgbe_adapter *adapter,
                                       struct devlink_info_req *req,
                                       struct ixgbe_info_ctx *ctx)
@@ -153,31 +237,77 @@ static int ixgbe_devlink_info_get_e610(struct ixgbe_adapter *adapter,
        if (err)
                return err;
 
-       ixgbe_info_fw_srev(adapter, ctx);
+       ixgbe_info_fw_srev(adapter, ctx, IXGBE_DL_VERSION_RUNNING);
        err = devlink_info_version_running_put(req, "fw.mgmt.srev", ctx->buf);
        if (err)
                return err;
 
-       ixgbe_info_orom_srev(adapter, ctx);
+       ixgbe_info_orom_srev(adapter, ctx, IXGBE_DL_VERSION_RUNNING);
        err = devlink_info_version_running_put(req, "fw.undi.srev", ctx->buf);
        if (err)
                return err;
 
-       ixgbe_info_nvm_ver(adapter, ctx);
+       ixgbe_info_nvm_ver(adapter, ctx, IXGBE_DL_VERSION_RUNNING);
        err = devlink_info_version_running_put(req, "fw.psid.api", ctx->buf);
        if (err)
                return err;
 
-       ixgbe_info_netlist_ver(adapter, ctx);
+       ixgbe_info_netlist_ver(adapter, ctx, IXGBE_DL_VERSION_RUNNING);
        err = devlink_info_version_running_put(req, "fw.netlist", ctx->buf);
        if (err)
                return err;
 
-       ixgbe_info_netlist_build(adapter, ctx);
+       ixgbe_info_netlist_build(adapter, ctx, IXGBE_DL_VERSION_RUNNING);
        return devlink_info_version_running_put(req, "fw.netlist.build",
                                                ctx->buf);
 }
 
+static int
+ixgbe_devlink_pending_info_get_e610(struct ixgbe_adapter *adapter,
+                                   struct devlink_info_req *req,
+                                   struct ixgbe_info_ctx *ctx)
+{
+       int err;
+
+       ixgbe_info_orom_ver(adapter, ctx, IXGBE_DL_VERSION_STORED);
+       err = devlink_info_version_stored_put(req,
+                                             DEVLINK_INFO_VERSION_GENERIC_FW_UNDI,
+                                             ctx->buf);
+       if (err)
+               return err;
+
+       ixgbe_info_eetrack(adapter, ctx, IXGBE_DL_VERSION_STORED);
+       err = devlink_info_version_stored_put(req,
+                                             DEVLINK_INFO_VERSION_GENERIC_FW_BUNDLE_ID,
+                                             ctx->buf);
+       if (err)
+               return err;
+
+       ixgbe_info_fw_srev(adapter, ctx, IXGBE_DL_VERSION_STORED);
+       err = devlink_info_version_stored_put(req, "fw.mgmt.srev", ctx->buf);
+       if (err)
+               return err;
+
+       ixgbe_info_orom_srev(adapter, ctx, IXGBE_DL_VERSION_STORED);
+       err = devlink_info_version_stored_put(req, "fw.undi.srev", ctx->buf);
+       if (err)
+               return err;
+
+       ixgbe_info_nvm_ver(adapter, ctx, IXGBE_DL_VERSION_STORED);
+       err = devlink_info_version_stored_put(req, "fw.psid.api", ctx->buf);
+       if (err)
+               return err;
+
+       ixgbe_info_netlist_ver(adapter, ctx, IXGBE_DL_VERSION_STORED);
+       err = devlink_info_version_stored_put(req, "fw.netlist", ctx->buf);
+       if (err)
+               return err;
+
+       ixgbe_info_netlist_build(adapter, ctx, IXGBE_DL_VERSION_STORED);
+       return devlink_info_version_stored_put(req, "fw.netlist.build",
+                                              ctx->buf);
+}
+
 static int ixgbe_devlink_info_get(struct devlink *devlink,
                                  struct devlink_info_req *req,
                                  struct netlink_ext_ack *extack)
@@ -206,21 +336,29 @@ static int ixgbe_devlink_info_get(struct devlink *devlink,
        if (err)
                goto free_ctx;
 
-       ixgbe_info_orom_ver(adapter, ctx);
+       ixgbe_info_orom_ver(adapter, ctx, IXGBE_DL_VERSION_RUNNING);
        err = devlink_info_version_running_put(req,
                                               DEVLINK_INFO_VERSION_GENERIC_FW_UNDI,
                                               ctx->buf);
        if (err)
                goto free_ctx;
 
-       ixgbe_info_eetrack(adapter, ctx);
+       ixgbe_info_eetrack(adapter, ctx, IXGBE_DL_VERSION_RUNNING);
        err = devlink_info_version_running_put(req,
                                               DEVLINK_INFO_VERSION_GENERIC_FW_BUNDLE_ID,
                                               ctx->buf);
        if (err || hw->mac.type != ixgbe_mac_e610)
                goto free_ctx;
 
+       err = ixgbe_set_ctx_dev_caps(hw, ctx, extack);
+       if (err)
+               goto free_ctx;
+
        err = ixgbe_devlink_info_get_e610(adapter, req, ctx);
+       if (err)
+               goto free_ctx;
+
+       err = ixgbe_devlink_pending_info_get_e610(adapter, req, ctx);
 free_ctx:
        kfree(ctx);
        return err;
index f856690106af427085b481e3eb8d0135a38d625e..24443db831eb19f1628e5c4a7efd5e7bc7112aff 100644 (file)
@@ -588,6 +588,15 @@ static bool ixgbe_parse_e610_caps(struct ixgbe_hw *hw,
                break;
        case IXGBE_ACI_CAPS_NVM_VER:
                break;
+       case IXGBE_ACI_CAPS_PENDING_NVM_VER:
+               caps->nvm_update_pending_nvm = true;
+               break;
+       case IXGBE_ACI_CAPS_PENDING_OROM_VER:
+               caps->nvm_update_pending_orom = true;
+               break;
+       case IXGBE_ACI_CAPS_PENDING_NET_VER:
+               caps->nvm_update_pending_netlist = true;
+               break;
        case IXGBE_ACI_CAPS_MAX_MTU:
                caps->max_mtu = number;
                break;
@@ -2932,6 +2941,23 @@ static int ixgbe_get_orom_ver_info(struct ixgbe_hw *hw,
        return ixgbe_get_orom_srev(hw, bank, &orom->srev);
 }
 
+/**
+ * ixgbe_get_inactive_orom_ver - Read Option ROM version from the inactive bank
+ * @hw: pointer to the HW structure
+ * @orom: storage for Option ROM version information
+ *
+ * Read the Option ROM version and security revision data for the inactive
+ * section of flash. Used to access version data for a pending update that has
+ * not yet been activated.
+ *
+ * Return: the exit code of the operation.
+ */
+int ixgbe_get_inactive_orom_ver(struct ixgbe_hw *hw,
+                               struct ixgbe_orom_info *orom)
+{
+       return ixgbe_get_orom_ver_info(hw, IXGBE_INACTIVE_FLASH_BANK, orom);
+}
+
 /**
  * ixgbe_get_nvm_ver_info - Read NVM version information
  * @hw: pointer to the HW struct
@@ -2975,6 +3001,22 @@ static int ixgbe_get_nvm_ver_info(struct ixgbe_hw *hw,
        return 0;
 }
 
+/**
+ * ixgbe_get_inactive_nvm_ver - Read Option ROM version from the inactive bank
+ * @hw: pointer to the HW structure
+ * @nvm: storage for Option ROM version information
+ *
+ * Read the NVM EETRACK ID, Map version, and security revision of the
+ * inactive NVM bank. Used to access version data for a pending update that
+ * has not yet been activated.
+ *
+ * Return: the exit code of the operation.
+ */
+int ixgbe_get_inactive_nvm_ver(struct ixgbe_hw *hw, struct ixgbe_nvm_info *nvm)
+{
+       return ixgbe_get_nvm_ver_info(hw, IXGBE_INACTIVE_FLASH_BANK, nvm);
+}
+
 /**
  * ixgbe_get_netlist_info - Read the netlist version information
  * @hw: pointer to the HW struct
@@ -3055,6 +3097,23 @@ free_id_blk:
        return err;
 }
 
+/**
+ * ixgbe_get_inactive_netlist_ver - Read netlist version from the inactive bank
+ * @hw: pointer to the HW struct
+ * @netlist: pointer to netlist version info structure
+ *
+ * Read the netlist version data from the inactive netlist bank. Used to
+ * extract version data of a pending flash update in order to display the
+ * version data.
+ *
+ * Return: the exit code of the operation.
+ */
+int ixgbe_get_inactive_netlist_ver(struct ixgbe_hw *hw,
+                                  struct ixgbe_netlist_info *netlist)
+{
+       return ixgbe_get_netlist_info(hw, IXGBE_INACTIVE_FLASH_BANK, netlist);
+}
+
 /**
  * ixgbe_get_flash_data - get flash data
  * @hw: pointer to the HW struct
index 2c971a34200b8c21bcd8edc7694fbfb750c52e01..7565a40d792fcd3bf5fb4e45392ba141a7f31b03 100644 (file)
@@ -67,6 +67,11 @@ int ixgbe_aci_read_nvm(struct ixgbe_hw *hw, u16 module_typeid, u32 offset,
                       u16 length, void *data, bool last_command,
                       bool read_shadow_ram);
 int ixgbe_nvm_validate_checksum(struct ixgbe_hw *hw);
+int ixgbe_get_inactive_orom_ver(struct ixgbe_hw *hw,
+                               struct ixgbe_orom_info *orom);
+int ixgbe_get_inactive_nvm_ver(struct ixgbe_hw *hw, struct ixgbe_nvm_info *nvm);
+int ixgbe_get_inactive_netlist_ver(struct ixgbe_hw *hw,
+                                  struct ixgbe_netlist_info *netlist);
 int ixgbe_read_sr_word_aci(struct ixgbe_hw  *hw, u16 offset, u16 *data);
 int ixgbe_read_flat_nvm(struct ixgbe_hw  *hw, u32 offset, u32 *length,
                        u8 *data, bool read_shadow_ram);