34 #include "debug/Vma.hh" 42 Addr max_stack_size,
Addr next_thread_stack_base,
44 : _ownerProcess(owner),
45 _pageBytes(owner->
system->getPageBytes()), _brkPoint(brk_point),
46 _stackBase(stack_base), _maxStackSize(max_stack_size),
47 _nextThreadStackBase(next_thread_stack_base),
48 _mmapEnd(mmap_end), _endBrkPoint(brk_point)
82 const AddrRange range(start_addr, end_addr);
84 if (vma.intersects(range))
92 for (
auto start = start_addr; start < end_addr;
95 panic(
"Someone allocated physical memory at VA %p without " 96 "creating a VMA!\n", start);
114 if (new_brk < old_brk) {
156 _endBrkPoint = page_aligned_brk;
164 const std::string& region_name,
int sim_fd,
Addr offset)
166 DPRINTF(Vma,
"memstate: creating vma (%s) [0x%x - 0x%x]\n",
167 region_name.c_str(), start_addr, start_addr +
length);
186 const AddrRange range(start_addr, end_addr);
190 if (vma->isStrictSuperset(range)) {
191 DPRINTF(Vma,
"memstate: split vma [0x%x - 0x%x] into " 192 "[0x%x - 0x%x] and [0x%x - 0x%x]\n",
193 vma->start(), vma->end(),
194 vma->start(), start_addr,
195 end_addr, vma->end());
201 _vmaList.back().sliceRegionRight(start_addr);
206 vma->sliceRegionLeft(end_addr);
213 }
else if (vma->isSubset(range)) {
214 DPRINTF(Vma,
"memstate: destroying vma [0x%x - 0x%x]\n",
215 vma->start(), vma->end());
223 }
else if (vma->intersects(range)) {
227 if (vma->start() < start_addr) {
228 DPRINTF(Vma,
"memstate: resizing vma [0x%x - 0x%x] " 229 "into [0x%x - 0x%x]\n",
230 vma->start(), vma->end(),
231 vma->start(), start_addr);
235 vma->sliceRegionRight(start_addr);
237 DPRINTF(Vma,
"memstate: resizing vma [0x%x - 0x%x] " 238 "into [0x%x - 0x%x]\n",
239 vma->start(), vma->end(),
240 end_addr, vma->end());
244 vma->sliceRegionLeft(end_addr);
260 tc->getDTBPtr()->flushAll();
261 tc->getITBPtr()->flushAll();
277 }
while (length > 0);
284 const AddrRange range(start_addr, end_addr);
288 if (vma->isStrictSuperset(range)) {
293 _vmaList.back().sliceRegionRight(start_addr);
299 _vmaList.back().sliceRegionLeft(end_addr);
305 vma->sliceRegionLeft(start_addr);
306 vma->sliceRegionRight(end_addr);
307 vma->remap(new_start_addr);
314 }
else if (vma->isSubset(range)) {
318 vma->remap(vma->start() - start_addr + new_start_addr);
319 }
else if (vma->intersects(range)) {
325 if (vma->start() < start_addr) {
329 _vmaList.back().sliceRegionRight(start_addr);
334 vma->sliceRegionLeft(start_addr);
335 vma->remap(new_start_addr);
340 _vmaList.back().sliceRegionLeft(end_addr);
345 vma->sliceRegionRight(end_addr);
346 vma->remap(new_start_addr + vma->start() - start_addr);
362 tc->getDTBPtr()->flushAll();
363 tc->getITBPtr()->flushAll();
381 }
while (length > 0);
392 if (vma.contains(vaddr)) {
402 if (vma.hasHostBuf()) {
412 vma.fillMemPages(vpage_start,
_pageBytes, virt_mem);
439 fatal(
"Maximum stack size exceeded\n");
442 inform(
"Increasing stack size by one page.");
462 DPRINTF(Vma,
"memstate: cannot extend vma for mmap region at %p. " 463 "Virtual address range is already reserved! Skipping a page " 464 "and trying again!\n", start);
469 DPRINTF(Vma,
"memstate: extending mmap region (old %p) (new %p)\n",
481 std::stringstream file_content;
484 std::stringstream line;
485 line << std::hex << vma.start() <<
"-";
486 line << std::hex << vma.end() <<
" ";
487 line <<
"r-xp 00000000 00:00 0 ";
488 line <<
"[" << vma.getName() <<
"]" << std::endl;
489 file_content << line.str();
492 return file_content.str();
#define panic(...)
This implements a cprintf based panic() function.
Process * _ownerProcess
Owner process of MemState.
virtual void remap(Addr vaddr, int64_t size, Addr new_vaddr)
#define fatal(...)
This implements a cprintf based fatal() function.
MemState & operator=(const MemState &in)
std::vector< ContextID > contextIds
void allocateMem(Addr vaddr, int64_t size, bool clobber=false)
void mapRegion(Addr start_addr, Addr length, const std::string &name="anon", int sim_fd=-1, Addr offset=0)
Add a new memory region.
Addr _endBrkPoint
Keeps record of the furthest mapped heap location.
T roundUp(const T &val, const U &align)
This function is used to align addresses in memory.
ThreadContext is the external interface to all thread state for anything outside of the CPU...
ThreadContext * getThreadContext(ContextID tid) const
void updateBrkRegion(Addr old_brk, Addr new_brk)
Change the end of a process' program break.
The AddrRange class encapsulates an address range, and supports a number of tests to check if two ran...
void resetOwner(Process *owner)
Change the Process owner in case this MemState is copied.
void unmapRegion(Addr start_addr, Addr length)
Unmap a pre-existing region.
bool fixupFault(Addr vaddr)
Attempt to fix up a fault at vaddr by allocating a page.
std::string printVmaList()
Print the list of VMAs in a format similar to /proc/self/maps.
std::vector< ThreadContext * > threadContexts
Addr _nextThreadStackBase
T roundDown(const T &val, const U &align)
This function is used to align addresses in memory.
std::list< VMA > _vmaList
The _vmaList member is a list of virtual memory areas in the target application space that have been ...
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
This class holds the memory state for the Process class and all of its derived, architecture-specific...
EmulationPageTable * pTable
virtual bool isUnmapped(Addr vaddr, int64_t size)
Check if any pages in a region are already allocated.
const Entry * lookup(Addr vaddr)
Lookup function.
virtual void unmap(Addr vaddr, int64_t size)
virtual bool mmapGrowsDown() const
Does mmap region grow upward or downward from mmapEnd? Most platforms grow downward, but a few (such as Alpha) grow upward instead, so they can override this method to return false.
bool isUnmapped(Addr start_addr, Addr length)
Check if any page in the virtual address range from start_addr to start_addr + length is already mapp...
void remapRegion(Addr start_addr, Addr new_start_addr, Addr length)
Remap a pre-existing region.
MemState(Process *owner, Addr brk_point, Addr stack_base, Addr max_stack_size, Addr next_thread_stack_base, Addr mmap_end)
Addr extendMmap(Addr length)