Go to the documentation of this file.
48 #include "debug/Checkpoint.hh"
49 #include "debug/SMMUv3.hh"
57 requestorId(params.
system->getRequestorId(this)),
58 requestPort(
name() +
".request", *this),
59 tableWalkPort(
name() +
".walker", *this),
60 controlPort(
name() +
".control", *this, params.reg_map),
61 irqInterfaceEnable(params.irq_interface_enable),
62 tlb(params.tlb_entries, params.tlb_assoc, params.tlb_policy, this),
63 configCache(params.cfg_entries, params.cfg_assoc, params.cfg_policy, this),
64 ipaCache(params.ipa_entries, params.ipa_assoc, params.ipa_policy, this),
69 params.walk_assoc, params.walk_policy,
this),
70 tlbEnable(params.tlb_enable),
71 configCacheEnable(params.cfg_enable),
72 ipaCacheEnable(params.ipa_enable),
73 walkCacheEnable(params.walk_enable),
74 tableWalkPortEnable(
false),
75 walkCacheNonfinalEnable(params.wc_nonfinal_enable),
76 walkCacheS1Levels(params.wc_s1_levels),
77 walkCacheS2Levels(params.wc_s2_levels),
78 requestPortWidth(params.request_port_width),
79 tlbSem(params.tlb_slots),
82 configSem(params.cfg_slots),
83 ipaSem(params.ipa_slots),
84 walkSem(params.walk_slots),
86 transSem(params.xlate_slots),
87 ptwSem(params.ptw_slots),
89 tlbLat(params.tlb_lat),
90 ifcSmmuLat(params.ifc_smmu_lat),
91 smmuIfcLat(params.smmu_ifc_lat),
92 configLat(params.cfg_lat),
93 ipaLat(params.ipa_lat),
94 walkLat(params.walk_lat),
96 deviceInterfaces(params.device_interfaces),
97 commandExecutor(
name() +
".cmd_exec", *
this),
98 regsMap(params.reg_map),
99 processCommandsEvent(
this)
102 "Invalid register map size: %#x different than SMMU_REG_SIZE = %#x\n",
106 memset(®s, 0,
sizeof(regs));
109 regs.idr0 = params.smmu_idr0;
110 regs.idr1 = params.smmu_idr1;
111 regs.idr2 = params.smmu_idr2;
112 regs.idr3 = params.smmu_idr3;
113 regs.idr4 = params.smmu_idr4;
114 regs.idr5 = params.smmu_idr5;
115 regs.iidr = params.smmu_iidr;
116 regs.aidr = params.smmu_aidr;
122 warn(
"SMMUv3 IDx register values unchecked\n");
124 for (
auto ifc : deviceInterfaces)
156 a.pkt->getAddr(),
a.pkt->getSize());
177 DPRINTF(
SMMUv3,
"[t] requestor HWTW resp addr=%#x size=%#x\n",
202 DPRINTF(
SMMUv3,
"[t] requestor HWTW retr addr=%#x size=%#x\n",
203 a.pkt->getAddr(),
a.pkt->getSize());
216 ifc->scheduleDeviceRetry();
228 panic(
"Not in timing or atomic mode!");
237 bool finished =
false;
240 action = proc->
run(pkt);
242 switch (action.
type) {
265 delay += action.
delay;
269 panic(
"ACTION_TERMINATE in atomic mode\n");
272 panic(
"Unknown action\n");
276 action.
delay = delay;
286 switch (action.
type) {
294 DPRINTF(
SMMUv3,
"[t] requestor HWTW req addr=%#x size=%#x\n",
344 DPRINTF(
SMMUv3,
"[t] ATS responder resp addr=%#x size=%#x\n",
362 panic(
"Unknown action\n");
380 panic(
"Not in timing or atomic mode!");
387 switch (cmd.dw0.
type) {
401 dev_interface->microTLB->invalidateSID(cmd.dw0.
sid);
402 dev_interface->mainTLB->invalidateSID(cmd.dw0.
sid);
408 const auto range = cmd.dw1.
range;
416 dev_interface->microTLB->invalidateAll();
417 dev_interface->mainTLB->invalidateAll();
421 const auto start_sid = cmd.dw0.
sid & ~((1 << (range + 1)) - 1);
422 const auto end_sid = start_sid + (1 << (range + 1)) - 1;
423 for (
auto sid = start_sid; sid <= end_sid; sid++) {
427 dev_interface->microTLB->invalidateSID(sid);
428 dev_interface->mainTLB->invalidateSID(sid);
441 dev_interface->microTLB->invalidateSSID(
443 dev_interface->mainTLB->invalidateSSID(
454 dev_interface->microTLB->invalidateSID(cmd.dw0.
sid);
455 dev_interface->mainTLB->invalidateSID(cmd.dw0.
sid);
463 dev_interface->microTLB->invalidateVMID(cmd.dw0.
vmid);
464 dev_interface->mainTLB->invalidateVMID(cmd.dw0.
vmid);
475 dev_interface->microTLB->invalidateASID(
477 dev_interface->mainTLB->invalidateASID(
490 dev_interface->microTLB->invalidateVAA(
492 dev_interface->mainTLB->invalidateVAA(
496 const bool leaf_only = cmd.dw1.leaf ? true :
false;
503 DPRINTF(
SMMUv3,
"CMD_TLBI_NH_VA va=%#08x asid=%#x vmid=%#x\n",
506 dev_interface->microTLB->invalidateVA(
508 dev_interface->mainTLB->invalidateVA(
512 const bool leaf_only = cmd.dw1.leaf ? true :
false;
534 dev_interface->microTLB->invalidateVMID(cmd.dw0.
vmid);
535 dev_interface->mainTLB->invalidateVMID(cmd.dw0.
vmid);
546 dev_interface->microTLB->invalidateAll();
547 dev_interface->mainTLB->invalidateAll();
557 panic(
"resume unimplemented");
561 warn(
"Unimplemented command %#x\n", cmd.dw0.
type);
573 switch (trans_granule) {
578 panic(
"Unknown translation granule size %d", trans_granule);
592 warn(
"smmu: secure registers (0x%x) are not implemented\n",
599 case sizeof(uint32_t):
600 pkt->
setLE<uint32_t>(*
reinterpret_cast<uint32_t *
>(reg_ptr));
602 case sizeof(uint64_t):
603 pkt->
setLE<uint64_t>(*
reinterpret_cast<uint64_t *
>(reg_ptr));
606 panic(
"smmu: unallowed access size: %d bytes\n", pkt->
getSize());
621 DPRINTF(
SMMUv3,
"writeControl: addr=%08x size=%d data=%16x\n",
623 pkt->
getSize() ==
sizeof(uint64_t) ?
624 pkt->
getLE<uint64_t>() : pkt->
getLE<uint32_t>());
628 assert(pkt->
getSize() ==
sizeof(uint32_t));
632 assert(pkt->
getSize() ==
sizeof(uint32_t));
634 warn(
"SMMUv3::%s No support for interrupt sources", __func__);
641 case offsetof(
SMMURegs, strtab_base_cfg):
642 case offsetof(
SMMURegs, eventq_cons):
643 case offsetof(
SMMURegs, eventq_irq_cfg1):
645 assert(pkt->
getSize() ==
sizeof(uint32_t));
647 pkt->
getLE<uint32_t>();
651 assert(pkt->
getSize() ==
sizeof(uint32_t));
653 warn(
"CMDQ is enabled: ignoring write to CMDQ_CONS\n");
656 pkt->
getLE<uint32_t>();
661 assert(pkt->
getSize() ==
sizeof(uint32_t));
663 pkt->
getLE<uint32_t>();
667 case offsetof(
SMMURegs, strtab_base):
668 case offsetof(
SMMURegs, eventq_irq_cfg0):
669 assert(pkt->
getSize() ==
sizeof(uint64_t));
671 pkt->
getLE<uint64_t>();
675 assert(pkt->
getSize() ==
sizeof(uint64_t));
677 warn(
"CMDQ is enabled: ignoring write to CMDQ_BASE\n");
680 pkt->
getLE<uint64_t>();
686 case offsetof(
SMMURegs, eventq_base):
687 assert(pkt->
getSize() ==
sizeof(uint64_t));
689 pkt->
getLE<uint64_t>();
695 assert(pkt->
getSize() ==
sizeof(uint64_t));
697 pkt->
getLE<uint64_t>();
704 warn(
"smmu: secure registers (0x%x) are not implemented\n",
707 warn(
"smmu: write to read-only/undefined register at 0x%x\n",
731 fatal(
"Request port is not connected.\n");
748 :
Stats::Group(parent),
756 using namespace Stats;
771 .
init(0, 2000000, 2000)
775 .
init(0, 2000000, 2000)
792 DPRINTF(Checkpoint,
"Serializing SMMUv3\n");
800 DPRINTF(Checkpoint,
"Unserializing SMMUv3\n");
808 if (
name ==
"request") {
810 }
else if (
name ==
"walker") {
812 }
else if (
name ==
"control") {
#define fatal(...)
This implements a cprintf based fatal() function.
void makeAtomicResponse()
void schedTimingResp(PacketPtr pkt)
Stats::Scalar steL1Fetches
void invalidateSSID(uint32_t sid, uint32_t ssid)
EventWrapper< SMMUv3, &SMMUv3::processCommands > processCommandsEvent
bool isTimingMode() const
Is the system in timing mode?
uint32_t payloadDelay
The extra pipelining delay from seeing the packet until the end of payload is transmitted by the comp...
SMMUv3Stats(Stats::Group *parent)
void invalidateIPA(Addr ipa, uint16_t vmid)
bool recvTimingResp(PacketPtr pkt)
const bool irqInterfaceEnable
const PageTableOps * getPageTableOps(uint8_t trans_granule)
uint64_t Tick
Tick count type.
SMMUv3DeviceInterface * ifc
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
SMMUAction runProcess(SMMUProcess *proc, PacketPtr pkt)
Stats::Scalar cdL1Fetches
void scheduleDeviceRetries()
SMMUAction run(PacketPtr pkt)
SMMUAction runProcessTiming(SMMUProcess *proc, PacketPtr pkt)
std::queue< SMMUAction > packetsTableWalkToRetry
SMMURequestPort requestPort
uint32_t headerDelay
The extra delay from seeing the packet until the header is transmitted.
The ClockedObject class extends the SimObject with a clock and accessor functions to relate ticks to ...
Derived & flags(Flags _flags)
Set the flags and marks this stat to print at the end of simulation.
@ Drained
Buffers drained, ready for serialization/handover.
void invalidateVAA(Addr va, uint16_t vmid, const bool leaf_only)
DrainState
Object drain/handover states.
SMMUAction runProcessAtomic(SMMUProcess *proc, PacketPtr pkt)
void invalidateVA(Addr va, uint16_t asid, uint16_t vmid, const bool leaf_only)
void schedule(Event &event, Tick when)
bool sendTimingReq(PacketPtr pkt)
Attempt to send a timing request to the responder port by calling its corresponding receive function.
virtual Port & getPort(const std::string &if_name, PortID idx=InvalidPortID)
Get a port with a given name and index.
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
SMMUv3(const SMMUv3Params &p)
bool inSecureBlock(uint32_t offs) const
Ports are used to interface objects to each other.
SMMUCommandExecProcess commandExecutor
SMMUTableWalkPort tableWalkPort
#define SERIALIZE_ARRAY(member, size)
Tick readControl(PacketPtr pkt)
std::vector< SMMUv3DeviceInterface * > deviceInterfaces
std::queue< SMMUAction > packetsToRetry
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
SMMUControlPort controlPort
const std::string & name()
void serialize(CheckpointOut &cp) const override
Serialize an object.
void invalidateVMID(uint16_t vmid)
Stats::Distribution translationTimeDist
void invalidateASID(uint16_t asid, uint16_t vmid)
uint8_t data[SMMU_REG_SIZE]
virtual const std::string name() const
void invalidateSID(uint32_t sid)
DrainState drain() override
Draining is the process of clearing out the states of SimObjects.These are the SimObjects that are pa...
void invalidateASID(uint16_t asid, uint16_t vmid)
void schedAtsTimingResp(PacketPtr pkt)
void tableWalkRecvReqRetry()
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.
#define UNSERIALIZE_ARRAY(member, size)
Distribution & init(Counter min, Counter max, Counter bkt)
Set the parameters of this distribution.
const FlagsType pdf
Print the percent of the total that this entry represents.
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
void invalidateVAA(Addr va, uint16_t vmid)
Tick writeControl(PacketPtr pkt)
SenderState * popSenderState()
Pop the top of the state stack and return a pointer to it.
bool isConnected() const
Is this port currently connected to a peer?
Tick nextCycle() const
Based on the clock of the object, determine the start tick of the first cycle that is at least one cy...
virtual Port & getPort(const std::string &name, PortID id=InvalidPortID) override
Get a port with a given name and index.
void setLE(T v)
Set the value in the data pointer to v as little endian.
std::ostream CheckpointOut
void invalidateVMID(uint16_t vmid)
const Params & params() const
virtual void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
bool tableWalkRecvTimingResp(PacketPtr pkt)
Tick sendAtomic(PacketPtr pkt)
Send an atomic request packet, where the data is moved and the state is updated in zero time,...
void invalidateVA(Addr va, uint16_t asid, uint16_t vmid)
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
Stats::Distribution ptwTimeDist
void unserialize(CheckpointIn &cp) override
Unserialize an object.
void sendRangeChange() const
Called by the owner to send a range change.
void processCommand(const SMMUCommand &cmd)
bool isAtomicMode() const
Is the system in atomic mode?
@ Draining
Draining buffers pending serialization/handover.
void invalidateVMID(uint16_t vmid)
#define panic(...)
This implements a cprintf based panic() function.
Generated on Tue Jun 22 2021 15:28:27 for gem5 by doxygen 1.8.17