| // SPDX-License-Identifier: GPL-2.0 |
| /* |
| * Copyright (C) 2022-2024 Jason A. Donenfeld <Jason@zx2c4.com>. All Rights Reserved. |
| */ |
| |
| #include <assert.h> |
| #include <pthread.h> |
| #include <stdint.h> |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <time.h> |
| #include <unistd.h> |
| #include <signal.h> |
| #include <sys/auxv.h> |
| #include <sys/mman.h> |
| #include <sys/random.h> |
| #include <sys/syscall.h> |
| #include <sys/types.h> |
| #include <linux/random.h> |
| |
| #include "../kselftest.h" |
| #include "parse_vdso.h" |
| |
| #ifndef timespecsub |
| #define timespecsub(tsp, usp, vsp) \ |
| do { \ |
| (vsp)->tv_sec = (tsp)->tv_sec - (usp)->tv_sec; \ |
| (vsp)->tv_nsec = (tsp)->tv_nsec - (usp)->tv_nsec; \ |
| if ((vsp)->tv_nsec < 0) { \ |
| (vsp)->tv_sec--; \ |
| (vsp)->tv_nsec += 1000000000L; \ |
| } \ |
| } while (0) |
| #endif |
| |
| static struct { |
| pthread_mutex_t lock; |
| void **states; |
| size_t len, cap; |
| } grnd_allocator = { |
| .lock = PTHREAD_MUTEX_INITIALIZER |
| }; |
| |
| static struct { |
| ssize_t(*fn)(void *, size_t, unsigned long, void *, size_t); |
| pthread_key_t key; |
| pthread_once_t initialized; |
| struct vgetrandom_opaque_params params; |
| } grnd_ctx = { |
| .initialized = PTHREAD_ONCE_INIT |
| }; |
| |
| static void *vgetrandom_get_state(void) |
| { |
| void *state = NULL; |
| |
| pthread_mutex_lock(&grnd_allocator.lock); |
| if (!grnd_allocator.len) { |
| size_t page_size = getpagesize(); |
| size_t new_cap; |
| size_t alloc_size, num = sysconf(_SC_NPROCESSORS_ONLN); /* Just a decent heuristic. */ |
| void *new_block, *new_states; |
| |
| alloc_size = (num * grnd_ctx.params.size_of_opaque_state + page_size - 1) & (~(page_size - 1)); |
| num = (page_size / grnd_ctx.params.size_of_opaque_state) * (alloc_size / page_size); |
| new_block = mmap(0, alloc_size, grnd_ctx.params.mmap_prot, grnd_ctx.params.mmap_flags, -1, 0); |
| if (new_block == MAP_FAILED) |
| goto out; |
| |
| new_cap = grnd_allocator.cap + num; |
| new_states = reallocarray(grnd_allocator.states, new_cap, sizeof(*grnd_allocator.states)); |
| if (!new_states) |
| goto unmap; |
| grnd_allocator.cap = new_cap; |
| grnd_allocator.states = new_states; |
| |
| for (size_t i = 0; i < num; ++i) { |
| if (((uintptr_t)new_block & (page_size - 1)) + grnd_ctx.params.size_of_opaque_state > page_size) |
| new_block = (void *)(((uintptr_t)new_block + page_size - 1) & (~(page_size - 1))); |
| grnd_allocator.states[i] = new_block; |
| new_block += grnd_ctx.params.size_of_opaque_state; |
| } |
| grnd_allocator.len = num; |
| goto success; |
| |
| unmap: |
| munmap(new_block, alloc_size); |
| goto out; |
| } |
| success: |
| state = grnd_allocator.states[--grnd_allocator.len]; |
| |
| out: |
| pthread_mutex_unlock(&grnd_allocator.lock); |
| return state; |
| } |
| |
| static void vgetrandom_put_state(void *state) |
| { |
| if (!state) |
| return; |
| pthread_mutex_lock(&grnd_allocator.lock); |
| grnd_allocator.states[grnd_allocator.len++] = state; |
| pthread_mutex_unlock(&grnd_allocator.lock); |
| } |
| |
| static void vgetrandom_init(void) |
| { |
| if (pthread_key_create(&grnd_ctx.key, vgetrandom_put_state) != 0) |
| return; |
| unsigned long sysinfo_ehdr = getauxval(AT_SYSINFO_EHDR); |
| if (!sysinfo_ehdr) { |
| printf("AT_SYSINFO_EHDR is not present!\n"); |
| exit(KSFT_SKIP); |
| } |
| vdso_init_from_sysinfo_ehdr(sysinfo_ehdr); |
| grnd_ctx.fn = (__typeof__(grnd_ctx.fn))vdso_sym("LINUX_2.6", "__vdso_getrandom"); |
| if (!grnd_ctx.fn) { |
| printf("__vdso_getrandom is missing!\n"); |
| exit(KSFT_FAIL); |
| } |
| if (grnd_ctx.fn(NULL, 0, 0, &grnd_ctx.params, ~0UL) != 0) { |
| printf("failed to fetch vgetrandom params!\n"); |
| exit(KSFT_FAIL); |
| } |
| } |
| |
| static ssize_t vgetrandom(void *buf, size_t len, unsigned long flags) |
| { |
| void *state; |
| |
| pthread_once(&grnd_ctx.initialized, vgetrandom_init); |
| state = pthread_getspecific(grnd_ctx.key); |
| if (!state) { |
| state = vgetrandom_get_state(); |
| if (pthread_setspecific(grnd_ctx.key, state) != 0) { |
| vgetrandom_put_state(state); |
| state = NULL; |
| } |
| if (!state) { |
| printf("vgetrandom_get_state failed!\n"); |
| exit(KSFT_FAIL); |
| } |
| } |
| return grnd_ctx.fn(buf, len, flags, state, grnd_ctx.params.size_of_opaque_state); |
| } |
| |
| enum { TRIALS = 25000000, THREADS = 256 }; |
| |
| static void *test_vdso_getrandom(void *) |
| { |
| for (size_t i = 0; i < TRIALS; ++i) { |
| unsigned int val; |
| ssize_t ret = vgetrandom(&val, sizeof(val), 0); |
| assert(ret == sizeof(val)); |
| } |
| return NULL; |
| } |
| |
| static void *test_libc_getrandom(void *) |
| { |
| for (size_t i = 0; i < TRIALS; ++i) { |
| unsigned int val; |
| ssize_t ret = getrandom(&val, sizeof(val), 0); |
| assert(ret == sizeof(val)); |
| } |
| return NULL; |
| } |
| |
| static void *test_syscall_getrandom(void *) |
| { |
| for (size_t i = 0; i < TRIALS; ++i) { |
| unsigned int val; |
| ssize_t ret = syscall(__NR_getrandom, &val, sizeof(val), 0); |
| assert(ret == sizeof(val)); |
| } |
| return NULL; |
| } |
| |
| static void bench_single(void) |
| { |
| struct timespec start, end, diff; |
| |
| clock_gettime(CLOCK_MONOTONIC, &start); |
| test_vdso_getrandom(NULL); |
| clock_gettime(CLOCK_MONOTONIC, &end); |
| timespecsub(&end, &start, &diff); |
| printf(" vdso: %u times in %lu.%09lu seconds\n", TRIALS, diff.tv_sec, diff.tv_nsec); |
| |
| clock_gettime(CLOCK_MONOTONIC, &start); |
| test_libc_getrandom(NULL); |
| clock_gettime(CLOCK_MONOTONIC, &end); |
| timespecsub(&end, &start, &diff); |
| printf(" libc: %u times in %lu.%09lu seconds\n", TRIALS, diff.tv_sec, diff.tv_nsec); |
| |
| clock_gettime(CLOCK_MONOTONIC, &start); |
| test_syscall_getrandom(NULL); |
| clock_gettime(CLOCK_MONOTONIC, &end); |
| timespecsub(&end, &start, &diff); |
| printf("syscall: %u times in %lu.%09lu seconds\n", TRIALS, diff.tv_sec, diff.tv_nsec); |
| } |
| |
| static void bench_multi(void) |
| { |
| struct timespec start, end, diff; |
| pthread_t threads[THREADS]; |
| |
| clock_gettime(CLOCK_MONOTONIC, &start); |
| for (size_t i = 0; i < THREADS; ++i) |
| assert(pthread_create(&threads[i], NULL, test_vdso_getrandom, NULL) == 0); |
| for (size_t i = 0; i < THREADS; ++i) |
| pthread_join(threads[i], NULL); |
| clock_gettime(CLOCK_MONOTONIC, &end); |
| timespecsub(&end, &start, &diff); |
| printf(" vdso: %u x %u times in %lu.%09lu seconds\n", TRIALS, THREADS, diff.tv_sec, diff.tv_nsec); |
| |
| clock_gettime(CLOCK_MONOTONIC, &start); |
| for (size_t i = 0; i < THREADS; ++i) |
| assert(pthread_create(&threads[i], NULL, test_libc_getrandom, NULL) == 0); |
| for (size_t i = 0; i < THREADS; ++i) |
| pthread_join(threads[i], NULL); |
| clock_gettime(CLOCK_MONOTONIC, &end); |
| timespecsub(&end, &start, &diff); |
| printf(" libc: %u x %u times in %lu.%09lu seconds\n", TRIALS, THREADS, diff.tv_sec, diff.tv_nsec); |
| |
| clock_gettime(CLOCK_MONOTONIC, &start); |
| for (size_t i = 0; i < THREADS; ++i) |
| assert(pthread_create(&threads[i], NULL, test_syscall_getrandom, NULL) == 0); |
| for (size_t i = 0; i < THREADS; ++i) |
| pthread_join(threads[i], NULL); |
| clock_gettime(CLOCK_MONOTONIC, &end); |
| timespecsub(&end, &start, &diff); |
| printf(" syscall: %u x %u times in %lu.%09lu seconds\n", TRIALS, THREADS, diff.tv_sec, diff.tv_nsec); |
| } |
| |
| static void fill(void) |
| { |
| uint8_t weird_size[323929]; |
| for (;;) |
| vgetrandom(weird_size, sizeof(weird_size), 0); |
| } |
| |
| static void kselftest(void) |
| { |
| uint8_t weird_size[1263]; |
| |
| ksft_print_header(); |
| ksft_set_plan(1); |
| |
| for (size_t i = 0; i < 1000; ++i) { |
| ssize_t ret = vgetrandom(weird_size, sizeof(weird_size), 0); |
| if (ret != sizeof(weird_size)) |
| exit(KSFT_FAIL); |
| } |
| |
| ksft_test_result_pass("getrandom: PASS\n"); |
| exit(KSFT_PASS); |
| } |
| |
| static void usage(const char *argv0) |
| { |
| fprintf(stderr, "Usage: %s [bench-single|bench-multi|fill]\n", argv0); |
| } |
| |
| int main(int argc, char *argv[]) |
| { |
| if (argc == 1) { |
| kselftest(); |
| return 0; |
| } |
| |
| if (argc != 2) { |
| usage(argv[0]); |
| return 1; |
| } |
| if (!strcmp(argv[1], "bench-single")) |
| bench_single(); |
| else if (!strcmp(argv[1], "bench-multi")) |
| bench_multi(); |
| else if (!strcmp(argv[1], "fill")) |
| fill(); |
| else { |
| usage(argv[0]); |
| return 1; |
| } |
| return 0; |
| } |