xref: /linux/drivers/gpu/drm/xe/xe_guc_klv_helpers.c (revision 946661e3bef8efa11ba8079d4ebafe6fc3b0aaad)
10ddc1e07SMichal Wajdeczko // SPDX-License-Identifier: MIT
20ddc1e07SMichal Wajdeczko /*
30ddc1e07SMichal Wajdeczko  * Copyright © 2024 Intel Corporation
40ddc1e07SMichal Wajdeczko  */
50ddc1e07SMichal Wajdeczko 
60ddc1e07SMichal Wajdeczko #include <linux/bitfield.h>
70ddc1e07SMichal Wajdeczko #include <drm/drm_print.h>
80ddc1e07SMichal Wajdeczko 
90ddc1e07SMichal Wajdeczko #include "abi/guc_klvs_abi.h"
100ddc1e07SMichal Wajdeczko #include "xe_guc_klv_helpers.h"
117aefee83SMichal Wajdeczko #include "xe_guc_klv_thresholds_set.h"
120ddc1e07SMichal Wajdeczko 
130ddc1e07SMichal Wajdeczko #define make_u64(hi, lo) ((u64)((u64)(u32)(hi) << 32 | (u32)(lo)))
140ddc1e07SMichal Wajdeczko 
150ddc1e07SMichal Wajdeczko /**
160ddc1e07SMichal Wajdeczko  * xe_guc_klv_key_to_string - Convert KLV key into friendly name.
170ddc1e07SMichal Wajdeczko  * @key: the `GuC KLV`_ key
180ddc1e07SMichal Wajdeczko  *
190ddc1e07SMichal Wajdeczko  * Return: name of the KLV key.
200ddc1e07SMichal Wajdeczko  */
xe_guc_klv_key_to_string(u16 key)210ddc1e07SMichal Wajdeczko const char *xe_guc_klv_key_to_string(u16 key)
220ddc1e07SMichal Wajdeczko {
230ddc1e07SMichal Wajdeczko 	switch (key) {
240ddc1e07SMichal Wajdeczko 	/* VGT POLICY keys */
250ddc1e07SMichal Wajdeczko 	case GUC_KLV_VGT_POLICY_SCHED_IF_IDLE_KEY:
260ddc1e07SMichal Wajdeczko 		return "sched_if_idle";
270ddc1e07SMichal Wajdeczko 	case GUC_KLV_VGT_POLICY_ADVERSE_SAMPLE_PERIOD_KEY:
280ddc1e07SMichal Wajdeczko 		return "sample_period";
290ddc1e07SMichal Wajdeczko 	case GUC_KLV_VGT_POLICY_RESET_AFTER_VF_SWITCH_KEY:
300ddc1e07SMichal Wajdeczko 		return "reset_engine";
310ddc1e07SMichal Wajdeczko 	/* VF CFG keys */
320ddc1e07SMichal Wajdeczko 	case GUC_KLV_VF_CFG_GGTT_START_KEY:
330ddc1e07SMichal Wajdeczko 		return "ggtt_start";
340ddc1e07SMichal Wajdeczko 	case GUC_KLV_VF_CFG_GGTT_SIZE_KEY:
350ddc1e07SMichal Wajdeczko 		return "ggtt_size";
360ddc1e07SMichal Wajdeczko 	case GUC_KLV_VF_CFG_LMEM_SIZE_KEY:
370ddc1e07SMichal Wajdeczko 		return "lmem_size";
380ddc1e07SMichal Wajdeczko 	case GUC_KLV_VF_CFG_NUM_CONTEXTS_KEY:
390ddc1e07SMichal Wajdeczko 		return "num_contexts";
400ddc1e07SMichal Wajdeczko 	case GUC_KLV_VF_CFG_TILE_MASK_KEY:
410ddc1e07SMichal Wajdeczko 		return "tile_mask";
420ddc1e07SMichal Wajdeczko 	case GUC_KLV_VF_CFG_NUM_DOORBELLS_KEY:
430ddc1e07SMichal Wajdeczko 		return "num_doorbells";
440ddc1e07SMichal Wajdeczko 	case GUC_KLV_VF_CFG_EXEC_QUANTUM_KEY:
450ddc1e07SMichal Wajdeczko 		return "exec_quantum";
460ddc1e07SMichal Wajdeczko 	case GUC_KLV_VF_CFG_PREEMPT_TIMEOUT_KEY:
470ddc1e07SMichal Wajdeczko 		return "preempt_timeout";
480ddc1e07SMichal Wajdeczko 	case GUC_KLV_VF_CFG_BEGIN_DOORBELL_ID_KEY:
490ddc1e07SMichal Wajdeczko 		return "begin_db_id";
500ddc1e07SMichal Wajdeczko 	case GUC_KLV_VF_CFG_BEGIN_CONTEXT_ID_KEY:
510ddc1e07SMichal Wajdeczko 		return "begin_ctx_id";
52*5a814e3aSMichal Wajdeczko 	case GUC_KLV_VF_CFG_SCHED_PRIORITY_KEY:
53*5a814e3aSMichal Wajdeczko 		return "sched_priority";
547aefee83SMichal Wajdeczko 
557aefee83SMichal Wajdeczko 	/* VF CFG threshold keys */
567aefee83SMichal Wajdeczko #define define_threshold_key_to_string_case(TAG, NAME, ...)	\
577aefee83SMichal Wajdeczko 								\
587aefee83SMichal Wajdeczko 	case MAKE_GUC_KLV_VF_CFG_THRESHOLD_KEY(TAG):		\
597aefee83SMichal Wajdeczko 		return #NAME;
607aefee83SMichal Wajdeczko 
617aefee83SMichal Wajdeczko 	/* private: auto-generated case statements */
627aefee83SMichal Wajdeczko 	MAKE_XE_GUC_KLV_THRESHOLDS_SET(define_threshold_key_to_string_case)
637aefee83SMichal Wajdeczko #undef define_threshold_key_to_string_case
647aefee83SMichal Wajdeczko 
650ddc1e07SMichal Wajdeczko 	default:
660ddc1e07SMichal Wajdeczko 		return "(unknown)";
670ddc1e07SMichal Wajdeczko 	}
680ddc1e07SMichal Wajdeczko }
690ddc1e07SMichal Wajdeczko 
700ddc1e07SMichal Wajdeczko /**
710ddc1e07SMichal Wajdeczko  * xe_guc_klv_print - Print content of the buffer with `GuC KLV`_.
720ddc1e07SMichal Wajdeczko  * @klvs: the buffer with KLVs
730ddc1e07SMichal Wajdeczko  * @num_dwords: number of dwords (u32) available in the buffer
740ddc1e07SMichal Wajdeczko  * @p: the &drm_printer
750ddc1e07SMichal Wajdeczko  *
760ddc1e07SMichal Wajdeczko  * The buffer may contain more than one KLV.
770ddc1e07SMichal Wajdeczko  */
xe_guc_klv_print(const u32 * klvs,u32 num_dwords,struct drm_printer * p)780ddc1e07SMichal Wajdeczko void xe_guc_klv_print(const u32 *klvs, u32 num_dwords, struct drm_printer *p)
790ddc1e07SMichal Wajdeczko {
800ddc1e07SMichal Wajdeczko 	while (num_dwords >= GUC_KLV_LEN_MIN) {
810ddc1e07SMichal Wajdeczko 		u32 key = FIELD_GET(GUC_KLV_0_KEY, klvs[0]);
820ddc1e07SMichal Wajdeczko 		u32 len = FIELD_GET(GUC_KLV_0_LEN, klvs[0]);
830ddc1e07SMichal Wajdeczko 
840ddc1e07SMichal Wajdeczko 		klvs += GUC_KLV_LEN_MIN;
850ddc1e07SMichal Wajdeczko 		num_dwords -= GUC_KLV_LEN_MIN;
860ddc1e07SMichal Wajdeczko 
870ddc1e07SMichal Wajdeczko 		if (num_dwords < len) {
880ddc1e07SMichal Wajdeczko 			drm_printf(p, "{ key %#06x : truncated %zu of %zu bytes %*ph } # %s\n",
890ddc1e07SMichal Wajdeczko 				   key, num_dwords * sizeof(u32), len * sizeof(u32),
900ddc1e07SMichal Wajdeczko 				   (int)(num_dwords * sizeof(u32)), klvs,
910ddc1e07SMichal Wajdeczko 				   xe_guc_klv_key_to_string(key));
920ddc1e07SMichal Wajdeczko 			return;
930ddc1e07SMichal Wajdeczko 		}
940ddc1e07SMichal Wajdeczko 
950ddc1e07SMichal Wajdeczko 		switch (len) {
960ddc1e07SMichal Wajdeczko 		case 0:
970ddc1e07SMichal Wajdeczko 			drm_printf(p, "{ key %#06x : no value } # %s\n",
980ddc1e07SMichal Wajdeczko 				   key, xe_guc_klv_key_to_string(key));
990ddc1e07SMichal Wajdeczko 			break;
1000ddc1e07SMichal Wajdeczko 		case 1:
1010ddc1e07SMichal Wajdeczko 			drm_printf(p, "{ key %#06x : 32b value %u } # %s\n",
1020ddc1e07SMichal Wajdeczko 				   key, klvs[0], xe_guc_klv_key_to_string(key));
1030ddc1e07SMichal Wajdeczko 			break;
1040ddc1e07SMichal Wajdeczko 		case 2:
1050ddc1e07SMichal Wajdeczko 			drm_printf(p, "{ key %#06x : 64b value %#llx } # %s\n",
1060ddc1e07SMichal Wajdeczko 				   key, make_u64(klvs[1], klvs[0]),
1070ddc1e07SMichal Wajdeczko 				   xe_guc_klv_key_to_string(key));
1080ddc1e07SMichal Wajdeczko 			break;
1090ddc1e07SMichal Wajdeczko 		default:
1100ddc1e07SMichal Wajdeczko 			drm_printf(p, "{ key %#06x : %zu bytes %*ph } # %s\n",
1110ddc1e07SMichal Wajdeczko 				   key, len * sizeof(u32), (int)(len * sizeof(u32)),
1120ddc1e07SMichal Wajdeczko 				   klvs, xe_guc_klv_key_to_string(key));
1130ddc1e07SMichal Wajdeczko 			break;
1140ddc1e07SMichal Wajdeczko 		}
1150ddc1e07SMichal Wajdeczko 
1160ddc1e07SMichal Wajdeczko 		klvs += len;
1170ddc1e07SMichal Wajdeczko 		num_dwords -= len;
1180ddc1e07SMichal Wajdeczko 	}
1190ddc1e07SMichal Wajdeczko 
1200ddc1e07SMichal Wajdeczko 	/* we don't expect any leftovers, fix if KLV header is ever changed */
1210ddc1e07SMichal Wajdeczko 	BUILD_BUG_ON(GUC_KLV_LEN_MIN > 1);
1220ddc1e07SMichal Wajdeczko }
1230ddc1e07SMichal Wajdeczko 
1240ddc1e07SMichal Wajdeczko /**
1250ddc1e07SMichal Wajdeczko  * xe_guc_klv_count - Count KLVs present in the buffer.
1260ddc1e07SMichal Wajdeczko  * @klvs: the buffer with KLVs
1270ddc1e07SMichal Wajdeczko  * @num_dwords: number of dwords (u32) in the buffer
1280ddc1e07SMichal Wajdeczko  *
1290ddc1e07SMichal Wajdeczko  * Return: number of recognized KLVs or
1300ddc1e07SMichal Wajdeczko  *         a negative error code if KLV buffer is truncated.
1310ddc1e07SMichal Wajdeczko  */
xe_guc_klv_count(const u32 * klvs,u32 num_dwords)1320ddc1e07SMichal Wajdeczko int xe_guc_klv_count(const u32 *klvs, u32 num_dwords)
1330ddc1e07SMichal Wajdeczko {
1340ddc1e07SMichal Wajdeczko 	int num_klvs = 0;
1350ddc1e07SMichal Wajdeczko 
1360ddc1e07SMichal Wajdeczko 	while (num_dwords >= GUC_KLV_LEN_MIN) {
1370ddc1e07SMichal Wajdeczko 		u32 len = FIELD_GET(GUC_KLV_0_LEN, klvs[0]);
1380ddc1e07SMichal Wajdeczko 
1390ddc1e07SMichal Wajdeczko 		if (num_dwords < len + GUC_KLV_LEN_MIN)
1400ddc1e07SMichal Wajdeczko 			break;
1410ddc1e07SMichal Wajdeczko 
1420ddc1e07SMichal Wajdeczko 		klvs += GUC_KLV_LEN_MIN + len;
1430ddc1e07SMichal Wajdeczko 		num_dwords -= GUC_KLV_LEN_MIN + len;
1440ddc1e07SMichal Wajdeczko 		num_klvs++;
1450ddc1e07SMichal Wajdeczko 	}
1460ddc1e07SMichal Wajdeczko 
1470ddc1e07SMichal Wajdeczko 	return num_dwords ? -ENODATA : num_klvs;
1480ddc1e07SMichal Wajdeczko }
149