]> www.infradead.org Git - users/hch/xfs.git/commitdiff
drm/xe/huc: Extract version and binary offset from new HuC headers
authorDaniele Ceraolo Spurio <daniele.ceraolospurio@intel.com>
Wed, 25 Oct 2023 17:57:42 +0000 (10:57 -0700)
committerRodrigo Vivi <rodrigo.vivi@intel.com>
Thu, 21 Dec 2023 16:43:22 +0000 (11:43 -0500)
The GSC-enabled HuC binary starts with a GSC header, which is followed
by the legacy-style CSS header and the binary itself. We can parse the
GSC headers to find the HuC version and the location of the binary to
be used for the DMA transfer.

The parsing function has been designed to be re-used for the GSC binary,
so the entry names are external parameters (because the GSC uses
different ones) and the CSS entry is optional (because the GSC doesn't
have it).

v2: move new code to uc_fw.c, better comments and error checking, split
    old code move to separate patch (Lucas), move headers and
    documentation to uc_fw_abi.h.

v3: use 2 separate loops, rework marker check (Lucas)

Signed-off-by: Daniele Ceraolo Spurio <daniele.ceraolospurio@intel.com>
Cc: Alan Previn <alan.previn.teres.alexis@intel.com>
Cc: John Harrison <John.C.Harrison@Intel.com>
Cc: Lucas De Marchi <lucas.demarchi@intel.com>
Reviewed-by: Lucas De Marchi <lucas.demarchi@intel.com>
Signed-off-by: Rodrigo Vivi <rodrigo.vivi@intel.com>
Documentation/gpu/xe/xe_firmware.rst
drivers/gpu/drm/xe/xe_uc_fw.c
drivers/gpu/drm/xe/xe_uc_fw.h
drivers/gpu/drm/xe/xe_uc_fw_abi.h
drivers/gpu/drm/xe/xe_uc_fw_types.h

index f1ac6f60893052417d00179769200b7b24f60ec0..afcb561cd37db227b3d0a22faea599545d4ff883 100644 (file)
@@ -10,6 +10,9 @@ Firmware Layout
 .. kernel-doc:: drivers/gpu/drm/xe/xe_uc_fw_abi.h
    :doc: CSS-based Firmware Layout
 
+.. kernel-doc:: drivers/gpu/drm/xe/xe_uc_fw_abi.h
+   :doc: GSC-based Firmware Layout
+
 Write Once Protected Content Memory (WOPCM) Layout
 ==================================================
 
index 189a298e54790e5673a8d45319886752fb81aaea..d1cc99f86ec013719f4a4595491c97ecd9d5e25d 100644 (file)
@@ -402,9 +402,125 @@ static int parse_css_header(struct xe_uc_fw *uc_fw, const void *fw_data, size_t
        return 0;
 }
 
+static bool is_cpd_header(const void *data)
+{
+       const u32 *marker = data;
+
+       return *marker == GSC_CPD_HEADER_MARKER;
+}
+
+static u32 entry_offset(const struct gsc_cpd_header_v2 *header, const char *name)
+{
+       const struct gsc_cpd_entry *entry;
+       int i;
+
+       entry = (void *)header + header->header_length;
+
+       for (i = 0; i < header->num_of_entries; i++, entry++)
+               if (strcmp(entry->name, name) == 0)
+                       return entry->offset & GSC_CPD_ENTRY_OFFSET_MASK;
+
+       return 0;
+}
+
+/* Refer to the "GSC-based Firmware Layout" documentation entry for details */
+static int parse_cpd_header(struct xe_uc_fw *uc_fw, const void *data, size_t size,
+                           const char *manifest_entry, const char *css_entry)
+{
+       struct xe_gt *gt = uc_fw_to_gt(uc_fw);
+       struct xe_device *xe = gt_to_xe(gt);
+       const struct gsc_cpd_header_v2 *header = data;
+       const struct gsc_manifest_header *manifest;
+       size_t min_size = sizeof(*header);
+       u32 offset;
+
+       /* manifest_entry is mandatory, css_entry is optional */
+       xe_assert(xe, manifest_entry);
+
+       if (size < min_size || !is_cpd_header(header))
+               return -ENOENT;
+
+       if (header->header_length < sizeof(struct gsc_cpd_header_v2)) {
+               xe_gt_err(gt, "invalid CPD header length %u!\n", header->header_length);
+               return -EINVAL;
+       }
+
+       min_size = header->header_length + sizeof(struct gsc_cpd_entry) * header->num_of_entries;
+       if (size < min_size) {
+               xe_gt_err(gt, "FW too small! %zu < %zu\n", size, min_size);
+               return -ENODATA;
+       }
+
+       /* Look for the manifest first */
+       offset = entry_offset(header, manifest_entry);
+       if (!offset) {
+               xe_gt_err(gt, "Failed to find %s manifest!\n",
+                         xe_uc_fw_type_repr(uc_fw->type));
+               return -ENODATA;
+       }
+
+       min_size = offset + sizeof(struct gsc_manifest_header);
+       if (size < min_size) {
+               xe_gt_err(gt, "FW too small! %zu < %zu\n", size, min_size);
+               return -ENODATA;
+       }
+
+       manifest = data + offset;
+
+       uc_fw->major_ver_found = manifest->fw_version.major;
+       uc_fw->minor_ver_found = manifest->fw_version.minor;
+       uc_fw->patch_ver_found = manifest->fw_version.hotfix;
+
+       /* then optionally look for the css header */
+       if (css_entry) {
+               int ret;
+
+               /*
+                * This section does not contain a CSS entry on DG2. We
+                * don't support DG2 HuC right now, so no need to handle
+                * it, just add a reminder in case that changes.
+                */
+               xe_assert(xe, xe->info.platform != XE_DG2);
+
+               offset = entry_offset(header, css_entry);
+
+               /* the CSS header parser will check that the CSS header fits */
+               if (offset > size) {
+                       xe_gt_err(gt, "FW too small! %zu < %u\n", size, offset);
+                       return -ENODATA;
+               }
+
+               ret = parse_css_header(uc_fw, data + offset, size - offset);
+               if (ret)
+                       return ret;
+
+               uc_fw->css_offset = offset;
+       }
+
+       return 0;
+}
+
 static int parse_headers(struct xe_uc_fw *uc_fw, const struct firmware *fw)
 {
-       return parse_css_header(uc_fw, fw->data, fw->size);
+       int ret;
+
+       /*
+        * All GuC releases and older HuC ones use CSS headers, while newer HuC
+        * releases use GSC CPD headers.
+        */
+       switch (uc_fw->type) {
+       case XE_UC_FW_TYPE_HUC:
+               ret = parse_cpd_header(uc_fw, fw->data, fw->size, "HUCP.man", "huc_fw");
+               if (!ret || ret != -ENOENT)
+                       return ret;
+               fallthrough;
+       case XE_UC_FW_TYPE_GUC:
+               return parse_css_header(uc_fw, fw->data, fw->size);
+       default:
+               return -EINVAL;
+       }
+
+       return 0;
 }
 
 int xe_uc_fw_init(struct xe_uc_fw *uc_fw)
@@ -510,7 +626,7 @@ static int uc_fw_xfer(struct xe_uc_fw *uc_fw, u32 offset, u32 dma_flags)
        xe_force_wake_assert_held(gt_to_fw(gt), XE_FW_GT);
 
        /* Set the source address for the uCode */
-       src_offset = uc_fw_ggtt_offset(uc_fw);
+       src_offset = uc_fw_ggtt_offset(uc_fw) + uc_fw->css_offset;
        xe_mmio_write32(gt, DMA_ADDR_0_LOW, lower_32_bits(src_offset));
        xe_mmio_write32(gt, DMA_ADDR_0_HIGH, upper_32_bits(src_offset));
 
index a519c77d496216947d43b51eba880e4f305b4314..1d1a0c156cdfd1ba45a86b6b287e5731b701d620 100644 (file)
@@ -21,7 +21,7 @@ void xe_uc_fw_print(struct xe_uc_fw *uc_fw, struct drm_printer *p);
 
 static inline u32 xe_uc_fw_rsa_offset(struct xe_uc_fw *uc_fw)
 {
-       return sizeof(struct uc_css_header) + uc_fw->ucode_size;
+       return sizeof(struct uc_css_header) + uc_fw->ucode_size + uc_fw->css_offset;
 }
 
 static inline void xe_uc_fw_change_status(struct xe_uc_fw *uc_fw,
index edae7bb3cd7203c3d364a2f1a9938e8efbbede2b..d6725c963251749349c4f708e0b56153479a53af 100644 (file)
@@ -85,4 +85,124 @@ struct uc_css_header {
 } __packed;
 static_assert(sizeof(struct uc_css_header) == 128);
 
+/**
+ * DOC: GSC-based Firmware Layout
+ *
+ * The GSC-based firmware structure is used for GSC releases on all platforms
+ * and for HuC releases starting from DG2/MTL. Older HuC releases use the
+ * CSS-based layout instead. Differently from the CSS headers, the GSC headers
+ * uses a directory + entries structure (i.e., there is array of addresses
+ * pointing to specific header extensions identified by a name). Although the
+ * header structures are the same, some of the entries are specific to GSC while
+ * others are specific to HuC. The manifest header entry, which includes basic
+ * information about the binary (like the version) is always present, but it is
+ * named differently based on the binary type.
+ *
+ * The HuC binary starts with a Code Partition Directory (CPD) header. The
+ * entries we're interested in for use in the driver are:
+ *
+ * 1. "HUCP.man": points to the manifest header for the HuC.
+ * 2. "huc_fw": points to the FW code. On platforms that support load via DMA
+ *    and 2-step HuC authentication (i.e. MTL+) this is a full CSS-based binary,
+ *    while if the GSC is the one doing the load (which only happens on DG2)
+ *    this section only contains the uCode.
+ *
+ * The GSC-based HuC firmware layout looks like this::
+ *
+ *     +================================================+
+ *     |  CPD Header                                    |
+ *     +================================================+
+ *     |  CPD entries[]                                 |
+ *     |      entry1                                    |
+ *     |      ...                                       |
+ *     |      entryX                                    |
+ *     |          "HUCP.man"                            |
+ *     |           ...                                  |
+ *     |           offset  >----------------------------|------o
+ *     |      ...                                       |      |
+ *     |      entryY                                    |      |
+ *     |          "huc_fw"                              |      |
+ *     |           ...                                  |      |
+ *     |           offset  >----------------------------|----------o
+ *     +================================================+      |   |
+ *                                                             |   |
+ *     +================================================+      |   |
+ *     |  Manifest Header                               |<-----o   |
+ *     |      ...                                       |          |
+ *     |      FW version                                |          |
+ *     |      ...                                       |          |
+ *     +================================================+          |
+ *                                                                 |
+ *     +================================================+          |
+ *     |  FW binary                                     |<---------o
+ *     |      CSS (MTL+ only)                           |
+ *     |      uCode                                     |
+ *     |      RSA Key (MTL+ only)                       |
+ *     |      ...                                       |
+ *     +================================================+
+ */
+
+struct gsc_version {
+       u16 major;
+       u16 minor;
+       u16 hotfix;
+       u16 build;
+} __packed;
+
+/* Code partition directory (CPD) structures */
+struct gsc_cpd_header_v2 {
+       u32 header_marker;
+#define GSC_CPD_HEADER_MARKER 0x44504324
+
+       u32 num_of_entries;
+       u8 header_version;
+       u8 entry_version;
+       u8 header_length; /* in bytes */
+       u8 flags;
+       u32 partition_name;
+       u32 crc32;
+} __packed;
+
+struct gsc_cpd_entry {
+       u8 name[12];
+
+       /*
+        * Bits 0-24: offset from the beginning of the code partition
+        * Bit 25: huffman compressed
+        * Bits 26-31: reserved
+        */
+       u32 offset;
+#define GSC_CPD_ENTRY_OFFSET_MASK GENMASK(24, 0)
+#define GSC_CPD_ENTRY_HUFFMAN_COMP BIT(25)
+
+       /*
+        * Module/Item length, in bytes. For Huffman-compressed modules, this
+        * refers to the uncompressed size. For software-compressed modules,
+        * this refers to the compressed size.
+        */
+       u32 length;
+
+       u8 reserved[4];
+} __packed;
+
+struct gsc_manifest_header {
+       u32 header_type; /* 0x4 for manifest type */
+       u32 header_length; /* in dwords */
+       u32 header_version;
+       u32 flags;
+       u32 vendor;
+       u32 date;
+       u32 size; /* In dwords, size of entire manifest (header + extensions) */
+       u32 header_id;
+       u32 internal_data;
+       struct gsc_version fw_version;
+       u32 security_version;
+       struct gsc_version meu_kit_version;
+       u32 meu_manifest_version;
+       u8 general_data[4];
+       u8 reserved3[56];
+       u32 modulus_size; /* in dwords */
+       u32 exponent_size; /* in dwords */
+} __packed;
+
 #endif
index 444bff83cdbe5ab3643646c66a8ae34765bdf66a..1650599303c85cc1b1d8c195abbcf1571cbe33c3 100644 (file)
@@ -113,6 +113,8 @@ struct xe_uc_fw {
        u32 rsa_size;
        /** @ucode_size: micro kernel size */
        u32 ucode_size;
+       /** @css_offset: offset within the blob at which the CSS is located */
+       u32 css_offset;
 
        /** @private_data_size: size of private data found in uC css header */
        u32 private_data_size;