• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // SPDX-License-Identifier: MIT
2 /*
3  * Copyright © 2016-2019 Intel Corporation
4  */
5 
6 #include <linux/bitfield.h>
7 #include <linux/firmware.h>
8 #include <drm/drm_print.h>
9 
10 #include "intel_uc_fw.h"
11 #include "intel_uc_fw_abi.h"
12 #include "i915_drv.h"
13 
14 static inline struct intel_gt *
____uc_fw_to_gt(struct intel_uc_fw * uc_fw,enum intel_uc_fw_type type)15 ____uc_fw_to_gt(struct intel_uc_fw *uc_fw, enum intel_uc_fw_type type)
16 {
17 	if (type == INTEL_UC_FW_TYPE_GUC)
18 		return container_of(uc_fw, struct intel_gt, uc.guc.fw);
19 
20 	GEM_BUG_ON(type != INTEL_UC_FW_TYPE_HUC);
21 	return container_of(uc_fw, struct intel_gt, uc.huc.fw);
22 }
23 
__uc_fw_to_gt(struct intel_uc_fw * uc_fw)24 static inline struct intel_gt *__uc_fw_to_gt(struct intel_uc_fw *uc_fw)
25 {
26 	GEM_BUG_ON(uc_fw->status == INTEL_UC_FIRMWARE_UNINITIALIZED);
27 	return ____uc_fw_to_gt(uc_fw, uc_fw->type);
28 }
29 
30 #ifdef CONFIG_DRM_I915_DEBUG_GUC
intel_uc_fw_change_status(struct intel_uc_fw * uc_fw,enum intel_uc_fw_status status)31 void intel_uc_fw_change_status(struct intel_uc_fw *uc_fw,
32 			       enum intel_uc_fw_status status)
33 {
34 	uc_fw->__status =  status;
35 	drm_dbg(&__uc_fw_to_gt(uc_fw)->i915->drm,
36 		"%s firmware -> %s\n",
37 		intel_uc_fw_type_repr(uc_fw->type),
38 		status == INTEL_UC_FIRMWARE_SELECTED ?
39 		uc_fw->path : intel_uc_fw_status_repr(status));
40 }
41 #endif
42 
43 /*
44  * List of required GuC and HuC binaries per-platform.
45  * Must be ordered based on platform + revid, from newer to older.
46  *
47  * Note that RKL uses the same firmware as TGL.
48  */
49 #define INTEL_UC_FIRMWARE_DEFS(fw_def, guc_def, huc_def) \
50 	fw_def(ROCKETLAKE,  0, guc_def(tgl, 49, 0, 1), huc_def(tgl,  7, 5, 0)) \
51 	fw_def(TIGERLAKE,   0, guc_def(tgl, 49, 0, 1), huc_def(tgl,  7, 5, 0)) \
52 	fw_def(ELKHARTLAKE, 0, guc_def(ehl, 49, 0, 1), huc_def(ehl,  9, 0, 0)) \
53 	fw_def(ICELAKE,     0, guc_def(icl, 49, 0, 1), huc_def(icl,  9, 0, 0)) \
54 	fw_def(COMETLAKE,   5, guc_def(cml, 49, 0, 1), huc_def(cml,  4, 0, 0)) \
55 	fw_def(COFFEELAKE,  0, guc_def(kbl, 49, 0, 1), huc_def(kbl,  4, 0, 0)) \
56 	fw_def(GEMINILAKE,  0, guc_def(glk, 49, 0, 1), huc_def(glk,  4, 0, 0)) \
57 	fw_def(KABYLAKE,    0, guc_def(kbl, 49, 0, 1), huc_def(kbl,  4, 0, 0)) \
58 	fw_def(BROXTON,     0, guc_def(bxt, 49, 0, 1), huc_def(bxt,  2, 0, 0)) \
59 	fw_def(SKYLAKE,     0, guc_def(skl, 49, 0, 1), huc_def(skl,  2, 0, 0))
60 
61 #define __MAKE_UC_FW_PATH(prefix_, name_, major_, minor_, patch_) \
62 	"i915/" \
63 	__stringify(prefix_) name_ \
64 	__stringify(major_) "." \
65 	__stringify(minor_) "." \
66 	__stringify(patch_) ".bin"
67 
68 #define MAKE_GUC_FW_PATH(prefix_, major_, minor_, patch_) \
69 	__MAKE_UC_FW_PATH(prefix_, "_guc_", major_, minor_, patch_)
70 
71 #define MAKE_HUC_FW_PATH(prefix_, major_, minor_, bld_num_) \
72 	__MAKE_UC_FW_PATH(prefix_, "_huc_", major_, minor_, bld_num_)
73 
74 /* All blobs need to be declared via MODULE_FIRMWARE() */
75 #define INTEL_UC_MODULE_FW(platform_, revid_, guc_, huc_) \
76 	MODULE_FIRMWARE(guc_); \
77 	MODULE_FIRMWARE(huc_);
78 
79 INTEL_UC_FIRMWARE_DEFS(INTEL_UC_MODULE_FW, MAKE_GUC_FW_PATH, MAKE_HUC_FW_PATH)
80 
81 /* The below structs and macros are used to iterate across the list of blobs */
82 struct __packed uc_fw_blob {
83 	u8 major;
84 	u8 minor;
85 	const char *path;
86 };
87 
88 #define UC_FW_BLOB(major_, minor_, path_) \
89 	{ .major = major_, .minor = minor_, .path = path_ }
90 
91 #define GUC_FW_BLOB(prefix_, major_, minor_, patch_) \
92 	UC_FW_BLOB(major_, minor_, \
93 		   MAKE_GUC_FW_PATH(prefix_, major_, minor_, patch_))
94 
95 #define HUC_FW_BLOB(prefix_, major_, minor_, bld_num_) \
96 	UC_FW_BLOB(major_, minor_, \
97 		   MAKE_HUC_FW_PATH(prefix_, major_, minor_, bld_num_))
98 
99 struct __packed uc_fw_platform_requirement {
100 	enum intel_platform p;
101 	u8 rev; /* first platform rev using this FW */
102 	const struct uc_fw_blob blobs[INTEL_UC_FW_NUM_TYPES];
103 };
104 
105 #define MAKE_FW_LIST(platform_, revid_, guc_, huc_) \
106 { \
107 	.p = INTEL_##platform_, \
108 	.rev = revid_, \
109 	.blobs[INTEL_UC_FW_TYPE_GUC] = guc_, \
110 	.blobs[INTEL_UC_FW_TYPE_HUC] = huc_, \
111 },
112 
113 static void
__uc_fw_auto_select(struct drm_i915_private * i915,struct intel_uc_fw * uc_fw)114 __uc_fw_auto_select(struct drm_i915_private *i915, struct intel_uc_fw *uc_fw)
115 {
116 	static const struct uc_fw_platform_requirement fw_blobs[] = {
117 		INTEL_UC_FIRMWARE_DEFS(MAKE_FW_LIST, GUC_FW_BLOB, HUC_FW_BLOB)
118 	};
119 	enum intel_platform p = INTEL_INFO(i915)->platform;
120 	u8 rev = INTEL_REVID(i915);
121 	int i;
122 
123 	for (i = 0; i < ARRAY_SIZE(fw_blobs) && p <= fw_blobs[i].p; i++) {
124 		if (p == fw_blobs[i].p && rev >= fw_blobs[i].rev) {
125 			const struct uc_fw_blob *blob =
126 					&fw_blobs[i].blobs[uc_fw->type];
127 			uc_fw->path = blob->path;
128 			uc_fw->major_ver_wanted = blob->major;
129 			uc_fw->minor_ver_wanted = blob->minor;
130 			break;
131 		}
132 	}
133 
134 	/* make sure the list is ordered as expected */
135 	if (IS_ENABLED(CONFIG_DRM_I915_SELFTEST)) {
136 		for (i = 1; i < ARRAY_SIZE(fw_blobs); i++) {
137 			if (fw_blobs[i].p < fw_blobs[i - 1].p)
138 				continue;
139 
140 			if (fw_blobs[i].p == fw_blobs[i - 1].p &&
141 			    fw_blobs[i].rev < fw_blobs[i - 1].rev)
142 				continue;
143 
144 			pr_err("invalid FW blob order: %s r%u comes before %s r%u\n",
145 			       intel_platform_name(fw_blobs[i - 1].p),
146 			       fw_blobs[i - 1].rev,
147 			       intel_platform_name(fw_blobs[i].p),
148 			       fw_blobs[i].rev);
149 
150 			uc_fw->path = NULL;
151 		}
152 	}
153 
154 	/* We don't want to enable GuC/HuC on pre-Gen11 by default */
155 	if (i915->params.enable_guc == -1 && p < INTEL_ICELAKE)
156 		uc_fw->path = NULL;
157 }
158 
__override_guc_firmware_path(struct drm_i915_private * i915)159 static const char *__override_guc_firmware_path(struct drm_i915_private *i915)
160 {
161 	if (i915->params.enable_guc & (ENABLE_GUC_SUBMISSION |
162 				       ENABLE_GUC_LOAD_HUC))
163 		return i915->params.guc_firmware_path;
164 	return "";
165 }
166 
__override_huc_firmware_path(struct drm_i915_private * i915)167 static const char *__override_huc_firmware_path(struct drm_i915_private *i915)
168 {
169 	if (i915->params.enable_guc & ENABLE_GUC_LOAD_HUC)
170 		return i915->params.huc_firmware_path;
171 	return "";
172 }
173 
__uc_fw_user_override(struct drm_i915_private * i915,struct intel_uc_fw * uc_fw)174 static void __uc_fw_user_override(struct drm_i915_private *i915, struct intel_uc_fw *uc_fw)
175 {
176 	const char *path = NULL;
177 
178 	switch (uc_fw->type) {
179 	case INTEL_UC_FW_TYPE_GUC:
180 		path = __override_guc_firmware_path(i915);
181 		break;
182 	case INTEL_UC_FW_TYPE_HUC:
183 		path = __override_huc_firmware_path(i915);
184 		break;
185 	}
186 
187 	if (unlikely(path)) {
188 		uc_fw->path = path;
189 		uc_fw->user_overridden = true;
190 	}
191 }
192 
193 /**
194  * intel_uc_fw_init_early - initialize the uC object and select the firmware
195  * @uc_fw: uC firmware
196  * @type: type of uC
197  *
198  * Initialize the state of our uC object and relevant tracking and select the
199  * firmware to fetch and load.
200  */
intel_uc_fw_init_early(struct intel_uc_fw * uc_fw,enum intel_uc_fw_type type)201 void intel_uc_fw_init_early(struct intel_uc_fw *uc_fw,
202 			    enum intel_uc_fw_type type)
203 {
204 	struct drm_i915_private *i915 = ____uc_fw_to_gt(uc_fw, type)->i915;
205 
206 	/*
207 	 * we use FIRMWARE_UNINITIALIZED to detect checks against uc_fw->status
208 	 * before we're looked at the HW caps to see if we have uc support
209 	 */
210 	BUILD_BUG_ON(INTEL_UC_FIRMWARE_UNINITIALIZED);
211 	GEM_BUG_ON(uc_fw->status);
212 	GEM_BUG_ON(uc_fw->path);
213 
214 	uc_fw->type = type;
215 
216 	if (HAS_GT_UC(i915)) {
217 		__uc_fw_auto_select(i915, uc_fw);
218 		__uc_fw_user_override(i915, uc_fw);
219 	}
220 
221 	intel_uc_fw_change_status(uc_fw, uc_fw->path ? *uc_fw->path ?
222 				  INTEL_UC_FIRMWARE_SELECTED :
223 				  INTEL_UC_FIRMWARE_DISABLED :
224 				  INTEL_UC_FIRMWARE_NOT_SUPPORTED);
225 }
226 
__force_fw_fetch_failures(struct intel_uc_fw * uc_fw,int e)227 static void __force_fw_fetch_failures(struct intel_uc_fw *uc_fw, int e)
228 {
229 	struct drm_i915_private *i915 = __uc_fw_to_gt(uc_fw)->i915;
230 	bool user = e == -EINVAL;
231 
232 	if (i915_inject_probe_error(i915, e)) {
233 		/* non-existing blob */
234 		uc_fw->path = "<invalid>";
235 		uc_fw->user_overridden = user;
236 	} else if (i915_inject_probe_error(i915, e)) {
237 		/* require next major version */
238 		uc_fw->major_ver_wanted += 1;
239 		uc_fw->minor_ver_wanted = 0;
240 		uc_fw->user_overridden = user;
241 	} else if (i915_inject_probe_error(i915, e)) {
242 		/* require next minor version */
243 		uc_fw->minor_ver_wanted += 1;
244 		uc_fw->user_overridden = user;
245 	} else if (uc_fw->major_ver_wanted &&
246 		   i915_inject_probe_error(i915, e)) {
247 		/* require prev major version */
248 		uc_fw->major_ver_wanted -= 1;
249 		uc_fw->minor_ver_wanted = 0;
250 		uc_fw->user_overridden = user;
251 	} else if (uc_fw->minor_ver_wanted &&
252 		   i915_inject_probe_error(i915, e)) {
253 		/* require prev minor version - hey, this should work! */
254 		uc_fw->minor_ver_wanted -= 1;
255 		uc_fw->user_overridden = user;
256 	} else if (user && i915_inject_probe_error(i915, e)) {
257 		/* officially unsupported platform */
258 		uc_fw->major_ver_wanted = 0;
259 		uc_fw->minor_ver_wanted = 0;
260 		uc_fw->user_overridden = true;
261 	}
262 }
263 
264 /**
265  * intel_uc_fw_fetch - fetch uC firmware
266  * @uc_fw: uC firmware
267  *
268  * Fetch uC firmware into GEM obj.
269  *
270  * Return: 0 on success, a negative errno code on failure.
271  */
intel_uc_fw_fetch(struct intel_uc_fw * uc_fw)272 int intel_uc_fw_fetch(struct intel_uc_fw *uc_fw)
273 {
274 	struct drm_i915_private *i915 = __uc_fw_to_gt(uc_fw)->i915;
275 	struct device *dev = i915->drm.dev;
276 	struct drm_i915_gem_object *obj;
277 	const struct firmware *fw = NULL;
278 	struct uc_css_header *css;
279 	size_t size;
280 	int err;
281 
282 	GEM_BUG_ON(!i915->wopcm.size);
283 	GEM_BUG_ON(!intel_uc_fw_is_enabled(uc_fw));
284 
285 	err = i915_inject_probe_error(i915, -ENXIO);
286 	if (err)
287 		goto fail;
288 
289 	__force_fw_fetch_failures(uc_fw, -EINVAL);
290 	__force_fw_fetch_failures(uc_fw, -ESTALE);
291 
292 	err = request_firmware(&fw, uc_fw->path, dev);
293 	if (err)
294 		goto fail;
295 
296 	/* Check the size of the blob before examining buffer contents */
297 	if (unlikely(fw->size < sizeof(struct uc_css_header))) {
298 		drm_warn(&i915->drm, "%s firmware %s: invalid size: %zu < %zu\n",
299 			 intel_uc_fw_type_repr(uc_fw->type), uc_fw->path,
300 			 fw->size, sizeof(struct uc_css_header));
301 		err = -ENODATA;
302 		goto fail;
303 	}
304 
305 	css = (struct uc_css_header *)fw->data;
306 
307 	/* Check integrity of size values inside CSS header */
308 	size = (css->header_size_dw - css->key_size_dw - css->modulus_size_dw -
309 		css->exponent_size_dw) * sizeof(u32);
310 	if (unlikely(size != sizeof(struct uc_css_header))) {
311 		drm_warn(&i915->drm,
312 			 "%s firmware %s: unexpected header size: %zu != %zu\n",
313 			 intel_uc_fw_type_repr(uc_fw->type), uc_fw->path,
314 			 fw->size, sizeof(struct uc_css_header));
315 		err = -EPROTO;
316 		goto fail;
317 	}
318 
319 	/* uCode size must calculated from other sizes */
320 	uc_fw->ucode_size = (css->size_dw - css->header_size_dw) * sizeof(u32);
321 
322 	/* now RSA */
323 	if (unlikely(css->key_size_dw != UOS_RSA_SCRATCH_COUNT)) {
324 		drm_warn(&i915->drm, "%s firmware %s: unexpected key size: %u != %u\n",
325 			 intel_uc_fw_type_repr(uc_fw->type), uc_fw->path,
326 			 css->key_size_dw, UOS_RSA_SCRATCH_COUNT);
327 		err = -EPROTO;
328 		goto fail;
329 	}
330 	uc_fw->rsa_size = css->key_size_dw * sizeof(u32);
331 
332 	/* At least, it should have header, uCode and RSA. Size of all three. */
333 	size = sizeof(struct uc_css_header) + uc_fw->ucode_size + uc_fw->rsa_size;
334 	if (unlikely(fw->size < size)) {
335 		drm_warn(&i915->drm, "%s firmware %s: invalid size: %zu < %zu\n",
336 			 intel_uc_fw_type_repr(uc_fw->type), uc_fw->path,
337 			 fw->size, size);
338 		err = -ENOEXEC;
339 		goto fail;
340 	}
341 
342 	/* Sanity check whether this fw is not larger than whole WOPCM memory */
343 	size = __intel_uc_fw_get_upload_size(uc_fw);
344 	if (unlikely(size >= i915->wopcm.size)) {
345 		drm_warn(&i915->drm, "%s firmware %s: invalid size: %zu > %zu\n",
346 			 intel_uc_fw_type_repr(uc_fw->type), uc_fw->path,
347 			 size, (size_t)i915->wopcm.size);
348 		err = -E2BIG;
349 		goto fail;
350 	}
351 
352 	/* Get version numbers from the CSS header */
353 	uc_fw->major_ver_found = FIELD_GET(CSS_SW_VERSION_UC_MAJOR,
354 					   css->sw_version);
355 	uc_fw->minor_ver_found = FIELD_GET(CSS_SW_VERSION_UC_MINOR,
356 					   css->sw_version);
357 
358 	if (uc_fw->major_ver_found != uc_fw->major_ver_wanted ||
359 	    uc_fw->minor_ver_found < uc_fw->minor_ver_wanted) {
360 		drm_notice(&i915->drm, "%s firmware %s: unexpected version: %u.%u != %u.%u\n",
361 			   intel_uc_fw_type_repr(uc_fw->type), uc_fw->path,
362 			   uc_fw->major_ver_found, uc_fw->minor_ver_found,
363 			   uc_fw->major_ver_wanted, uc_fw->minor_ver_wanted);
364 		if (!intel_uc_fw_is_overridden(uc_fw)) {
365 			err = -ENOEXEC;
366 			goto fail;
367 		}
368 	}
369 
370 	if (uc_fw->type == INTEL_UC_FW_TYPE_GUC)
371 		uc_fw->private_data_size = css->private_data_size;
372 
373 	obj = i915_gem_object_create_shmem_from_data(i915, fw->data, fw->size);
374 	if (IS_ERR(obj)) {
375 		err = PTR_ERR(obj);
376 		goto fail;
377 	}
378 
379 	uc_fw->obj = obj;
380 	uc_fw->size = fw->size;
381 	intel_uc_fw_change_status(uc_fw, INTEL_UC_FIRMWARE_AVAILABLE);
382 
383 	release_firmware(fw);
384 	return 0;
385 
386 fail:
387 	intel_uc_fw_change_status(uc_fw, err == -ENOENT ?
388 				  INTEL_UC_FIRMWARE_MISSING :
389 				  INTEL_UC_FIRMWARE_ERROR);
390 
391 	drm_notice(&i915->drm, "%s firmware %s: fetch failed with error %d\n",
392 		   intel_uc_fw_type_repr(uc_fw->type), uc_fw->path, err);
393 	drm_info(&i915->drm, "%s firmware(s) can be downloaded from %s\n",
394 		 intel_uc_fw_type_repr(uc_fw->type), INTEL_UC_FIRMWARE_URL);
395 
396 	release_firmware(fw);		/* OK even if fw is NULL */
397 	return err;
398 }
399 
uc_fw_ggtt_offset(struct intel_uc_fw * uc_fw)400 static u32 uc_fw_ggtt_offset(struct intel_uc_fw *uc_fw)
401 {
402 	struct i915_ggtt *ggtt = __uc_fw_to_gt(uc_fw)->ggtt;
403 	struct drm_mm_node *node = &ggtt->uc_fw;
404 
405 	GEM_BUG_ON(!drm_mm_node_allocated(node));
406 	GEM_BUG_ON(upper_32_bits(node->start));
407 	GEM_BUG_ON(upper_32_bits(node->start + node->size - 1));
408 
409 	return lower_32_bits(node->start);
410 }
411 
uc_fw_bind_ggtt(struct intel_uc_fw * uc_fw)412 static void uc_fw_bind_ggtt(struct intel_uc_fw *uc_fw)
413 {
414 	struct drm_i915_gem_object *obj = uc_fw->obj;
415 	struct i915_ggtt *ggtt = __uc_fw_to_gt(uc_fw)->ggtt;
416 	struct i915_vma dummy = {
417 		.node.start = uc_fw_ggtt_offset(uc_fw),
418 		.node.size = obj->base.size,
419 		.pages = obj->mm.pages,
420 		.vm = &ggtt->vm,
421 	};
422 
423 	GEM_BUG_ON(!i915_gem_object_has_pinned_pages(obj));
424 	GEM_BUG_ON(dummy.node.size > ggtt->uc_fw.size);
425 
426 	/* uc_fw->obj cache domains were not controlled across suspend */
427 	drm_clflush_sg(dummy.pages);
428 
429 	ggtt->vm.insert_entries(&ggtt->vm, &dummy, I915_CACHE_NONE, 0);
430 }
431 
uc_fw_unbind_ggtt(struct intel_uc_fw * uc_fw)432 static void uc_fw_unbind_ggtt(struct intel_uc_fw *uc_fw)
433 {
434 	struct drm_i915_gem_object *obj = uc_fw->obj;
435 	struct i915_ggtt *ggtt = __uc_fw_to_gt(uc_fw)->ggtt;
436 	u64 start = uc_fw_ggtt_offset(uc_fw);
437 
438 	ggtt->vm.clear_range(&ggtt->vm, start, obj->base.size);
439 }
440 
uc_fw_xfer(struct intel_uc_fw * uc_fw,u32 dst_offset,u32 dma_flags)441 static int uc_fw_xfer(struct intel_uc_fw *uc_fw, u32 dst_offset, u32 dma_flags)
442 {
443 	struct intel_gt *gt = __uc_fw_to_gt(uc_fw);
444 	struct intel_uncore *uncore = gt->uncore;
445 	u64 offset;
446 	int ret;
447 
448 	ret = i915_inject_probe_error(gt->i915, -ETIMEDOUT);
449 	if (ret)
450 		return ret;
451 
452 	intel_uncore_forcewake_get(uncore, FORCEWAKE_ALL);
453 
454 	/* Set the source address for the uCode */
455 	offset = uc_fw_ggtt_offset(uc_fw);
456 	GEM_BUG_ON(upper_32_bits(offset) & 0xFFFF0000);
457 	intel_uncore_write_fw(uncore, DMA_ADDR_0_LOW, lower_32_bits(offset));
458 	intel_uncore_write_fw(uncore, DMA_ADDR_0_HIGH, upper_32_bits(offset));
459 
460 	/* Set the DMA destination */
461 	intel_uncore_write_fw(uncore, DMA_ADDR_1_LOW, dst_offset);
462 	intel_uncore_write_fw(uncore, DMA_ADDR_1_HIGH, DMA_ADDRESS_SPACE_WOPCM);
463 
464 	/*
465 	 * Set the transfer size. The header plus uCode will be copied to WOPCM
466 	 * via DMA, excluding any other components
467 	 */
468 	intel_uncore_write_fw(uncore, DMA_COPY_SIZE,
469 			      sizeof(struct uc_css_header) + uc_fw->ucode_size);
470 
471 	/* Start the DMA */
472 	intel_uncore_write_fw(uncore, DMA_CTRL,
473 			      _MASKED_BIT_ENABLE(dma_flags | START_DMA));
474 
475 	/* Wait for DMA to finish */
476 	ret = intel_wait_for_register_fw(uncore, DMA_CTRL, START_DMA, 0, 100);
477 	if (ret)
478 		drm_err(&gt->i915->drm, "DMA for %s fw failed, DMA_CTRL=%u\n",
479 			intel_uc_fw_type_repr(uc_fw->type),
480 			intel_uncore_read_fw(uncore, DMA_CTRL));
481 
482 	/* Disable the bits once DMA is over */
483 	intel_uncore_write_fw(uncore, DMA_CTRL, _MASKED_BIT_DISABLE(dma_flags));
484 
485 	intel_uncore_forcewake_put(uncore, FORCEWAKE_ALL);
486 
487 	return ret;
488 }
489 
490 /**
491  * intel_uc_fw_upload - load uC firmware using custom loader
492  * @uc_fw: uC firmware
493  * @dst_offset: destination offset
494  * @dma_flags: flags for flags for dma ctrl
495  *
496  * Loads uC firmware and updates internal flags.
497  *
498  * Return: 0 on success, non-zero on failure.
499  */
intel_uc_fw_upload(struct intel_uc_fw * uc_fw,u32 dst_offset,u32 dma_flags)500 int intel_uc_fw_upload(struct intel_uc_fw *uc_fw, u32 dst_offset, u32 dma_flags)
501 {
502 	struct intel_gt *gt = __uc_fw_to_gt(uc_fw);
503 	int err;
504 
505 	/* make sure the status was cleared the last time we reset the uc */
506 	GEM_BUG_ON(intel_uc_fw_is_loaded(uc_fw));
507 
508 	err = i915_inject_probe_error(gt->i915, -ENOEXEC);
509 	if (err)
510 		return err;
511 
512 	if (!intel_uc_fw_is_loadable(uc_fw))
513 		return -ENOEXEC;
514 
515 	/* Call custom loader */
516 	uc_fw_bind_ggtt(uc_fw);
517 	err = uc_fw_xfer(uc_fw, dst_offset, dma_flags);
518 	uc_fw_unbind_ggtt(uc_fw);
519 	if (err)
520 		goto fail;
521 
522 	intel_uc_fw_change_status(uc_fw, INTEL_UC_FIRMWARE_TRANSFERRED);
523 	return 0;
524 
525 fail:
526 	i915_probe_error(gt->i915, "Failed to load %s firmware %s (%d)\n",
527 			 intel_uc_fw_type_repr(uc_fw->type), uc_fw->path,
528 			 err);
529 	intel_uc_fw_change_status(uc_fw, INTEL_UC_FIRMWARE_FAIL);
530 	return err;
531 }
532 
intel_uc_fw_init(struct intel_uc_fw * uc_fw)533 int intel_uc_fw_init(struct intel_uc_fw *uc_fw)
534 {
535 	int err;
536 
537 	/* this should happen before the load! */
538 	GEM_BUG_ON(intel_uc_fw_is_loaded(uc_fw));
539 
540 	if (!intel_uc_fw_is_available(uc_fw))
541 		return -ENOEXEC;
542 
543 	err = i915_gem_object_pin_pages(uc_fw->obj);
544 	if (err) {
545 		DRM_DEBUG_DRIVER("%s fw pin-pages err=%d\n",
546 				 intel_uc_fw_type_repr(uc_fw->type), err);
547 		intel_uc_fw_change_status(uc_fw, INTEL_UC_FIRMWARE_FAIL);
548 	}
549 
550 	return err;
551 }
552 
intel_uc_fw_fini(struct intel_uc_fw * uc_fw)553 void intel_uc_fw_fini(struct intel_uc_fw *uc_fw)
554 {
555 	if (i915_gem_object_has_pinned_pages(uc_fw->obj))
556 		i915_gem_object_unpin_pages(uc_fw->obj);
557 
558 	intel_uc_fw_change_status(uc_fw, INTEL_UC_FIRMWARE_AVAILABLE);
559 }
560 
561 /**
562  * intel_uc_fw_cleanup_fetch - cleanup uC firmware
563  * @uc_fw: uC firmware
564  *
565  * Cleans up uC firmware by releasing the firmware GEM obj.
566  */
intel_uc_fw_cleanup_fetch(struct intel_uc_fw * uc_fw)567 void intel_uc_fw_cleanup_fetch(struct intel_uc_fw *uc_fw)
568 {
569 	if (!intel_uc_fw_is_available(uc_fw))
570 		return;
571 
572 	i915_gem_object_put(fetch_and_zero(&uc_fw->obj));
573 
574 	intel_uc_fw_change_status(uc_fw, INTEL_UC_FIRMWARE_SELECTED);
575 }
576 
577 /**
578  * intel_uc_fw_copy_rsa - copy fw RSA to buffer
579  *
580  * @uc_fw: uC firmware
581  * @dst: dst buffer
582  * @max_len: max number of bytes to copy
583  *
584  * Return: number of copied bytes.
585  */
intel_uc_fw_copy_rsa(struct intel_uc_fw * uc_fw,void * dst,u32 max_len)586 size_t intel_uc_fw_copy_rsa(struct intel_uc_fw *uc_fw, void *dst, u32 max_len)
587 {
588 	struct sg_table *pages = uc_fw->obj->mm.pages;
589 	u32 size = min_t(u32, uc_fw->rsa_size, max_len);
590 	u32 offset = sizeof(struct uc_css_header) + uc_fw->ucode_size;
591 
592 	GEM_BUG_ON(!intel_uc_fw_is_available(uc_fw));
593 
594 	return sg_pcopy_to_buffer(pages->sgl, pages->nents, dst, size, offset);
595 }
596 
597 /**
598  * intel_uc_fw_dump - dump information about uC firmware
599  * @uc_fw: uC firmware
600  * @p: the &drm_printer
601  *
602  * Pretty printer for uC firmware.
603  */
intel_uc_fw_dump(const struct intel_uc_fw * uc_fw,struct drm_printer * p)604 void intel_uc_fw_dump(const struct intel_uc_fw *uc_fw, struct drm_printer *p)
605 {
606 	drm_printf(p, "%s firmware: %s\n",
607 		   intel_uc_fw_type_repr(uc_fw->type), uc_fw->path);
608 	drm_printf(p, "\tstatus: %s\n",
609 		   intel_uc_fw_status_repr(uc_fw->status));
610 	drm_printf(p, "\tversion: wanted %u.%u, found %u.%u\n",
611 		   uc_fw->major_ver_wanted, uc_fw->minor_ver_wanted,
612 		   uc_fw->major_ver_found, uc_fw->minor_ver_found);
613 	drm_printf(p, "\tuCode: %u bytes\n", uc_fw->ucode_size);
614 	drm_printf(p, "\tRSA: %u bytes\n", uc_fw->rsa_size);
615 }
616