advice: allow disabling the automatic hint in advise_if_enabled()
[git.git] / trace2 / tr2_tls.c
blob4f75392952b6a429682d8989fe7146487ce0cb6a
1 #include "git-compat-util.h"
2 #include "strbuf.h"
3 #include "thread-utils.h"
4 #include "trace.h"
5 #include "trace2/tr2_tls.h"
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.
12 #define TR2_REGION_NESTING_INITIAL_SIZE (100)
14 static struct tr2tls_thread_ctx *tr2tls_thread_main;
15 static uint64_t tr2tls_us_start_process;
17 static pthread_mutex_t tr2tls_mutex;
18 static pthread_key_t tr2tls_key;
20 static int tr2_next_thread_id; /* modify under lock */
22 void tr2tls_start_process_clock(void)
24 if (tr2tls_us_start_process)
25 return;
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).
33 tr2tls_us_start_process = getnanotime() / 1000;
36 struct tr2tls_thread_ctx *tr2tls_create_self(const char *thread_base_name,
37 uint64_t us_thread_start)
39 struct tr2tls_thread_ctx *ctx = xcalloc(1, sizeof(*ctx));
40 struct strbuf buf = STRBUF_INIT;
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.
47 ctx->alloc = TR2_REGION_NESTING_INITIAL_SIZE;
48 ctx->array_us_start = (uint64_t *)xcalloc(ctx->alloc, sizeof(uint64_t));
49 ctx->array_us_start[ctx->nr_open_regions++] = us_thread_start;
51 ctx->thread_id = tr2tls_locked_increment(&tr2_next_thread_id);
53 strbuf_init(&buf, 0);
54 if (ctx->thread_id)
55 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);
61 pthread_setspecific(tr2tls_key, ctx);
63 return ctx;
66 struct tr2tls_thread_ctx *tr2tls_get_self(void)
68 struct tr2tls_thread_ctx *ctx;
70 if (!HAVE_THREADS)
71 return tr2tls_thread_main;
73 ctx = pthread_getspecific(tr2tls_key);
76 * 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.
80 if (!ctx)
81 ctx = tr2tls_create_self("unknown", getnanotime() / 1000);
83 return ctx;
86 int tr2tls_is_main_thread(void)
88 if (!HAVE_THREADS)
89 return 1;
91 return pthread_getspecific(tr2tls_key) == tr2tls_thread_main;
94 void tr2tls_unset_self(void)
96 struct tr2tls_thread_ctx *ctx;
98 ctx = tr2tls_get_self();
100 pthread_setspecific(tr2tls_key, NULL);
102 free((char *)ctx->thread_name);
103 free(ctx->array_us_start);
104 free(ctx);
107 void tr2tls_push_self(uint64_t us_now)
109 struct tr2tls_thread_ctx *ctx = tr2tls_get_self();
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;
115 void tr2tls_pop_self(void)
117 struct tr2tls_thread_ctx *ctx = tr2tls_get_self();
119 if (!ctx->nr_open_regions)
120 BUG("no open regions in thread '%s'", ctx->thread_name);
122 ctx->nr_open_regions--;
125 void tr2tls_pop_unwind_self(void)
127 struct tr2tls_thread_ctx *ctx = tr2tls_get_self();
129 while (ctx->nr_open_regions > 1)
130 tr2tls_pop_self();
133 uint64_t tr2tls_region_elasped_self(uint64_t us)
135 struct tr2tls_thread_ctx *ctx;
136 uint64_t us_start;
138 ctx = tr2tls_get_self();
139 if (!ctx->nr_open_regions)
140 return 0;
142 us_start = ctx->array_us_start[ctx->nr_open_regions - 1];
144 return us - us_start;
147 uint64_t tr2tls_absolute_elapsed(uint64_t us)
149 if (!tr2tls_thread_main)
150 return 0;
152 return us - tr2tls_us_start_process;
155 void tr2tls_init(void)
157 tr2tls_start_process_clock();
159 pthread_key_create(&tr2tls_key, NULL);
160 init_recursive_mutex(&tr2tls_mutex);
162 tr2tls_thread_main =
163 tr2tls_create_self("main", tr2tls_us_start_process);
166 void tr2tls_release(void)
168 tr2tls_unset_self();
169 tr2tls_thread_main = NULL;
171 pthread_mutex_destroy(&tr2tls_mutex);
172 pthread_key_delete(tr2tls_key);
175 int tr2tls_locked_increment(int *p)
177 int current_value;
179 pthread_mutex_lock(&tr2tls_mutex);
180 current_value = *p;
181 *p = current_value + 1;
182 pthread_mutex_unlock(&tr2tls_mutex);
184 return current_value;
187 void tr2tls_lock(void)
189 pthread_mutex_lock(&tr2tls_mutex);
192 void tr2tls_unlock(void)
194 pthread_mutex_unlock(&tr2tls_mutex);