amdgpu_psp.c 88.2 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
/*
 * Copyright 2016 Advanced Micro Devices, Inc.
 *
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 * and/or sell copies of the Software, and to permit persons to whom the
 * Software is furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
 * OTHER DEALINGS IN THE SOFTWARE.
 *
 * Author: Huang Rui
 *
 */

#include <linux/firmware.h>
27
#include <drm/drm_drv.h>
28

29 30 31
#include "amdgpu.h"
#include "amdgpu_psp.h"
#include "amdgpu_ucode.h"
32
#include "amdgpu_xgmi.h"
33 34
#include "soc15_common.h"
#include "psp_v3_1.h"
35
#include "psp_v10_0.h"
36
#include "psp_v11_0.h"
37
#include "psp_v11_0_8.h"
38
#include "psp_v12_0.h"
39
#include "psp_v13_0.h"
40

41
#include "amdgpu_ras.h"
42
#include "amdgpu_securedisplay.h"
43
#include "amdgpu_atomfirmware.h"
44

45 46 47
static int psp_sysfs_init(struct amdgpu_device *adev);
static void psp_sysfs_fini(struct amdgpu_device *adev);

48
static int psp_load_smu_fw(struct psp_context *psp);
49
static int psp_ta_unload(struct psp_context *psp, struct ta_context *context);
50
static int psp_ta_load(struct psp_context *psp, struct ta_context *context);
51 52
static int psp_rap_terminate(struct psp_context *psp);
static int psp_securedisplay_terminate(struct psp_context *psp);
53

54 55 56 57 58 59 60 61 62 63 64 65 66
/*
 * Due to DF Cstate management centralized to PMFW, the firmware
 * loading sequence will be updated as below:
 *   - Load KDB
 *   - Load SYS_DRV
 *   - Load tOS
 *   - Load PMFW
 *   - Setup TMR
 *   - Load other non-psp fw
 *   - Load ASD
 *   - Load XGMI/RAS/HDCP/DTM TA if any
 *
 * This new sequence is required for
67
 *   - Arcturus and onwards
68 69 70 71 72 73
 *   - Navi12 and onwards
 */
static void psp_check_pmfw_centralized_cstate_management(struct psp_context *psp)
{
	struct amdgpu_device *adev = psp->adev;

74 75
	if (amdgpu_sriov_vf(adev)) {
		psp->pmfw_centralized_cstate_management = false;
76
		return;
77
	}
78

79
	switch (adev->ip_versions[MP0_HWIP][0]) {
80 81 82 83 84 85 86
	case IP_VERSION(11, 0, 4):
	case IP_VERSION(11, 0, 7):
	case IP_VERSION(11, 0, 9):
	case IP_VERSION(11, 0, 11):
	case IP_VERSION(11, 0, 12):
	case IP_VERSION(11, 0, 13):
	case IP_VERSION(13, 0, 2):
87
		psp->pmfw_centralized_cstate_management = true;
88 89 90 91 92
		break;
	default:
		psp->pmfw_centralized_cstate_management = false;
		break;
	}
93 94
}

95 96 97
static int psp_early_init(void *handle)
{
	struct amdgpu_device *adev = (struct amdgpu_device *)handle;
98
	struct psp_context *psp = &adev->psp;
99

100
	switch (adev->ip_versions[MP0_HWIP][0]) {
101
	case IP_VERSION(9, 0, 0):
102
		psp_v3_1_set_psp_funcs(psp);
103
		psp->autoload_supported = false;
104
		break;
105 106
	case IP_VERSION(10, 0, 0):
	case IP_VERSION(10, 0, 1):
107
		psp_v10_0_set_psp_funcs(psp);
108
		psp->autoload_supported = false;
109
		break;
110 111
	case IP_VERSION(11, 0, 2):
	case IP_VERSION(11, 0, 4):
112 113 114
		psp_v11_0_set_psp_funcs(psp);
		psp->autoload_supported = false;
		break;
115 116 117 118 119 120 121 122
	case IP_VERSION(11, 0, 0):
	case IP_VERSION(11, 0, 5):
	case IP_VERSION(11, 0, 9):
	case IP_VERSION(11, 0, 7):
	case IP_VERSION(11, 0, 11):
	case IP_VERSION(11, 5, 0):
	case IP_VERSION(11, 0, 12):
	case IP_VERSION(11, 0, 13):
123
		psp_v11_0_set_psp_funcs(psp);
124
		psp->autoload_supported = true;
125
		break;
126 127
	case IP_VERSION(11, 0, 3):
	case IP_VERSION(12, 0, 1):
128 129
		psp_v12_0_set_psp_funcs(psp);
		break;
130
	case IP_VERSION(13, 0, 2):
131 132
		psp_v13_0_set_psp_funcs(psp);
		break;
133 134
	case IP_VERSION(13, 0, 1):
	case IP_VERSION(13, 0, 3):
135 136 137
		psp_v13_0_set_psp_funcs(psp);
		psp->autoload_supported = true;
		break;
138
	case IP_VERSION(11, 0, 8):
139 140 141 142 143
		if (adev->apu_flags & AMD_APU_IS_CYAN_SKILLFISH2) {
			psp_v11_0_8_set_psp_funcs(psp);
			psp->autoload_supported = false;
		}
		break;
144 145 146 147 148 149
	default:
		return -EINVAL;
	}

	psp->adev = adev;

150 151
	psp_check_pmfw_centralized_cstate_management(psp);

152 153 154
	return 0;
}

155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192
static void psp_memory_training_fini(struct psp_context *psp)
{
	struct psp_memory_training_context *ctx = &psp->mem_train_ctx;

	ctx->init = PSP_MEM_TRAIN_NOT_SUPPORT;
	kfree(ctx->sys_cache);
	ctx->sys_cache = NULL;
}

static int psp_memory_training_init(struct psp_context *psp)
{
	int ret;
	struct psp_memory_training_context *ctx = &psp->mem_train_ctx;

	if (ctx->init != PSP_MEM_TRAIN_RESERVE_SUCCESS) {
		DRM_DEBUG("memory training is not supported!\n");
		return 0;
	}

	ctx->sys_cache = kzalloc(ctx->train_data_size, GFP_KERNEL);
	if (ctx->sys_cache == NULL) {
		DRM_ERROR("alloc mem_train_ctx.sys_cache failed!\n");
		ret = -ENOMEM;
		goto Err_out;
	}

	DRM_DEBUG("train_data_size:%llx,p2c_train_data_offset:%llx,c2p_train_data_offset:%llx.\n",
		  ctx->train_data_size,
		  ctx->p2c_train_data_offset,
		  ctx->c2p_train_data_offset);
	ctx->init = PSP_MEM_TRAIN_INIT_SUCCESS;
	return 0;

Err_out:
	psp_memory_training_fini(psp);
	return ret;
}

193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260
/*
 * Helper funciton to query psp runtime database entry
 *
 * @adev: amdgpu_device pointer
 * @entry_type: the type of psp runtime database entry
 * @db_entry: runtime database entry pointer
 *
 * Return false if runtime database doesn't exit or entry is invalid
 * or true if the specific database entry is found, and copy to @db_entry
 */
static bool psp_get_runtime_db_entry(struct amdgpu_device *adev,
				     enum psp_runtime_entry_type entry_type,
				     void *db_entry)
{
	uint64_t db_header_pos, db_dir_pos;
	struct psp_runtime_data_header db_header = {0};
	struct psp_runtime_data_directory db_dir = {0};
	bool ret = false;
	int i;

	db_header_pos = adev->gmc.mc_vram_size - PSP_RUNTIME_DB_OFFSET;
	db_dir_pos = db_header_pos + sizeof(struct psp_runtime_data_header);

	/* read runtime db header from vram */
	amdgpu_device_vram_access(adev, db_header_pos, (uint32_t *)&db_header,
			sizeof(struct psp_runtime_data_header), false);

	if (db_header.cookie != PSP_RUNTIME_DB_COOKIE_ID) {
		/* runtime db doesn't exist, exit */
		dev_warn(adev->dev, "PSP runtime database doesn't exist\n");
		return false;
	}

	/* read runtime database entry from vram */
	amdgpu_device_vram_access(adev, db_dir_pos, (uint32_t *)&db_dir,
			sizeof(struct psp_runtime_data_directory), false);

	if (db_dir.entry_count >= PSP_RUNTIME_DB_DIAG_ENTRY_MAX_COUNT) {
		/* invalid db entry count, exit */
		dev_warn(adev->dev, "Invalid PSP runtime database entry count\n");
		return false;
	}

	/* look up for requested entry type */
	for (i = 0; i < db_dir.entry_count && !ret; i++) {
		if (db_dir.entry_list[i].entry_type == entry_type) {
			switch (entry_type) {
			case PSP_RUNTIME_ENTRY_TYPE_BOOT_CONFIG:
				if (db_dir.entry_list[i].size < sizeof(struct psp_runtime_boot_cfg_entry)) {
					/* invalid db entry size */
					dev_warn(adev->dev, "Invalid PSP runtime database entry size\n");
					return false;
				}
				/* read runtime database entry */
				amdgpu_device_vram_access(adev, db_header_pos + db_dir.entry_list[i].offset,
							  (uint32_t *)db_entry, sizeof(struct psp_runtime_boot_cfg_entry), false);
				ret = true;
				break;
			default:
				ret = false;
				break;
			}
		}
	}

	return ret;
}

261 262 263 264 265
static int psp_sw_init(void *handle)
{
	struct amdgpu_device *adev = (struct amdgpu_device *)handle;
	struct psp_context *psp = &adev->psp;
	int ret;
266
	struct psp_runtime_boot_cfg_entry boot_cfg_entry;
267
	struct psp_memory_training_context *mem_training_ctx = &psp->mem_train_ctx;
268

269 270 271 272 273 274
	psp->cmd = kzalloc(sizeof(struct psp_gfx_cmd_resp), GFP_KERNEL);
	if (!psp->cmd) {
		DRM_ERROR("Failed to allocate memory to command buffer!\n");
		ret = -ENOMEM;
	}

275 276 277 278 279 280
	if (!amdgpu_sriov_vf(adev)) {
		ret = psp_init_microcode(psp);
		if (ret) {
			DRM_ERROR("Failed to load psp firmware!\n");
			return ret;
		}
281
	} else if (amdgpu_sriov_vf(adev) &&
282
		   adev->ip_versions[MP0_HWIP][0] == IP_VERSION(13, 0, 2)) {
283 284 285 286 287
		ret = psp_init_ta_microcode(psp, "aldebaran");
		if (ret) {
			DRM_ERROR("Failed to initialize ta microcode!\n");
			return ret;
		}
288 289
	}

290 291 292
	memset(&boot_cfg_entry, 0, sizeof(boot_cfg_entry));
	if (psp_get_runtime_db_entry(adev,
				PSP_RUNTIME_ENTRY_TYPE_BOOT_CONFIG,
293
				&boot_cfg_entry)) {
294
		psp->boot_cfg_bitmask = boot_cfg_entry.boot_cfg_bitmask;
295 296 297 298 299 300 301 302
		if ((psp->boot_cfg_bitmask) &
		    BOOT_CFG_FEATURE_TWO_STAGE_DRAM_TRAINING) {
			/* If psp runtime database exists, then
			 * only enable two stage memory training
			 * when TWO_STAGE_DRAM_TRAINING bit is set
			 * in runtime database */
			mem_training_ctx->enable_mem_training = true;
		}
303

304 305 306 307 308
	} else {
		/* If psp runtime database doesn't exist or
		 * is invalid, force enable two stage memory
		 * training */
		mem_training_ctx->enable_mem_training = true;
309
	}
310 311 312 313 314 315 316 317 318 319 320 321 322

	if (mem_training_ctx->enable_mem_training) {
		ret = psp_memory_training_init(psp);
		if (ret) {
			DRM_ERROR("Failed to initialize memory training!\n");
			return ret;
		}

		ret = psp_mem_training(psp, PSP_MEM_TRAIN_COLD_BOOT);
		if (ret) {
			DRM_ERROR("Failed to process memory training!\n");
			return ret;
		}
323 324
	}

325 326
	if (adev->ip_versions[MP0_HWIP][0] == IP_VERSION(11, 0, 0) ||
	    adev->ip_versions[MP0_HWIP][0] == IP_VERSION(11, 0, 7)) {
327 328 329 330 331 332
		ret= psp_sysfs_init(adev);
		if (ret) {
			return ret;
		}
	}

333 334 335 336 337
	return 0;
}

static int psp_sw_fini(void *handle)
{
338
	struct amdgpu_device *adev = (struct amdgpu_device *)handle;
339 340
	struct psp_context *psp = &adev->psp;
	struct psp_gfx_cmd_resp *cmd = psp->cmd;
341

342 343 344 345
	psp_memory_training_fini(psp);
	if (psp->sos_fw) {
		release_firmware(psp->sos_fw);
		psp->sos_fw = NULL;
346
	}
347 348 349
	if (psp->asd_fw) {
		release_firmware(psp->asd_fw);
		psp->asd_fw = NULL;
350
	}
351 352 353
	if (psp->ta_fw) {
		release_firmware(psp->ta_fw);
		psp->ta_fw = NULL;
354
	}
355

356 357
	if (adev->ip_versions[MP0_HWIP][0] == IP_VERSION(11, 0, 0) ||
	    adev->ip_versions[MP0_HWIP][0] == IP_VERSION(11, 0, 7))
358 359
		psp_sysfs_fini(adev);

360 361 362
	kfree(cmd);
	cmd = NULL;

363 364 365 366 367 368 369 370 371 372
	return 0;
}

int psp_wait_for(struct psp_context *psp, uint32_t reg_index,
		 uint32_t reg_val, uint32_t mask, bool check_changed)
{
	uint32_t val;
	int i;
	struct amdgpu_device *adev = psp->adev;

373
	if (psp->adev->no_hw_access)
374 375
		return 0;

376
	for (i = 0; i < adev->usec_timeout; i++) {
377
		val = RREG32(reg_index);
378 379 380 381 382 383 384 385 386 387 388 389 390
		if (check_changed) {
			if (val != reg_val)
				return 0;
		} else {
			if ((val & mask) == reg_val)
				return 0;
		}
		udelay(1);
	}

	return -ETIME;
}

391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428
static const char *psp_gfx_cmd_name(enum psp_gfx_cmd_id cmd_id)
{
	switch (cmd_id) {
	case GFX_CMD_ID_LOAD_TA:
		return "LOAD_TA";
	case GFX_CMD_ID_UNLOAD_TA:
		return "UNLOAD_TA";
	case GFX_CMD_ID_INVOKE_CMD:
		return "INVOKE_CMD";
	case GFX_CMD_ID_LOAD_ASD:
		return "LOAD_ASD";
	case GFX_CMD_ID_SETUP_TMR:
		return "SETUP_TMR";
	case GFX_CMD_ID_LOAD_IP_FW:
		return "LOAD_IP_FW";
	case GFX_CMD_ID_DESTROY_TMR:
		return "DESTROY_TMR";
	case GFX_CMD_ID_SAVE_RESTORE:
		return "SAVE_RESTORE_IP_FW";
	case GFX_CMD_ID_SETUP_VMR:
		return "SETUP_VMR";
	case GFX_CMD_ID_DESTROY_VMR:
		return "DESTROY_VMR";
	case GFX_CMD_ID_PROG_REG:
		return "PROG_REG";
	case GFX_CMD_ID_GET_FW_ATTESTATION:
		return "GET_FW_ATTESTATION";
	case GFX_CMD_ID_LOAD_TOC:
		return "ID_LOAD_TOC";
	case GFX_CMD_ID_AUTOLOAD_RLC:
		return "AUTOLOAD_RLC";
	case GFX_CMD_ID_BOOT_CFG:
		return "BOOT_CFG";
	default:
		return "UNKNOWN CMD";
	}
}

429 430 431
static int
psp_cmd_submit_buf(struct psp_context *psp,
		   struct amdgpu_firmware_info *ucode,
432
		   struct psp_gfx_cmd_resp *cmd, uint64_t fence_mc_addr)
433 434
{
	int ret;
435
	int index, idx;
436
	int timeout = 20000;
437
	bool ras_intr = false;
438
	bool skip_unsupport = false;
439

440
	if (psp->adev->no_hw_access)
441 442
		return 0;

443
	if (!drm_dev_enter(adev_to_drm(psp->adev), &idx))
444 445
		return 0;

446
	memset(psp->cmd_buf_mem, 0, PSP_CMD_BUFFER_SIZE);
447

448
	memcpy(psp->cmd_buf_mem, cmd, sizeof(struct psp_gfx_cmd_resp));
449

450
	index = atomic_inc_return(&psp->fence_value);
451
	ret = psp_ring_cmd_submit(psp, psp->cmd_buf_mc_addr, fence_mc_addr, index);
452 453
	if (ret) {
		atomic_dec(&psp->fence_value);
454
		goto exit;
455
	}
456

457
	amdgpu_device_invalidate_hdp(psp->adev, NULL);
458 459
	while (*((unsigned int *)psp->fence_buf) != index) {
		if (--timeout == 0)
460
			break;
461 462 463 464 465
		/*
		 * Shouldn't wait for timeout when err_event_athub occurs,
		 * because gpu reset thread triggered and lock resource should
		 * be released for psp resume sequence.
		 */
466 467
		ras_intr = amdgpu_ras_intr_triggered();
		if (ras_intr)
468
			break;
469
		usleep_range(10, 100);
470
		amdgpu_device_invalidate_hdp(psp->adev, NULL);
471
	}
472

473 474 475
	/* We allow TEE_ERROR_NOT_SUPPORTED for VMR command and PSP_ERR_UNKNOWN_COMMAND in SRIOV */
	skip_unsupport = (psp->cmd_buf_mem->resp.status == TEE_ERROR_NOT_SUPPORTED ||
		psp->cmd_buf_mem->resp.status == PSP_ERR_UNKNOWN_COMMAND) && amdgpu_sriov_vf(psp->adev);
476

477 478
	memcpy((void*)&cmd->resp, (void*)&psp->cmd_buf_mem->resp, sizeof(struct psp_gfx_resp));

479 480 481 482 483 484 485
	/* In some cases, psp response status is not 0 even there is no
	 * problem while the command is submitted. Some version of PSP FW
	 * doesn't write 0 to that field.
	 * So here we would like to only print a warning instead of an error
	 * during psp initialization to avoid breaking hw_init and it doesn't
	 * return -EINVAL.
	 */
486
	if (!skip_unsupport && (psp->cmd_buf_mem->resp.status || !timeout) && !ras_intr) {
487
		if (ucode)
488 489 490 491
			DRM_WARN("failed to load ucode %s(0x%X) ",
				  amdgpu_ucode_name(ucode->ucode_id), ucode->ucode_id);
		DRM_WARN("psp gfx command %s(0x%X) failed and response status is (0x%X)\n",
			 psp_gfx_cmd_name(psp->cmd_buf_mem->cmd_id), psp->cmd_buf_mem->cmd_id,
492
			 psp->cmd_buf_mem->resp.status);
493
		if (!timeout) {
494 495
			ret = -EINVAL;
			goto exit;
496
		}
497 498
	}

J
James Zhu 已提交
499 500 501 502 503
	if (ucode) {
		ucode->tmr_mc_addr_lo = psp->cmd_buf_mem->resp.fw_addr_lo;
		ucode->tmr_mc_addr_hi = psp->cmd_buf_mem->resp.fw_addr_hi;
	}

504 505
exit:
	drm_dev_exit(idx);
506 507 508
	return ret;
}

509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524
static struct psp_gfx_cmd_resp *acquire_psp_cmd_buf(struct psp_context *psp)
{
	struct psp_gfx_cmd_resp *cmd = psp->cmd;

	mutex_lock(&psp->mutex);

	memset(cmd, 0, sizeof(struct psp_gfx_cmd_resp));

	return cmd;
}

void release_psp_cmd_buf(struct psp_context *psp)
{
	mutex_unlock(&psp->mutex);
}

525 526
static void psp_prep_tmr_cmd_buf(struct psp_context *psp,
				 struct psp_gfx_cmd_resp *cmd,
527
				 uint64_t tmr_mc, struct amdgpu_bo *tmr_bo)
528
{
529 530 531 532
	struct amdgpu_device *adev = psp->adev;
	uint32_t size = amdgpu_bo_size(tmr_bo);
	uint64_t tmr_pa = amdgpu_gmc_vram_pa(adev, tmr_bo);

533
	if (amdgpu_sriov_vf(psp->adev))
534 535 536
		cmd->cmd_id = GFX_CMD_ID_SETUP_VMR;
	else
		cmd->cmd_id = GFX_CMD_ID_SETUP_TMR;
537 538
	cmd->cmd.cmd_setup_tmr.buf_phy_addr_lo = lower_32_bits(tmr_mc);
	cmd->cmd.cmd_setup_tmr.buf_phy_addr_hi = upper_32_bits(tmr_mc);
539
	cmd->cmd.cmd_setup_tmr.buf_size = size;
540 541 542
	cmd->cmd.cmd_setup_tmr.bitfield.virt_phy_addr = 1;
	cmd->cmd.cmd_setup_tmr.system_phy_addr_lo = lower_32_bits(tmr_pa);
	cmd->cmd.cmd_setup_tmr.system_phy_addr_hi = upper_32_bits(tmr_pa);
543 544
}

545 546 547 548 549 550 551 552 553 554 555 556 557 558
static void psp_prep_load_toc_cmd_buf(struct psp_gfx_cmd_resp *cmd,
				      uint64_t pri_buf_mc, uint32_t size)
{
	cmd->cmd_id = GFX_CMD_ID_LOAD_TOC;
	cmd->cmd.cmd_load_toc.toc_phy_addr_lo = lower_32_bits(pri_buf_mc);
	cmd->cmd.cmd_load_toc.toc_phy_addr_hi = upper_32_bits(pri_buf_mc);
	cmd->cmd.cmd_load_toc.toc_size = size;
}

/* Issue LOAD TOC cmd to PSP to part toc and calculate tmr size needed */
static int psp_load_toc(struct psp_context *psp,
			uint32_t *tmr_size)
{
	int ret;
559
	struct psp_gfx_cmd_resp *cmd = acquire_psp_cmd_buf(psp);
560 561

	/* Copy toc to psp firmware private buffer */
562
	psp_copy_fw(psp, psp->toc.start_addr, psp->toc.size_bytes);
563

564
	psp_prep_load_toc_cmd_buf(cmd, psp->fw_pri_mc_addr, psp->toc.size_bytes);
565 566 567 568 569

	ret = psp_cmd_submit_buf(psp, NULL, cmd,
				 psp->fence_buf_mc_addr);
	if (!ret)
		*tmr_size = psp->cmd_buf_mem->resp.tmr_size;
570

571 572
	release_psp_cmd_buf(psp);

573 574 575
	return ret;
}

576 577 578 579
/* Set up Trusted Memory Region */
static int psp_tmr_init(struct psp_context *psp)
{
	int ret;
580
	int tmr_size;
581 582
	void *tmr_buf;
	void **pptr;
583 584

	/*
585 586
	 * According to HW engineer, they prefer the TMR address be "naturally
	 * aligned" , e.g. the start address be an integer divide of TMR size.
587 588 589 590
	 *
	 * Note: this memory need be reserved till the driver
	 * uninitializes.
	 */
591
	tmr_size = PSP_TMR_SIZE(psp->adev);
592 593 594

	/* For ASICs support RLC autoload, psp will parse the toc
	 * and calculate the total size of TMR needed */
595
	if (!amdgpu_sriov_vf(psp->adev) &&
596 597
	    psp->toc.start_addr &&
	    psp->toc.size_bytes &&
598 599 600 601 602 603 604 605
	    psp->fw_pri_buf) {
		ret = psp_load_toc(psp, &tmr_size);
		if (ret) {
			DRM_ERROR("Failed to load toc\n");
			return ret;
		}
	}

606
	pptr = amdgpu_sriov_vf(psp->adev) ? &tmr_buf : NULL;
607
	ret = amdgpu_bo_create_kernel(psp->adev, tmr_size, PSP_TMR_SIZE(psp->adev),
608
				      AMDGPU_GEM_DOMAIN_VRAM,
609
				      &psp->tmr_bo, &psp->tmr_mc_addr, pptr);
610 611 612 613

	return ret;
}

614 615
static bool psp_skip_tmr(struct psp_context *psp)
{
616
	switch (psp->adev->ip_versions[MP0_HWIP][0]) {
617 618 619
	case IP_VERSION(11, 0, 9):
	case IP_VERSION(11, 0, 7):
	case IP_VERSION(13, 0, 2):
620 621 622 623 624 625
		return true;
	default:
		return false;
	}
}

626 627 628
static int psp_tmr_load(struct psp_context *psp)
{
	int ret;
629
	struct psp_gfx_cmd_resp *cmd;
630

631 632 633 634 635 636
	/* For Navi12 and CHIP_SIENNA_CICHLID SRIOV, do not set up TMR.
	 * Already set up by host driver.
	 */
	if (amdgpu_sriov_vf(psp->adev) && psp_skip_tmr(psp))
		return 0;

637 638
	cmd = acquire_psp_cmd_buf(psp);

639
	psp_prep_tmr_cmd_buf(psp, cmd, psp->tmr_mc_addr, psp->tmr_bo);
640 641
	DRM_INFO("reserve 0x%lx from 0x%llx for PSP TMR\n",
		 amdgpu_bo_size(psp->tmr_bo), psp->tmr_mc_addr);
642 643

	ret = psp_cmd_submit_buf(psp, NULL, cmd,
644
				 psp->fence_buf_mc_addr);
645

646 647
	release_psp_cmd_buf(psp);

648 649 650
	return ret;
}

651
static void psp_prep_tmr_unload_cmd_buf(struct psp_context *psp,
652
				        struct psp_gfx_cmd_resp *cmd)
653 654 655 656 657 658 659 660 661 662
{
	if (amdgpu_sriov_vf(psp->adev))
		cmd->cmd_id = GFX_CMD_ID_DESTROY_VMR;
	else
		cmd->cmd_id = GFX_CMD_ID_DESTROY_TMR;
}

static int psp_tmr_unload(struct psp_context *psp)
{
	int ret;
663
	struct psp_gfx_cmd_resp *cmd = acquire_psp_cmd_buf(psp);
664 665 666 667 668 669 670

	psp_prep_tmr_unload_cmd_buf(psp, cmd);
	DRM_INFO("free PSP TMR buffer\n");

	ret = psp_cmd_submit_buf(psp, NULL, cmd,
				 psp->fence_buf_mc_addr);

671 672
	release_psp_cmd_buf(psp);

673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692
	return ret;
}

static int psp_tmr_terminate(struct psp_context *psp)
{
	int ret;
	void *tmr_buf;
	void **pptr;

	ret = psp_tmr_unload(psp);
	if (ret)
		return ret;

	/* free TMR memory buffer */
	pptr = amdgpu_sriov_vf(psp->adev) ? &tmr_buf : NULL;
	amdgpu_bo_free_kernel(&psp->tmr_bo, &psp->tmr_mc_addr, pptr);

	return 0;
}

693 694 695 696
int psp_get_fw_attestation_records_addr(struct psp_context *psp,
					uint64_t *output_ptr)
{
	int ret;
697
	struct psp_gfx_cmd_resp *cmd;
698 699 700 701 702 703 704

	if (!output_ptr)
		return -EINVAL;

	if (amdgpu_sriov_vf(psp->adev))
		return 0;

705 706
	cmd = acquire_psp_cmd_buf(psp);

707 708 709 710 711 712 713 714 715 716
	cmd->cmd_id = GFX_CMD_ID_GET_FW_ATTESTATION;

	ret = psp_cmd_submit_buf(psp, NULL, cmd,
				 psp->fence_buf_mc_addr);

	if (!ret) {
		*output_ptr = ((uint64_t)cmd->resp.uresp.fwar_db_info.fwar_db_addr_lo) +
			      ((uint64_t)cmd->resp.uresp.fwar_db_info.fwar_db_addr_hi << 32);
	}

717 718
	release_psp_cmd_buf(psp);

719 720 721
	return ret;
}

722 723 724
static int psp_boot_config_get(struct amdgpu_device *adev, uint32_t *boot_cfg)
{
	struct psp_context *psp = &adev->psp;
725
	struct psp_gfx_cmd_resp *cmd;
726 727 728 729 730
	int ret;

	if (amdgpu_sriov_vf(adev))
		return 0;

731
	cmd = acquire_psp_cmd_buf(psp);
732 733 734 735 736 737 738 739 740 741

	cmd->cmd_id = GFX_CMD_ID_BOOT_CFG;
	cmd->cmd.boot_cfg.sub_cmd = BOOTCFG_CMD_GET;

	ret = psp_cmd_submit_buf(psp, NULL, cmd, psp->fence_buf_mc_addr);
	if (!ret) {
		*boot_cfg =
			(cmd->resp.uresp.boot_cfg.boot_cfg & BOOT_CONFIG_GECC) ? 1 : 0;
	}

742 743
	release_psp_cmd_buf(psp);

744 745 746
	return ret;
}

747
static int psp_boot_config_set(struct amdgpu_device *adev, uint32_t boot_cfg)
748
{
749
	int ret;
750
	struct psp_context *psp = &adev->psp;
751
	struct psp_gfx_cmd_resp *cmd;
752

753
	if (amdgpu_sriov_vf(adev))
754 755
		return 0;

756
	cmd = acquire_psp_cmd_buf(psp);
757 758 759

	cmd->cmd_id = GFX_CMD_ID_BOOT_CFG;
	cmd->cmd.boot_cfg.sub_cmd = BOOTCFG_CMD_SET;
760 761
	cmd->cmd.boot_cfg.boot_config = boot_cfg;
	cmd->cmd.boot_cfg.boot_config_valid = boot_cfg;
762

763 764 765 766 767
	ret = psp_cmd_submit_buf(psp, NULL, cmd, psp->fence_buf_mc_addr);

	release_psp_cmd_buf(psp);

	return ret;
768 769
}

770 771
static int psp_rl_load(struct amdgpu_device *adev)
{
772
	int ret;
773
	struct psp_context *psp = &adev->psp;
774
	struct psp_gfx_cmd_resp *cmd;
775

776
	if (!is_psp_fw_valid(psp->rl))
777 778
		return 0;

779 780
	cmd = acquire_psp_cmd_buf(psp);

781
	memset(psp->fw_pri_buf, 0, PSP_1_MEG);
782
	memcpy(psp->fw_pri_buf, psp->rl.start_addr, psp->rl.size_bytes);
783 784 785 786

	cmd->cmd_id = GFX_CMD_ID_LOAD_IP_FW;
	cmd->cmd.cmd_load_ip_fw.fw_phy_addr_lo = lower_32_bits(psp->fw_pri_mc_addr);
	cmd->cmd.cmd_load_ip_fw.fw_phy_addr_hi = upper_32_bits(psp->fw_pri_mc_addr);
787
	cmd->cmd.cmd_load_ip_fw.fw_size = psp->rl.size_bytes;
788 789
	cmd->cmd.cmd_load_ip_fw.fw_type = GFX_FW_TYPE_REG_LIST;

790 791 792 793 794
	ret = psp_cmd_submit_buf(psp, NULL, cmd, psp->fence_buf_mc_addr);

	release_psp_cmd_buf(psp);

	return ret;
795 796
}

797
static int psp_asd_load(struct psp_context *psp)
798
{
799
	return psp_ta_load(psp, &psp->asd_context);
H
Huang Rui 已提交
800 801
}

802
static int psp_asd_initialize(struct psp_context *psp)
803 804 805
{
	int ret;

806 807 808 809
	/* If PSP version doesn't match ASD version, asd loading will be failed.
	 * add workaround to bypass it for sriov now.
	 * TODO: add version check to make it common
	 */
C
Candice Li 已提交
810
	if (amdgpu_sriov_vf(psp->adev) || !psp->asd_context.bin_desc.size_bytes)
811 812
		return 0;

813 814 815
	psp->asd_context.mem_context.shared_mc_addr  = 0;
	psp->asd_context.mem_context.shared_mem_size = PSP_ASD_SHARED_MEM_SIZE;
	psp->asd_context.ta_load_type                = GFX_CMD_ID_LOAD_ASD;
816

817 818 819
	ret = psp_asd_load(psp);
	if (!ret)
		psp->asd_context.initialized = true;
820

821 822 823
	return ret;
}

824 825
static void psp_prep_ta_unload_cmd_buf(struct psp_gfx_cmd_resp *cmd,
				       uint32_t session_id)
826 827
{
	cmd->cmd_id = GFX_CMD_ID_UNLOAD_TA;
828
	cmd->cmd.cmd_unload_ta.session_id = session_id;
829 830
}

831
static int psp_ta_unload(struct psp_context *psp, struct ta_context *context)
832 833 834 835
{
	int ret;
	struct psp_gfx_cmd_resp *cmd = acquire_psp_cmd_buf(psp);

836
	psp_prep_ta_unload_cmd_buf(cmd, context->session_id);
837 838 839 840 841 842 843 844

	ret = psp_cmd_submit_buf(psp, NULL, cmd, psp->fence_buf_mc_addr);

	release_psp_cmd_buf(psp);

	return ret;
}

845
static int psp_asd_unload(struct psp_context *psp)
846
{
847
	return psp_ta_unload(psp, &psp->asd_context);
848 849 850
}

static int psp_asd_terminate(struct psp_context *psp)
851 852 853 854 855 856
{
	int ret;

	if (amdgpu_sriov_vf(psp->adev))
		return 0;

857
	if (!psp->asd_context.initialized)
858 859
		return 0;

860
	ret = psp_asd_unload(psp);
861 862

	if (!ret)
863
		psp->asd_context.initialized = false;
864 865 866 867

	return ret;
}

868 869 870 871 872 873 874 875 876 877 878
static void psp_prep_reg_prog_cmd_buf(struct psp_gfx_cmd_resp *cmd,
		uint32_t id, uint32_t value)
{
	cmd->cmd_id = GFX_CMD_ID_PROG_REG;
	cmd->cmd.cmd_setup_reg_prog.reg_value = value;
	cmd->cmd.cmd_setup_reg_prog.reg_id = id;
}

int psp_reg_program(struct psp_context *psp, enum psp_reg_prog_id reg,
		uint32_t value)
{
879
	struct psp_gfx_cmd_resp *cmd;
880 881 882 883 884
	int ret = 0;

	if (reg >= PSP_REG_LAST)
		return -EINVAL;

885 886
	cmd = acquire_psp_cmd_buf(psp);

887 888
	psp_prep_reg_prog_cmd_buf(cmd, reg, value);
	ret = psp_cmd_submit_buf(psp, NULL, cmd, psp->fence_buf_mc_addr);
889 890
	if (ret)
		DRM_ERROR("PSP failed to program reg id %d", reg);
891

892 893
	release_psp_cmd_buf(psp);

894 895 896
	return ret;
}

897 898
static void psp_prep_ta_load_cmd_buf(struct psp_gfx_cmd_resp *cmd,
				     uint64_t ta_bin_mc,
C
Candice Li 已提交
899
				     struct ta_context *context)
900
{
901
	cmd->cmd_id				= context->ta_load_type;
902
	cmd->cmd.cmd_load_ta.app_phy_addr_lo 	= lower_32_bits(ta_bin_mc);
903
	cmd->cmd.cmd_load_ta.app_phy_addr_hi	= upper_32_bits(ta_bin_mc);
C
Candice Li 已提交
904
	cmd->cmd.cmd_load_ta.app_len		= context->bin_desc.size_bytes;
905

C
Candice Li 已提交
906 907 908 909 910
	cmd->cmd.cmd_load_ta.cmd_buf_phy_addr_lo =
		lower_32_bits(context->mem_context.shared_mc_addr);
	cmd->cmd.cmd_load_ta.cmd_buf_phy_addr_hi =
		upper_32_bits(context->mem_context.shared_mc_addr);
	cmd->cmd.cmd_load_ta.cmd_buf_len = context->mem_context.shared_mem_size;
911 912
}

913
static int psp_ta_init_shared_buf(struct psp_context *psp,
914
				  struct ta_mem_context *mem_ctx)
915 916 917 918
{
	int ret;

	/*
919 920 921
	* Allocate 16k memory aligned to 4k from Frame Buffer (local
	* physical) for ta to host memory
	*/
922 923
	ret = amdgpu_bo_create_kernel(psp->adev, mem_ctx->shared_mem_size,
				      PAGE_SIZE, AMDGPU_GEM_DOMAIN_VRAM,
924 925 926
				      &mem_ctx->shared_bo,
				      &mem_ctx->shared_mc_addr,
				      &mem_ctx->shared_buf);
927 928 929 930

	return ret;
}

931 932 933 934 935 936 937 938
static void psp_ta_free_shared_buf(struct ta_mem_context *mem_ctx)
{
	amdgpu_bo_free_kernel(&mem_ctx->shared_bo, &mem_ctx->shared_mc_addr,
			      &mem_ctx->shared_buf);
}

static int psp_xgmi_init_shared_buf(struct psp_context *psp)
{
939
	return psp_ta_init_shared_buf(psp, &psp->xgmi_context.context.mem_context);
940 941
}

942 943 944 945
static void psp_prep_ta_invoke_cmd_buf(struct psp_gfx_cmd_resp *cmd,
				       uint32_t ta_cmd_id,
				       uint32_t session_id)
{
946 947 948
	cmd->cmd_id				= GFX_CMD_ID_INVOKE_CMD;
	cmd->cmd.cmd_invoke_cmd.session_id	= session_id;
	cmd->cmd.cmd_invoke_cmd.ta_cmd_id	= ta_cmd_id;
949 950
}

951
static int psp_ta_invoke(struct psp_context *psp,
952
		  uint32_t ta_cmd_id,
953
		  struct ta_context *context)
954 955
{
	int ret;
956
	struct psp_gfx_cmd_resp *cmd = acquire_psp_cmd_buf(psp);
957

958
	psp_prep_ta_invoke_cmd_buf(cmd, ta_cmd_id, context->session_id);
959 960 961 962

	ret = psp_cmd_submit_buf(psp, NULL, cmd,
				 psp->fence_buf_mc_addr);

963 964
	release_psp_cmd_buf(psp);

965 966 967
	return ret;
}

968
static int psp_ta_load(struct psp_context *psp, struct ta_context *context)
969 970
{
	int ret;
971
	struct psp_gfx_cmd_resp *cmd;
972

973 974
	cmd = acquire_psp_cmd_buf(psp);

C
Candice Li 已提交
975 976
	psp_copy_fw(psp, context->bin_desc.start_addr,
		    context->bin_desc.size_bytes);
977

978
	psp_prep_ta_load_cmd_buf(cmd, psp->fw_pri_mc_addr, context);
979 980 981 982 983

	ret = psp_cmd_submit_buf(psp, NULL, cmd,
				 psp->fence_buf_mc_addr);

	if (!ret) {
984
		context->session_id = cmd->resp.session_id;
985 986
	}

987 988
	release_psp_cmd_buf(psp);

989 990 991
	return ret;
}

992 993
static int psp_xgmi_load(struct psp_context *psp)
{
C
Candice Li 已提交
994
	return psp_ta_load(psp, &psp->xgmi_context.context);
995 996
}

997 998
static int psp_xgmi_unload(struct psp_context *psp)
{
999
	return psp_ta_unload(psp, &psp->xgmi_context.context);
1000 1001
}

1002 1003
int psp_xgmi_invoke(struct psp_context *psp, uint32_t ta_cmd_id)
{
1004
	return psp_ta_invoke(psp, ta_cmd_id, &psp->xgmi_context.context);
1005 1006
}

1007
int psp_xgmi_terminate(struct psp_context *psp)
1008 1009
{
	int ret;
1010 1011 1012
	struct amdgpu_device *adev = psp->adev;

	/* XGMI TA unload currently is not supported on Arcturus/Aldebaran A+A */
1013 1014
	if (adev->ip_versions[MP0_HWIP][0] == IP_VERSION(11, 0, 4) ||
	    (adev->ip_versions[MP0_HWIP][0] == IP_VERSION(13, 0, 2) &&
1015
	     adev->gmc.xgmi.connected_to_cpu))
1016
		return 0;
1017

1018
	if (!psp->xgmi_context.context.initialized)
1019 1020 1021 1022 1023 1024
		return 0;

	ret = psp_xgmi_unload(psp);
	if (ret)
		return ret;

1025
	psp->xgmi_context.context.initialized = false;
1026 1027

	/* free xgmi shared memory */
1028
	psp_ta_free_shared_buf(&psp->xgmi_context.context.mem_context);
1029 1030 1031 1032

	return 0;
}

1033
int psp_xgmi_initialize(struct psp_context *psp, bool set_extended_data, bool load_ta)
1034 1035 1036 1037
{
	struct ta_xgmi_shared_memory *xgmi_cmd;
	int ret;

1038
	if (!psp->ta_fw ||
C
Candice Li 已提交
1039 1040
	    !psp->xgmi_context.context.bin_desc.size_bytes ||
	    !psp->xgmi_context.context.bin_desc.start_addr)
1041 1042
		return -ENOENT;

1043 1044 1045
	if (!load_ta)
		goto invoke;

1046
	psp->xgmi_context.context.mem_context.shared_mem_size = PSP_XGMI_SHARED_MEM_SIZE;
1047
	psp->xgmi_context.context.ta_load_type = GFX_CMD_ID_LOAD_TA;
1048

1049
	if (!psp->xgmi_context.context.initialized) {
1050 1051 1052 1053 1054 1055 1056
		ret = psp_xgmi_init_shared_buf(psp);
		if (ret)
			return ret;
	}

	/* Load XGMI TA */
	ret = psp_xgmi_load(psp);
1057 1058 1059
	if (!ret)
		psp->xgmi_context.context.initialized = true;
	else
1060 1061
		return ret;

1062
invoke:
1063
	/* Initialize XGMI session */
1064
	xgmi_cmd = (struct ta_xgmi_shared_memory *)(psp->xgmi_context.context.mem_context.shared_buf);
1065
	memset(xgmi_cmd, 0, sizeof(struct ta_xgmi_shared_memory));
1066
	xgmi_cmd->flag_extend_link_record = set_extended_data;
1067 1068 1069 1070 1071 1072 1073
	xgmi_cmd->cmd_id = TA_COMMAND_XGMI__INITIALIZE;

	ret = psp_xgmi_invoke(psp, xgmi_cmd->cmd_id);

	return ret;
}

1074 1075 1076 1077 1078
int psp_xgmi_get_hive_id(struct psp_context *psp, uint64_t *hive_id)
{
	struct ta_xgmi_shared_memory *xgmi_cmd;
	int ret;

1079
	xgmi_cmd = (struct ta_xgmi_shared_memory *)psp->xgmi_context.context.mem_context.shared_buf;
1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098
	memset(xgmi_cmd, 0, sizeof(struct ta_xgmi_shared_memory));

	xgmi_cmd->cmd_id = TA_COMMAND_XGMI__GET_HIVE_ID;

	/* Invoke xgmi ta to get hive id */
	ret = psp_xgmi_invoke(psp, xgmi_cmd->cmd_id);
	if (ret)
		return ret;

	*hive_id = xgmi_cmd->xgmi_out_message.get_hive_id.hive_id;

	return 0;
}

int psp_xgmi_get_node_id(struct psp_context *psp, uint64_t *node_id)
{
	struct ta_xgmi_shared_memory *xgmi_cmd;
	int ret;

1099
	xgmi_cmd = (struct ta_xgmi_shared_memory *)psp->xgmi_context.context.mem_context.shared_buf;
1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113
	memset(xgmi_cmd, 0, sizeof(struct ta_xgmi_shared_memory));

	xgmi_cmd->cmd_id = TA_COMMAND_XGMI__GET_NODE_ID;

	/* Invoke xgmi ta to get the node id */
	ret = psp_xgmi_invoke(psp, xgmi_cmd->cmd_id);
	if (ret)
		return ret;

	*node_id = xgmi_cmd->xgmi_out_message.get_node_id.node_id;

	return 0;
}

1114 1115
static bool psp_xgmi_peer_link_info_supported(struct psp_context *psp)
{
1116
	return psp->adev->ip_versions[MP0_HWIP][0] == IP_VERSION(13, 0, 2) &&
1117
		psp->xgmi_context.context.bin_desc.feature_version >= 0x2000000b;
1118 1119
}

1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165
/*
 * Chips that support extended topology information require the driver to
 * reflect topology information in the opposite direction.  This is
 * because the TA has already exceeded its link record limit and if the
 * TA holds bi-directional information, the driver would have to do
 * multiple fetches instead of just two.
 */
static void psp_xgmi_reflect_topology_info(struct psp_context *psp,
					struct psp_xgmi_node_info node_info)
{
	struct amdgpu_device *mirror_adev;
	struct amdgpu_hive_info *hive;
	uint64_t src_node_id = psp->adev->gmc.xgmi.node_id;
	uint64_t dst_node_id = node_info.node_id;
	uint8_t dst_num_hops = node_info.num_hops;
	uint8_t dst_num_links = node_info.num_links;

	hive = amdgpu_get_xgmi_hive(psp->adev);
	list_for_each_entry(mirror_adev, &hive->device_list, gmc.xgmi.head) {
		struct psp_xgmi_topology_info *mirror_top_info;
		int j;

		if (mirror_adev->gmc.xgmi.node_id != dst_node_id)
			continue;

		mirror_top_info = &mirror_adev->psp.xgmi_context.top_info;
		for (j = 0; j < mirror_top_info->num_nodes; j++) {
			if (mirror_top_info->nodes[j].node_id != src_node_id)
				continue;

			mirror_top_info->nodes[j].num_hops = dst_num_hops;
			/*
			 * prevent 0 num_links value re-reflection since reflection
			 * criteria is based on num_hops (direct or indirect).
			 *
			 */
			if (dst_num_links)
				mirror_top_info->nodes[j].num_links = dst_num_links;

			break;
		}

		break;
	}
}

1166 1167
int psp_xgmi_get_topology_info(struct psp_context *psp,
			       int number_devices,
1168 1169
			       struct psp_xgmi_topology_info *topology,
			       bool get_extended_data)
1170 1171 1172 1173 1174 1175 1176 1177 1178 1179
{
	struct ta_xgmi_shared_memory *xgmi_cmd;
	struct ta_xgmi_cmd_get_topology_info_input *topology_info_input;
	struct ta_xgmi_cmd_get_topology_info_output *topology_info_output;
	int i;
	int ret;

	if (!topology || topology->num_nodes > TA_XGMI__MAX_CONNECTED_NODES)
		return -EINVAL;

1180
	xgmi_cmd = (struct ta_xgmi_shared_memory *)psp->xgmi_context.context.mem_context.shared_buf;
1181
	memset(xgmi_cmd, 0, sizeof(struct ta_xgmi_shared_memory));
1182
	xgmi_cmd->flag_extend_link_record = get_extended_data;
1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204

	/* Fill in the shared memory with topology information as input */
	topology_info_input = &xgmi_cmd->xgmi_in_message.get_topology_info;
	xgmi_cmd->cmd_id = TA_COMMAND_XGMI__GET_GET_TOPOLOGY_INFO;
	topology_info_input->num_nodes = number_devices;

	for (i = 0; i < topology_info_input->num_nodes; i++) {
		topology_info_input->nodes[i].node_id = topology->nodes[i].node_id;
		topology_info_input->nodes[i].num_hops = topology->nodes[i].num_hops;
		topology_info_input->nodes[i].is_sharing_enabled = topology->nodes[i].is_sharing_enabled;
		topology_info_input->nodes[i].sdma_engine = topology->nodes[i].sdma_engine;
	}

	/* Invoke xgmi ta to get the topology information */
	ret = psp_xgmi_invoke(psp, TA_COMMAND_XGMI__GET_GET_TOPOLOGY_INFO);
	if (ret)
		return ret;

	/* Read the output topology information from the shared memory */
	topology_info_output = &xgmi_cmd->xgmi_out_message.get_topology_info;
	topology->num_nodes = xgmi_cmd->xgmi_out_message.get_topology_info.num_nodes;
	for (i = 0; i < topology->num_nodes; i++) {
1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216 1217
		/* extended data will either be 0 or equal to non-extended data */
		if (topology_info_output->nodes[i].num_hops)
			topology->nodes[i].num_hops = topology_info_output->nodes[i].num_hops;

		/* non-extended data gets everything here so no need to update */
		if (!get_extended_data) {
			topology->nodes[i].node_id = topology_info_output->nodes[i].node_id;
			topology->nodes[i].is_sharing_enabled =
					topology_info_output->nodes[i].is_sharing_enabled;
			topology->nodes[i].sdma_engine =
					topology_info_output->nodes[i].sdma_engine;
		}

1218 1219
	}

1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231
	/* Invoke xgmi ta again to get the link information */
	if (psp_xgmi_peer_link_info_supported(psp)) {
		struct ta_xgmi_cmd_get_peer_link_info_output *link_info_output;

		xgmi_cmd->cmd_id = TA_COMMAND_XGMI__GET_PEER_LINKS;

		ret = psp_xgmi_invoke(psp, TA_COMMAND_XGMI__GET_PEER_LINKS);

		if (ret)
			return ret;

		link_info_output = &xgmi_cmd->xgmi_out_message.get_link_info;
1232 1233 1234 1235 1236
		for (i = 0; i < topology->num_nodes; i++) {
			/* accumulate num_links on extended data */
			topology->nodes[i].num_links = get_extended_data ?
					topology->nodes[i].num_links +
							link_info_output->nodes[i].num_links :
1237
					link_info_output->nodes[i].num_links;
1238 1239 1240 1241 1242 1243

			/* reflect the topology information for bi-directionality */
			if (psp->xgmi_context.supports_extended_data &&
					get_extended_data && topology->nodes[i].num_hops)
				psp_xgmi_reflect_topology_info(psp, topology->nodes[i]);
		}
1244 1245
	}

1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259
	return 0;
}

int psp_xgmi_set_topology_info(struct psp_context *psp,
			       int number_devices,
			       struct psp_xgmi_topology_info *topology)
{
	struct ta_xgmi_shared_memory *xgmi_cmd;
	struct ta_xgmi_cmd_get_topology_info_input *topology_info_input;
	int i;

	if (!topology || topology->num_nodes > TA_XGMI__MAX_CONNECTED_NODES)
		return -EINVAL;

1260
	xgmi_cmd = (struct ta_xgmi_shared_memory *)psp->xgmi_context.context.mem_context.shared_buf;
1261 1262 1263 1264 1265 1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277
	memset(xgmi_cmd, 0, sizeof(struct ta_xgmi_shared_memory));

	topology_info_input = &xgmi_cmd->xgmi_in_message.get_topology_info;
	xgmi_cmd->cmd_id = TA_COMMAND_XGMI__SET_TOPOLOGY_INFO;
	topology_info_input->num_nodes = number_devices;

	for (i = 0; i < topology_info_input->num_nodes; i++) {
		topology_info_input->nodes[i].node_id = topology->nodes[i].node_id;
		topology_info_input->nodes[i].num_hops = topology->nodes[i].num_hops;
		topology_info_input->nodes[i].is_sharing_enabled = 1;
		topology_info_input->nodes[i].sdma_engine = topology->nodes[i].sdma_engine;
	}

	/* Invoke xgmi ta to set topology information */
	return psp_xgmi_invoke(psp, TA_COMMAND_XGMI__SET_TOPOLOGY_INFO);
}

1278 1279 1280
// ras begin
static int psp_ras_init_shared_buf(struct psp_context *psp)
{
1281
	return psp_ta_init_shared_buf(psp, &psp->ras_context.context.mem_context);
1282 1283 1284 1285
}

static int psp_ras_load(struct psp_context *psp)
{
C
Candice Li 已提交
1286
	return psp_ta_load(psp, &psp->ras_context.context);
1287 1288 1289 1290
}

static int psp_ras_unload(struct psp_context *psp)
{
1291
	return psp_ta_unload(psp, &psp->ras_context.context);
1292 1293 1294 1295
}

int psp_ras_invoke(struct psp_context *psp, uint32_t ta_cmd_id)
{
1296 1297 1298
	struct ta_ras_shared_memory *ras_cmd;
	int ret;

1299
	ras_cmd = (struct ta_ras_shared_memory *)psp->ras_context.context.mem_context.shared_buf;
1300

1301 1302 1303 1304 1305 1306
	/*
	 * TODO: bypass the loading in sriov for now
	 */
	if (amdgpu_sriov_vf(psp->adev))
		return 0;

1307
	ret = psp_ta_invoke(psp, ta_cmd_id, &psp->ras_context.context);
1308

1309 1310 1311
	if (amdgpu_ras_intr_triggered())
		return ret;

1312 1313 1314 1315 1316 1317 1318 1319 1320 1321 1322 1323 1324 1325 1326
	if (ras_cmd->if_version > RAS_TA_HOST_IF_VER)
	{
		DRM_WARN("RAS: Unsupported Interface");
		return -EINVAL;
	}

	if (!ret) {
		if (ras_cmd->ras_out_message.flags.err_inject_switch_disable_flag) {
			dev_warn(psp->adev->dev, "ECC switch disabled\n");

			ras_cmd->ras_status = TA_RAS_STATUS__ERROR_RAS_NOT_AVAILABLE;
		}
		else if (ras_cmd->ras_out_message.flags.reg_access_failure_flag)
			dev_warn(psp->adev->dev,
				 "RAS internal register access blocked\n");
1327 1328 1329 1330 1331

		if (ras_cmd->ras_status == TA_RAS_STATUS__ERROR_UNSUPPORTED_IP)
		    dev_warn(psp->adev->dev, "RAS WARNING: cmd failed due to unsupported ip\n");
		else if (ras_cmd->ras_status)
		    dev_warn(psp->adev->dev, "RAS WARNING: ras status = 0x%X\n", ras_cmd->ras_status);
1332 1333 1334
	}

	return ret;
1335 1336 1337 1338 1339 1340 1341 1342
}

int psp_ras_enable_features(struct psp_context *psp,
		union ta_ras_cmd_input *info, bool enable)
{
	struct ta_ras_shared_memory *ras_cmd;
	int ret;

1343
	if (!psp->ras_context.context.initialized)
1344 1345
		return -EINVAL;

1346
	ras_cmd = (struct ta_ras_shared_memory *)psp->ras_context.context.mem_context.shared_buf;
1347 1348 1349 1350 1351 1352 1353 1354 1355 1356 1357 1358 1359
	memset(ras_cmd, 0, sizeof(struct ta_ras_shared_memory));

	if (enable)
		ras_cmd->cmd_id = TA_RAS_COMMAND__ENABLE_FEATURES;
	else
		ras_cmd->cmd_id = TA_RAS_COMMAND__DISABLE_FEATURES;

	ras_cmd->ras_in_message = *info;

	ret = psp_ras_invoke(psp, ras_cmd->cmd_id);
	if (ret)
		return -EINVAL;

1360
	return 0;
1361 1362 1363 1364 1365 1366
}

static int psp_ras_terminate(struct psp_context *psp)
{
	int ret;

1367 1368 1369 1370 1371 1372
	/*
	 * TODO: bypass the terminate in sriov for now
	 */
	if (amdgpu_sriov_vf(psp->adev))
		return 0;

1373
	if (!psp->ras_context.context.initialized)
1374 1375 1376 1377 1378 1379
		return 0;

	ret = psp_ras_unload(psp);
	if (ret)
		return ret;

1380
	psp->ras_context.context.initialized = false;
1381 1382

	/* free ras shared memory */
1383
	psp_ta_free_shared_buf(&psp->ras_context.context.mem_context);
1384 1385 1386 1387 1388 1389 1390

	return 0;
}

static int psp_ras_initialize(struct psp_context *psp)
{
	int ret;
1391 1392
	uint32_t boot_cfg = 0xFF;
	struct amdgpu_device *adev = psp->adev;
1393
	struct ta_ras_shared_memory *ras_cmd;
1394

1395 1396 1397
	/*
	 * TODO: bypass the initialize in sriov for now
	 */
1398
	if (amdgpu_sriov_vf(adev))
1399 1400
		return 0;

C
Candice Li 已提交
1401 1402
	if (!adev->psp.ras_context.context.bin_desc.size_bytes ||
	    !adev->psp.ras_context.context.bin_desc.start_addr) {
1403
		dev_info(adev->dev, "RAS: optional ras ta ucode is not available\n");
1404 1405 1406
		return 0;
	}

1407 1408 1409 1410 1411 1412 1413 1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447
	if (amdgpu_atomfirmware_dynamic_boot_config_supported(adev)) {
		/* query GECC enablement status from boot config
		 * boot_cfg: 1: GECC is enabled or 0: GECC is disabled
		 */
		ret = psp_boot_config_get(adev, &boot_cfg);
		if (ret)
			dev_warn(adev->dev, "PSP get boot config failed\n");

		if (!amdgpu_ras_is_supported(psp->adev, AMDGPU_RAS_BLOCK__UMC)) {
			if (!boot_cfg) {
				dev_info(adev->dev, "GECC is disabled\n");
			} else {
				/* disable GECC in next boot cycle if ras is
				 * disabled by module parameter amdgpu_ras_enable
				 * and/or amdgpu_ras_mask, or boot_config_get call
				 * is failed
				 */
				ret = psp_boot_config_set(adev, 0);
				if (ret)
					dev_warn(adev->dev, "PSP set boot config failed\n");
				else
					dev_warn(adev->dev, "GECC will be disabled in next boot cycle "
						 "if set amdgpu_ras_enable and/or amdgpu_ras_mask to 0x0\n");
			}
		} else {
			if (1 == boot_cfg) {
				dev_info(adev->dev, "GECC is enabled\n");
			} else {
				/* enable GECC in next boot cycle if it is disabled
				 * in boot config, or force enable GECC if failed to
				 * get boot configuration
				 */
				ret = psp_boot_config_set(adev, BOOT_CONFIG_GECC);
				if (ret)
					dev_warn(adev->dev, "PSP set boot config failed\n");
				else
					dev_warn(adev->dev, "GECC will be enabled in next boot cycle\n");
			}
		}
	}

1448
	psp->ras_context.context.mem_context.shared_mem_size = PSP_RAS_SHARED_MEM_SIZE;
1449
	psp->ras_context.context.ta_load_type = GFX_CMD_ID_LOAD_TA;
1450

1451
	if (!psp->ras_context.context.initialized) {
1452 1453 1454 1455 1456
		ret = psp_ras_init_shared_buf(psp);
		if (ret)
			return ret;
	}

1457 1458 1459
	ras_cmd = (struct ta_ras_shared_memory *)psp->ras_context.context.mem_context.shared_buf;
	memset(ras_cmd, 0, sizeof(struct ta_ras_shared_memory));

1460
	if (amdgpu_ras_is_poison_mode_supported(adev))
1461
		ras_cmd->ras_in_message.init_flags.poison_mode_en = 1;
1462
	if (!adev->gmc.xgmi.connected_to_cpu)
1463 1464
		ras_cmd->ras_in_message.init_flags.dgpu_mode = 1;

1465 1466
	ret = psp_ras_load(psp);

1467 1468 1469 1470 1471 1472 1473 1474 1475
	if (!ret && !ras_cmd->ras_status)
		psp->ras_context.context.initialized = true;
	else {
		if (ras_cmd->ras_status)
			dev_warn(psp->adev->dev, "RAS Init Status: 0x%X\n", ras_cmd->ras_status);
		amdgpu_ras_fini(psp->adev);
	}

	return ret;
1476
}
1477 1478 1479 1480 1481 1482 1483

int psp_ras_trigger_error(struct psp_context *psp,
			  struct ta_ras_trigger_error_input *info)
{
	struct ta_ras_shared_memory *ras_cmd;
	int ret;

1484
	if (!psp->ras_context.context.initialized)
1485 1486
		return -EINVAL;

1487
	ras_cmd = (struct ta_ras_shared_memory *)psp->ras_context.context.mem_context.shared_buf;
1488 1489 1490 1491 1492 1493 1494 1495 1496 1497 1498 1499 1500 1501
	memset(ras_cmd, 0, sizeof(struct ta_ras_shared_memory));

	ras_cmd->cmd_id = TA_RAS_COMMAND__TRIGGER_ERROR;
	ras_cmd->ras_in_message.trigger_error = *info;

	ret = psp_ras_invoke(psp, ras_cmd->cmd_id);
	if (ret)
		return -EINVAL;

	/* If err_event_athub occurs error inject was successful, however
	   return status from TA is no long reliable */
	if (amdgpu_ras_intr_triggered())
		return 0;

1502
	if (ras_cmd->ras_status)
1503 1504 1505
		return -EINVAL;

	return 0;
1506
}
1507 1508
// ras end

B
Bhawanpreet Lakha 已提交
1509 1510 1511
// HDCP start
static int psp_hdcp_init_shared_buf(struct psp_context *psp)
{
1512
	return psp_ta_init_shared_buf(psp, &psp->hdcp_context.context.mem_context);
B
Bhawanpreet Lakha 已提交
1513 1514 1515 1516
}

static int psp_hdcp_load(struct psp_context *psp)
{
C
Candice Li 已提交
1517
	return psp_ta_load(psp, &psp->hdcp_context.context);
B
Bhawanpreet Lakha 已提交
1518
}
C
Candice Li 已提交
1519

B
Bhawanpreet Lakha 已提交
1520 1521 1522 1523
static int psp_hdcp_initialize(struct psp_context *psp)
{
	int ret;

1524 1525 1526 1527 1528 1529
	/*
	 * TODO: bypass the initialize in sriov for now
	 */
	if (amdgpu_sriov_vf(psp->adev))
		return 0;

C
Candice Li 已提交
1530 1531
	if (!psp->hdcp_context.context.bin_desc.size_bytes ||
	    !psp->hdcp_context.context.bin_desc.start_addr) {
1532
		dev_info(psp->adev->dev, "HDCP: optional hdcp ta ucode is not available\n");
1533 1534 1535
		return 0;
	}

1536
	psp->hdcp_context.context.mem_context.shared_mem_size = PSP_HDCP_SHARED_MEM_SIZE;
1537
	psp->hdcp_context.context.ta_load_type = GFX_CMD_ID_LOAD_TA;
1538

1539
	if (!psp->hdcp_context.context.initialized) {
B
Bhawanpreet Lakha 已提交
1540 1541 1542 1543 1544 1545
		ret = psp_hdcp_init_shared_buf(psp);
		if (ret)
			return ret;
	}

	ret = psp_hdcp_load(psp);
1546 1547 1548 1549
	if (!ret) {
		psp->hdcp_context.context.initialized = true;
		mutex_init(&psp->hdcp_context.mutex);
	}
B
Bhawanpreet Lakha 已提交
1550

1551
	return ret;
B
Bhawanpreet Lakha 已提交
1552 1553 1554 1555
}

static int psp_hdcp_unload(struct psp_context *psp)
{
1556
	return psp_ta_unload(psp, &psp->hdcp_context.context);
B
Bhawanpreet Lakha 已提交
1557 1558 1559 1560 1561 1562 1563 1564 1565 1566
}

int psp_hdcp_invoke(struct psp_context *psp, uint32_t ta_cmd_id)
{
	/*
	 * TODO: bypass the loading in sriov for now
	 */
	if (amdgpu_sriov_vf(psp->adev))
		return 0;

1567
	return psp_ta_invoke(psp, ta_cmd_id, &psp->hdcp_context.context);
B
Bhawanpreet Lakha 已提交
1568 1569 1570 1571 1572 1573
}

static int psp_hdcp_terminate(struct psp_context *psp)
{
	int ret;

1574 1575 1576 1577 1578 1579
	/*
	 * TODO: bypass the terminate in sriov for now
	 */
	if (amdgpu_sriov_vf(psp->adev))
		return 0;

1580 1581
	if (!psp->hdcp_context.context.initialized) {
		if (psp->hdcp_context.context.mem_context.shared_buf)
1582 1583 1584 1585
			goto out;
		else
			return 0;
	}
B
Bhawanpreet Lakha 已提交
1586 1587 1588 1589 1590

	ret = psp_hdcp_unload(psp);
	if (ret)
		return ret;

1591
	psp->hdcp_context.context.initialized = false;
B
Bhawanpreet Lakha 已提交
1592

1593
out:
B
Bhawanpreet Lakha 已提交
1594
	/* free hdcp shared memory */
1595
	psp_ta_free_shared_buf(&psp->hdcp_context.context.mem_context);
B
Bhawanpreet Lakha 已提交
1596 1597 1598 1599 1600

	return 0;
}
// HDCP end

B
Bhawanpreet Lakha 已提交
1601 1602 1603
// DTM start
static int psp_dtm_init_shared_buf(struct psp_context *psp)
{
1604
	return psp_ta_init_shared_buf(psp, &psp->dtm_context.context.mem_context);
B
Bhawanpreet Lakha 已提交
1605 1606 1607 1608
}

static int psp_dtm_load(struct psp_context *psp)
{
C
Candice Li 已提交
1609
	return psp_ta_load(psp, &psp->dtm_context.context);
B
Bhawanpreet Lakha 已提交
1610 1611 1612 1613 1614 1615
}

static int psp_dtm_initialize(struct psp_context *psp)
{
	int ret;

1616 1617 1618 1619 1620 1621
	/*
	 * TODO: bypass the initialize in sriov for now
	 */
	if (amdgpu_sriov_vf(psp->adev))
		return 0;

C
Candice Li 已提交
1622 1623
	if (!psp->dtm_context.context.bin_desc.size_bytes ||
	    !psp->dtm_context.context.bin_desc.start_addr) {
1624
		dev_info(psp->adev->dev, "DTM: optional dtm ta ucode is not available\n");
1625 1626 1627
		return 0;
	}

1628
	psp->dtm_context.context.mem_context.shared_mem_size = PSP_DTM_SHARED_MEM_SIZE;
1629
	psp->dtm_context.context.ta_load_type = GFX_CMD_ID_LOAD_TA;
1630

1631
	if (!psp->dtm_context.context.initialized) {
B
Bhawanpreet Lakha 已提交
1632 1633 1634 1635 1636 1637
		ret = psp_dtm_init_shared_buf(psp);
		if (ret)
			return ret;
	}

	ret = psp_dtm_load(psp);
1638 1639 1640 1641
	if (!ret) {
		psp->dtm_context.context.initialized = true;
		mutex_init(&psp->dtm_context.mutex);
	}
B
Bhawanpreet Lakha 已提交
1642

1643
	return ret;
B
Bhawanpreet Lakha 已提交
1644 1645
}

1646 1647
static int psp_dtm_unload(struct psp_context *psp)
{
1648
	return psp_ta_unload(psp, &psp->dtm_context.context);
1649 1650
}

B
Bhawanpreet Lakha 已提交
1651 1652 1653 1654 1655 1656 1657 1658
int psp_dtm_invoke(struct psp_context *psp, uint32_t ta_cmd_id)
{
	/*
	 * TODO: bypass the loading in sriov for now
	 */
	if (amdgpu_sriov_vf(psp->adev))
		return 0;

1659
	return psp_ta_invoke(psp, ta_cmd_id, &psp->dtm_context.context);
B
Bhawanpreet Lakha 已提交
1660 1661 1662 1663 1664 1665
}

static int psp_dtm_terminate(struct psp_context *psp)
{
	int ret;

1666 1667 1668 1669 1670 1671
	/*
	 * TODO: bypass the terminate in sriov for now
	 */
	if (amdgpu_sriov_vf(psp->adev))
		return 0;

1672 1673
	if (!psp->dtm_context.context.initialized) {
		if (psp->dtm_context.context.mem_context.shared_buf)
1674 1675 1676 1677
			goto out;
		else
			return 0;
	}
B
Bhawanpreet Lakha 已提交
1678

1679
	ret = psp_dtm_unload(psp);
B
Bhawanpreet Lakha 已提交
1680 1681 1682
	if (ret)
		return ret;

1683
	psp->dtm_context.context.initialized = false;
B
Bhawanpreet Lakha 已提交
1684

1685
out:
1686 1687
	/* free dtm shared memory */
	psp_ta_free_shared_buf(&psp->dtm_context.context.mem_context);
B
Bhawanpreet Lakha 已提交
1688 1689 1690 1691 1692

	return 0;
}
// DTM end

W
Wenhui Sheng 已提交
1693 1694 1695
// RAP start
static int psp_rap_init_shared_buf(struct psp_context *psp)
{
1696
	return psp_ta_init_shared_buf(psp, &psp->rap_context.context.mem_context);
W
Wenhui Sheng 已提交
1697 1698 1699 1700
}

static int psp_rap_load(struct psp_context *psp)
{
C
Candice Li 已提交
1701
	return psp_ta_load(psp, &psp->rap_context.context);
W
Wenhui Sheng 已提交
1702 1703 1704 1705
}

static int psp_rap_unload(struct psp_context *psp)
{
1706
	return psp_ta_unload(psp, &psp->rap_context.context);
W
Wenhui Sheng 已提交
1707 1708 1709 1710 1711
}

static int psp_rap_initialize(struct psp_context *psp)
{
	int ret;
1712
	enum ta_rap_status status = TA_RAP_STATUS__SUCCESS;
W
Wenhui Sheng 已提交
1713 1714 1715 1716 1717 1718 1719

	/*
	 * TODO: bypass the initialize in sriov for now
	 */
	if (amdgpu_sriov_vf(psp->adev))
		return 0;

C
Candice Li 已提交
1720 1721
	if (!psp->rap_context.context.bin_desc.size_bytes ||
	    !psp->rap_context.context.bin_desc.start_addr) {
W
Wenhui Sheng 已提交
1722 1723 1724 1725
		dev_info(psp->adev->dev, "RAP: optional rap ta ucode is not available\n");
		return 0;
	}

1726
	psp->rap_context.context.mem_context.shared_mem_size = PSP_RAP_SHARED_MEM_SIZE;
1727
	psp->rap_context.context.ta_load_type = GFX_CMD_ID_LOAD_TA;
1728

1729
	if (!psp->rap_context.context.initialized) {
W
Wenhui Sheng 已提交
1730 1731 1732 1733 1734 1735
		ret = psp_rap_init_shared_buf(psp);
		if (ret)
			return ret;
	}

	ret = psp_rap_load(psp);
1736 1737 1738 1739
	if (!ret) {
		psp->rap_context.context.initialized = true;
		mutex_init(&psp->rap_context.mutex);
	} else
W
Wenhui Sheng 已提交
1740 1741
		return ret;

1742 1743
	ret = psp_rap_invoke(psp, TA_CMD_RAP__INITIALIZE, &status);
	if (ret || status != TA_RAP_STATUS__SUCCESS) {
1744
		psp_rap_terminate(psp);
W
Wenhui Sheng 已提交
1745

1746 1747 1748 1749
		dev_warn(psp->adev->dev, "RAP TA initialize fail (%d) status %d.\n",
			 ret, status);

		return ret;
W
Wenhui Sheng 已提交
1750 1751 1752 1753 1754 1755 1756 1757 1758
	}

	return 0;
}

static int psp_rap_terminate(struct psp_context *psp)
{
	int ret;

1759
	if (!psp->rap_context.context.initialized)
W
Wenhui Sheng 已提交
1760 1761 1762 1763
		return 0;

	ret = psp_rap_unload(psp);

1764
	psp->rap_context.context.initialized = false;
W
Wenhui Sheng 已提交
1765 1766

	/* free rap shared memory */
1767
	psp_ta_free_shared_buf(&psp->rap_context.context.mem_context);
W
Wenhui Sheng 已提交
1768 1769 1770 1771

	return ret;
}

1772
int psp_rap_invoke(struct psp_context *psp, uint32_t ta_cmd_id, enum ta_rap_status *status)
W
Wenhui Sheng 已提交
1773 1774
{
	struct ta_rap_shared_memory *rap_cmd;
1775
	int ret = 0;
W
Wenhui Sheng 已提交
1776

1777
	if (!psp->rap_context.context.initialized)
1778
		return 0;
W
Wenhui Sheng 已提交
1779 1780 1781 1782 1783 1784 1785 1786

	if (ta_cmd_id != TA_CMD_RAP__INITIALIZE &&
	    ta_cmd_id != TA_CMD_RAP__VALIDATE_L0)
		return -EINVAL;

	mutex_lock(&psp->rap_context.mutex);

	rap_cmd = (struct ta_rap_shared_memory *)
1787
		  psp->rap_context.context.mem_context.shared_buf;
W
Wenhui Sheng 已提交
1788 1789 1790 1791 1792
	memset(rap_cmd, 0, sizeof(struct ta_rap_shared_memory));

	rap_cmd->cmd_id = ta_cmd_id;
	rap_cmd->validation_method_id = METHOD_A;

1793
	ret = psp_ta_invoke(psp, rap_cmd->cmd_id, &psp->rap_context.context);
1794 1795 1796 1797 1798
	if (ret)
		goto out_unlock;

	if (status)
		*status = rap_cmd->rap_status;
W
Wenhui Sheng 已提交
1799

1800
out_unlock:
W
Wenhui Sheng 已提交
1801 1802
	mutex_unlock(&psp->rap_context.mutex);

1803
	return ret;
W
Wenhui Sheng 已提交
1804 1805 1806
}
// RAP end

1807 1808 1809
/* securedisplay start */
static int psp_securedisplay_init_shared_buf(struct psp_context *psp)
{
1810
	return psp_ta_init_shared_buf(
1811
		psp, &psp->securedisplay_context.context.mem_context);
1812 1813 1814 1815
}

static int psp_securedisplay_load(struct psp_context *psp)
{
C
Candice Li 已提交
1816
	return psp_ta_load(psp, &psp->securedisplay_context.context);
1817 1818 1819 1820
}

static int psp_securedisplay_unload(struct psp_context *psp)
{
1821
	return psp_ta_unload(psp, &psp->securedisplay_context.context);
1822 1823 1824 1825 1826 1827 1828 1829 1830 1831 1832 1833 1834
}

static int psp_securedisplay_initialize(struct psp_context *psp)
{
	int ret;
	struct securedisplay_cmd *securedisplay_cmd;

	/*
	 * TODO: bypass the initialize in sriov for now
	 */
	if (amdgpu_sriov_vf(psp->adev))
		return 0;

C
Candice Li 已提交
1835 1836
	if (!psp->securedisplay_context.context.bin_desc.size_bytes ||
	    !psp->securedisplay_context.context.bin_desc.start_addr) {
1837 1838 1839 1840
		dev_info(psp->adev->dev, "SECUREDISPLAY: securedisplay ta ucode is not available\n");
		return 0;
	}

1841 1842
	psp->securedisplay_context.context.mem_context.shared_mem_size =
		PSP_SECUREDISPLAY_SHARED_MEM_SIZE;
1843
	psp->securedisplay_context.context.ta_load_type = GFX_CMD_ID_LOAD_TA;
1844

1845
	if (!psp->securedisplay_context.context.initialized) {
1846 1847 1848 1849 1850 1851
		ret = psp_securedisplay_init_shared_buf(psp);
		if (ret)
			return ret;
	}

	ret = psp_securedisplay_load(psp);
1852 1853 1854 1855
	if (!ret) {
		psp->securedisplay_context.context.initialized = true;
		mutex_init(&psp->securedisplay_context.mutex);
	} else
1856 1857 1858 1859 1860 1861 1862
		return ret;

	psp_prep_securedisplay_cmd_buf(psp, &securedisplay_cmd,
			TA_SECUREDISPLAY_COMMAND__QUERY_TA);

	ret = psp_securedisplay_invoke(psp, TA_SECUREDISPLAY_COMMAND__QUERY_TA);
	if (ret) {
1863
		psp_securedisplay_terminate(psp);
1864 1865 1866 1867 1868 1869 1870 1871 1872 1873 1874 1875 1876 1877 1878 1879 1880 1881 1882 1883 1884 1885 1886
		dev_err(psp->adev->dev, "SECUREDISPLAY TA initialize fail.\n");
		return -EINVAL;
	}

	if (securedisplay_cmd->status != TA_SECUREDISPLAY_STATUS__SUCCESS) {
		psp_securedisplay_parse_resp_status(psp, securedisplay_cmd->status);
		dev_err(psp->adev->dev, "SECUREDISPLAY: query securedisplay TA failed. ret 0x%x\n",
			securedisplay_cmd->securedisplay_out_message.query_ta.query_cmd_ret);
	}

	return 0;
}

static int psp_securedisplay_terminate(struct psp_context *psp)
{
	int ret;

	/*
	 * TODO:bypass the terminate in sriov for now
	 */
	if (amdgpu_sriov_vf(psp->adev))
		return 0;

1887
	if (!psp->securedisplay_context.context.initialized)
1888 1889 1890 1891 1892 1893
		return 0;

	ret = psp_securedisplay_unload(psp);
	if (ret)
		return ret;

1894
	psp->securedisplay_context.context.initialized = false;
1895 1896

	/* free securedisplay shared memory */
1897
	psp_ta_free_shared_buf(&psp->securedisplay_context.context.mem_context);
1898 1899 1900 1901 1902 1903 1904 1905

	return ret;
}

int psp_securedisplay_invoke(struct psp_context *psp, uint32_t ta_cmd_id)
{
	int ret;

1906
	if (!psp->securedisplay_context.context.initialized)
1907 1908 1909 1910 1911 1912 1913 1914
		return -EINVAL;

	if (ta_cmd_id != TA_SECUREDISPLAY_COMMAND__QUERY_TA &&
	    ta_cmd_id != TA_SECUREDISPLAY_COMMAND__SEND_ROI_CRC)
		return -EINVAL;

	mutex_lock(&psp->securedisplay_context.mutex);

1915
	ret = psp_ta_invoke(psp, ta_cmd_id, &psp->securedisplay_context.context);
1916 1917 1918 1919 1920 1921 1922

	mutex_unlock(&psp->securedisplay_context.mutex);

	return ret;
}
/* SECUREDISPLAY end */

1923
static int psp_hw_start(struct psp_context *psp)
1924
{
1925
	struct amdgpu_device *adev = psp->adev;
1926 1927
	int ret;

1928
	if (!amdgpu_sriov_vf(adev)) {
1929
		if ((is_psp_fw_valid(psp->kdb)) &&
1930 1931 1932 1933
		    (psp->funcs->bootloader_load_kdb != NULL)) {
			ret = psp_bootloader_load_kdb(psp);
			if (ret) {
				DRM_ERROR("PSP load kdb failed!\n");
1934 1935 1936 1937
				return ret;
			}
		}

1938 1939
		if ((is_psp_fw_valid(psp->spl)) &&
		    (psp->funcs->bootloader_load_spl != NULL)) {
1940 1941 1942
			ret = psp_bootloader_load_spl(psp);
			if (ret) {
				DRM_ERROR("PSP load spl failed!\n");
1943 1944 1945 1946
				return ret;
			}
		}

1947 1948 1949 1950
		if ((is_psp_fw_valid(psp->sys)) &&
		    (psp->funcs->bootloader_load_sysdrv != NULL)) {
			ret = psp_bootloader_load_sysdrv(psp);
			if (ret) {
1951 1952 1953 1954 1955 1956 1957 1958 1959 1960 1961 1962 1963 1964 1965 1966 1967 1968 1969 1970 1971 1972 1973 1974 1975 1976 1977 1978
				DRM_ERROR("PSP load sys drv failed!\n");
				return ret;
			}
		}

		if ((is_psp_fw_valid(psp->soc_drv)) &&
		    (psp->funcs->bootloader_load_soc_drv != NULL)) {
			ret = psp_bootloader_load_soc_drv(psp);
			if (ret) {
				DRM_ERROR("PSP load soc drv failed!\n");
				return ret;
			}
		}

		if ((is_psp_fw_valid(psp->intf_drv)) &&
		    (psp->funcs->bootloader_load_intf_drv != NULL)) {
			ret = psp_bootloader_load_intf_drv(psp);
			if (ret) {
				DRM_ERROR("PSP load intf drv failed!\n");
				return ret;
			}
		}

		if ((is_psp_fw_valid(psp->dbg_drv)) &&
		    (psp->funcs->bootloader_load_dbg_drv != NULL)) {
			ret = psp_bootloader_load_dbg_drv(psp);
			if (ret) {
				DRM_ERROR("PSP load dbg drv failed!\n");
1979 1980
				return ret;
			}
1981
		}
1982

1983 1984 1985 1986 1987 1988 1989
		if ((is_psp_fw_valid(psp->sos)) &&
		    (psp->funcs->bootloader_load_sos != NULL)) {
			ret = psp_bootloader_load_sos(psp);
			if (ret) {
				DRM_ERROR("PSP load sos failed!\n");
				return ret;
			}
1990
		}
1991
	}
1992

1993
	ret = psp_ring_create(psp, PSP_RING_TYPE__KM);
1994 1995
	if (ret) {
		DRM_ERROR("PSP create ring failed!\n");
1996
		return ret;
1997
	}
1998

1999 2000 2001 2002 2003 2004
	ret = psp_tmr_init(psp);
	if (ret) {
		DRM_ERROR("PSP tmr init failed!\n");
		return ret;
	}

2005
	/*
2006
	 * For ASICs with DF Cstate management centralized
2007 2008 2009
	 * to PMFW, TMR setup should be performed after PMFW
	 * loaded and before other non-psp firmware loaded.
	 */
2010 2011 2012
	if (psp->pmfw_centralized_cstate_management) {
		ret = psp_load_smu_fw(psp);
		if (ret)
2013
			return ret;
2014 2015 2016 2017 2018 2019
	}

	ret = psp_tmr_load(psp);
	if (ret) {
		DRM_ERROR("PSP load tmr failed!\n");
		return ret;
2020
	}
2021

2022 2023 2024
	return 0;
}

2025 2026 2027 2028 2029 2030 2031 2032 2033 2034
static int psp_get_fw_type(struct amdgpu_firmware_info *ucode,
			   enum psp_gfx_fw_type *type)
{
	switch (ucode->ucode_id) {
	case AMDGPU_UCODE_ID_SDMA0:
		*type = GFX_FW_TYPE_SDMA0;
		break;
	case AMDGPU_UCODE_ID_SDMA1:
		*type = GFX_FW_TYPE_SDMA1;
		break;
2035 2036 2037 2038 2039 2040 2041 2042 2043 2044 2045 2046 2047 2048 2049 2050 2051 2052
	case AMDGPU_UCODE_ID_SDMA2:
		*type = GFX_FW_TYPE_SDMA2;
		break;
	case AMDGPU_UCODE_ID_SDMA3:
		*type = GFX_FW_TYPE_SDMA3;
		break;
	case AMDGPU_UCODE_ID_SDMA4:
		*type = GFX_FW_TYPE_SDMA4;
		break;
	case AMDGPU_UCODE_ID_SDMA5:
		*type = GFX_FW_TYPE_SDMA5;
		break;
	case AMDGPU_UCODE_ID_SDMA6:
		*type = GFX_FW_TYPE_SDMA6;
		break;
	case AMDGPU_UCODE_ID_SDMA7:
		*type = GFX_FW_TYPE_SDMA7;
		break;
2053 2054 2055 2056 2057 2058
	case AMDGPU_UCODE_ID_CP_MES:
		*type = GFX_FW_TYPE_CP_MES;
		break;
	case AMDGPU_UCODE_ID_CP_MES_DATA:
		*type = GFX_FW_TYPE_MES_STACK;
		break;
2059 2060 2061 2062 2063 2064 2065 2066 2067 2068 2069 2070 2071 2072 2073 2074 2075 2076 2077 2078 2079 2080 2081 2082 2083 2084 2085 2086 2087 2088 2089 2090 2091
	case AMDGPU_UCODE_ID_CP_CE:
		*type = GFX_FW_TYPE_CP_CE;
		break;
	case AMDGPU_UCODE_ID_CP_PFP:
		*type = GFX_FW_TYPE_CP_PFP;
		break;
	case AMDGPU_UCODE_ID_CP_ME:
		*type = GFX_FW_TYPE_CP_ME;
		break;
	case AMDGPU_UCODE_ID_CP_MEC1:
		*type = GFX_FW_TYPE_CP_MEC;
		break;
	case AMDGPU_UCODE_ID_CP_MEC1_JT:
		*type = GFX_FW_TYPE_CP_MEC_ME1;
		break;
	case AMDGPU_UCODE_ID_CP_MEC2:
		*type = GFX_FW_TYPE_CP_MEC;
		break;
	case AMDGPU_UCODE_ID_CP_MEC2_JT:
		*type = GFX_FW_TYPE_CP_MEC_ME2;
		break;
	case AMDGPU_UCODE_ID_RLC_G:
		*type = GFX_FW_TYPE_RLC_G;
		break;
	case AMDGPU_UCODE_ID_RLC_RESTORE_LIST_CNTL:
		*type = GFX_FW_TYPE_RLC_RESTORE_LIST_SRM_CNTL;
		break;
	case AMDGPU_UCODE_ID_RLC_RESTORE_LIST_GPM_MEM:
		*type = GFX_FW_TYPE_RLC_RESTORE_LIST_GPM_MEM;
		break;
	case AMDGPU_UCODE_ID_RLC_RESTORE_LIST_SRM_MEM:
		*type = GFX_FW_TYPE_RLC_RESTORE_LIST_SRM_MEM;
		break;
2092 2093 2094 2095 2096 2097
	case AMDGPU_UCODE_ID_RLC_IRAM:
		*type = GFX_FW_TYPE_RLC_IRAM;
		break;
	case AMDGPU_UCODE_ID_RLC_DRAM:
		*type = GFX_FW_TYPE_RLC_DRAM_BOOT;
		break;
2098 2099 2100 2101 2102 2103 2104 2105 2106 2107 2108 2109 2110 2111 2112
	case AMDGPU_UCODE_ID_SMC:
		*type = GFX_FW_TYPE_SMU;
		break;
	case AMDGPU_UCODE_ID_UVD:
		*type = GFX_FW_TYPE_UVD;
		break;
	case AMDGPU_UCODE_ID_UVD1:
		*type = GFX_FW_TYPE_UVD1;
		break;
	case AMDGPU_UCODE_ID_VCE:
		*type = GFX_FW_TYPE_VCE;
		break;
	case AMDGPU_UCODE_ID_VCN:
		*type = GFX_FW_TYPE_VCN;
		break;
2113 2114 2115
	case AMDGPU_UCODE_ID_VCN1:
		*type = GFX_FW_TYPE_VCN1;
		break;
2116 2117 2118 2119 2120 2121
	case AMDGPU_UCODE_ID_DMCU_ERAM:
		*type = GFX_FW_TYPE_DMCU_ERAM;
		break;
	case AMDGPU_UCODE_ID_DMCU_INTV:
		*type = GFX_FW_TYPE_DMCU_ISR;
		break;
2122 2123 2124 2125 2126 2127
	case AMDGPU_UCODE_ID_VCN0_RAM:
		*type = GFX_FW_TYPE_VCN0_RAM;
		break;
	case AMDGPU_UCODE_ID_VCN1_RAM:
		*type = GFX_FW_TYPE_VCN1_RAM;
		break;
2128 2129 2130
	case AMDGPU_UCODE_ID_DMCUB:
		*type = GFX_FW_TYPE_DMUB;
		break;
2131 2132 2133 2134 2135 2136 2137 2138
	case AMDGPU_UCODE_ID_MAXIMUM:
	default:
		return -EINVAL;
	}

	return 0;
}

2139 2140 2141 2142
static void psp_print_fw_hdr(struct psp_context *psp,
			     struct amdgpu_firmware_info *ucode)
{
	struct amdgpu_device *adev = psp->adev;
2143
	struct common_firmware_header *hdr;
2144 2145 2146 2147 2148 2149 2150 2151 2152 2153

	switch (ucode->ucode_id) {
	case AMDGPU_UCODE_ID_SDMA0:
	case AMDGPU_UCODE_ID_SDMA1:
	case AMDGPU_UCODE_ID_SDMA2:
	case AMDGPU_UCODE_ID_SDMA3:
	case AMDGPU_UCODE_ID_SDMA4:
	case AMDGPU_UCODE_ID_SDMA5:
	case AMDGPU_UCODE_ID_SDMA6:
	case AMDGPU_UCODE_ID_SDMA7:
2154 2155 2156
		hdr = (struct common_firmware_header *)
			adev->sdma.instance[ucode->ucode_id - AMDGPU_UCODE_ID_SDMA0].fw->data;
		amdgpu_ucode_print_sdma_hdr(hdr);
2157 2158
		break;
	case AMDGPU_UCODE_ID_CP_CE:
2159 2160
		hdr = (struct common_firmware_header *)adev->gfx.ce_fw->data;
		amdgpu_ucode_print_gfx_hdr(hdr);
2161 2162
		break;
	case AMDGPU_UCODE_ID_CP_PFP:
2163 2164
		hdr = (struct common_firmware_header *)adev->gfx.pfp_fw->data;
		amdgpu_ucode_print_gfx_hdr(hdr);
2165 2166
		break;
	case AMDGPU_UCODE_ID_CP_ME:
2167 2168
		hdr = (struct common_firmware_header *)adev->gfx.me_fw->data;
		amdgpu_ucode_print_gfx_hdr(hdr);
2169 2170
		break;
	case AMDGPU_UCODE_ID_CP_MEC1:
2171 2172
		hdr = (struct common_firmware_header *)adev->gfx.mec_fw->data;
		amdgpu_ucode_print_gfx_hdr(hdr);
2173 2174
		break;
	case AMDGPU_UCODE_ID_RLC_G:
2175 2176
		hdr = (struct common_firmware_header *)adev->gfx.rlc_fw->data;
		amdgpu_ucode_print_rlc_hdr(hdr);
2177 2178
		break;
	case AMDGPU_UCODE_ID_SMC:
2179 2180
		hdr = (struct common_firmware_header *)adev->pm.fw->data;
		amdgpu_ucode_print_smc_hdr(hdr);
2181 2182 2183 2184 2185 2186
		break;
	default:
		break;
	}
}

2187 2188 2189 2190 2191 2192 2193 2194 2195 2196 2197 2198 2199 2200 2201 2202 2203 2204
static int psp_prep_load_ip_fw_cmd_buf(struct amdgpu_firmware_info *ucode,
				       struct psp_gfx_cmd_resp *cmd)
{
	int ret;
	uint64_t fw_mem_mc_addr = ucode->mc_addr;

	cmd->cmd_id = GFX_CMD_ID_LOAD_IP_FW;
	cmd->cmd.cmd_load_ip_fw.fw_phy_addr_lo = lower_32_bits(fw_mem_mc_addr);
	cmd->cmd.cmd_load_ip_fw.fw_phy_addr_hi = upper_32_bits(fw_mem_mc_addr);
	cmd->cmd.cmd_load_ip_fw.fw_size = ucode->ucode_size;

	ret = psp_get_fw_type(ucode, &cmd->cmd.cmd_load_ip_fw.fw_type);
	if (ret)
		DRM_ERROR("Unknown firmware type\n");

	return ret;
}

2205
static int psp_execute_non_psp_fw_load(struct psp_context *psp,
2206
			          struct amdgpu_firmware_info *ucode)
2207 2208
{
	int ret = 0;
2209
	struct psp_gfx_cmd_resp *cmd = acquire_psp_cmd_buf(psp);
2210

2211 2212 2213 2214 2215
	ret = psp_prep_load_ip_fw_cmd_buf(ucode, cmd);
	if (!ret) {
		ret = psp_cmd_submit_buf(psp, ucode, cmd,
					 psp->fence_buf_mc_addr);
	}
2216

2217
	release_psp_cmd_buf(psp);
2218 2219 2220 2221

	return ret;
}

2222 2223 2224
static int psp_load_smu_fw(struct psp_context *psp)
{
	int ret;
2225
	struct amdgpu_device *adev = psp->adev;
2226
	struct amdgpu_firmware_info *ucode =
2227
			&adev->firmware.ucode[AMDGPU_UCODE_ID_SMC];
2228
	struct amdgpu_ras *ras = psp->ras_context.ras;
2229 2230 2231 2232

	if (!ucode->fw || amdgpu_sriov_vf(psp->adev))
		return 0;

2233
	if ((amdgpu_in_reset(adev) &&
2234
	     ras && adev->ras_enabled &&
2235 2236
	     (adev->ip_versions[MP0_HWIP][0] == IP_VERSION(11, 0, 4) ||
	      adev->ip_versions[MP0_HWIP][0] == IP_VERSION(11, 0, 2)))) {
2237 2238 2239 2240 2241 2242
		ret = amdgpu_dpm_set_mp1_state(adev, PP_MP1_STATE_UNLOAD);
		if (ret) {
			DRM_WARN("Failed to set MP1 state prepare for reload\n");
		}
	}

2243
	ret = psp_execute_non_psp_fw_load(psp, ucode);
2244 2245 2246 2247 2248 2249 2250 2251 2252 2253 2254 2255 2256 2257 2258 2259 2260 2261 2262 2263 2264 2265 2266 2267 2268 2269 2270 2271 2272 2273 2274 2275 2276 2277 2278 2279 2280 2281 2282 2283 2284 2285 2286 2287 2288

	if (ret)
		DRM_ERROR("PSP load smu failed!\n");

	return ret;
}

static bool fw_load_skip_check(struct psp_context *psp,
			       struct amdgpu_firmware_info *ucode)
{
	if (!ucode->fw)
		return true;

	if (ucode->ucode_id == AMDGPU_UCODE_ID_SMC &&
	    (psp_smu_reload_quirk(psp) ||
	     psp->autoload_supported ||
	     psp->pmfw_centralized_cstate_management))
		return true;

	if (amdgpu_sriov_vf(psp->adev) &&
	   (ucode->ucode_id == AMDGPU_UCODE_ID_SDMA0
	    || ucode->ucode_id == AMDGPU_UCODE_ID_SDMA1
	    || ucode->ucode_id == AMDGPU_UCODE_ID_SDMA2
	    || ucode->ucode_id == AMDGPU_UCODE_ID_SDMA3
	    || ucode->ucode_id == AMDGPU_UCODE_ID_SDMA4
	    || ucode->ucode_id == AMDGPU_UCODE_ID_SDMA5
	    || ucode->ucode_id == AMDGPU_UCODE_ID_SDMA6
	    || ucode->ucode_id == AMDGPU_UCODE_ID_SDMA7
	    || ucode->ucode_id == AMDGPU_UCODE_ID_RLC_G
	    || ucode->ucode_id == AMDGPU_UCODE_ID_RLC_RESTORE_LIST_CNTL
	    || ucode->ucode_id == AMDGPU_UCODE_ID_RLC_RESTORE_LIST_GPM_MEM
	    || ucode->ucode_id == AMDGPU_UCODE_ID_RLC_RESTORE_LIST_SRM_MEM
	    || ucode->ucode_id == AMDGPU_UCODE_ID_SMC))
		/*skip ucode loading in SRIOV VF */
		return true;

	if (psp->autoload_supported &&
	    (ucode->ucode_id == AMDGPU_UCODE_ID_CP_MEC1_JT ||
	     ucode->ucode_id == AMDGPU_UCODE_ID_CP_MEC2_JT))
		/* skip mec JT when autoload is enabled */
		return true;

	return false;
}

2289 2290 2291 2292 2293 2294 2295 2296 2297
int psp_load_fw_list(struct psp_context *psp,
		     struct amdgpu_firmware_info **ucode_list, int ucode_count)
{
	int ret = 0, i;
	struct amdgpu_firmware_info *ucode;

	for (i = 0; i < ucode_count; ++i) {
		ucode = ucode_list[i];
		psp_print_fw_hdr(psp, ucode);
2298
		ret = psp_execute_non_psp_fw_load(psp, ucode);
2299 2300 2301 2302 2303 2304
		if (ret)
			return ret;
	}
	return ret;
}

2305
static int psp_load_non_psp_fw(struct psp_context *psp)
2306 2307
{
	int i, ret;
2308
	struct amdgpu_firmware_info *ucode;
2309
	struct amdgpu_device *adev = psp->adev;
2310

2311 2312 2313
	if (psp->autoload_supported &&
	    !psp->pmfw_centralized_cstate_management) {
		ret = psp_load_smu_fw(psp);
2314 2315 2316 2317
		if (ret)
			return ret;
	}

2318 2319 2320 2321
	for (i = 0; i < adev->firmware.max_ucodes; i++) {
		ucode = &adev->firmware.ucode[i];

		if (ucode->ucode_id == AMDGPU_UCODE_ID_SMC &&
2322 2323 2324 2325
		    !fw_load_skip_check(psp, ucode)) {
			ret = psp_load_smu_fw(psp);
			if (ret)
				return ret;
2326
			continue;
2327
		}
2328

2329
		if (fw_load_skip_check(psp, ucode))
2330
			continue;
2331

2332
		if (psp->autoload_supported &&
2333 2334 2335
		    (adev->ip_versions[MP0_HWIP][0] == IP_VERSION(11, 0, 7) ||
		     adev->ip_versions[MP0_HWIP][0] == IP_VERSION(11, 0, 11) ||
		     adev->ip_versions[MP0_HWIP][0] == IP_VERSION(11, 0, 12)) &&
2336 2337 2338 2339 2340 2341 2342
		    (ucode->ucode_id == AMDGPU_UCODE_ID_SDMA1 ||
		     ucode->ucode_id == AMDGPU_UCODE_ID_SDMA2 ||
		     ucode->ucode_id == AMDGPU_UCODE_ID_SDMA3))
			/* PSP only receive one SDMA fw for sienna_cichlid,
			 * as all four sdma fw are same */
			continue;

2343 2344
		psp_print_fw_hdr(psp, ucode);

2345
		ret = psp_execute_non_psp_fw_load(psp, ucode);
2346
		if (ret)
2347
			return ret;
2348

2349
		/* Start rlc autoload after psp recieved all the gfx firmware */
2350
		if (psp->autoload_supported && ucode->ucode_id == (amdgpu_sriov_vf(adev) ?
2351
		    AMDGPU_UCODE_ID_CP_MEC2 : AMDGPU_UCODE_ID_RLC_G)) {
2352
			ret = psp_rlc_autoload_start(psp);
2353 2354 2355 2356 2357
			if (ret) {
				DRM_ERROR("Failed to start rlc autoload\n");
				return ret;
			}
		}
2358 2359
	}

2360 2361 2362 2363 2364 2365
	return 0;
}

static int psp_load_fw(struct amdgpu_device *adev)
{
	int ret;
2366 2367
	struct psp_context *psp = &adev->psp;

2368
	if (amdgpu_sriov_vf(adev) && amdgpu_in_reset(adev)) {
2369
		psp_ring_stop(psp, PSP_RING_TYPE__KM); /* should not destroy ring, only stop */
2370
		goto skip_memalloc;
2371
	}
2372

2373 2374 2375 2376 2377 2378 2379 2380 2381 2382 2383 2384 2385 2386
	if (amdgpu_sriov_vf(adev)) {
		ret = amdgpu_bo_create_kernel(adev, PSP_1_MEG, PSP_1_MEG,
						AMDGPU_GEM_DOMAIN_VRAM,
						&psp->fw_pri_bo,
						&psp->fw_pri_mc_addr,
						&psp->fw_pri_buf);
	} else {
		ret = amdgpu_bo_create_kernel(adev, PSP_1_MEG, PSP_1_MEG,
						AMDGPU_GEM_DOMAIN_GTT,
						&psp->fw_pri_bo,
						&psp->fw_pri_mc_addr,
						&psp->fw_pri_buf);
	}

2387 2388
	if (ret)
		goto failed;
2389 2390

	ret = amdgpu_bo_create_kernel(adev, PSP_FENCE_BUFFER_SIZE, PAGE_SIZE,
2391 2392 2393 2394
					AMDGPU_GEM_DOMAIN_VRAM,
					&psp->fence_buf_bo,
					&psp->fence_buf_mc_addr,
					&psp->fence_buf);
2395
	if (ret)
2396
		goto failed;
2397 2398 2399 2400 2401

	ret = amdgpu_bo_create_kernel(adev, PSP_CMD_BUFFER_SIZE, PAGE_SIZE,
				      AMDGPU_GEM_DOMAIN_VRAM,
				      &psp->cmd_buf_bo, &psp->cmd_buf_mc_addr,
				      (void **)&psp->cmd_buf_mem);
2402
	if (ret)
2403
		goto failed;
2404 2405 2406

	memset(psp->fence_buf, 0, PSP_FENCE_BUFFER_SIZE);

2407
	ret = psp_ring_init(psp, PSP_RING_TYPE__KM);
2408 2409
	if (ret) {
		DRM_ERROR("PSP ring init failed!\n");
2410
		goto failed;
2411
	}
2412

2413
skip_memalloc:
2414
	ret = psp_hw_start(psp);
2415
	if (ret)
2416
		goto failed;
2417

2418
	ret = psp_load_non_psp_fw(psp);
2419
	if (ret)
2420
		goto failed;
2421

2422
	ret = psp_asd_initialize(psp);
2423 2424 2425 2426 2427
	if (ret) {
		DRM_ERROR("PSP load asd failed!\n");
		return ret;
	}

2428 2429 2430 2431 2432 2433
	ret = psp_rl_load(adev);
	if (ret) {
		DRM_ERROR("PSP load RL failed!\n");
		return ret;
	}

2434
	if (psp->ta_fw) {
2435 2436 2437 2438 2439 2440 2441 2442 2443 2444 2445 2446 2447 2448
		ret = psp_ras_initialize(psp);
		if (ret)
			dev_err(psp->adev->dev,
					"RAS: Failed to initialize RAS\n");

		ret = psp_hdcp_initialize(psp);
		if (ret)
			dev_err(psp->adev->dev,
				"HDCP: Failed to initialize HDCP\n");

		ret = psp_dtm_initialize(psp);
		if (ret)
			dev_err(psp->adev->dev,
				"DTM: Failed to initialize DTM\n");
W
Wenhui Sheng 已提交
2449 2450 2451 2452 2453

		ret = psp_rap_initialize(psp);
		if (ret)
			dev_err(psp->adev->dev,
				"RAP: Failed to initialize RAP\n");
2454 2455 2456 2457 2458

		ret = psp_securedisplay_initialize(psp);
		if (ret)
			dev_err(psp->adev->dev,
				"SECUREDISPLAY: Failed to initialize SECUREDISPLAY\n");
2459 2460
	}

2461 2462 2463
	return 0;

failed:
2464 2465 2466 2467 2468
	/*
	 * all cleanup jobs (xgmi terminate, ras terminate,
	 * ring destroy, cmd/fence/fw buffers destory,
	 * psp->cmd destory) are delayed to psp_hw_fini
	 */
2469 2470 2471 2472 2473 2474 2475 2476 2477
	return ret;
}

static int psp_hw_init(void *handle)
{
	int ret;
	struct amdgpu_device *adev = (struct amdgpu_device *)handle;

	mutex_lock(&adev->firmware.mutex);
2478 2479 2480 2481 2482 2483 2484
	/*
	 * This sequence is just used on hw_init only once, no need on
	 * resume.
	 */
	ret = amdgpu_ucode_init_bo(adev);
	if (ret)
		goto failed;
2485 2486 2487 2488 2489 2490 2491 2492

	ret = psp_load_fw(adev);
	if (ret) {
		DRM_ERROR("PSP firmware loading failed\n");
		goto failed;
	}

	mutex_unlock(&adev->firmware.mutex);
2493
	return 0;
2494 2495 2496 2497 2498 2499 2500 2501 2502 2503 2504 2505

failed:
	adev->firmware.load_type = AMDGPU_FW_LOAD_DIRECT;
	mutex_unlock(&adev->firmware.mutex);
	return -EINVAL;
}

static int psp_hw_fini(void *handle)
{
	struct amdgpu_device *adev = (struct amdgpu_device *)handle;
	struct psp_context *psp = &adev->psp;

2506
	if (psp->ta_fw) {
2507
		psp_ras_terminate(psp);
2508
		psp_securedisplay_terminate(psp);
W
Wenhui Sheng 已提交
2509
		psp_rap_terminate(psp);
B
Bhawanpreet Lakha 已提交
2510
		psp_dtm_terminate(psp);
B
Bhawanpreet Lakha 已提交
2511 2512
		psp_hdcp_terminate(psp);
	}
2513

2514
	psp_asd_terminate(psp);
2515

2516
	psp_tmr_terminate(psp);
2517
	psp_ring_destroy(psp, PSP_RING_TYPE__KM);
2518

H
Huang Rui 已提交
2519 2520 2521 2522
	amdgpu_bo_free_kernel(&psp->fw_pri_bo,
			      &psp->fw_pri_mc_addr, &psp->fw_pri_buf);
	amdgpu_bo_free_kernel(&psp->fence_buf_bo,
			      &psp->fence_buf_mc_addr, &psp->fence_buf);
2523 2524
	amdgpu_bo_free_kernel(&psp->cmd_buf_bo, &psp->cmd_buf_mc_addr,
			      (void **)&psp->cmd_buf_mem);
2525

2526 2527 2528 2529 2530
	return 0;
}

static int psp_suspend(void *handle)
{
E
Evan Quan 已提交
2531 2532 2533 2534
	int ret;
	struct amdgpu_device *adev = (struct amdgpu_device *)handle;
	struct psp_context *psp = &adev->psp;

2535
	if (adev->gmc.xgmi.num_physical_nodes > 1 &&
2536
	    psp->xgmi_context.context.initialized) {
2537 2538 2539 2540 2541 2542 2543
		ret = psp_xgmi_terminate(psp);
		if (ret) {
			DRM_ERROR("Failed to terminate xgmi ta\n");
			return ret;
		}
	}

2544
	if (psp->ta_fw) {
2545 2546 2547 2548 2549
		ret = psp_ras_terminate(psp);
		if (ret) {
			DRM_ERROR("Failed to terminate ras ta\n");
			return ret;
		}
B
Bhawanpreet Lakha 已提交
2550 2551 2552 2553 2554
		ret = psp_hdcp_terminate(psp);
		if (ret) {
			DRM_ERROR("Failed to terminate hdcp ta\n");
			return ret;
		}
B
Bhawanpreet Lakha 已提交
2555 2556 2557 2558 2559
		ret = psp_dtm_terminate(psp);
		if (ret) {
			DRM_ERROR("Failed to terminate dtm ta\n");
			return ret;
		}
W
Wenhui Sheng 已提交
2560 2561 2562 2563 2564
		ret = psp_rap_terminate(psp);
		if (ret) {
			DRM_ERROR("Failed to terminate rap ta\n");
			return ret;
		}
2565 2566 2567 2568 2569
		ret = psp_securedisplay_terminate(psp);
		if (ret) {
			DRM_ERROR("Failed to terminate securedisplay ta\n");
			return ret;
		}
2570 2571
	}

2572
	ret = psp_asd_terminate(psp);
2573
	if (ret) {
2574
		DRM_ERROR("Failed to terminate asd\n");
2575 2576 2577
		return ret;
	}

2578 2579
	ret = psp_tmr_terminate(psp);
	if (ret) {
2580
		DRM_ERROR("Failed to terminate tmr\n");
2581 2582 2583
		return ret;
	}

E
Evan Quan 已提交
2584 2585 2586 2587 2588 2589
	ret = psp_ring_stop(psp, PSP_RING_TYPE__KM);
	if (ret) {
		DRM_ERROR("PSP ring stop failed\n");
		return ret;
	}

2590 2591 2592 2593 2594 2595 2596
	return 0;
}

static int psp_resume(void *handle)
{
	int ret;
	struct amdgpu_device *adev = (struct amdgpu_device *)handle;
2597
	struct psp_context *psp = &adev->psp;
2598

2599 2600
	DRM_INFO("PSP is resuming...\n");

2601 2602 2603 2604 2605 2606
	if (psp->mem_train_ctx.enable_mem_training) {
		ret = psp_mem_training(psp, PSP_MEM_TRAIN_RESUME);
		if (ret) {
			DRM_ERROR("Failed to process memory training!\n");
			return ret;
		}
2607 2608
	}

2609 2610
	mutex_lock(&adev->firmware.mutex);

2611
	ret = psp_hw_start(psp);
2612
	if (ret)
2613 2614
		goto failed;

2615
	ret = psp_load_non_psp_fw(psp);
2616 2617
	if (ret)
		goto failed;
2618

2619
	ret = psp_asd_initialize(psp);
2620 2621 2622 2623 2624 2625
	if (ret) {
		DRM_ERROR("PSP load asd failed!\n");
		goto failed;
	}

	if (adev->gmc.xgmi.num_physical_nodes > 1) {
2626
		ret = psp_xgmi_initialize(psp, false, true);
2627 2628 2629 2630 2631 2632 2633 2634
		/* Warning the XGMI seesion initialize failure
		 * Instead of stop driver initialization
		 */
		if (ret)
			dev_err(psp->adev->dev,
				"XGMI: Failed to initialize XGMI session\n");
	}

2635
	if (psp->ta_fw) {
2636 2637 2638 2639 2640 2641 2642 2643 2644 2645 2646 2647 2648 2649
		ret = psp_ras_initialize(psp);
		if (ret)
			dev_err(psp->adev->dev,
					"RAS: Failed to initialize RAS\n");

		ret = psp_hdcp_initialize(psp);
		if (ret)
			dev_err(psp->adev->dev,
				"HDCP: Failed to initialize HDCP\n");

		ret = psp_dtm_initialize(psp);
		if (ret)
			dev_err(psp->adev->dev,
				"DTM: Failed to initialize DTM\n");
W
Wenhui Sheng 已提交
2650 2651 2652 2653 2654

		ret = psp_rap_initialize(psp);
		if (ret)
			dev_err(psp->adev->dev,
				"RAP: Failed to initialize RAP\n");
2655 2656 2657 2658 2659

		ret = psp_securedisplay_initialize(psp);
		if (ret)
			dev_err(psp->adev->dev,
				"SECUREDISPLAY: Failed to initialize SECUREDISPLAY\n");
2660 2661
	}

2662 2663
	mutex_unlock(&adev->firmware.mutex);

2664 2665 2666 2667 2668
	return 0;

failed:
	DRM_ERROR("PSP resume failed\n");
	mutex_unlock(&adev->firmware.mutex);
2669 2670 2671
	return ret;
}

2672
int psp_gpu_reset(struct amdgpu_device *adev)
2673
{
2674 2675
	int ret;

2676 2677 2678
	if (adev->firmware.load_type != AMDGPU_FW_LOAD_PSP)
		return 0;

2679 2680 2681 2682 2683
	mutex_lock(&adev->psp.mutex);
	ret = psp_mode1_reset(&adev->psp);
	mutex_unlock(&adev->psp.mutex);

	return ret;
2684 2685
}

2686 2687 2688
int psp_rlc_autoload_start(struct psp_context *psp)
{
	int ret;
2689
	struct psp_gfx_cmd_resp *cmd = acquire_psp_cmd_buf(psp);
2690 2691 2692 2693 2694

	cmd->cmd_id = GFX_CMD_ID_AUTOLOAD_RLC;

	ret = psp_cmd_submit_buf(psp, NULL, cmd,
				 psp->fence_buf_mc_addr);
2695

2696 2697
	release_psp_cmd_buf(psp);

2698 2699 2700
	return ret;
}

2701 2702 2703 2704 2705 2706 2707 2708 2709 2710
int psp_update_vcn_sram(struct amdgpu_device *adev, int inst_idx,
			uint64_t cmd_gpu_addr, int cmd_size)
{
	struct amdgpu_firmware_info ucode = {0};

	ucode.ucode_id = inst_idx ? AMDGPU_UCODE_ID_VCN1_RAM :
		AMDGPU_UCODE_ID_VCN0_RAM;
	ucode.mc_addr = cmd_gpu_addr;
	ucode.ucode_size = cmd_size;

2711
	return psp_execute_non_psp_fw_load(&adev->psp, &ucode);
2712 2713
}

2714 2715 2716 2717 2718 2719
int psp_ring_cmd_submit(struct psp_context *psp,
			uint64_t cmd_buf_mc_addr,
			uint64_t fence_mc_addr,
			int index)
{
	unsigned int psp_write_ptr_reg = 0;
2720
	struct psp_gfx_rb_frame *write_frame;
2721 2722 2723 2724 2725 2726 2727 2728 2729 2730 2731 2732 2733 2734 2735 2736 2737 2738 2739 2740 2741 2742 2743 2744 2745 2746 2747 2748 2749 2750 2751 2752 2753 2754 2755
	struct psp_ring *ring = &psp->km_ring;
	struct psp_gfx_rb_frame *ring_buffer_start = ring->ring_mem;
	struct psp_gfx_rb_frame *ring_buffer_end = ring_buffer_start +
		ring->ring_size / sizeof(struct psp_gfx_rb_frame) - 1;
	struct amdgpu_device *adev = psp->adev;
	uint32_t ring_size_dw = ring->ring_size / 4;
	uint32_t rb_frame_size_dw = sizeof(struct psp_gfx_rb_frame) / 4;

	/* KM (GPCOM) prepare write pointer */
	psp_write_ptr_reg = psp_ring_get_wptr(psp);

	/* Update KM RB frame pointer to new frame */
	/* write_frame ptr increments by size of rb_frame in bytes */
	/* psp_write_ptr_reg increments by size of rb_frame in DWORDs */
	if ((psp_write_ptr_reg % ring_size_dw) == 0)
		write_frame = ring_buffer_start;
	else
		write_frame = ring_buffer_start + (psp_write_ptr_reg / rb_frame_size_dw);
	/* Check invalid write_frame ptr address */
	if ((write_frame < ring_buffer_start) || (ring_buffer_end < write_frame)) {
		DRM_ERROR("ring_buffer_start = %p; ring_buffer_end = %p; write_frame = %p\n",
			  ring_buffer_start, ring_buffer_end, write_frame);
		DRM_ERROR("write_frame is pointing to address out of bounds\n");
		return -EINVAL;
	}

	/* Initialize KM RB frame */
	memset(write_frame, 0, sizeof(struct psp_gfx_rb_frame));

	/* Update KM RB frame */
	write_frame->cmd_buf_addr_hi = upper_32_bits(cmd_buf_mc_addr);
	write_frame->cmd_buf_addr_lo = lower_32_bits(cmd_buf_mc_addr);
	write_frame->fence_addr_hi = upper_32_bits(fence_mc_addr);
	write_frame->fence_addr_lo = lower_32_bits(fence_mc_addr);
	write_frame->fence_value = index;
2756
	amdgpu_device_flush_hdp(adev, NULL);
2757 2758 2759 2760 2761 2762 2763

	/* Update the write Pointer in DWORDs */
	psp_write_ptr_reg = (psp_write_ptr_reg + rb_frame_size_dw) % ring_size_dw;
	psp_ring_set_wptr(psp, psp_write_ptr_reg);
	return 0;
}

2764 2765 2766 2767
int psp_init_asd_microcode(struct psp_context *psp,
			   const char *chip_name)
{
	struct amdgpu_device *adev = psp->adev;
2768
	char fw_name[PSP_FW_NAME_LEN];
2769 2770 2771 2772 2773 2774 2775 2776 2777 2778 2779 2780 2781 2782 2783 2784 2785 2786
	const struct psp_firmware_header_v1_0 *asd_hdr;
	int err = 0;

	if (!chip_name) {
		dev_err(adev->dev, "invalid chip name for asd microcode\n");
		return -EINVAL;
	}

	snprintf(fw_name, sizeof(fw_name), "amdgpu/%s_asd.bin", chip_name);
	err = request_firmware(&adev->psp.asd_fw, fw_name, adev->dev);
	if (err)
		goto out;

	err = amdgpu_ucode_validate(adev->psp.asd_fw);
	if (err)
		goto out;

	asd_hdr = (const struct psp_firmware_header_v1_0 *)adev->psp.asd_fw->data;
C
Candice Li 已提交
2787 2788 2789 2790
	adev->psp.asd_context.bin_desc.fw_version = le32_to_cpu(asd_hdr->header.ucode_version);
	adev->psp.asd_context.bin_desc.feature_version = le32_to_cpu(asd_hdr->sos.fw_version);
	adev->psp.asd_context.bin_desc.size_bytes = le32_to_cpu(asd_hdr->header.ucode_size_bytes);
	adev->psp.asd_context.bin_desc.start_addr = (uint8_t *)asd_hdr +
2791 2792 2793 2794 2795 2796 2797 2798 2799
				le32_to_cpu(asd_hdr->header.ucode_array_offset_bytes);
	return 0;
out:
	dev_err(adev->dev, "fail to initialize asd microcode\n");
	release_firmware(adev->psp.asd_fw);
	adev->psp.asd_fw = NULL;
	return err;
}

2800 2801 2802 2803
int psp_init_toc_microcode(struct psp_context *psp,
			   const char *chip_name)
{
	struct amdgpu_device *adev = psp->adev;
2804
	char fw_name[PSP_FW_NAME_LEN];
2805 2806 2807 2808 2809 2810 2811 2812 2813 2814 2815 2816 2817 2818 2819 2820 2821 2822
	const struct psp_firmware_header_v1_0 *toc_hdr;
	int err = 0;

	if (!chip_name) {
		dev_err(adev->dev, "invalid chip name for toc microcode\n");
		return -EINVAL;
	}

	snprintf(fw_name, sizeof(fw_name), "amdgpu/%s_toc.bin", chip_name);
	err = request_firmware(&adev->psp.toc_fw, fw_name, adev->dev);
	if (err)
		goto out;

	err = amdgpu_ucode_validate(adev->psp.toc_fw);
	if (err)
		goto out;

	toc_hdr = (const struct psp_firmware_header_v1_0 *)adev->psp.toc_fw->data;
2823 2824 2825 2826
	adev->psp.toc.fw_version = le32_to_cpu(toc_hdr->header.ucode_version);
	adev->psp.toc.feature_version = le32_to_cpu(toc_hdr->sos.fw_version);
	adev->psp.toc.size_bytes = le32_to_cpu(toc_hdr->header.ucode_size_bytes);
	adev->psp.toc.start_addr = (uint8_t *)toc_hdr +
2827 2828 2829 2830 2831 2832 2833 2834 2835
				le32_to_cpu(toc_hdr->header.ucode_array_offset_bytes);
	return 0;
out:
	dev_err(adev->dev, "fail to request/validate toc microcode\n");
	release_firmware(adev->psp.toc_fw);
	adev->psp.toc_fw = NULL;
	return err;
}

2836 2837 2838 2839 2840 2841 2842 2843 2844 2845 2846 2847 2848 2849 2850 2851 2852 2853 2854 2855 2856 2857 2858 2859 2860 2861 2862 2863 2864 2865 2866 2867 2868 2869 2870 2871 2872 2873 2874 2875 2876 2877 2878 2879 2880 2881 2882 2883 2884 2885
static int parse_sos_bin_descriptor(struct psp_context *psp,
				   const struct psp_fw_bin_desc *desc,
				   const struct psp_firmware_header_v2_0 *sos_hdr)
{
	uint8_t *ucode_start_addr  = NULL;

	if (!psp || !desc || !sos_hdr)
		return -EINVAL;

	ucode_start_addr  = (uint8_t *)sos_hdr +
			    le32_to_cpu(desc->offset_bytes) +
			    le32_to_cpu(sos_hdr->header.ucode_array_offset_bytes);

	switch (desc->fw_type) {
	case PSP_FW_TYPE_PSP_SOS:
		psp->sos.fw_version        = le32_to_cpu(desc->fw_version);
		psp->sos.feature_version   = le32_to_cpu(desc->fw_version);
		psp->sos.size_bytes        = le32_to_cpu(desc->size_bytes);
		psp->sos.start_addr 	   = ucode_start_addr;
		break;
	case PSP_FW_TYPE_PSP_SYS_DRV:
		psp->sys.fw_version        = le32_to_cpu(desc->fw_version);
		psp->sys.feature_version   = le32_to_cpu(desc->fw_version);
		psp->sys.size_bytes        = le32_to_cpu(desc->size_bytes);
		psp->sys.start_addr        = ucode_start_addr;
		break;
	case PSP_FW_TYPE_PSP_KDB:
		psp->kdb.fw_version        = le32_to_cpu(desc->fw_version);
		psp->kdb.feature_version   = le32_to_cpu(desc->fw_version);
		psp->kdb.size_bytes        = le32_to_cpu(desc->size_bytes);
		psp->kdb.start_addr        = ucode_start_addr;
		break;
	case PSP_FW_TYPE_PSP_TOC:
		psp->toc.fw_version        = le32_to_cpu(desc->fw_version);
		psp->toc.feature_version   = le32_to_cpu(desc->fw_version);
		psp->toc.size_bytes        = le32_to_cpu(desc->size_bytes);
		psp->toc.start_addr        = ucode_start_addr;
		break;
	case PSP_FW_TYPE_PSP_SPL:
		psp->spl.fw_version        = le32_to_cpu(desc->fw_version);
		psp->spl.feature_version   = le32_to_cpu(desc->fw_version);
		psp->spl.size_bytes        = le32_to_cpu(desc->size_bytes);
		psp->spl.start_addr        = ucode_start_addr;
		break;
	case PSP_FW_TYPE_PSP_RL:
		psp->rl.fw_version         = le32_to_cpu(desc->fw_version);
		psp->rl.feature_version    = le32_to_cpu(desc->fw_version);
		psp->rl.size_bytes         = le32_to_cpu(desc->size_bytes);
		psp->rl.start_addr         = ucode_start_addr;
		break;
2886 2887 2888 2889 2890 2891 2892 2893 2894 2895 2896 2897 2898 2899 2900 2901 2902 2903
	case PSP_FW_TYPE_PSP_SOC_DRV:
		psp->soc_drv.fw_version         = le32_to_cpu(desc->fw_version);
		psp->soc_drv.feature_version    = le32_to_cpu(desc->fw_version);
		psp->soc_drv.size_bytes         = le32_to_cpu(desc->size_bytes);
		psp->soc_drv.start_addr         = ucode_start_addr;
		break;
	case PSP_FW_TYPE_PSP_INTF_DRV:
		psp->intf_drv.fw_version        = le32_to_cpu(desc->fw_version);
		psp->intf_drv.feature_version   = le32_to_cpu(desc->fw_version);
		psp->intf_drv.size_bytes        = le32_to_cpu(desc->size_bytes);
		psp->intf_drv.start_addr        = ucode_start_addr;
		break;
	case PSP_FW_TYPE_PSP_DBG_DRV:
		psp->dbg_drv.fw_version         = le32_to_cpu(desc->fw_version);
		psp->dbg_drv.feature_version    = le32_to_cpu(desc->fw_version);
		psp->dbg_drv.size_bytes         = le32_to_cpu(desc->size_bytes);
		psp->dbg_drv.start_addr         = ucode_start_addr;
		break;
2904 2905 2906 2907 2908 2909 2910 2911
	default:
		dev_warn(psp->adev->dev, "Unsupported PSP FW type: %d\n", desc->fw_type);
		break;
	}

	return 0;
}

2912 2913 2914 2915
static int psp_init_sos_base_fw(struct amdgpu_device *adev)
{
	const struct psp_firmware_header_v1_0 *sos_hdr;
	const struct psp_firmware_header_v1_3 *sos_hdr_v1_3;
2916
	uint8_t *ucode_array_start_addr;
2917 2918

	sos_hdr = (const struct psp_firmware_header_v1_0 *)adev->psp.sos_fw->data;
2919 2920
	ucode_array_start_addr = (uint8_t *)sos_hdr +
		le32_to_cpu(sos_hdr->header.ucode_array_offset_bytes);
2921

2922
	if (adev->gmc.xgmi.connected_to_cpu ||
2923
	    (adev->ip_versions[MP0_HWIP][0] != IP_VERSION(13, 0, 2))) {
2924 2925
		adev->psp.sos.fw_version = le32_to_cpu(sos_hdr->header.ucode_version);
		adev->psp.sos.feature_version = le32_to_cpu(sos_hdr->sos.fw_version);
2926

2927 2928
		adev->psp.sys.size_bytes = le32_to_cpu(sos_hdr->sos.offset_bytes);
		adev->psp.sys.start_addr = ucode_array_start_addr;
2929

2930 2931
		adev->psp.sos.size_bytes = le32_to_cpu(sos_hdr->sos.size_bytes);
		adev->psp.sos.start_addr = ucode_array_start_addr +
2932
				le32_to_cpu(sos_hdr->sos.offset_bytes);
2933
		adev->psp.xgmi_context.supports_extended_data = false;
2934 2935 2936 2937
	} else {
		/* Load alternate PSP SOS FW */
		sos_hdr_v1_3 = (const struct psp_firmware_header_v1_3 *)adev->psp.sos_fw->data;

2938 2939
		adev->psp.sos.fw_version = le32_to_cpu(sos_hdr_v1_3->sos_aux.fw_version);
		adev->psp.sos.feature_version = le32_to_cpu(sos_hdr_v1_3->sos_aux.fw_version);
2940

2941 2942
		adev->psp.sys.size_bytes = le32_to_cpu(sos_hdr_v1_3->sys_drv_aux.size_bytes);
		adev->psp.sys.start_addr = ucode_array_start_addr +
2943 2944
			le32_to_cpu(sos_hdr_v1_3->sys_drv_aux.offset_bytes);

2945 2946
		adev->psp.sos.size_bytes = le32_to_cpu(sos_hdr_v1_3->sos_aux.size_bytes);
		adev->psp.sos.start_addr = ucode_array_start_addr +
2947
			le32_to_cpu(sos_hdr_v1_3->sos_aux.offset_bytes);
2948
		adev->psp.xgmi_context.supports_extended_data = true;
2949 2950
	}

2951
	if ((adev->psp.sys.size_bytes == 0) || (adev->psp.sos.size_bytes == 0)) {
2952 2953 2954 2955 2956 2957 2958
		dev_warn(adev->dev, "PSP SOS FW not available");
		return -EINVAL;
	}

	return 0;
}

2959 2960 2961 2962
int psp_init_sos_microcode(struct psp_context *psp,
			   const char *chip_name)
{
	struct amdgpu_device *adev = psp->adev;
2963
	char fw_name[PSP_FW_NAME_LEN];
2964 2965 2966
	const struct psp_firmware_header_v1_0 *sos_hdr;
	const struct psp_firmware_header_v1_1 *sos_hdr_v1_1;
	const struct psp_firmware_header_v1_2 *sos_hdr_v1_2;
2967
	const struct psp_firmware_header_v1_3 *sos_hdr_v1_3;
2968
	const struct psp_firmware_header_v2_0 *sos_hdr_v2_0;
2969
	int err = 0;
2970
	uint8_t *ucode_array_start_addr;
2971
	int fw_index = 0;
2972 2973 2974 2975 2976 2977 2978 2979 2980 2981 2982 2983 2984 2985 2986 2987

	if (!chip_name) {
		dev_err(adev->dev, "invalid chip name for sos microcode\n");
		return -EINVAL;
	}

	snprintf(fw_name, sizeof(fw_name), "amdgpu/%s_sos.bin", chip_name);
	err = request_firmware(&adev->psp.sos_fw, fw_name, adev->dev);
	if (err)
		goto out;

	err = amdgpu_ucode_validate(adev->psp.sos_fw);
	if (err)
		goto out;

	sos_hdr = (const struct psp_firmware_header_v1_0 *)adev->psp.sos_fw->data;
2988 2989
	ucode_array_start_addr = (uint8_t *)sos_hdr +
		le32_to_cpu(sos_hdr->header.ucode_array_offset_bytes);
2990 2991 2992 2993
	amdgpu_ucode_print_psp_hdr(&sos_hdr->header);

	switch (sos_hdr->header.header_version_major) {
	case 1:
2994 2995 2996 2997
		err = psp_init_sos_base_fw(adev);
		if (err)
			goto out;

2998 2999
		if (sos_hdr->header.header_version_minor == 1) {
			sos_hdr_v1_1 = (const struct psp_firmware_header_v1_1 *)adev->psp.sos_fw->data;
3000 3001
			adev->psp.toc.size_bytes = le32_to_cpu(sos_hdr_v1_1->toc.size_bytes);
			adev->psp.toc.start_addr = (uint8_t *)adev->psp.sys.start_addr +
3002
					le32_to_cpu(sos_hdr_v1_1->toc.offset_bytes);
3003 3004
			adev->psp.kdb.size_bytes = le32_to_cpu(sos_hdr_v1_1->kdb.size_bytes);
			adev->psp.kdb.start_addr = (uint8_t *)adev->psp.sys.start_addr +
3005
					le32_to_cpu(sos_hdr_v1_1->kdb.offset_bytes);
3006 3007 3008
		}
		if (sos_hdr->header.header_version_minor == 2) {
			sos_hdr_v1_2 = (const struct psp_firmware_header_v1_2 *)adev->psp.sos_fw->data;
3009 3010
			adev->psp.kdb.size_bytes = le32_to_cpu(sos_hdr_v1_2->kdb.size_bytes);
			adev->psp.kdb.start_addr = (uint8_t *)adev->psp.sys.start_addr +
3011
						    le32_to_cpu(sos_hdr_v1_2->kdb.offset_bytes);
3012
		}
3013 3014
		if (sos_hdr->header.header_version_minor == 3) {
			sos_hdr_v1_3 = (const struct psp_firmware_header_v1_3 *)adev->psp.sos_fw->data;
3015 3016
			adev->psp.toc.size_bytes = le32_to_cpu(sos_hdr_v1_3->v1_1.toc.size_bytes);
			adev->psp.toc.start_addr = ucode_array_start_addr +
3017
				le32_to_cpu(sos_hdr_v1_3->v1_1.toc.offset_bytes);
3018 3019
			adev->psp.kdb.size_bytes = le32_to_cpu(sos_hdr_v1_3->v1_1.kdb.size_bytes);
			adev->psp.kdb.start_addr = ucode_array_start_addr +
3020
				le32_to_cpu(sos_hdr_v1_3->v1_1.kdb.offset_bytes);
3021 3022
			adev->psp.spl.size_bytes = le32_to_cpu(sos_hdr_v1_3->spl.size_bytes);
			adev->psp.spl.start_addr = ucode_array_start_addr +
3023
				le32_to_cpu(sos_hdr_v1_3->spl.offset_bytes);
3024 3025
			adev->psp.rl.size_bytes = le32_to_cpu(sos_hdr_v1_3->rl.size_bytes);
			adev->psp.rl.start_addr = ucode_array_start_addr +
3026
				le32_to_cpu(sos_hdr_v1_3->rl.offset_bytes);
3027
		}
3028
		break;
3029 3030 3031 3032 3033 3034 3035 3036 3037 3038 3039 3040 3041 3042 3043 3044 3045
	case 2:
		sos_hdr_v2_0 = (const struct psp_firmware_header_v2_0 *)adev->psp.sos_fw->data;

		if (le32_to_cpu(sos_hdr_v2_0->psp_fw_bin_count) >= UCODE_MAX_PSP_PACKAGING) {
			dev_err(adev->dev, "packed SOS count exceeds maximum limit\n");
			err = -EINVAL;
			goto out;
		}

		for (fw_index = 0; fw_index < le32_to_cpu(sos_hdr_v2_0->psp_fw_bin_count); fw_index++) {
			err = parse_sos_bin_descriptor(psp,
						       &sos_hdr_v2_0->psp_fw_bin[fw_index],
						       sos_hdr_v2_0);
			if (err)
				goto out;
		}
		break;
3046 3047 3048 3049 3050 3051 3052 3053 3054 3055 3056 3057 3058 3059 3060 3061 3062
	default:
		dev_err(adev->dev,
			"unsupported psp sos firmware\n");
		err = -EINVAL;
		goto out;
	}

	return 0;
out:
	dev_err(adev->dev,
		"failed to init sos firmware\n");
	release_firmware(adev->psp.sos_fw);
	adev->psp.sos_fw = NULL;

	return err;
}

3063
static int parse_ta_bin_descriptor(struct psp_context *psp,
3064
				   const struct psp_fw_bin_desc *desc,
3065
				   const struct ta_firmware_header_v2_0 *ta_hdr)
3066 3067 3068 3069 3070 3071
{
	uint8_t *ucode_start_addr  = NULL;

	if (!psp || !desc || !ta_hdr)
		return -EINVAL;

3072 3073 3074
	ucode_start_addr  = (uint8_t *)ta_hdr +
			    le32_to_cpu(desc->offset_bytes) +
			    le32_to_cpu(ta_hdr->header.ucode_array_offset_bytes);
3075 3076 3077

	switch (desc->fw_type) {
	case TA_FW_TYPE_PSP_ASD:
C
Candice Li 已提交
3078 3079 3080 3081
		psp->asd_context.bin_desc.fw_version        = le32_to_cpu(desc->fw_version);
		psp->asd_context.bin_desc.feature_version   = le32_to_cpu(desc->fw_version);
		psp->asd_context.bin_desc.size_bytes        = le32_to_cpu(desc->size_bytes);
		psp->asd_context.bin_desc.start_addr        = ucode_start_addr;
3082 3083
		break;
	case TA_FW_TYPE_PSP_XGMI:
C
Candice Li 已提交
3084 3085 3086
		psp->xgmi_context.context.bin_desc.feature_version  = le32_to_cpu(desc->fw_version);
		psp->xgmi_context.context.bin_desc.size_bytes       = le32_to_cpu(desc->size_bytes);
		psp->xgmi_context.context.bin_desc.start_addr       = ucode_start_addr;
3087 3088
		break;
	case TA_FW_TYPE_PSP_RAS:
C
Candice Li 已提交
3089 3090 3091
		psp->ras_context.context.bin_desc.feature_version   = le32_to_cpu(desc->fw_version);
		psp->ras_context.context.bin_desc.size_bytes        = le32_to_cpu(desc->size_bytes);
		psp->ras_context.context.bin_desc.start_addr        = ucode_start_addr;
3092 3093
		break;
	case TA_FW_TYPE_PSP_HDCP:
C
Candice Li 已提交
3094 3095 3096
		psp->hdcp_context.context.bin_desc.feature_version  = le32_to_cpu(desc->fw_version);
		psp->hdcp_context.context.bin_desc.size_bytes       = le32_to_cpu(desc->size_bytes);
		psp->hdcp_context.context.bin_desc.start_addr       = ucode_start_addr;
3097 3098
		break;
	case TA_FW_TYPE_PSP_DTM:
C
Candice Li 已提交
3099 3100 3101
		psp->dtm_context.context.bin_desc.feature_version  = le32_to_cpu(desc->fw_version);
		psp->dtm_context.context.bin_desc.size_bytes       = le32_to_cpu(desc->size_bytes);
		psp->dtm_context.context.bin_desc.start_addr       = ucode_start_addr;
3102
		break;
W
Wenhui Sheng 已提交
3103
	case TA_FW_TYPE_PSP_RAP:
C
Candice Li 已提交
3104 3105 3106
		psp->rap_context.context.bin_desc.feature_version  = le32_to_cpu(desc->fw_version);
		psp->rap_context.context.bin_desc.size_bytes       = le32_to_cpu(desc->size_bytes);
		psp->rap_context.context.bin_desc.start_addr       = ucode_start_addr;
W
Wenhui Sheng 已提交
3107
		break;
3108
	case TA_FW_TYPE_PSP_SECUREDISPLAY:
C
Candice Li 已提交
3109 3110 3111 3112 3113 3114
		psp->securedisplay_context.context.bin_desc.feature_version =
			le32_to_cpu(desc->fw_version);
		psp->securedisplay_context.context.bin_desc.size_bytes =
			le32_to_cpu(desc->size_bytes);
		psp->securedisplay_context.context.bin_desc.start_addr =
			ucode_start_addr;
3115
		break;
3116 3117 3118 3119 3120 3121 3122 3123 3124 3125 3126 3127
	default:
		dev_warn(psp->adev->dev, "Unsupported TA type: %d\n", desc->fw_type);
		break;
	}

	return 0;
}

int psp_init_ta_microcode(struct psp_context *psp,
			  const char *chip_name)
{
	struct amdgpu_device *adev = psp->adev;
3128
	char fw_name[PSP_FW_NAME_LEN];
3129 3130 3131 3132 3133 3134 3135 3136 3137 3138 3139 3140 3141 3142 3143 3144 3145 3146 3147 3148 3149 3150 3151 3152 3153 3154
	const struct ta_firmware_header_v2_0 *ta_hdr;
	int err = 0;
	int ta_index = 0;

	if (!chip_name) {
		dev_err(adev->dev, "invalid chip name for ta microcode\n");
		return -EINVAL;
	}

	snprintf(fw_name, sizeof(fw_name), "amdgpu/%s_ta.bin", chip_name);
	err = request_firmware(&adev->psp.ta_fw, fw_name, adev->dev);
	if (err)
		goto out;

	err = amdgpu_ucode_validate(adev->psp.ta_fw);
	if (err)
		goto out;

	ta_hdr = (const struct ta_firmware_header_v2_0 *)adev->psp.ta_fw->data;

	if (le16_to_cpu(ta_hdr->header.header_version_major) != 2) {
		dev_err(adev->dev, "unsupported TA header version\n");
		err = -EINVAL;
		goto out;
	}

3155
	if (le32_to_cpu(ta_hdr->ta_fw_bin_count) >= UCODE_MAX_PSP_PACKAGING) {
3156 3157 3158 3159 3160 3161 3162 3163 3164 3165 3166 3167 3168 3169 3170 3171 3172 3173 3174 3175 3176
		dev_err(adev->dev, "packed TA count exceeds maximum limit\n");
		err = -EINVAL;
		goto out;
	}

	for (ta_index = 0; ta_index < le32_to_cpu(ta_hdr->ta_fw_bin_count); ta_index++) {
		err = parse_ta_bin_descriptor(psp,
					      &ta_hdr->ta_fw_bin[ta_index],
					      ta_hdr);
		if (err)
			goto out;
	}

	return 0;
out:
	dev_err(adev->dev, "fail to initialize ta microcode\n");
	release_firmware(adev->psp.ta_fw);
	adev->psp.ta_fw = NULL;
	return err;
}

3177 3178 3179 3180 3181 3182 3183 3184 3185 3186 3187 3188
static int psp_set_clockgating_state(void *handle,
				     enum amd_clockgating_state state)
{
	return 0;
}

static int psp_set_powergating_state(void *handle,
				     enum amd_powergating_state state)
{
	return 0;
}

3189 3190 3191 3192 3193
static ssize_t psp_usbc_pd_fw_sysfs_read(struct device *dev,
					 struct device_attribute *attr,
					 char *buf)
{
	struct drm_device *ddev = dev_get_drvdata(dev);
3194
	struct amdgpu_device *adev = drm_to_adev(ddev);
3195 3196 3197
	uint32_t fw_ver;
	int ret;

3198 3199 3200 3201 3202
	if (!adev->ip_blocks[AMD_IP_BLOCK_TYPE_PSP].status.late_initialized) {
		DRM_INFO("PSP block is not ready yet.");
		return -EBUSY;
	}

3203 3204 3205 3206 3207 3208 3209 3210 3211
	mutex_lock(&adev->psp.mutex);
	ret = psp_read_usbc_pd_fw(&adev->psp, &fw_ver);
	mutex_unlock(&adev->psp.mutex);

	if (ret) {
		DRM_ERROR("Failed to read USBC PD FW, err = %d", ret);
		return ret;
	}

3212
	return sysfs_emit(buf, "%x\n", fw_ver);
3213 3214 3215 3216 3217 3218 3219 3220
}

static ssize_t psp_usbc_pd_fw_sysfs_write(struct device *dev,
						       struct device_attribute *attr,
						       const char *buf,
						       size_t count)
{
	struct drm_device *ddev = dev_get_drvdata(dev);
3221
	struct amdgpu_device *adev = drm_to_adev(ddev);
3222
	int ret, idx;
3223 3224
	char fw_name[100];
	const struct firmware *usbc_pd_fw;
3225 3226 3227
	struct amdgpu_bo *fw_buf_bo = NULL;
	uint64_t fw_pri_mc_addr;
	void *fw_pri_cpu_addr;
3228

3229 3230 3231 3232
	if (!adev->ip_blocks[AMD_IP_BLOCK_TYPE_PSP].status.late_initialized) {
		DRM_INFO("PSP block is not ready yet.");
		return -EBUSY;
	}
3233

3234 3235 3236
	if (!drm_dev_enter(ddev, &idx))
		return -ENODEV;

3237 3238 3239 3240 3241
	snprintf(fw_name, sizeof(fw_name), "amdgpu/%s", buf);
	ret = request_firmware(&usbc_pd_fw, fw_name, adev->dev);
	if (ret)
		goto fail;

3242 3243 3244 3245 3246 3247
	/* LFB address which is aligned to 1MB boundary per PSP request */
	ret = amdgpu_bo_create_kernel(adev, usbc_pd_fw->size, 0x100000,
						AMDGPU_GEM_DOMAIN_VRAM,
						&fw_buf_bo,
						&fw_pri_mc_addr,
						&fw_pri_cpu_addr);
3248 3249 3250
	if (ret)
		goto rel_buf;

3251
	memcpy_toio(fw_pri_cpu_addr, usbc_pd_fw->data, usbc_pd_fw->size);
3252 3253

	mutex_lock(&adev->psp.mutex);
3254
	ret = psp_load_usbc_pd_fw(&adev->psp, fw_pri_mc_addr);
3255 3256
	mutex_unlock(&adev->psp.mutex);

3257 3258
	amdgpu_bo_free_kernel(&fw_buf_bo, &fw_pri_mc_addr, &fw_pri_cpu_addr);

3259 3260 3261 3262 3263
rel_buf:
	release_firmware(usbc_pd_fw);
fail:
	if (ret) {
		DRM_ERROR("Failed to load USBC PD FW, err = %d", ret);
3264
		count = ret;
3265 3266
	}

3267
	drm_dev_exit(idx);
3268 3269 3270
	return count;
}

3271 3272 3273 3274
void psp_copy_fw(struct psp_context *psp, uint8_t *start_addr, uint32_t bin_size)
{
	int idx;

3275
	if (!drm_dev_enter(adev_to_drm(psp->adev), &idx))
3276 3277 3278 3279 3280 3281 3282 3283
		return;

	memset(psp->fw_pri_buf, 0, PSP_1_MEG);
	memcpy(psp->fw_pri_buf, start_addr, bin_size);

	drm_dev_exit(idx);
}

3284 3285 3286 3287
static DEVICE_ATTR(usbc_pd_fw, S_IRUGO | S_IWUSR,
		   psp_usbc_pd_fw_sysfs_read,
		   psp_usbc_pd_fw_sysfs_write);

3288 3289 3290 3291
int is_psp_fw_valid(struct psp_bin_desc bin)
{
	return bin.size_bytes;
}
3292

3293 3294 3295
const struct amd_ip_funcs psp_ip_funcs = {
	.name = "psp",
	.early_init = psp_early_init,
3296
	.late_init = NULL,
3297 3298 3299 3300 3301 3302 3303
	.sw_init = psp_sw_init,
	.sw_fini = psp_sw_fini,
	.hw_init = psp_hw_init,
	.hw_fini = psp_hw_fini,
	.suspend = psp_suspend,
	.resume = psp_resume,
	.is_idle = NULL,
3304
	.check_soft_reset = NULL,
3305
	.wait_for_idle = NULL,
3306
	.soft_reset = NULL,
3307 3308 3309 3310
	.set_clockgating_state = psp_set_clockgating_state,
	.set_powergating_state = psp_set_powergating_state,
};

3311 3312 3313 3314 3315 3316 3317 3318 3319 3320 3321 3322 3323 3324 3325
static int psp_sysfs_init(struct amdgpu_device *adev)
{
	int ret = device_create_file(adev->dev, &dev_attr_usbc_pd_fw);

	if (ret)
		DRM_ERROR("Failed to create USBC PD FW control file!");

	return ret;
}

static void psp_sysfs_fini(struct amdgpu_device *adev)
{
	device_remove_file(adev->dev, &dev_attr_usbc_pd_fw);
}

3326 3327 3328 3329 3330 3331 3332 3333
const struct amdgpu_ip_block_version psp_v3_1_ip_block =
{
	.type = AMD_IP_BLOCK_TYPE_PSP,
	.major = 3,
	.minor = 1,
	.rev = 0,
	.funcs = &psp_ip_funcs,
};
H
Huang Rui 已提交
3334 3335 3336 3337 3338 3339 3340 3341 3342

const struct amdgpu_ip_block_version psp_v10_0_ip_block =
{
	.type = AMD_IP_BLOCK_TYPE_PSP,
	.major = 10,
	.minor = 0,
	.rev = 0,
	.funcs = &psp_ip_funcs,
};
3343 3344 3345 3346 3347 3348 3349 3350 3351

const struct amdgpu_ip_block_version psp_v11_0_ip_block =
{
	.type = AMD_IP_BLOCK_TYPE_PSP,
	.major = 11,
	.minor = 0,
	.rev = 0,
	.funcs = &psp_ip_funcs,
};
3352

3353 3354 3355 3356 3357 3358 3359 3360
const struct amdgpu_ip_block_version psp_v11_0_8_ip_block = {
	.type = AMD_IP_BLOCK_TYPE_PSP,
	.major = 11,
	.minor = 0,
	.rev = 8,
	.funcs = &psp_ip_funcs,
};

3361 3362 3363 3364 3365 3366 3367 3368
const struct amdgpu_ip_block_version psp_v12_0_ip_block =
{
	.type = AMD_IP_BLOCK_TYPE_PSP,
	.major = 12,
	.minor = 0,
	.rev = 0,
	.funcs = &psp_ip_funcs,
};
3369 3370 3371 3372 3373 3374 3375 3376

const struct amdgpu_ip_block_version psp_v13_0_ip_block = {
	.type = AMD_IP_BLOCK_TYPE_PSP,
	.major = 13,
	.minor = 0,
	.rev = 0,
	.funcs = &psp_ip_funcs,
};