Hi Jan,

On 30.10.25 13:23, Jan Beulich wrote:
On 30.10.2025 00:54, Grygorii Strashko wrote:
From: Grygorii Strashko <[email protected]>

The Cache Disable mode data is used only by VMX code, so move it from
common HVM structures into VMX specific structures:
- move "uc_lock", "is_in_uc_mode" fields from struct hvm_domain to struct
vmx_domain;
- move "cache_mode" field from struct hvm_vcpu to struct vmx_vcpu.

Hence, the "is_in_uc_mode" field is used directly in mm/shadow/multi.c
_sh_propagate(), introduce the hvm_is_in_uc_mode() macro to avoid direct
access to this field and account for INTEL_VMX configuration.

Signed-off-by: Grygorii Strashko <[email protected]>

Requested-by: Andrew ?

ok


--- a/xen/arch/x86/hvm/vmx/vmx.c
+++ b/xen/arch/x86/hvm/vmx/vmx.c
@@ -583,6 +583,7 @@ static int cf_check vmx_domain_initialise(struct domain *d)
      int rc;
d->arch.ctxt_switch = &csw;
+    spin_lock_init(&d->arch.hvm.vmx.uc_lock);

I don't think this is the best place; in any event it wants to be separated from
adjacent code by a blank line. I'd prefer if it was put ...

      /*
       * Work around CVE-2018-12207?  The hardware domain is already permitted

... below this CVE workaround.

ok


--- a/xen/arch/x86/include/asm/hvm/vmx/vmcs.h
+++ b/xen/arch/x86/include/asm/hvm/vmx/vmcs.h
@@ -46,7 +46,9 @@ struct ept_data {
#define _VMX_DOMAIN_PML_ENABLED 0
  #define VMX_DOMAIN_PML_ENABLED     (1ul << _VMX_DOMAIN_PML_ENABLED)
+
  struct vmx_domain {
+    spinlock_t uc_lock;
      mfn_t apic_access_mfn;
      /* VMX_DOMAIN_* */
      unsigned int status;

Any reason to make this the very first field of the struct? It might better
live adjacent to the other field you move; there's going to be some padding
anyway, afaict.

I've tried to put fields in holes and checked with pahole.

With current change it is:
struct vmx_domain {
        spinlock_t                 uc_lock;              /*     0     8 */
        mfn_t                      apic_access_mfn;      /*     8     8 */
        unsigned int               status;               /*    16     4 */
        _Bool                      exec_sp;              /*    20     1 */
        _Bool                      is_in_uc_mode;        /*    21     1 */

        /* size: 24, cachelines: 1, members: 5 */
        /* padding: 2 */
        /* last cacheline: 24 bytes */
};

It seems can be grouped like below?:
struct vmx_domain {
        mfn_t                      apic_access_mfn;      /*     0     8 */
        unsigned int               status;               /*     8     4 */
        _Bool                      exec_sp;              /*    12     1 */
        _Bool                      is_in_uc_mode;        /*    13     1 */

        /* XXX 2 bytes hole, try to pack */

        spinlock_t                 uc_lock;              /*    16     8 */

        /* size: 24, cachelines: 1, members: 5 */
        /* sum members: 22, holes: 1, sum holes: 2 */
        /* last cacheline: 24 bytes */
};


@@ -56,6 +58,12 @@ struct vmx_domain {
       * around CVE-2018-12207 as appropriate.
       */
      bool exec_sp;
+    /*
+     * If one of vcpus of this domain is in no_fill_mode or
+     * mtrr/pat between vcpus is not the same, set is_in_uc_mode.
+     * Protected by uc_lock.
+     */
+    bool is_in_uc_mode;

Imo while moving, the is_ prefix could also be dropped. It doesn't convey any
extra information on top of the in_, and I think we prefer is_*() also as
typically function(-like) predicates. (I.e. in hvm_is_in_uc_mode() I'm fine
with the name.)

ok


@@ -93,6 +101,9 @@ struct pi_blocking_vcpu {
      spinlock_t           *lock;
  };
+#define NORMAL_CACHE_MODE 0
+#define NO_FILL_CACHE_MODE         2

As you necessarily touch all use sites, could we switch to the more common
CACHE_MODE_* at this occasion? Also imo these want to live ...

@@ -156,6 +167,9 @@ struct vmx_vcpu {
uint8_t lbr_flags; + /* Which cache mode is this VCPU in (CR0:CD/NW)? */
+    uint8_t              cache_mode;

... right next to the field they belong to.

ok.

--
Best regards,
-grygorii


Reply via email to