libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
mman.cxx
1// libcosmos
2#include <cosmos/error/ApiError.hxx>
3#include <cosmos/error/UsageError.hxx>
4#include <cosmos/proc/mman.hxx>
5#include <cosmos/utils.hxx>
6
7namespace cosmos::mem {
8
9void* map(const size_t length, const MapSettings &settings) {
10 // type is non-bitmask value that is added to the actual system call flags.
11 // for type safety we use two separate types in the API.
12 const auto flags = settings.flags.raw() | to_integral(settings.type);
13
14 auto addr = ::mmap(
15 settings.addr, length, settings.access.raw(),
16 flags, to_integral(settings.fd.raw()), settings.offset);
17
18 if (addr == MAP_FAILED) {
19 cosmos_throw (ApiError("mmap()"));
20 }
21
22 return addr;
23}
24
25void unmap(void *addr, const size_t length) {
26 if (::munmap(addr, length) != 0) {
27 cosmos_throw (ApiError("munmap()"));
28 }
29}
30
31void sync(void *addr, const size_t length, const SyncFlags flags) {
32 if (::msync(addr, length, flags.raw()) != 0) {
33 cosmos_throw (ApiError("msync()"));
34 }
35}
36
37void* remap(void *old_addr, const size_t old_size, const size_t new_size, const RemapFlags flags, std::optional<void*> new_addr) {
38 if (flags[RemapFlag::FIXED] && !new_addr) {
39 cosmos_throw (UsageError("missing new_addr argument along with FIXED"));
40 } else if (!flags[RemapFlag::FIXED] && new_addr) {
41 cosmos_throw (UsageError("unexpected new_addr argument without FIXED"));
42 }
43
44 auto ret = ::mremap(old_addr, old_size, new_size, flags.raw(), new_addr ? *new_addr : MAP_FAILED);
45
46 if (ret == MAP_FAILED) {
47 cosmos_throw (ApiError("mremap()"));
48 }
49
50 return ret;
51}
52
53void lock(void *addr, const size_t length, const LockFlags flags) {
54 if (::mlock2(addr, length, flags.raw()) != 0) {
55 cosmos_throw (ApiError("mlock2()"));
56 }
57}
58
59void unlock(void *addr, const size_t length) {
60 if (::munlock(addr, length) != 0) {
61 cosmos_throw (ApiError("munlock()"));
62 }
63}
64
65void lock_all(const LockAllFlags flags) {
66 if (::mlockall(flags.raw()) != 0) {
67 cosmos_throw (ApiError("mlockall()"));
68 }
69}
70
71void unlock_all() {
72 if (::munlockall() != 0) {
73 cosmos_throw (ApiError("munlockall()"));
74 }
75}
76
77void protect(void *addr, const size_t length, const AccessFlags flags, const ProtectFlags extra) {
78
79 const auto raw_flags = flags.raw() | extra.raw();
80
81 if (::mprotect(addr, length, raw_flags) != 0) {
82 cosmos_throw (ApiError("mprotect()"));
83 }
84}
85
86void MapFlags::setTLBPageSize(size_t page_size) {
87 if (!page_size || (page_size & (page_size-1)) != 0) {
88 // it's not a power of 2
89 cosmos_throw (UsageError("non-log2 TLB page size encountered"));
90 }
91
92 EnumBaseType log2 = 1;
93
94 while ((page_size & 0x1) != 1) {
95 log2++;
96 page_size >>= 1;
97 }
98
99 constexpr auto MAX_TLB_LOG2 = 0x3F;
100
101 if (log2 > MAX_TLB_LOG2) {
102 // a maximum of 6 bits can be used for specifying the log2 for
103 // the page size
104 cosmos_throw (UsageError("requested TLB page size is too large"));
105 }
106
107 // set all TLB page bits to zero first
108 this->reset(MapFlags{MAX_TLB_LOG2});
109 // now set the required bits for the selected TLB page size
110 this->set(MapFlags{log2 << MAP_HUGE_SHIFT});
111}
112
113} // end ns
BitMask & set(const All)
Definition BitMask.hxx:77
Exception type for logical usage errors within the application.
Flags used in MapSettings.
Definition mman.hxx:108
void setTLBPageSize(size_t page_size)
Sets the TLB page size if MapFlag::HUGETLB is set.
Definition mman.cxx:86