1 #include "git-compat-util.h"
2 #include "thread-utils.h"
4 #include "trace2/tr2_tls.h"
7 * Initialize size of the thread stack for nested regions.
8 * This is used to store nested region start times. Note that
9 * this stack is per-thread and not per-trace-key.
11 #define TR2_REGION_NESTING_INITIAL_SIZE (100)
13 static struct tr2tls_thread_ctx
*tr2tls_thread_main
;
14 static uint64_t tr2tls_us_start_process
;
16 static pthread_mutex_t tr2tls_mutex
;
17 static pthread_key_t tr2tls_key
;
19 static int tr2_next_thread_id
; /* modify under lock */
21 void tr2tls_start_process_clock(void)
23 if (tr2tls_us_start_process
)
27 * Keep the absolute start time of the process (i.e. the main
28 * process) in a fixed variable since other threads need to
29 * access it. This allows them to do that without a lock on
30 * main thread's array data (because of reallocs).
32 tr2tls_us_start_process
= getnanotime() / 1000;
35 struct tr2tls_thread_ctx
*tr2tls_create_self(const char *thread_base_name
,
36 uint64_t us_thread_start
)
38 struct tr2tls_thread_ctx
*ctx
= xcalloc(1, sizeof(*ctx
));
39 struct strbuf buf
= STRBUF_INIT
;
42 * Implicitly "tr2tls_push_self()" to capture the thread's start
43 * time in array_us_start[0]. For the main thread this gives us the
44 * application run time.
46 ctx
->alloc
= TR2_REGION_NESTING_INITIAL_SIZE
;
47 ctx
->array_us_start
= (uint64_t *)xcalloc(ctx
->alloc
, sizeof(uint64_t));
48 ctx
->array_us_start
[ctx
->nr_open_regions
++] = us_thread_start
;
50 ctx
->thread_id
= tr2tls_locked_increment(&tr2_next_thread_id
);
54 strbuf_addf(&buf
, "th%02d:", ctx
->thread_id
);
55 strbuf_addstr(&buf
, thread_base_name
);
56 if (buf
.len
> TR2_MAX_THREAD_NAME
)
57 strbuf_setlen(&buf
, TR2_MAX_THREAD_NAME
);
58 ctx
->thread_name
= strbuf_detach(&buf
, NULL
);
60 pthread_setspecific(tr2tls_key
, ctx
);
65 struct tr2tls_thread_ctx
*tr2tls_get_self(void)
67 struct tr2tls_thread_ctx
*ctx
;
70 return tr2tls_thread_main
;
72 ctx
= pthread_getspecific(tr2tls_key
);
75 * If the current thread's thread-proc did not call
76 * trace2_thread_start(), then the thread will not have any
77 * thread-local storage. Create it now and silently continue.
80 ctx
= tr2tls_create_self("unknown", getnanotime() / 1000);
85 int tr2tls_is_main_thread(void)
90 return pthread_getspecific(tr2tls_key
) == tr2tls_thread_main
;
93 void tr2tls_unset_self(void)
95 struct tr2tls_thread_ctx
*ctx
;
97 ctx
= tr2tls_get_self();
99 pthread_setspecific(tr2tls_key
, NULL
);
101 free((char *)ctx
->thread_name
);
102 free(ctx
->array_us_start
);
106 void tr2tls_push_self(uint64_t us_now
)
108 struct tr2tls_thread_ctx
*ctx
= tr2tls_get_self();
110 ALLOC_GROW(ctx
->array_us_start
, ctx
->nr_open_regions
+ 1, ctx
->alloc
);
111 ctx
->array_us_start
[ctx
->nr_open_regions
++] = us_now
;
114 void tr2tls_pop_self(void)
116 struct tr2tls_thread_ctx
*ctx
= tr2tls_get_self();
118 if (!ctx
->nr_open_regions
)
119 BUG("no open regions in thread '%s'", ctx
->thread_name
);
121 ctx
->nr_open_regions
--;
124 void tr2tls_pop_unwind_self(void)
126 struct tr2tls_thread_ctx
*ctx
= tr2tls_get_self();
128 while (ctx
->nr_open_regions
> 1)
132 uint64_t tr2tls_region_elasped_self(uint64_t us
)
134 struct tr2tls_thread_ctx
*ctx
;
137 ctx
= tr2tls_get_self();
138 if (!ctx
->nr_open_regions
)
141 us_start
= ctx
->array_us_start
[ctx
->nr_open_regions
- 1];
143 return us
- us_start
;
146 uint64_t tr2tls_absolute_elapsed(uint64_t us
)
148 if (!tr2tls_thread_main
)
151 return us
- tr2tls_us_start_process
;
154 void tr2tls_init(void)
156 tr2tls_start_process_clock();
158 pthread_key_create(&tr2tls_key
, NULL
);
159 init_recursive_mutex(&tr2tls_mutex
);
162 tr2tls_create_self("main", tr2tls_us_start_process
);
165 void tr2tls_release(void)
168 tr2tls_thread_main
= NULL
;
170 pthread_mutex_destroy(&tr2tls_mutex
);
171 pthread_key_delete(tr2tls_key
);
174 int tr2tls_locked_increment(int *p
)
178 pthread_mutex_lock(&tr2tls_mutex
);
180 *p
= current_value
+ 1;
181 pthread_mutex_unlock(&tr2tls_mutex
);
183 return current_value
;
186 void tr2tls_lock(void)
188 pthread_mutex_lock(&tr2tls_mutex
);
191 void tr2tls_unlock(void)
193 pthread_mutex_unlock(&tr2tls_mutex
);