xref: /linux/drivers/gpu/drm/xe/xe_nvm.c (revision 9156bf442ee56c0f883aa4c81af9c8471eef6846)
1 // SPDX-License-Identifier: MIT
2 /*
3  * Copyright(c) 2019-2025, Intel Corporation. All rights reserved.
4  */
5 
6 #include <linux/intel_dg_nvm_aux.h>
7 #include <linux/pci.h>
8 
9 #include "xe_device_types.h"
10 #include "xe_mmio.h"
11 #include "xe_nvm.h"
12 #include "xe_pcode_api.h"
13 #include "regs/xe_gsc_regs.h"
14 #include "xe_sriov.h"
15 
16 #define GEN12_GUNIT_NVM_BASE 0x00102040
17 #define GEN12_DEBUG_NVM_BASE 0x00101018
18 
19 #define GEN12_CNTL_PROTECTED_NVM_REG 0x0010100C
20 
21 #define GEN12_GUNIT_NVM_SIZE 0x80
22 #define GEN12_DEBUG_NVM_SIZE 0x4
23 
24 #define NVM_NON_POSTED_ERASE_CHICKEN_BIT BIT(13)
25 
26 #define HECI_FW_STATUS_2_NVM_ACCESS_MODE BIT(3)
27 
28 static const struct intel_dg_nvm_region regions[INTEL_DG_NVM_REGIONS] = {
29 	[0] = { .name = "DESCRIPTOR", },
30 	[2] = { .name = "GSC", },
31 	[9] = { .name = "PADDING", },
32 	[11] = { .name = "OptionROM", },
33 	[12] = { .name = "DAM", },
34 };
35 
36 static void xe_nvm_release_dev(struct device *dev)
37 {
38 	struct auxiliary_device *aux = container_of(dev, struct auxiliary_device, dev);
39 	struct intel_dg_nvm_dev *nvm = container_of(aux, struct intel_dg_nvm_dev, aux_dev);
40 
41 	kfree(nvm);
42 }
43 
44 static bool xe_nvm_non_posted_erase(struct xe_device *xe)
45 {
46 	struct xe_mmio *mmio = xe_root_tile_mmio(xe);
47 
48 	switch (xe->info.platform) {
49 	case XE_CRESCENTISLAND:
50 	case XE_BATTLEMAGE:
51 		return !(xe_mmio_read32(mmio, XE_REG(GEN12_CNTL_PROTECTED_NVM_REG)) &
52 			 NVM_NON_POSTED_ERASE_CHICKEN_BIT);
53 	default:
54 		return false;
55 	}
56 }
57 
58 static bool xe_nvm_writable_override(struct xe_device *xe)
59 {
60 	struct xe_mmio *mmio = xe_root_tile_mmio(xe);
61 	bool writable_override;
62 	struct xe_reg reg;
63 	u32 test_bit;
64 
65 	switch (xe->info.platform) {
66 	case XE_CRESCENTISLAND:
67 		reg = PCODE_SCRATCH(0);
68 		test_bit = FDO_MODE;
69 		break;
70 	case XE_BATTLEMAGE:
71 		reg = HECI_FWSTS2(DG2_GSC_HECI2_BASE);
72 		test_bit = HECI_FW_STATUS_2_NVM_ACCESS_MODE;
73 		break;
74 	case XE_PVC:
75 		reg = HECI_FWSTS2(PVC_GSC_HECI2_BASE);
76 		test_bit = HECI_FW_STATUS_2_NVM_ACCESS_MODE;
77 		break;
78 	case XE_DG2:
79 		reg = HECI_FWSTS2(DG2_GSC_HECI2_BASE);
80 		test_bit = HECI_FW_STATUS_2_NVM_ACCESS_MODE;
81 		break;
82 	case XE_DG1:
83 		reg = HECI_FWSTS2(DG1_GSC_HECI2_BASE);
84 		test_bit = HECI_FW_STATUS_2_NVM_ACCESS_MODE;
85 		break;
86 	default:
87 		drm_err(&xe->drm, "Unknown platform\n");
88 		return true;
89 	}
90 
91 	writable_override = !(xe_mmio_read32(mmio, reg) & test_bit);
92 	if (writable_override)
93 		drm_info(&xe->drm, "NVM access overridden by jumper\n");
94 	return writable_override;
95 }
96 
97 int xe_nvm_init(struct xe_device *xe)
98 {
99 	struct pci_dev *pdev = to_pci_dev(xe->drm.dev);
100 	struct auxiliary_device *aux_dev;
101 	struct intel_dg_nvm_dev *nvm;
102 	int ret;
103 
104 	if (!xe->info.has_gsc_nvm)
105 		return 0;
106 
107 	/* No access to internal NVM from VFs */
108 	if (IS_SRIOV_VF(xe))
109 		return 0;
110 
111 	/* Nvm pointer should be NULL here */
112 	if (WARN_ON(xe->nvm))
113 		return -EFAULT;
114 
115 	xe->nvm = kzalloc(sizeof(*nvm), GFP_KERNEL);
116 	if (!xe->nvm)
117 		return -ENOMEM;
118 
119 	nvm = xe->nvm;
120 
121 	nvm->writable_override = xe_nvm_writable_override(xe);
122 	nvm->non_posted_erase = xe_nvm_non_posted_erase(xe);
123 	nvm->bar.parent = &pdev->resource[0];
124 	nvm->bar.start = GEN12_GUNIT_NVM_BASE + pdev->resource[0].start;
125 	nvm->bar.end = nvm->bar.start + GEN12_GUNIT_NVM_SIZE - 1;
126 	nvm->bar.flags = IORESOURCE_MEM;
127 	nvm->bar.desc = IORES_DESC_NONE;
128 	nvm->regions = regions;
129 
130 	nvm->bar2.parent = &pdev->resource[0];
131 	nvm->bar2.start = GEN12_DEBUG_NVM_BASE + pdev->resource[0].start;
132 	nvm->bar2.end = nvm->bar2.start + GEN12_DEBUG_NVM_SIZE - 1;
133 	nvm->bar2.flags = IORESOURCE_MEM;
134 	nvm->bar2.desc = IORES_DESC_NONE;
135 
136 	aux_dev = &nvm->aux_dev;
137 
138 	aux_dev->name = "nvm";
139 	aux_dev->id = (pci_domain_nr(pdev->bus) << 16) | pci_dev_id(pdev);
140 	aux_dev->dev.parent = &pdev->dev;
141 	aux_dev->dev.release = xe_nvm_release_dev;
142 
143 	ret = auxiliary_device_init(aux_dev);
144 	if (ret) {
145 		drm_err(&xe->drm, "xe-nvm aux init failed %d\n", ret);
146 		goto err;
147 	}
148 
149 	ret = auxiliary_device_add(aux_dev);
150 	if (ret) {
151 		drm_err(&xe->drm, "xe-nvm aux add failed %d\n", ret);
152 		auxiliary_device_uninit(aux_dev);
153 		goto err;
154 	}
155 	return 0;
156 
157 err:
158 	kfree(nvm);
159 	xe->nvm = NULL;
160 	return ret;
161 }
162 
163 void xe_nvm_fini(struct xe_device *xe)
164 {
165 	struct intel_dg_nvm_dev *nvm = xe->nvm;
166 
167 	if (!xe->info.has_gsc_nvm)
168 		return;
169 
170 	/* No access to internal NVM from VFs */
171 	if (IS_SRIOV_VF(xe))
172 		return;
173 
174 	/* Nvm pointer should not be NULL here */
175 	if (WARN_ON(!nvm))
176 		return;
177 
178 	auxiliary_device_delete(&nvm->aux_dev);
179 	auxiliary_device_uninit(&nvm->aux_dev);
180 	xe->nvm = NULL;
181 }
182