lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-Id: <20251202050844.2520762-4-yilun.xu@linux.intel.com>
Date: Tue,  2 Dec 2025 13:08:41 +0800
From: Xu Yilun <yilun.xu@...ux.intel.com>
To: x86@...nel.org,
	dave.hansen@...ux.intel.com,
	kas@...nel.org,
	linux-kernel@...r.kernel.org
Cc: chao.gao@...el.com,
	rick.p.edgecombe@...el.com,
	dan.j.williams@...el.com,
	baolu.lu@...ux.intel.com,
	yilun.xu@...ux.intel.com,
	yilun.xu@...el.com,
	zhenzhong.duan@...el.com,
	kvm@...r.kernel.org,
	adrian.hunter@...el.com
Subject: [PATCH 3/6] x86/virt/tdx: Refactor metadata reading with a clearer for loop

Replace confusing "ret = ret ?: " code pattern with a normal for loop.

The existing code pattern is compact and friendly to auto-generation.
However, hiding the stop-on-failure logic within the sequential
execution pattern reduces readability. Revive the field mapping table
which lists the field_ids to read and where to store each readout value.
Iterate the table with a normal for loop.

For now this metadata reading process doesn't work well with array typed
fields. Use dedicated method to read these 2 array typed fields. Will
improve in later patches.

Cc: Kai Huang <kai.huang@...el.com>
Signed-off-by: Xu Yilun <yilun.xu@...ux.intel.com>
---
 arch/x86/virt/vmx/tdx/tdx_global_metadata.c | 172 ++++++++++----------
 1 file changed, 86 insertions(+), 86 deletions(-)

diff --git a/arch/x86/virt/vmx/tdx/tdx_global_metadata.c b/arch/x86/virt/vmx/tdx/tdx_global_metadata.c
index 0dfb3a9995fe..3db87c4accd6 100644
--- a/arch/x86/virt/vmx/tdx/tdx_global_metadata.c
+++ b/arch/x86/virt/vmx/tdx/tdx_global_metadata.c
@@ -7,8 +7,50 @@
  * Include this file to other C file instead.
  */
 
-static int read_sys_metadata_field(u64 field_id, u64 *data)
+struct field_mapping {
+	u64 field_id;
+	int offset;
+	int size;
+};
+
+#define TD_SYSINFO_MAP(_field_id, _member)				\
+	{ .field_id = _field_id,					\
+	  .offset = offsetof(struct tdx_sys_info, _member),		\
+	  .size = sizeof_field(struct tdx_sys_info, _member) }
+
+/* Map TD_SYSINFO fields into 'struct tdx_sys_info': */
+static const struct field_mapping mappings[] = {
+	TD_SYSINFO_MAP(0x0A00000300000008, features.tdx_features0),
+
+	TD_SYSINFO_MAP(0x9100000100000008, tdmr.max_tdmrs),
+	TD_SYSINFO_MAP(0x9100000100000009, tdmr.max_reserved_per_tdmr),
+	TD_SYSINFO_MAP(0x9100000100000010, tdmr.pamt_4k_entry_size),
+	TD_SYSINFO_MAP(0x9100000100000011, tdmr.pamt_2m_entry_size),
+	TD_SYSINFO_MAP(0x9100000100000012, tdmr.pamt_1g_entry_size),
+
+	TD_SYSINFO_MAP(0x9800000100000000, td_ctrl.tdr_base_size),
+	TD_SYSINFO_MAP(0x9800000100000100, td_ctrl.tdcs_base_size),
+	TD_SYSINFO_MAP(0x9800000100000200, td_ctrl.tdvps_base_size),
+
+	TD_SYSINFO_MAP(0x1900000300000000, td_conf.attributes_fixed0),
+	TD_SYSINFO_MAP(0x1900000300000001, td_conf.attributes_fixed1),
+	TD_SYSINFO_MAP(0x1900000300000002, td_conf.xfam_fixed0),
+	TD_SYSINFO_MAP(0x1900000300000003, td_conf.xfam_fixed1),
+	TD_SYSINFO_MAP(0x9900000100000004, td_conf.num_cpuid_config),
+	TD_SYSINFO_MAP(0x9900000100000008, td_conf.max_vcpus_per_td),
+};
+
+/* Populate the following fields in special manner, separate them out. */
+static const struct field_mapping cpuid_config_leaves =
+	TD_SYSINFO_MAP(0x9900000300000400, td_conf.cpuid_config_leaves[0]);
+
+static const struct field_mapping cpuid_config_values =
+	TD_SYSINFO_MAP(0x9900000300000500, td_conf.cpuid_config_values[0][0]);
+
+static int read_sys_metadata_field(u64 field_id, int offset, int size,
+				   struct tdx_sys_info *ts)
 {
+	void *field = ((void *)ts) + offset;
 	struct tdx_module_args args = {};
 	int ret;
 
@@ -22,97 +64,55 @@ static int read_sys_metadata_field(u64 field_id, u64 *data)
 	if (ret)
 		return ret;
 
-	*data = args.r8;
+	memcpy(field, &args.r8, size);
 
 	return 0;
 }
 
-static int get_tdx_sys_info_features(struct tdx_sys_info_features *sysinfo_features)
-{
-	int ret = 0;
-	u64 val;
-
-	if (!ret && !(ret = read_sys_metadata_field(0x0A00000300000008, &val)))
-		sysinfo_features->tdx_features0 = val;
-
-	return ret;
-}
-
-static int get_tdx_sys_info_tdmr(struct tdx_sys_info_tdmr *sysinfo_tdmr)
-{
-	int ret = 0;
-	u64 val;
-
-	if (!ret && !(ret = read_sys_metadata_field(0x9100000100000008, &val)))
-		sysinfo_tdmr->max_tdmrs = val;
-	if (!ret && !(ret = read_sys_metadata_field(0x9100000100000009, &val)))
-		sysinfo_tdmr->max_reserved_per_tdmr = val;
-	if (!ret && !(ret = read_sys_metadata_field(0x9100000100000010, &val)))
-		sysinfo_tdmr->pamt_4k_entry_size = val;
-	if (!ret && !(ret = read_sys_metadata_field(0x9100000100000011, &val)))
-		sysinfo_tdmr->pamt_2m_entry_size = val;
-	if (!ret && !(ret = read_sys_metadata_field(0x9100000100000012, &val)))
-		sysinfo_tdmr->pamt_1g_entry_size = val;
-
-	return ret;
-}
-
-static int get_tdx_sys_info_td_ctrl(struct tdx_sys_info_td_ctrl *sysinfo_td_ctrl)
-{
-	int ret = 0;
-	u64 val;
-
-	if (!ret && !(ret = read_sys_metadata_field(0x9800000100000000, &val)))
-		sysinfo_td_ctrl->tdr_base_size = val;
-	if (!ret && !(ret = read_sys_metadata_field(0x9800000100000100, &val)))
-		sysinfo_td_ctrl->tdcs_base_size = val;
-	if (!ret && !(ret = read_sys_metadata_field(0x9800000100000200, &val)))
-		sysinfo_td_ctrl->tdvps_base_size = val;
-
-	return ret;
-}
-
-static int get_tdx_sys_info_td_conf(struct tdx_sys_info_td_conf *sysinfo_td_conf)
-{
-	int ret = 0;
-	u64 val;
-	int i, j;
-
-	if (!ret && !(ret = read_sys_metadata_field(0x1900000300000000, &val)))
-		sysinfo_td_conf->attributes_fixed0 = val;
-	if (!ret && !(ret = read_sys_metadata_field(0x1900000300000001, &val)))
-		sysinfo_td_conf->attributes_fixed1 = val;
-	if (!ret && !(ret = read_sys_metadata_field(0x1900000300000002, &val)))
-		sysinfo_td_conf->xfam_fixed0 = val;
-	if (!ret && !(ret = read_sys_metadata_field(0x1900000300000003, &val)))
-		sysinfo_td_conf->xfam_fixed1 = val;
-	if (!ret && !(ret = read_sys_metadata_field(0x9900000100000004, &val)))
-		sysinfo_td_conf->num_cpuid_config = val;
-	if (!ret && !(ret = read_sys_metadata_field(0x9900000100000008, &val)))
-		sysinfo_td_conf->max_vcpus_per_td = val;
-	if (sysinfo_td_conf->num_cpuid_config > ARRAY_SIZE(sysinfo_td_conf->cpuid_config_leaves))
-		return -EINVAL;
-	for (i = 0; i < sysinfo_td_conf->num_cpuid_config; i++)
-		if (!ret && !(ret = read_sys_metadata_field(0x9900000300000400 + i, &val)))
-			sysinfo_td_conf->cpuid_config_leaves[i] = val;
-	if (sysinfo_td_conf->num_cpuid_config > ARRAY_SIZE(sysinfo_td_conf->cpuid_config_values))
-		return -EINVAL;
-	for (i = 0; i < sysinfo_td_conf->num_cpuid_config; i++)
-		for (j = 0; j < 2; j++)
-			if (!ret && !(ret = read_sys_metadata_field(0x9900000300000500 + i * 2 + j, &val)))
-				sysinfo_td_conf->cpuid_config_values[i][j] = val;
-
-	return ret;
-}
-
 static int get_tdx_sys_info(struct tdx_sys_info *sysinfo)
 {
-	int ret = 0;
+	struct tdx_sys_info_td_conf *td_conf = &sysinfo->td_conf;
+	int ret, i;
+
+	/* Populate 'tdx_sys_info' fields using the mapping structure above: */
+	for (i = 0; i < ARRAY_SIZE(mappings); i++) {
+		ret = read_sys_metadata_field(mappings[i].field_id,
+					      mappings[i].offset,
+					      mappings[i].size,
+					      sysinfo);
+		if (ret)
+			return ret;
+	}
+
+	if (td_conf->num_cpuid_config > ARRAY_SIZE(td_conf->cpuid_config_leaves) ||
+	    td_conf->num_cpuid_config > ARRAY_SIZE(td_conf->cpuid_config_values))
+		return -EINVAL;
 
-	ret = ret ?: get_tdx_sys_info_features(&sysinfo->features);
-	ret = ret ?: get_tdx_sys_info_tdmr(&sysinfo->tdmr);
-	ret = ret ?: get_tdx_sys_info_td_ctrl(&sysinfo->td_ctrl);
-	ret = ret ?: get_tdx_sys_info_td_conf(&sysinfo->td_conf);
+	/*
+	 * Populate 2 special fields, td_conf.cpuid_config_leaves[] and
+	 * td_conf.cpuid_config_values[][]
+	 */
+	for (i = 0; i < td_conf->num_cpuid_config; i++) {
+		ret = read_sys_metadata_field(cpuid_config_leaves.field_id + i,
+					      cpuid_config_leaves.offset +
+						cpuid_config_leaves.size * i,
+					      cpuid_config_leaves.size,
+					      sysinfo);
+		if (ret)
+			return ret;
+	}
+
+	for (i = 0;
+	     i < td_conf->num_cpuid_config * ARRAY_SIZE(td_conf->cpuid_config_values[0]);
+	     i++) {
+		ret = read_sys_metadata_field(cpuid_config_values.field_id + i,
+					      cpuid_config_values.offset +
+						cpuid_config_values.size * i,
+					      cpuid_config_values.size,
+					      sysinfo);
+		if (ret)
+			return ret;
+	}
 
-	return ret;
+	return 0;
 }
-- 
2.25.1


Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ