52 #include "debug/TLB.hh" 63 :
BaseTLB(p), configAddress(0), size(p->size),
64 tlb(size), lruSeq(0), m5opRange(p->
system->m5opRange())
67 fatal(
"TLBs must have a non-zero size.\n");
69 for (
int x = 0;
x <
size;
x++) {
70 tlb[
x].trieHandle = NULL;
85 for (
unsigned i = 1;
i <
size;
i++) {
90 assert(
tlb[lru].trieHandle);
92 tlb[lru].trieHandle = NULL;
102 assert(newEntry->
vaddr == vpn);
114 newEntry->
vaddr = vpn;
124 if (entry && update_lru)
133 for (
unsigned i = 0;
i <
size;
i++) {
134 if (
tlb[
i].trieHandle) {
136 tlb[
i].trieHandle = NULL;
151 DPRINTF(
TLB,
"Invalidating all non global entries.\n");
152 for (
unsigned i = 0;
i <
size;
i++) {
153 if (
tlb[
i].trieHandle && !
tlb[
i].global) {
155 tlb[
i].trieHandle = NULL;
182 pkt->
setData((uint8_t *)&data);
197 DPRINTF(
TLB,
"Addresses references internal memory.\n");
201 panic(
"CPUID memory space not yet implemented!\n");
207 return std::make_shared<GeneralProtection>(0);
209 req->setPaddr(req->getVaddr());
210 req->setLocalAccessor(
213 return localMiscRegAccess(read, regNum, tc, pkt);
225 assert(!(IOPort & ~0xFFFF));
226 if (IOPort == 0xCF8 && req->getSize() == 4) {
227 req->setPaddr(req->getVaddr());
228 req->setLocalAccessor(
231 return localMiscRegAccess(
235 }
else if ((IOPort & ~
mask(2)) == 0xCFC) {
239 if (
bits(configAddress, 31, 31)) {
241 mbits(configAddress, 30, 2) |
252 panic(
"Access to unrecognized internal address space %#x.\n",
261 Addr paddr = req->getPaddr();
267 req->setLocalAccessor(
271 PseudoInst::pseudoInst<X86PseudoInstABI>(tc, func, ret);
279 LocalApicBase localApicBase =
299 paddr - apicRange.
start()));
309 Mode mode,
bool &delayedResponse,
bool timing)
315 delayedResponse =
false;
324 DPRINTF(
TLB,
"Translating vaddr %#x.\n", vaddr);
332 if (m5Reg.mode != LongMode) {
333 DPRINTF(
TLB,
"Not in long mode. Checking segment protection.\n");
338 return std::make_shared<GeneralProtection>(0);
342 if (!attr.writable && (mode ==
Write || storeCheck))
343 return std::make_shared<GeneralProtection>(0);
344 if (!attr.readable && mode ==
Read)
345 return std::make_shared<GeneralProtection>(0);
346 expandDown = attr.expandDown;
352 unsigned logSize = sizeOverride ? (unsigned)m5Reg.altAddr
353 : (
unsigned)m5Reg.defAddr;
354 int size = (1 << logSize) * 8;
356 Addr endOffset = offset + req->getSize() - 1;
358 DPRINTF(
TLB,
"Checking an expand down segment.\n");
359 warn_once(
"Expand down segments are untested.\n");
360 if (offset <= limit || endOffset <= limit)
361 return std::make_shared<GeneralProtection>(0);
363 if (offset > limit || endOffset > limit)
364 return std::make_shared<GeneralProtection>(0);
382 "address %#x at pc %#x.\n",
391 if (timing || fault !=
NoFault) {
393 delayedResponse =
true;
403 return std::make_shared<PageFault>(
vaddr,
true,
mode,
407 DPRINTF(
TLB,
"Mapping %#x to %#x\n", alignedVaddr,
419 "doing protection checks.\n", entry->
paddr);
421 bool inUser = (m5Reg.cpl == 3 &&
424 bool badWrite = (!entry->
writable && (inUser || cr0.wp));
425 if ((inUser && !entry->
user) || (mode ==
Write && badWrite)) {
429 return std::make_shared<PageFault>(
vaddr,
true,
mode, inUser,
432 if (storeCheck && badWrite) {
435 return std::make_shared<PageFault>(
vaddr,
true,
Write, inUser,
440 DPRINTF(
TLB,
"Translated %#x -> %#x.\n", vaddr, paddr);
441 req->setPaddr(paddr);
447 DPRINTF(
TLB,
"Translated %#x -> %#x.\n", vaddr, vaddr);
448 req->setPaddr(vaddr);
453 DPRINTF(
TLB,
"Translated %#x -> %#x.\n", vaddr, vaddr);
454 req->setPaddr(vaddr);
463 bool delayedResponse;
464 return TLB::translate(req, tc, NULL, mode, delayedResponse,
false);
478 paddr =
insertBits(addr, logBytes - 1, 0, vaddr);
492 return std::make_shared<PageFault>(
vaddr,
true,
mode,
true,
false);
496 DPRINTF(
TLB,
"Translated (functional) %#x -> %#x.\n", vaddr, paddr);
497 req->setPaddr(paddr);
505 bool delayedResponse;
508 TLB::translate(req, tc, translation, mode, delayedResponse,
true);
509 if (!delayedResponse)
510 translation->
finish(fault, req, tc, mode);
524 using namespace Stats;
528 .
desc(
"TLB accesses on read requests");
532 .
desc(
"TLB accesses on write requests");
536 .
desc(
"TLB misses on read requests");
540 .
desc(
"TLB misses on write requests");
553 for (uint32_t
x = 0;
x <
size;
x++) {
554 if (
tlb[
x].trieHandle != NULL)
555 tlb[
x].serializeSection(cp,
csprintf(
"Entry%d", _count++));
566 fatal(
"TLB size less than the one in checkpoint!");
571 for (uint32_t
x = 0;
x < _size;
x++) {
590 X86TLBParams::create()
#define panic(...)
This implements a cprintf based panic() function.
const Addr PhysAddrPrefixPciConfig
Ports are used to interface objects to each other.
virtual void setMiscReg(RegIndex misc_reg, RegVal val)=0
Handle insert(Key key, unsigned width, Value *val)
Method which inserts a key/value pair into the trie.
decltype(nullptr) constexpr NoFault
Cycles is a wrapper class for representing cycle counts, i.e.
#define fatal(...)
This implements a cprintf based fatal() function.
void unserialize(CheckpointIn &cp) override
Unserialize an object.
void setConfigAddress(uint32_t addr)
Port * getTableWalkerPort() override
Get the table walker port.
std::vector< TlbEntry > tlb
Declaration of a request, the overall memory request consisting of the parts of the request that are ...
bool contains(const Addr &a) const
Determine if the range contains an address.
std::shared_ptr< Request > RequestPtr
virtual void markDelayed()=0
Signal that the translation has been delayed due to a hw page table walk.
static const unsigned MaxBits
static void decodeAddrOffset(Addr offset, uint8_t &func)
Fault start(ThreadContext *_tc, BaseTLB::Translation *translation, const RequestPtr &req, BaseTLB::Mode mode)
bool FullSystem
The FullSystem variable can be used to determine the current mode of simulation.
virtual Process * getProcessPtr()=0
Fault finalizePhysical(const RequestPtr &req, ThreadContext *tc, Mode mode) const override
Do post-translation physical address finalization.
const Addr IntAddrPrefixCPUID
Bitfield< 14 > expandDown
Fault translate(const RequestPtr &req, ThreadContext *tc, Translation *translation, Mode mode, bool &delayedResponse, bool timing)
ThreadContext is the external interface to all thread state for anything outside of the CPU...
The request is to an uncacheable address.
void setLE(T v)
Set the value in the data pointer to v as little endian.
const Addr IntAddrPrefixMask
The AddrRange class encapsulates an address range, and supports a number of tests to check if two ran...
#define UNSERIALIZE_SCALAR(scalar)
std::string csprintf(const char *format, const Args &...args)
static MiscRegIndex MISCREG_SEG_ATTR(int index)
static MiscRegIndex MISCREG_SEG_LIMIT(int index)
Fault startFunctional(ThreadContext *_tc, Addr &addr, unsigned &logBytes, BaseTLB::Mode mode)
TlbEntryTrie::Handle trieHandle
void setData(const uint8_t *p)
Copy data into the packet from the provided pointer.
void regStats() override
Callback to set stat parameters.
void translateTiming(const RequestPtr &req, ThreadContext *tc, Translation *translation, Mode mode) override
Value * lookup(Key key)
Method which looks up the Value corresponding to a particular key.
void writeData(uint8_t *p) const
Copy data from the packet to the memory at the provided pointer.
T insertBits(T val, int first, int last, B bit_val)
Returns val with bits first to last set to the LSBs of bit_val.
const Addr IntAddrPrefixMSR
virtual void finish(const Fault &fault, const RequestPtr &req, ThreadContext *tc, Mode mode)=0
virtual Addr instAddr() const =0
static MiscRegIndex MISCREG_SEG_SEL(int index)
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
const Request::FlagsType M5_VAR_USED SegmentFlagMask
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
Fault translateAtomic(const RequestPtr &req, ThreadContext *tc, Mode mode) override
const Addr IntAddrPrefixIO
#define SERIALIZE_SCALAR(scalar)
bool msrAddrToIndex(MiscRegIndex ®Num, Addr addr)
Find and return the misc reg corresponding to an MSR address.
void flushAll() override
Remove all entries from the TLB.
Derived & name(const std::string &name)
Set the name and marks this stat to print at the end of simulation.
Value * remove(Handle handle)
Method to delete a value from the trie.
virtual const std::string name() const
BitfieldType< SegDescriptorLimit > limit
TlbEntry * lookup(Addr va, bool update_lru=true)
EmulationPageTable * pTable
virtual RegVal readMiscRegNoEffect(RegIndex misc_reg) const =0
Declarations of a non-full system Page Table.
bool fixupFault(Addr vaddr)
Attempt to fix up a fault at vaddr by allocating a page on the stack.
static MiscRegIndex MISCREG_SEG_BASE(int index)
std::ostream CheckpointOut
void serialize(CheckpointOut &cp) const override
Serialize an object.
This is exposed globally, independent of the ISA.
void demapPage(Addr va, uint64_t asn) override
Fault translateInt(bool read, RequestPtr req, ThreadContext *tc)
virtual ContextID contextId() const =0
const Entry * lookup(Addr vaddr)
Lookup function.
const Addr PhysAddrPrefixIO
The request is required to be strictly ordered by CPU models and is non-speculative.
Addr start() const
Get the start address of the range.
Derived & desc(const std::string &_desc)
Set the description and marks this stat to print at the end of simulation.
Fault translateFunctional(const RequestPtr &req, ThreadContext *tc, 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 regStats()
Callback to set stat parameters.
T mbits(T val, int first, int last)
Mask off the given bits in place like bits() but without shifting.
T bits(T val, int first, int last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it...
static Addr x86LocalAPICAddress(const uint8_t id, const uint16_t addr)
virtual RegVal readMiscReg(RegIndex misc_reg)=0
std::shared_ptr< FaultBase > Fault
void unserializeSection(CheckpointIn &cp, const char *name)
Unserialize an a child object.
TlbEntry * insert(Addr vpn, const TlbEntry &entry)