49#include "debug/TLB.hh"
50#include "debug/TLBVerbose.hh"
52#include "params/RiscvTLB.hh"
60using namespace RiscvISA;
70 return (
static_cast<Addr>(
asid) << 48) | vpn;
74 BaseTLB(
p), size(
p.size),
tlb(size),
75 lruSeq(0), stats(this), pma(
p.pma_checker),
78 for (
size_t x = 0;
x < size;
x++) {
79 tlb[
x].trieHandle = NULL;
80 freeList.push_back(&
tlb[
x]);
100 for (
size_t i = 1;
i < size;
i++) {
101 if (
tlb[
i].lruSeq <
tlb[lru].lruSeq)
115 entry->
lruSeq = nextSeq();
117 if (
mode == BaseMMU::Write)
118 stats.writeAccesses++;
120 stats.readAccesses++;
123 if (
mode == BaseMMU::Write)
129 if (
mode == BaseMMU::Write)
135 DPRINTF(TLBVerbose,
"lookup(vpn=%#x, asid=%#x): %s ppn %#x\n",
136 vpn,
asid, entry ?
"hit" :
"miss", entry ? entry->
paddr : 0);
145 DPRINTF(
TLB,
"insert(vpn=%#x, asid=%#x): ppn=%#x pte=%#x size=%#x\n",
149 TlbEntry *newEntry = lookup(vpn, entry.
asid, BaseMMU::Read,
true);
152 newEntry->
pte = entry.
pte;
153 assert(newEntry->
vaddr == vpn);
157 if (freeList.empty())
160 newEntry = freeList.front();
161 freeList.pop_front();
165 newEntry->
lruSeq = nextSeq();
166 newEntry->
vaddr = vpn;
168 trie.insert(key, TlbEntryTrie::MaxBits - entry.
logBytes, newEntry);
173TLB::demapPage(
Addr vpn, uint64_t asid)
177 if (vpn == 0 && asid == 0)
180 DPRINTF(
TLB,
"flush(vpn=%#x, asid=%#x)\n", vpn, asid);
181 if (vpn != 0 && asid != 0) {
182 TlbEntry *newEntry = lookup(vpn, asid, BaseMMU::Read,
true);
184 remove(newEntry -
tlb.data());
187 for (
size_t i = 0;
i < size;
i++) {
188 if (tlb[i].trieHandle) {
190 if ((vpn == 0 || (vpn & mask) == tlb[i].vaddr) &&
203 for (
size_t i = 0;
i < size;
i++) {
204 if (tlb[i].trieHandle)
210TLB::remove(
size_t idx)
212 DPRINTF(
TLB,
"remove(vpn=%#x, asid=%#x): ppn=%#x pte=%#x size=%#x\n",
216 assert(
tlb[idx].trieHandle);
217 trie.remove(
tlb[idx].trieHandle);
218 tlb[idx].trieHandle = NULL;
219 freeList.push_back(&
tlb[idx]);
228 if (
mode == BaseMMU::Read && !pte.r) {
229 DPRINTF(
TLB,
"PTE has no read perm, raising PF\n");
232 else if (
mode == BaseMMU::Write && !pte.w) {
233 DPRINTF(
TLB,
"PTE has no write perm, raising PF\n");
236 else if (
mode == BaseMMU::Execute && !pte.x) {
237 DPRINTF(
TLB,
"PTE has no exec perm, raising PF\n");
243 if (pmode == PrivilegeMode::PRV_U && !pte.u) {
244 DPRINTF(
TLB,
"PTE is not user accessible, raising PF\n");
247 else if (pmode == PrivilegeMode::PRV_S && pte.u &&
status.sum == 0) {
248 DPRINTF(
TLB,
"PTE is only user accessible, raising PF\n");
260 if (
mode == BaseMMU::Read)
261 code = ExceptionCode::LOAD_PAGE;
262 else if (
mode == BaseMMU::Write)
263 code = ExceptionCode::STORE_PAGE;
265 code = ExceptionCode::INST_PAGE;
266 return std::make_shared<AddressFault>(
vaddr, code);
273 assert(
e !=
nullptr);
274 return e->paddr << PageShift | (
vaddr &
mask(
e->logBytes));
289 Fault fault = walker->start(tc, translation, req,
mode);
290 if (translation !=
nullptr || fault !=
NoFault) {
296 assert(
e !=
nullptr);
305 if (
mode == BaseMMU::Write && !
e->pte.w) {
306 DPRINTF(
TLB,
"Dirty bit not set, repeating PT walk\n");
307 fault = walker->start(tc, translation, req,
mode);
308 if (translation !=
nullptr || fault !=
NoFault) {
318 DPRINTF(TLBVerbose,
"translate(vpn=%#x, asid=%#x): %#x\n",
319 vaddr, satp.asid, paddr);
320 req->setPaddr(paddr);
330 if (
mode != BaseMMU::Execute &&
status.mprv == 1)
345 if (pmode == PrivilegeMode::PRV_M || satp.mode == AddrXlateMode::BARE)
346 req->setFlags(Request::PHYSICAL);
349 if (req->getFlags() & Request::PHYSICAL) {
353 req->setPaddr(req->getVaddr());
356 fault = doTranslate(req, tc, translation,
mode, delayed);
362 if (!delayed && fault ==
NoFault &&
bits(req->getPaddr(), 63)) {
364 if (
mode == BaseMMU::Read)
365 code = ExceptionCode::LOAD_ACCESS;
366 else if (
mode == BaseMMU::Write)
367 code = ExceptionCode::STORE_ACCESS;
369 code = ExceptionCode::INST_ACCESS;
370 fault = std::make_shared<AddressFault>(req->getVaddr(), code);
373 if (!delayed && fault ==
NoFault) {
379 fault = pmp->pmpCheck(req,
mode, pmode, tc);
391 assert(req->getSize() > 0);
392 if (req->getVaddr() + req->getSize() - 1 < req->getVaddr())
393 return std::make_shared<GenericPageTableFault>(req->getVaddr());
397 Fault fault = p->pTable->translate(req);
410 return translate(req, tc,
nullptr, mode, delayed);
419 Fault fault = translate(req, tc, translation, mode, delayed);
421 translation->
finish(fault, req, tc, mode);
427TLB::translateFunctional(
const RequestPtr &req, ThreadContext *tc,
430 const Addr vaddr = req->getVaddr();
434 MMU *mmu =
static_cast<MMU *
>(tc->getMMUPtr());
436 PrivilegeMode pmode = mmu->getMemPriv(tc, mode);
437 SATP satp = tc->readMiscReg(MISCREG_SATP);
438 if (pmode != PrivilegeMode::PRV_M &&
439 satp.mode != AddrXlateMode::BARE) {
440 Walker *walker = mmu->getDataWalker();
442 Fault fault = walker->startFunctional(
443 tc, paddr, logBytes, mode);
444 if (fault != NoFault)
447 Addr masked_addr = vaddr & mask(logBytes);
448 paddr |= masked_addr;
452 Process *process = tc->getProcessPtr();
453 const auto *pte = process->pTable->lookup(vaddr);
455 if (!pte && mode != BaseMMU::Execute) {
457 if (process->fixupFault(vaddr)) {
459 pte = process->pTable->lookup(vaddr);
464 return std::make_shared<GenericPageTableFault>(req->getVaddr());
466 paddr = pte->paddr | process->pTable->pageOffset(vaddr);
469 DPRINTF(TLB,
"Translated (functional) %#x -> %#x.\n", vaddr, paddr);
470 req->setPaddr(paddr);
475TLB::finalizePhysical(
const RequestPtr &req,
476 ThreadContext *tc, BaseMMU::Mode mode)
const
482TLB::serialize(CheckpointOut &cp)
const
485 uint32_t _size = size - freeList.size();
490 for (uint32_t x = 0;
x < size;
x++) {
491 if (tlb[x].trieHandle != NULL)
492 tlb[
x].serializeSection(cp,
csprintf(
"Entry%d", _count++));
497TLB::unserialize(CheckpointIn &cp)
503 fatal(
"TLB size less than the one in checkpoint!");
508 for (uint32_t x = 0;
x < _size;
x++) {
509 TlbEntry *newEntry = freeList.front();
510 freeList.pop_front();
514 newEntry->trieHandle = trie.insert(key,
515 TlbEntryTrie::MaxBits - newEntry->logBytes, newEntry);
520 : statistics::
Group(parent),
521 ADD_STAT(readHits, statistics::units::Count::get(),
"read hits"),
522 ADD_STAT(readMisses, statistics::units::Count::get(),
"read misses"),
523 ADD_STAT(readAccesses, statistics::units::Count::get(),
"read accesses"),
524 ADD_STAT(writeHits, statistics::units::Count::get(),
"write hits"),
525 ADD_STAT(writeMisses, statistics::units::Count::get(),
"write misses"),
526 ADD_STAT(writeAccesses, statistics::units::Count::get(),
"write accesses"),
527 ADD_STAT(hits, statistics::units::Count::get(),
528 "Total TLB (read and write) hits", readHits + writeHits),
529 ADD_STAT(misses, statistics::units::Count::get(),
530 "Total TLB (read and write) misses", readMisses + writeMisses),
531 ADD_STAT(accesses, statistics::units::Count::get(),
532 "Total TLB (read and write) accesses",
533 readAccesses + writeAccesses)
virtual void finish(const Fault &fault, const RequestPtr &req, ThreadContext *tc, BaseMMU::Mode mode)=0
virtual void markDelayed()=0
Signal that the translation has been delayed due to a hw page table walk.
Ports are used to interface objects to each other.
ThreadContext is the external interface to all thread state for anything outside of the CPU.
virtual RegVal readMiscReg(RegIndex misc_reg)=0
virtual Process * getProcessPtr()=0
Port * getTableWalkerPort() override
Get the table walker port.
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
#define fatal(...)
This implements a cprintf based fatal() function.
void unserializeSection(CheckpointIn &cp, const char *name)
Unserialize an a child object.
Declaration of IniFile object.
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
std::shared_ptr< FaultBase > Fault
static Addr buildKey(Addr vpn, uint16_t asid)
std::shared_ptr< Request > RequestPtr
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
bool FullSystem
The FullSystem variable can be used to determine the current mode of simulation.
std::string csprintf(const char *format, const Args &...args)
constexpr decltype(nullptr) NoFault
Declarations of a non-full system Page Table.
#define UNSERIALIZE_SCALAR(scalar)
#define SERIALIZE_SCALAR(scalar)
TlbEntryTrie::Handle trieHandle