Go to the documentation of this file.
52 #include "debug/TLB.hh"
66 :
BaseTLB(
p), configAddress(0), size(
p.size),
67 tlb(size), lruSeq(0), m5opRange(
p.
system->m5opRange()), stats(this)
70 fatal(
"TLBs must have a non-zero size.\n");
72 for (
int x = 0;
x <
size;
x++) {
73 tlb[
x].trieHandle = NULL;
88 for (
unsigned i = 1;
i <
size;
i++) {
93 assert(
tlb[lru].trieHandle);
95 tlb[lru].trieHandle = NULL;
105 assert(newEntry->vaddr == vpn);
117 newEntry->vaddr = vpn;
118 newEntry->trieHandle =
127 if (entry && update_lru)
136 for (
unsigned i = 0;
i <
size;
i++) {
137 if (
tlb[
i].trieHandle) {
139 tlb[
i].trieHandle = NULL;
154 DPRINTF(
TLB,
"Invalidating all non global entries.\n");
155 for (
unsigned i = 0;
i <
size;
i++) {
156 if (
tlb[
i].trieHandle && !
tlb[
i].global) {
158 tlb[
i].trieHandle = NULL;
179 localMiscRegAccess(
bool read,
RegIndex regNum,
200 DPRINTF(
TLB,
"Addresses references internal memory.\n");
204 panic(
"CPUID memory space not yet implemented!\n");
210 return std::make_shared<GeneralProtection>(0);
212 req->setPaddr(req->getVaddr());
213 req->setLocalAccessor(
216 return localMiscRegAccess(read, regNum, tc, pkt);
228 assert(!(IOPort & ~0xFFFF));
229 if (IOPort == 0xCF8 && req->getSize() == 4) {
230 req->setPaddr(req->getVaddr());
231 req->setLocalAccessor(
234 return localMiscRegAccess(
238 }
else if ((IOPort & ~
mask(2)) == 0xCFC) {
255 panic(
"Access to unrecognized internal address space %#x.\n",
264 Addr paddr = req->getPaddr();
270 req->setLocalAccessor(
274 pseudo_inst::pseudoInst<X86PseudoInstABI, true>(tc, func, ret);
282 LocalApicBase localApicBase =
302 paddr - apicRange.
start()));
318 delayedResponse =
false;
332 const int addrSize = 8 << logAddrSize;
333 const Addr addrMask =
mask(addrSize);
339 if (m5Reg.mode != LongMode) {
340 DPRINTF(
TLB,
"Not in long mode. Checking segment protection.\n");
351 return std::make_shared<GeneralProtection>(0);
356 DPRINTF(
TLB,
"Tried to write to unwritable segment.\n");
357 return std::make_shared<GeneralProtection>(0);
360 DPRINTF(
TLB,
"Tried to read from unreadble segment.\n");
361 return std::make_shared<GeneralProtection>(0);
375 DPRINTF(
TLB,
"Checking an expand down segment.\n");
376 warn_once(
"Expand down segments are untested.\n");
378 return std::make_shared<GeneralProtection>(0);
383 return std::make_shared<GeneralProtection>(0);
401 "address %#x at pc %#x.\n",
410 if (timing || fault !=
NoFault) {
412 delayedResponse =
true;
422 return std::make_shared<PageFault>(
vaddr,
true,
mode,
425 Addr alignedVaddr =
p->pTable->pageAlign(
vaddr);
426 DPRINTF(
TLB,
"Mapping %#x to %#x\n", alignedVaddr,
429 p->pTable->pid(), alignedVaddr, pte->
paddr,
438 "doing protection checks.\n", entry->
paddr);
442 bool badWrite = (!entry->
writable && (inUser || cr0.wp));
443 if ((inUser && !entry->
user) ||
448 return std::make_shared<PageFault>(
vaddr,
true,
mode, inUser,
451 if (storeCheck && badWrite) {
454 return std::make_shared<PageFault>(
460 req->setPaddr(paddr);
467 req->setPaddr(
vaddr);
473 req->setPaddr(
vaddr);
483 bool delayedResponse;
513 return std::make_shared<PageFault>(
vaddr,
true,
mode,
true,
false);
518 req->setPaddr(paddr);
526 bool delayedResponse;
530 if (!delayedResponse)
543 : statistics::
Group(parent),
544 ADD_STAT(rdAccesses, statistics::units::Count::get(),
545 "TLB accesses on read requests"),
546 ADD_STAT(wrAccesses, statistics::units::Count::get(),
547 "TLB accesses on write requests"),
548 ADD_STAT(rdMisses, statistics::units::Count::get(),
549 "TLB misses on read requests"),
550 ADD_STAT(wrMisses, statistics::units::Count::get(),
551 "TLB misses on write requests")
564 for (uint32_t
x = 0;
x <
size;
x++) {
565 if (
tlb[
x].trieHandle != NULL)
566 tlb[
x].serializeSection(cp,
csprintf(
"Entry%d", _count++));
577 fatal(
"TLB size less than the one in checkpoint!");
582 for (uint32_t
x = 0;
x < _size;
x++) {
#define fatal(...)
This implements a cprintf based fatal() function.
std::vector< TlbEntry > tlb
Addr instAddr() const
Returns the memory address of the instruction this PC points to.
virtual RegVal readMiscReg(RegIndex misc_reg)=0
constexpr decltype(nullptr) NoFault
Addr start() const
Get the start address of the range.
constexpr Request::FlagsType SegmentFlagMask
TlbEntryTrie::Handle trieHandle
static Addr x86LocalAPICAddress(const uint8_t id, const uint16_t addr)
#define UNSERIALIZE_SCALAR(scalar)
void unserialize(CheckpointIn &cp) override
Unserialize an object.
const Addr IntAddrPrefixMSR
void setData(const uint8_t *p)
Copy data into the packet from the provided pointer.
void translateTiming(const RequestPtr &req, ThreadContext *tc, BaseMMU::Translation *translation, BaseMMU::Mode mode) override
bool contains(const Addr &a) const
Determine if the range contains an address.
virtual const PCStateBase & pcState() const =0
Fault translateAtomic(const RequestPtr &req, ThreadContext *tc, BaseMMU::Mode mode) override
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
virtual void markDelayed()=0
Signal that the translation has been delayed due to a hw page table walk.
Fault start(ThreadContext *_tc, BaseMMU::Translation *translation, const RequestPtr &req, BaseMMU::Mode mode)
Value * lookup(Key key)
Method which looks up the Value corresponding to a particular key.
virtual ContextID contextId() const =0
static RegIndex segAttr(int index)
statistics::Scalar rdMisses
std::string csprintf(const char *format, const Args &...args)
const Entry * lookup(Addr vaddr)
Lookup function.
BitfieldType< SegDescriptorLimit > limit
constexpr T mbits(T val, unsigned first, unsigned last)
Mask off the given bits in place like bits() but without shifting.
@ READ_MODIFY_WRITE
This request is a read which will be followed by a write.
Fault translateInt(bool read, RequestPtr req, ThreadContext *tc)
void serialize(CheckpointOut &cp) const override
Serialize an object.
Fault translateFunctional(const RequestPtr &req, ThreadContext *tc, BaseMMU::Mode mode) override
constexpr auto AddrSizeFlagShift
bool fixupFault(Addr vaddr)
Attempt to fix up a fault at vaddr by allocating a page on the stack.
Cycles is a wrapper class for representing cycle counts, i.e.
Fault finalizePhysical(const RequestPtr &req, ThreadContext *tc, BaseMMU::Mode mode) const override
Do post-translation physical address finalization.
static RegIndex segBase(int index)
Value * remove(Handle handle)
Method to delete a value from the trie.
void setConfigAddress(uint32_t addr)
@ UNCACHEABLE
The request is to an uncacheable address.
ThreadContext is the external interface to all thread state for anything outside of the CPU.
void unserializeSection(CheckpointIn &cp, const char *name)
Unserialize an a child object.
std::shared_ptr< FaultBase > Fault
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
EmulationPageTable * pTable
std::shared_ptr< Request > RequestPtr
const Addr IntAddrPrefixMask
gem5::X86ISA::TLB::TlbStats stats
statistics::Scalar wrMisses
constexpr T insertBits(T val, unsigned first, unsigned last, B bit_val)
Returns val with bits first to last set to the LSBs of bit_val.
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
const Addr IntAddrPrefixCPUID
virtual RegVal readMiscRegNoEffect(RegIndex misc_reg) const =0
static void decodeAddrOffset(Addr offset, uint8_t &func)
void writeData(uint8_t *p) const
Copy data from the packet to the memory at the provided pointer.
Fault translate(const RequestPtr &req, ThreadContext *tc, BaseMMU::Translation *translation, BaseMMU::Mode mode, bool &delayedResponse, bool timing)
const Addr PhysAddrPrefixPciConfig
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
TlbEntry * lookup(Addr va, bool update_lru=true)
static const unsigned MaxBits
#define SERIALIZE_SCALAR(scalar)
Bitfield< 14 > expandDown
void demapPage(Addr va, uint64_t asn) override
TlbEntry * insert(Addr vpn, const TlbEntry &entry)
@ 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.
statistics::Scalar rdAccesses
TlbStats(statistics::Group *parent)
constexpr auto CPL0FlagBit
constexpr auto AddrSizeFlagMask
Fault startFunctional(ThreadContext *_tc, Addr &addr, unsigned &logBytes, BaseMMU::Mode mode)
Ports are used to interface objects to each other.
bool msrAddrToIndex(RegIndex ®_num, Addr addr)
Find and return the misc reg corresponding to an MSR address.
virtual void setMiscReg(RegIndex misc_reg, RegVal val)=0
void flushAll() override
Remove all entries from the TLB.
Port * getTableWalkerPort() override
Get the table walker port.
statistics::Scalar wrAccesses
std::ostream CheckpointOut
const Addr IntAddrPrefixIO
void setLE(T v)
Set the value in the data pointer to v as little endian.
Handle insert(Key key, unsigned width, Value *val)
Method which inserts a key/value pair into the trie.
The AddrRange class encapsulates an address range, and supports a number of tests to check if two ran...
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
virtual void finish(const Fault &fault, const RequestPtr &req, ThreadContext *tc, BaseMMU::Mode mode)=0
static RegIndex segLimit(int index)
const Addr PhysAddrPrefixIO
#define panic(...)
This implements a cprintf based panic() function.
Generated on Wed Jul 13 2022 10:38:55 for gem5 by doxygen 1.8.17