diff options
Diffstat (limited to 'src/dev/arm')
-rw-r--r-- | src/dev/arm/gic_v3.cc | 30 | ||||
-rw-r--r-- | src/dev/arm/gic_v3.hh | 60 | ||||
-rw-r--r-- | src/dev/arm/gic_v3_distributor.cc | 157 | ||||
-rw-r--r-- | src/dev/arm/gic_v3_distributor.hh | 74 | ||||
-rw-r--r-- | src/dev/arm/gic_v3_redistributor.cc | 53 | ||||
-rw-r--r-- | src/dev/arm/gic_v3_redistributor.hh | 62 |
6 files changed, 217 insertions, 219 deletions
diff --git a/src/dev/arm/gic_v3.cc b/src/dev/arm/gic_v3.cc index 2fb90ea49..a6bd3d8fd 100644 --- a/src/dev/arm/gic_v3.cc +++ b/src/dev/arm/gic_v3.cc @@ -45,10 +45,6 @@ Gicv3::Gicv3(const Params * p) { } -Gicv3::~Gicv3() -{ -} - void Gicv3::init() { @@ -112,7 +108,7 @@ Gicv3::read(PacketPtr pkt) "Invalid redistributor_id!"); panic_if(!redistributors[redistributor_id], "Redistributor is null!"); resp = redistributors[redistributor_id]->read(daddr, size, - is_secure_access); + is_secure_access); delay = params()->redist_pio_delay; DPRINTF(GIC, "Gicv3::read(): (redistributor %d) context_id %d " "register %#x size %d is_secure_access %d (value %#x)\n", @@ -179,7 +175,7 @@ Gicv3::sendInt(uint32_t int_id) void Gicv3::clearInt(uint32_t number) { - distributor->intDeasserted(number); + distributor->deassertSPI(number); } void @@ -201,7 +197,7 @@ Gicv3::clearPPInt(uint32_t num, uint32_t cpu) void Gicv3::postInt(uint32_t cpu, ArmISA::InterruptTypes int_type) { - postDelayedInt(cpu, int_type); + platform->intrctrl->post(cpu, int_type, 0); } void @@ -210,14 +206,8 @@ Gicv3::deassertInt(uint32_t cpu, ArmISA::InterruptTypes int_type) platform->intrctrl->clear(cpu, int_type, 0); } -void -Gicv3::postDelayedInt(uint32_t cpu, ArmISA::InterruptTypes int_type) -{ - platform->intrctrl->post(cpu, int_type, 0); -} - Gicv3Redistributor * -Gicv3::getRedistributorByAffinity(uint32_t affinity) +Gicv3::getRedistributorByAffinity(uint32_t affinity) const { for (auto & redistributor : redistributors) { if (redistributor->getAffinity() == affinity) { @@ -234,14 +224,12 @@ Gicv3::serialize(CheckpointOut & cp) const distributor->serializeSection(cp, "distributor"); for (uint32_t redistributor_id = 0; - redistributor_id < redistributors.size(); - redistributor_id++) + redistributor_id < redistributors.size(); redistributor_id++) redistributors[redistributor_id]->serializeSection(cp, csprintf("redistributors.%i", redistributor_id)); for (uint32_t cpu_interface_id = 0; - cpu_interface_id < cpuInterfaces.size(); - cpu_interface_id++) + cpu_interface_id < cpuInterfaces.size(); cpu_interface_id++) cpuInterfaces[cpu_interface_id]->serializeSection(cp, csprintf("cpuInterface.%i", cpu_interface_id)); } @@ -254,14 +242,12 @@ Gicv3::unserialize(CheckpointIn & cp) distributor->unserializeSection(cp, "distributor"); for (uint32_t redistributor_id = 0; - redistributor_id < redistributors.size(); - redistributor_id++) + redistributor_id < redistributors.size(); redistributor_id++) redistributors[redistributor_id]->unserializeSection(cp, csprintf("redistributors.%i", redistributor_id)); for (uint32_t cpu_interface_id = 0; - cpu_interface_id < cpuInterfaces.size(); - cpu_interface_id++) + cpu_interface_id < cpuInterfaces.size(); cpu_interface_id++) cpuInterfaces[cpu_interface_id]->unserializeSection(cp, csprintf("cpuInterface.%i", cpu_interface_id)); } diff --git a/src/dev/arm/gic_v3.hh b/src/dev/arm/gic_v3.hh index a24263969..e38d9ba03 100644 --- a/src/dev/arm/gic_v3.hh +++ b/src/dev/arm/gic_v3.hh @@ -34,14 +34,15 @@ #include "dev/arm/base_gic.hh" #include "params/Gicv3.hh" +class Gicv3CPUInterface; class Gicv3Distributor; class Gicv3Redistributor; -class Gicv3CPUInterface; class Gicv3 : public BaseGic { protected: + typedef Gicv3Params Params; Gicv3Distributor * distributor; std::vector<Gicv3Redistributor *> redistributors; std::vector<Gicv3CPUInterface *> cpuInterfaces; @@ -51,7 +52,7 @@ class Gicv3 : public BaseGic public: - // Special interrupt IDs + // Special interrupt IDs, as per SPEC 2.2.1 section static const int INTID_SECURE = 1020; static const int INTID_NONSECURE = 1021; static const int INTID_SPURIOUS = 1023; @@ -61,6 +62,7 @@ class Gicv3 : public BaseGic // Number of Private Peripheral Interrupts static const int PPI_MAX = 16; + // Interrupt states for PPIs, SGIs and SPIs, as per SPEC 4.1.2 section typedef enum { INT_INACTIVE, INT_PENDING, @@ -68,6 +70,7 @@ class Gicv3 : public BaseGic INT_ACTIVE_PENDING, } IntStatus; + // Interrupt groups, as per SPEC section 4.6 typedef enum { G0S, G1S, @@ -79,49 +82,53 @@ class Gicv3 : public BaseGic INT_EDGE_TRIGGERED, } IntTriggerType; - typedef Gicv3Params Params; + protected: - const Params * - params() const + void clearInt(uint32_t int_id) override; + void clearPPInt(uint32_t int_id, uint32_t cpu) override; + + inline AddrRangeList + getAddrRanges() const override { - return dynamic_cast<const Params *>(_params); + return addrRanges; } - Gicv3(const Params * p); - ~Gicv3(); void init() override; void initState() override; - AddrRangeList - getAddrRanges() const override + const Params * + params() const { - return addrRanges; + return dynamic_cast<const Params *>(_params); } Tick read(PacketPtr pkt) override; - Tick write(PacketPtr pkt) override; + void reset(); void sendInt(uint32_t int_id) override; - void clearInt(uint32_t int_id) override; void sendPPInt(uint32_t int_id, uint32_t cpu) override; - void clearPPInt(uint32_t int_id, uint32_t cpu) override; - void serialize(CheckpointOut & cp) const override; void unserialize(CheckpointIn & cp) override; + Tick write(PacketPtr pkt) override; - Gicv3Distributor * - getDistributor() const - { - return distributor; - } + public: - Gicv3CPUInterface * + Gicv3(const Params * p); + void deassertInt(uint32_t cpu, ArmISA::InterruptTypes int_type); + + inline Gicv3CPUInterface * getCPUInterface(int cpu_id) const { assert(cpu_id < cpuInterfaces.size() and cpuInterfaces[cpu_id]); return cpuInterfaces[cpu_id]; } - Gicv3Redistributor * + inline Gicv3Distributor * + getDistributor() const + { + return distributor; + } + + inline Gicv3Redistributor * getRedistributor(ContextID context_id) const { assert(context_id < redistributors.size() and @@ -129,14 +136,9 @@ class Gicv3 : public BaseGic return redistributors[context_id]; } - Gicv3Redistributor * getRedistributorByAffinity(uint32_t affinity); + Gicv3Redistributor * + getRedistributorByAffinity(uint32_t affinity) const; void postInt(uint32_t cpu, ArmISA::InterruptTypes int_type); - void postDelayedInt(uint32_t cpu, ArmISA::InterruptTypes int_type); - void deassertInt(uint32_t cpu, ArmISA::InterruptTypes int_type); - - protected: - - void reset(); }; #endif //__DEV_ARM_GICV3_H__ diff --git a/src/dev/arm/gic_v3_distributor.cc b/src/dev/arm/gic_v3_distributor.cc index 5eee07d87..148de5a19 100644 --- a/src/dev/arm/gic_v3_distributor.cc +++ b/src/dev/arm/gic_v3_distributor.cc @@ -27,6 +27,7 @@ * * Authors: Jairo Balart */ + #include "dev/arm/gic_v3_distributor.hh" #include <algorithm> @@ -36,21 +37,21 @@ #include "dev/arm/gic_v3_cpu_interface.hh" #include "dev/arm/gic_v3_redistributor.hh" -const AddrRange Gicv3Distributor::GICD_IGROUPR(0x0080, 0x00ff); -const AddrRange Gicv3Distributor::GICD_ISENABLER(0x0100, 0x017f); -const AddrRange Gicv3Distributor::GICD_ICENABLER(0x0180, 0x01ff); -const AddrRange Gicv3Distributor::GICD_ISPENDR(0x0200, 0x027f); -const AddrRange Gicv3Distributor::GICD_ICPENDR(0x0280, 0x02ff); -const AddrRange Gicv3Distributor::GICD_ISACTIVER(0x0300, 0x037f); -const AddrRange Gicv3Distributor::GICD_ICACTIVER(0x0380, 0x03ff); +const AddrRange Gicv3Distributor::GICD_IGROUPR (0x0080, 0x00ff); +const AddrRange Gicv3Distributor::GICD_ISENABLER (0x0100, 0x017f); +const AddrRange Gicv3Distributor::GICD_ICENABLER (0x0180, 0x01ff); +const AddrRange Gicv3Distributor::GICD_ISPENDR (0x0200, 0x027f); +const AddrRange Gicv3Distributor::GICD_ICPENDR (0x0280, 0x02ff); +const AddrRange Gicv3Distributor::GICD_ISACTIVER (0x0300, 0x037f); +const AddrRange Gicv3Distributor::GICD_ICACTIVER (0x0380, 0x03ff); const AddrRange Gicv3Distributor::GICD_IPRIORITYR(0x0400, 0x07ff); -const AddrRange Gicv3Distributor::GICD_ITARGETSR(0x0800, 0x08ff); -const AddrRange Gicv3Distributor::GICD_ICFGR(0x0c00, 0x0cff); -const AddrRange Gicv3Distributor::GICD_IGRPMODR(0x0d00, 0x0d7f); -const AddrRange Gicv3Distributor::GICD_NSACR(0x0e00, 0x0eff); -const AddrRange Gicv3Distributor::GICD_CPENDSGIR(0x0f10, 0x0f1f); -const AddrRange Gicv3Distributor::GICD_SPENDSGIR(0x0f20, 0x0f2f); -const AddrRange Gicv3Distributor::GICD_IROUTER(0x6000, 0x7fe0); +const AddrRange Gicv3Distributor::GICD_ITARGETSR (0x0800, 0x08ff); +const AddrRange Gicv3Distributor::GICD_ICFGR (0x0c00, 0x0cff); +const AddrRange Gicv3Distributor::GICD_IGRPMODR (0x0d00, 0x0d7f); +const AddrRange Gicv3Distributor::GICD_NSACR (0x0e00, 0x0eff); +const AddrRange Gicv3Distributor::GICD_CPENDSGIR (0x0f10, 0x0f1f); +const AddrRange Gicv3Distributor::GICD_SPENDSGIR (0x0f20, 0x0f2f); +const AddrRange Gicv3Distributor::GICD_IROUTER (0x6000, 0x7fe0); Gicv3Distributor::Gicv3Distributor(Gicv3 * gic, uint32_t it_lines) : gic(gic), @@ -68,10 +69,6 @@ Gicv3Distributor::Gicv3Distributor(Gicv3 * gic, uint32_t it_lines) panic_if(it_lines > Gicv3::INTID_SECURE, "Invalid value for it_lines!"); } -Gicv3Distributor::~Gicv3Distributor() -{ -} - void Gicv3Distributor::init() { @@ -132,13 +129,13 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) } for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { val |= irqGroup[int_id] << i; } return val; - // Interrupt Set-Enable Registers } else if (GICD_ISENABLER.contains(addr)) { + // Interrupt Set-Enable Registers uint64_t val = 0x0; int first_intid = (addr - GICD_ISENABLER.start()) * 8; @@ -147,7 +144,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) } for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { continue; @@ -167,7 +165,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) } for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { continue; @@ -178,6 +177,7 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) return val; } else if (GICD_ISPENDR.contains(addr)) { + // Interrupt Set-Pending Registers uint64_t val = 0x0; int first_intid = (addr - GICD_ISPENDR.start()) * 8; @@ -186,7 +186,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) } for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { if (irqNsacr[int_id] == 0) { @@ -200,6 +201,7 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) return val; } else if (GICD_ICPENDR.contains(addr)) { + // Interrupt Clear-Pending Registers uint64_t val = 0x0; int first_intid = (addr - GICD_ICPENDR.start()) * 8; @@ -208,7 +210,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) } for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { if (irqNsacr[int_id] < 2) { @@ -221,8 +224,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) } return val; - // Interrupt Set-Active Registers } else if (GICD_ISACTIVER.contains(addr)) { + // Interrupt Set-Active Registers int first_intid = (addr - GICD_ISACTIVER.start()) * 8; if (isNotSPI(first_intid)) { @@ -232,7 +235,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) uint64_t val = 0x0; for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { // Group 0 or Secure Group 1 interrupts are RAZ/WI @@ -245,8 +249,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) } return val; - // Interrupt Clear-Active Registers } else if (GICD_ICACTIVER.contains(addr)) { + // Interrupt Clear-Active Registers int first_intid = (addr - GICD_ICACTIVER.start()) * 8; if (isNotSPI(first_intid)) { @@ -256,7 +260,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) uint64_t val = 0x0; for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { if (irqNsacr[int_id] < 2) { @@ -268,8 +273,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) } return val; - // Interrupt Priority Registers } else if (GICD_IPRIORITYR.contains(addr)) { + // Interrupt Priority Registers uint64_t val = 0x0; int first_intid = addr - GICD_IPRIORITYR.start(); @@ -278,7 +283,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) } for (int i = 0, int_id = first_intid; i < size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + uint8_t prio = irqPriority[int_id]; if (!DS && !is_secure_access) { @@ -301,8 +307,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) warn("Gicv3Distributor::read(): " "GICD_ITARGETSR is RAZ/WI, legacy not supported!\n"); return 0; - // Interrupt Configuration Registers } else if (GICD_ICFGR.contains(addr)) { + // Interrupt Configuration Registers int first_intid = (addr - GICD_ICFGR.start()) * 4; if (isNotSPI(first_intid)) { @@ -312,7 +318,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) uint64_t val = 0x0; for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i = i + 2, int_id++) { + i = i + 2, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { continue; @@ -343,16 +350,15 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) uint64_t val = 0x0; for (int i = 0, int_id = first_intid; - i < 8 * size && int_id < itLines; i++, int_id++) { + i < 8 * size && int_id < itLines; i++, int_id++) { val |= irqGrpmod[int_id] << i; } return val; } } - - // Non-secure Access Control Registers } else if (GICD_NSACR.contains(addr)) { + // Non-secure Access Control Registers // 2 bits per interrupt int first_intid = (addr - GICD_NSACR.start()) * 4; @@ -367,7 +373,7 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) uint64_t val = 0x0; for (int i = 0, int_id = first_intid; - i < 8 * size && int_id < itLines; i = i + 2, int_id++) { + i < 8 * size && int_id < itLines; i = i + 2, int_id++) { val |= irqNsacr[int_id] << i; } @@ -479,10 +485,8 @@ Gicv3Distributor::read(Addr addr, size_t size, bool is_secure_access) // Optional register, RAZ/WI return 0x0; - case GICD_PIDR0: { // Peripheral ID0 Register - uint8_t part_0 = 0x92; // Part number, bits[7:0] - return part_0; - } + case GICD_PIDR0: // Peripheral ID0 Register + return 0x92; // Part number, bits[7:0] case GICD_PIDR1: { // Peripheral ID1 Register uint8_t des_0 = 0xB; // JEP106 identification code, bits[3:0] @@ -534,15 +538,15 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, } for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { irqGroup[int_id] = data & (1 << i) ? 1 : 0; DPRINTF(GIC, "Gicv3Distributor::write(): int_id %d group %d\n", int_id, irqGroup[int_id]); } return; - // Interrupt Set-Enable Registers } else if (GICD_ISENABLER.contains(addr)) { + // Interrupt Set-Enable Registers int first_intid = (addr - GICD_ISENABLER.start()) * 8; if (isNotSPI(first_intid)) { @@ -550,7 +554,8 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, } for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { continue; @@ -574,7 +579,8 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, int first_intid = (addr - GICD_ICENABLER.start()) * 8; for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { continue; @@ -594,6 +600,7 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, return; } else if (GICD_ISPENDR.contains(addr)) { + // Interrupt Set-Pending Registers int first_intid = (addr - GICD_ISPENDR.start()) * 8; if (isNotSPI(first_intid)) { @@ -601,7 +608,8 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, } for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { if (irqNsacr[int_id] == 0) { @@ -622,6 +630,7 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, updateAndInformCPUInterfaces(); return; } else if (GICD_ICPENDR.contains(addr)) { + // Interrupt Clear-Pending Registers int first_intid = (addr - GICD_ICPENDR.start()) * 8; if (isNotSPI(first_intid)) { @@ -629,7 +638,8 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, } for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { if (irqNsacr[int_id] < 2) { @@ -647,8 +657,8 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, updateAndInformCPUInterfaces(); return; - // Interrupt Set-Active Registers } else if (GICD_ISACTIVER.contains(addr)) { + // Interrupt Set-Active Registers int first_intid = (addr - GICD_ISACTIVER.start()) * 8; if (isNotSPI(first_intid)) { @@ -656,7 +666,8 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, } for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { continue; @@ -679,7 +690,8 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, } for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i++, int_id++) { + i++, int_id++) { + if (nsAccessToSecInt(int_id, is_secure_access)) { continue; @@ -698,8 +710,8 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, } return; - // Interrupt Priority Registers } else if (GICD_IPRIORITYR.contains(addr)) { + // Interrupt Priority Registers int first_intid = addr - GICD_IPRIORITYR.start(); if (isNotSPI(first_intid)) { @@ -731,9 +743,13 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, warn("Gicv3Distributor::write(): " "GICD_ITARGETSR is RAZ/WI, legacy not supported!\n"); return; - // Interrupt Configuration Registers } else if (GICD_ICFGR.contains(addr)) { - /* Here only the odd bits are used; even bits are RES0 */ + // Interrupt Configuration Registers + // for x = 0 to 15: + // GICD_ICFGR[2x] = RES0 + // GICD_ICFGR[2x + 1] = + // 0 level-sensitive + // 1 edge-triggered int first_intid = (addr - GICD_ICFGR.start()) * 4; if (isNotSPI(first_intid)) { @@ -741,7 +757,7 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, } for (int i = 0, int_id = first_intid; i < 8 * size && int_id < itLines; - i = i + 2, int_id++) { + i = i + 2, int_id++) { irqConfig[int_id] = data & (0x2 << i) ? Gicv3::INT_EDGE_TRIGGERED : Gicv3::INT_LEVEL_SENSITIVE; @@ -766,7 +782,7 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, } for (int i = 0, int_id = first_intid; - i < 8 * size && int_id < itLines; i++, int_id++) { + i < 8 * size && int_id < itLines; i++, int_id++) { irqGrpmod[int_id] = data & (0x1 << i); } @@ -774,8 +790,8 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, } } - // Non-secure Access Control Registers } else if (GICD_NSACR.contains(addr)) { + // Non-secure Access Control Registers // 2 bits per interrupt int first_intid = (addr - GICD_NSACR.start()) * 4; @@ -788,7 +804,7 @@ Gicv3Distributor::write(Addr addr, uint64_t data, size_t size, } for (int i = 0, int_id = first_intid; - i < 8 * size && int_id < itLines; i = i + 2, int_id++) { + i < 8 * size && int_id < itLines; i = i + 2, int_id++) { irqNsacr[int_id] = (data >> (2 * int_id)) & 0x3; } @@ -924,7 +940,7 @@ Gicv3Distributor::sendInt(uint32_t int_id) } void -Gicv3Distributor::intDeasserted(uint32_t int_id) +Gicv3Distributor::deassertSPI(uint32_t int_id) { panic_if(int_id < Gicv3::SGI_MAX + Gicv3::PPI_MAX, "Invalid SPI!"); panic_if(int_id > itLines, "Invalid SPI!"); @@ -965,12 +981,12 @@ Gicv3Distributor::update() // Find the highest priority pending SPI for (int int_id = Gicv3::SGI_MAX + Gicv3::PPI_MAX; int_id < itLines; - int_id++) { + int_id++) { Gicv3::GroupId int_group = getIntGroup(int_id); bool group_enabled = groupEnabled(int_group); if (irqPending[int_id] && irqEnabled[int_id] && - !irqActive[int_id] && group_enabled) { + !irqActive[int_id] && group_enabled) { IROUTER affinity_routing = irqAffinityRouting[int_id]; Gicv3Redistributor * target_redistributor = nullptr; @@ -1005,13 +1021,13 @@ Gicv3Distributor::update() uint32_t target_cpu = target_redistributor->cpuId; if ((irqPriority[int_id] < target_cpu_interface->hppi.prio) || - /* - * Multiple pending ints with same priority. - * Implementation choice which one to signal. - * Our implementation selects the one with the lower id. - */ - (irqPriority[int_id] == target_cpu_interface->hppi.prio && - int_id < target_cpu_interface->hppi.intid)) { + /* + * Multiple pending ints with same priority. + * Implementation choice which one to signal. + * Our implementation selects the one with the lower id. + */ + (irqPriority[int_id] == target_cpu_interface->hppi.prio && + int_id < target_cpu_interface->hppi.intid)) { target_cpu_interface->hppi.intid = int_id; target_cpu_interface->hppi.prio = irqPriority[int_id]; target_cpu_interface->hppi.group = int_group; @@ -1026,16 +1042,15 @@ Gicv3Distributor::update() redistributor_i->getCPUInterface(); if (!new_hppi[i] && cpu_interface_i->hppi.prio != 0xff && - cpu_interface_i->hppi.intid >= - (Gicv3::SGI_MAX + Gicv3::PPI_MAX) && - cpu_interface_i->hppi.intid < Gicv3::INTID_SECURE) { + cpu_interface_i->hppi.intid >= (Gicv3::SGI_MAX + Gicv3::PPI_MAX) && + cpu_interface_i->hppi.intid < Gicv3::INTID_SECURE) { fullUpdate(); } } } Gicv3::IntStatus -Gicv3Distributor::intStatus(uint32_t int_id) +Gicv3Distributor::intStatus(uint32_t int_id) const { panic_if(int_id < Gicv3::SGI_MAX + Gicv3::PPI_MAX, "Invalid SPI!"); panic_if(int_id > itLines, "Invalid SPI!"); @@ -1054,7 +1069,7 @@ Gicv3Distributor::intStatus(uint32_t int_id) } Gicv3::GroupId -Gicv3Distributor::getIntGroup(int int_id) +Gicv3Distributor::getIntGroup(int int_id) const { panic_if(int_id < Gicv3::SGI_MAX + Gicv3::PPI_MAX, "Invalid SPI!"); panic_if(int_id > itLines, "Invalid SPI!"); diff --git a/src/dev/arm/gic_v3_distributor.hh b/src/dev/arm/gic_v3_distributor.hh index 334eceee5..6b7b0cbb1 100644 --- a/src/dev/arm/gic_v3_distributor.hh +++ b/src/dev/arm/gic_v3_distributor.hh @@ -106,21 +106,21 @@ class Gicv3Distributor : public Serializable static const AddrRange GICD_IROUTER; BitUnion64(IROUTER) - Bitfield<63, 40> res0_1; - Bitfield<39, 32> Aff3; - Bitfield<31> IRM; - Bitfield<30, 24> res0_2; - Bitfield<23, 16> Aff2; - Bitfield<15, 8> Aff1; - Bitfield<7, 0> Aff0; + Bitfield<63, 40> res0_1; + Bitfield<39, 32> Aff3; + Bitfield<31> IRM; + Bitfield<30, 24> res0_2; + Bitfield<23, 16> Aff2; + Bitfield<15, 8> Aff1; + Bitfield<7, 0> Aff0; EndBitUnion(IROUTER) - static const uint32_t GICD_CTLR_ENABLEGRP0 = 1 << 0; + static const uint32_t GICD_CTLR_ENABLEGRP0 = 1 << 0; + static const uint32_t GICD_CTLR_ENABLEGRP1 = 1 << 0; static const uint32_t GICD_CTLR_ENABLEGRP1NS = 1 << 1; - static const uint32_t GICD_CTLR_ENABLEGRP1S = 1 << 2; - static const uint32_t GICD_CTLR_ENABLEGRP1 = 1 << 0; - static const uint32_t GICD_CTLR_ENABLEGRP1A = 1 << 1; - static const uint32_t GICD_CTLR_DS = 1 << 6; + static const uint32_t GICD_CTLR_ENABLEGRP1A = 1 << 1; + static const uint32_t GICD_CTLR_ENABLEGRP1S = 1 << 2; + static const uint32_t GICD_CTLR_DS = 1 << 6; bool ARE; bool DS; @@ -141,19 +141,15 @@ class Gicv3Distributor : public Serializable static const uint32_t ADDR_RANGE_SIZE = 0x10000; - Gicv3Distributor(Gicv3 * gic, uint32_t it_lines); - ~Gicv3Distributor(); - void init(); - void initState(); + protected: - uint64_t read(Addr addr, size_t size, bool is_secure_access); - void write(Addr addr, uint64_t data, size_t size, - bool is_secure_access); - void serialize(CheckpointOut & cp) const override; - void unserialize(CheckpointIn & cp) override; + void activateIRQ(uint32_t int_id); + void deactivateIRQ(uint32_t int_id); + void fullUpdate(); + Gicv3::GroupId getIntGroup(int int_id) const; - bool - groupEnabled(Gicv3::GroupId group) + inline bool + groupEnabled(Gicv3::GroupId group) const { if (DS == 0) { switch (group) { @@ -186,16 +182,9 @@ class Gicv3Distributor : public Serializable } } - void sendInt(uint32_t int_id); - void intDeasserted(uint32_t int_id); - Gicv3::IntStatus intStatus(uint32_t int_id); - void updateAndInformCPUInterfaces(); - void update(); - void fullUpdate(); - void activateIRQ(uint32_t int_id); - void deactivateIRQ(uint32_t int_id); + Gicv3::IntStatus intStatus(uint32_t int_id) const; - inline bool isNotSPI(uint8_t int_id) + inline bool isNotSPI(uint8_t int_id) const { if (int_id < (Gicv3::SGI_MAX + Gicv3::PPI_MAX) || int_id >= itLines) { return true; @@ -204,15 +193,28 @@ class Gicv3Distributor : public Serializable } } - inline bool nsAccessToSecInt(uint8_t int_id, bool is_secure_access) + inline bool nsAccessToSecInt(uint8_t int_id, bool is_secure_access) const { return !DS && !is_secure_access && getIntGroup(int_id) != Gicv3::G1NS; } - protected: - void reset(); - Gicv3::GroupId getIntGroup(int int_id); + void serialize(CheckpointOut & cp) const override; + void unserialize(CheckpointIn & cp) override; + void update(); + void updateAndInformCPUInterfaces(); + + public: + + Gicv3Distributor(Gicv3 * gic, uint32_t it_lines); + + void deassertSPI(uint32_t int_id); + void init(); + void initState(); + uint64_t read(Addr addr, size_t size, bool is_secure_access); + void sendInt(uint32_t int_id); + void write(Addr addr, uint64_t data, size_t size, + bool is_secure_access); }; #endif //__DEV_ARM_GICV3_DISTRIBUTOR_H__ diff --git a/src/dev/arm/gic_v3_redistributor.cc b/src/dev/arm/gic_v3_redistributor.cc index 62565079b..01ec9c511 100644 --- a/src/dev/arm/gic_v3_redistributor.cc +++ b/src/dev/arm/gic_v3_redistributor.cc @@ -37,7 +37,7 @@ #include "mem/fs_translating_port_proxy.hh" const AddrRange Gicv3Redistributor::GICR_IPRIORITYR(SGI_base + 0x0400, - SGI_base + 0x041f); + SGI_base + 0x041f); Gicv3Redistributor::Gicv3Redistributor(Gicv3 * gic, uint32_t cpu_id) : gic(gic), @@ -55,10 +55,6 @@ Gicv3Redistributor::Gicv3Redistributor(Gicv3 * gic, uint32_t cpu_id) { } -Gicv3Redistributor::~Gicv3Redistributor() -{ -} - void Gicv3Redistributor::init() { @@ -191,8 +187,7 @@ Gicv3Redistributor::read(Addr addr, size_t size, bool is_secure_access) } case GICR_PIDR0: { // Peripheral ID0 Register - uint8_t part_0 = 0x92; // Part number, bits[7:0] - return part_0; + return 0x92; // Part number, bits[7:0] } case GICR_PIDR1: { // Peripheral ID1 Register @@ -299,7 +294,7 @@ Gicv3Redistributor::read(Addr addr, size_t size, bool is_secure_access) uint32_t first_int_id = addr == GICR_ICFGR0 ? 0 : Gicv3::SGI_MAX; for (int i = 0, int_id = first_int_id; i < 32; - i = i + 2, int_id++) { + i = i + 2, int_id++) { if (!distributor->DS && !is_secure_access) { // RAZ/WI for non-secure accesses for secure interrupts if (getIntGroup(int_id) != Gicv3::G1NS) { @@ -346,7 +341,7 @@ Gicv3Redistributor::read(Addr addr, size_t size, bool is_secure_access) value = 0; } else { for (int i = 0, int_id = 0; i < 8 * size; - i = i + 2, int_id++) { + i = i + 2, int_id++) { value |= irqNsacr[int_id] << i; } } @@ -439,11 +434,11 @@ Gicv3Redistributor::write(Addr addr, uint64_t data, size_t size, } if (not peInLowPowerState and - (data & GICR_WAKER_ProcessorSleep)) { + (data & GICR_WAKER_ProcessorSleep)) { DPRINTF(GIC, "Gicv3Redistributor::write(): " "PE entering in low power state\n"); } else if (peInLowPowerState and - not(data & GICR_WAKER_ProcessorSleep)) { + not(data & GICR_WAKER_ProcessorSleep)) { DPRINTF(GIC, "Gicv3Redistributor::write(): powering up PE\n"); } @@ -596,7 +591,7 @@ Gicv3Redistributor::write(Addr addr, uint64_t data, size_t size, int first_intid = Gicv3::SGI_MAX; for (int i = 0, int_id = first_intid; i < 8 * size; - i = i + 2, int_id++) { + i = i + 2, int_id++) { if (!distributor->DS && !is_secure_access) { // RAZ/WI for non-secure accesses for secure interrupts if (getIntGroup(int_id) != Gicv3::G1NS) { @@ -604,9 +599,9 @@ Gicv3Redistributor::write(Addr addr, uint64_t data, size_t size, } } - irqConfig[int_id] = data & (0x2 << i) - ? Gicv3::INT_EDGE_TRIGGERED : - Gicv3::INT_LEVEL_SENSITIVE; + irqConfig[int_id] = data & (0x2 << i) ? + Gicv3::INT_EDGE_TRIGGERED : + Gicv3::INT_LEVEL_SENSITIVE; DPRINTF(GIC, "Gicv3Redistributor::write(): " "int_id %d (PPI) config %d\n", int_id, irqConfig[int_id]); @@ -640,7 +635,7 @@ Gicv3Redistributor::write(Addr addr, uint64_t data, size_t size, // RAZ/WI } else { for (int i = 0, int_id = 0; i < 8 * size; - i = i + 2, int_id++) { + i = i + 2, int_id++) { irqNsacr[int_id] = (data >> i) & 0x3; } } @@ -773,7 +768,7 @@ Gicv3Redistributor::sendSGI(uint32_t int_id, Gicv3::GroupId group, bool ns) int nsaccess = irqNsacr[int_id]; if ((int_group == Gicv3::G0S && nsaccess < 1) || - (int_group == Gicv3::G1S && nsaccess < 2)) { + (int_group == Gicv3::G1S && nsaccess < 2)) { return; } } @@ -785,7 +780,7 @@ Gicv3Redistributor::sendSGI(uint32_t int_id, Gicv3::GroupId group, bool ns) } Gicv3::IntStatus -Gicv3Redistributor::intStatus(uint32_t int_id) +Gicv3Redistributor::intStatus(uint32_t int_id) const { assert(int_id < Gicv3::SGI_MAX + Gicv3::PPI_MAX); @@ -818,13 +813,13 @@ Gicv3Redistributor::update() if (irqPending[int_id] && irqEnabled[int_id] && !irqActive[int_id] && group_enabled) { if ((irqPriority[int_id] < cpuInterface->hppi.prio) || - /* - * Multiple pending ints with same priority. - * Implementation choice which one to signal. - * Our implementation selects the one with the lower id. - */ - (irqPriority[int_id] == cpuInterface->hppi.prio && - int_id < cpuInterface->hppi.intid)) { + /* + * Multiple pending ints with same priority. + * Implementation choice which one to signal. + * Our implementation selects the one with the lower id. + */ + (irqPriority[int_id] == cpuInterface->hppi.prio && + int_id < cpuInterface->hppi.intid)) { cpuInterface->hppi.intid = int_id; cpuInterface->hppi.prio = irqPriority[int_id]; cpuInterface->hppi.group = int_group; @@ -870,7 +865,7 @@ Gicv3Redistributor::update() } if (!new_hppi && cpuInterface->hppi.prio != 0xff && - cpuInterface->hppi.intid < Gicv3::SGI_MAX + Gicv3::PPI_MAX) { + cpuInterface->hppi.intid < Gicv3::SGI_MAX + Gicv3::PPI_MAX) { distributor->fullUpdate(); } } @@ -934,7 +929,7 @@ Gicv3Redistributor::updateAndInformCPUInterface() } Gicv3::GroupId -Gicv3Redistributor::getIntGroup(int int_id) +Gicv3Redistributor::getIntGroup(int int_id) const { assert(int_id < (Gicv3::SGI_MAX + Gicv3::PPI_MAX)); @@ -973,7 +968,7 @@ Gicv3Redistributor::deactivateIRQ(uint32_t int_id) } uint32_t -Gicv3Redistributor::getAffinity() +Gicv3Redistributor::getAffinity() const { ThreadContext * tc = gic->getSystem()->getThreadContext(cpuId); uint64_t mpidr = getMPIDR(gic->getSystem(), tc); @@ -990,7 +985,7 @@ Gicv3Redistributor::getAffinity() } bool -Gicv3Redistributor::canBeSelectedFor1toNInterrupt(Gicv3::GroupId group) +Gicv3Redistributor::canBeSelectedFor1toNInterrupt(Gicv3::GroupId group) const { if (peInLowPowerState) { return false; diff --git a/src/dev/arm/gic_v3_redistributor.hh b/src/dev/arm/gic_v3_redistributor.hh index 664173641..578cba105 100644 --- a/src/dev/arm/gic_v3_redistributor.hh +++ b/src/dev/arm/gic_v3_redistributor.hh @@ -35,8 +35,8 @@ #include "dev/arm/gic_v3.hh" #include "sim/serialize.hh" -class Gicv3Distributor; class Gicv3CPUInterface; +class Gicv3Distributor; class Gicv3Redistributor : public Serializable { @@ -56,14 +56,14 @@ class Gicv3Redistributor : public Serializable * GICv3 defines 2 contiguous 64KB frames for each redistributor. * Order of frames must be RD_base, SGI_base. */ - static const uint32_t RD_base = 0x0; + static const uint32_t RD_base = 0x0; static const uint32_t SGI_base = 0x10000; enum { // Control Register GICR_CTLR = RD_base + 0x0000, // Implementer Identification Register - GICR_IIDR = RD_base + 0x0004, + GICR_IIDR = RD_base + 0x0004, // Type Register GICR_TYPER = RD_base + 0x0008, // Wake Register @@ -99,21 +99,21 @@ class Gicv3Redistributor : public Serializable // Interrupt Clear-Enable Register 0 GICR_ICENABLER0 = SGI_base + 0x0180, // Interrupt Set-Pending Register 0 - GICR_ISPENDR0 = SGI_base + 0x0200, + GICR_ISPENDR0 = SGI_base + 0x0200, // Interrupt Clear-Pending Register 0 - GICR_ICPENDR0 = SGI_base + 0x0280, + GICR_ICPENDR0 = SGI_base + 0x0280, // Interrupt Set-Active Register 0 GICR_ISACTIVER0 = SGI_base + 0x0300, // Interrupt Clear-Active Register 0 GICR_ICACTIVER0 = SGI_base + 0x0380, // SGI Configuration Register - GICR_ICFGR0 = SGI_base + 0x0c00, + GICR_ICFGR0 = SGI_base + 0x0c00, // PPI Configuration Register - GICR_ICFGR1 = SGI_base + 0x0c04, + GICR_ICFGR1 = SGI_base + 0x0c04, // Interrupt Group Modifier Register 0 - GICR_IGRPMODR0 = SGI_base + 0x0d00, + GICR_IGRPMODR0 = SGI_base + 0x0d00, // Non-secure Access Control Register - GICR_NSACR = SGI_base + 0x0e00, + GICR_NSACR = SGI_base + 0x0e00, }; // Interrupt Priority Registers @@ -164,9 +164,9 @@ class Gicv3Redistributor : public Serializable std::vector<LPIConfigurationTableEntry> lpiConfigurationTable; static const uint32_t GICR_CTLR_ENABLE_LPIS = 1 << 0; - static const uint32_t GICR_CTLR_DPG0 = 1 << 24; + static const uint32_t GICR_CTLR_DPG0 = 1 << 24; static const uint32_t GICR_CTLR_DPG1NS = 1 << 25; - static const uint32_t GICR_CTLR_DPG1S = 1 << 26; + static const uint32_t GICR_CTLR_DPG1S = 1 << 26; public: @@ -181,39 +181,37 @@ class Gicv3Redistributor : public Serializable static const uint32_t SMALLEST_LPI_ID = 8192; - Gicv3Redistributor(Gicv3 * gic, uint32_t cpu_id); - ~Gicv3Redistributor(); - void init(); - void initState(); - uint64_t read(Addr addr, size_t size, bool is_secure_access); - void write(Addr addr, uint64_t data, size_t size, - bool is_secure_access); - void sendPPInt(uint32_t int_id); - void sendSGI(uint32_t int_id, Gicv3::GroupId group, bool ns); - void serialize(CheckpointOut & cp) const override; - void unserialize(CheckpointIn & cp) override; - uint32_t getAffinity(); + void activateIRQ(uint32_t int_id); + bool canBeSelectedFor1toNInterrupt(Gicv3::GroupId group) const; + void deactivateIRQ(uint32_t int_id); - Gicv3CPUInterface * + inline Gicv3CPUInterface * getCPUInterface() const { return cpuInterface; } - bool canBeSelectedFor1toNInterrupt(Gicv3::GroupId group); + Gicv3::GroupId getIntGroup(int int_id) const; + Gicv3::IntStatus intStatus(uint32_t int_id) const; void setClrLPI(uint64_t data, bool set); - - protected: - void reset(); + void sendSGI(uint32_t int_id, Gicv3::GroupId group, bool ns); + void serialize(CheckpointOut & cp) const override; + void unserialize(CheckpointIn & cp) override; void update(); void updateAndInformCPUInterface(); - Gicv3::IntStatus intStatus(uint32_t int_id); - Gicv3::GroupId getIntGroup(int int_id); - void activateIRQ(uint32_t int_id); - void deactivateIRQ(uint32_t int_id); + + public: + + Gicv3Redistributor(Gicv3 * gic, uint32_t cpu_id); void invalLpiConfig(uint32_t lpi_entry_index); + uint32_t getAffinity() const; + void init(); + void initState(); + uint64_t read(Addr addr, size_t size, bool is_secure_access); + void sendPPInt(uint32_t int_id); + void write(Addr addr, uint64_t data, size_t size, bool is_secure_access); }; #endif //__DEV_ARM_GICV3_REDISTRIBUTOR_H__ |