Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

cvm: Refactor ohcl/ovmm emulator for flexible register retrieval and update #482

Open
wants to merge 26 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 19 commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
9d84290
emulator changes, rebased on emulator cache commit
babayet2 Dec 9, 2024
0f03330
remove register wrappers
babayet2 Dec 11, 2024
0c7f163
emulator changes, rebased on emulator cache commit
babayet2 Dec 11, 2024
6d5d20c
use state object for all backends
babayet2 Dec 13, 2024
586ecc8
update names for clarity, fix compiler warnings
babayet2 Dec 14, 2024
a17af8c
fmt fix
babayet2 Dec 14, 2024
2a6181e
comments
babayet2 Dec 14, 2024
662c3d3
compiling again after arm rebase
babayet2 Dec 21, 2024
44f5274
remove unnecessary iced_x86 imports
babayet2 Dec 26, 2024
59996ff
restrict bitness functions to x86emu crate
babayet2 Dec 27, 2024
5a180be
fmt fix
babayet2 Dec 27, 2024
0b88c8b
seperate out imports
babayet2 Dec 27, 2024
6119a7e
remove emulation cache for snp
babayet2 Dec 27, 2024
6c47dda
kill load_registers for tdx/snp
babayet2 Dec 27, 2024
5734648
remove gp_sign_extend from EmulatorSupport trait
babayet2 Dec 30, 2024
71fad42
pass cache from backing into emulate call
babayet2 Dec 31, 2024
4b8441c
use cache for mshv
babayet2 Dec 31, 2024
d0df517
remove generic "emulate fast path", fmt fix
babayet2 Dec 31, 2024
ec1605b
move register flushes back to original locations
babayet2 Dec 31, 2024
0e1f88a
temporarily stub out mshv impl for non cvm impls
babayet2 Jan 3, 2025
3c251d0
replace segment/gp usize indexes with enums
babayet2 Jan 4, 2025
0c384ae
only cache expensive get/set in mshv emu
babayet2 Jan 4, 2025
5e5b91d
misc feedback from Steven
babayet2 Jan 4, 2025
47e9574
fmt fix
babayet2 Jan 4, 2025
ccbf28d
non-cvm emulator sup impls compiling
babayet2 Jan 7, 2025
2a85175
mshv compiler errors, and flush rsp to cpu_context
babayet2 Jan 7, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
29 changes: 11 additions & 18 deletions openhcl/virt_mshv_vtl/src/processor/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,7 @@ use virt::Processor;
use virt::StopVp;
use virt::VpHaltReason;
use virt::VpIndex;
use virt_support_x86emu::emulate::EmulatorSupport;
babayet2 marked this conversation as resolved.
Show resolved Hide resolved
use vm_topology::processor::TargetVpInfo;
use vmcore::vmtime::VmTimeAccess;
use vtl_array::VtlArray;
Expand Down Expand Up @@ -187,7 +188,7 @@ mod private {

pub trait BackingPrivate: 'static + Sized + InspectMut + Sized {
type HclBacking: hcl::ioctl::Backing;
type EmulationCache: Default;
type EmulationCache;
type Shared;

fn shared(shared: &BackingShared) -> &Self::Shared;
Expand Down Expand Up @@ -1032,24 +1033,20 @@ impl<'a, T: Backing> UhProcessor<'a, T> {
devices: &D,
interruption_pending: bool,
vtl: GuestVtl,
cache: T::EmulationCache,
) -> Result<(), VpHaltReason<UhRunVpError>>
where
for<'b> UhEmulationState<'b, 'a, D, T>:
virt_support_x86emu::emulate::EmulatorSupport<Error = UhRunVpError>,
for<'b> UhEmulationState<'b, 'a, D, T>: EmulatorSupport<Error = UhRunVpError>,
{
let guest_memory = &self.partition.gm[vtl];
virt_support_x86emu::emulate::emulate(
&mut UhEmulationState {
vp: &mut *self,
interruption_pending,
devices,
vtl,
cache: T::EmulationCache::default(),
},
guest_memory,
let mut emulation_state = UhEmulationState {
vp: &mut *self,
interruption_pending,
devices,
)
.await
vtl,
cache,
};
virt_support_x86emu::emulate::emulate(&mut emulation_state, guest_memory, devices).await
}

/// Emulates an instruction due to a memory access exit.
Expand Down Expand Up @@ -1174,10 +1171,6 @@ struct UhEmulationState<'a, 'b, T: CpuIo, U: Backing> {
interruption_pending: bool,
devices: &'a T,
vtl: GuestVtl,
#[cfg_attr(
guest_arch = "x86_64",
expect(dead_code, reason = "not used yet in x86_64")
)]
cache: U::EmulationCache,
}

Expand Down
201 changes: 121 additions & 80 deletions openhcl/virt_mshv_vtl/src/processor/mshv/x64.rs
Original file line number Diff line number Diff line change
Expand Up @@ -84,6 +84,7 @@ use x86defs::xsave::XsaveHeader;
use x86defs::xsave::XFEATURE_SSE;
use x86defs::xsave::XFEATURE_X87;
use x86defs::RFlags;
use x86defs::SegmentRegister;
use zerocopy::AsBytes;
use zerocopy::FromBytes;
use zerocopy::FromZeroes;
Expand Down Expand Up @@ -125,7 +126,7 @@ struct ProcessorStatsX86 {
impl BackingPrivate for HypervisorBackedX86 {
type HclBacking = MshvX64;
type Shared = ();
type EmulationCache = ();
type EmulationCache = x86emu::CpuState;
babayet2 marked this conversation as resolved.
Show resolved Hide resolved

fn shared(_: &BackingShared) -> &Self::Shared {
&()
Expand Down Expand Up @@ -704,27 +705,36 @@ impl<'a, 'b> InterceptHandler<'a, 'b> {
== self.vp.partition.monitor_page.gpa()
&& message.header.intercept_access_type == HvInterceptAccessType::WRITE
{
let instruction_bytes = message.instruction_bytes;
let instruction_bytes = &instruction_bytes[..message.instruction_byte_count as usize];
let tlb_lock_held = message.memory_access_info.gva_gpa_valid()
|| message.memory_access_info.tlb_locked();
let mut state = self.vp.emulator_state(self.intercepted_vtl);
let cache = self.vp.emulation_cache(self.intercepted_vtl);
let guest_memory = &self.vp.partition.gm[self.intercepted_vtl];
let mut emulation_state = UhEmulationState {
vp: &mut *self.vp,
interruption_pending,
devices: dev,
vtl: self.intercepted_vtl,
cache,
};
if let Some(bit) = virt_support_x86emu::emulate::emulate_mnf_write_fast_path(
instruction_bytes,
&mut state,
&mut emulation_state,
guest_memory,
dev,
interruption_pending,
tlb_lock_held,
) {
self.vp.set_emulator_state(self.intercepted_vtl, &state);
emulation_state.flush();
if let Some(connection_id) = self.vp.partition.monitor_page.write_bit(bit) {
signal_mnf(dev, connection_id);
}
return Ok(());
}
}

let cache = self.vp.emulation_cache(self.intercepted_vtl);
smalis-msft marked this conversation as resolved.
Show resolved Hide resolved

self.vp
.emulate(dev, interruption_pending, self.intercepted_vtl)
.emulate(dev, interruption_pending, self.intercepted_vtl, cache)
.await?;
Ok(())
}
Expand All @@ -745,8 +755,9 @@ impl<'a, 'b> InterceptHandler<'a, 'b> {
let interruption_pending = message.header.execution_state.interruption_pending();

if message.access_info.string_op() || message.access_info.rep_prefix() {
let cache = self.vp.emulation_cache(self.intercepted_vtl);
self.vp
.emulate(dev, interruption_pending, self.intercepted_vtl)
.emulate(dev, interruption_pending, self.intercepted_vtl, cache)
.await
} else {
let next_rip = next_rip(&message.header);
Expand Down Expand Up @@ -1141,62 +1152,6 @@ impl UhProcessor<'_, HypervisorBackedX86> {
.expect("set_vp_register should succeed for pending event");
}

fn emulator_state(&mut self, vtl: GuestVtl) -> x86emu::CpuState {
const NAMES: &[HvX64RegisterName] = &[
HvX64RegisterName::Rsp,
HvX64RegisterName::Es,
HvX64RegisterName::Ds,
HvX64RegisterName::Fs,
HvX64RegisterName::Gs,
HvX64RegisterName::Ss,
HvX64RegisterName::Cr0,
HvX64RegisterName::Efer,
];
let mut values = [FromZeroes::new_zeroed(); NAMES.len()];
self.runner
.get_vp_registers(vtl, NAMES, &mut values)
.expect("register query should not fail");

let [rsp, es, ds, fs, gs, ss, cr0, efer] = values;

let mut gps = self.runner.cpu_context().gps;
gps[x86emu::CpuState::RSP] = rsp.as_u64();

let message = self.runner.exit_message();
let header = HvX64InterceptMessageHeader::ref_from_prefix(message.payload()).unwrap();

x86emu::CpuState {
gps,
segs: [
from_seg(es.into()),
from_seg(header.cs_segment),
from_seg(ss.into()),
from_seg(ds.into()),
from_seg(fs.into()),
from_seg(gs.into()),
],
rip: header.rip,
rflags: header.rflags.into(),
cr0: cr0.as_u64(),
efer: efer.as_u64(),
}
}

fn set_emulator_state(&mut self, vtl: GuestVtl, state: &x86emu::CpuState) {
self.runner
.set_vp_registers(
vtl,
[
(HvX64RegisterName::Rip, state.rip),
(HvX64RegisterName::Rflags, state.rflags.into()),
(HvX64RegisterName::Rsp, state.gps[x86emu::CpuState::RSP]),
],
)
.unwrap();

self.runner.cpu_context_mut().gps = state.gps;
}

fn set_vsm_partition_config(
&mut self,
vtl: GuestVtl,
Expand Down Expand Up @@ -1301,11 +1256,72 @@ impl UhProcessor<'_, HypervisorBackedX86> {

Ok(())
}

///Eagerly load registers for emulation
babayet2 marked this conversation as resolved.
Show resolved Hide resolved
fn emulation_cache(&mut self, vtl: GuestVtl) -> x86emu::CpuState {
const NAMES: &[HvX64RegisterName] = &[
HvX64RegisterName::Rsp,
HvX64RegisterName::Es,
HvX64RegisterName::Ds,
HvX64RegisterName::Fs,
HvX64RegisterName::Gs,
HvX64RegisterName::Ss,
HvX64RegisterName::Cr0,
HvX64RegisterName::Efer,
];
let mut values = [FromZeroes::new_zeroed(); NAMES.len()];
self.runner
.get_vp_registers(vtl, NAMES, &mut values)
.expect("register query should not fail");

let [rsp, es, ds, fs, gs, ss, cr0, efer] = values;

let mut gps = self.runner.cpu_context().gps;
gps[x86emu::CpuState::RSP] = rsp.as_u64();

let message = self.runner.exit_message();
let header = HvX64InterceptMessageHeader::ref_from_prefix(message.payload()).unwrap();
babayet2 marked this conversation as resolved.
Show resolved Hide resolved

x86emu::CpuState {
gps,
segs: [
from_seg(es.into()),
from_seg(header.cs_segment),
from_seg(ss.into()),
from_seg(ds.into()),
from_seg(fs.into()),
from_seg(gs.into()),
],
rip: header.rip,
rflags: header.rflags.into(),
cr0: cr0.as_u64(),
efer: efer.as_u64(),
}
}
}

impl<T: CpuIo> EmulatorSupport for UhEmulationState<'_, '_, T, HypervisorBackedX86> {
type Error = UhRunVpError;

fn flush(&mut self) {
self.vp
.runner
.set_vp_registers(
self.vtl,
[
(HvX64RegisterName::Rip, self.cache.rip),
(HvX64RegisterName::Rflags, self.cache.rflags.into()),
(
HvX64RegisterName::Rsp,
self.cache.gps[x86emu::CpuState::RSP],
),
],
)
.unwrap();

self.vp.runner.cpu_context_mut().gps = self.cache.gps;
}

fn vp_index(&self) -> VpIndex {
self.vp.vp_index()
}
Expand All @@ -1314,15 +1330,51 @@ impl<T: CpuIo> EmulatorSupport for UhEmulationState<'_, '_, T, HypervisorBackedX
self.vp.partition.caps.vendor
}

fn state(&mut self) -> Result<x86emu::CpuState, Self::Error> {
Ok(self.vp.emulator_state(self.vtl))
fn gp(&mut self, reg: usize) -> u64 {
self.cache.gps[reg]
}

fn set_state(&mut self, state: x86emu::CpuState) -> Result<(), Self::Error> {
self.vp.set_emulator_state(self.vtl, &state);
fn set_gp(&mut self, reg: usize, v: u64) {
self.cache.gps[reg] = v;
}

fn xmm(&mut self, index: usize) -> u128 {
u128::from_le_bytes(self.vp.runner.cpu_context().fx_state.xmm[index])
}

fn set_xmm(&mut self, index: usize, v: u128) -> Result<(), Self::Error> {
self.vp.runner.cpu_context_mut().fx_state.xmm[index] = v.to_le_bytes();
Ok(())
}

fn rip(&mut self) -> u64 {
self.cache.rip
}

fn set_rip(&mut self, v: u64) {
self.cache.rip = v;
}

fn segment(&mut self, index: usize) -> SegmentRegister {
self.cache.segs[index]
}

fn efer(&mut self) -> u64 {
self.cache.efer
}

fn cr0(&mut self) -> u64 {
self.cache.cr0
}

fn rflags(&mut self) -> RFlags {
self.cache.rflags
}

fn set_rflags(&mut self, v: RFlags) {
self.cache.rflags = v;
}

fn instruction_bytes(&self) -> &[u8] {
let message = self.vp.runner.exit_message();
match message.header.typ {
Expand Down Expand Up @@ -1525,17 +1577,6 @@ impl<T: CpuIo> EmulatorSupport for UhEmulationState<'_, '_, T, HypervisorBackedX
.expect("set_vp_registers hypercall for setting pending event should not fail");
}

fn get_xmm(&mut self, reg: usize) -> Result<u128, Self::Error> {
Ok(u128::from_le_bytes(
self.vp.runner.cpu_context().fx_state.xmm[reg],
))
}

fn set_xmm(&mut self, reg: usize, value: u128) -> Result<(), Self::Error> {
self.vp.runner.cpu_context_mut().fx_state.xmm[reg] = value.to_le_bytes();
Ok(())
}

fn check_monitor_write(&self, gpa: u64, bytes: &[u8]) -> bool {
self.vp
.partition
Expand Down
Loading
Loading