accel/habanalabs: extract and save the FW's SW major/minor/sub-minor
authorDafna Hirschfeld <dhirschfeld@habana.ai>
Thu, 16 Mar 2023 08:20:44 +0000 (10:20 +0200)
committerOded Gabbay <ogabbay@kernel.org>
Mon, 5 Jun 2023 12:31:33 +0000 (15:31 +0300)
It is not always possible to know the FW's SW version from the inner FW
version. Therefore we should extract the general SW version in addition
to the FW version and use it in functions like
'hl_is_fw_ver_below_1_9' etc.

Signed-off-by: Dafna Hirschfeld <dhirschfeld@habana.ai>
Reviewed-by: Oded Gabbay <ogabbay@kernel.org>
Signed-off-by: Oded Gabbay <ogabbay@kernel.org>
drivers/accel/habanalabs/common/firmware_if.c
drivers/accel/habanalabs/common/habanalabs.h

index 6150ab6ba810472a7b63949a5b9656d6390d3194..62052cfe694ffb3bff8529882a1500b33726a769 100644 (file)
@@ -93,6 +93,71 @@ static char *extract_u32_until_given_char(char *str, u32 *ver_num, char given_ch
        return ch;
 }
 
+/**
+ * hl_get_sw_major_minor_subminor() - extract the FW's SW version major, minor, sub-minor
+ *                                   from the version string
+ * @hdev: pointer to the hl_device
+ * @fw_str: the FW's version string
+ *
+ * The extracted version is set in the hdev fields: fw_sw_{major/minor/sub_minor}_ver.
+ *
+ * fw_str is expected to have one of two possible formats, examples:
+ * 1) 'Preboot version hl-gaudi2-1.9.0-fw-42.0.1-sec-3'
+ * 2) 'Preboot version hl-gaudi2-1.9.0-rc-fw-42.0.1-sec-3'
+ * In those examples, the SW major,minor,subminor are correspondingly: 1,9,0.
+ *
+ * Return: 0 for success or a negative error code for failure.
+ */
+static int hl_get_sw_major_minor_subminor(struct hl_device *hdev, const char *fw_str)
+{
+       char *end, *start;
+
+       end = strnstr(fw_str, "-rc-", VERSION_MAX_LEN);
+       if (end == fw_str)
+               return -EINVAL;
+
+       if (!end)
+               end = strnstr(fw_str, "-fw-", VERSION_MAX_LEN);
+
+       if (end == fw_str)
+               return -EINVAL;
+
+       if (!end)
+               return -EINVAL;
+
+       for (start = end - 1; start != fw_str; start--) {
+               if (*start == '-')
+                       break;
+       }
+
+       if (start == fw_str)
+               return -EINVAL;
+
+       /* start/end point each to the starting and ending hyphen of the sw version e.g. -1.9.0- */
+       start++;
+       start = extract_u32_until_given_char(start, &hdev->fw_sw_major_ver, '.');
+       if (!start)
+               goto err_zero_ver;
+
+       start++;
+       start = extract_u32_until_given_char(start, &hdev->fw_sw_minor_ver, '.');
+       if (!start)
+               goto err_zero_ver;
+
+       start++;
+       start = extract_u32_until_given_char(start, &hdev->fw_sw_sub_minor_ver, '-');
+       if (!start)
+               goto err_zero_ver;
+
+       return 0;
+
+err_zero_ver:
+       hdev->fw_sw_major_ver = 0;
+       hdev->fw_sw_minor_ver = 0;
+       hdev->fw_sw_sub_minor_ver = 0;
+       return -EINVAL;
+}
+
 /**
  * hl_get_preboot_major_minor() - extract the FW's version major, minor from the version string.
  * @hdev: pointer to the hl_device
@@ -2172,6 +2237,7 @@ static int hl_fw_dynamic_read_device_fw_version(struct hl_device *hdev,
        struct asic_fixed_properties *prop = &hdev->asic_prop;
        char *preboot_ver, *boot_ver;
        char btl_ver[32];
+       int rc;
 
        switch (fwc) {
        case FW_COMP_BOOT_FIT:
@@ -2185,20 +2251,20 @@ static int hl_fw_dynamic_read_device_fw_version(struct hl_device *hdev,
                break;
        case FW_COMP_PREBOOT:
                strscpy(prop->preboot_ver, fw_version, VERSION_MAX_LEN);
-               preboot_ver = strnstr(prop->preboot_ver, "Preboot",
-                                               VERSION_MAX_LEN);
+               preboot_ver = strnstr(prop->preboot_ver, "Preboot", VERSION_MAX_LEN);
+               dev_info(hdev->dev, "preboot full version: '%s'\n", preboot_ver);
+
                if (preboot_ver && preboot_ver != prop->preboot_ver) {
                        strscpy(btl_ver, prop->preboot_ver,
                                min((int) (preboot_ver - prop->preboot_ver), 31));
                        dev_info(hdev->dev, "%s\n", btl_ver);
                }
 
+               rc = hl_get_sw_major_minor_subminor(hdev, preboot_ver);
+               if (rc)
+                       return rc;
                preboot_ver = extract_fw_ver_from_str(prop->preboot_ver);
                if (preboot_ver) {
-                       int rc;
-
-                       dev_info(hdev->dev, "preboot version %s\n", preboot_ver);
-
                        rc = hl_get_preboot_major_minor(hdev, preboot_ver);
                        kfree(preboot_ver);
                        if (rc)
index 57661cb516218d658e91c17186ecced4cb62d05e..d6f454b1cde4832857bdd1b7978a323afbf683d8 100644 (file)
@@ -3227,6 +3227,9 @@ struct hl_reset_info {
  * @stream_master_qid_arr: pointer to array with QIDs of master streams.
  * @fw_inner_major_ver: the major of current loaded preboot inner version.
  * @fw_inner_minor_ver: the minor of current loaded preboot inner version.
+ * @fw_sw_major_ver: the major of current loaded preboot SW version.
+ * @fw_sw_minor_ver: the minor of current loaded preboot SW version.
+ * @fw_sw_sub_minor_ver: the sub-minor of current loaded preboot SW version.
  * @dram_used_mem: current DRAM memory consumption.
  * @memory_scrub_val: the value to which the dram will be scrubbed to using cb scrub_device_dram
  * @timeout_jiffies: device CS timeout value.
@@ -3414,6 +3417,9 @@ struct hl_device {
        u32                             *stream_master_qid_arr;
        u32                             fw_inner_major_ver;
        u32                             fw_inner_minor_ver;
+       u32                             fw_sw_major_ver;
+       u32                             fw_sw_minor_ver;
+       u32                             fw_sw_sub_minor_ver;
        atomic64_t                      dram_used_mem;
        u64                             memory_scrub_val;
        u64                             timeout_jiffies;