diff options
Diffstat (limited to 'src/cpu/kvm/vm.cc')
-rw-r--r-- | src/cpu/kvm/vm.cc | 80 |
1 files changed, 71 insertions, 9 deletions
diff --git a/src/cpu/kvm/vm.cc b/src/cpu/kvm/vm.cc index 5ce42a5ac..4ef11b2ee 100644 --- a/src/cpu/kvm/vm.cc +++ b/src/cpu/kvm/vm.cc @@ -1,4 +1,5 @@ /* + * Copyright 2014 Google, Inc. * Copyright (c) 2012 ARM Limited * All rights reserved * @@ -124,6 +125,16 @@ Kvm::capCoalescedMMIO() const return checkExtension(KVM_CAP_COALESCED_MMIO); } +int +Kvm::capNumMemSlots() const +{ +#ifdef KVM_CAP_NR_MEMSLOTS + return checkExtension(KVM_CAP_NR_MEMSLOTS); +#else + return 0; +#endif +} + bool Kvm::capOneReg() const { @@ -288,6 +299,10 @@ KvmVM::KvmVM(KvmVMParams *params) started(false), nextVCPUID(0) { + maxMemorySlot = kvm.capNumMemSlots(); + /* If we couldn't determine how memory slots there are, guess 32. */ + if (!maxMemorySlot) + maxMemorySlot = 32; /* Setup the coalesced MMIO regions */ for (int i = 0; i < params->coalescedMMIO.size(); ++i) coalesceMMIO(params->coalescedMMIO[i]); @@ -323,7 +338,13 @@ KvmVM::delayedStartup() DPRINTF(Kvm, "Mapping region: 0x%p -> 0x%llx [size: 0x%llx]\n", pmem, range.start(), range.size()); - setUserMemoryRegion(slot, pmem, range, 0 /* flags */); + if (range.interleaved()) { + panic("Tried to map an interleaved memory range into " + "a KVM VM.\n"); + } + + const MemSlot slot = allocMemSlot(range.size()); + setupMemSlot(slot, pmem, range.start(), 0/* flags */); } else { DPRINTF(Kvm, "Zero-region not mapped: [0x%llx]\n", range.start()); hack("KVM: Zero memory handled as IO\n"); @@ -331,17 +352,58 @@ KvmVM::delayedStartup() } } +const KvmVM::MemSlot +KvmVM::allocMemSlot(uint64_t size) +{ + if (!size) + panic("Memory slots must have non-zero size.\n"); + + std::vector<MemorySlot>::iterator pos; + for (pos = memorySlots.begin(); pos != memorySlots.end(); pos++) { + if (!pos->size) { + pos->size = size; + pos->active = false; + return pos->slot; + } + } + + uint32_t nextSlot = memorySlots.size(); + if (nextSlot > maxMemorySlot) + panic("Out of memory slots.\n"); + + MemorySlot slot; + slot.size = size; + slot.slot = nextSlot; + slot.active = false; + + memorySlots.push_back(slot); + return MemSlot(slot.slot); +} + void -KvmVM::setUserMemoryRegion(uint32_t slot, - void *host_addr, AddrRange guest_range, - uint32_t flags) +KvmVM::setupMemSlot(const KvmVM::MemSlot num, void *host_addr, Addr guest, + uint32_t flags) +{ + MemorySlot &slot = memorySlots.at(num.num); + slot.active = true; + setUserMemoryRegion(num.num, host_addr, guest, slot.size, flags); +} + +void +KvmVM::disableMemSlot(const KvmVM::MemSlot num) { - if (guest_range.interleaved()) - panic("Tried to map an interleaved memory range into a KVM VM.\n"); + MemorySlot &slot = memorySlots.at(num.num); + if (slot.active) + setUserMemoryRegion(num.num, NULL, 0, 0, 0); + slot.active = false; +} - setUserMemoryRegion(slot, host_addr, - guest_range.start(), guest_range.size(), - flags); +void +KvmVM::freeMemSlot(const KvmVM::MemSlot num) +{ + disableMemSlot(num.num); + MemorySlot &slot = memorySlots.at(num.num); + slot.size = 0; } void |