summaryrefslogtreecommitdiff
path: root/mem/page_table.cc
diff options
context:
space:
mode:
Diffstat (limited to 'mem/page_table.cc')
-rw-r--r--mem/page_table.cc111
1 files changed, 37 insertions, 74 deletions
diff --git a/mem/page_table.cc b/mem/page_table.cc
index cd93c0692..63b60de24 100644
--- a/mem/page_table.cc
+++ b/mem/page_table.cc
@@ -34,33 +34,25 @@
#include <map>
#include <fstream>
-using namespace std;
-
+#include "base/bitfield.hh"
#include "base/intmath.hh"
#include "base/trace.hh"
-#include "mem/physical.hh"
#include "mem/page_table.hh"
#include "sim/builder.hh"
#include "sim/sim_object.hh"
+#include "sim/system.hh"
+
+using namespace std;
-PageTable::PageTable(const std::string &name)
- : SimObject(name)
+PageTable::PageTable(System *_system, Addr _pageSize)
+ : pageSize(_pageSize), offsetMask(mask(floorLog2(_pageSize))),
+ system(_system)
{
+ assert(isPowerOf2(pageSize));
}
PageTable::~PageTable()
{
- //Iterate the page table freeing the memoruy
- //Addr addr;
- //std::map<Addr,Addr>::iterator iter;
-
- //iter = pTable.begin();
- //while(iter != pTable.end())
- //{
- //delete [] (uint8_t *)iter->second;
-// iter ++;
- // }
-
}
Fault
@@ -89,78 +81,49 @@ PageTable::page_check(Addr addr, int size) const
}
-Fault
-PageTable::translate(CpuRequestPtr &req)
-{
-//Should I check here for accesses that are > VMPageSize?
- req->paddr = translate(req->vaddr, req->asid);
- return page_check(req->paddr, req->size);
-}
-Addr
-PageTable::translate(Addr vaddr, unsigned asid)
+void
+PageTable::allocate(Addr vaddr, int size)
{
- Addr hash_addr;
- std::map<Addr,Addr>::iterator iter;
-
- //DPRINTF(PageTable,"PageTable: Virtual Address %#x Translating for ASID %i\n",
- // vaddr,asid);
-
- //Create the hash_addr
- //Combine vaddr and asid
- hash_addr = vaddr & (~(VMPageSize - 1)) | asid;
-
- //DPRINTF(PageTable,"PageTable: Hash Address %#x\n",hash_addr);
-
- //Look into the page table
- iter=pTable.find(hash_addr);
-
- //bool page_fault = true;
+ // starting address must be page aligned
+ assert(pageOffset(vaddr) == 0);
- //Store the translated address if found, and return
- if (iter != pTable.end()) //Found??
- {
- Addr return_addr = iter->second + (vaddr & (VMPageSize - 1));
+ for (; size > 0; size -= pageSize, vaddr += pageSize) {
+ std::map<Addr,Addr>::iterator iter = pTable.find(vaddr);
- return return_addr;
- }
- else//Alocate a new page, register translation
- {
- Addr return_addr;
-
- //DPRINTF(PageTable,"PageTable: Page Not Found. Allocating new page\n");
-
- Addr new_page = mem->new_page();
-
- pTable[hash_addr] = new_page;
-
- return_addr = new_page + (vaddr & (VMPageSize - 1));
+ if (iter != pTable.end()) {
+ // already mapped
+ fatal("PageTable::allocate: address 0x%x already mapped", vaddr);
+ }
- return return_addr;
+ pTable[vaddr] = system->new_page();
}
}
-BEGIN_DECLARE_SIM_OBJECT_PARAMS(PageTable)
- SimObjectParam<PhysicalMemory *> physmem;
-END_DECLARE_SIM_OBJECT_PARAMS(PageTable)
+bool
+PageTable::translate(Addr vaddr, Addr &paddr)
+{
+ Addr page_addr = pageAlign(vaddr);
+ std::map<Addr,Addr>::iterator iter = pTable.find(page_addr);
-BEGIN_INIT_SIM_OBJECT_PARAMS(PageTable)
+ if (iter == pTable.end()) {
+ return false;
+ }
- INIT_PARAM_DFLT(physmem, "Pointer to functional memory", NULL)
+ paddr = iter->second + pageOffset(vaddr);
+ return true;
+}
-END_INIT_SIM_OBJECT_PARAMS(PageTable)
-CREATE_SIM_OBJECT(PageTable)
+Fault
+PageTable::translate(CpuRequestPtr &req)
{
- PageTable *pTable = new PageTable(getInstanceName());
-
- if (physmem)
- pTable->setPhysMem(physmem);
-
- return pTable;
+ assert(pageAlign(req->vaddr + req->size - 1) == pageAlign(req->vaddr));
+ if (!translate(req->vaddr, req->paddr)) {
+ return Machine_Check_Fault;
+ }
+ return page_check(req->paddr, req->size);
}
-
-REGISTER_SIM_OBJECT("PageTable", PageTable)