mirror of
https://github.com/jemalloc/jemalloc.git
synced 2026-06-01 17:54:16 +03:00
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.
337 lines
10 KiB
C
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);
|
|
}
|