Go to the documentation of this file.
63 #include "debug/Checkpoint.hh"
64 #include "debug/TLB.hh"
65 #include "debug/TLBVerbose.hh"
69 #include "params/ArmTLB.hh"
78 isStage2(
p.is_stage2), stage2Req(false), stage2DescReq(false), _attr(0),
79 directToStage2(false), tableWalker(
p.walker), stage2Tlb(NULL),
80 stage2Mmu(NULL),
test(nullptr), stats(this), rangeMRU(1),
82 isHyp(false),
asid(0), vmid(0), hcr(0), dacr(0),
83 miscRegValid(false), miscRegContext(0), curTranType(NormalTran)
140 const Addr paddr = req->getPaddr();
145 req->setLocalAccessor(
150 PseudoInst::pseudoInst<RegABI64>(tc, func, ret);
152 PseudoInst::pseudoInst<RegABI32>(tc, func, ret);
175 while (retval == NULL &&
x <
size) {
176 if ((!ignore_asn &&
table[
x].match(
va, asn,
vmid, hyp, secure,
false,
177 target_el, in_host)) ||
178 (ignore_asn &&
table[
x].match(
va,
vmid, hyp, secure, target_el,
184 for (
int i =
x;
i > 0;
i--)
186 table[0] = tmp_entry;
196 DPRINTF(TLBVerbose,
"Lookup %#x, asn %#x -> %s vmn 0x%x hyp %d secure %d "
197 "ppn %#x size: %#x pa: %#x ap:%d ns:%d nstid:%d g:%d asid: %d "
199 va, asn, retval ?
"hit" :
"miss",
vmid, hyp, secure,
200 retval ? retval->
pfn : 0, retval ? retval->
size : 0,
201 retval ? retval->
pAddr(
va) : 0, retval ? retval->
ap : 0,
202 retval ? retval->
ns : 0, retval ? retval->
nstid : 0,
203 retval ? retval->
global : 0, retval ? retval->
asid : 0,
204 retval ? retval->
el : 0);
213 DPRINTF(
TLB,
"Inserting entry into TLB with pfn:%#x size:%#x vpn: %#x"
214 " asid:%d vmid:%d N:%d global:%d valid:%d nc:%d xn:%d"
215 " ap:%#x domain:%#x ns:%d nstid:%d isHyp:%d\n", entry.
pfn,
222 DPRINTF(
TLB,
" - Replacing Valid entry %#x, asn %d vmn %d ppn %#x "
223 "size: %#x ap:%d ns:%d nstid:%d g:%d isHyp:%d el: %d\n",
232 for (
int i =
size - 1;
i > 0; --
i)
280 DPRINTF(
TLB,
"Flushing all TLB entries (%s lookup)\n",
286 const bool el_match =
te->checkELMatch(
310 DPRINTF(
TLB,
"Flushing all TLB entries (%s lookup)\n",
316 const bool el_match =
te->checkELMatch(
340 DPRINTF(
TLB,
"Flushing all TLB entries (%s lookup)\n",
346 const bool el_match =
te->checkELMatch(
372 DPRINTF(
TLB,
"Flushing all NS TLB entries (%s lookup)\n",
373 (hyp ?
"hyp" :
"non-hyp"));
378 const bool el_match =
te->checkELMatch(tlbi_op.
targetEL,
false);
380 if (
te->valid &&
te->nstid &&
te->isHyp == hyp && el_match) {
400 DPRINTF(
TLB,
"Flushing TLB entries with mva: %#x, asid: %#x "
401 "(%s lookup)\n", tlbi_op.
addr, tlbi_op.
asid,
411 DPRINTF(
TLB,
"Flushing TLB entries with asid: %#x (%s lookup)\n",
419 if (
te->valid &&
te->asid == tlbi_op.
asid &&
436 DPRINTF(
TLB,
"Flushing TLB entries with mva: %#x (%s lookup)\n",
452 bool hyp = target_el ==
EL2;
454 te =
lookup(mva, asn,
vmid, hyp, secure_lookup,
true, ignore_asn,
457 if (secure_lookup == !
te->nstid) {
462 te =
lookup(mva, asn,
vmid, hyp, secure_lookup,
true, ignore_asn,
487 TLB *otlb =
dynamic_cast<TLB*
>(_otlb);
504 panic(
"Incompatible TLB type!");
509 :
Stats::Group(parent),
517 "Number of times an entry is inserted into the TLB"),
520 "Number of times TLB was flushed by MVA"),
522 "Number of times TLB was flushed by MVA & ASID"),
524 "Number of times TLB was flushed by ASID"),
526 "Number of entries that have been flushed from TLB"),
528 "Number of TLB faults due to alignment restrictions"),
530 "Number of TLB faults due to prefetch"),
532 "Number of TLB faults due to domain restrictions"),
534 "Number of TLB faults due to permissions restrictions"),
536 readHits + readMisses),
538 writeHits + writeMisses),
540 instHits + instMisses),
542 readHits + writeHits + instHits),
544 readMisses + writeMisses + instMisses),
546 readAccesses + writeAccesses + instAccesses)
558 Translation *translation,
bool &delay,
bool timing)
561 Addr vaddr_tainted = req->getVaddr();
567 vaddr = vaddr_tainted;
577 return std::make_shared<DataAbort>(
589 if (!
p->pTable->translate(
vaddr, paddr))
590 return std::make_shared<GenericPageTableFault>(vaddr_tainted);
591 req->setPaddr(paddr);
601 if (req->isCacheMaintenance()) {
621 return std::make_shared<DataAbort>(
633 return std::make_shared<DataAbort>(
641 if (
te->nonCacheable) {
643 if (req->isPrefetch()) {
646 return std::make_shared<PrefetchAbort>(
652 if (!
te->longDescFormat) {
653 switch ((
dacr >> (
static_cast<uint8_t
>(
te->domain) * 2)) & 0x3) {
656 DPRINTF(
TLB,
"TLB Fault: Data abort on domain. DACR: %#x"
657 " domain: %#x write:%d\n",
dacr,
658 static_cast<uint8_t
>(
te->domain), is_write);
663 return std::make_shared<PrefetchAbort>(
668 return std::make_shared<DataAbort>(
676 panic(
"UNPRED domain\n");
683 uint8_t ap =
te->longDescFormat ?
te->ap << 1 :
te->ap;
684 uint8_t hap =
te->hap;
686 if (
sctlr.afe == 1 ||
te->longDescFormat)
690 bool isWritable =
true;
699 DPRINTF(
TLB,
"Access permissions 0, checking rs:%#x\n",
702 switch ((
int)
sctlr.rs) {
707 abt = is_write || !is_priv;
723 abt = !is_priv && is_write;
724 isWritable = is_priv;
730 panic(
"UNPRED premissions\n");
732 abt = !is_priv || is_write;
741 panic(
"Unknown permissions %#x\n", ap);
745 bool hapAbt = is_write ? !(hap & 2) : !(hap & 1);
746 bool xn =
te->xn || (isWritable &&
sctlr.wxn) ||
747 (ap == 3 &&
sctlr.uwxn && is_priv);
748 if (is_fetch && (abt || xn ||
749 (
te->longDescFormat &&
te->pxn && is_priv) ||
752 DPRINTF(
TLB,
"TLB Fault: Prefetch abort on permission check. AP:%d "
753 "priv:%d write:%d ns:%d sif:%d sctlr.afe: %d \n",
754 ap, is_priv, is_write,
te->ns,
scr.sif,
sctlr.afe);
757 return std::make_shared<PrefetchAbort>(
761 }
else if (abt | hapAbt) {
763 DPRINTF(
TLB,
"TLB Fault: Data abort on permission check. AP:%d priv:%d"
764 " write:%d\n", ap, is_priv, is_write);
765 return std::make_shared<DataAbort>(
789 Addr vaddr_tainted = req->getVaddr();
796 bool is_write = !req->isCacheClean() &&
mode ==
Write;
797 bool is_atomic = req->isAtomic();
808 return std::make_shared<DataAbort>(
809 vaddr_tainted,
te->domain, is_write,
820 return std::make_shared<DataAbort>(
823 is_atomic ?
false : is_write,
830 if (
te->nonCacheable) {
832 if (req->isPrefetch()) {
835 return std::make_shared<PrefetchAbort>(
842 uint8_t ap = 0x3 & (
te->ap);
847 uint8_t pxn =
te->pxn;
848 bool r = (!is_write && !is_fetch);
859 bool grant_read =
true;
860 DPRINTF(TLBVerbose,
"Checking permissions: ap:%d, xn:%d, pxn:%d, r:%d, "
861 "w:%d, x:%d, is_priv: %d, wxn: %d\n", ap, xn,
862 pxn,
r,
w,
x, is_priv,
wxn);
869 uint8_t hap = 0x3 &
te->hap;
870 grant_read = hap & 0x1;
874 }
else if (is_atomic) {
876 }
else if (is_write) {
885 grant_read = ap & 0x1;
886 uint8_t
perm = (ap << 2) | (xn << 1) | pxn;
896 grant =
r ||
w || (
x && !
wxn);
923 uint8_t
perm = (ap << 2) | (xn << 1) | pxn;
927 grant =
r ||
w || (
x && !
wxn);
965 uint8_t
perm = (ap & 0x2) | xn;
968 grant =
r ||
w || (
x && !
wxn);
990 DPRINTF(
TLB,
"TLB Fault: Prefetch abort on permission check. "
991 "AP:%d priv:%d write:%d ns:%d sif:%d "
993 ap, is_priv, is_write,
te->ns,
scr.sif,
sctlr.afe);
996 return std::make_shared<PrefetchAbort>(
1002 DPRINTF(
TLB,
"TLB Fault: Data abort on permission check. AP:%d "
1003 "priv:%d write:%d\n", ap, is_priv, is_write);
1004 return std::make_shared<DataAbort>(
1005 vaddr_tainted,
te->domain,
1006 (is_atomic && !grant_read) ?
false : is_write,
1027 (!req->isCacheMaintenance() ||
1040 bool is_atomic = req->isAtomic();
1041 req->setPaddr(
vaddr);
1055 f = std::make_shared<PrefetchAbort>(
vaddr,
1058 f = std::make_shared<DataAbort>(
vaddr,
1067 if (long_desc_format ||
sctlr.tre == 0 ||
nmrr.ir0 == 0 ||
1068 nmrr.or0 == 0 ||
prrr.tr0 != 0x2) {
1069 if (!req->isCacheMaintenance()) {
1079 &&
hcr.e2h == 1 &&
hcr.tge == 1) ? 0:
hcr.dc;
1086 temp_te.
innerAttrs = i_cacheability? 0x2: 0x0;
1087 temp_te.
outerAttrs = i_cacheability? 0x2: 0x0;
1098 DPRINTF(TLBVerbose,
"(No MMU) setting memory attributes: shareable: "
1099 "%d, innerAttrs: %d, outerAttrs: %d, isStage2: %d\n",
1109 Translation *translation,
bool &delay,
bool timing,
1118 Addr vaddr_tainted = req->getVaddr();
1121 functional, &mergeTe);
1123 if ((
te == NULL) && (fault ==
NoFault)) delay =
true;
1130 "Setting memory attributes: shareable: %d, innerAttrs: %d, "
1131 "outerAttrs: %d, mtype: %d, isStage2: %d\n",
1132 te->shareable,
te->innerAttrs,
te->outerAttrs,
1136 if (
te->nonCacheable && !req->isCacheMaintenance())
1151 if (!is_fetch && fault ==
NoFault &&
1158 return std::make_shared<DataAbort>(
1181 Translation *translation,
bool &delay,
bool timing,
1185 assert(!(timing && functional));
1189 Addr vaddr_tainted = req->getVaddr();
1195 vaddr = vaddr_tainted;
1205 "CPSR is priv:%d UserMode:%d secure:%d S1S2NsTran:%d\n",
1208 DPRINTF(
TLB,
"translateFs addr %#x, mode %d, st2 %d, scr %#x sctlr %#x "
1209 "flags %#lx tranType 0x%x\n", vaddr_tainted,
mode,
isStage2,
1212 if ((req->isInstFetch() && (!
sctlr.i)) ||
1213 ((!req->isInstFetch()) && (!
sctlr.c))){
1214 if (!req->isCacheMaintenance()) {
1223 return std::make_shared<DataAbort>(
1235 else if (
hcr.dc == 1)
1244 DPRINTF(TLBVerbose,
"Translating %s=%#x context=%d\n",
1248 functional,
vaddr, tranMethod);
1254 if (
sd->enabled() && fault ==
NoFault) {
1255 fault =
sd->testDebug(tc, req,
mode);
1296 fault =
translateFs(req, tc,
mode, NULL, delay,
false, tranType,
true);
1315 assert(translation);
1328 fault =
translateFs(req, tc,
mode, translation, delay,
true, tranType);
1331 DPRINTF(TLBVerbose,
"Translation returning delay=%d fault=%d\n", delay, fault !=
1340 if (translation && (callFromS2 || !
stage2Req || req->hasPaddr() ||
1367 DPRINTF(TLBVerbose,
"TLB variables changed!\n");
1388 uint64_t ttbr_asid =
ttbcr.a1 ?
1397 uint64_t ttbr_asid =
ttbcr.a1 ?
1409 uint64_t ttbr_asid =
ttbcr.a1 ?
1421 uint64_t ttbr_asid =
ttbcr.a1 ?
1492 asid = context_id.asid;
1557 panic(
"Unknown translation mode!\n");
1563 Translation *translation,
bool timing,
bool functional,
1574 Addr vaddr_tainted = req->getVaddr();
1581 vaddr = vaddr_tainted;
1586 if (req->isPrefetch()) {
1591 return std::make_shared<PrefetchAbort>(
1604 DPRINTF(
TLB,
"TLB Miss: Starting hardware table walker for %#x(%d:%d)\n",
1608 translation, timing, functional, is_secure,
1611 if (timing || fault !=
NoFault) {
1634 Translation *translation,
bool timing,
bool functional,
1644 fault =
getTE(&s2Te, req, tc,
mode, translation, timing, functional,
1647 if ((s2Te != NULL) && (fault ==
NoFault)) {
1659 Addr vaddr_tainted = req->getVaddr();
1662 fault =
getTE(&s1Te, req, tc,
mode, translation, timing, functional,
1665 if ((s1Te != NULL) && (fault ==
NoFault)) {
1675 fault = s2Lookup->
getTe(tc, mergeTe);
1690 DPRINTF(TLBVerbose,
"s2TLB: reqVa %#x, reqPa %#x, fault %p\n",
1691 vaddr_tainted, req->hasPaddr() ? req->getPaddr() : ~0, fault);
1711 fatal_if(!
ti,
"%s is not a valid ARM TLB tester\n", _ti->
name());
1720 if (!
test || !req->hasSize() || req->getSize() == 0 ||
1721 req->isCacheMaintenance()) {
virtual void finish(const Fault &fault, const RequestPtr &req, ThreadContext *tc, Mode mode)=0
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)
int computeAddrTop(ThreadContext *tc, bool selbit, bool isInstr, TCR tcr, ExceptionLevel el)
@ SECURE
The request targets the secure memory space.
void setTestInterface(SimObject *ti)
bool HaveVirtHostExt(ThreadContext *tc)
TLBIVMALL makeStage2() const
Fault translateFs(const RequestPtr &req, ThreadContext *tc, Mode mode, Translation *translation, bool &delay, bool timing, ArmTranslationType tranType, bool functional=false)
TlbEntry * lookup(Addr vpn, uint16_t asn, uint8_t vmid, bool hyp, bool secure, bool functional, bool ignore_asn, ExceptionLevel target_el, bool in_host)
Lookup an entry in the TLB.
Fault translateSe(const RequestPtr &req, ThreadContext *tc, Mode mode, Translation *translation, bool &delay, bool timing)
Fault translateMmuOn(ThreadContext *tc, const RequestPtr &req, Mode mode, Translation *translation, bool &delay, bool timing, bool functional, Addr vaddr, ArmFault::TranMethod tranMethod)
Fault finalizePhysical(const RequestPtr &req, ThreadContext *tc, Mode mode) const override
Do post-translation physical address finalization.
const AddrRange & m5opRange() const
Range used by memory-mapped m5 pseudo-ops if enabled.
int snsBankedIndex(MiscRegIndex reg, ThreadContext *tc)
Stats::Scalar prefetchFaults
Stats::Scalar flushTlbAsid
Stats::Scalar domainFaults
static ExceptionLevel currEL(const ThreadContext *tc)
@ CACHE_BLOCK_ZERO
This is a write that is targeted and zeroing an entire cache block.
bool haveLargeAsid64() const
ProbePointArg generates a point for the class of Arg.
Stats::Scalar writeMisses
void setAttr(uint64_t attr)
Accessor functions for memory attributes for last accessed TLB entry.
void drainResume() override
Resume execution after a successful drain.
bool contains(const Addr &a) const
Determine if the range contains an address.
std::shared_ptr< Request > RequestPtr
Implementaton of AArch64 TLBI VMALLE1(IS)/VMALLS112E1(IS) instructions.
ArmISA::TLB::TlbStats stats
static void decodeAddrOffset(Addr offset, uint8_t &func)
bool FullSystem
The FullSystem variable can be used to determine the current mode of simulation.
virtual Process * getProcessPtr()=0
Stats::Scalar flushTlbMva
Stats::Scalar permsFaults
TLB Invalidate by ASID match.
bool haveVirtualization() const
Returns true if this system implements the virtualization Extensions.
TLBIALL makeStage2() const
TLBIALLN makeStage2() const
bool ELIs64(ThreadContext *tc, ExceptionLevel el)
void _flushMva(Addr mva, uint64_t asn, bool secure_lookup, bool ignore_asn, ExceptionLevel target_el, bool in_host)
Remove any entries that match both a va and asn.
bool IsSecureEL2Enabled(ThreadContext *tc)
Fault translateMmuOff(ThreadContext *tc, const RequestPtr &req, Mode mode, TLB::ArmTranslationType tranType, Addr vaddr, bool long_desc_format)
Fault checkPermissions64(TlbEntry *te, const RequestPtr &req, Mode mode, ThreadContext *tc)
void regProbePoints() override
Register probe points for this object.
ArmTranslationType curTranType
virtual void annotate(AnnotationIDs id, uint64_t val)
TlbStats(Stats::Group *parent)
ThreadContext is the external interface to all thread state for anything outside of the CPU.
DmaPort & getDMAPort()
Get the port that ultimately belongs to the stage-two MMU, but is used by the two table walkers,...
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
std::shared_ptr< FaultBase > Fault
@ MISCREG_ID_AA64MMFR1_EL1
Ports are used to interface objects to each other.
@ STRICT_ORDER
The request is required to be strictly ordered by CPU models and is non-speculative.
Fault checkPermissions(TlbEntry *te, const RequestPtr &req, Mode mode)
virtual ContextID contextId() const =0
static ExceptionLevel tranTypeEL(CPSR cpsr, ArmTranslationType type)
Determine the EL to use for the purpose of a translation given a specific translation type.
Addr pAddr(Addr va) const
bool haveVirtualization() const
void setAttributes(bool lpae)
Stats::Scalar flushedEntries
bool longDescFormatInUse(ThreadContext *tc)
TLB Invalidate by VA, All ASID.
constexpr decltype(nullptr) NoFault
bool checkPAN(ThreadContext *tc, uint8_t ap, const RequestPtr &req, Mode mode)
Fault testWalk(Addr pa, Addr size, Addr va, bool is_secure, Mode mode, TlbEntry::DomainType domain, LookupLevel lookup_level)
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
TableWalker * tableWalker
Addr purifyTaggedAddr(Addr addr, ThreadContext *tc, ExceptionLevel el, TCR tcr, bool isInstr)
Removes the tag from tagged addresses if that mode is enabled.
Fault getTe(ThreadContext *tc, TlbEntry *destTe)
TLBIALLEL makeStage2() const
ProbeManager * getProbeManager()
Get the probe manager for this object.
void updateMiscReg(ThreadContext *tc, ArmTranslationType tranType=NormalTran)
SelfDebug * getSelfDebug() const
uint8_t physAddrRange() const
virtual const std::string name() const
void setMMU(Stage2MMU *m, RequestorID requestor_id)
virtual void markDelayed()=0
Signal that the translation has been delayed due to a hw page table walk.
TLBIMVAA makeStage2() const
TLBIIPA is basically a TLBIMVAA for stage2 TLBs.
Addr start() const
Get the start address of the range.
void init() override
setup all the back pointers
ProbePoints::PMUUPtr ppRefills
PMU probe for TLB refills.
void takeOverFrom(BaseTLB *otlb) override
Take over from an old tlb context.
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
@ UNCACHEABLE
The request is to an uncacheable address.
virtual RegVal readMiscReg(RegIndex misc_reg)=0
static bool haveEL(ThreadContext *tc, ArmISA::ExceptionLevel el)
Return true if the system implements a specific exception level.
Cycles is a wrapper class for representing cycle counts, i.e.
Implementaton of AArch64 TLBI ALLE(1,2,3)(IS) instructions.
void setLE(T v)
Set the value in the data pointer to v as little endian.
TLB Invalidate by Intermediate Physical Address.
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
Fault translateAtomic(const RequestPtr &req, ThreadContext *tc, Mode mode, ArmTranslationType tranType)
Port * getTableWalkerPort() override
Get the table walker port.
Fault testTranslation(const RequestPtr &req, Mode mode, TlbEntry::DomainType domain)
Fault getTE(TlbEntry **te, const RequestPtr &req, ThreadContext *tc, Mode mode, Translation *translation, bool timing, bool functional, bool is_secure, ArmTranslationType tranType)
Fault getResultTe(TlbEntry **te, const RequestPtr &req, ThreadContext *tc, Mode mode, Translation *translation, bool timing, bool functional, TlbEntry *mergeTe)
void setMMU(Stage2MMU *m, RequestorID requestor_id)
bool inAArch64(ThreadContext *tc)
void flush(const TLBIALL &tlbi_op)
Reset the entire TLB.
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
Stats::Scalar alignFaults
void flushAll() override
Reset the entire TLB.
Stats::Scalar flushTlbMvaAsid
bool translateFunctional(ThreadContext *tc, Addr vaddr, Addr &paddr)
Do a functional lookup on the TLB (for debugging) and don't modify any internal state.
Fault translateComplete(const RequestPtr &req, ThreadContext *tc, Translation *translation, Mode mode, ArmTranslationType tranType, bool callFromS2)
void translateTiming(const RequestPtr &req, ThreadContext *tc, Translation *translation, Mode mode, ArmTranslationType tranType)
bool isSecure(ThreadContext *tc)
#define panic(...)
This implements a cprintf based panic() function.
TLB Invalidate All, Non-Secure.
void insert(Addr vaddr, TlbEntry &pte)
Abstract superclass for simulation objects.
Generated on Tue Jun 22 2021 15:28:20 for gem5 by doxygen 1.8.17