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
use std::fs;
use std::io;
use std::os::unix::io::FromRawFd;
use crate::cap::{CapBool, CapInt};
use crate::vm::Vm;
use crate::{ioctl, kvm_sys, libcret};
pub struct Kvm {
kvm: fs::File,
}
impl Kvm {
pub fn new() -> io::Result<Kvm> {
let kvm = libcret(unsafe {
libc::open("/dev/kvm\0".as_ptr().cast(), libc::O_RDWR | libc::O_CLOEXEC)
})
.map(|fd| unsafe { fs::File::from_raw_fd(fd) })?;
assert_eq!(
kvm_sys::KVM_API_VERSION,
ioctl(&kvm, kvm_sys::KVM_GET_API_VERSION, 0)?
);
Ok(Kvm { kvm })
}
fn get_vpcu_mmap_size(&self) -> io::Result<usize> {
ioctl(&self.kvm, kvm_sys::KVM_GET_VCPU_MMAP_SIZE, 0).map(|size| size as usize)
}
pub fn create_vm(&self) -> io::Result<Vm> {
let vm = ioctl(&self.kvm, kvm_sys::KVM_CREATE_VM, 0 )
.map(|fd| unsafe { fs::File::from_raw_fd(fd) })?;
let vcpu_mmap_size = self.get_vpcu_mmap_size()?;
Ok(Vm::new(vm, vcpu_mmap_size))
}
pub fn check_extenstion(&self, cap: CapBool) -> bool {
let ret = ioctl(&self.kvm, kvm_sys::KVM_CHECK_EXTENSION, cap.into());
matches!(ret, Ok(ret) if ret > 0)
}
pub fn check_extenstion_int(&self, cap: CapInt) -> i32 {
let ret = ioctl(&self.kvm, kvm_sys::KVM_CHECK_EXTENSION, cap.into());
ret.unwrap_or(0)
}
}