|  | @@ -36,8 +36,11 @@
 | 
	
		
			
				|  |  |  #include <grpc/support/alloc.h>
 | 
	
		
			
				|  |  |  #include <grpc/support/log.h>
 | 
	
		
			
				|  |  |  #include <grpc/support/string_util.h>
 | 
	
		
			
				|  |  | +#include <grpc/support/sync.h>
 | 
	
		
			
				|  |  | +#include <grpc/support/thd.h>
 | 
	
		
			
				|  |  |  #include <grpc/support/useful.h>
 | 
	
		
			
				|  |  |  #include <inttypes.h>
 | 
	
		
			
				|  |  | +#include <string.h>
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  #include "src/core/lib/support/string.h"
 | 
	
		
			
				|  |  |  #include "test/core/util/test_config.h"
 | 
	
	
		
			
				|  | @@ -65,9 +68,12 @@ static void test(const char *name, size_t init_size, const size_t *allocs,
 | 
	
		
			
				|  |  |    void **ps = gpr_zalloc(sizeof(*ps) * nallocs);
 | 
	
		
			
				|  |  |    for (size_t i = 0; i < nallocs; i++) {
 | 
	
		
			
				|  |  |      ps[i] = gpr_arena_alloc(a, allocs[i]);
 | 
	
		
			
				|  |  | +    // ensure no duplicate results
 | 
	
		
			
				|  |  |      for (size_t j = 0; j < i; j++) {
 | 
	
		
			
				|  |  |        GPR_ASSERT(ps[i] != ps[j]);
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  | +    // ensure writable
 | 
	
		
			
				|  |  | +    memset(ps[i], 1, allocs[i]);
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |    gpr_arena_destroy(a);
 | 
	
		
			
				|  |  |  }
 | 
	
	
		
			
				|  | @@ -76,6 +82,42 @@ static void test(const char *name, size_t init_size, const size_t *allocs,
 | 
	
		
			
				|  |  |    static const size_t allocs_##name[] = {__VA_ARGS__}; \
 | 
	
		
			
				|  |  |    test(#name, init_size, allocs_##name, GPR_ARRAY_SIZE(allocs_##name))
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +#define CONCURRENT_TEST_ITERATIONS 100000
 | 
	
		
			
				|  |  | +#define CONCURRENT_TEST_THREADS 100
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +typedef struct {
 | 
	
		
			
				|  |  | +  gpr_event ev_start;
 | 
	
		
			
				|  |  | +  gpr_arena *arena;
 | 
	
		
			
				|  |  | +} concurrent_test_args;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +static void concurrent_test_body(void *arg) {
 | 
	
		
			
				|  |  | +  concurrent_test_args *a = arg;
 | 
	
		
			
				|  |  | +  gpr_event_wait(&a->ev_start, gpr_inf_future(GPR_CLOCK_REALTIME));
 | 
	
		
			
				|  |  | +  for (size_t i = 0; i < CONCURRENT_TEST_ITERATIONS; i++) {
 | 
	
		
			
				|  |  | +    *(char *)gpr_arena_alloc(a->arena, 1) = (char)i;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +static void concurrent_test(void) {
 | 
	
		
			
				|  |  | +  concurrent_test_args args;
 | 
	
		
			
				|  |  | +  gpr_event_init(&args.ev_start);
 | 
	
		
			
				|  |  | +  args.arena = gpr_arena_create(1024);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  gpr_thd_id thds[CONCURRENT_TEST_THREADS];
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  for (int i = 0; i < CONCURRENT_TEST_THREADS; i++) {
 | 
	
		
			
				|  |  | +    gpr_thd_options opt = gpr_thd_options_default();
 | 
	
		
			
				|  |  | +    gpr_thd_options_is_joinable(&opt);
 | 
	
		
			
				|  |  | +    gpr_thd_new(&thds[i], concurrent_test_body, &args, &opt);
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  gpr_event_set(&args.ev_start, (void *)1);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  for (int i = 0; i < CONCURRENT_TEST_THREADS; i++) {
 | 
	
		
			
				|  |  | +    gpr_thd_join(thds[i]);
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  int main(int argc, char *argv[]) {
 | 
	
		
			
				|  |  |    grpc_test_init(argc, argv);
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -86,6 +128,7 @@ int main(int argc, char *argv[]) {
 | 
	
		
			
				|  |  |    TEST(1_3, 1, 3);
 | 
	
		
			
				|  |  |    TEST(1_inc, 1, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11);
 | 
	
		
			
				|  |  |    TEST(6_123, 6, 1, 2, 3);
 | 
	
		
			
				|  |  | +  concurrent_test();
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    return 0;
 | 
	
		
			
				|  |  |  }
 |