#include "kvmxx.hh" #include "exception.hh" #include #include #include #include #include #include namespace kvm { static long check_error(long r) { if (r == -1) { throw errno_exception(errno); } return r; } fd::fd(int fd) : _fd(fd) { } fd::fd(const fd& other) : _fd(::dup(other._fd)) { check_error(_fd); } fd::fd(std::string device_node, int flags) : _fd(::open(device_node.c_str(), flags)) { check_error(_fd); } long fd::ioctl(unsigned nr, long arg) { return check_error(::ioctl(_fd, nr, arg)); } vcpu::vcpu(vm& vm, int id) : _vm(vm), _fd(vm._fd.ioctl(KVM_CREATE_VCPU, id)), _shared(NULL) , _mmap_size(_vm._system._fd.ioctl(KVM_GET_VCPU_MMAP_SIZE, 0)) { kvm_run *shared = static_cast(::mmap(NULL, _mmap_size, PROT_READ | PROT_WRITE, MAP_SHARED, _fd.get(), 0)); if (shared == MAP_FAILED) { throw errno_exception(errno); } _shared = shared; } vcpu::~vcpu() { munmap(_shared, _mmap_size); } void vcpu::run() { _fd.ioctl(KVM_RUN, 0); } kvm_regs vcpu::regs() { kvm_regs regs; _fd.ioctlp(KVM_GET_REGS, ®s); return regs; } void vcpu::set_regs(const kvm_regs& regs) { _fd.ioctlp(KVM_SET_REGS, const_cast(®s)); } kvm_sregs vcpu::sregs() { kvm_sregs sregs; _fd.ioctlp(KVM_GET_SREGS, &sregs); return sregs; } void vcpu::set_sregs(const kvm_sregs& sregs) { _fd.ioctlp(KVM_SET_SREGS, const_cast(&sregs)); } class vcpu::kvm_msrs_ptr { public: explicit kvm_msrs_ptr(size_t nmsrs); ~kvm_msrs_ptr() { ::free(_kvm_msrs); } kvm_msrs* operator->() { return _kvm_msrs; } kvm_msrs* get() { return _kvm_msrs; } private: kvm_msrs* _kvm_msrs; }; vcpu::kvm_msrs_ptr::kvm_msrs_ptr(size_t nmsrs) : _kvm_msrs(0) { size_t size = sizeof(kvm_msrs) + sizeof(kvm_msr_entry) * nmsrs; _kvm_msrs = static_cast(::malloc(size)); if (!_kvm_msrs) { throw std::bad_alloc(); } } std::vector vcpu::msrs(std::vector indices) { kvm_msrs_ptr msrs(indices.size()); msrs->nmsrs = indices.size(); for (unsigned i = 0; i < msrs->nmsrs; ++i) { msrs->entries[i].index = indices[i]; } _fd.ioctlp(KVM_GET_MSRS, msrs.get()); return std::vector(msrs->entries, msrs->entries + msrs->nmsrs); } void vcpu::set_msrs(const std::vector& msrs) { kvm_msrs_ptr _msrs(msrs.size()); _msrs->nmsrs = msrs.size(); std::copy(msrs.begin(), msrs.end(), _msrs->entries); _fd.ioctlp(KVM_SET_MSRS, _msrs.get()); } void vcpu::set_debug(uint64_t dr[8], bool enabled, bool singlestep) { kvm_guest_debug gd; gd.control = 0; if (enabled) { gd.control |= KVM_GUESTDBG_ENABLE; } if (singlestep) { gd.control |= KVM_GUESTDBG_SINGLESTEP; } for (int i = 0; i < 8; ++i) { gd.arch.debugreg[i] = dr[i]; } _fd.ioctlp(KVM_SET_GUEST_DEBUG, &gd); } vm::vm(system& system) : _system(system), _fd(system._fd.ioctl(KVM_CREATE_VM, 0)) { } void vm::set_memory_region(int slot, void *addr, uint64_t gpa, size_t len, uint32_t flags) { struct kvm_userspace_memory_region umr; umr.slot = slot; umr.flags = flags; umr.guest_phys_addr = gpa; umr.memory_size = len; umr.userspace_addr = reinterpret_cast(addr); _fd.ioctlp(KVM_SET_USER_MEMORY_REGION, &umr); } void vm::get_dirty_log(int slot, void *log) { struct kvm_dirty_log kdl; kdl.slot = slot; kdl.dirty_bitmap = log; _fd.ioctlp(KVM_GET_DIRTY_LOG, &kdl); } void vm::set_tss_addr(uint32_t addr) { _fd.ioctl(KVM_SET_TSS_ADDR, addr); } void vm::set_ept_identity_map_addr(uint64_t addr) { _fd.ioctlp(KVM_SET_IDENTITY_MAP_ADDR, &addr); } system::system(std::string device_node) : _fd(device_node, O_RDWR) { } bool system::check_extension(int extension) { return _fd.ioctl(KVM_CHECK_EXTENSION, extension); } int system::get_extension_int(int extension) { return _fd.ioctl(KVM_CHECK_EXTENSION, extension); } };