Go to the documentation of this file.
   44 #include "debug/IPR.hh" 
   45 #include "debug/TLB.hh" 
   61     : 
BaseTLB(
p), size(
p.size), usedEntries(0), lastReplaced(0),
 
   62       cacheState(0), cacheValid(false)
 
   66         fatal(
"SPARC T1 TLB registers don't support more than 64 TLB entries");
 
   93         if (!
t->pte.locked()) {
 
  110     va &= ~(
PTE.size()-1);
 
  113         "TLB: Inserting Entry; va=%#x pa=%#x pid=%d cid=%d r=%d entryid=%d\n",
 
  114         va, 
PTE.paddr(), partition_id, context_id, (
int)real, entry);
 
  118         if (
tlb[
x].range.real == real &&
 
  126                 DPRINTF(
TLB, 
"TLB: Conflicting entry %#X , deleting it\n", 
x);
 
  139         assert(entry < size && entry >= 0);
 
  140         new_entry = &
tlb[entry];
 
  151                     goto insertAllLocked;
 
  152             } 
while (
tlb[
x].pte.locked());
 
  167     if (new_entry->
valid)
 
  178     new_entry->
used = 
true;;
 
  179     new_entry->
valid = 
true;
 
  189         new_entry->
used = 
true;
 
  203     DPRINTF(
TLB, 
"TLB: Looking up entry va=%#x pid=%d cid=%d r=%d\n",
 
  204             va, partition_id, context_id, real);
 
  221     DPRINTF(
TLB, 
"TLB: Valid entry found pa: %#x size: %#x\n", 
t->pte.paddr(),
 
  226     if (!
t->used && update_used) {
 
  243     for (
int x = 0; 
x < 
size; 
x++) {
 
  245            DPRINTFN(
"%4d:  %#2x:%#2x %c %#4x %#8x %#8x %#16x\n",
 
  259     DPRINTF(IPR, 
"TLB: Demapping Page va=%#x pid=%#d cid=%d r=%d\n",
 
  260             va, partition_id, context_id, real);
 
  274         DPRINTF(IPR, 
"TLB: Demapped page\n");
 
  275         i->second->valid = 
false;
 
  276         if (
i->second->used) {
 
  277             i->second->used = 
false;
 
  288     DPRINTF(IPR, 
"TLB: Demapping Context pid=%#d cid=%d\n",
 
  289             partition_id, context_id);
 
  291     for (
int x = 0; 
x < 
size; 
x++) {
 
  292         if (
tlb[
x].range.contextId == context_id &&
 
  310     DPRINTF(
TLB, 
"TLB: Demapping All pid=%#d\n", partition_id);
 
  312     for (
int x = 0; 
x < 
size; 
x++) {
 
  313         if (
tlb[
x].valid && !
tlb[
x].pte.locked() &&
 
  332     for (
int x = 0; 
x < 
size; 
x++) {
 
  345         panic(
"entry: %d\n", entry);
 
  347     assert(entry < 
size);
 
  348     if (
tlb[entry].valid)
 
  351         return (uint64_t)-1ll;
 
  357     assert(entry < 
size);
 
  359     if (!
tlb[entry].valid)
 
  360         return (uint64_t)-1ll;
 
  364     tag |= (uint64_t)
tlb[entry].range.partitionId << 61;
 
  400     DPRINTF(
TLB, 
"TLB: Writing Tag Access: va: %#X ctx: %#X value: %#X\n",
 
  410     DPRINTF(
TLB, 
"TLB: Fault: A=%#x w=%d ct=%d ft=%d asi=%d\n",
 
  411             a, (
int)write, ct, ft, asi);
 
  426     DPRINTF(
TLB, 
"TLB: ITB Request to translate va=%#x size=%d\n",
 
  427             vaddr, req->getSize());
 
  446     bool addr_mask = 
bits(tlbdata,3,3);
 
  447     bool lsu_im = 
bits(tlbdata,4,4);
 
  449     int part_id = 
bits(tlbdata,15,8);
 
  450     int tl = 
bits(tlbdata,18,16);
 
  451     int pri_context = 
bits(tlbdata,47,32);
 
  457     DPRINTF(
TLB, 
"TLB: priv:%d hpriv:%d red:%d lsuim:%d part_id: %#X\n",
 
  467         context = pri_context;
 
  481         return std::make_shared<MemAddressNotAligned>();
 
  489         return std::make_shared<InstructionAccessException>();
 
  500     if (
e == NULL || !
e->valid) {
 
  503             return std::make_shared<InstructionRealTranslationMiss>();
 
  506                 return std::make_shared<FastInstructionAccessMMUMiss>();
 
  508                 return std::make_shared<FastInstructionAccessMMUMiss>(
 
  514     if (!
priv && 
e->pte.priv()) {
 
  517         return std::make_shared<InstructionAccessException>();
 
  525     req->setPaddr(
e->pte.translate(
vaddr));
 
  541     asi = (
ASI)req->getArchFlags();
 
  542     bool implicit = 
false;
 
  546     DPRINTF(
TLB, 
"TLB: DTB Request to translate va=%#x size=%d asi=%#x\n",
 
  558         if (
hpriv && implicit) {
 
  570                 Addr ce_va = 
ce->range.va;
 
  572                     ce_va < vaddr + size && ce_va + ce->range.size > 
vaddr &&
 
  573                     (!write || 
ce->pte.writable())) {
 
  574                     req->setPaddr(
ce->pte.translate(
vaddr));
 
  575                     if (
ce->pte.sideffect() || (
ce->pte.paddr() >> 39) & 1) {
 
  585                 Addr ce_va = 
ce->range.va;
 
  587                     ce_va < vaddr + size && ce_va + ce->range.size > 
vaddr &&
 
  588                     (!write || 
ce->pte.writable())) {
 
  589                     req->setPaddr(
ce->pte.translate(
vaddr));
 
  590                     if (
ce->pte.sideffect() || (
ce->pte.paddr() >> 39) & 1) {
 
  603     bool addr_mask = 
bits(tlbdata,3,3);
 
  604     bool lsu_dm = 
bits(tlbdata,5,5);
 
  606     int part_id = 
bits(tlbdata,15,8);
 
  607     int tl = 
bits(tlbdata,18,16);
 
  608     int pri_context = 
bits(tlbdata,47,32);
 
  609     int sec_context = 
bits(tlbdata,63,48);
 
  617     DPRINTF(
TLB, 
"TLB: priv:%d hpriv:%d red:%d lsudm:%d part_id: %#X\n",
 
  628             context = pri_context;
 
  635             return std::make_shared<PrivilegedAction>();
 
  640             return std::make_shared<DataAccessException>();
 
  644             context = pri_context;
 
  647             context = sec_context;
 
  654             context = pri_context;
 
  658     if (!implicit && asi != 
ASI_P && asi != 
ASI_S) {
 
  660             panic(
"Little Endian ASIs not supported\n");
 
  663             panic(
"Partial Store ASIs not supported\n");
 
  666             panic(
"Cmt ASI registers not implmented\n");
 
  669             goto handleIntRegAccess;
 
  671             goto handleMmuRegAccess;
 
  673             goto handleScratchRegAccess;
 
  675             goto handleQueueRegAccess;
 
  677             goto handleSparcErrorRegAccess;
 
  681             panic(
"Accessing ASI %#X. Should we?\n", asi);
 
  687         return std::make_shared<MemAddressNotAligned>();
 
  695         return std::make_shared<DataAccessException>();
 
  710     if (
e == NULL || !
e->valid) {
 
  712         DPRINTF(
TLB, 
"TLB: DTB Failed to find matching TLB entry\n");
 
  714             return std::make_shared<DataRealTranslationMiss>();
 
  717                 return std::make_shared<FastDataAccessMMUMiss>();
 
  719                 return std::make_shared<FastDataAccessMMUMiss>(
 
  725     if (!
priv && 
e->pte.priv()) {
 
  728         return std::make_shared<DataAccessException>();
 
  731     if (write && !
e->pte.writable()) {
 
  734         return std::make_shared<FastDataAccessProtection>();
 
  740         return std::make_shared<DataAccessException>();
 
  746         return std::make_shared<DataAccessException>();
 
  749     if (
e->pte.sideffect() || (
e->pte.paddr() >> 39) & 1)
 
  768     req->setPaddr(
e->pte.translate(
vaddr));
 
  777             return std::make_shared<DataAccessException>();
 
  779              return std::make_shared<PrivilegedAction>();
 
  785         return std::make_shared<DataAccessException>();
 
  791 handleScratchRegAccess:
 
  794         return std::make_shared<DataAccessException>();
 
  798 handleQueueRegAccess:
 
  801         return std::make_shared<PrivilegedAction>();
 
  805         return std::make_shared<DataAccessException>();
 
  809 handleSparcErrorRegAccess:
 
  813             return std::make_shared<DataAccessException>();
 
  815              return std::make_shared<PrivilegedAction>();
 
  822     DPRINTF(
TLB, 
"TLB: DTB Translating local access\n");
 
  823     req->setLocalAccessor(
 
  824         [
this,write](ThreadContext *tc, 
PacketPtr pkt) -> Cycles
 
  829     req->setPaddr(req->getVaddr());
 
  863     bool addr_mask = 
bits(tlbdata,3,3);
 
  864     bool data_real = !
bits(tlbdata,5,5);
 
  865     bool inst_real = !
bits(tlbdata,4,4);
 
  866     bool ctx_zero  = 
bits(tlbdata,18,16) > 0;
 
  867     int part_id = 
bits(tlbdata,15,8);
 
  868     int pri_context = 
bits(tlbdata,47,32);
 
  880         req->setPaddr(
vaddr);
 
  889             return std::make_shared<InstructionAccessException>();
 
  891             return std::make_shared<DataAccessException>();
 
  894     tbe = 
lookup(
vaddr, part_id, real, ctx_zero ? 0 : pri_context, 
false);
 
  910     for (
int x = 0; 
x < 4; 
x++) {
 
  911         ttetag = 
betoh(
mem.read<uint64_t>(tsbs[
x]));
 
  912         if (ttetag.
valid() && ttetag.
va() == va_tag) {
 
  913             uint64_t entry = 
mem.read<uint64_t>(tsbs[
x]) + 
sizeof(uint64_t);
 
  916             DPRINTF(
TLB, 
"Virtual(%#x)->Physical(%#x) found in TTE\n",
 
  925             return std::make_shared<InstructionRealTranslationMiss>();
 
  927             return std::make_shared<FastInstructionAccessMMUMiss>();
 
  929             return std::make_shared<FastInstructionAccessMMUMiss>(
vaddr);
 
  932             return std::make_shared<DataRealTranslationMiss>();
 
  934             return std::make_shared<FastDataAccessMMUMiss>();
 
  936             return std::make_shared<FastDataAccessMMUMiss>(
vaddr);
 
  962     DPRINTF(IPR, 
"Memory Mapped IPR Read: asi=%#X a=%#x\n",
 
  963          (uint32_t)pkt->
req->getArchFlags(), pkt->
getAddr());
 
 1037         pkt->
setBE((uint64_t)0);
 
 1056             goto doMmuReadError;
 
 1078                 goto doMmuReadError;
 
 1117                         tc->
getCpuPtr()->getInterruptController(0));
 
 1125                         tc->
getCpuPtr()->getInterruptController(0));
 
 1133         panic(
"need to impl DTB::doMmuRegRead() got asi=%#x, va=%#x\n",
 
 1145     ASI asi = (
ASI)pkt->
req->getArchFlags();
 
 1151     int entry_insert = -1;
 
 1158     DPRINTF(IPR, 
"Memory Mapped IPR Write: asi=%#X a=%#x d=%#X\n",
 
 1159          (uint32_t)asi, 
va, 
data);
 
 1177             goto doMmuWriteError;
 
 1235         inform(
"Ignoring write to SPARC ERROR regsiter\n");
 
 1250             goto doMmuWriteError;
 
 1254         entry_insert = 
bits(
va, 8,3);
 
 1257         assert(entry_insert != -1 || 
mbits(
va,10,9) == 
va);
 
 1259         va_insert = 
mbits(ta_insert, 63,13);
 
 1260         ct_insert = 
mbits(ta_insert, 12,0);
 
 1262         real_insert = 
bits(
va, 9,9);
 
 1265         itb->
insert(va_insert, part_insert, ct_insert, real_insert,
 
 1269         entry_insert = 
bits(
va, 8,3);
 
 1272         assert(entry_insert != -1 || 
mbits(
va,10,9) == 
va);
 
 1274         va_insert = 
mbits(ta_insert, 63,13);
 
 1275         ct_insert = 
mbits(ta_insert, 12,0);
 
 1277         real_insert = 
bits(
va, 9,9);
 
 1280         insert(va_insert, part_insert, ct_insert, real_insert, pte,
 
 1314             panic(
"Invalid type for IMMU demap\n");
 
 1329             goto doMmuWriteError;
 
 1363             panic(
"Invalid type for IMMU demap\n");
 
 1372                         tc->
getCpuPtr()->getInterruptController(0));
 
 1381                 getCpuPtr()->postInterrupt(0, 
bits(
data, 5, 0), 0);
 
 1385         panic(
"need to impl DTB::doMmuRegWrite() got asi=%#x, va=%#x d=%#x\n",
 
 1421         uint64_t c0_config, uint64_t cX_tsb, uint64_t cX_config)
 
 1434     uint64_t ptr = 
mbits(tsb,63,13);
 
 1435     bool split = 
bits(tsb,12,12);
 
 1436     int tsb_size = 
bits(tsb,3,0);
 
 1437     int page_size = (
ps == 
Ps0) ? 
bits(config, 2,0) : 
bits(config,10,8);
 
 1439     if (
ps == 
Ps1  && split)
 
 1440         ptr |= 1ULL << (13 + tsb_size);
 
 1441     ptr |= (
tag_access >> (9 + page_size * 3)) & 
mask(12+tsb_size, 4);
 
 1456         free_list.push_back(entry - 
tlb);
 
 1470     for (
int x = 0; 
x < 
size; 
x++) {
 
 1482     if (oldSize != 
size)
 
 1483         panic(
"Don't support unserializing different sized TLBs\n");
 
 1490     for (
int idx : free_list)
 
 1503     for (
int x = 0; 
x < 
size; 
x++) {
 
  
void GetTsbPtr(ThreadContext *tc, Addr addr, int ctx, Addr *ptrs)
@ ASI_DTLB_DATA_ACCESS_REG
@ ASI_DMMU_TSB_PS0_PTR_REG
#define fatal(...)
This implements a cprintf based fatal() function.
static void ignore(const char *expr)
virtual RegVal readMiscReg(RegIndex misc_reg)=0
constexpr decltype(nullptr) NoFault
virtual System * getSystemPtr()=0
@ ASI_DMMU_TSB_PS1_PTR_REG
Fault translateAtomic(const RequestPtr &req, ThreadContext *tc, BaseMMU::Mode mode) override
void serialize(CheckpointOut &cp) const
void flushAll() override
Remove all entries from the TLB.
#define UNSERIALIZE_SCALAR(scalar)
@ ASI_IMMU_CTXT_NONZERO_TSB_BASE_PS1
#define UNSERIALIZE_CONTAINER(member)
Cycles doMmuRegWrite(ThreadContext *tc, Packet *pkt)
void insert(Addr vpn, int partition_id, int context_id, bool real, const PageTableEntry &PTE, int entry=-1)
Insert a PTE into the TLB.
const Addr StartVAddrHole
RequestPtr req
A pointer to the original request.
bool asiIsSecondary(ASI asi)
@ ASI_IMMU_TSB_PS0_PTR_REG
virtual BaseMMU * getMMUPtr()=0
void writeTagAccess(Addr va, int context)
Cycles doMmuRegRead(ThreadContext *tc, Packet *pkt)
bool asiIsPartialStore(ASI asi)
bool translate(Addr vaddr, Addr &paddr)
Translate function.
bool asiIsLittle(ASI asi)
uint64_t TteRead(int entry)
Give an entry id, read that tlb entries' tte.
bool asiIsNoFault(ASI asi)
uint64_t TagRead(int entry)
Given an entry id, read that tlb entries' tag.
std::string csprintf(const char *format, const Args &...args)
void makeAtomicResponse()
constexpr T mbits(T val, unsigned first, unsigned last)
Mask off the given bits in place like bits() but without shifting.
iterator insert(TlbRange &r, TlbEntry *d)
@ ASI_DMMU_CTXT_ZERO_TSB_BASE_PS0
iterator find(const TlbRange &r)
constexpr uint64_t mask(unsigned nbits)
Generate a 64-bit mask of 'nbits' 1s, right justified.
unsigned int cacheLineSize() const
Get the cache line size of the system.
@ ASI_IMMU_CTXT_NONZERO_TSB_BASE_PS0
Cycles is a wrapper class for representing cycle counts, i.e.
void unserialize(CheckpointIn &cp)
Fault translateFunctional(const RequestPtr &req, ThreadContext *tc, BaseMMU::Mode mode) override
@ ASI_IMMU_CTXT_ZERO_CONFIG
bool asiIsInterrupt(ASI asi)
@ UNCACHEABLE
The request is to an uncacheable address.
@ ASI_IMMU_CTXT_ZERO_TSB_BASE_PS0
ThreadContext is the external interface to all thread state for anything outside of the CPU.
std::shared_ptr< FaultBase > Fault
T getBE() const
Get the data in the packet byte swapped from big endian to host endian.
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
EmulationPageTable * pTable
bool validVirtualAddress(Addr va, bool am)
Checks if the virtual address provided is a valid one.
@ ASI_DMMU_CTXT_ZERO_CONFIG
std::shared_ptr< Request > RequestPtr
void unserialize(CheckpointIn &cp) override
Unserialize an object.
void translateTiming(const RequestPtr &req, ThreadContext *tc, BaseMMU::Translation *translation, BaseMMU::Mode mode) override
void setBE(T v)
Set the value in the data pointer to v as big endian.
This object is a proxy for a port or other object which implements the functional response protocol,...
void populate(uint64_t e, EntryType t=sun4u)
void serialize(CheckpointOut &cp) const override
Serialize an object.
bool asiIsScratchPad(ASI asi)
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
@ ASI_DMMU_CTXT_NONZERO_TSB_BASE_PS0
uint64_t get_vec(int int_num)
virtual RegVal readMiscRegNoEffect(RegIndex misc_reg) const =0
bool asiIsPrimary(ASI asi)
void demapContext(int partition_id, int context_id)
Remove all entries that match a given context/partition id.
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
void demapAll(int partition_id)
Remove all non-locked entries from the tlb that match partition id.
TlbEntry * lookup(Addr va, int partition_id, bool real, int context_id=0, bool update_used=true)
lookup an entry in the TLB based on the partition id, and real bit if real is true or the partition i...
@ ASI_IMMU_CTXT_ZERO_TSB_BASE_PS1
bool asiIsAsIfUser(ASI asi)
#define SERIALIZE_SCALAR(scalar)
@ ASI_ITLB_DATA_ACCESS_REG
@ MISCREG_QUEUE_CPU_MONDO_HEAD
@ STRICT_ORDER
The request is required to be strictly ordered by CPU models and is non-speculative.
virtual Process * getProcessPtr()=0
bool FullSystem
The FullSystem variable can be used to determine the current mode of simulation.
Addr translate(Addr vaddr) const
Fault translateData(const RequestPtr &req, ThreadContext *tc, bool write)
@ ASI_IMMU_CTXT_NONZERO_CONFIG
#define SERIALIZE_CONTAINER(member)
virtual void setMiscReg(RegIndex misc_reg, RegVal val)=0
void paramIn(CheckpointIn &cp, const std::string &name, ExtMachInst &machInst)
@ ASI_DMMU_CTXT_NONZERO_CONFIG
void writeSfsr(bool write, ContextType ct, bool se, FaultTypes ft, int asi)
@ ASI_SPARC_ERROR_STATUS_REG
constexpr int findMsbSet(uint64_t val)
Returns the bit position of the MSB that is set in the input.
std::ostream CheckpointOut
bool asiIsUnPriv(ASI asi)
virtual BaseCPU * getCpuPtr()=0
std::list< TlbEntry * > freeList
@ ASI_IMMU_TSB_PS1_PTR_REG
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
Fault finalizePhysical(const RequestPtr &req, ThreadContext *tc, BaseMMU::Mode mode) const override
Do post-translation physical address finalization.
bool asiIsNucleus(ASI asi)
uint64_t MakeTsbPtr(TsbPageSize ps, uint64_t tag_access, uint64_t c0_tsb, uint64_t c0_config, uint64_t cX_tsb, uint64_t cX_config)
virtual void finish(const Fault &fault, const RequestPtr &req, ThreadContext *tc, BaseMMU::Mode mode)=0
@ MISCREG_MMU_P_CONTEXT
MMU Internal Registers.
@ ASI_DMMU_CTXT_ZERO_TSB_BASE_PS1
Fault translateInst(const RequestPtr &req, ThreadContext *tc)
@ MISCREG_SCRATCHPAD_R0
Scratchpad regiscers.
bool asiIsSparcError(ASI asi)
#define panic(...)
This implements a cprintf based panic() function.
@ ASI_DMMU_CTXT_NONZERO_TSB_BASE_PS1
void demapPage(Addr va, int partition_id, bool real, int context_id)
Remve all entries that match a certain partition id, (contextid), and va).
Generated on Thu Jul 28 2022 13:32:08 for gem5 by  doxygen 1.8.17