Go to the documentation of this file.
40 #include <linux/kvm.h>
45 #include "debug/KvmInt.hh"
47 #include "params/BaseArmKvmCPU.hh"
48 #include "params/GenericTimer.hh"
53 using namespace ArmISA;
55 #define INTERRUPT_ID(type, vcpu, irq) ( \
56 ((type) << KVM_ARM_IRQ_TYPE_SHIFT) | \
57 ((vcpu) << KVM_ARM_IRQ_VCPU_SHIFT) | \
58 ((irq) << KVM_ARM_IRQ_NUM_SHIFT))
60 #define INTERRUPT_VCPU_IRQ(vcpu) \
61 INTERRUPT_ID(KVM_ARM_IRQ_TYPE_CPU, vcpu, KVM_ARM_IRQ_CPU_IRQ)
63 #define INTERRUPT_VCPU_FIQ(vcpu) \
64 INTERRUPT_ID(KVM_ARM_IRQ_TYPE_CPU, vcpu, KVM_ARM_IRQ_CPU_FIQ)
77 uint64_t vtime_counter = 0;
78 UncontendedMutex vtime_mutex;
84 irqAsserted(false), fiqAsserted(false),
85 virtTimerPin(nullptr), prevDeviceIRQLevel(0)
102 struct kvm_vcpu_init target_config;
103 memset(&target_config, 0,
sizeof(target_config));
105 vm.kvmArmPreferredTarget(target_config);
107 target_config.features[0] |= (1 << KVM_ARM_VCPU_EL1_32BIT);
113 ->getGenericTimer()->
params().int_virt->get(
tc);
120 const bool simFIQ(interrupt->checkRaw(
INT_FIQ));
121 const bool simIRQ(interrupt->checkRaw(
INT_IRQ));
125 DPRINTF(KvmInt,
"KVM: Update FIQ state: %i\n", simFIQ);
129 DPRINTF(KvmInt,
"KVM: Update IRQ state: %i\n", simIRQ);
134 "FIQ raised by the simulated interrupt controller " \
135 "despite in-kernel GIC emulation. This is probably a bug.");
138 "IRQ raised by the simulated interrupt controller " \
139 "despite in-kernel GIC emulation. This is probably a bug.");
148 uint64_t device_irq_level =
152 (device_irq_level & KVM_ARM_DEV_EL1_VTIMER)) {
154 DPRINTF(KvmInt,
"In-kernel vtimer IRQ asserted\n");
159 !(device_irq_level & KVM_ARM_DEV_EL1_VTIMER)) {
161 DPRINTF(KvmInt,
"In-kernel vtimer IRQ disasserted\n");
176 std::lock_guard<UncontendedMutex>
l(vtime_mutex);
177 if (vtime_counter++ == 0)
184 std::lock_guard<UncontendedMutex>
l(vtime_mutex);
185 if (--vtime_counter == 0)
186 getOneReg(KVM_REG_ARM_TIMER_CNT, &vtime);
198 kvm_reg_list regs_probe;
204 std::unique_ptr<kvm_reg_list> regs;
205 const size_t size(
sizeof(kvm_reg_list) +
206 regs_probe.n *
sizeof(uint64_t));
207 regs.reset((kvm_reg_list *)
operator new(size));
208 regs->n = regs_probe.n;
210 panic(
"Failed to determine register list size.\n");
221 if (
ioctl(KVM_ARM_VCPU_INIT, (
void *)&
init) == -1)
222 panic(
"KVM: Failed to initialize vCPU\n");
228 if (
ioctl(KVM_GET_REG_LIST, (
void *)®s) == -1) {
229 if (errno == E2BIG) {
232 panic(
"KVM: Failed to get vCPU register list (errno: %i)\n",
std::vector< BaseInterrupts * > interrupts
void kvmArmVCpuInit(const kvm_vcpu_init &init)
Tell the kernel to initialize this CPU.
int ioctl(int request, long p1) const
vCPU ioctl interface.
BaseArmKvmCPU(const BaseArmKvmCPUParams ¶ms)
Tick kvmRun(Tick ticks) override
Request KVM to run the guest for a given number of ticks.
struct kvm_run * getKvmRunState()
Get a pointer to the kvm_run structure containing all the input and output parameters from kvmRun().
bool irqAsserted
Cached state of the IRQ line.
void getOneReg(uint64_t id, void *addr) const
void startup() override
startup() is the final initialization call before simulation.
virtual void ioctlRun()
Execute the KVM_RUN ioctl.
const Params & params() const
Base class for KVM based CPU models.
ThreadContext * tc
ThreadContext object, provides an interface for external objects to modify this thread's state.
bool fiqAsserted
Cached state of the FIQ line.
uint64_t Tick
Tick count type.
virtual void clear()=0
Clear a signalled interrupt.
#define INTERRUPT_VCPU_IRQ(vcpu)
void setOneReg(uint64_t id, const void *addr)
Get/Set single register using the KVM_(SET|GET)_ONE_REG API.
void setIRQLine(uint32_t irq, bool high)
Set the status of an IRQ line using KVM_IRQ_LINE.
const long vcpuID
KVM internal ID of the vCPU.
RegIndexVector _regIndexList
Cached copy of the list of registers supported by KVM.
uint64_t prevDeviceIRQLevel
KVM records whether each in-kernel device IRQ is asserted or disasserted in the kvmRunState->s....
#define warn_if(cond,...)
Conditional warning macro that checks the supplied condition and only prints a warning if the conditi...
ArmInterruptPin * virtTimerPin
If the user-space GIC and the kernel-space timer are used simultaneously, set up this interrupt pin t...
const RegIndexVector & getRegList() const
Get a list of registers supported by getOneReg() and setOneReg().
const FlagsType init
This Stat is Initialized.
virtual Tick kvmRun(Tick ticks)
Request KVM to run the guest for a given number of ticks.
virtual void raise()=0
Signal an interrupt.
void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
void startup() override
startup() is the final initialization call before simulation.
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
#define INTERRUPT_VCPU_FIQ(vcpu)
void ioctlRun() override
Override for synchronizing state in kvm_run.
bool hasKernelIRQChip() const
Is in-kernel IRQ chip emulation enabled?
#define panic(...)
This implements a cprintf based panic() function.
Generated on Wed Jul 28 2021 12:10:21 for gem5 by doxygen 1.8.17