]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
x86/virt/seamldr: Shut down the current TDX module
authorChao Gao <chao.gao@intel.com>
Wed, 20 May 2026 22:29:05 +0000 (15:29 -0700)
committerDave Hansen <dave.hansen@linux.intel.com>
Wed, 3 Jun 2026 15:59:24 +0000 (08:59 -0700)
The first step of TDX module updates is shutting down the current TDX
module. This step also packs state information that needs to be
preserved across updates, called "handoff data". This handoff data is
consumed by the updated module and stored internally in the SEAM range and
hidden from the kernel.

Since the handoff data layout may change between modules, the handoff
data is versioned. Each module has a native handoff version and
provides backward support for several older versions.

The complete handoff versioning protocol is complex as it supports both
module upgrades and downgrades. See details in "Intel Trust Domain
Extensions (Intel TDX) Module Base Architecture Specification", Chapter
"Handoff Versioning".

Ideally, the kernel needs to retrieve the handoff versions supported by
the current module and the new module and select a version supported by
both. But since this implementation only supports module upgrades, simply
request handoff data from the current module using its highest supported
version. That is sufficient for this upgrade-only implementation.

Retrieve the module's handoff version from TDX global metadata and add an
update step to shut down the module. Module shutdown only needs to run on
one CPU.

Don't cache the handoff information in tdx_sysinfo. It is used only for
module shutdown, and is present only when the TDX module supports updates.
Caching it in get_tdx_sys_info() would require extra update-support guards
and refreshing the cached value across module updates.

[ dhansen: fix up function variables, remove 'cpu'.
     Return from tdx_module_shutdown() early if handoff call fails. ]

Signed-off-by: Chao Gao <chao.gao@intel.com>
Signed-off-by: Dave Hansen <dave.hansen@linux.intel.com>
Reviewed-by: Tony Lindgren <tony.lindgren@linux.intel.com>
Reviewed-by: Xu Yilun <yilun.xu@linux.intel.com>
Reviewed-by: Kai Huang <kai.huang@intel.com>
Reviewed-by: Kiryl Shutsemau (Meta) <kas@kernel.org>
Link: https://patch.msgid.link/20260520133909.409394-17-chao.gao@intel.com
arch/x86/include/asm/tdx_global_metadata.h
arch/x86/virt/vmx/tdx/seamldr.c
arch/x86/virt/vmx/tdx/tdx.c
arch/x86/virt/vmx/tdx/tdx.h
arch/x86/virt/vmx/tdx/tdx_global_metadata.c

index 40689c8dc67eba79726f406e473474dbcb34f524..41150d546589c15b9adbfb80743700db61a40fbe 100644 (file)
@@ -40,6 +40,10 @@ struct tdx_sys_info_td_conf {
        u64 cpuid_config_values[128][2];
 };
 
+struct tdx_sys_info_handoff {
+       u16 module_hv;
+};
+
 struct tdx_sys_info {
        struct tdx_sys_info_version version;
        struct tdx_sys_info_features features;
index b03dce213102d82373704271ef511e58128a6f0f..3fe1d397ecd116ac53fde98fa556d16519a9b2fb 100644 (file)
@@ -18,6 +18,7 @@
 #include <asm/seamldr.h>
 
 #include "seamcall_internal.h"
+#include "tdx.h"
 
 /* P-SEAMLDR SEAMCALL leaf function */
 #define P_SEAMLDR_INFO                 0x8000000000000000
@@ -196,6 +197,7 @@ static int init_seamldr_params(struct seamldr_params *params,
  */
 enum module_update_state {
        MODULE_UPDATE_START,
+       MODULE_UPDATE_SHUTDOWN,
        MODULE_UPDATE_DONE,
 };
 
@@ -247,8 +249,16 @@ static int do_seamldr_install_module(void *seamldr_params)
 {
        enum module_update_state curstate = MODULE_UPDATE_START;
        enum module_update_state newstate;
+       bool is_lead_cpu = false;
        int ret = 0;
 
+       /*
+        * Some steps must be run on exactly one CPU. Pick a "lead" CPU to
+        * execute those steps. Use CPU 0 because it is always online.
+        */
+       if (smp_processor_id() == 0)
+               is_lead_cpu = true;
+
        do {
                newstate = READ_ONCE(update_ctrl.state);
 
@@ -259,7 +269,10 @@ static int do_seamldr_install_module(void *seamldr_params)
 
                curstate = newstate;
                switch (curstate) {
-               /* TODO: add the update steps. */
+               case MODULE_UPDATE_SHUTDOWN:
+                       if (is_lead_cpu)
+                               ret = tdx_module_shutdown();
+                       break;
                default:
                        break;
                }
index 53cf99c41dbbf7151976d38ca6a0bb226555bace..3fe01b546fb8a54ec671ba68a8502a4386414ac4 100644 (file)
@@ -328,7 +328,7 @@ err:
        return ret;
 }
 
-static __init int read_sys_metadata_field(u64 field_id, u64 *data)
+static int read_sys_metadata_field(u64 field_id, u64 *data)
 {
        struct tdx_module_args args = {};
        int ret;
@@ -1274,6 +1274,29 @@ static __init int tdx_enable(void)
 }
 subsys_initcall(tdx_enable);
 
+int tdx_module_shutdown(void)
+{
+       struct tdx_sys_info_handoff handoff = {};
+       struct tdx_module_args args = {};
+       int ret;
+
+       ret = get_tdx_sys_info_handoff(&handoff);
+       /*
+        * Handoff information is required for proper
+        * shutdown. Refuse to shut down without it.
+        */
+       if (ret)
+               return ret;
+
+       /*
+        * Use the module's handoff version as it is the highest the
+        * module can produce and most likely supported by newer modules.
+        */
+       args.rcx = handoff.module_hv;
+
+       return seamcall_prerr(TDH_SYS_SHUTDOWN, &args);
+}
+
 static bool is_pamt_page(unsigned long phys)
 {
        struct tdmr_info_list *tdmr_list = &tdx_tdmr_list;
index 76c5fb1e1ffe62537c837adeb63e3e65ed7b8ca2..f0c20dea0388a27cca2f2e87298efe299869d243 100644 (file)
@@ -46,6 +46,7 @@
 #define TDH_PHYMEM_PAGE_WBINVD         41
 #define TDH_VP_WR                      43
 #define TDH_SYS_CONFIG                 45
+#define TDH_SYS_SHUTDOWN               52
 #define TDH_SYS_DISABLE                        69
 
 /*
@@ -108,4 +109,6 @@ struct tdmr_info_list {
        int max_tdmrs;  /* How many 'tdmr_info's are allocated */
 };
 
+int tdx_module_shutdown(void);
+
 #endif
index d54d4227990c202c34ee224b9fce1a63dbbe224c..e793dec688abd3fbf45a4b2bea9456792b44d59e 100644 (file)
@@ -100,6 +100,19 @@ static __init int get_tdx_sys_info_td_conf(struct tdx_sys_info_td_conf *sysinfo_
        return ret;
 }
 
+static int get_tdx_sys_info_handoff(struct tdx_sys_info_handoff *sysinfo_handoff)
+{
+       int ret;
+       u64 val;
+
+       ret = read_sys_metadata_field(0x8900000100000000, &val);
+       if (ret)
+               return ret;
+
+       sysinfo_handoff->module_hv = val;
+       return 0;
+}
+
 static __init int get_tdx_sys_info(struct tdx_sys_info *sysinfo)
 {
        int ret = 0;