-
Notifications
You must be signed in to change notification settings - Fork 11.9k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[Support][Memory] Add memfd based fallback for strict W^X Linux systems
PaX's MPROTECT feature as well as SELinux's deny_execmem policy prevent a transition from a once writable memory mapping to an executable one. This obviously breaks JIT code generation. As both of these are Linux specific and the Linux kernel gained memfd support almost 10 years ago in v3.17, making use of it to implement a fallback seems like a viable option to get JIT code generation fixed for such systems. Implement a detour through a memfd for systems that are detected to deny the W<->X transition of memory mappings. For PaX this can be easily achieved by evaluating the per-process PaX flags, if access to /proc/self/status is available. For others, and as a fallback for PaX, a runtime test is done once. This enables such systems to make use of JIT code generation without completely abandoning their W^X policy. Signed-off-by: Mathias Krause <[email protected]>
- Loading branch information
1 parent
e4163c0
commit 3a22789
Showing
3 changed files
with
170 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,105 @@ | ||
//===- Unix/MemoryLinux.h - Linux specific Helper Fuctions ------*- C++ -*-===// | ||
// | ||
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | ||
// See https://llvm.org/LICENSE.txt for license information. | ||
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | ||
// | ||
//===----------------------------------------------------------------------===// | ||
// | ||
// This file defines Linux specific helper functions for memory management. | ||
// | ||
//===----------------------------------------------------------------------===// | ||
|
||
#ifndef LLVM_LIB_SUPPORT_UNIX_MEMORYLINUX_H | ||
#define LLVM_LIB_SUPPORT_UNIX_MEMORYLINUX_H | ||
|
||
#ifndef __linux__ | ||
#error Linux only support header! | ||
#endif | ||
|
||
#include "llvm/Support/Process.h" | ||
|
||
#include <stdio.h> | ||
#include <stdlib.h> | ||
#include <string.h> | ||
#include <unistd.h> | ||
|
||
#include <sys/mman.h> | ||
#include <sys/syscall.h> | ||
|
||
#ifndef MFD_CLOEXEC | ||
#define MFD_CLOEXEC 1U | ||
#endif | ||
|
||
namespace llvm { | ||
namespace sys { | ||
namespace { | ||
|
||
static inline bool isPermissionError(int err) { | ||
// PaX uses EPERM, SELinux uses EACCES | ||
return err == EPERM || err == EACCES; | ||
} | ||
|
||
// FIXME: Make this either more low-level C'ish or C++'ish | ||
static inline bool execProtChangeNeedsNewMapping() { | ||
static int status = -1; | ||
|
||
if (status != -1) | ||
return status; | ||
|
||
// Try to get the status from /proc/self/status, looking for PaX flags. | ||
FILE *f = fopen("/proc/self/status", "re"); | ||
if (f) { | ||
char *buf = NULL; | ||
size_t len; | ||
|
||
while (getline(&buf, &len, f) != -1) { | ||
if (strncmp(buf, "PaX:", 4)) | ||
continue; | ||
|
||
// Look for 'm', indicating PaX MPROTECT is disabled. | ||
status = !strchr(buf + 4, 'm'); | ||
break; | ||
} | ||
|
||
fclose(f); | ||
free(buf); | ||
|
||
if (status != -1) | ||
return status; | ||
} | ||
|
||
// Create a temporary writable mapping and try to make it executable. If | ||
// this fails, test 'errno' to ensure it failed because we were not allowed | ||
// to create such a mapping and not because of some transient error. | ||
size_t size = Process::getPageSizeEstimate(); | ||
void *addr = ::mmap(NULL, size, PROT_READ | PROT_WRITE, | ||
MAP_PRIVATE | MAP_ANONYMOUS, -1, 0); | ||
if (addr == MAP_FAILED) { | ||
// Must be low on memory or have too many mappings already, not much we can | ||
// do here. | ||
status = 0; | ||
} else { | ||
if (::mprotect(addr, size, PROT_READ | PROT_EXEC) < 0) | ||
status = isPermissionError(errno); | ||
else | ||
status = 0; | ||
::munmap(addr, size); | ||
} | ||
|
||
return status; | ||
} | ||
|
||
static inline int memfd_create(const char *name, int flags) { | ||
#ifdef SYS_memfd_create | ||
return syscall(SYS_memfd_create, name, flags); | ||
#else | ||
return -1; | ||
#endif | ||
} | ||
|
||
} // anonymous namespace | ||
} // namespace sys | ||
} // namespace llvm | ||
|
||
#endif |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters