59 #include "debug/Checkpoint.hh" 60 #include "debug/TLB.hh" 61 #include "debug/TLBVerbose.hh" 65 #include "params/ArmTLB.hh" 73 TLB::TLB(
const ArmTLBParams *
p)
75 isStage2(p->is_stage2), stage2Req(false), stage2DescReq(false), _attr(0),
76 directToStage2(false), tableWalker(p->walker), stage2Tlb(NULL),
77 stage2Mmu(NULL),
test(nullptr), rangeMRU(1),
78 aarch64(false), aarch64EL(
EL0), isPriv(false), isSecure(false),
79 isHyp(false),
asid(0), vmid(0), hcr(0), dacr(0),
80 miscRegValid(false), miscRegContext(0), curTranType(NormalTran)
136 const Addr paddr = req->getPaddr();
141 req->setLocalAccessor(
145 PseudoInst::pseudoInst<PseudoInstABI>(tc, func, ret);
165 while (retval == NULL && x <
size) {
166 if ((!ignore_asn &&
table[x].match(va, asn, vmid, hyp, secure,
false,
168 (ignore_asn &&
table[x].match(va, vmid, hyp, secure, target_el))) {
173 for (
int i = x;
i > 0;
i--)
175 table[0] = tmp_entry;
185 DPRINTF(TLBVerbose,
"Lookup %#x, asn %#x -> %s vmn 0x%x hyp %d secure %d " 186 "ppn %#x size: %#x pa: %#x ap:%d ns:%d nstid:%d g:%d asid: %d " 188 va, asn, retval ?
"hit" :
"miss", vmid, hyp, secure,
189 retval ? retval->
pfn : 0, retval ? retval->
size : 0,
190 retval ? retval->
pAddr(va) : 0, retval ? retval->
ap : 0,
191 retval ? retval->
ns : 0, retval ? retval->
nstid : 0,
192 retval ? retval->
global : 0, retval ? retval->
asid : 0,
193 retval ? retval->
el : 0);
202 DPRINTF(
TLB,
"Inserting entry into TLB with pfn:%#x size:%#x vpn: %#x" 203 " asid:%d vmid:%d N:%d global:%d valid:%d nc:%d xn:%d" 204 " ap:%#x domain:%#x ns:%d nstid:%d isHyp:%d\n", entry.
pfn,
211 DPRINTF(
TLB,
" - Replacing Valid entry %#x, asn %d vmn %d ppn %#x " 212 "size: %#x ap:%d ns:%d nstid:%d g:%d isHyp:%d el: %d\n",
221 for (
int i =
size - 1;
i > 0; --
i)
247 DPRINTF(
TLB,
"Flushing all TLB entries (%s lookup)\n",
248 (secure_lookup ?
"secure" :
"non-secure"));
253 const bool el_match = ignore_el ?
256 if (te->
valid && secure_lookup == !te->
nstid &&
257 (te->
vmid ==
vmid || secure_lookup) && el_match) {
278 bool hyp = target_el ==
EL2;
280 DPRINTF(
TLB,
"Flushing all NS TLB entries (%s lookup)\n",
281 (hyp ?
"hyp" :
"non-hyp"));
286 const bool el_match = ignore_el ?
310 DPRINTF(
TLB,
"Flushing TLB entries with mva: %#x, asid: %#x " 311 "(%s lookup)\n", mva, asn, (secure_lookup ?
312 "secure" :
"non-secure"));
313 _flushMva(mva, asn, secure_lookup,
false, target_el);
320 DPRINTF(
TLB,
"Flushing TLB entries with asid: %#x (%s lookup)\n", asn,
321 (secure_lookup ?
"secure" :
"non-secure"));
328 if (te->
valid && te->
asid == asn && secure_lookup == !te->
nstid &&
329 (te->
vmid ==
vmid || secure_lookup) &&
344 DPRINTF(
TLB,
"Flushing TLB entries with mva: %#x (%s lookup)\n", mva,
345 (secure_lookup ?
"secure" :
"non-secure"));
346 _flushMva(mva, 0xbeef, secure_lookup,
true, target_el);
358 bool hyp = target_el ==
EL2;
360 te =
lookup(mva, asn,
vmid, hyp, secure_lookup,
false, ignore_asn,
363 if (secure_lookup == !te->
nstid) {
368 te =
lookup(mva, asn,
vmid, hyp, secure_lookup,
false, ignore_asn,
391 TLB *otlb =
dynamic_cast<TLB*
>(_otlb);
408 panic(
"Incompatible TLB type!");
418 .
desc(
"ITB inst hits")
423 .
desc(
"ITB inst misses")
428 .
desc(
"ITB inst accesses")
433 .
desc(
"DTB read hits")
438 .
desc(
"DTB read misses")
443 .
desc(
"DTB read accesses")
448 .
desc(
"DTB write hits")
453 .
desc(
"DTB write misses")
458 .
desc(
"DTB write accesses")
473 .
desc(
"DTB accesses")
478 .
desc(
"Number of times complete TLB was flushed")
483 .
desc(
"Number of times TLB was flushed by MVA")
487 .
name(
name() +
".flush_tlb_mva_asid")
488 .
desc(
"Number of times TLB was flushed by MVA & ASID")
493 .
desc(
"Number of times TLB was flushed by ASID")
498 .
desc(
"Number of entries that have been flushed from TLB")
503 .
desc(
"Number of TLB faults due to alignment restrictions")
508 .
desc(
"Number of TLB faults due to prefetch")
513 .
desc(
"Number of TLB faults due to domain restrictions")
518 .
desc(
"Number of TLB faults due to permissions restrictions")
537 Translation *translation,
bool &delay,
bool timing)
540 Addr vaddr_tainted = req->getVaddr();
546 vaddr = vaddr_tainted;
549 bool is_fetch = (mode ==
Execute);
550 bool is_write = (mode ==
Write);
556 return std::make_shared<DataAbort>(
569 return std::make_shared<GenericPageTableFault>(vaddr_tainted);
570 req->setPaddr(paddr);
580 if (req->isCacheMaintenance()) {
586 bool is_fetch = (mode ==
Execute);
587 bool is_write = (mode ==
Write);
600 return std::make_shared<DataAbort>(
612 return std::make_shared<DataAbort>(
622 if (req->isPrefetch()) {
625 return std::make_shared<PrefetchAbort>(
632 switch ((
dacr >> (static_cast<uint8_t>(te->
domain) * 2)) & 0x3) {
635 DPRINTF(
TLB,
"TLB Fault: Data abort on domain. DACR: %#x" 636 " domain: %#x write:%d\n",
dacr,
637 static_cast<uint8_t>(te->
domain), is_write);
642 return std::make_shared<PrefetchAbort>(
647 return std::make_shared<DataAbort>(
655 panic(
"UNPRED domain\n");
663 uint8_t hap = te->
hap;
669 bool isWritable =
true;
678 DPRINTF(
TLB,
"Access permissions 0, checking rs:%#x\n",
681 switch ((
int)
sctlr.rs) {
686 abt = is_write || !is_priv;
702 abt = !is_priv && is_write;
703 isWritable = is_priv;
709 panic(
"UNPRED premissions\n");
711 abt = !is_priv || is_write;
720 panic(
"Unknown permissions %#x\n", ap);
724 bool hapAbt = is_write ? !(hap & 2) : !(hap & 1);
725 bool xn = te->
xn || (isWritable &&
sctlr.wxn) ||
726 (ap == 3 &&
sctlr.uwxn && is_priv);
727 if (is_fetch && (abt || xn ||
731 DPRINTF(
TLB,
"TLB Fault: Prefetch abort on permission check. AP:%d " 732 "priv:%d write:%d ns:%d sif:%d sctlr.afe: %d \n",
733 ap, is_priv, is_write, te->
ns,
scr.sif,
sctlr.afe);
736 return std::make_shared<PrefetchAbort>(
740 }
else if (abt | hapAbt) {
742 DPRINTF(
TLB,
"TLB Fault: Data abort on permission check. AP:%d priv:%d" 743 " write:%d\n", ap, is_priv, is_write);
744 return std::make_shared<DataAbort>(
768 Addr vaddr_tainted = req->getVaddr();
773 bool is_fetch = (mode ==
Execute);
775 bool is_write = !req->isCacheClean() && mode ==
Write;
776 bool is_atomic = req->isAtomic();
787 return std::make_shared<DataAbort>(
788 vaddr_tainted, te->
domain, is_write,
799 return std::make_shared<DataAbort>(
802 is_atomic ? false : is_write,
811 if (req->isPrefetch()) {
814 return std::make_shared<PrefetchAbort>(
821 uint8_t ap = 0x3 & (te->
ap);
825 uint8_t pxn = te->
pxn;
826 bool r = !is_write && !is_fetch;
834 bool grant_read =
true;
835 DPRINTF(TLBVerbose,
"Checking permissions: ap:%d, xn:%d, pxn:%d, r:%d, " 836 "w:%d, x:%d\n", ap, xn, pxn, r, w, x);
843 uint8_t hap = 0x3 & te->
hap;
844 grant_read = hap & 0x1;
847 grant = !
sctlr.wxn && !xn;
848 }
else if (is_write) {
857 grant_read = ap & 0x1;
858 uint8_t
perm = (ap << 2) | (xn << 1) | pxn;
868 grant = r || w || (x && !
sctlr.wxn);
895 uint8_t
perm = (ap << 2) | (xn << 1) | pxn;
899 grant = r || w || (x && !
sctlr.wxn);
937 uint8_t
perm = (ap & 0x2) | xn;
940 grant = r || w || (x && !
sctlr.wxn) ;
962 DPRINTF(
TLB,
"TLB Fault: Prefetch abort on permission check. " 963 "AP:%d priv:%d write:%d ns:%d sif:%d " 965 ap, is_priv, is_write, te->
ns,
scr.sif,
sctlr.afe);
968 return std::make_shared<PrefetchAbort>(
974 DPRINTF(
TLB,
"TLB Fault: Data abort on permission check. AP:%d " 975 "priv:%d write:%d\n", ap, is_priv, is_write);
976 return std::make_shared<DataAbort>(
977 vaddr_tainted, te->
domain,
978 (is_atomic && !grant_read) ?
false : is_write,
998 if (mmfr1.pan &&
cpsr.pan && (ap & 0x1) && mode !=
Execute &&
999 (!req->isCacheMaintenance() ||
1011 bool is_fetch = (mode ==
Execute);
1012 req->setPaddr(vaddr);
1019 if (long_desc_format ||
sctlr.tre == 0 ||
nmrr.ir0 == 0 ||
1020 nmrr.or0 == 0 ||
prrr.tr0 != 0x2) {
1021 if (!req->isCacheMaintenance()) {
1047 DPRINTF(TLBVerbose,
"(No MMU) setting memory attributes: shareable: " 1048 "%d, innerAttrs: %d, outerAttrs: %d, isStage2: %d\n",
1058 Translation *translation,
bool &delay,
bool timing,
1063 bool is_fetch = (mode ==
Execute);
1067 Addr vaddr_tainted = req->getVaddr();
1070 functional, &mergeTe);
1072 if ((te == NULL) && (fault ==
NoFault)) delay =
true;
1079 "Setting memory attributes: shareable: %d, innerAttrs: %d, " 1080 "outerAttrs: %d, mtype: %d, isStage2: %d\n",
1105 bool is_write = (mode ==
Write);
1106 return std::make_shared<DataAbort>(
1129 Translation *translation,
bool &delay,
bool timing,
1133 assert(!(timing && functional));
1137 Addr vaddr_tainted = req->getVaddr();
1143 vaddr = vaddr_tainted;
1146 bool is_fetch = (mode ==
Execute);
1147 bool is_write = (mode ==
Write);
1153 "CPSR is priv:%d UserMode:%d secure:%d S1S2NsTran:%d\n",
1156 DPRINTF(
TLB,
"translateFs addr %#x, mode %d, st2 %d, scr %#x sctlr %#x " 1157 "flags %#lx tranType 0x%x\n", vaddr_tainted, mode,
isStage2,
1160 if ((req->isInstFetch() && (!
sctlr.i)) ||
1161 ((!req->isInstFetch()) && (!
sctlr.c))){
1162 if (!req->isCacheMaintenance()) {
1171 return std::make_shared<DataAbort>(
1185 DPRINTF(TLBVerbose,
"Translating %s=%#x context=%d\n",
1189 functional, vaddr, tranMethod);
1207 fault =
translateFs(req, tc, mode, NULL, delay,
false, tranType);
1209 fault =
translateSe(req, tc, mode, NULL, delay,
false);
1228 fault =
translateFs(req, tc, mode, NULL, delay,
false, tranType,
true);
1230 fault =
translateSe(req, tc, mode, NULL, delay,
false);
1247 assert(translation);
1260 fault =
translateFs(req, tc, mode, translation, delay,
true, tranType);
1262 fault =
translateSe(req, tc, mode, translation, delay,
true);
1263 DPRINTF(TLBVerbose,
"Translation returning delay=%d fault=%d\n", delay, fault !=
1271 if (translation && (callFromS2 || !
stage2Req || req->hasPaddr() || fault !=
NoFault)) {
1273 translation->
finish(fault, req, tc, mode);
1297 DPRINTF(TLBVerbose,
"TLB variables changed!\n");
1311 switch (aarch64EL) {
1317 uint64_t ttbr_asid =
ttbcr.a1 ?
1343 isHyp &= (tranType &
S1CTran) == 0;
1349 !(tranType &
S1CTran) && (aarch64EL <
EL2) &&
1377 asid = context_id.asid;
1392 isHyp &= (tranType &
S1CTran) == 0;
1441 panic(
"Unknown translation mode!\n");
1447 Translation *translation,
bool timing,
bool functional,
1455 bool is_fetch = (mode ==
Execute);
1456 bool is_write = (mode ==
Write);
1458 Addr vaddr_tainted = req->getVaddr();
1465 vaddr = vaddr_tainted;
1469 if (req->isPrefetch()) {
1474 return std::make_shared<PrefetchAbort>(
1487 DPRINTF(
TLB,
"TLB Miss: Starting hardware table walker for %#x(%d:%d)\n",
1491 translation, timing, functional, is_secure,
1494 if (timing || fault !=
NoFault) {
1516 Translation *translation,
bool timing,
bool functional,
1526 fault =
getTE(&s2Te, req, tc, mode, translation, timing, functional,
1529 if ((s2Te != NULL) && (fault ==
NoFault)) {
1541 Addr vaddr_tainted = req->getVaddr();
1544 fault =
getTE(&s1Te, req, tc, mode, translation, timing, functional,
1547 if ((s1Te != NULL) && (fault ==
NoFault)) {
1555 req, translation, mode, timing, functional,
curTranType);
1556 fault = s2Lookup->
getTe(tc, mergeTe);
1571 DPRINTF(TLBVerbose,
"s2TLB: reqVa %#x, reqPa %#x, fault %p\n",
1572 vaddr_tainted, req->hasPaddr() ? req->getPaddr() : ~0, fault);
1592 fatal_if(!ti,
"%s is not a valid ARM TLB tester\n", _ti->
name());
1601 if (!
test || !req->hasSize() || req->getSize() == 0 ||
1602 req->isCacheMaintenance()) {
1605 return test->translationCheck(req,
isPriv, mode, domain);
1616 return test->walkCheck(pa, size, va, is_secure,
isPriv, mode,
1617 domain, lookup_level);
1623 ArmTLBParams::create()
#define panic(...)
This implements a cprintf based panic() function.
ProbePoints::PMUUPtr ppRefills
PMU probe for TLB refills.
Ports are used to interface objects to each other.
void translateTiming(const RequestPtr &req, ThreadContext *tc, Translation *translation, Mode mode, ArmTranslationType tranType)
Fault checkPermissions64(TlbEntry *te, const RequestPtr &req, Mode mode, ThreadContext *tc)
Fault getResultTe(TlbEntry **te, const RequestPtr &req, ThreadContext *tc, Mode mode, Translation *translation, bool timing, bool functional, TlbEntry *mergeTe)
decltype(nullptr) constexpr NoFault
Cycles is a wrapper class for representing cycle counts, i.e.
Fault getTE(TlbEntry **te, const RequestPtr &req, ThreadContext *tc, Mode mode, Translation *translation, bool timing, bool functional, bool is_secure, ArmTranslationType tranType)
void flushAllNs(ExceptionLevel target_el, bool ignore_el=false)
Remove all entries in the non secure world, depending on whether they were allocated in hyp mode or n...
void flushMva(Addr mva, bool secure_lookup, ExceptionLevel target_el)
Remove all entries that match the va regardless of asn.
DmaPort & getDMAPort()
Get the port that ultimately belongs to the stage-two MMU, but is used by the two table walkers...
Declaration of a request, the overall memory request consisting of the parts of the request that are ...
bool contains(const Addr &a) const
Determine if the range contains an address.
std::shared_ptr< Request > RequestPtr
void flushAllSecurity(bool secure_lookup, ExceptionLevel target_el, bool ignore_el=false)
Reset the entire TLB.
virtual void markDelayed()=0
Signal that the translation has been delayed due to a hw page table walk.
static void decodeAddrOffset(Addr offset, uint8_t &func)
The request targets the secure memory space.
bool FullSystem
The FullSystem variable can be used to determine the current mode of simulation.
virtual Process * getProcessPtr()=0
static ExceptionLevel tranTypeEL(CPSR cpsr, ArmTranslationType type)
Determine the EL to use for the purpose of a translation given a specific translation type...
Overload hash function for BasicBlockRange type.
std::string print() const
void regStats() override
Callback to set stat parameters.
Stats::Scalar prefetchFaults
bool checkPAN(ThreadContext *tc, uint8_t ap, const RequestPtr &req, Mode mode)
void drainResume() override
Resume execution after a successful drain.
Port * getTableWalkerPort() override
Get the table walker port.
Addr purifyTaggedAddr(Addr addr, ThreadContext *tc, ExceptionLevel el, TCR tcr, bool isInstr)
Removes the tag from tagged addresses if that mode is enabled.
ThreadContext is the external interface to all thread state for anything outside of the CPU...
static ExceptionLevel currEL(const ThreadContext *tc)
Fault walk(const RequestPtr &req, ThreadContext *tc, uint16_t asid, uint8_t _vmid, bool _isHyp, TLB::Mode mode, TLB::Translation *_trans, bool timing, bool functional, bool secure, TLB::ArmTranslationType tranType, bool _stage2Req)
The request is to an uncacheable address.
void setLE(T v)
Set the value in the data pointer to v as little endian.
Fault testWalk(Addr pa, Addr size, Addr va, bool is_secure, Mode mode, TlbEntry::DomainType domain, LookupLevel lookup_level)
Stats::Formula writeAccesses
TableWalker * tableWalker
bool haveVirtualization() const
Stats::Scalar flushedEntries
This is a write that is targeted and zeroing an entire cache block.
bool ELIs64(ThreadContext *tc, ExceptionLevel el)
bool haveLargeAsid64() const
Fault getTe(ThreadContext *tc, TlbEntry *destTe)
Fault finalizePhysical(const RequestPtr &req, ThreadContext *tc, Mode mode) const override
Do post-translation physical address finalization.
Stats::Scalar permsFaults
Stats::Scalar writeMisses
bool translate(Addr vaddr, Addr &paddr)
Translate function.
void _flushMva(Addr mva, uint64_t asn, bool secure_lookup, bool ignore_asn, ExceptionLevel target_el)
Remove any entries that match both a va and asn.
virtual void annotate(AnnotationIDs id, uint64_t val)
void flushAsid(uint64_t asn, bool secure_lookup, ExceptionLevel target_el)
Remove any entries that match the asn.
void setMMU(Stage2MMU *m, MasterID master_id)
Stats::Scalar domainFaults
void updateMiscReg(ThreadContext *tc, ArmTranslationType tranType=NormalTran)
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
Fault translateFs(const RequestPtr &req, ThreadContext *tc, Mode mode, Translation *translation, bool &delay, bool timing, ArmTranslationType tranType, bool functional=false)
virtual void finish(const Fault &fault, const RequestPtr &req, ThreadContext *tc, Mode mode)=0
bool checkELMatch(ExceptionLevel target_el) const
Addr pAddr(Addr va) const
bool translateFunctional(ThreadContext *tc, Addr vaddr, Addr &paddr)
Do a functional lookup on the TLB (for debugging) and don't modify any internal state.
bool haveVirtualization() const
Returns true if this system implements the virtualization Extensions.
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Declaration of IniFile object.
Fault checkPermissions(TlbEntry *te, const RequestPtr &req, Mode mode)
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
Stats::Scalar flushTlbMva
void init() override
setup all the back pointers
ProbeManager * getProbeManager()
Get the probe manager for this object.
Stats::Scalar flushTlbMvaAsid
Derived & name(const std::string &name)
Set the name and marks this stat to print at the end of simulation.
const AddrRange & m5opRange() const
Range used by memory-mapped m5 pseudo-ops if enabled.
virtual const std::string name() const
ProbePointArg generates a point for the class of Arg.
void flushIpaVmid(Addr ipa, bool secure_lookup, ExceptionLevel target_el)
Invalidate all entries in the stage 2 TLB that match the given ipa and the current VMID...
Stats::Formula readAccesses
EmulationPageTable * pTable
Stats::Scalar alignFaults
void flushMvaAsid(Addr mva, uint64_t asn, bool secure_lookup, ExceptionLevel target_el)
Remove any entries that match both a va and asn.
Declarations of a non-full system Page Table.
bool longDescFormatInUse(ThreadContext *tc)
TlbEntry * lookup(Addr vpn, uint16_t asn, uint8_t vmid, bool hyp, bool secure, bool functional, bool ignore_asn, ExceptionLevel target_el)
Lookup an entry in the TLB.
void insert(Addr vaddr, TlbEntry &pte)
void regProbePoints() override
Register probe points for this object.
Fault translateSe(const RequestPtr &req, ThreadContext *tc, Mode mode, Translation *translation, bool &delay, bool timing)
void setMMU(Stage2MMU *m, MasterID master_id)
int snsBankedIndex(MiscRegIndex reg, ThreadContext *tc)
Fault translateMmuOff(ThreadContext *tc, const RequestPtr &req, Mode mode, TLB::ArmTranslationType tranType, Addr vaddr, bool long_desc_format)
virtual ContextID contextId() const =0
Stats::Formula instAccesses
The request is required to be strictly ordered by CPU models and is non-speculative.
Addr start() const
Get the start address of the range.
Derived & desc(const std::string &_desc)
Set the description and marks this stat to print at the end of simulation.
Stats::Scalar flushTlbAsid
virtual void regStats()
Callback to set stat parameters.
bool inSecureState(ThreadContext *tc)
Fault translateComplete(const RequestPtr &req, ThreadContext *tc, Translation *translation, Mode mode, ArmTranslationType tranType, bool callFromS2)
T bits(T val, int first, int last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it...
void takeOverFrom(BaseTLB *otlb) override
Take over from an old tlb context.
ArmTranslationType curTranType
virtual RegVal readMiscReg(RegIndex misc_reg)=0
void setAttr(uint64_t attr)
Accessor functions for memory attributes for last accessed TLB entry.
std::shared_ptr< FaultBase > Fault
Fault translateMmuOn(ThreadContext *tc, const RequestPtr &req, Mode mode, Translation *translation, bool &delay, bool timing, bool functional, Addr vaddr, ArmFault::TranMethod tranMethod)
Fault translateAtomic(const RequestPtr &req, ThreadContext *tc, Mode mode, ArmTranslationType tranType)
Abstract superclass for simulation objects.
void setAttributes(bool lpae)
void setTestInterface(SimObject *ti)
Fault testTranslation(const RequestPtr &req, Mode mode, TlbEntry::DomainType domain)