gic.cc revision 11943
1/* 2 * Copyright (c) 2015-2017 ARM Limited 3 * All rights reserved 4 * 5 * The license below extends only to copyright in the software and shall 6 * not be construed as granting a license to any other intellectual 7 * property including but not limited to intellectual property relating 8 * to a hardware implementation of the functionality of the software 9 * licensed hereunder. You may use the software subject to the license 10 * terms below provided that you ensure that this notice is replicated 11 * unmodified and in its entirety in all distributions of the software, 12 * modified or unmodified, in source code or in binary form. 13 * 14 * Redistribution and use in source and binary forms, with or without 15 * modification, are permitted provided that the following conditions are 16 * met: redistributions of source code must retain the above copyright 17 * notice, this list of conditions and the following disclaimer; 18 * redistributions in binary form must reproduce the above copyright 19 * notice, this list of conditions and the following disclaimer in the 20 * documentation and/or other materials provided with the distribution; 21 * neither the name of the copyright holders nor the names of its 22 * contributors may be used to endorse or promote products derived from 23 * this software without specific prior written permission. 24 * 25 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 26 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 27 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 28 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 29 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 30 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 31 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 32 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 33 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 34 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 35 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 36 * 37 * Authors: Andreas Sandberg 38 * Curtis Dunham 39 */ 40 41#include "arch/arm/kvm/gic.hh" 42 43#include <linux/kvm.h> 44 45#include "arch/arm/kvm/base_cpu.hh" 46#include "debug/GIC.hh" 47#include "debug/Interrupt.hh" 48#include "params/MuxingKvmGic.hh" 49 50KvmKernelGicV2::KvmKernelGicV2(KvmVM &_vm, Addr cpu_addr, Addr dist_addr, 51 unsigned it_lines) 52 : cpuRange(RangeSize(cpu_addr, KVM_VGIC_V2_CPU_SIZE)), 53 distRange(RangeSize(dist_addr, KVM_VGIC_V2_DIST_SIZE)), 54 vm(_vm), 55 kdev(vm.createDevice(KVM_DEV_TYPE_ARM_VGIC_V2)) 56{ 57 kdev.setAttr<uint64_t>( 58 KVM_DEV_ARM_VGIC_GRP_ADDR, KVM_VGIC_V2_ADDR_TYPE_DIST, dist_addr); 59 kdev.setAttr<uint64_t>( 60 KVM_DEV_ARM_VGIC_GRP_ADDR, KVM_VGIC_V2_ADDR_TYPE_CPU, cpu_addr); 61 62 kdev.setAttr<uint32_t>(KVM_DEV_ARM_VGIC_GRP_NR_IRQS, 0, it_lines); 63} 64 65KvmKernelGicV2::~KvmKernelGicV2() 66{ 67} 68 69void 70KvmKernelGicV2::setSPI(unsigned spi) 71{ 72 setIntState(KVM_ARM_IRQ_TYPE_SPI, 0, spi, true); 73} 74 75void 76KvmKernelGicV2::clearSPI(unsigned spi) 77{ 78 setIntState(KVM_ARM_IRQ_TYPE_SPI, 0, spi, false); 79} 80 81void 82KvmKernelGicV2::setPPI(unsigned vcpu, unsigned ppi) 83{ 84 setIntState(KVM_ARM_IRQ_TYPE_PPI, vcpu, ppi, true); 85} 86 87void 88KvmKernelGicV2::clearPPI(unsigned vcpu, unsigned ppi) 89{ 90 setIntState(KVM_ARM_IRQ_TYPE_PPI, vcpu, ppi, false); 91} 92 93void 94KvmKernelGicV2::setIntState(unsigned type, unsigned vcpu, unsigned irq, 95 bool high) 96{ 97 assert(type <= KVM_ARM_IRQ_TYPE_MASK); 98 assert(vcpu <= KVM_ARM_IRQ_VCPU_MASK); 99 assert(irq <= KVM_ARM_IRQ_NUM_MASK); 100 const uint32_t line( 101 (type << KVM_ARM_IRQ_TYPE_SHIFT) | 102 (vcpu << KVM_ARM_IRQ_VCPU_SHIFT) | 103 (irq << KVM_ARM_IRQ_NUM_SHIFT)); 104 105 vm.setIRQLine(line, high); 106} 107 108uint32_t 109KvmKernelGicV2::getGicReg(unsigned group, unsigned vcpu, unsigned offset) 110{ 111 uint64_t reg; 112 113 assert(vcpu <= KVM_ARM_IRQ_VCPU_MASK); 114 const uint32_t attr( 115 (vcpu << KVM_DEV_ARM_VGIC_CPUID_SHIFT) | 116 (offset << KVM_DEV_ARM_VGIC_OFFSET_SHIFT)); 117 118 kdev.getAttrPtr(group, attr, ®); 119 return (uint32_t) reg; 120} 121 122void 123KvmKernelGicV2::setGicReg(unsigned group, unsigned vcpu, unsigned offset, 124 unsigned value) 125{ 126 uint64_t reg = value; 127 128 assert(vcpu <= KVM_ARM_IRQ_VCPU_MASK); 129 const uint32_t attr( 130 (vcpu << KVM_DEV_ARM_VGIC_CPUID_SHIFT) | 131 (offset << KVM_DEV_ARM_VGIC_OFFSET_SHIFT)); 132 133 kdev.setAttrPtr(group, attr, ®); 134} 135 136uint32_t 137KvmKernelGicV2::readDistributor(ContextID ctx, Addr daddr) 138{ 139 auto vcpu = vm.contextIdToVCpuId(ctx); 140 return getGicReg(KVM_DEV_ARM_VGIC_GRP_DIST_REGS, vcpu, daddr); 141} 142 143uint32_t 144KvmKernelGicV2::readCpu(ContextID ctx, Addr daddr) 145{ 146 auto vcpu = vm.contextIdToVCpuId(ctx); 147 return getGicReg(KVM_DEV_ARM_VGIC_GRP_CPU_REGS, vcpu, daddr); 148} 149 150void 151KvmKernelGicV2::writeDistributor(ContextID ctx, Addr daddr, uint32_t data) 152{ 153 auto vcpu = vm.contextIdToVCpuId(ctx); 154 setGicReg(KVM_DEV_ARM_VGIC_GRP_DIST_REGS, vcpu, daddr, data); 155} 156 157void 158KvmKernelGicV2::writeCpu(ContextID ctx, Addr daddr, uint32_t data) 159{ 160 auto vcpu = vm.contextIdToVCpuId(ctx); 161 setGicReg(KVM_DEV_ARM_VGIC_GRP_CPU_REGS, vcpu, daddr, data); 162} 163 164 165 166MuxingKvmGic::MuxingKvmGic(const MuxingKvmGicParams *p) 167 : Pl390(p), 168 system(*p->system), 169 kernelGic(nullptr), 170 usingKvm(false) 171{ 172 if (auto vm = system.getKvmVM()) { 173 kernelGic = new KvmKernelGicV2(*vm, p->cpu_addr, p->dist_addr, 174 p->it_lines); 175 } 176} 177 178MuxingKvmGic::~MuxingKvmGic() 179{ 180} 181 182void 183MuxingKvmGic::loadState(CheckpointIn &cp) 184{ 185 Pl390::loadState(cp); 186} 187 188void 189MuxingKvmGic::startup() 190{ 191 Pl390::startup(); 192 usingKvm = (kernelGic != nullptr) && validKvmEnvironment(); 193 if (usingKvm) 194 fromPl390ToKvm(); 195} 196 197DrainState 198MuxingKvmGic::drain() 199{ 200 if (usingKvm) 201 fromKvmToPl390(); 202 return Pl390::drain(); 203} 204 205void 206MuxingKvmGic::drainResume() 207{ 208 Pl390::drainResume(); 209 bool use_kvm = (kernelGic != nullptr) && validKvmEnvironment(); 210 if (use_kvm != usingKvm) { 211 // Should only occur due to CPU switches 212 if (use_kvm) // from simulation to KVM emulation 213 fromPl390ToKvm(); 214 // otherwise, drain() already sync'd the state back to the Pl390 215 216 usingKvm = use_kvm; 217 } 218} 219 220void 221MuxingKvmGic::serialize(CheckpointOut &cp) const 222{ 223 // drain() already ensured Pl390 updated with KvmGic state if necessary 224 Pl390::serialize(cp); 225} 226 227void 228MuxingKvmGic::unserialize(CheckpointIn &cp) 229{ 230 Pl390::unserialize(cp); 231} 232 233Tick 234MuxingKvmGic::read(PacketPtr pkt) 235{ 236 if (!usingKvm) 237 return Pl390::read(pkt); 238 239 panic("MuxingKvmGic: PIO from gem5 is currently unsupported\n"); 240} 241 242Tick 243MuxingKvmGic::write(PacketPtr pkt) 244{ 245 if (!usingKvm) 246 return Pl390::write(pkt); 247 248 panic("MuxingKvmGic: PIO from gem5 is currently unsupported\n"); 249} 250 251void 252MuxingKvmGic::sendInt(uint32_t num) 253{ 254 if (!usingKvm) 255 return Pl390::sendInt(num); 256 257 DPRINTF(Interrupt, "Set SPI %d\n", num); 258 kernelGic->setSPI(num); 259} 260 261void 262MuxingKvmGic::clearInt(uint32_t num) 263{ 264 if (!usingKvm) 265 return Pl390::clearInt(num); 266 267 DPRINTF(Interrupt, "Clear SPI %d\n", num); 268 kernelGic->clearSPI(num); 269} 270 271void 272MuxingKvmGic::sendPPInt(uint32_t num, uint32_t cpu) 273{ 274 if (!usingKvm) 275 return Pl390::sendPPInt(num, cpu); 276 DPRINTF(Interrupt, "Set PPI %d:%d\n", cpu, num); 277 kernelGic->setPPI(cpu, num); 278} 279 280void 281MuxingKvmGic::clearPPInt(uint32_t num, uint32_t cpu) 282{ 283 if (!usingKvm) 284 return Pl390::clearPPInt(num, cpu); 285 286 DPRINTF(Interrupt, "Clear PPI %d:%d\n", cpu, num); 287 kernelGic->clearPPI(cpu, num); 288} 289 290bool 291MuxingKvmGic::validKvmEnvironment() const 292{ 293 if (system.threadContexts.empty()) 294 return false; 295 296 for (auto tc : system.threadContexts) { 297 if (dynamic_cast<BaseArmKvmCPU*>(tc->getCpuPtr()) == nullptr) { 298 return false; 299 } 300 } 301 return true; 302} 303 304void 305MuxingKvmGic::copyDistRegister(BaseGicRegisters* from, BaseGicRegisters* to, 306 ContextID ctx, Addr daddr) 307{ 308 auto val = from->readDistributor(ctx, daddr); 309 DPRINTF(GIC, "copy dist 0x%x 0x%08x\n", daddr, val); 310 to->writeDistributor(ctx, daddr, val); 311} 312 313void 314MuxingKvmGic::copyCpuRegister(BaseGicRegisters* from, BaseGicRegisters* to, 315 ContextID ctx, Addr daddr) 316{ 317 auto val = from->readCpu(ctx, daddr); 318 DPRINTF(GIC, "copy cpu 0x%x 0x%08x\n", daddr, val); 319 to->writeCpu(ctx, daddr, val); 320} 321 322void 323MuxingKvmGic::copyBankedDistRange(BaseGicRegisters* from, BaseGicRegisters* to, 324 Addr daddr, size_t size) 325{ 326 for (int ctx = 0; ctx < system._numContexts; ++ctx) 327 for (auto a = daddr; a < daddr + size; a += 4) 328 copyDistRegister(from, to, ctx, a); 329} 330 331void 332MuxingKvmGic::clearBankedDistRange(BaseGicRegisters* to, 333 Addr daddr, size_t size) 334{ 335 for (int ctx = 0; ctx < system._numContexts; ++ctx) 336 for (auto a = daddr; a < daddr + size; a += 4) 337 to->writeDistributor(ctx, a, 0xFFFFFFFF); 338} 339 340void 341MuxingKvmGic::copyDistRange(BaseGicRegisters* from, BaseGicRegisters* to, 342 Addr daddr, size_t size) 343{ 344 for (auto a = daddr; a < daddr + size; a += 4) 345 copyDistRegister(from, to, 0, a); 346} 347 348void 349MuxingKvmGic::clearDistRange(BaseGicRegisters* to, 350 Addr daddr, size_t size) 351{ 352 for (auto a = daddr; a < daddr + size; a += 4) 353 to->writeDistributor(0, a, 0xFFFFFFFF); 354} 355 356void 357MuxingKvmGic::copyGicState(BaseGicRegisters* from, BaseGicRegisters* to) 358{ 359 Addr set, clear; 360 size_t size; 361 362 /// CPU state (GICC_*) 363 // Copy CPU Interface Control Register (CTLR), 364 // Interrupt Priority Mask Register (PMR), and 365 // Binary Point Register (BPR) 366 for (int ctx = 0; ctx < system._numContexts; ++ctx) { 367 copyCpuRegister(from, to, ctx, GICC_CTLR); 368 copyCpuRegister(from, to, ctx, GICC_PMR); 369 copyCpuRegister(from, to, ctx, GICC_BPR); 370 } 371 372 373 /// Distributor state (GICD_*) 374 // Copy Distributor Control Register (CTLR) 375 copyDistRegister(from, to, 0, GICD_CTLR); 376 377 // Copy interrupt-enabled statuses (I[CS]ENABLERn; R0 is per-CPU banked) 378 set = Pl390::GICD_ISENABLER.start(); 379 clear = Pl390::GICD_ICENABLER.start(); 380 size = Pl390::itLines / 8; 381 clearBankedDistRange(to, clear, 4); 382 copyBankedDistRange(from, to, set, 4); 383 384 set += 4, clear += 4, size -= 4; 385 clearDistRange(to, clear, size); 386 copyDistRange(from, to, set, size); 387 388 // Copy pending interrupts (I[CS]PENDRn; R0 is per-CPU banked) 389 set = Pl390::GICD_ISPENDR.start(); 390 clear = Pl390::GICD_ICPENDR.start(); 391 size = Pl390::itLines / 8; 392 clearBankedDistRange(to, clear, 4); 393 copyBankedDistRange(from, to, set, 4); 394 395 set += 4, clear += 4, size -= 4; 396 clearDistRange(to, clear, size); 397 copyDistRange(from, to, set, size); 398 399 // Copy active interrupts (I[CS]ACTIVERn; R0 is per-CPU banked) 400 set = Pl390::GICD_ISACTIVER.start(); 401 clear = Pl390::GICD_ICACTIVER.start(); 402 size = Pl390::itLines / 8; 403 clearBankedDistRange(to, clear, 4); 404 copyBankedDistRange(from, to, set, 4); 405 406 set += 4, clear += 4, size -= 4; 407 clearDistRange(to, clear, size); 408 copyDistRange(from, to, set, size); 409 410 // Copy interrupt priorities (IPRIORITYRn; R0-7 are per-CPU banked) 411 set = Pl390::GICD_IPRIORITYR.start(); 412 copyBankedDistRange(from, to, set, 32); 413 414 set += 32; 415 size = Pl390::itLines - 32; 416 copyDistRange(from, to, set, size); 417 418 // Copy interrupt processor target regs (ITARGETRn; R0-7 are read-only) 419 set = Pl390::GICD_ITARGETSR.start() + 32; 420 size = Pl390::itLines - 32; 421 copyDistRange(from, to, set, size); 422 423 // Copy interrupt configuration registers (ICFGRn) 424 set = Pl390::GICD_ICFGR.start(); 425 size = Pl390::itLines / 4; 426 copyDistRange(from, to, set, size); 427} 428 429void 430MuxingKvmGic::fromPl390ToKvm() 431{ 432 copyGicState(static_cast<Pl390*>(this), kernelGic); 433} 434 435void 436MuxingKvmGic::fromKvmToPl390() 437{ 438 copyGicState(kernelGic, static_cast<Pl390*>(this)); 439 440 // the values read for the Interrupt Priority Mask Register (PMR) 441 // have been shifted by three bits due to its having been emulated by 442 // a VGIC with only 5 PMR bits in its VMCR register. Presently the 443 // Linux kernel does not repair this inaccuracy, so we correct it here. 444 for (int cpu = 0; cpu < system._numContexts; ++cpu) { 445 cpuPriority[cpu] <<= 3; 446 assert((cpuPriority[cpu] & ~0xff) == 0); 447 } 448} 449 450MuxingKvmGic * 451MuxingKvmGicParams::create() 452{ 453 return new MuxingKvmGic(this); 454} 455