summaryrefslogtreecommitdiff
path: root/mem/translating_port.cc
diff options
context:
space:
mode:
Diffstat (limited to 'mem/translating_port.cc')
-rw-r--r--mem/translating_port.cc28
1 files changed, 18 insertions, 10 deletions
diff --git a/mem/translating_port.cc b/mem/translating_port.cc
index 052d350b9..e385a74b6 100644
--- a/mem/translating_port.cc
+++ b/mem/translating_port.cc
@@ -47,29 +47,37 @@ TranslatingPort::readBlobFunctional(Addr addr, uint8_t *p, int size)
for (ChunkGenerator gen(addr, size, VMPageSize); !gen.done(); gen.next()) {
- if (!pTable->translate(gen.addr(),paddr))
- return Machine_Check_Fault;
+ if (!pTable->translate(gen.addr(),paddr))
+ return Machine_Check_Fault;
- port->readBlobFunctional(paddr, p + prevSize, gen.size());
- prevSize += gen.size();
+ port->readBlobFunctional(paddr, p + prevSize, gen.size());
+ prevSize += gen.size();
}
return No_Fault;
}
Fault
-TranslatingPort::writeBlobFunctional(Addr addr, uint8_t *p, int size)
+TranslatingPort::writeBlobFunctional(Addr addr, uint8_t *p, int size,
+ bool alloc)
{
Addr paddr;
int prevSize = 0;
for (ChunkGenerator gen(addr, size, VMPageSize); !gen.done(); gen.next()) {
- if (!pTable->translate(gen.addr(),paddr))
- return Machine_Check_Fault;
-
- port->writeBlobFunctional(paddr, p + prevSize, gen.size());
- prevSize += gen.size();
+ if (!pTable->translate(gen.addr(), paddr)) {
+ if (alloc) {
+ pTable->allocate(roundDown(gen.addr(), VMPageSize),
+ VMPageSize);
+ pTable->translate(gen.addr(), paddr);
+ } else {
+ return Machine_Check_Fault;
+ }
+ }
+
+ port->writeBlobFunctional(paddr, p + prevSize, gen.size());
+ prevSize += gen.size();
}
return No_Fault;