use virt::CpuidLeaf;
use virt::IsolationType;
use vm_topology::processor::ProcessorTopology;
use vm_topology::processor::x86::X86Topology;
const MAX_CPUS: usize = 2048;
pub fn hv_cpuid_leaves(
topology: &ProcessorTopology<X86Topology>,
isolation: IsolationType,
access_vsm: bool,
hv_version: [u32; 4],
vtom: Option<u64>,
) -> Vec<CpuidLeaf> {
let hardware_isolated = isolation.is_hardware_isolated();
let split_u128 = |x: u128| -> [u32; 4] { zerocopy::transmute!(x) };
let privileges = {
let mut privileges = hvdef::HvPartitionPrivilege::new()
.with_access_partition_reference_counter(true)
.with_access_hypercall_msrs(true)
.with_access_vp_index(true)
.with_access_frequency_msrs(true)
.with_access_synic_msrs(true)
.with_access_synthetic_timer_msrs(true)
.with_access_apic_msrs(true)
.with_access_vp_runtime_msr(true)
.with_access_partition_reference_tsc(true)
.with_start_virtual_processor(true)
.with_access_vsm(access_vsm)
.with_enable_extended_gva_ranges_flush_va_list(true);
if hardware_isolated {
privileges = privileges
.with_isolation(true)
.with_access_guest_idle_msr(true);
}
privileges
};
let mut hv_cpuid = vec![
CpuidLeaf::new(
hvdef::HV_CPUID_FUNCTION_HV_VENDOR_AND_MAX_FUNCTION,
[
if hardware_isolated {
hvdef::HV_CPUID_FUNCTION_MS_HV_ISOLATION_CONFIGURATION
} else {
hvdef::HV_CPUID_FUNCTION_MS_HV_IMPLEMENTATION_LIMITS
},
u32::from_le_bytes(*b"Micr"),
u32::from_le_bytes(*b"osof"),
u32::from_le_bytes(*b"t Hv"),
],
),
CpuidLeaf::new(
hvdef::HV_CPUID_FUNCTION_HV_INTERFACE,
[u32::from_le_bytes(*b"Hv#1"), 0, 0, 0],
),
CpuidLeaf::new(hvdef::HV_CPUID_FUNCTION_MS_HV_VERSION, hv_version),
CpuidLeaf::new(hvdef::HV_CPUID_FUNCTION_MS_HV_FEATURES, {
let mut features = hvdef::HvFeatures::new()
.with_privileges(privileges)
.with_frequency_regs_available(true)
.with_direct_synthetic_timers(true)
.with_extended_gva_ranges_for_flush_virtual_address_list_available(true);
if hardware_isolated {
features = features.with_guest_idle_available(true);
}
if cfg!(guest_arch = "x86_64") {
features = features.with_xmm_registers_for_fast_hypercall_available(true);
}
split_u128(features.into())
}),
CpuidLeaf::new(hvdef::HV_CPUID_FUNCTION_MS_HV_ENLIGHTENMENT_INFORMATION, {
let use_apic_msrs = match topology.apic_mode() {
vm_topology::processor::x86::ApicMode::XApic => {
true
}
vm_topology::processor::x86::ApicMode::X2ApicSupported
| vm_topology::processor::x86::ApicMode::X2ApicEnabled => {
isolation == IsolationType::Snp
}
};
let mut enlightenments = hvdef::HvEnlightenmentInformation::new()
.with_deprecate_auto_eoi(true)
.with_use_relaxed_timing(true)
.with_use_ex_processor_masks(true)
.with_use_apic_msrs(use_apic_msrs);
if hardware_isolated {
enlightenments = enlightenments.with_long_spin_wait_count(!0); if isolation != IsolationType::Tdx {
enlightenments = enlightenments
.with_use_hypercall_for_remote_flush_and_local_flush_entire(true)
}
};
split_u128(enlightenments.into())
}),
CpuidLeaf::new(
hvdef::HV_CPUID_FUNCTION_MS_HV_IMPLEMENTATION_LIMITS,
[MAX_CPUS as u32, MAX_CPUS as u32, 0, 0],
),
];
if hardware_isolated {
hv_cpuid.append(&mut vec![
CpuidLeaf::new(
hvdef::HV_CPUID_FUNCTION_MS_HV_HARDWARE_FEATURES,
split_u128(
hvdef::HvHardwareFeatures::new()
.with_apic_overlay_assist_in_use(true)
.with_msr_bitmaps_in_use(true)
.with_second_level_address_translation_in_use(true)
.with_dma_remapping_in_use(false)
.with_interrupt_remapping_in_use(false)
.into(),
),
),
CpuidLeaf::new(
hvdef::HV_CPUID_FUNCTION_MS_HV_ISOLATION_CONFIGURATION,
split_u128(
hvdef::HvIsolationConfiguration::new()
.with_paravisor_present(true)
.with_isolation_type(isolation.to_hv().0)
.with_shared_gpa_boundary_active(true)
.with_shared_gpa_boundary_bits(
vtom.expect("cvm requires vtom").trailing_zeros() as u8,
)
.into(),
),
),
]);
}
hv_cpuid
}