libcosmos
Linux C++ System Programming Library
|
#include <optional>
#include <sys/mman.h>
#include <linux/mman.h>
#include <cosmos/BitMask.hxx>
#include <cosmos/fs/FileDescriptor.hxx>
Go to the source code of this file.
Classes | |
class | cosmos::mem::MapFlags |
Flags used in MapSettings. More... | |
struct | cosmos::mem::MapSettings |
Collection of settings used in cosmos::mem::map(). More... | |
Macros | |
#define | PROT_SAO 0x10 |
Typedefs | |
using | cosmos::mem::AccessFlags = BitMask<AccessFlag> |
A mask of memory page access settings. | |
using | cosmos::mem::ProtectFlags = BitMask<ProtectFlag> |
A mask of extra settings used in mem::protect(). | |
using | cosmos::mem::RemapFlags = BitMask<RemapFlag> |
using | cosmos::mem::SyncFlags = BitMask<SyncFlag> |
using | cosmos::mem::LockFlags = BitMask<LockFlag> |
using | cosmos::mem::LockAllFlags = BitMask<LockAllFlag> |
Enumerations | |
enum class | cosmos::mem::MapType : int { SHARED = MAP_SHARED , SHARED_VALIDATE = MAP_SHARED_VALIDATE , PRIVATE = MAP_PRIVATE } |
The basic type of a memory mapping to be created. More... | |
enum class | cosmos::mem::AccessFlag : int { EXEC = PROT_EXEC , READ = PROT_READ , WRITE = PROT_WRITE , NONE = PROT_NONE , SEM = PROT_SEM , SAO = PROT_SAO } |
Different memory page access permissions. More... | |
enum class | cosmos::mem::MapFlag : int { ANONYMOUS = MAP_ANONYMOUS , FIXED = MAP_FIXED , FIXED_NOREPLACE = MAP_FIXED_NOREPLACE , GROWSDOWN = MAP_GROWSDOWN , HUGETLB = MAP_HUGETLB , LOCKED = MAP_LOCKED , NONBLOCK = MAP_NONBLOCK , NORESERVE = MAP_NORESERVE , POPULATE = MAP_POPULATE , STACK = MAP_STACK , SYNC = MAP_SYNC , UNINITIALIZED = MAP_UNINITIALIZED } |
Flags that influence properties of memory mappings. More... | |
enum class | cosmos::mem::ProtectFlag : int { GROWSUP = PROT_GROWSUP , GROWSDOWN = PROT_GROWSDOWN } |
Extra flags used with mem::protect(). More... | |
enum class | cosmos::mem::RemapFlag : int { MAYMOVE = MREMAP_MAYMOVE , FIXED = MREMAP_FIXED , DONTUNMAP = MREMAP_DONTUNMAP } |
Flags used with cosmos::mem::remap(). More... | |
enum class | cosmos::mem::SyncFlag : int { ASYNC = MS_ASYNC , SYNC = MS_SYNC , INVALIDATE = MS_INVALIDATE } |
Flags used with cosmos::mem::sync(). More... | |
enum class | cosmos::mem::LockFlag : unsigned int { LOCK_ON_FAULT = MLOCK_ONFAULT } |
Flags used with cosmos::mem::lock(). More... | |
enum class | cosmos::mem::LockAllFlag : int { CURRENT = MCL_CURRENT , FUTURE = MCL_FUTURE , ONFAULT = MCL_ONFAULT } |
Flags passed to cosmos::mem::lockall(). More... | |
Functions | |
void * | cosmos::mem::map (const size_t length, const MapSettings &settings) |
Request a memory mapping of the given length using the provided settings. | |
void | cosmos::mem::unmap (void *addr, const size_t length) |
Unmap an existing mapping at the given address and of the given length. | |
void | cosmos::mem::protect (void *addr, const size_t length, const AccessFlags flags, const ProtectFlags extra={}) |
Change memory protection settings of an existing mapping. | |
void * | cosmos::mem::remap (void *old_addr, const size_t old_size, const size_t new_size, const RemapFlags flags={}, std::optional< void * > new_addr={}) |
Expand or shrink an existing memory mapping. | |
void | cosmos::mem::sync (void *addr, const size_t length, const SyncFlags flags=SyncFlags{SyncFlag::SYNC}) |
Synchronize changes in a memory mapping with the file backing it. | |
void | cosmos::mem::lock (void *addr, const size_t length, const LockFlags flags={}) |
Lock pages in memory, preventing memory from being paged to the swap area. | |
void | cosmos::mem::unlock (void *addr, const size_t length) |
Unlock previously locked pages. | |
void | cosmos::mem::lock_all (const LockAllFlags flags) |
Locks all current and/or future pages in memory. | |
void | cosmos::mem::unlock_all () |
Unlock all current process memory pages.. | |
This header contains memory mapping related functionality.
Definition in file mman.hxx.
using cosmos::mem::AccessFlags = BitMask<AccessFlag> |
using cosmos::mem::LockAllFlags = BitMask<LockAllFlag> |
using cosmos::mem::LockFlags = BitMask<LockFlag> |
using cosmos::mem::ProtectFlags = BitMask<ProtectFlag> |
using cosmos::mem::RemapFlags = BitMask<RemapFlag> |
using cosmos::mem::SyncFlags = BitMask<SyncFlag> |
|
strong |
Different memory page access permissions.
Definition at line 38 of file mman.hxx.
|
strong |
Flags passed to cosmos::mem::lockall().
Enumerator | |
---|---|
CURRENT | lock all currently loaded pages in memory. |
FUTURE | lock all pages loaded in the future in memory. |
ONFAULT | lock all current/future pages, but don't pre-fault them. |
|
strong |
Flags used with cosmos::mem::lock().
Enumerator | |
---|---|
LOCK_ON_FAULT | lock all pages that are already resident, the rest will be locked after a page fault occurs. |
Definition at line 313 of file mman.hxx.
|
strong |
Flags that influence properties of memory mappings.
Enumerator | |
---|---|
ANONYMOUS | Create a mapping that is not backed by a file, contents are initialized to zero. The offset should be zero and the file descriptor invalid. |
FIXED | Map memory exactly at the given hint address, replacing already existing mappings at the address. |
FIXED_NOREPLACE | Like FIXED but don't replace existing mappings, fail with EEXISTS instead. |
GROWSDOWN | Create a mapping suitable for stacks, including automatic growing via a guard page. |
HUGETLB | Allocate the mapping using hugetlb page sizes, see also MapFlags::setTLBPageSize(). |
LOCKED | Mark the memory to be locked similar to mem::lock(), but no major faults will be prevented. |
NONBLOCK | Used in conjunction with POPULATE, currently causes POPULATE to do nothing though. |
NORESERVE | Do not reserve swap space for this mapping. Writes may fail with SIGSEGV if no physical memory is available. |
POPULATE | Pre-fault page tables for a mapping. This is reduce blocking on page faults later; failure to populate the mapping does not cause an error of mmap(), though. |
STACK | Allocate the mapping at an address suitable for a thread stack (currently has no effect on Linux). |
SYNC | Synchronous writes for files supporting DAX (direct memory access). Using this flag only works in combination with MapType::SHARED_VALIDATE, otherwise it is ignored. For supported files, if suitable CPU instructions are used for writing to memory, it is guaranteed that the state of the memory is also found on the underlying persistent device. |
UNINITIALIZED | Don't clear anonymous pages. Only possible if CONFIG_MMAP_ALLOW_UNITIALIZED is set in the kernel |
Definition at line 51 of file mman.hxx.
|
strong |
The basic type of a memory mapping to be created.
Enumerator | |
---|---|
SHARED | Creates a shared memory mapping that can be shared with other processes. |
SHARED_VALIDATE | Same as SHARED but the MapFlags will be validated for unknown flags. |
PRIVATE | A private copy-on-write mapping that isn't shared with other processes. |
Definition at line 23 of file mman.hxx.
|
strong |
Extra flags used with mem::protect().
Enumerator | |
---|---|
GROWSUP | Apply protection settings up to the end of mapping that grows upwards. |
GROWSDOWN | Apply protection settings down to the beginning of mapping that grows downwards. |
|
strong |
Flags used with cosmos::mem::remap().
Enumerator | |
---|---|
MAYMOVE | allow to move the mapping to a new starting address. |
FIXED | request the mapping to be placed at a fixed supplied address, similar to MapFlag::FIXED; this requires the MAYMOVE flag to be set as well. |
DONTUNMAP | used only together with MAYMOVE; keep the original mapping available for special memory algorithms like |
|
strong |
Flags used with cosmos::mem::sync().
At least one of ASYNC or SYNC, but not both, must be present when passing flags to sync().
Definition at line 281 of file mman.hxx.
COSMOS_API void cosmos::mem::lock | ( | void * | addr, |
const size_t | length, | ||
const LockFlags | flags = {} ) |
Lock pages in memory, preventing memory from being paged to the swap area.
The given address range will be pre-faulted upon return (unless LockFlag::LOCK_ON_FAULT is set in flags
) and will be prevented from being swapped out. The main applications for this feature are real-time requirements or security considerations (preventing sensitive data from ending up on disk).
Memory locks do not stack, i.e. calling lock() multiple times doesn't change the state, a single unlock() will remove the lock.
Memory locks are not maintained across process forks or execve().
addr
down to the nearest page size.On error an ApiError is thrown with one of the following Errno values:
addr + length
resulted in an overflow.flags
have been encountered.Definition at line 53 of file mman.cxx.
COSMOS_API void cosmos::mem::lock_all | ( | const LockAllFlags | flags | ) |
Locks all current and/or future pages in memory.
Depending on the settings in flags
this call locks all currently loaded memory pages in memory, as well as mappings possibly created in the future. The locking logic is the same as described in cosmos::mem::lock().
On error an ApiError with one of the following Errno values is thrown:
flags
encountered, or LockAllFlag::ONFAULT was specified without any of the other flags. Definition at line 65 of file mman.cxx.
COSMOS_API void * cosmos::mem::map | ( | const size_t | length, |
const MapSettings & | settings ) |
Request a memory mapping of the given length using the provided settings.
On error an ApiError is thrown with one of the following Errno values:
settings.fd
is not a regular file, or the OpenMode of the file does not match (lacking read or write permission, depending on settings.type
and settings.access
.settings.fd
is not a valid file descriptor and settings.flags
does not contain MapFlag::ANONYMOUS.settings.offset
, settings.addr
or length
are offending.settings.type
is invalid (shouldn't happen if a proper MapType constant is used)length
is zero.settings.fd
does not support memory mapping.settings.addr
exceeds the virtual address space of the CPU.unsigned long
can hold.settings.access
has AccessFlag::EXEC set, but the file to be mapped resides on a filesystem mounted no-exec.Definition at line 9 of file mman.cxx.
COSMOS_API void cosmos::mem::protect | ( | void * | addr, |
const size_t | length, | ||
const AccessFlags | flags, | ||
const ProtectFlags | extra = {} ) |
Change memory protection settings of an existing mapping.
Failure to change the memory protection causes an ApiError to be thrown, with one of the following Errno values:
addr
is not a valid pointer or not page alignedextra
has both ProtectFlag::GROWSUP and ProtectFlag::GROWSDOWN set.flags
encountered.Definition at line 77 of file mman.cxx.
COSMOS_API void * cosmos::mem::remap | ( | void * | old_addr, |
const size_t | old_size, | ||
const size_t | new_size, | ||
const RemapFlags | flags = {}, | ||
std::optional< void * > | new_addr = {} ) |
Expand or shrink an existing memory mapping.
old_addr
needs to be page aligned. If old_size
is zero, and old_addr
refers to a memory mapping of MapType::SHARED, then this call will create a new mapping of the same pages. new_addr
behaves similar as the addr
argument in mem::map(), depending on the RemapFlag::FIXED flag.
If it is not possible to change the size of the mapping, then an error is thrown, except if RemapFlag::MAYMOVE is specified, in which case a new address may be returned for the mapping.
Resizing or moving a mapping that is currently locked will cause the new mapping also to be locked.
Failure to change the mapping will cause an ApiError to be thrown with one of the following Errno values:
RLIMIT_MEMLOCK
resource limit.old_addr
, old_addr + old_size
] is not consisting of all valid addresses for the calling process. This can also happen if all addresses are valid, but mixed types of memory mappings are found in this range.old_addr
is not page aligned.flags
have been specified.new_size
was zero.new_size
or new_addr
are invalid.new_addr
, new_addr
+ new_size
] overlap with the range of [old_addr
, old_addr
+ old_size
].flags
without RemapFlag::MAYMOVE being set as well.flags
, but one or more pages in the old range are not private-anonymous.flags
, but new_size
is not equal to old_size
.old_size
is zero, but old_addr
does not refer to a mapping of MapType::SHARED.old_size
is zero, but RemapFlag::MAYMOVE is not set in flags
.flags
.flags
, which would cause a new mapping to be created that would exceed the available virtual memory, or the maximum number of allowed mappings. Definition at line 37 of file mman.cxx.
COSMOS_API void cosmos::mem::sync | ( | void * | addr, |
const size_t | length, | ||
const SyncFlags | flags = SyncFlags{SyncFlag::SYNC} ) |
Synchronize changes in a memory mapping with the file backing it.
When writing changes to a memory mapping, then it is undefined when these changes will actually be written back to the file backing the mapping. Only after unmapping, the changes are guaranteed to be written back.
Using this call, changes can be written back explicitly.
On error an ApiError with one of the following Errno values is thrown:
flags
, but a memory lock exists for the specified address range.addr
is not on a page boundary, or bad values have been found in flags
, or both SyncFlag::SYNC and SyncFlag::ASYNC have been encountered.Definition at line 31 of file mman.cxx.
COSMOS_API void cosmos::mem::unlock | ( | void * | addr, |
const size_t | length ) |
Unlock previously locked pages.
After successful return from this call the affected address range can be swapped out by the kernel again.
On error an ApiError is thrown with one of the following Errno values:
Errno::INVALID_ARG, Errno::AGAIN, Errno::NO_MEMORY or Errno::PERMISSION, as documented in cosmos::mem::lock().
Definition at line 59 of file mman.cxx.
COSMOS_API void cosmos::mem::unlock_all | ( | ) |
COSMOS_API void cosmos::mem::unmap | ( | void * | addr, |
const size_t | length ) |
Unmap an existing mapping at the given address and of the given length.
As with mapping memory, unmapping memory can also cause an ApiError to be thrown, with one of the following Errno values:
addr
or length
are offendingDefinition at line 25 of file mman.cxx.