2020-04-24 23:36:52 +00:00
|
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
let
|
|
|
|
|
cfg = config.environment.memoryAllocator;
|
|
|
|
|
|
|
|
|
|
# The set of alternative malloc(3) providers.
|
|
|
|
|
providers = {
|
|
|
|
|
graphene-hardened = {
|
|
|
|
|
libPath = "${pkgs.graphene-hardened-malloc}/lib/libhardened_malloc.so";
|
|
|
|
|
description = ''
|
2024-05-15 15:35:15 +00:00
|
|
|
|
Hardened memory allocator coming from GrapheneOS project.
|
|
|
|
|
The default configuration template has all normal optional security
|
|
|
|
|
features enabled and is quite aggressive in terms of sacrificing
|
|
|
|
|
performance and memory usage for security.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
graphene-hardened-light = {
|
|
|
|
|
libPath = "${pkgs.graphene-hardened-malloc}/lib/libhardened_malloc-light.so";
|
|
|
|
|
description = ''
|
|
|
|
|
Hardened memory allocator coming from GrapheneOS project.
|
|
|
|
|
The light configuration template disables the slab quarantines,
|
|
|
|
|
write after free check, slot randomization and raises the guard
|
|
|
|
|
slab interval from 1 to 8 but leaves zero-on-free and slab canaries enabled.
|
|
|
|
|
The light configuration has solid performance and memory usage while still
|
|
|
|
|
being far more secure than mainstream allocators with much better security
|
|
|
|
|
properties.
|
2020-04-24 23:36:52 +00:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
jemalloc = {
|
|
|
|
|
libPath = "${pkgs.jemalloc}/lib/libjemalloc.so";
|
|
|
|
|
description = ''
|
|
|
|
|
A general purpose allocator that emphasizes fragmentation avoidance
|
|
|
|
|
and scalable concurrency support.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2022-01-19 23:45:15 +00:00
|
|
|
|
scudo = let
|
|
|
|
|
platformMap = {
|
|
|
|
|
aarch64-linux = "aarch64";
|
|
|
|
|
x86_64-linux = "x86_64";
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
systemPlatform = platformMap.${pkgs.stdenv.hostPlatform.system} or (throw "scudo not supported on ${pkgs.stdenv.hostPlatform.system}");
|
|
|
|
|
in {
|
2023-05-24 13:37:59 +00:00
|
|
|
|
libPath = "${pkgs.llvmPackages_14.compiler-rt}/lib/linux/libclang_rt.scudo-${systemPlatform}.so";
|
2020-04-24 23:36:52 +00:00
|
|
|
|
description = ''
|
|
|
|
|
A user-mode allocator based on LLVM Sanitizer’s CombinedAllocator,
|
|
|
|
|
which aims at providing additional mitigations against heap based
|
|
|
|
|
vulnerabilities, while maintaining good performance.
|
|
|
|
|
'';
|
|
|
|
|
};
|
2021-09-22 15:38:15 +00:00
|
|
|
|
|
|
|
|
|
mimalloc = {
|
|
|
|
|
libPath = "${pkgs.mimalloc}/lib/libmimalloc.so";
|
|
|
|
|
description = ''
|
|
|
|
|
A compact and fast general purpose allocator, which may
|
|
|
|
|
optionally be built with mitigations against various heap
|
|
|
|
|
vulnerabilities.
|
|
|
|
|
'';
|
|
|
|
|
};
|
2020-04-24 23:36:52 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
providerConf = providers.${cfg.provider};
|
|
|
|
|
|
|
|
|
|
# An output that contains only the shared library, to avoid
|
|
|
|
|
# needlessly bloating the system closure
|
|
|
|
|
mallocLib = pkgs.runCommand "malloc-provider-${cfg.provider}"
|
|
|
|
|
rec {
|
|
|
|
|
preferLocalBuild = true;
|
|
|
|
|
allowSubstitutes = false;
|
|
|
|
|
origLibPath = providerConf.libPath;
|
|
|
|
|
libName = baseNameOf origLibPath;
|
|
|
|
|
}
|
|
|
|
|
''
|
|
|
|
|
mkdir -p $out/lib
|
|
|
|
|
cp -L $origLibPath $out/lib/$libName
|
|
|
|
|
'';
|
|
|
|
|
|
|
|
|
|
# The full path to the selected provider shlib.
|
|
|
|
|
providerLibPath = "${mallocLib}/lib/${mallocLib.libName}";
|
|
|
|
|
in
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
meta = {
|
2024-09-19 14:19:46 +00:00
|
|
|
|
maintainers = [ lib.maintainers.joachifm ];
|
2020-04-24 23:36:52 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
options = {
|
2024-09-19 14:19:46 +00:00
|
|
|
|
environment.memoryAllocator.provider = lib.mkOption {
|
|
|
|
|
type = lib.types.enum ([ "libc" ] ++ lib.attrNames providers);
|
2020-04-24 23:36:52 +00:00
|
|
|
|
default = "libc";
|
2024-04-21 15:54:59 +00:00
|
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
|
The system-wide memory allocator.
|
|
|
|
|
|
|
|
|
|
Briefly, the system-wide memory allocator providers are:
|
2022-09-09 14:08:57 +00:00
|
|
|
|
|
|
|
|
|
- `libc`: the standard allocator provided by libc
|
2024-09-19 14:19:46 +00:00
|
|
|
|
${lib.concatStringsSep "\n" (lib.mapAttrsToList
|
|
|
|
|
(name: value: "- `${name}`: ${lib.replaceStrings [ "\n" ] [ " " ] value.description}")
|
2020-04-24 23:36:52 +00:00
|
|
|
|
providers)}
|
|
|
|
|
|
2022-09-09 14:08:57 +00:00
|
|
|
|
::: {.warning}
|
2020-04-24 23:36:52 +00:00
|
|
|
|
Selecting an alternative allocator (i.e., anything other than
|
2022-09-09 14:08:57 +00:00
|
|
|
|
`libc`) may result in instability, data loss,
|
2020-04-24 23:36:52 +00:00
|
|
|
|
and/or service failure.
|
2022-09-09 14:08:57 +00:00
|
|
|
|
:::
|
2020-04-24 23:36:52 +00:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
2024-09-19 14:19:46 +00:00
|
|
|
|
config = lib.mkIf (cfg.provider != "libc") {
|
2020-04-24 23:36:52 +00:00
|
|
|
|
environment.etc."ld-nix.so.preload".text = ''
|
|
|
|
|
${providerLibPath}
|
|
|
|
|
'';
|
2021-05-20 23:08:51 +00:00
|
|
|
|
security.apparmor.includes = {
|
|
|
|
|
"abstractions/base" = ''
|
|
|
|
|
r /etc/ld-nix.so.preload,
|
|
|
|
|
r ${config.environment.etc."ld-nix.so.preload".source},
|
2021-09-22 15:38:15 +00:00
|
|
|
|
include "${pkgs.apparmorRulesFromClosure {
|
|
|
|
|
name = "mallocLib";
|
|
|
|
|
baseRules = ["mr $path/lib/**.so*"];
|
|
|
|
|
} [ mallocLib ] }"
|
2021-05-20 23:08:51 +00:00
|
|
|
|
'';
|
|
|
|
|
};
|
2020-04-24 23:36:52 +00:00
|
|
|
|
};
|
|
|
|
|
}
|