KVM: selftests: Add X86_PROPERTY_* framework to retrieve CPUID values
Introduce X86_PROPERTY_* to allow retrieving values/properties from CPUID leafs, e.g. MAXPHYADDR from CPUID.0x80000008. Use the same core code as X86_FEATURE_*, the primary difference is that properties are multi-bit values, whereas features enumerate a single bit. Add this_cpu_has_p() to allow querying whether or not a property exists based on the maximum leaf associated with the property, e.g. MAXPHYADDR doesn't exist if the max leaf for 0x8000_xxxx is less than 0x8000_0008. Use the new property infrastructure in vm_compute_max_gfn() to prove that the code works as intended. Future patches will convert additional selftests code. Signed-off-by: Sean Christopherson <seanjc@google.com> Link: https://lore.kernel.org/r/20221006005125.680782-4-seanjc@google.com
This commit is contained in:
parent
ee37955366
commit
53a7dc0f21
2 changed files with 76 additions and 10 deletions
|
@ -168,6 +168,48 @@ struct kvm_x86_cpu_feature {
|
||||||
#define X86_FEATURE_KVM_HC_MAP_GPA_RANGE KVM_X86_CPU_FEATURE(0x40000001, 0, EAX, 16)
|
#define X86_FEATURE_KVM_HC_MAP_GPA_RANGE KVM_X86_CPU_FEATURE(0x40000001, 0, EAX, 16)
|
||||||
#define X86_FEATURE_KVM_MIGRATION_CONTROL KVM_X86_CPU_FEATURE(0x40000001, 0, EAX, 17)
|
#define X86_FEATURE_KVM_MIGRATION_CONTROL KVM_X86_CPU_FEATURE(0x40000001, 0, EAX, 17)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Same idea as X86_FEATURE_XXX, but X86_PROPERTY_XXX retrieves a multi-bit
|
||||||
|
* value/property as opposed to a single-bit feature. Again, pack the info
|
||||||
|
* into a 64-bit value to pass by value with no overhead.
|
||||||
|
*/
|
||||||
|
struct kvm_x86_cpu_property {
|
||||||
|
u32 function;
|
||||||
|
u8 index;
|
||||||
|
u8 reg;
|
||||||
|
u8 lo_bit;
|
||||||
|
u8 hi_bit;
|
||||||
|
};
|
||||||
|
#define KVM_X86_CPU_PROPERTY(fn, idx, gpr, low_bit, high_bit) \
|
||||||
|
({ \
|
||||||
|
struct kvm_x86_cpu_property property = { \
|
||||||
|
.function = fn, \
|
||||||
|
.index = idx, \
|
||||||
|
.reg = KVM_CPUID_##gpr, \
|
||||||
|
.lo_bit = low_bit, \
|
||||||
|
.hi_bit = high_bit, \
|
||||||
|
}; \
|
||||||
|
\
|
||||||
|
static_assert(low_bit < high_bit); \
|
||||||
|
static_assert((fn & 0xc0000000) == 0 || \
|
||||||
|
(fn & 0xc0000000) == 0x40000000 || \
|
||||||
|
(fn & 0xc0000000) == 0x80000000 || \
|
||||||
|
(fn & 0xc0000000) == 0xc0000000); \
|
||||||
|
static_assert(idx < BIT(sizeof(property.index) * BITS_PER_BYTE)); \
|
||||||
|
property; \
|
||||||
|
})
|
||||||
|
|
||||||
|
#define X86_PROPERTY_MAX_BASIC_LEAF KVM_X86_CPU_PROPERTY(0, 0, EAX, 0, 31)
|
||||||
|
|
||||||
|
#define X86_PROPERTY_MAX_KVM_LEAF KVM_X86_CPU_PROPERTY(0x40000000, 0, EAX, 0, 31)
|
||||||
|
|
||||||
|
#define X86_PROPERTY_MAX_EXT_LEAF KVM_X86_CPU_PROPERTY(0x80000000, 0, EAX, 0, 31)
|
||||||
|
#define X86_PROPERTY_MAX_PHY_ADDR KVM_X86_CPU_PROPERTY(0x80000008, 0, EAX, 0, 7)
|
||||||
|
#define X86_PROPERTY_PHYS_ADDR_REDUCTION KVM_X86_CPU_PROPERTY(0x8000001F, 0, EBX, 6, 11)
|
||||||
|
|
||||||
|
#define X86_PROPERTY_MAX_CENTAUR_LEAF KVM_X86_CPU_PROPERTY(0xC0000000, 0, EAX, 0, 31)
|
||||||
|
|
||||||
|
|
||||||
/* Page table bitfield declarations */
|
/* Page table bitfield declarations */
|
||||||
#define PTE_PRESENT_MASK BIT_ULL(0)
|
#define PTE_PRESENT_MASK BIT_ULL(0)
|
||||||
#define PTE_WRITABLE_MASK BIT_ULL(1)
|
#define PTE_WRITABLE_MASK BIT_ULL(1)
|
||||||
|
@ -455,6 +497,32 @@ static inline bool this_cpu_has(struct kvm_x86_cpu_feature feature)
|
||||||
feature.reg, feature.bit, feature.bit);
|
feature.reg, feature.bit, feature.bit);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline uint32_t this_cpu_property(struct kvm_x86_cpu_property property)
|
||||||
|
{
|
||||||
|
return __this_cpu_has(property.function, property.index,
|
||||||
|
property.reg, property.lo_bit, property.hi_bit);
|
||||||
|
}
|
||||||
|
|
||||||
|
static __always_inline bool this_cpu_has_p(struct kvm_x86_cpu_property property)
|
||||||
|
{
|
||||||
|
uint32_t max_leaf;
|
||||||
|
|
||||||
|
switch (property.function & 0xc0000000) {
|
||||||
|
case 0:
|
||||||
|
max_leaf = this_cpu_property(X86_PROPERTY_MAX_BASIC_LEAF);
|
||||||
|
break;
|
||||||
|
case 0x40000000:
|
||||||
|
max_leaf = this_cpu_property(X86_PROPERTY_MAX_KVM_LEAF);
|
||||||
|
break;
|
||||||
|
case 0x80000000:
|
||||||
|
max_leaf = this_cpu_property(X86_PROPERTY_MAX_EXT_LEAF);
|
||||||
|
break;
|
||||||
|
case 0xc0000000:
|
||||||
|
max_leaf = this_cpu_property(X86_PROPERTY_MAX_CENTAUR_LEAF);
|
||||||
|
}
|
||||||
|
return max_leaf >= property.function;
|
||||||
|
}
|
||||||
|
|
||||||
#define SET_XMM(__var, __xmm) \
|
#define SET_XMM(__var, __xmm) \
|
||||||
asm volatile("movq %0, %%"#__xmm : : "r"(__var) : #__xmm)
|
asm volatile("movq %0, %%"#__xmm : : "r"(__var) : #__xmm)
|
||||||
|
|
||||||
|
|
|
@ -1229,7 +1229,8 @@ unsigned long vm_compute_max_gfn(struct kvm_vm *vm)
|
||||||
{
|
{
|
||||||
const unsigned long num_ht_pages = 12 << (30 - vm->page_shift); /* 12 GiB */
|
const unsigned long num_ht_pages = 12 << (30 - vm->page_shift); /* 12 GiB */
|
||||||
unsigned long ht_gfn, max_gfn, max_pfn;
|
unsigned long ht_gfn, max_gfn, max_pfn;
|
||||||
uint32_t eax, ebx, ecx, edx, max_ext_leaf;
|
uint32_t eax, ebx, ecx, edx;
|
||||||
|
uint8_t maxphyaddr;
|
||||||
|
|
||||||
max_gfn = (1ULL << (vm->pa_bits - vm->page_shift)) - 1;
|
max_gfn = (1ULL << (vm->pa_bits - vm->page_shift)) - 1;
|
||||||
|
|
||||||
|
@ -1252,17 +1253,14 @@ unsigned long vm_compute_max_gfn(struct kvm_vm *vm)
|
||||||
* reduced due to SME by bits 11:6 of CPUID[0x8000001f].EBX. Use
|
* reduced due to SME by bits 11:6 of CPUID[0x8000001f].EBX. Use
|
||||||
* the old conservative value if MAXPHYADDR is not enumerated.
|
* the old conservative value if MAXPHYADDR is not enumerated.
|
||||||
*/
|
*/
|
||||||
cpuid(0x80000000, &eax, &ebx, &ecx, &edx);
|
if (!this_cpu_has_p(X86_PROPERTY_MAX_PHY_ADDR))
|
||||||
max_ext_leaf = eax;
|
|
||||||
if (max_ext_leaf < 0x80000008)
|
|
||||||
goto done;
|
goto done;
|
||||||
|
|
||||||
cpuid(0x80000008, &eax, &ebx, &ecx, &edx);
|
maxphyaddr = this_cpu_property(X86_PROPERTY_MAX_PHY_ADDR);
|
||||||
max_pfn = (1ULL << ((eax & 0xff) - vm->page_shift)) - 1;
|
max_pfn = (1ULL << (maxphyaddr - vm->page_shift)) - 1;
|
||||||
if (max_ext_leaf >= 0x8000001f) {
|
|
||||||
cpuid(0x8000001f, &eax, &ebx, &ecx, &edx);
|
if (this_cpu_has_p(X86_PROPERTY_PHYS_ADDR_REDUCTION))
|
||||||
max_pfn >>= (ebx >> 6) & 0x3f;
|
max_pfn >>= this_cpu_property(X86_PROPERTY_PHYS_ADDR_REDUCTION);
|
||||||
}
|
|
||||||
|
|
||||||
ht_gfn = max_pfn - num_ht_pages;
|
ht_gfn = max_pfn - num_ht_pages;
|
||||||
done:
|
done:
|
||||||
|
|
Loading…
Add table
Reference in a new issue