Go to the documentation of this file.
43 #include "debug/IPR.hh"
44 #include "debug/TLB.hh"
57 :
BaseTLB(
p), size(
p.size), usedEntries(0), lastReplaced(0),
58 cacheState(0), cacheValid(false)
62 fatal(
"SPARC T1 TLB registers don't support more than 64 TLB entries");
89 if (!
t->pte.locked()) {
109 "TLB: Inserting Entry; va=%#x pa=%#x pid=%d cid=%d r=%d entryid=%d\n",
110 va, PTE.
paddr(), partition_id, context_id, (
int)real, entry);
114 if (
tlb[
x].range.real == real &&
122 DPRINTF(
TLB,
"TLB: Conflicting entry %#X , deleting it\n",
x);
135 assert(entry < size && entry >= 0);
136 new_entry = &
tlb[entry];
147 goto insertAllLocked;
148 }
while (
tlb[
x].pte.locked());
163 if (new_entry->
valid)
173 new_entry->
pte = PTE;
174 new_entry->
used =
true;;
175 new_entry->
valid =
true;
185 new_entry->
used =
true;
199 DPRINTF(
TLB,
"TLB: Looking up entry va=%#x pid=%d cid=%d r=%d\n",
200 va, partition_id, context_id, real);
217 DPRINTF(
TLB,
"TLB: Valid entry found pa: %#x size: %#x\n",
t->pte.paddr(),
222 if (!
t->used && update_used) {
239 for (
int x = 0;
x <
size;
x++) {
241 DPRINTFN(
"%4d: %#2x:%#2x %c %#4x %#8x %#8x %#16x\n",
255 DPRINTF(IPR,
"TLB: Demapping Page va=%#x pid=%#d cid=%d r=%d\n",
256 va, partition_id, context_id, real);
270 DPRINTF(IPR,
"TLB: Demapped page\n");
271 i->second->valid =
false;
272 if (
i->second->used) {
273 i->second->used =
false;
284 DPRINTF(IPR,
"TLB: Demapping Context pid=%#d cid=%d\n",
285 partition_id, context_id);
287 for (
int x = 0;
x <
size;
x++) {
288 if (
tlb[
x].range.contextId == context_id &&
306 DPRINTF(
TLB,
"TLB: Demapping All pid=%#d\n", partition_id);
308 for (
int x = 0;
x <
size;
x++) {
309 if (
tlb[
x].valid && !
tlb[
x].pte.locked() &&
328 for (
int x = 0;
x <
size;
x++) {
341 panic(
"entry: %d\n", entry);
343 assert(entry <
size);
344 if (
tlb[entry].valid)
347 return (uint64_t)-1ll;
353 assert(entry <
size);
355 if (!
tlb[entry].valid)
356 return (uint64_t)-1ll;
360 tag |= (uint64_t)
tlb[entry].range.partitionId << 61;
362 tag |= (uint64_t)~
tlb[entry].pte._size() << 56;
396 DPRINTF(
TLB,
"TLB: Writing Tag Access: va: %#X ctx: %#X value: %#X\n",
406 DPRINTF(
TLB,
"TLB: Fault: A=%#x w=%d ct=%d ft=%d asi=%d\n",
407 a, (
int)write, ct, ft, asi);
422 DPRINTF(
TLB,
"TLB: ITB Request to translate va=%#x size=%d\n",
423 vaddr, req->getSize());
442 bool addr_mask =
bits(tlbdata,3,3);
443 bool lsu_im =
bits(tlbdata,4,4);
445 int part_id =
bits(tlbdata,15,8);
446 int tl =
bits(tlbdata,18,16);
447 int pri_context =
bits(tlbdata,47,32);
453 DPRINTF(
TLB,
"TLB: priv:%d hpriv:%d red:%d lsuim:%d part_id: %#X\n",
463 context = pri_context;
477 return std::make_shared<MemAddressNotAligned>();
485 return std::make_shared<InstructionAccessException>();
496 if (
e == NULL || !
e->valid) {
499 return std::make_shared<InstructionRealTranslationMiss>();
502 return std::make_shared<FastInstructionAccessMMUMiss>();
504 return std::make_shared<FastInstructionAccessMMUMiss>(
510 if (!
priv &&
e->pte.priv()) {
513 return std::make_shared<InstructionAccessException>();
521 req->setPaddr(
e->pte.translate(
vaddr));
537 asi = (
ASI)req->getArchFlags();
538 bool implicit =
false;
542 DPRINTF(
TLB,
"TLB: DTB Request to translate va=%#x size=%d asi=%#x\n",
554 if (
hpriv && implicit) {
566 Addr ce_va =
ce->range.va;
568 ce_va < vaddr + size && ce_va + ce->range.size >
vaddr &&
569 (!write ||
ce->pte.writable())) {
570 req->setPaddr(
ce->pte.translate(
vaddr));
571 if (
ce->pte.sideffect() || (
ce->pte.paddr() >> 39) & 1) {
581 Addr ce_va =
ce->range.va;
583 ce_va < vaddr + size && ce_va + ce->range.size >
vaddr &&
584 (!write ||
ce->pte.writable())) {
585 req->setPaddr(
ce->pte.translate(
vaddr));
586 if (
ce->pte.sideffect() || (
ce->pte.paddr() >> 39) & 1) {
599 bool addr_mask =
bits(tlbdata,3,3);
600 bool lsu_dm =
bits(tlbdata,5,5);
602 int part_id =
bits(tlbdata,15,8);
603 int tl =
bits(tlbdata,18,16);
604 int pri_context =
bits(tlbdata,47,32);
605 int sec_context =
bits(tlbdata,63,48);
613 DPRINTF(
TLB,
"TLB: priv:%d hpriv:%d red:%d lsudm:%d part_id: %#X\n",
624 context = pri_context;
631 return std::make_shared<PrivilegedAction>();
636 return std::make_shared<DataAccessException>();
640 context = pri_context;
643 context = sec_context;
650 context = pri_context;
654 if (!implicit && asi !=
ASI_P && asi !=
ASI_S) {
656 panic(
"Little Endian ASIs not supported\n");
659 panic(
"Partial Store ASIs not supported\n");
662 panic(
"Cmt ASI registers not implmented\n");
665 goto handleIntRegAccess;
667 goto handleMmuRegAccess;
669 goto handleScratchRegAccess;
671 goto handleQueueRegAccess;
673 goto handleSparcErrorRegAccess;
677 panic(
"Accessing ASI %#X. Should we?\n", asi);
683 return std::make_shared<MemAddressNotAligned>();
691 return std::make_shared<DataAccessException>();
706 if (
e == NULL || !
e->valid) {
708 DPRINTF(
TLB,
"TLB: DTB Failed to find matching TLB entry\n");
710 return std::make_shared<DataRealTranslationMiss>();
713 return std::make_shared<FastDataAccessMMUMiss>();
715 return std::make_shared<FastDataAccessMMUMiss>(
721 if (!
priv &&
e->pte.priv()) {
724 return std::make_shared<DataAccessException>();
727 if (write && !
e->pte.writable()) {
730 return std::make_shared<FastDataAccessProtection>();
736 return std::make_shared<DataAccessException>();
742 return std::make_shared<DataAccessException>();
745 if (
e->pte.sideffect() || (
e->pte.paddr() >> 39) & 1)
764 req->setPaddr(
e->pte.translate(
vaddr));
773 return std::make_shared<DataAccessException>();
775 return std::make_shared<PrivilegedAction>();
781 return std::make_shared<DataAccessException>();
787 handleScratchRegAccess:
790 return std::make_shared<DataAccessException>();
794 handleQueueRegAccess:
797 return std::make_shared<PrivilegedAction>();
801 return std::make_shared<DataAccessException>();
805 handleSparcErrorRegAccess:
809 return std::make_shared<DataAccessException>();
811 return std::make_shared<PrivilegedAction>();
818 DPRINTF(
TLB,
"TLB: DTB Translating local access\n");
819 req->setLocalAccessor(
825 req->setPaddr(req->getVaddr());
857 bool addr_mask =
bits(tlbdata,3,3);
858 bool data_real = !
bits(tlbdata,5,5);
859 bool inst_real = !
bits(tlbdata,4,4);
860 bool ctx_zero =
bits(tlbdata,18,16) > 0;
861 int part_id =
bits(tlbdata,15,8);
862 int pri_context =
bits(tlbdata,47,32);
865 bool real = (
mode ==
Execute) ? inst_real : data_real;
874 req->setPaddr(
vaddr);
883 return std::make_shared<InstructionAccessException>();
885 return std::make_shared<DataAccessException>();
888 tbe =
lookup(
vaddr, part_id, real, ctx_zero ? 0 : pri_context,
false);
904 for (
int x = 0;
x < 4;
x++) {
905 ttetag =
betoh(
mem.read<uint64_t>(tsbs[
x]));
906 if (ttetag.
valid() && ttetag.
va() == va_tag) {
907 uint64_t entry =
mem.read<uint64_t>(tsbs[
x]) +
sizeof(uint64_t);
910 DPRINTF(
TLB,
"Virtual(%#x)->Physical(%#x) found in TTE\n",
919 return std::make_shared<InstructionRealTranslationMiss>();
921 return std::make_shared<FastInstructionAccessMMUMiss>();
923 return std::make_shared<FastInstructionAccessMMUMiss>(
vaddr);
926 return std::make_shared<DataRealTranslationMiss>();
928 return std::make_shared<FastDataAccessMMUMiss>();
930 return std::make_shared<FastDataAccessMMUMiss>(
vaddr);
956 DPRINTF(IPR,
"Memory Mapped IPR Read: asi=%#X a=%#x\n",
957 (uint32_t)pkt->
req->getArchFlags(), pkt->
getAddr());
1031 pkt->
setBE((uint64_t)0);
1050 goto doMmuReadError;
1072 goto doMmuReadError;
1127 panic(
"need to impl DTB::doMmuRegRead() got asi=%#x, va=%#x\n",
1139 ASI asi = (
ASI)pkt->
req->getArchFlags();
1145 int entry_insert = -1;
1152 DPRINTF(IPR,
"Memory Mapped IPR Write: asi=%#X a=%#x d=%#X\n",
1153 (uint32_t)asi,
va,
data);
1171 goto doMmuWriteError;
1229 inform(
"Ignoring write to SPARC ERROR regsiter\n");
1245 goto doMmuWriteError;
1249 entry_insert =
bits(
va, 8,3);
1252 assert(entry_insert != -1 ||
mbits(
va,10,9) ==
va);
1254 va_insert =
mbits(ta_insert, 63,13);
1255 ct_insert =
mbits(ta_insert, 12,0);
1257 real_insert =
bits(
va, 9,9);
1260 itb->
insert(va_insert, part_insert, ct_insert, real_insert,
1264 entry_insert =
bits(
va, 8,3);
1267 assert(entry_insert != -1 ||
mbits(
va,10,9) ==
va);
1269 va_insert =
mbits(ta_insert, 63,13);
1270 ct_insert =
mbits(ta_insert, 12,0);
1272 real_insert =
bits(
va, 9,9);
1275 insert(va_insert, part_insert, ct_insert, real_insert, pte,
1309 panic(
"Invalid type for IMMU demap\n");
1325 goto doMmuWriteError;
1359 panic(
"Invalid type for IMMU demap\n");
1377 getCpuPtr()->postInterrupt(0,
bits(
data, 5, 0), 0);
1381 panic(
"need to impl DTB::doMmuRegWrite() got asi=%#x, va=%#x d=%#x\n",
1417 uint64_t c0_config, uint64_t cX_tsb, uint64_t cX_config)
1430 uint64_t ptr =
mbits(tsb,63,13);
1431 bool split =
bits(tsb,12,12);
1432 int tsb_size =
bits(tsb,3,0);
1433 int page_size = (
ps ==
Ps0) ?
bits(config, 2,0) :
bits(config,10,8);
1435 if (
ps ==
Ps1 && split)
1436 ptr |=
ULL(1) << (13 + tsb_size);
1437 ptr |= (
tag_access >> (9 + page_size * 3)) &
mask(12+tsb_size, 4);
1452 free_list.push_back(entry -
tlb);
1466 for (
int x = 0;
x <
size;
x++) {
1478 if (oldSize !=
size)
1479 panic(
"Don't support unserializing different sized TLBs\n");
1486 for (
int idx : free_list)
1499 for (
int x = 0;
x <
size;
x++) {
#define fatal(...)
This implements a cprintf based fatal() function.
virtual RegVal readMiscRegNoEffect(RegIndex misc_reg) const =0
uint64_t TteRead(int entry)
Give an entry id, read that tlb entries' tte.
virtual void finish(const Fault &fault, const RequestPtr &req, ThreadContext *tc, Mode mode)=0
@ ASI_DMMU_CTXT_ZERO_TSB_BASE_PS1
@ ASI_IMMU_CTXT_NONZERO_TSB_BASE_PS1
void makeAtomicResponse()
T getBE() const
Get the data in the packet byte swapped from big endian to host endian.
void demapContext(int partition_id, int context_id)
Remove all entries that match a given context/partition id.
void demapPage(Addr va, int partition_id, bool real, int context_id)
Remve all entries that match a certain partition id, (contextid), and va).
std::list< TlbEntry * > freeList
@ ASI_ITLB_DATA_ACCESS_REG
#define UNSERIALIZE_SCALAR(scalar)
bool asiIsUnPriv(ASI asi)
#define UNSERIALIZE_CONTAINER(member)
void GetTsbPtr(ThreadContext *tc, Addr addr, int ctx, Addr *ptrs)
@ ASI_DMMU_CTXT_NONZERO_TSB_BASE_PS0
Fault translateAtomic(const RequestPtr &req, ThreadContext *tc, Mode mode) override
bool asiIsLittle(ASI asi)
iterator insert(TlbRange &r, TlbEntry *d)
virtual BaseMMU * getMMUPtr()=0
Fault translateFunctional(const RequestPtr &req, ThreadContext *tc, Mode mode) override
Addr translate(Addr vaddr) const
bool asiIsPartialStore(ASI asi)
EmulationPageTable * pTable
Fault translateData(const RequestPtr &req, ThreadContext *tc, bool write)
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...
std::shared_ptr< Request > RequestPtr
RequestPtr req
A pointer to the original request.
Cycles doMmuRegWrite(ThreadContext *tc, Packet *pkt)
virtual PortProxy & getPhysProxy()=0
bool FullSystem
The FullSystem variable can be used to determine the current mode of simulation.
void writeSfsr(bool write, ContextType ct, bool se, FaultTypes ft, int asi)
virtual Process * getProcessPtr()=0
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)
bool asiIsNucleus(ASI asi)
void setBE(T v)
Set the value in the data pointer to v as big endian.
void serialize(CheckpointOut &cp) const
Cycles doMmuRegRead(ThreadContext *tc, Packet *pkt)
bool asiIsSecondary(ASI asi)
ThreadContext is the external interface to all thread state for anything outside of the CPU.
void flushAll() override
Remove all entries from the TLB.
void insert(Addr vpn, int partition_id, int context_id, bool real, const PageTableEntry &PTE, int entry=-1)
Insert a PTE into the TLB.
@ ASI_DMMU_CTXT_NONZERO_CONFIG
@ ASI_DTLB_DATA_ACCESS_REG
std::shared_ptr< FaultBase > Fault
bool asiIsInterrupt(ASI asi)
@ STRICT_ORDER
The request is required to be strictly ordered by CPU models and is non-speculative.
void serialize(CheckpointOut &cp) const override
Serialize an object.
@ ASI_DMMU_TSB_PS0_PTR_REG
@ ASI_IMMU_CTXT_ZERO_TSB_BASE_PS0
void clearInterrupt(ThreadID tid, int int_num, int index)
@ MISCREG_QUEUE_CPU_MONDO_HEAD
void demapAll(int partition_id)
Remove all non-locked entries from the tlb that match partition id.
constexpr T mbits(T val, unsigned first, unsigned last)
Mask off the given bits in place like bits() but without shifting.
constexpr decltype(nullptr) NoFault
void unserialize(CheckpointIn &cp)
void populate(uint64_t e, EntryType t=sun4u)
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
@ ASI_IMMU_CTXT_NONZERO_TSB_BASE_PS0
bool asiIsPrimary(ASI asi)
#define SERIALIZE_SCALAR(scalar)
bool asiIsNoFault(ASI asi)
void unserialize(CheckpointIn &cp) override
Unserialize an object.
@ MISCREG_SCRATCHPAD_R0
Scratchpad regiscers.
constexpr int findMsbSet(uint64_t val)
Returns the bit position of the MSB that is set in the input.
void translateTiming(const RequestPtr &req, ThreadContext *tc, Translation *translation, Mode mode) override
@ ASI_IMMU_CTXT_ZERO_CONFIG
Fault translateInst(const RequestPtr &req, ThreadContext *tc)
uint64_t get_vec(int int_num)
This object is a proxy for a port or other object which implements the functional response protocol,...
@ ASI_DMMU_CTXT_NONZERO_TSB_BASE_PS1
#define SERIALIZE_CONTAINER(member)
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
@ ASI_IMMU_CTXT_NONZERO_CONFIG
virtual void setMiscReg(RegIndex misc_reg, RegVal val)=0
void writeTagAccess(Addr va, int context)
bool validVirtualAddress(Addr va, bool am)
Checks if the virtual address provided is a valid one.
Cycles is a wrapper class for representing cycle counts, i.e.
@ ASI_DMMU_CTXT_ZERO_CONFIG
const Addr StartVAddrHole
std::ostream CheckpointOut
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.
Fault finalizePhysical(const RequestPtr &req, ThreadContext *tc, Mode mode) const override
Do post-translation physical address finalization.
static void ignore(const char *expr)
@ ASI_IMMU_TSB_PS1_PTR_REG
@ ASI_IMMU_TSB_PS0_PTR_REG
@ MISCREG_MMU_P_CONTEXT
MMU Internal Registers.
@ ASI_DMMU_TSB_PS1_PTR_REG
void paramIn(CheckpointIn &cp, const std::string &name, ExtMachInst &machInst)
@ ASI_SPARC_ERROR_STATUS_REG
bool asiIsSparcError(ASI asi)
virtual BaseCPU * getCpuPtr()=0
bool translate(Addr vaddr, Addr &paddr)
Translate function.
BaseInterrupts * getInterruptController(ThreadID tid)
@ ASI_IMMU_CTXT_ZERO_TSB_BASE_PS1
std::string csprintf(const char *format, const Args &...args)
uint64_t TagRead(int entry)
Given an entry id, read that tlb entries' tag.
#define ULL(N)
uint64_t constant
virtual System * getSystemPtr()=0
iterator find(const TlbRange &r)
#define panic(...)
This implements a cprintf based panic() function.
bool asiIsAsIfUser(ASI asi)
@ ASI_DMMU_CTXT_ZERO_TSB_BASE_PS0
Generated on Tue Mar 23 2021 19:41:19 for gem5 by doxygen 1.8.17