123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139 |
- #include "test/jemalloc_test.h"
- static int data_cleanup_count;
- void
- data_cleanup(int *data) {
- if (data_cleanup_count == 0) {
- assert_x_eq(*data, MALLOC_TSD_TEST_DATA_INIT,
- "Argument passed into cleanup function should match tsd "
- "value");
- }
- ++data_cleanup_count;
- /*
- * Allocate during cleanup for two rounds, in order to assure that
- * jemalloc's internal tsd reinitialization happens.
- */
- bool reincarnate = false;
- switch (*data) {
- case MALLOC_TSD_TEST_DATA_INIT:
- *data = 1;
- reincarnate = true;
- break;
- case 1:
- *data = 2;
- reincarnate = true;
- break;
- case 2:
- return;
- default:
- not_reached();
- }
- if (reincarnate) {
- void *p = mallocx(1, 0);
- assert_ptr_not_null(p, "Unexpeced mallocx() failure");
- dallocx(p, 0);
- }
- }
- static void *
- thd_start(void *arg) {
- int d = (int)(uintptr_t)arg;
- void *p;
- tsd_t *tsd = tsd_fetch();
- assert_x_eq(tsd_test_data_get(tsd), MALLOC_TSD_TEST_DATA_INIT,
- "Initial tsd get should return initialization value");
- p = malloc(1);
- assert_ptr_not_null(p, "Unexpected malloc() failure");
- tsd_test_data_set(tsd, d);
- assert_x_eq(tsd_test_data_get(tsd), d,
- "After tsd set, tsd get should return value that was set");
- d = 0;
- assert_x_eq(tsd_test_data_get(tsd), (int)(uintptr_t)arg,
- "Resetting local data should have no effect on tsd");
- tsd_test_callback_set(tsd, &data_cleanup);
- free(p);
- return NULL;
- }
- TEST_BEGIN(test_tsd_main_thread) {
- thd_start((void *)(uintptr_t)0xa5f3e329);
- }
- TEST_END
- TEST_BEGIN(test_tsd_sub_thread) {
- thd_t thd;
- data_cleanup_count = 0;
- thd_create(&thd, thd_start, (void *)MALLOC_TSD_TEST_DATA_INIT);
- thd_join(thd, NULL);
- /*
- * We reincarnate twice in the data cleanup, so it should execute at
- * least 3 times.
- */
- assert_x_ge(data_cleanup_count, 3,
- "Cleanup function should have executed multiple times.");
- }
- TEST_END
- static void *
- thd_start_reincarnated(void *arg) {
- tsd_t *tsd = tsd_fetch();
- assert(tsd);
- void *p = malloc(1);
- assert_ptr_not_null(p, "Unexpected malloc() failure");
- /* Manually trigger reincarnation. */
- assert_ptr_not_null(tsd_arena_get(tsd),
- "Should have tsd arena set.");
- tsd_cleanup((void *)tsd);
- assert_ptr_null(*tsd_arenap_get_unsafe(tsd),
- "TSD arena should have been cleared.");
- assert_u_eq(tsd->state, tsd_state_purgatory,
- "TSD state should be purgatory\n");
- free(p);
- assert_u_eq(tsd->state, tsd_state_reincarnated,
- "TSD state should be reincarnated\n");
- p = mallocx(1, MALLOCX_TCACHE_NONE);
- assert_ptr_not_null(p, "Unexpected malloc() failure");
- assert_ptr_null(*tsd_arenap_get_unsafe(tsd),
- "Should not have tsd arena set after reincarnation.");
- free(p);
- tsd_cleanup((void *)tsd);
- assert_ptr_null(*tsd_arenap_get_unsafe(tsd),
- "TSD arena should have been cleared after 2nd cleanup.");
- return NULL;
- }
- TEST_BEGIN(test_tsd_reincarnation) {
- thd_t thd;
- thd_create(&thd, thd_start_reincarnated, NULL);
- thd_join(thd, NULL);
- }
- TEST_END
- int
- main(void) {
- /* Ensure tsd bootstrapped. */
- if (nallocx(1, 0) == 0) {
- malloc_printf("Initialization error");
- return test_status_fail;
- }
- return test_no_reentrancy(
- test_tsd_main_thread,
- test_tsd_sub_thread,
- test_tsd_reincarnation);
- }
|