48 #include "debug/Checkpoint.hh" 49 #include "debug/SMMUv3.hh" 57 masterId(params->
system->getMasterId(this)),
58 masterPort(
name() +
".master", *this),
59 masterTableWalkPort(
name() +
".master_walker", *this),
60 controlPort(
name() +
".control", *this, params->reg_map),
61 tlb(params->tlb_entries, params->tlb_assoc, params->tlb_policy),
62 configCache(params->cfg_entries, params->cfg_assoc, params->cfg_policy),
63 ipaCache(params->ipa_entries, params->ipa_assoc, params->ipa_policy),
64 walkCache({ { params->walk_S1L0, params->walk_S1L1,
65 params->walk_S1L2, params->walk_S1L3,
66 params->walk_S2L0, params->walk_S2L1,
67 params->walk_S2L2, params->walk_S2L3 } },
100 "Invalid register map size: %#x different than SMMU_REG_SIZE = %#x\n",
120 warn(
"SMMUv3 IDx register values unchecked\n");
214 ifc->scheduleDeviceRetry();
226 panic(
"Not in timing or atomic mode!");
235 bool finished =
false;
238 action = proc->
run(pkt);
240 switch (action.
type) {
263 delay += action.
delay;
267 panic(
"ACTION_TERMINATE in atomic mode\n");
270 panic(
"Unknown action\n");
274 action.
delay = delay;
284 switch (action.
type) {
359 panic(
"Unknown action\n");
377 panic(
"Not in timing or atomic mode!");
384 switch (cmd.dw0.
type) {
398 slave_interface->microTLB->invalidateSID(cmd.dw0.
sid);
399 slave_interface->mainTLB->invalidateSID(cmd.dw0.
sid);
405 const auto range = cmd.dw1.
range;
413 slave_interface->microTLB->invalidateAll();
414 slave_interface->mainTLB->invalidateAll();
418 const auto start_sid = cmd.dw0.
sid & ~((1 << (range + 1)) - 1);
419 const auto end_sid = start_sid + (1 << (range + 1)) - 1;
420 for (
auto sid = start_sid; sid <= end_sid; sid++) {
424 slave_interface->microTLB->invalidateSID(sid);
425 slave_interface->mainTLB->invalidateSID(sid);
438 slave_interface->microTLB->invalidateSSID(
440 slave_interface->mainTLB->invalidateSSID(
451 slave_interface->microTLB->invalidateSID(cmd.dw0.
sid);
452 slave_interface->mainTLB->invalidateSID(cmd.dw0.
sid);
460 slave_interface->microTLB->invalidateVMID(cmd.dw0.
vmid);
461 slave_interface->mainTLB->invalidateVMID(cmd.dw0.
vmid);
472 slave_interface->microTLB->invalidateASID(
474 slave_interface->mainTLB->invalidateASID(
487 slave_interface->microTLB->invalidateVAA(
489 slave_interface->mainTLB->invalidateVAA(
493 const bool leaf_only = cmd.dw1.leaf ? true :
false;
500 DPRINTF(
SMMUv3,
"CMD_TLBI_NH_VA va=%#08x asid=%#x vmid=%#x\n",
503 slave_interface->microTLB->invalidateVA(
505 slave_interface->mainTLB->invalidateVA(
509 const bool leaf_only = cmd.dw1.leaf ? true :
false;
531 slave_interface->microTLB->invalidateVMID(cmd.dw0.
vmid);
532 slave_interface->mainTLB->invalidateVMID(cmd.dw0.
vmid);
543 slave_interface->microTLB->invalidateAll();
544 slave_interface->mainTLB->invalidateAll();
554 panic(
"resume unimplemented");
558 warn(
"Unimplemented command %#x\n", cmd.dw0.
type);
570 switch (trans_granule) {
575 panic(
"Unknown translation granule size %d", trans_granule);
589 warn(
"smmu: secure registers (0x%x) are not implemented\n",
596 case sizeof(uint32_t):
597 pkt->
setLE<uint32_t>(*reinterpret_cast<uint32_t *>(reg_ptr));
599 case sizeof(uint64_t):
600 pkt->
setLE<uint64_t>(*reinterpret_cast<uint64_t *>(reg_ptr));
603 panic(
"smmu: unallowed access size: %d bytes\n", pkt->
getSize());
618 DPRINTF(
SMMUv3,
"writeControl: addr=%08x size=%d data=%16x\n",
620 pkt->
getSize() ==
sizeof(uint64_t) ?
621 pkt->
getLE<uint64_t>() : pkt->
getLE<uint32_t>());
625 assert(pkt->
getSize() ==
sizeof(uint32_t));
631 case offsetof(
SMMURegs, strtab_base_cfg):
632 case offsetof(
SMMURegs, eventq_cons):
633 case offsetof(
SMMURegs, eventq_irq_cfg1):
635 assert(pkt->
getSize() ==
sizeof(uint32_t));
637 pkt->
getLE<uint32_t>();
641 assert(pkt->
getSize() ==
sizeof(uint32_t));
643 warn(
"CMDQ is enabled: ignoring write to CMDQ_CONS\n");
646 pkt->
getLE<uint32_t>();
651 assert(pkt->
getSize() ==
sizeof(uint32_t));
653 pkt->
getLE<uint32_t>();
657 case offsetof(
SMMURegs, strtab_base):
658 case offsetof(
SMMURegs, eventq_irq_cfg0):
659 assert(pkt->
getSize() ==
sizeof(uint64_t));
661 pkt->
getLE<uint64_t>();
665 assert(pkt->
getSize() ==
sizeof(uint64_t));
667 warn(
"CMDQ is enabled: ignoring write to CMDQ_BASE\n");
670 pkt->
getLE<uint64_t>();
676 case offsetof(
SMMURegs, eventq_base):
677 assert(pkt->
getSize() ==
sizeof(uint64_t));
679 pkt->
getLE<uint64_t>();
685 assert(pkt->
getSize() ==
sizeof(uint64_t));
687 pkt->
getLE<uint64_t>();
694 warn(
"smmu: secure registers (0x%x) are not implemented\n",
697 warn(
"smmu: write to read-only/undefined register at 0x%x\n",
721 fatal(
"Master port is not connected.\n");
742 using namespace Stats;
758 .
desc(
"STE L1 fetches")
768 .
desc(
"CD L1 fetches")
777 .
init(0, 2000000, 2000)
778 .
name(
name() +
".translationTimeDist")
779 .
desc(
"Time to translate address")
783 .
init(0, 2000000, 2000)
785 .
desc(
"Time to walk page tables")
802 DPRINTF(Checkpoint,
"Serializing SMMUv3\n");
810 DPRINTF(Checkpoint,
"Unserializing SMMUv3\n");
818 if (name ==
"master") {
820 }
else if (name ==
"master_walker") {
822 }
else if (name ==
"control") {
830 SMMUv3Params::create()
#define panic(...)
This implements a cprintf based panic() function.
void unserialize(CheckpointIn &cp) override
Unserialize an object.
std::queue< SMMUAction > packetsTableWalkToRetry
const FlagsType pdf
Print the percent of the total that this entry represents.
Ports are used to interface objects to each other.
void invalidateVAA(Addr va, uint16_t vmid, const bool leaf_only)
bool inSecureBlock(uint32_t offs) const
void sendRangeChange() const
Called by the owner to send a range change.
SMMUAction runProcess(SMMUProcess *proc, PacketPtr pkt)
Stats::Distribution translationTimeDist
#define fatal(...)
This implements a cprintf based fatal() function.
std::queue< SMMUAction > packetsToRetry
const std::string & name()
SMMUAction run(PacketPtr pkt)
void invalidateVMID(uint16_t vmid)
virtual void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
void serialize(CheckpointOut &cp) const override
Serialize an object.
void schedAtsTimingResp(PacketPtr pkt)
Tick readControl(PacketPtr pkt)
Tick writeControl(PacketPtr pkt)
void invalidateVMID(uint16_t vmid)
void scheduleSlaveRetries()
const unsigned walkCacheS1Levels
void invalidateSSID(uint32_t sid, uint32_t ssid)
const unsigned masterPortWidth
EventWrapper< SMMUv3, &SMMUv3::processCommands > processCommandsEvent
SMMUv3SlaveInterface * ifc
bool sendTimingReq(PacketPtr pkt)
Attempt to send a timing request to the slave port by calling its corresponding receive function...
bool isConnected() const
Is this port currently connected to a peer?
Stats::Scalar steL1Fetches
void invalidateVA(Addr va, uint16_t asid, uint16_t vmid, const bool leaf_only)
void processCommand(const SMMUCommand &cmd)
Derived & flags(Flags _flags)
Set the flags and marks this stat to print at the end of simulation.
const Params * params() const
DrainState
Object drain/handover states.
virtual Port & getPort(const std::string &name, PortID id=InvalidPortID) override
Get a port with a given name and index.
std::vector< SMMUv3SlaveInterface * > slaveInterfaces
void setLE(T v)
Set the value in the data pointer to v as little endian.
Stats::Scalar cdL1Fetches
void masterRecvReqRetry()
SMMUControlPort controlPort
Draining buffers pending serialization/handover.
bool isAtomicMode() const
Is the system in atomic mode?
virtual Port & getPort(const std::string &if_name, PortID idx=InvalidPortID)
Get a port with a given name and index.
std::string csprintf(const char *format, const Args &...args)
void masterTableWalkRecvReqRetry()
uint32_t headerDelay
The extra delay from seeing the packet until the header is transmitted.
SMMUCommandExecProcess commandExecutor
void makeAtomicResponse()
uint64_t Tick
Tick count type.
Stats::Distribution ptwTimeDist
The ClockedObject class extends the SimObject with a clock and accessor functions to relate ticks to ...
bool masterRecvTimingResp(PacketPtr pkt)
void invalidateIPA(Addr ipa, uint16_t vmid)
void invalidateSID(uint32_t sid)
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
void invalidateASID(uint16_t asid, uint16_t vmid)
void schedule(Event &event, Tick when)
Distribution & init(Counter min, Counter max, Counter bkt)
Set the parameters of this distribution.
void schedTimingResp(PacketPtr pkt)
Tick nextCycle() const
Based on the clock of the object, determine the start tick of the first cycle that is at least one cy...
DrainState drain() override
Notify an object that it needs to drain its state.
Defines global host-dependent types: Counter, Tick, and (indirectly) {int,uint}{8,16,32,64}_t.
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
SMMUSemaphore masterPortSem
uint32_t payloadDelay
The extra pipelining delay from seeing the packet until the end of payload is transmitted by the comp...
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
#define SERIALIZE_ARRAY(member, size)
SMMUMasterPort masterPort
This is an implementation of the SMMUv3 architecture.
Derived & name(const std::string &name)
Set the name and marks this stat to print at the end of simulation.
virtual const std::string name() const
#define UNSERIALIZE_ARRAY(member, size)
void invalidateVAA(Addr va, uint16_t vmid)
bool masterTableWalkRecvTimingResp(PacketPtr pkt)
SMMUAction runProcessTiming(SMMUProcess *proc, PacketPtr pkt)
std::ostream CheckpointOut
uint8_t data[SMMU_REG_SIZE]
const PageTableOps * getPageTableOps(uint8_t trans_granule)
const bool walkCacheNonfinalEnable
void invalidateVMID(uint16_t vmid)
const bool walkCacheEnable
T getLE() const
Get the data in the packet byte swapped from little endian to host endian.
void pushSenderState(SenderState *sender_state)
Push a new sender state to the packet and make the current sender state the predecessor of the new on...
Addr start() const
Get the start address of the range.
Addr size() const
Get the size of the address range.
SenderState * popSenderState()
Pop the top of the state stack and return a pointer to it.
Derived & desc(const std::string &_desc)
Set the description and marks this stat to print at the end of simulation.
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
const bool ipaCacheEnable
const unsigned walkCacheS2Levels
virtual void regStats()
Callback to set stat parameters.
bool isTimingMode() const
Is the system in timing mode?
void invalidateVA(Addr va, uint16_t asid, uint16_t vmid)
Tick sendAtomic(PacketPtr pkt)
Send an atomic request packet, where the data is moved and the state is updated in zero time...
void regStats(const std::string &name) override
const bool configCacheEnable
SMMUAction runProcessAtomic(SMMUProcess *proc, PacketPtr pkt)
virtual void regStats(const std::string &name)
virtual void regStats() override
Callback to set stat parameters.
void invalidateASID(uint16_t asid, uint16_t vmid)
SMMUMasterTableWalkPort masterTableWalkPort