Go to the documentation of this file.
42 #include <linux/kvm.h>
43 #include <sys/ioctl.h>
45 #include <sys/types.h>
52 #include "debug/Kvm.hh"
54 #include "params/KvmVM.hh"
63 constexpr
int ExpectedKvmApiVersion = 12;
64 static_assert(KVM_API_VERSION == ExpectedKvmApiVersion,
65 "Unsupported KVM version");
72 : kvmFD(-1), apiVersion(-1), vcpuMMapSize(0)
74 static bool created =
false;
76 warn_once(
"Use of multiple KvmVMs is currently untested!");
80 kvmFD = ::open(
"/dev/kvm", O_RDWR);
82 fatal(
"KVM: Failed to open /dev/kvm\n");
86 fatal(
"KVM: Incompatible API version\n");
90 panic(
"KVM: Failed to get virtual CPU MMAP size\n");
128 #ifdef KVM_CAP_USER_NMI
144 #ifdef KVM_CAP_NR_MEMSLOTS
154 #ifdef KVM_CAP_ONE_REG
170 #ifdef KVM_CAP_VCPU_EVENTS
180 #ifdef KVM_CAP_DEBUGREGS
208 #if defined(__i386__) || defined(__x86_64__)
210 Kvm::getSupportedCPUID(
struct kvm_cpuid2 &
cpuid)
const
212 if (
ioctl(KVM_GET_SUPPORTED_CPUID, (
void *)&
cpuid) == -1) {
216 panic(
"KVM: Failed to get supported CPUID (errno: %i)\n", errno);
221 const Kvm::CPUIDVector &
222 Kvm::getSupportedCPUID()
const
224 if (supportedCPUIDCache.empty()) {
225 std::unique_ptr<
struct kvm_cpuid2, void(*)(
void *
p)>
226 cpuid(
nullptr, [](
void *
p) {
operator delete(
p); });
229 cpuid.reset((
struct kvm_cpuid2 *)
operator new(
230 sizeof(kvm_cpuid2) +
i *
sizeof(kvm_cpuid_entry2)));
234 }
while (!getSupportedCPUID(*
cpuid));
235 supportedCPUIDCache.assign(
cpuid->entries,
239 return supportedCPUIDCache;
243 Kvm::getSupportedMSRs(
struct kvm_msr_list &msrs)
const
245 if (
ioctl(KVM_GET_MSR_INDEX_LIST, (
void *)&msrs) == -1) {
249 panic(
"KVM: Failed to get supported CPUID (errno: %i)\n", errno);
254 const Kvm::MSRIndexVector &
255 Kvm::getSupportedMSRs()
const
257 if (supportedMSRCache.empty()) {
258 std::unique_ptr<
struct kvm_msr_list, void(*)(
void *
p)>
259 msrs(
nullptr, [](
void *
p) {
operator delete(
p); });
262 msrs.reset((
struct kvm_msr_list *)
operator new(
263 sizeof(kvm_msr_list) +
i *
sizeof(uint32_t)));
267 }
while (!getSupportedMSRs(*msrs));
268 supportedMSRCache.assign(msrs->indices, msrs->indices + msrs->nmsrs);
271 return supportedMSRCache;
274 #endif // x86-specific
280 int ret =
ioctl(KVM_CHECK_EXTENSION, extension);
282 panic(
"KVM: ioctl failed when checking for extension\n");
291 return ::ioctl(
kvmFD, request, p1);
299 vmFD =
ioctl(KVM_CREATE_VM);
301 panic(
"Failed to create KVM VM\n");
310 vmFD(kvm->createVM()),
312 _hasKernelIRQChip(false),
320 for (
int i = 0;
i <
params.coalescedMMIO.size(); ++
i)
337 if (close(
vmFD) == -1)
338 warn(
"kvm VM: notifyFork failed to close vmFD\n");
364 DPRINTF(
Kvm,
"Mapping %i memory region(s)\n", memories.size());
365 for (
int slot(0); slot < memories.size(); ++slot) {
366 if (!memories[slot].kvmMap) {
367 DPRINTF(
Kvm,
"Skipping region marked as not usable by KVM\n");
371 const AddrRange &range(memories[slot].range);
372 void *pmem(memories[slot].pmem);
375 DPRINTF(
Kvm,
"Mapping region: 0x%p -> 0x%llx [size: 0x%llx]\n",
379 panic(
"Tried to map an interleaved memory range into "
387 hack(
"KVM: Zero memory handled as IO\n");
396 panic(
"Memory slots must have non-zero size.\n");
409 panic(
"Out of memory slots.\n");
413 slot.
slot = nextSlot;
448 void *host_addr,
Addr guest_addr,
449 uint64_t
len, uint32_t flags)
451 struct kvm_userspace_memory_region
m;
453 memset(&
m, 0,
sizeof(
m));
456 m.guest_phys_addr = (uint64_t)guest_addr;
458 m.userspace_addr = (__u64)host_addr;
460 if (
ioctl(KVM_SET_USER_MEMORY_REGION, (
void *)&
m) == -1) {
461 panic(
"Failed to setup KVM memory region:\n"
462 "\tHost Address: 0x%p\n"
463 "\tGuest Address: 0x%llx\n",
466 m.userspace_addr,
m.guest_phys_addr,
467 m.memory_size,
m.flags);
480 struct kvm_coalesced_mmio_zone zone;
486 DPRINTF(
Kvm,
"KVM: Registering coalesced MMIO region [0x%x, 0x%x]\n",
487 zone.addr, zone.addr + zone.size - 1);
488 if (
ioctl(KVM_REGISTER_COALESCED_MMIO, (
void *)&zone) == -1)
489 panic(
"KVM: Failed to register coalesced MMIO region (%i)\n",
496 if (
ioctl(KVM_SET_TSS_ADDR, (
unsigned long)tss_address) == -1)
497 panic(
"KVM: Failed to set VM TSS address\n");
504 panic(
"KvmVM::createIRQChip called twice.\n");
506 if (
ioctl(KVM_CREATE_IRQCHIP) != -1) {
509 warn(
"KVM: Failed to create in-kernel IRQ chip (errno: %i)\n",
518 struct kvm_irq_level kvm_level;
521 kvm_level.level =
high ? 1 : 0;
523 if (
ioctl(KVM_IRQ_LINE, &kvm_level) == -1)
524 panic(
"KVM: Failed to set IRQ line level (errno: %i)\n",
531 #if defined(KVM_CREATE_DEVICE)
532 struct kvm_create_device dev = {
type, 0, flags };
534 if (
ioctl(KVM_CREATE_DEVICE, &dev) == -1) {
535 panic(
"KVM: Failed to create device (errno: %i)\n",
541 panic(
"Kernel headers don't support KVM_CREATE_DEVICE\n");
548 panic_if(
system !=
nullptr,
"setSystem() can only be called once");
549 panic_if(
s ==
nullptr,
"setSystem() called with null System*");
556 assert(
system !=
nullptr);
566 fd =
ioctl(KVM_CREATE_VCPU, vcpuID);
568 panic(
"KVM: Failed to create virtual CPU");
579 #if defined(__aarch64__)
581 KvmVM::kvmArmPreferredTarget(
struct kvm_vcpu_init &target)
const
583 if (
ioctl(KVM_ARM_PREFERRED_TARGET, &target) == -1) {
584 panic(
"KVM: Failed to get ARM preferred CPU target (errno: %i)\n",
595 return ::ioctl(
vmFD, request, p1);
#define fatal(...)
This implements a cprintf based fatal() function.
int vmFD
KVM VM file descriptor.
int createVCPU(long vcpuID)
Create a new vCPU within a VM.
bool capSetTSSAddress() const
Support for KvmVM::setTSSAddress()
void freeMemSlot(const MemSlot slot)
Free a previously allocated memory slot.
bool capUserNMI() const
Support for BaseKvmCPU::kvmNonMaskableInterrupt().
std::vector< BackingStoreEntry > getBackingStore() const
Get the pointers to the backing store for external host access.
void setUserMemoryRegion(uint32_t slot, void *host_addr, Addr guest_addr, uint64_t len, uint32_t flags)
Setup a region of physical memory in the guest.
static Kvm * instance
Singleton instance.
Addr start() const
Get the start address of the range.
int capCoalescedMMIO() const
Check if coalesced MMIO is supported and which page in the MMAP'ed structure it stores requests in.
void createIRQChip()
Create an in-kernel interrupt controller.
void setTSSAddress(Addr tss_address)
Setup a shared three-page memory region used by the internals of KVM.
long allocVCPUID()
Allocate a new vCPU ID within the VM.
void disableMemSlot(const MemSlot slot)
Disable a memory slot.
void setSystem(System *s)
Initialize system pointer.
bool capExtendedCPUID() const
Support for BaseKvmCPU::setCPUID2 and getSupportedCPUID().
int ioctl(int request, long p1) const
KVM VM ioctl interface.
void coalesceMMIO(Addr start, int size)
Request coalescing MMIO for a memory range.
bool capUserMemory() const
Support for KvmVM::setUserMemoryRegion()
KvmVM(const KvmVMParams ¶ms)
bool capXCRs() const
Support for getting and setting the x86 XCRs.
bool capXSave() const
Support for getting and setting the kvm_xsave structure.
bool capIRQChip() const
Support for creating an in-kernel IRQ chip model.
int checkExtension(int extension) const
Check for the presence of an extension to the KVM API.
bool capVCPUEvents() const
Support for getting and setting the kvm_vcpu_events structure.
bool interleaved() const
Determine if the range is interleaved or not.
const Params & params() const
Base class for KVM based CPU models.
int vcpuMMapSize
Size of the MMAPed vCPU parameter area.
bool _hasKernelIRQChip
Do we have in-kernel IRQ-chip emulation enabled?
int kvmFD
KVM VM file descriptor.
bool started
Has delayedStartup() already been called?
void setIRQLine(uint32_t irq, bool high)
Set the status of an IRQ line using KVM_IRQ_LINE.
Addr size() const
Get the size of the address range.
void delayedStartup()
Delayed initialization, executed once before the first CPU starts.
int ioctl(int request, long p1) const
Main VM ioctl interface.
Abstract superclass for simulation objects.
long contextIdToVCpuId(ContextID ctx) const
Get the VCPUID for a given context.
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Kvm * kvm
Global KVM interface.
void cpuStartup()
VM CPU initialization code.
Structures tracking memory slots.
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
const MemSlot allocMemSlot(uint64_t size)
Allocate a memory slot within the VM.
bool capDebugRegs() const
Support for getting and setting the kvm_debugregs structure.
long nextVCPUID
Next unallocated vCPU ID.
int ContextID
Globally unique thread context ID.
int capNumMemSlots() const
Attempt to determine how many memory slots are available.
memory::PhysicalMemory & getPhysMem()
Get a pointer to access the physical memory of the system.
The AddrRange class encapsulates an address range, and supports a number of tests to check if two ran...
int apiVersion
KVM API version.
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
void notifyFork()
Notify a child process of a fork.
std::vector< MemorySlot > memorySlots
bool capOneReg() const
Support for reading and writing single registers.
#define panic(...)
This implements a cprintf based panic() function.
int createVM()
Create a KVM Virtual Machine.
void setupMemSlot(const MemSlot slot, void *host_addr, Addr guest_addr, uint32_t flags)
Setup a region of physical memory in the guest.
int createDevice(uint32_t type, uint32_t flags=0)
Create an in-kernel device model.
Generated on Wed May 4 2022 12:13:53 for gem5 by doxygen 1.8.17