forked from firecracker-microvm/firecracker
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmod.rs
139 lines (116 loc) · 4.26 KB
/
mod.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
// Copyright 2019 Amazon.com, Inc. or its affiliates. All Rights Reserved.
// SPDX-License-Identifier: Apache-2.0
pub mod amd;
pub mod common;
pub mod intel;
pub use kvm_bindings::kvm_cpuid_entry2;
use kvm_ioctls::CpuId;
use brand_string::BrandString;
use brand_string::Reg as BsReg;
use common::get_vendor_id;
/// Structure containing the specifications of the VM
pub struct VmSpec {
/// The vendor id of the CPU
cpu_vendor_id: [u8; 12],
/// The id of the current logical cpu in the range [0..cpu_count].
cpu_id: u8,
/// The total number of logical cpus.
cpu_count: u8,
/// Specifies whether hyper-threading is enabled.
ht_enabled: bool,
/// The desired brand string for the guest.
brand_string: BrandString,
}
impl VmSpec {
/// Creates a new instance of VmSpec with the specified parameters
/// The brand string is deduced from the vendor_id
pub fn new(cpu_id: u8, cpu_count: u8, ht_enabled: bool) -> Result<VmSpec, Error> {
let cpu_vendor_id = get_vendor_id().map_err(Error::InternalError)?;
Ok(VmSpec {
cpu_vendor_id,
cpu_id,
cpu_count,
ht_enabled,
brand_string: BrandString::from_vendor_id(&cpu_vendor_id),
})
}
/// Returns an immutable reference to cpu_vendor_id
pub fn cpu_vendor_id(&self) -> &[u8; 12] {
&self.cpu_vendor_id
}
}
/// Errors associated with processing the CPUID leaves.
#[derive(Debug, Clone)]
pub enum Error {
/// A FamStructWrapper operation has failed
FamError(vmm_sys_util::fam::Error),
/// A call to an internal helper method failed
InternalError(super::common::Error),
/// The maximum number of addressable logical CPUs cannot be stored in an `u8`.
VcpuCountOverflow,
}
pub type EntryTransformerFn =
fn(entry: &mut kvm_cpuid_entry2, vm_spec: &VmSpec) -> Result<(), Error>;
/// Generic trait that provides methods for transforming the cpuid
pub trait CpuidTransformer {
/// Trait main function. It processes the cpuid and makes the desired transformations.
/// The default logic can be overwritten if needed. For example see `AmdCpuidTransformer`.
fn process_cpuid(&self, cpuid: &mut CpuId, vm_spec: &VmSpec) -> Result<(), Error> {
self.process_entries(cpuid, vm_spec)
}
/// Iterates through all the cpuid entries and calls the associated transformer for each one.
fn process_entries(&self, cpuid: &mut CpuId, vm_spec: &VmSpec) -> Result<(), Error> {
for entry in cpuid.as_mut_slice().iter_mut() {
let maybe_transformer_fn = self.entry_transformer_fn(entry);
if let Some(transformer_fn) = maybe_transformer_fn {
transformer_fn(entry, vm_spec)?;
}
}
Ok(())
}
/// Gets the associated transformer for a cpuid entry
fn entry_transformer_fn(&self, _entry: &mut kvm_cpuid_entry2) -> Option<EntryTransformerFn> {
None
}
}
#[cfg(test)]
mod test {
use super::*;
use kvm_bindings::kvm_cpuid_entry2;
const PROCESSED_FN: u32 = 1;
const EXPECTED_INDEX: u32 = 100;
fn transform_entry(entry: &mut kvm_cpuid_entry2, _vm_spec: &VmSpec) -> Result<(), Error> {
entry.index = EXPECTED_INDEX;
Ok(())
}
struct MockCpuidTransformer {}
impl CpuidTransformer for MockCpuidTransformer {
fn entry_transformer_fn(&self, entry: &mut kvm_cpuid_entry2) -> Option<EntryTransformerFn> {
match entry.function {
PROCESSED_FN => Some(transform_entry),
_ => None,
}
}
}
#[test]
fn test_process_cpuid() {
let num_entries = 5;
let mut cpuid = CpuId::new(num_entries);
let vm_spec = VmSpec::new(0, 1, false);
cpuid.as_mut_slice()[0].function = PROCESSED_FN;
assert!(MockCpuidTransformer {}
.process_cpuid(&mut cpuid, &vm_spec.unwrap())
.is_ok());
assert!(cpuid.as_mut_slice().len() == num_entries);
for entry in cpuid.as_mut_slice().iter() {
match entry.function {
PROCESSED_FN => {
assert_eq!(entry.index, EXPECTED_INDEX);
}
_ => {
assert_ne!(entry.index, EXPECTED_INDEX);
}
}
}
}
}