63 #include "debug/Checkpoint.hh"
64 #include "debug/TLB.hh"
65 #include "debug/TLBVerbose.hh"
69 #include "params/ArmTLB.hh"
77 using namespace ArmISA;
81 isStage2(
p.is_stage2), stage2Req(false), stage2DescReq(false), _attr(0),
82 directToStage2(false), tableWalker(nullptr), stage2Tlb(nullptr),
83 test(nullptr), stats(this), rangeMRU(1),
85 isHyp(false),
asid(0), vmid(0), hcr(0), dacr(0),
86 miscRegValid(false), miscRegContext(0), curTranType(NormalTran)
140 const Addr paddr = req->getPaddr();
145 req->setLocalAccessor(
150 pseudo_inst::pseudoInst<RegABI64>(tc, func, ret);
152 pseudo_inst::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);
232 DPRINTF(
TLB,
"Inserting entry into TLB with pfn:%#x size:%#x vpn: %#x"
233 " asid:%d vmid:%d N:%d global:%d valid:%d nc:%d xn:%d"
234 " ap:%#x domain:%#x ns:%d nstid:%d isHyp:%d\n", entry.
pfn,
241 DPRINTF(
TLB,
" - Replacing Valid entry %#x, asn %d vmn %d ppn %#x "
242 "size: %#x ap:%d ns:%d nstid:%d g:%d isHyp:%d el: %d\n",
251 for (
int i =
size - 1;
i > 0; --
i)
294 DPRINTF(
TLB,
"Flushing all TLB entries (%s lookup)\n",
300 const bool el_match =
te->checkELMatch(
318 DPRINTF(
TLB,
"Flushing all TLB entries (%s lookup)\n",
324 const bool el_match =
te->checkELMatch(
341 DPRINTF(
TLB,
"Flushing all TLB entries (%s lookup)\n",
347 const bool el_match =
te->checkELMatch(
367 DPRINTF(
TLB,
"Flushing all NS TLB entries (%s lookup)\n",
368 (hyp ?
"hyp" :
"non-hyp"));
373 const bool el_match =
te->checkELMatch(tlbi_op.
targetEL,
false);
375 if (
te->valid &&
te->nstid &&
te->isHyp == hyp && el_match) {
390 DPRINTF(
TLB,
"Flushing TLB entries with mva: %#x, asid: %#x "
391 "(%s lookup)\n", tlbi_op.
addr, tlbi_op.
asid,
401 DPRINTF(
TLB,
"Flushing TLB entries with asid: %#x (%s lookup)\n",
409 if (
te->valid &&
te->asid == tlbi_op.
asid &&
426 DPRINTF(
TLB,
"Flushing TLB entries with mva: %#x (%s lookup)\n",
442 bool hyp = target_el ==
EL2;
444 te =
lookup(mva, asn,
vmid, hyp, secure_lookup,
true, ignore_asn,
447 if (secure_lookup == !
te->nstid) {
452 te =
lookup(mva, asn,
vmid, hyp, secure_lookup,
true, ignore_asn,
468 TLB *otlb =
dynamic_cast<TLB*
>(_otlb);
484 panic(
"Incompatible TLB type!");
489 : statistics::
Group(parent),
490 ADD_STAT(instHits, statistics::units::Count::get(),
"ITB inst hits"),
491 ADD_STAT(instMisses, statistics::units::Count::get(),
"ITB inst misses"),
492 ADD_STAT(readHits, statistics::units::Count::get(),
"DTB read hits"),
493 ADD_STAT(readMisses, statistics::units::Count::get(),
"DTB read misses"),
494 ADD_STAT(writeHits, statistics::units::Count::get(),
"DTB write hits"),
495 ADD_STAT(writeMisses, statistics::units::Count::get(),
"DTB write misses"),
496 ADD_STAT(inserts, statistics::units::Count::get(),
497 "Number of times an entry is inserted into the TLB"),
498 ADD_STAT(flushTlb, statistics::units::Count::get(),
499 "Number of times complete TLB was flushed"),
500 ADD_STAT(flushTlbMva, statistics::units::Count::get(),
501 "Number of times TLB was flushed by MVA"),
502 ADD_STAT(flushTlbMvaAsid, statistics::units::Count::get(),
503 "Number of times TLB was flushed by MVA & ASID"),
504 ADD_STAT(flushTlbAsid, statistics::units::Count::get(),
505 "Number of times TLB was flushed by ASID"),
506 ADD_STAT(flushedEntries, statistics::units::Count::get(),
507 "Number of entries that have been flushed from TLB"),
508 ADD_STAT(alignFaults, statistics::units::Count::get(),
509 "Number of TLB faults due to alignment restrictions"),
510 ADD_STAT(prefetchFaults, statistics::units::Count::get(),
511 "Number of TLB faults due to prefetch"),
512 ADD_STAT(domainFaults, statistics::units::Count::get(),
513 "Number of TLB faults due to domain restrictions"),
514 ADD_STAT(permsFaults, statistics::units::Count::get(),
515 "Number of TLB faults due to permissions restrictions"),
516 ADD_STAT(readAccesses, statistics::units::Count::get(),
"DTB read accesses",
517 readHits + readMisses),
518 ADD_STAT(writeAccesses, statistics::units::Count::get(),
"DTB write accesses",
519 writeHits + writeMisses),
520 ADD_STAT(instAccesses, statistics::units::Count::get(),
"ITB inst accesses",
521 instHits + instMisses),
522 ADD_STAT(hits, statistics::units::Count::get(),
523 "Total TLB (inst and data) hits",
524 readHits + writeHits + instHits),
525 ADD_STAT(misses, statistics::units::Count::get(),
526 "Total TLB (inst and data) misses",
527 readMisses + writeMisses + instMisses),
528 ADD_STAT(accesses, statistics::units::Count::get(),
529 "Total TLB (inst and data) accesses",
530 readAccesses + writeAccesses + instAccesses)
545 Addr vaddr_tainted = req->getVaddr();
551 vaddr = vaddr_tainted;
558 if (sctlr.a || !(flags & AllowUnaligned)) {
559 if (
vaddr &
mask(flags & AlignmentMask)) {
561 return std::make_shared<DataAbort>(
563 TlbEntry::DomainType::NoAccess, is_write,
573 if (!
p->pTable->translate(
vaddr, paddr))
574 return std::make_shared<GenericPageTableFault>(vaddr_tainted);
575 req->setPaddr(paddr);
585 if (req->isCacheMaintenance()) {
593 bool is_priv = isPriv && !(flags & UserMode);
603 if (isStage2 && req->isPTWalk() && hcr.ptw &&
604 (
te->mtype != TlbEntry::MemoryType::Normal)) {
605 return std::make_shared<DataAbort>(
608 isStage2, tranMethod);
614 if (
te->mtype != TlbEntry::MemoryType::Normal) {
615 if (
vaddr &
mask(flags & AlignmentMask)) {
617 return std::make_shared<DataAbort>(
618 vaddr, TlbEntry::DomainType::NoAccess, is_write,
625 if (
te->nonCacheable) {
627 if (req->isPrefetch()) {
630 return std::make_shared<PrefetchAbort>(
632 isStage2, tranMethod);
636 if (!
te->longDescFormat) {
637 switch ((dacr >> (
static_cast<uint8_t
>(
te->domain) * 2)) & 0x3) {
639 stats.domainFaults++;
640 DPRINTF(
TLB,
"TLB Fault: Data abort on domain. DACR: %#x"
641 " domain: %#x write:%d\n", dacr,
642 static_cast<uint8_t
>(
te->domain), is_write);
647 return std::make_shared<PrefetchAbort>(
650 isStage2, tranMethod);
652 return std::make_shared<DataAbort>(
655 isStage2, tranMethod);
660 panic(
"UNPRED domain\n");
667 uint8_t ap =
te->longDescFormat ?
te->ap << 1 :
te->ap;
668 uint8_t hap =
te->hap;
670 if (sctlr.afe == 1 ||
te->longDescFormat)
674 bool isWritable =
true;
683 DPRINTF(
TLB,
"Access permissions 0, checking rs:%#x\n",
686 switch ((
int)sctlr.rs) {
691 abt = is_write || !is_priv;
707 abt = !is_priv && is_write;
708 isWritable = is_priv;
714 panic(
"UNPRED premissions\n");
716 abt = !is_priv || is_write;
725 panic(
"Unknown permissions %#x\n", ap);
729 bool hapAbt = is_write ? !(hap & 2) : !(hap & 1);
730 bool xn =
te->xn || (isWritable && sctlr.wxn) ||
731 (ap == 3 && sctlr.uwxn && is_priv);
732 if (is_fetch && (abt || xn ||
733 (
te->longDescFormat &&
te->pxn && is_priv) ||
736 DPRINTF(
TLB,
"TLB Fault: Prefetch abort on permission check. AP:%d "
737 "priv:%d write:%d ns:%d sif:%d sctlr.afe: %d \n",
738 ap, is_priv, is_write,
te->ns, scr.sif,sctlr.afe);
741 return std::make_shared<PrefetchAbort>(
744 isStage2, tranMethod);
745 }
else if (abt | hapAbt) {
747 DPRINTF(
TLB,
"TLB Fault: Data abort on permission check. AP:%d priv:%d"
748 " write:%d\n", ap, is_priv, is_write);
749 return std::make_shared<DataAbort>(
752 isStage2 | !abt, tranMethod);
769 if (req->isCacheClean() && aarch64EL !=
EL0 && !isStage2) {
773 Addr vaddr_tainted = req->getVaddr();
781 bool is_atomic = req->isAtomic();
782 GEM5_VAR_USED
bool is_priv = isPriv && !(flags & UserMode);
784 updateMiscReg(tc, curTranType);
790 if (isStage2 && req->isPTWalk() && hcr.ptw &&
791 (
te->mtype != TlbEntry::MemoryType::Normal)) {
792 return std::make_shared<DataAbort>(
793 vaddr_tainted,
te->domain, is_write,
801 if (
te->mtype != TlbEntry::MemoryType::Normal) {
802 if (
vaddr &
mask(flags & AlignmentMask)) {
804 return std::make_shared<DataAbort>(
806 TlbEntry::DomainType::NoAccess,
807 is_atomic ?
false : is_write,
814 if (
te->nonCacheable) {
816 if (req->isPrefetch()) {
819 return std::make_shared<PrefetchAbort>(
826 uint8_t ap = 0x3 & (
te->ap);
829 bool wxn = sctlr.wxn;
831 uint8_t pxn =
te->pxn;
832 bool r = (!is_write && !is_fetch);
843 bool grant_read =
true;
844 DPRINTF(TLBVerbose,
"Checking permissions: ap:%d, xn:%d, pxn:%d, r:%d, "
845 "w:%d, x:%d, is_priv: %d, wxn: %d\n", ap, xn,
846 pxn,
r,
w,
x, is_priv,
wxn);
853 uint8_t hap = 0x3 &
te->hap;
854 grant_read = hap & 0x1;
858 }
else if (is_atomic) {
860 }
else if (is_write) {
869 grant_read = ap & 0x1;
870 uint8_t
perm = (ap << 2) | (xn << 1) | pxn;
880 grant =
r ||
w || (
x && !
wxn);
901 if (checkPAN(tc, ap, req,
mode, is_priv)) {
907 uint8_t
perm = (ap << 2) | (xn << 1) | pxn;
911 grant =
r ||
w || (
x && !
wxn);
941 if (hcr.e2h && checkPAN(tc, ap, req,
mode, is_priv)) {
949 uint8_t
perm = (ap & 0x2) | xn;
952 grant =
r ||
w || (
x && !
wxn);
974 DPRINTF(
TLB,
"TLB Fault: Prefetch abort on permission check. "
975 "AP:%d priv:%d write:%d ns:%d sif:%d "
977 ap, is_priv, is_write,
te->ns, scr.sif, sctlr.afe);
980 return std::make_shared<PrefetchAbort>(
986 DPRINTF(
TLB,
"TLB Fault: Data abort on permission check. AP:%d "
987 "priv:%d write:%d\n", ap, is_priv, is_write);
988 return std::make_shared<DataAbort>(
989 vaddr_tainted,
te->domain,
990 (is_atomic && !grant_read) ?
false : is_write,
1013 if (req->isCacheMaintenance() &&
1017 }
else if (!is_priv && !(hcr.e2h && !hcr.tge)) {
1033 bool is_atomic = req->isAtomic();
1034 req->setPaddr(
vaddr);
1044 int addr_sz =
bits(
vaddr, topbit, physAddrRange);
1048 f = std::make_shared<PrefetchAbort>(
vaddr,
1051 f = std::make_shared<DataAbort>(
vaddr,
1052 TlbEntry::DomainType::NoAccess,
1060 if (long_desc_format || sctlr.tre == 0 || nmrr.ir0 == 0 ||
1061 nmrr.or0 == 0 || prrr.tr0 != 0x2) {
1062 if (!req->isCacheMaintenance()) {
1072 && hcr.e2h == 1 && hcr.tge == 1) ? 0: hcr.dc;
1073 bool i_cacheability = sctlr.i && !sctlr.m;
1075 (isHyp && !(tranType & S1CTran))) {
1077 temp_te.
mtype = is_fetch ? TlbEntry::MemoryType::Normal
1078 : TlbEntry::MemoryType::StronglyOrdered;
1079 temp_te.
innerAttrs = i_cacheability? 0x2: 0x0;
1080 temp_te.
outerAttrs = i_cacheability? 0x2: 0x0;
1084 temp_te.
mtype = TlbEntry::MemoryType::Normal;
1091 DPRINTF(TLBVerbose,
"(No MMU) setting memory attributes: shareable: "
1092 "%d, innerAttrs: %d, outerAttrs: %d, isStage2: %d\n",
1097 return testTranslation(req,
mode, TlbEntry::DomainType::NoAccess);
1103 bool &delay,
bool timing,
1112 Addr vaddr_tainted = req->getVaddr();
1114 Fault fault = getResultTe(&
te, req, tc,
mode, translation, timing,
1115 functional, &mergeTe);
1117 if ((
te == NULL) && (fault ==
NoFault)) delay =
true;
1124 "Setting memory attributes: shareable: %d, innerAttrs: %d, "
1125 "outerAttrs: %d, mtype: %d, isStage2: %d\n",
1126 te->shareable,
te->innerAttrs,
te->outerAttrs,
1127 static_cast<uint8_t
>(
te->mtype), isStage2);
1128 setAttr(
te->attributes);
1130 if (
te->nonCacheable && !req->isCacheMaintenance())
1136 if (
te->mtype != TlbEntry::MemoryType::Normal)
1145 if (!is_fetch && fault ==
NoFault &&
1146 (
vaddr &
mask(flags & AlignmentMask)) &&
1147 (
te->mtype != TlbEntry::MemoryType::Normal)) {
1150 stats.alignFaults++;
1152 return std::make_shared<DataAbort>(
1154 TlbEntry::DomainType::NoAccess, is_write,
1161 fault = testTranslation(req,
mode,
te->domain);
1179 assert(!(timing && functional));
1181 updateMiscReg(tc, tranType);
1183 Addr vaddr_tainted = req->getVaddr();
1189 vaddr = vaddr_tainted;
1199 "CPSR is priv:%d UserMode:%d secure:%d S1S2NsTran:%d\n",
1200 isPriv, flags & UserMode,
isSecure, tranType & S1S2NsTran);
1202 DPRINTF(
TLB,
"translateFs addr %#x, mode %d, st2 %d, scr %#x sctlr %#x "
1203 "flags %#lx tranType 0x%x\n", vaddr_tainted,
mode, isStage2,
1204 scr, sctlr, flags, tranType);
1206 if ((req->isInstFetch() && (!sctlr.i)) ||
1207 ((!req->isInstFetch()) && (!sctlr.c))){
1208 if (!req->isCacheMaintenance()) {
1214 if (sctlr.a || !(flags & AllowUnaligned)) {
1215 if (
vaddr &
mask(flags & AlignmentMask)) {
1216 stats.alignFaults++;
1217 return std::make_shared<DataAbort>(
1219 TlbEntry::DomainType::NoAccess, is_write,
1229 else if (hcr.dc == 1)
1234 if ((isStage2 && !
vm) || (!isStage2 && !sctlr.m)) {
1235 fault = translateMmuOff(tc, req,
mode, tranType,
vaddr,
1238 DPRINTF(TLBVerbose,
"Translating %s=%#x context=%d\n",
1239 isStage2 ?
"IPA" :
"VA", vaddr_tainted,
asid);
1241 fault = translateMmuOn(tc, req,
mode, translation, delay, timing,
1242 functional,
vaddr, tranMethod);
1248 if (
sd->enabled() && fault ==
NoFault) {
1249 fault =
sd->testDebug(tc, req,
mode);
1259 updateMiscReg(tc, tranType);
1261 if (directToStage2) {
1263 return stage2Tlb->translateAtomic(req, tc,
mode, tranType);
1269 fault = translateFs(req, tc,
mode, NULL, delay,
false, tranType);
1271 fault = translateSe(req, tc,
mode, NULL, delay,
false);
1280 updateMiscReg(tc, tranType);
1282 if (directToStage2) {
1284 return stage2Tlb->translateFunctional(req, tc,
mode, tranType);
1290 fault = translateFs(req, tc,
mode, NULL, delay,
false, tranType,
true);
1292 fault = translateSe(req, tc,
mode, NULL, delay,
false);
1302 updateMiscReg(tc, tranType);
1304 if (directToStage2) {
1306 stage2Tlb->translateTiming(req, tc, translation,
mode, tranType);
1310 assert(translation);
1312 translateComplete(req, tc, translation,
mode, tranType, isStage2);
1323 fault = translateFs(req, tc,
mode, translation, delay,
true, tranType);
1325 fault = translateSe(req, tc,
mode, translation, delay,
true);
1326 DPRINTF(TLBVerbose,
"Translation returning delay=%d fault=%d\n", delay, fault !=
1335 if (translation && (callFromS2 || !stage2Req || req->hasPaddr() ||
1348 return &tableWalker->getTableWalkerPort();
1358 switch (mmfr1.vmidbits) {
1373 panic(
"Reserved ID_AA64MMFR1_EL1.VMIDBits value: %#x",
1386 if (miscRegValid && miscRegContext == tc->
contextId() &&
1387 ((tranType == curTranType) || isStage2)) {
1391 DPRINTF(TLBVerbose,
"TLB variables changed!\n");
1396 !(tranType & HypMode) && !(tranType & S1S2NsTran);
1398 aarch64EL = tranTypeEL(cpsr, tranType);
1406 switch (aarch64EL) {
1412 uint64_t ttbr_asid = ttbcr.a1 ?
1416 (haveLargeAsid64 && ttbcr.as) ? 63 : 55, 48);
1421 uint64_t ttbr_asid = ttbcr.a1 ?
1425 (haveLargeAsid64 && ttbcr.as) ? 63 : 55, 48);
1433 uint64_t ttbr_asid = ttbcr.a1 ?
1437 (haveLargeAsid64 && ttbcr.as) ? 63 : 55, 48);
1445 uint64_t ttbr_asid = ttbcr.a1 ?
1449 (haveLargeAsid64 && ttbcr.as) ? 63 : 55, 48);
1462 isPriv = aarch64EL !=
EL0;
1463 if (haveVirtualization) {
1465 isHyp = aarch64EL ==
EL2;
1466 isHyp |= tranType & HypMode;
1467 isHyp &= (tranType & S1S2NsTran) == 0;
1468 isHyp &= (tranType & S1CTran) == 0;
1474 if (hcr.e2h == 1 && (aarch64EL ==
EL2
1475 || (hcr.tge ==1 && aarch64EL ==
EL0))) {
1477 directToStage2 =
false;
1479 stage2DescReq =
false;
1485 stage2Req = isStage2 ||
1486 (
vm && !isHyp && sec &&
1487 !(tranType & S1CTran) && (aarch64EL <
EL2) &&
1488 !(tranType & S1E1Tran));
1489 stage2DescReq = isStage2 || (
vm && !isHyp && sec &&
1491 directToStage2 = !isStage2 && stage2Req && !sctlr.m;
1496 directToStage2 =
false;
1498 stage2DescReq =
false;
1516 asid = context_id.asid;
1526 if (haveVirtualization) {
1529 isHyp |= tranType & HypMode;
1530 isHyp &= (tranType & S1S2NsTran) == 0;
1531 isHyp &= (tranType & S1CTran) == 0;
1539 stage2Req = hcr.vm && !isStage2 && !isHyp && sec &&
1540 !(tranType & S1CTran);
1541 stage2DescReq = hcr.vm && !isStage2 && !isHyp && sec;
1542 directToStage2 = stage2Req && !sctlr.m;
1547 directToStage2 =
false;
1548 stage2DescReq =
false;
1551 miscRegValid =
true;
1553 curTranType = tranType;
1581 panic(
"Unknown translation mode!\n");
1593 updateMiscReg(tc, tranType);
1596 Addr vaddr_tainted = req->getVaddr();
1603 vaddr = vaddr_tainted;
1608 if (req->isPrefetch()) {
1612 stats.prefetchFaults++;
1613 return std::make_shared<PrefetchAbort>(
1619 DPRINTF(
TLB,
"TLB Miss: Starting hardware table walker for %#x(%d:%d)\n",
1620 vaddr_tainted,
asid, vmid);
1622 fault = tableWalker->walk(req, tc,
asid, vmid, isHyp,
mode,
1623 translation, timing, functional, is_secure,
1624 tranType, stage2DescReq);
1626 if (timing || fault !=
NoFault) {
1631 target_el,
false,
mode);
1652 fault = getTE(&s2Te, req, tc,
mode, translation, timing, functional,
1655 if ((s2Te != NULL) && (fault ==
NoFault)) {
1657 fault = checkPermissions64(s2Te, req,
mode, tc);
1659 fault = checkPermissions(s2Te, req,
mode);
1667 Addr vaddr_tainted = req->getVaddr();
1670 fault = getTE(&s1Te, req, tc,
mode, translation, timing, functional,
1673 if ((s1Te != NULL) && (fault ==
NoFault)) {
1676 fault = checkPermissions64(s1Te, req,
mode, tc);
1678 fault = checkPermissions(s1Te, req,
mode);
1679 if (stage2Req & (fault ==
NoFault)) {
1683 fault = s2Lookup->
getTe(tc, mergeTe);
1698 DPRINTF(TLBVerbose,
"s2TLB: reqVa %#x, reqPa %#x, fault %p\n",
1699 vaddr_tainted, req->hasPaddr() ? req->getPaddr() : ~0, fault);
1719 fatal_if(!
ti,
"%s is not a valid ARM TLB tester\n", _ti->
name());
1728 if (!
test || !req->hasSize() || req->getSize() == 0 ||
1729 req->isCacheMaintenance()) {