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