-
Notifications
You must be signed in to change notification settings - Fork 92
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Only ~16 bits of library ASLR entropy when vm.mmap_rnd_bits
is 18
#323
Comments
another unlikely possibility: 2 bits are wasted on randomizing 4k aligned addresses instead of 16k no, the section alignment is anyway 64k and the addresses are aligned to that |
I can replicate this result. Just as a sanity check of the experimental setup, I tried replacing the subprocess with Here's my optimised version of the experiment, which spawns fewer subprocesses and therefore runs about 5x faster: (not hugely important but it makes testing things faster) import multiprocessing
import subprocess
# Number of times we run the experiment
ITERATIONS: int = 1000
# Number of addresses we collect per experiment
SAMPLE_SIZE: int = 603
def experiment(sample_size: int) -> bool:
"""Grabs the libc base address sample_size times, returns whether any of them are duplicates"""
seen: set[int] = set()
for _ in range(sample_size):
lines = subprocess.run(
["cat", "/proc/self/maps"],
capture_output=True,
).stdout.decode().split("\n")
addr = [int(l.split("-")[0], 16) for l in lines if "libc" in l][0]
if addr in seen:
return True
seen.add(addr)
return False
with multiprocessing.Pool(multiprocessing.cpu_count() // 2) as pool:
print(sum(pool.map(experiment, [SAMPLE_SIZE] * ITERATIONS)) / ITERATIONS) I noticed checking the addresses of By observation, Unproven hypothesis: if you have a 16k kernel with 64k-aligned userland binaries, you throw away 2 bits of entropy relative to whatever |
Thanks for the feedback. I tried running the script on some other aarch64 platforms.
The takeaway here is that systems with 4k pages have this same problem, but even worse by another ~2 bits. I'm going to keep going back in time on the rpi to see if this has always been the behavior, or if this is a regression. |
@kenballus did the binaries you tested against on those systems have 4k-aligned segments, too? If my hypothesis is correct, loading a 64k-aligned binary on a 4k kernel with vm.mmap_rnd_bits=18 will result in only ~14 bits of entropy. (If you want a test binary, |
(This might be expected behavior. Apologies in advance.)
I saw this blog post about ASLR behaving weirdly on recent x86 Linux kernels, and was curious if Asahi was also affected.
By default,$2^{18}$ equally-likely possible values for the base address of libc.
vm.mmap_rnd_bits
is 18 on Asahi Fedora 40. Thus, we should expect that there are something likeUnder that assumption, if we collect 603 libc base addresses, we should expect to see about a 50% chance of there being at least one duplicate among them1.
I wrote this little Python script to collect a bunch of libc base addresses and check for duplicates, then report how often it hits at least one duplicate:
The math would lead us to expect this script to spit out something around
0.5
, but it reports something around0.9
on my M1 MBA. (Feel free to run it for yourself; only takes a minute or two). This would suggest that the distribution of libc base addresses has less than 18 bits of entropy.If you change
SAMPLE_SIZE
to302
, then the script will report a value close to0.5
. By the same math as before, this would suggest about 16 bits of effective entropy, instead of 18.I'm puzzled as to why we seem to observe only 25% as many possible addresses as we should. Some potential reasons:
I'd really appreciate it if anyone has any ideas about the cause of this, or could suggest a more appropriate place to bring this up.
Thanks!
Footnotes
This can be found by setting $n=2^{18}$ and solving for $k$ in this. You can check your work here. ↩
The text was updated successfully, but these errors were encountered: