/* Copyright (c) 2003-2004, Roger Dingledine * Copyright (c) 2004-2006, Roger Dingledine, Nick Mathewson. * Copyright (c) 2007-2020, The Tor Project, Inc. */ /* See LICENSE for licensing information */ /** * \file map_anon.c * \brief Manage anonymous mappings. **/ #include "orconfig.h" #include "lib/malloc/map_anon.h" #include "lib/malloc/malloc.h" #include "lib/err/torerr.h" #ifdef HAVE_SYS_MMAN_H #include #endif #ifdef HAVE_SYS_TYPES_H #include #endif #ifdef HAVE_MACH_VM_INHERIT_H #include #endif #ifdef _WIN32 #include #endif #include #include /** * Macro to get the high bytes of a size_t, if there are high bytes. * Windows needs this; other operating systems define a size_t that does * what it should. */ #if SIZEOF_SIZE_T > 4 #define HIGH_SIZE_T_BYTES(sz) ((sz) >> 32) #else #define HIGH_SIZE_T_BYTES(sz) (0) #endif /* Here we define a MINHERIT macro that is minherit() or madvise(), depending * on what we actually want. * * If there's a flag that sets pages to zero after fork, we define FLAG_ZERO * to be that flag. If there's a flag unmaps pages after fork, we define * FLAG_NOINHERIT to be that flag. */ #if defined(HAVE_MINHERIT) #define MINHERIT minherit #ifdef INHERIT_ZERO #define FLAG_ZERO INHERIT_ZERO #elif defined(MAP_INHERIT_ZERO) #define FLAG_ZERO MAP_INHERIT_ZERO #endif #ifdef INHERIT_NONE #define FLAG_NOINHERIT INHERIT_NONE #elif defined(VM_INHERIT_NONE) #define FLAG_NOINHERIT VM_INHERIT_NONE #elif defined(MAP_INHERIT_NONE) #define FLAG_NOINHERIT MAP_INHERIT_NONE #endif /* defined(INHERIT_NONE) || ... */ #elif defined(HAVE_MADVISE) #define MINHERIT madvise #ifdef MADV_WIPEONFORK #define FLAG_ZERO MADV_WIPEONFORK #endif #ifdef MADV_DONTFORK #define FLAG_NOINHERIT MADV_DONTFORK #endif #endif /* defined(HAVE_MINHERIT) || ... */ #if defined(HAVE_MINHERIT) && !defined(FLAG_ZERO) && !defined(FLAG_NOINHERIT) #warning "minherit() is defined, but we couldn't find the right flag for it." #warning "This is probably a bug in Tor's support for this platform." #endif /** * Helper: try to prevent the sz bytes at mem from being swapped * to disk. Return 0 on success or if the facility is not available on this * OS; return -1 on failure. */ static int lock_mem(void *mem, size_t sz) { #ifdef _WIN32 return VirtualLock(mem, sz) ? 0 : -1; #elif defined(HAVE_MLOCK) return mlock(mem, sz); #else (void) mem; (void) sz; return 0; #endif /* defined(_WIN32) || ... */ } /** * Helper: try to prevent the sz bytes at mem from appearing in * a core dump. Return 0 on success or if the facility is not available on * this OS; return -1 on failure. */ static int nodump_mem(void *mem, size_t sz) { #if defined(MADV_DONTDUMP) int rv = madvise(mem, sz, MADV_DONTDUMP); if (rv == 0) { return 0; } else if (errno == ENOSYS || errno == EINVAL) { return 0; // syscall not supported, or flag not supported. } else { tor_log_err_sigsafe("Unexpected error from madvise: ", strerror(errno), NULL); return -1; } #else /* !defined(MADV_DONTDUMP) */ (void) mem; (void) sz; return 0; #endif /* defined(MADV_DONTDUMP) */ } /** * Helper: try to prevent the sz bytes at mem from being * accessible in child processes -- ideally by having them set to 0 after a * fork, and if that doesn't work, by having them unmapped after a fork. * Return 0 on success or if the facility is not available on this OS; return * -1 on failure. * * If we successfully make the memory uninheritable, adjust the value of * *inherit_result_out. */ static int noinherit_mem(void *mem, size_t sz, inherit_res_t *inherit_result_out) { #ifdef FLAG_ZERO int r = MINHERIT(mem, sz, FLAG_ZERO); if (r == 0) { *inherit_result_out = INHERIT_RES_ZERO; return 0; } #endif /* defined(FLAG_ZERO) */ #ifdef FLAG_NOINHERIT int r2 = MINHERIT(mem, sz, FLAG_NOINHERIT); if (r2 == 0) { *inherit_result_out = INHERIT_RES_DROP; return 0; } #endif /* defined(FLAG_NOINHERIT) */ #if defined(FLAG_ZERO) || defined(FLAG_NOINHERIT) /* At least one operation was tried, and neither succeeded. */ if (errno == ENOSYS || errno == EINVAL) { /* Syscall not supported, or flag not supported. */ return 0; } else { tor_log_err_sigsafe("Unexpected error from minherit: ", strerror(errno), NULL); return -1; } #else /* !(defined(FLAG_ZERO) || defined(FLAG_NOINHERIT)) */ (void)inherit_result_out; (void)mem; (void)sz; return 0; #endif /* defined(FLAG_ZERO) || defined(FLAG_NOINHERIT) */ } /** * Return a new anonymous memory mapping that holds sz bytes. * * Memory mappings are unlike the results from malloc() in that they are * handled separately by the operating system, and as such can have different * kernel-level flags set on them. * * The "flags" argument may be zero or more of ANONMAP_PRIVATE and * ANONMAP_NOINHERIT. * * Memory returned from this function must be released with * tor_munmap_anonymous(). * * If inherit_result_out is non-NULL, set it to one of * INHERIT_RES_KEEP, INHERIT_RES_DROP, or INHERIT_RES_ZERO, depending on the * properties of the returned memory. * * [Note: OS people use the word "anonymous" here to mean that the memory * isn't associated with any file. This has *nothing* to do with the kind of * anonymity that Tor is trying to provide.] */ void * tor_mmap_anonymous(size_t sz, unsigned flags, inherit_res_t *inherit_result_out) { void *ptr; inherit_res_t itmp=0; if (inherit_result_out == NULL) { inherit_result_out = &itmp; } *inherit_result_out = INHERIT_RES_KEEP; #if defined(_WIN32) HANDLE mapping = CreateFileMapping(INVALID_HANDLE_VALUE, NULL, /*attributes*/ PAGE_READWRITE, HIGH_SIZE_T_BYTES(sz), sz & 0xffffffff, NULL /* name */); raw_assert(mapping != NULL); ptr = MapViewOfFile(mapping, FILE_MAP_WRITE, 0, 0, /* Offset */ 0 /* Extend to end of mapping */); raw_assert(ptr); CloseHandle(mapping); /* mapped view holds a reference */ #elif defined(HAVE_SYS_MMAN_H) ptr = mmap(NULL, sz, PROT_READ|PROT_WRITE, MAP_ANON|MAP_PRIVATE, -1, 0); raw_assert(ptr != MAP_FAILED); raw_assert(ptr != NULL); #else ptr = tor_malloc_zero(sz); #endif /* defined(_WIN32) || ... */ if (flags & ANONMAP_PRIVATE) { int lock_result = lock_mem(ptr, sz); raw_assert(lock_result == 0); int nodump_result = nodump_mem(ptr, sz); raw_assert(nodump_result == 0); } if (flags & ANONMAP_NOINHERIT) { int noinherit_result = noinherit_mem(ptr, sz, inherit_result_out); raw_assert(noinherit_result == 0); } return ptr; } /** * Release sz bytes of memory that were previously mapped at * mapping by tor_mmap_anonymous(). **/ void tor_munmap_anonymous(void *mapping, size_t sz) { if (!mapping) return; #if defined(_WIN32) (void)sz; UnmapViewOfFile(mapping); #elif defined(HAVE_SYS_MMAN_H) munmap(mapping, sz); #else (void)sz; tor_free(mapping); #endif /* defined(_WIN32) || ... */ }