blob: 7b023c1bfc65fa8524d46e15c2df7b1d57945da1 [file] [log] [blame]
Elijah Newren36bf1952023-02-24 00:09:24 +00001#include "git-compat-util.h"
Elijah Newren0a4d5b92023-12-23 17:14:58 +00002#include "strbuf.h"
Jeff Hostetleree4512e2019-02-22 14:25:01 -08003#include "thread-utils.h"
Elijah Newren36bf1952023-02-24 00:09:24 +00004#include "trace.h"
Jeff Hostetleree4512e2019-02-22 14:25:01 -08005#include "trace2/tr2_tls.h"
6
7/*
8 * Initialize size of the thread stack for nested regions.
9 * This is used to store nested region start times. Note that
10 * this stack is per-thread and not per-trace-key.
11 */
12#define TR2_REGION_NESTING_INITIAL_SIZE (100)
13
14static struct tr2tls_thread_ctx *tr2tls_thread_main;
Jeff Hostetlera0897242019-04-15 13:39:43 -070015static uint64_t tr2tls_us_start_process;
Jeff Hostetleree4512e2019-02-22 14:25:01 -080016
17static pthread_mutex_t tr2tls_mutex;
18static pthread_key_t tr2tls_key;
19
20static int tr2_next_thread_id; /* modify under lock */
21
Jeff Hostetlera0897242019-04-15 13:39:43 -070022void tr2tls_start_process_clock(void)
Jeff Hostetleree4512e2019-02-22 14:25:01 -080023{
Jeff Hostetlera0897242019-04-15 13:39:43 -070024 if (tr2tls_us_start_process)
25 return;
26
27 /*
28 * Keep the absolute start time of the process (i.e. the main
29 * process) in a fixed variable since other threads need to
30 * access it. This allows them to do that without a lock on
31 * main thread's array data (because of reallocs).
32 */
33 tr2tls_us_start_process = getnanotime() / 1000;
34}
35
Jeff Hostetlera70839c2022-10-24 13:41:03 +000036struct tr2tls_thread_ctx *tr2tls_create_self(const char *thread_base_name,
Jeff Hostetlera0897242019-04-15 13:39:43 -070037 uint64_t us_thread_start)
38{
Jeff Hostetleree4512e2019-02-22 14:25:01 -080039 struct tr2tls_thread_ctx *ctx = xcalloc(1, sizeof(*ctx));
Jeff Hostetler24a4c452022-10-24 13:41:05 +000040 struct strbuf buf = STRBUF_INIT;
Jeff Hostetleree4512e2019-02-22 14:25:01 -080041
42 /*
43 * Implicitly "tr2tls_push_self()" to capture the thread's start
44 * time in array_us_start[0]. For the main thread this gives us the
45 * application run time.
46 */
47 ctx->alloc = TR2_REGION_NESTING_INITIAL_SIZE;
48 ctx->array_us_start = (uint64_t *)xcalloc(ctx->alloc, sizeof(uint64_t));
Jeff Hostetlera0897242019-04-15 13:39:43 -070049 ctx->array_us_start[ctx->nr_open_regions++] = us_thread_start;
Jeff Hostetleree4512e2019-02-22 14:25:01 -080050
51 ctx->thread_id = tr2tls_locked_increment(&tr2_next_thread_id);
52
Jeff Hostetler24a4c452022-10-24 13:41:05 +000053 strbuf_init(&buf, 0);
Jeff Hostetleree4512e2019-02-22 14:25:01 -080054 if (ctx->thread_id)
Jeff Hostetler24a4c452022-10-24 13:41:05 +000055 strbuf_addf(&buf, "th%02d:", ctx->thread_id);
56 strbuf_addstr(&buf, thread_base_name);
57 if (buf.len > TR2_MAX_THREAD_NAME)
58 strbuf_setlen(&buf, TR2_MAX_THREAD_NAME);
59 ctx->thread_name = strbuf_detach(&buf, NULL);
Jeff Hostetleree4512e2019-02-22 14:25:01 -080060
61 pthread_setspecific(tr2tls_key, ctx);
62
63 return ctx;
64}
65
66struct tr2tls_thread_ctx *tr2tls_get_self(void)
67{
Jeff Hostetler5fdae9d2019-05-21 12:33:59 -070068 struct tr2tls_thread_ctx *ctx;
69
70 if (!HAVE_THREADS)
71 return tr2tls_thread_main;
72
73 ctx = pthread_getspecific(tr2tls_key);
Jeff Hostetleree4512e2019-02-22 14:25:01 -080074
75 /*
Jeff Hostetler5bbb9252022-10-24 13:41:01 +000076 * If the current thread's thread-proc did not call
77 * trace2_thread_start(), then the thread will not have any
78 * thread-local storage. Create it now and silently continue.
Jeff Hostetleree4512e2019-02-22 14:25:01 -080079 */
80 if (!ctx)
Jeff Hostetlera0897242019-04-15 13:39:43 -070081 ctx = tr2tls_create_self("unknown", getnanotime() / 1000);
Jeff Hostetleree4512e2019-02-22 14:25:01 -080082
83 return ctx;
84}
85
86int tr2tls_is_main_thread(void)
87{
Jeff Hostetler5fdae9d2019-05-21 12:33:59 -070088 if (!HAVE_THREADS)
89 return 1;
Jeff Hostetleree4512e2019-02-22 14:25:01 -080090
Jeff Hostetler5fdae9d2019-05-21 12:33:59 -070091 return pthread_getspecific(tr2tls_key) == tr2tls_thread_main;
Jeff Hostetleree4512e2019-02-22 14:25:01 -080092}
93
94void tr2tls_unset_self(void)
95{
96 struct tr2tls_thread_ctx *ctx;
97
98 ctx = tr2tls_get_self();
99
100 pthread_setspecific(tr2tls_key, NULL);
101
Jeff Hostetler24a4c452022-10-24 13:41:05 +0000102 free((char *)ctx->thread_name);
Jeff Hostetleree4512e2019-02-22 14:25:01 -0800103 free(ctx->array_us_start);
104 free(ctx);
105}
106
107void tr2tls_push_self(uint64_t us_now)
108{
109 struct tr2tls_thread_ctx *ctx = tr2tls_get_self();
110
111 ALLOC_GROW(ctx->array_us_start, ctx->nr_open_regions + 1, ctx->alloc);
112 ctx->array_us_start[ctx->nr_open_regions++] = us_now;
113}
114
115void tr2tls_pop_self(void)
116{
117 struct tr2tls_thread_ctx *ctx = tr2tls_get_self();
118
119 if (!ctx->nr_open_regions)
Jeff Hostetler24a4c452022-10-24 13:41:05 +0000120 BUG("no open regions in thread '%s'", ctx->thread_name);
Jeff Hostetleree4512e2019-02-22 14:25:01 -0800121
122 ctx->nr_open_regions--;
123}
124
125void tr2tls_pop_unwind_self(void)
126{
127 struct tr2tls_thread_ctx *ctx = tr2tls_get_self();
128
129 while (ctx->nr_open_regions > 1)
130 tr2tls_pop_self();
131}
132
133uint64_t tr2tls_region_elasped_self(uint64_t us)
134{
135 struct tr2tls_thread_ctx *ctx;
136 uint64_t us_start;
137
138 ctx = tr2tls_get_self();
139 if (!ctx->nr_open_regions)
140 return 0;
141
142 us_start = ctx->array_us_start[ctx->nr_open_regions - 1];
143
144 return us - us_start;
145}
146
147uint64_t tr2tls_absolute_elapsed(uint64_t us)
148{
149 if (!tr2tls_thread_main)
150 return 0;
151
Jeff Hostetlera0897242019-04-15 13:39:43 -0700152 return us - tr2tls_us_start_process;
Jeff Hostetleree4512e2019-02-22 14:25:01 -0800153}
154
Patrick Steinhardt64d9ada2024-09-26 13:46:45 +0200155static void tr2tls_key_destructor(void *payload)
156{
157 struct tr2tls_thread_ctx *ctx = payload;
158 free((char *)ctx->thread_name);
159 free(ctx->array_us_start);
160 free(ctx);
161}
162
Jeff Hostetleree4512e2019-02-22 14:25:01 -0800163void tr2tls_init(void)
164{
Jeff Hostetlera0897242019-04-15 13:39:43 -0700165 tr2tls_start_process_clock();
166
Patrick Steinhardt64d9ada2024-09-26 13:46:45 +0200167 pthread_key_create(&tr2tls_key, tr2tls_key_destructor);
Jeff Hostetleree4512e2019-02-22 14:25:01 -0800168 init_recursive_mutex(&tr2tls_mutex);
169
Jeff Hostetlera0897242019-04-15 13:39:43 -0700170 tr2tls_thread_main =
171 tr2tls_create_self("main", tr2tls_us_start_process);
Jeff Hostetleree4512e2019-02-22 14:25:01 -0800172}
173
174void tr2tls_release(void)
175{
176 tr2tls_unset_self();
177 tr2tls_thread_main = NULL;
178
179 pthread_mutex_destroy(&tr2tls_mutex);
180 pthread_key_delete(tr2tls_key);
181}
182
183int tr2tls_locked_increment(int *p)
184{
185 int current_value;
186
187 pthread_mutex_lock(&tr2tls_mutex);
188 current_value = *p;
189 *p = current_value + 1;
190 pthread_mutex_unlock(&tr2tls_mutex);
191
192 return current_value;
193}
Jeff Hostetler8ad57562022-10-24 13:41:06 +0000194
195void tr2tls_lock(void)
196{
197 pthread_mutex_lock(&tr2tls_mutex);
198}
199
200void tr2tls_unlock(void)
201{
202 pthread_mutex_unlock(&tr2tls_mutex);
203}