gem5 [DEVELOP-FOR-25.0]
Loading...
Searching...
No Matches
mem_state.cc
Go to the documentation of this file.
1/*
2 * Copyright (c) 2017-2020 Advanced Micro Devices, Inc.
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are
7 * met: redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer;
9 * redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution;
12 * neither the name of the copyright holders nor the names of its
13 * contributors may be used to endorse or promote products derived from
14 * this software without specific prior written permission.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29#include "sim/mem_state.hh"
30
31#include <cassert>
32
33#include "arch/generic/mmu.hh"
34#include "debug/Vma.hh"
36#include "sim/process.hh"
38#include "sim/system.hh"
39#include "sim/vma.hh"
40
41namespace gem5
42{
43
44MemState::MemState(Process *owner, Addr brk_point, Addr stack_base,
45 Addr max_stack_size, Addr next_thread_stack_base,
46 Addr mmap_end)
47 : _ownerProcess(owner),
48 _pageBytes(owner->pTable->pageSize()), _brkPoint(brk_point),
49 _stackBase(stack_base), _stackSize(max_stack_size),
50 _maxStackSize(max_stack_size), _stackMin(stack_base - max_stack_size),
51 _nextThreadStackBase(next_thread_stack_base),
52 _mmapEnd(mmap_end)
53{
54}
55
58{
59 if (this == &in)
60 return *this;
61
69 _mmapEnd = in._mmapEnd;
70 _vmaList = in._vmaList; /* This assignment does a deep copy. */
71
72 return *this;
73}
74
75void
77{
78 _ownerProcess = owner;
79}
80
81bool
82MemState::isUnmapped(Addr start_addr, Addr length)
83{
84 Addr end_addr = start_addr + length;
85 const AddrRange range(start_addr, end_addr);
86 for (const auto &vma : _vmaList) {
87 if (vma.intersects(range))
88 return false;
89 }
90
95 for (auto start = start_addr; start < end_addr;
96 start += _pageBytes) {
97 if (_ownerProcess->pTable->lookup(start) != nullptr) {
98 panic("Someone allocated physical memory at VA %p without "
99 "creating a VMA!\n", start);
100 return false;
101 }
102 }
103 return true;
104}
105
106void
108{
115 auto page_aligned_new_brk = roundUp(new_brk, _pageBytes);
116 auto page_aligned_old_brk = roundUp(old_brk, _pageBytes);
117
130 if (new_brk < old_brk) {
131 const auto length = page_aligned_old_brk - page_aligned_new_brk;
132 if (length > 0)
133 unmapRegion(page_aligned_new_brk, length);
134 _brkPoint = new_brk;
135 return;
136 }
137
147 if (page_aligned_new_brk > page_aligned_old_brk) {
148 auto length = page_aligned_new_brk - page_aligned_old_brk;
154 if (!isUnmapped(page_aligned_old_brk, length)) {
155 return;
156 }
157
165 mapRegion(page_aligned_old_brk, length, "heap");
166 }
167
168 _brkPoint = new_brk;
169}
170
171void
172MemState::mapRegion(Addr start_addr, Addr length,
173 const std::string& region_name, int sim_fd, Addr offset)
174{
175 DPRINTF(Vma, "memstate: creating vma (%s) [0x%x - 0x%x]\n",
176 region_name.c_str(), start_addr, start_addr + length);
177
182 assert(isUnmapped(start_addr, length));
183
187 _vmaList.emplace_back(AddrRange(start_addr, start_addr + length),
188 _pageBytes, region_name, sim_fd, offset);
189}
190
191void
192MemState::unmapRegion(Addr start_addr, Addr length)
193{
194 Addr end_addr = start_addr + length;
195 const AddrRange range(start_addr, end_addr);
196
197 auto vma = std::begin(_vmaList);
198 while (vma != std::end(_vmaList)) {
199 if (vma->isStrictSuperset(range)) {
200 DPRINTF(Vma, "memstate: split vma [0x%x - 0x%x] into "
201 "[0x%x - 0x%x] and [0x%x - 0x%x]\n",
202 vma->start(), vma->end(),
203 vma->start(), start_addr,
204 end_addr, vma->end());
209 _vmaList.push_back(*vma);
210 _vmaList.back().sliceRegionRight(start_addr);
211
215 vma->sliceRegionLeft(end_addr);
216
221 break;
222 } else if (vma->isSubset(range)) {
223 DPRINTF(Vma, "memstate: destroying vma [0x%x - 0x%x]\n",
224 vma->start(), vma->end());
228 vma = _vmaList.erase(vma);
229
230 continue;
231
232 } else if (vma->intersects(range)) {
236 if (vma->start() < 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 vma->start(), start_addr);
244 vma->sliceRegionRight(start_addr);
245 } else {
246 DPRINTF(Vma, "memstate: resizing vma [0x%x - 0x%x] "
247 "into [0x%x - 0x%x]\n",
248 vma->start(), vma->end(),
249 end_addr, vma->end());
253 vma->sliceRegionLeft(end_addr);
254 }
255 }
256
257 vma++;
258 }
259
268 for (auto *tc: _ownerProcess->system->threads) {
269 tc->getMMUPtr()->flushAll();
270 }
271
275 _ownerProcess->deallocateMem(start_addr, length);
276}
277
278void
279MemState::remapRegion(Addr start_addr, Addr new_start_addr, Addr length)
280{
284 unmapRegion(new_start_addr, length);
285
286 Addr end_addr = start_addr + length;
287 const AddrRange range(start_addr, end_addr);
288
289 auto vma = std::begin(_vmaList);
290 while (vma != std::end(_vmaList)) {
291 if (vma->isStrictSuperset(range)) {
295 _vmaList.push_back(*vma);
296 _vmaList.back().sliceRegionRight(start_addr);
297
301 _vmaList.push_back(*vma);
302 _vmaList.back().sliceRegionLeft(end_addr);
303
308 vma->sliceRegionLeft(start_addr);
309 vma->sliceRegionRight(end_addr);
310 vma->remap(new_start_addr);
311
316 break;
317 } else if (vma->isSubset(range)) {
321 vma->remap(vma->start() - start_addr + new_start_addr);
322 } else if (vma->intersects(range)) {
326 _vmaList.push_back(*vma);
327
328 if (vma->start() < start_addr) {
332 _vmaList.back().sliceRegionRight(start_addr);
333
337 vma->sliceRegionLeft(start_addr);
338 vma->remap(new_start_addr);
339 } else {
343 _vmaList.back().sliceRegionLeft(end_addr);
344
348 vma->sliceRegionRight(end_addr);
349 vma->remap(new_start_addr + vma->start() - start_addr);
350 }
351 }
352
353 vma++;
354 }
355
364 for (auto *tc: _ownerProcess->system->threads) {
365 tc->getMMUPtr()->flushAll();
366 }
367
368 do {
369 if (!_ownerProcess->pTable->isUnmapped(start_addr, _pageBytes))
370 _ownerProcess->pTable->remap(start_addr, _pageBytes,
371 new_start_addr);
372
373 start_addr += _pageBytes;
374 new_start_addr += _pageBytes;
375
382 length -= _pageBytes;
383 } while (length > 0);
384}
385
386bool
388{
393 for (const auto &vma : _vmaList) {
394 if (vma.contains(vaddr)) {
395 Addr vpage_start = roundDown(vaddr, _pageBytes);
396 _ownerProcess->allocateMem(vpage_start, _pageBytes);
397
404 if (vma.hasHostBuf()) {
409 for (auto &cid : _ownerProcess->contextIds) {
410 auto *tc = _ownerProcess->system->threads[cid];
412 virt_mem(tc, SETranslatingPortProxy::Always);
413 vma.fillMemPages(vpage_start, _pageBytes, virt_mem);
414 }
415 }
416 return true;
417 }
418 }
419
427 if (vaddr >= _stackMin && vaddr < _stackBase) {
429 return true;
430 }
431
437 while (vaddr < _stackMin) {
440 fatal("Maximum stack size exceeded\n");
441 }
442 _ownerProcess->allocateMem(_stackMin, _pageBytes);
443 inform("Increasing stack size by one page.");
444 }
445 return true;
446 }
447
448 return false;
449}
450
451Addr
453{
454 Addr start = _mmapEnd;
455
456 if (_ownerProcess->mmapGrowsDown())
457 start = _mmapEnd - length;
458
459 // Look for a contiguous region of free virtual memory. We can't assume
460 // that the region beyond mmap_end is free because of fixed mappings from
461 // the user.
462 while (!isUnmapped(start, length)) {
463 DPRINTF(Vma, "memstate: cannot extend vma for mmap region at %p. "
464 "Virtual address range is already reserved! Skipping a page "
465 "and trying again!\n", start);
466 start = (_ownerProcess->mmapGrowsDown()) ? start - _pageBytes :
467 start + _pageBytes;
468 }
469
470 DPRINTF(Vma, "memstate: extending mmap region (old %p) (new %p)\n",
471 _mmapEnd,
472 _ownerProcess->mmapGrowsDown() ? start : start + length);
473
474 _mmapEnd = _ownerProcess->mmapGrowsDown() ? start : start + length;
475
476 return start;
477}
478
479std::string
481{
482 std::stringstream file_content;
483
484 for (auto vma : _vmaList) {
485 std::stringstream line;
486 line << std::hex << vma.start() << "-";
487 line << std::hex << vma.end() << " ";
488 line << "r-xp 00000000 00:00 0 ";
489 line << "[" << vma.getName() << "]" << std::endl;
490 file_content << line.str();
491 }
492
493 return file_content.str();
494}
495
496} // namespace gem5
#define DPRINTF(x,...)
Definition trace.hh:209
The AddrRange class encapsulates an address range, and supports a number of tests to check if two ran...
Definition addr_range.hh:82
This class holds the memory state for the Process class and all of its derived, architecture-specific...
Definition mem_state.hh:68
std::list< VMA > _vmaList
The _vmaList member is a list of virtual memory areas in the target application space that have been ...
Definition mem_state.hh:292
Process * _ownerProcess
Owner process of MemState.
Definition mem_state.hh:269
Addr extendMmap(Addr length)
Definition mem_state.cc:452
bool fixupFault(Addr vaddr)
Attempt to fix up a fault at vaddr by allocating a page.
Definition mem_state.cc:387
Addr _nextThreadStackBase
Definition mem_state.hh:277
void updateBrkRegion(Addr old_brk, Addr new_brk)
Change the end of a process' program break.
Definition mem_state.cc:107
std::string printVmaList()
Print the list of VMAs in a format similar to /proc/self/maps.
Definition mem_state.cc:480
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...
Definition mem_state.cc:82
void resetOwner(Process *owner)
Change the Process owner in case this MemState is copied.
Definition mem_state.cc:76
MemState & operator=(const MemState &in)
Definition mem_state.cc:57
MemState(Process *owner, Addr brk_point, Addr stack_base, Addr max_stack_size, Addr next_thread_stack_base, Addr mmap_end)
Definition mem_state.cc:44
void unmapRegion(Addr start_addr, Addr length)
Unmap a pre-existing region.
Definition mem_state.cc:192
void remapRegion(Addr start_addr, Addr new_start_addr, Addr length)
Remap a pre-existing region.
Definition mem_state.cc:279
void mapRegion(Addr start_addr, Addr length, const std::string &name="anon", int sim_fd=-1, Addr offset=0)
Add a new memory region.
Definition mem_state.cc:172
static constexpr T roundDown(const T &val, const U &align)
This function is used to align addresses in memory.
Definition intmath.hh:279
static constexpr T roundUp(const T &val, const U &align)
This function is used to align addresses in memory.
Definition intmath.hh:260
#define panic(...)
This implements a cprintf based panic() function.
Definition logging.hh:220
#define fatal(...)
This implements a cprintf based fatal() function.
Definition logging.hh:232
#define inform(...)
Definition logging.hh:289
Bitfield< 23, 0 > offset
Definition types.hh:144
Bitfield< 7 > vma
Definition vector.hh:79
Copyright (c) 2024 Arm Limited All rights reserved.
Definition binary32.hh:36
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition types.hh:147

Generated on Mon May 26 2025 09:19:13 for gem5 by doxygen 1.13.2