jemalloc/test/unit/arenas_management.c
Slobodan Predolac b92420d309 Replace arena->tcache_ql with cache_bin_array_descriptor_ql walks
Drop the duplicate arena->tcache_ql; stats merging walks the
cache_bin_array_descriptor_ql directly. Rename the protecting mutex
from tcache_ql_mtx to cache_bin_array_descriptor_ql_mtx to match. Add
an assertion in test_thread_migrate_arena that the dissociate-time
flush zeros cache_bin->tstats.nrequests.
2026-05-13 17:50:41 -04:00

337 lines
10 KiB
C

#include "test/jemalloc_test.h"
#include "test/arena_util.h"
#include "jemalloc/internal/arenas_management.h"
/*
* a0* stuff tested in test/unit/a0, fork related interface in
* test/unit/fork. arena_choose_hard is exercised indirectly on each
* thread's first allocation.
*/
TEST_BEGIN(test_narenas_total_get_consistent) {
unsigned total = narenas_total_get();
expect_u_ge(total, narenas_auto,
"narenas_total (%u) must be >= narenas_auto (%u)", total,
narenas_auto);
expect_u_gt(total, 0, "narenas_total must be positive after init");
}
TEST_END
TEST_BEGIN(test_narenas_total_set_roundtrip) {
unsigned saved = narenas_total_get();
narenas_total_set(saved);
expect_u_eq(narenas_total_get(), saved,
"narenas_total_set/get round-trip mismatch");
}
TEST_END
TEST_BEGIN(test_narenas_auto_set_roundtrip) {
unsigned saved = narenas_auto;
narenas_auto_set(saved);
expect_u_eq(
narenas_auto, saved, "narenas_auto_set round-trip mismatch");
}
TEST_END
TEST_BEGIN(test_manual_arena_base_set_roundtrip) {
unsigned saved = manual_arena_base;
manual_arena_base_set(saved);
expect_u_eq(manual_arena_base, saved,
"manual_arena_base_set round-trip mismatch");
}
TEST_END
TEST_BEGIN(test_arena_set_roundtrip) {
tsdn_t *tsdn = tsd_tsdn(tsd_fetch());
arena_t *a0 = arena_get(tsdn, 0, false);
expect_ptr_not_null(a0, "arena 0 should exist after init");
arena_set(0, a0);
expect_ptr_eq(arena_get(tsdn, 0, false), a0,
"arena_set round-trip mismatch for ind=0");
}
TEST_END
TEST_BEGIN(test_arena_init_creates_arena) {
tsdn_t *tsdn = tsd_tsdn(tsd_fetch());
unsigned before = narenas_total_get();
expect_u_lt(before, MALLOCX_ARENA_LIMIT,
"Cannot create arena: at MALLOCX_ARENA_LIMIT");
arena_t *arena = arena_init(tsdn, before, &arena_config_default);
expect_ptr_not_null(arena, "arena_init failed for ind=%u", before);
expect_u_eq(narenas_total_get(), before + 1,
"narenas_total did not increment after arena_init");
expect_ptr_eq(arena_get(tsdn, before, false), arena,
"arena_get does not return the freshly-initialized arena");
}
TEST_END
TEST_BEGIN(test_arena_init_idempotent_auto) {
test_skip_if(narenas_auto < 1);
tsdn_t *tsdn = tsd_tsdn(tsd_fetch());
arena_t *a0 = arena_get(tsdn, 0, false);
expect_ptr_not_null(a0, "arena 0 should exist after init");
unsigned before = narenas_total_get();
arena_t *again = arena_init(tsdn, 0, &arena_config_default);
expect_ptr_eq(again, a0,
"arena_init for an existing auto arena should return same pointer");
expect_u_eq(narenas_total_get(), before,
"narenas_total should not change for idempotent arena_init");
}
TEST_END
/*
* test_thread_migrate_arena spawns one worker thread, binds it to arena1,
* then migrates it to arena2 via the thread-level helper. We verify the
* nthreads counters on both arenas, that the migrating thread's tsd_arena
* was re-pointed, and that its tcache (if active) was reassociated.
*/
static unsigned migrate_a1_ind;
static unsigned migrate_a2_ind;
static atomic_b_t migrate_done;
static atomic_b_t migrate_go_exit;
static void *
migrate_worker(void *unused) {
unsigned old_ind;
size_t sz = sizeof(unsigned);
expect_d_eq(mallctl("thread.arena", (void *)&old_ind, &sz,
(void *)&migrate_a1_ind, sizeof(unsigned)),
0, "thread.arena bind failed");
tsd_t *tsd = tsd_fetch();
tsdn_t *tsdn = tsd_tsdn(tsd);
arena_t *a1 = arena_get(tsdn, migrate_a1_ind, false);
arena_t *a2 = arena_get(tsdn, migrate_a2_ind, false);
expect_ptr_not_null(a1, "arena1 should exist");
expect_ptr_not_null(a2, "arena2 should exist");
/*
* Populate cache_bin tstats with explicit small allocs so the
* migrate's flush has something to merge.
*/
szind_t test_binind = sz_size2index(8);
if (config_stats && tcache_available(tsd)) {
void *p[16];
for (size_t i = 0; i < ARRAY_SIZE(p); i++) {
p[i] = malloc(8);
}
for (size_t i = 0; i < ARRAY_SIZE(p); i++) {
free(p[i]);
}
cache_bin_t *cb = &tsd_tcachep_get(tsd)->bins[test_binind];
expect_u64_gt(cb->tstats.nrequests, 0,
"Small allocs should accumulate cache_bin tstats");
}
thread_migrate_arena(tsd, a1, a2);
expect_ptr_eq(
tsd_arena_get(tsd), a2, "tsd_arena was not updated to newarena");
if (tcache_available(tsd)) {
expect_ptr_eq(tsd_tcache_slowp_get(tsd)->arena, a2,
"tcache should be reassociated with newarena");
}
if (config_stats && tcache_available(tsd)) {
cache_bin_t *cb = &tsd_tcachep_get(tsd)->bins[test_binind];
expect_u64_eq(cb->tstats.nrequests, 0,
"cache_bin tstats should be 0 after migrate flush");
}
/*
* Symmetric check: post-migrate allocations should accumulate against
* a2, not a1. Refresh stats, allocate N items, refresh again, and
* verify a2's bin nrequests grew while a1's did not.
*/
if (config_stats && tcache_available(tsd)) {
char ctl_a1[64], ctl_a2[64];
uint64_t a1_pre, a2_pre, a1_post, a2_post;
size_t sz_u64 = sizeof(uint64_t);
uint64_t epoch = 1;
malloc_snprintf(ctl_a1, sizeof(ctl_a1),
"stats.arenas.%u.bins.%u.nrequests",
migrate_a1_ind, test_binind);
malloc_snprintf(ctl_a2, sizeof(ctl_a2),
"stats.arenas.%u.bins.%u.nrequests",
migrate_a2_ind, test_binind);
expect_d_eq(mallctl("epoch", NULL, NULL, &epoch,
sizeof(uint64_t)), 0, "epoch refresh");
expect_d_eq(mallctl(ctl_a1, &a1_pre, &sz_u64, NULL, 0), 0,
"read a1 nrequests baseline");
expect_d_eq(mallctl(ctl_a2, &a2_pre, &sz_u64, NULL, 0), 0,
"read a2 nrequests baseline");
void *p[24];
for (size_t i = 0; i < ARRAY_SIZE(p); i++) {
p[i] = malloc(8);
}
for (size_t i = 0; i < ARRAY_SIZE(p); i++) {
free(p[i]);
}
/*
* Flushing the tcache merges cache_bin tstats into the arena's
* bin stats (epoch refresh alone does not).
*/
expect_d_eq(mallctl("thread.tcache.flush", NULL, NULL, NULL,
0), 0, "thread.tcache.flush");
expect_d_eq(mallctl("epoch", NULL, NULL, &epoch,
sizeof(uint64_t)), 0, "epoch refresh");
expect_d_eq(mallctl(ctl_a1, &a1_post, &sz_u64, NULL, 0), 0,
"read a1 nrequests post");
expect_d_eq(mallctl(ctl_a2, &a2_post, &sz_u64, NULL, 0), 0,
"read a2 nrequests post");
expect_u64_eq(a1_post, a1_pre,
"a1 nrequests should be unchanged by post-migrate allocs");
expect_u64_ge(a2_post - a2_pre, (uint64_t)ARRAY_SIZE(p),
"a2 nrequests should reflect post-migrate allocs");
}
atomic_store_b(&migrate_done, true, ATOMIC_RELEASE);
while (!atomic_load_b(&migrate_go_exit, ATOMIC_ACQUIRE)) {
/* Hold the binding so main can read post-migration counts. */
}
return NULL;
}
TEST_BEGIN(test_thread_migrate_arena) {
atomic_store_b(&migrate_done, false, ATOMIC_RELEASE);
atomic_store_b(&migrate_go_exit, false, ATOMIC_RELEASE);
migrate_a1_ind = do_arena_create(-1, -1);
migrate_a2_ind = do_arena_create(-1, -1);
thd_t thd;
thd_create(&thd, migrate_worker, NULL);
while (!atomic_load_b(&migrate_done, ATOMIC_ACQUIRE)) {
/* Wait for the migrator to publish results. */
}
tsdn_t *tsdn = tsdn_fetch();
arena_t *a1 = arena_get(tsdn, migrate_a1_ind, false);
arena_t *a2 = arena_get(tsdn, migrate_a2_ind, false);
expect_u_eq(arena_nthreads_get(a1, false), 0,
"arena1 should have 0 threads after migration");
expect_u_eq(arena_nthreads_get(a2, false), 1,
"arena2 should have 1 thread after migration");
atomic_store_b(&migrate_go_exit, true, ATOMIC_RELEASE);
thd_join(thd, NULL);
do_arena_destroy(migrate_a1_ind);
do_arena_destroy(migrate_a2_ind);
}
TEST_END
static atomic_b_t cleanup_skip_worker;
static void *
arena_cleanup_worker(void *unused) {
tsd_t *tsd = tsd_fetch();
/* Bind tsd to both an external and internal arena. */
free(malloc(1));
arena_t *a = tsd_arena_get(tsd);
if (a == NULL) {
atomic_store_b(&cleanup_skip_worker, true, ATOMIC_RELEASE);
return NULL;
}
unsigned pre_nt = arena_nthreads_get(a, false);
arena_cleanup(tsd);
expect_ptr_null(
tsd_arena_get(tsd), "tsd_arena should be NULL after arena_cleanup");
expect_u_eq(arena_nthreads_get(a, false), pre_nt - 1,
"external nthreads should decrease by 1 after arena_cleanup");
arena_cleanup(tsd);
expect_ptr_null(tsd_arena_get(tsd),
"tsd_arena should remain NULL after second arena_cleanup");
expect_u_eq(arena_nthreads_get(a, false), pre_nt - 1,
"external nthreads should not change on idempotent arena_cleanup");
return NULL;
}
TEST_BEGIN(test_arena_cleanup) {
atomic_store_b(&cleanup_skip_worker, false, ATOMIC_RELEASE);
thd_t thd;
thd_create(&thd, arena_cleanup_worker, NULL);
thd_join(thd, NULL);
if (atomic_load_b(&cleanup_skip_worker, ATOMIC_ACQUIRE)) {
test_skip(
"Worker tsd_arena was NULL after malloc; "
"cannot exercise arena_cleanup path");
}
}
TEST_END
static void *
iarena_cleanup_worker(void *unused) {
tsd_t *tsd = tsd_fetch();
/* Bind tsd to both an external and internal arena. */
free(malloc(1));
arena_t *ia = tsd_iarena_get(tsd);
if (ia == NULL) {
atomic_store_b(&cleanup_skip_worker, true, ATOMIC_RELEASE);
return NULL;
}
unsigned pre_nt = arena_nthreads_get(ia, true);
iarena_cleanup(tsd);
expect_ptr_null(tsd_iarena_get(tsd),
"tsd_iarena should be NULL after iarena_cleanup");
expect_u_eq(arena_nthreads_get(ia, true), pre_nt - 1,
"internal nthreads should decrease by 1 after iarena_cleanup");
iarena_cleanup(tsd);
expect_ptr_null(tsd_iarena_get(tsd),
"tsd_iarena should remain NULL after second iarena_cleanup");
expect_u_eq(arena_nthreads_get(ia, true), pre_nt - 1,
"internal nthreads should not change on idempotent iarena_cleanup");
return NULL;
}
TEST_BEGIN(test_iarena_cleanup) {
atomic_store_b(&cleanup_skip_worker, false, ATOMIC_RELEASE);
thd_t thd;
thd_create(&thd, iarena_cleanup_worker, NULL);
thd_join(thd, NULL);
if (atomic_load_b(&cleanup_skip_worker, ATOMIC_ACQUIRE)) {
test_skip(
"Worker tsd_iarena was NULL after malloc; "
"cannot exercise iarena_cleanup path");
}
}
TEST_END
int
main(void) {
return test(test_narenas_total_get_consistent,
test_narenas_total_set_roundtrip, test_narenas_auto_set_roundtrip,
test_manual_arena_base_set_roundtrip, test_arena_set_roundtrip,
test_arena_init_creates_arena, test_arena_init_idempotent_auto,
test_thread_migrate_arena, test_arena_cleanup, test_iarena_cleanup);
}