44#include "debug/Plic.hh"
47#include "params/Plic.hh"
48#include "params/PlicBase.hh"
54using namespace RiscvISA;
60 nContext(params.n_contexts),
61 registers(params.
name, pioAddr, this),
70 assert(src_id < nSrc && src_id >= 0);
73 int src_index = src_id >> 5;
74 int src_offset = src_id & 0x1F;
77 std::bitset<32> pending_bits(pending);
78 pending_bits[src_offset] = 1;
79 pending = (uint32_t) pending_bits.to_ulong();
88 "Int post request - source: %#x, current priority: %#x\n",
99 assert(src_id <
nSrc);
103 int src_index = src_id >> 5;
104 int src_offset = src_id & 0x1F;
106 std::bitset<32> pending_bits(pending);
107 pending_bits[src_offset] = 0;
108 pending = (uint32_t) pending_bits.to_ulong();
116 "Int clear request - source: %#x, current priority: %#x\n",
129 "Read request - addr: %#x, size: %#x, atomic:%d\n",
149 "Write request - addr: %#x, size: %#x\n",
189 "Device init - %d contexts, %d sources, %d pending registers\n",
198 using namespace std::placeholders;
202 reserved.emplace_back(
"reserved0", reserve0_size);
205 reserved.emplace_back(
"reserved1", reserve1_size);
208 reserved.emplace_back(
"reserved2", reserve2_size);
211 reserved.emplace_back(
"reserved3", reserve3_size);
225 std::string(
"priority") + std::to_string(
i), 0);
229 std::string(
"pending") + std::to_string(
i), 0);
236 std::string(
"enable") + std::to_string(
i)
237 +
"_" + std::to_string(
j), 0);
240 std::string(
"enable_hole") + std::to_string(
i), enable_hole_size);
243 std::string(
"threshold") + std::to_string(
i), 0);
245 std::string(
"claim") + std::to_string(
i), 0);
247 std::string(
"claim_hole") + std::to_string(
i), claim_hole_size);
298 int src_index = src_id >> 5;
299 int src_offset = src_id & 0x1F;
311 "Priority updated - src: %d, val: %d\n",
317 const int src32_id,
const int context_id)
321 for (
int i = 0;
i < 32;
i ++) {
322 int src_id = (src32_id << 5) +
i;
329 "Enable updated - context: %d, src32: %d, val: %#x\n",
330 context_id, src32_id,
reg.get());
335 const int context_id)
340 "Threshold updated - context: %d, val: %d\n",
341 context_id,
reg.get());
347 if (
lastID[context_id] == 0) {
350 int src_index = max_int_id >> 5;
351 int src_offset = max_int_id & 0x1F;
355 lastID[context_id] = max_int_id;
357 "Claim success - context: %d, interrupt ID: %d\n",
358 context_id, max_int_id);
360 reg.update(max_int_id);
364 "Claim already cleared - context: %d, interrupt ID: %d\n",
365 context_id, max_int_id);
369 warn(
"PLIC claim repeated (not completed) - context: %d, last: %d",
370 context_id,
lastID[context_id]);
371 return lastID[context_id];
387 "Complete - context: %d, interrupt ID: %d\n",
388 context_id,
reg.get());
400 uint32_t max_priority;
405 new_output.maxID[
i] = max_id;
406 new_output.maxPriority[
i] = max_priority;
419 DPRINTF(
Plic,
"Update scheduled - tick: %d\n", next_update);
447 int thread_id =
i >> 1;
448 int int_id = (
i & 1) ?
449 ExceptionCode::INT_EXT_SUPER : ExceptionCode::INT_EXT_MACHINE;
456 DPRINTF(
Plic,
"Int posted - thread: %d, int id: %d, ",
459 tc->getCpuPtr()->postInterrupt(tc->threadId(), int_id, 0);
462 DPRINTF(
Plic,
"Int filtered - thread: %d, int id: %d, ",
466 tc->getCpuPtr()->clearInterrupt(tc->threadId(), int_id, 0);
483 for (
auto const ®_inner:
reg) {
484 paramOut(cp, reg_inner.name(), reg_inner);
494 paramOut(cp, std::string(
"output_tick") +
495 std::to_string(n_outputs), it.first);
497 std::to_string(n_outputs), it.second.maxID);
499 std::to_string(n_outputs), it.second.maxPriority);
526 for (
auto ®_inner:
reg) {
527 paramIn(cp, reg_inner.name(), reg_inner);
536 for (
int i = 0;
i < n_outputs;
i++) {
540 paramIn(cp, std::string(
"output_tick") +
541 std::to_string(
i), output_tick);
543 std::to_string(
i), output_id);
545 std::to_string(
i), output_pri);
Tick pioDelay
Delay that the device experinces on an access.
Addr pioSize
Size that the device's address range.
Tick cyclesToTicks(Cycles c) const
Cycles is a wrapper class for representing cycle counts, i.e.
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
void makeResponse()
Take a request packet and modify it in place to be suitable for returning as a response to that reque...
T * getPtr()
get a pointer to the data ptr.
AtomicOpFunctor * getAtomicOp() const
Accessor function to atomic op.
MemCmd cmd
The command field of the packet.
void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
std::vector< Register32 > priority
const Addr thresholdPadding
std::vector< Register32 > pending
std::vector< RegisterRaz > enable_holes
const Addr thresholdStart
std::vector< Register32 > threshold
std::vector< std::vector< Register32 > > enable
std::vector< RegisterRaz > reserved
std::vector< RegisterRaz > claim_holes
std::vector< Register32 > claim
Tick write(PacketPtr pkt) override
Pure virtual function that the device must implement.
std::vector< uint32_t > lastID
int nContext
Number of interrupt contexts = nThread * 2 e.g.
std::map< Tick, PlicOutput > outputQueue
void updateOutput()
Trigger:
gem5::Plic::PlicRegisters registers
Plic(const Params ¶ms)
void post(int src_id) override
Interrupt interface.
EventFunctionWrapper update
Tick read(PacketPtr pkt) override
PioDevice funcitons.
PlicRegisters::Register32 Register32
void writeEnable(Register32 ®, const uint32_t &data, const int src32_id, const int context_id)
void unserialize(CheckpointIn &cp) override
Unserialize an object.
void clear(int src_id) override
void writeThreshold(Register32 ®, const uint32_t &data, const int context_id)
uint32_t readClaim(Register32 ®, const int context_id)
std::vector< std::vector< uint32_t > > effPriority
void propagateOutput()
Trigger:
int nSrc32
Number of 32-bit pending registers needed = ceil(nSrc / 32)
void init() override
SimObject functions.
std::vector< uint32_t > pendingPriority
void writeClaim(Register32 ®, const uint32_t &data, const int context_id)
void serialize(CheckpointOut &cp) const override
Serialize an object.
void writePriority(Register32 ®, const uint32_t &data, const int src_id)
Register read / write callbacks.
void addRegister(RegisterAdder reg)
virtual void read(Addr addr, void *buf, Addr bytes)
virtual void write(Addr addr, const void *buf, Addr bytes)
static constexpr T divCeil(const T &a, const U &b)
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
bool scheduled() const
Determine if the current event is scheduled.
void schedule(Event &event, Tick when)
#define UNSERIALIZE_CONTAINER(member)
decltype(std::begin(std::declval< const T & >()), std::end(std::declval< const T & >()), void()) arrayParamOut(CheckpointOut &os, const std::string &name, const T ¶m)
#define SERIALIZE_CONTAINER(member)
Bitfield< 3, 0 > priority
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
Tick curTick()
The universal simulation clock.
std::ostream CheckpointOut
void paramOut(CheckpointOut &cp, const std::string &name, ExtMachInst const &machInst)
void paramIn(CheckpointIn &cp, const std::string &name, ExtMachInst &machInst)
uint64_t Tick
Tick count type.
void arrayParamIn(CheckpointIn &cp, const std::string &name, CircleBuf< T > ¶m)
Declaration of the Packet class.
#define UNSERIALIZE_SCALAR(scalar)
#define SERIALIZE_SCALAR(scalar)
NOTE: This implementation of CLINT is based on the SiFive U54MC datasheet: https://sifive....
std::vector< uint32_t > maxPriority
std::vector< uint32_t > maxID
const std::string & name()