• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2   This file is part of drd, a thread error detector.
3 
4   Copyright (C) 2006-2012 Bart Van Assche <bvanassche@acm.org>.
5 
6   This program is free software; you can redistribute it and/or
7   modify it under the terms of the GNU General Public License as
8   published by the Free Software Foundation; either version 2 of the
9   License, or (at your option) any later version.
10 
11   This program is distributed in the hope that it will be useful, but
12   WITHOUT ANY WARRANTY; without even the implied warranty of
13   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14   General Public License for more details.
15 
16   You should have received a copy of the GNU General Public License
17   along with this program; if not, write to the Free Software
18   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
19   02111-1307, USA.
20 
21   The GNU General Public License is contained in the file COPYING.
22 */
23 
24 
25 #include "drd_error.h"
26 #include "drd_barrier.h"
27 #include "drd_clientobj.h"
28 #include "drd_cond.h"
29 #include "drd_mutex.h"
30 #include "drd_segment.h"
31 #include "drd_semaphore.h"
32 #include "drd_suppression.h"
33 #include "drd_thread.h"
34 #include "pub_tool_vki.h"
35 #include "pub_tool_basics.h"      // Addr, SizeT
36 #include "pub_tool_libcassert.h"  // tl_assert()
37 #include "pub_tool_libcbase.h"    // VG_(strlen)()
38 #include "pub_tool_libcprint.h"   // VG_(printf)()
39 #include "pub_tool_libcproc.h"    // VG_(getenv)()
40 #include "pub_tool_machine.h"
41 #include "pub_tool_mallocfree.h"  // VG_(malloc)(), VG_(free)()
42 #include "pub_tool_options.h"     // VG_(clo_backtrace_size)
43 #include "pub_tool_threadstate.h" // VG_(get_pthread_id)()
44 
45 
46 
47 /* Local functions. */
48 
49 static void thread_append_segment(const DrdThreadId tid, Segment* const sg);
50 static void thread_discard_segment(const DrdThreadId tid, Segment* const sg);
51 static void thread_compute_conflict_set(struct bitmap** conflict_set,
52                                         const DrdThreadId tid);
53 static Bool thread_conflict_set_up_to_date(const DrdThreadId tid);
54 
55 
56 /* Local variables. */
57 
58 static ULong    s_context_switch_count;
59 static ULong    s_discard_ordered_segments_count;
60 static ULong    s_compute_conflict_set_count;
61 static ULong    s_update_conflict_set_count;
62 static ULong    s_update_conflict_set_new_sg_count;
63 static ULong    s_update_conflict_set_sync_count;
64 static ULong    s_update_conflict_set_join_count;
65 static ULong    s_conflict_set_bitmap_creation_count;
66 static ULong    s_conflict_set_bitmap2_creation_count;
67 static ThreadId s_vg_running_tid  = VG_INVALID_THREADID;
68 DrdThreadId     DRD_(g_drd_running_tid) = DRD_INVALID_THREADID;
69 ThreadInfo      DRD_(g_threadinfo)[DRD_N_THREADS];
70 struct bitmap*  DRD_(g_conflict_set);
71 static Bool     s_trace_context_switches = False;
72 static Bool     s_trace_conflict_set = False;
73 static Bool     s_trace_conflict_set_bm = False;
74 static Bool     s_trace_fork_join = False;
75 static Bool     s_segment_merging = True;
76 static Bool     s_new_segments_since_last_merge;
77 static int      s_segment_merge_interval = 10;
78 static unsigned s_join_list_vol = 10;
79 static unsigned s_deletion_head;
80 static unsigned s_deletion_tail;
81 
82 
83 /* Function definitions. */
84 
85 /** Enables/disables context switch tracing. */
DRD_(thread_trace_context_switches)86 void DRD_(thread_trace_context_switches)(const Bool t)
87 {
88    tl_assert(t == False || t == True);
89    s_trace_context_switches = t;
90 }
91 
92 /** Enables/disables conflict set tracing. */
DRD_(thread_trace_conflict_set)93 void DRD_(thread_trace_conflict_set)(const Bool t)
94 {
95    tl_assert(t == False || t == True);
96    s_trace_conflict_set = t;
97 }
98 
99 /** Enables/disables conflict set bitmap tracing. */
DRD_(thread_trace_conflict_set_bm)100 void DRD_(thread_trace_conflict_set_bm)(const Bool t)
101 {
102    tl_assert(t == False || t == True);
103    s_trace_conflict_set_bm = t;
104 }
105 
106 /** Report whether fork/join tracing is enabled. */
DRD_(thread_get_trace_fork_join)107 Bool DRD_(thread_get_trace_fork_join)(void)
108 {
109    return s_trace_fork_join;
110 }
111 
112 /** Enables/disables fork/join tracing. */
DRD_(thread_set_trace_fork_join)113 void DRD_(thread_set_trace_fork_join)(const Bool t)
114 {
115    tl_assert(t == False || t == True);
116    s_trace_fork_join = t;
117 }
118 
119 /** Enables/disables segment merging. */
DRD_(thread_set_segment_merging)120 void DRD_(thread_set_segment_merging)(const Bool m)
121 {
122    tl_assert(m == False || m == True);
123    s_segment_merging = m;
124 }
125 
126 /** Get the segment merging interval. */
DRD_(thread_get_segment_merge_interval)127 int DRD_(thread_get_segment_merge_interval)(void)
128 {
129    return s_segment_merge_interval;
130 }
131 
132 /** Set the segment merging interval. */
DRD_(thread_set_segment_merge_interval)133 void DRD_(thread_set_segment_merge_interval)(const int i)
134 {
135    s_segment_merge_interval = i;
136 }
137 
DRD_(thread_set_join_list_vol)138 void DRD_(thread_set_join_list_vol)(const int jlv)
139 {
140    s_join_list_vol = jlv;
141 }
142 
DRD_(thread_init)143 void DRD_(thread_init)(void)
144 {
145 }
146 
147 /**
148  * Convert Valgrind's ThreadId into a DrdThreadId.
149  *
150  * @return DRD thread ID upon success and DRD_INVALID_THREADID if the passed
151  *         Valgrind ThreadId does not yet exist.
152  */
DRD_(VgThreadIdToDrdThreadId)153 DrdThreadId DRD_(VgThreadIdToDrdThreadId)(const ThreadId tid)
154 {
155    int i;
156 
157    if (tid == VG_INVALID_THREADID)
158       return DRD_INVALID_THREADID;
159 
160    for (i = 1; i < DRD_N_THREADS; i++)
161    {
162       if (DRD_(g_threadinfo)[i].vg_thread_exists == True
163           && DRD_(g_threadinfo)[i].vg_threadid == tid)
164       {
165          return i;
166       }
167    }
168 
169    return DRD_INVALID_THREADID;
170 }
171 
172 /** Allocate a new DRD thread ID for the specified Valgrind thread ID. */
DRD_(VgThreadIdToNewDrdThreadId)173 static DrdThreadId DRD_(VgThreadIdToNewDrdThreadId)(const ThreadId tid)
174 {
175    int i;
176 
177    tl_assert(DRD_(VgThreadIdToDrdThreadId)(tid) == DRD_INVALID_THREADID);
178 
179    for (i = 1; i < DRD_N_THREADS; i++)
180    {
181       if (!DRD_(g_threadinfo)[i].valid)
182       {
183          tl_assert(! DRD_(IsValidDrdThreadId)(i));
184 
185          DRD_(g_threadinfo)[i].valid         = True;
186          DRD_(g_threadinfo)[i].vg_thread_exists = True;
187          DRD_(g_threadinfo)[i].vg_threadid   = tid;
188          DRD_(g_threadinfo)[i].pt_threadid   = INVALID_POSIX_THREADID;
189          DRD_(g_threadinfo)[i].stack_min     = 0;
190          DRD_(g_threadinfo)[i].stack_min_min = 0;
191          DRD_(g_threadinfo)[i].stack_startup = 0;
192          DRD_(g_threadinfo)[i].stack_max     = 0;
193          DRD_(thread_set_name)(i, "");
194          DRD_(g_threadinfo)[i].on_alt_stack        = False;
195          DRD_(g_threadinfo)[i].is_recording_loads  = True;
196          DRD_(g_threadinfo)[i].is_recording_stores = True;
197          DRD_(g_threadinfo)[i].pthread_create_nesting_level = 0;
198          DRD_(g_threadinfo)[i].synchr_nesting = 0;
199          DRD_(g_threadinfo)[i].deletion_seq = s_deletion_tail - 1;
200          tl_assert(DRD_(g_threadinfo)[i].sg_first == NULL);
201          tl_assert(DRD_(g_threadinfo)[i].sg_last == NULL);
202 
203          tl_assert(DRD_(IsValidDrdThreadId)(i));
204 
205          return i;
206       }
207    }
208 
209    VG_(printf)(
210 "\nSorry, but the maximum number of threads supported by DRD has been exceeded."
211 "Aborting.\n");
212 
213    tl_assert(False);
214 
215    return DRD_INVALID_THREADID;
216 }
217 
218 /** Convert a POSIX thread ID into a DRD thread ID. */
DRD_(PtThreadIdToDrdThreadId)219 DrdThreadId DRD_(PtThreadIdToDrdThreadId)(const PThreadId tid)
220 {
221    int i;
222 
223    if (tid != INVALID_POSIX_THREADID)
224    {
225       for (i = 1; i < DRD_N_THREADS; i++)
226       {
227          if (DRD_(g_threadinfo)[i].posix_thread_exists
228              && DRD_(g_threadinfo)[i].pt_threadid == tid)
229          {
230             return i;
231          }
232       }
233    }
234    return DRD_INVALID_THREADID;
235 }
236 
237 /** Convert a DRD thread ID into a Valgrind thread ID. */
DRD_(DrdThreadIdToVgThreadId)238 ThreadId DRD_(DrdThreadIdToVgThreadId)(const DrdThreadId tid)
239 {
240    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
241              && tid != DRD_INVALID_THREADID);
242 
243    return (DRD_(g_threadinfo)[tid].vg_thread_exists
244            ? DRD_(g_threadinfo)[tid].vg_threadid
245            : VG_INVALID_THREADID);
246 }
247 
248 #ifdef ENABLE_DRD_CONSISTENCY_CHECKS
249 /**
250  * Sanity check of the doubly linked list of segments referenced by a
251  * ThreadInfo struct.
252  * @return True if sane, False if not.
253  */
DRD_(sane_ThreadInfo)254 static Bool DRD_(sane_ThreadInfo)(const ThreadInfo* const ti)
255 {
256    Segment* p;
257 
258    for (p = ti->first; p; p = p->next) {
259       if (p->next && p->next->prev != p)
260          return False;
261       if (p->next == 0 && p != ti->last)
262          return False;
263    }
264    for (p = ti->last; p; p = p->prev) {
265       if (p->prev && p->prev->next != p)
266          return False;
267       if (p->prev == 0 && p != ti->first)
268          return False;
269    }
270    return True;
271 }
272 #endif
273 
274 /**
275  * Create the first segment for a newly started thread.
276  *
277  * This function is called from the handler installed via
278  * VG_(track_pre_thread_ll_create)(). The Valgrind core invokes this handler
279  * from the context of the creator thread, before the new thread has been
280  * created.
281  *
282  * @param[in] creator    DRD thread ID of the creator thread.
283  * @param[in] vg_created Valgrind thread ID of the created thread.
284  *
285  * @return DRD thread ID of the created thread.
286  */
DRD_(thread_pre_create)287 DrdThreadId DRD_(thread_pre_create)(const DrdThreadId creator,
288                                     const ThreadId vg_created)
289 {
290    DrdThreadId created;
291 
292    tl_assert(DRD_(VgThreadIdToDrdThreadId)(vg_created) == DRD_INVALID_THREADID);
293    created = DRD_(VgThreadIdToNewDrdThreadId)(vg_created);
294    tl_assert(0 <= (int)created && created < DRD_N_THREADS
295              && created != DRD_INVALID_THREADID);
296 
297    tl_assert(DRD_(g_threadinfo)[created].sg_first == NULL);
298    tl_assert(DRD_(g_threadinfo)[created].sg_last == NULL);
299    /* Create an initial segment for the newly created thread. */
300    thread_append_segment(created, DRD_(sg_new)(creator, created));
301 
302    return created;
303 }
304 
305 /**
306  * Initialize DRD_(g_threadinfo)[] for a newly created thread. Must be called
307  * after the thread has been created and before any client instructions are run
308  * on the newly created thread, e.g. from the handler installed via
309  * VG_(track_pre_thread_first_insn)().
310  *
311  * @param[in] vg_created Valgrind thread ID of the newly created thread.
312  *
313  * @return DRD thread ID for the new thread.
314  */
DRD_(thread_post_create)315 DrdThreadId DRD_(thread_post_create)(const ThreadId vg_created)
316 {
317    const DrdThreadId created = DRD_(VgThreadIdToDrdThreadId)(vg_created);
318 
319    tl_assert(0 <= (int)created && created < DRD_N_THREADS
320              && created != DRD_INVALID_THREADID);
321 
322    DRD_(g_threadinfo)[created].stack_max
323       = VG_(thread_get_stack_max)(vg_created);
324    DRD_(g_threadinfo)[created].stack_startup
325       = DRD_(g_threadinfo)[created].stack_max;
326    DRD_(g_threadinfo)[created].stack_min
327       = DRD_(g_threadinfo)[created].stack_max;
328    DRD_(g_threadinfo)[created].stack_min_min
329       = DRD_(g_threadinfo)[created].stack_max;
330    DRD_(g_threadinfo)[created].stack_size
331       = VG_(thread_get_stack_size)(vg_created);
332    tl_assert(DRD_(g_threadinfo)[created].stack_max != 0);
333 
334    return created;
335 }
336 
DRD_(thread_delayed_delete)337 static void DRD_(thread_delayed_delete)(const DrdThreadId tid)
338 {
339    int j;
340 
341    DRD_(g_threadinfo)[tid].vg_thread_exists = False;
342    DRD_(g_threadinfo)[tid].posix_thread_exists = False;
343    DRD_(g_threadinfo)[tid].deletion_seq = s_deletion_head++;
344 #if 0
345    VG_(message)(Vg_DebugMsg, "Adding thread %d to the deletion list\n", tid);
346 #endif
347    if (s_deletion_head - s_deletion_tail >= s_join_list_vol) {
348       for (j = 0; j < DRD_N_THREADS; ++j) {
349          if (DRD_(IsValidDrdThreadId)(j)
350              && DRD_(g_threadinfo)[j].deletion_seq == s_deletion_tail)
351          {
352             s_deletion_tail++;
353 #if 0
354             VG_(message)(Vg_DebugMsg, "Delayed delete of thread %d\n", j);
355 #endif
356             DRD_(thread_delete)(j, False);
357             break;
358          }
359       }
360    }
361 }
362 
363 /**
364  * Process VG_USERREQ__POST_THREAD_JOIN. This client request is invoked just
365  * after thread drd_joiner joined thread drd_joinee.
366  */
DRD_(thread_post_join)367 void DRD_(thread_post_join)(DrdThreadId drd_joiner, DrdThreadId drd_joinee)
368 {
369    tl_assert(DRD_(IsValidDrdThreadId)(drd_joiner));
370    tl_assert(DRD_(IsValidDrdThreadId)(drd_joinee));
371 
372    DRD_(thread_new_segment)(drd_joiner);
373    DRD_(thread_combine_vc_join)(drd_joiner, drd_joinee);
374    DRD_(thread_new_segment)(drd_joinee);
375 
376    if (s_trace_fork_join)
377    {
378       const ThreadId joiner = DRD_(DrdThreadIdToVgThreadId)(drd_joiner);
379       const unsigned msg_size = 256;
380       char* msg;
381 
382       msg = VG_(malloc)("drd.main.dptj.1", msg_size);
383       tl_assert(msg);
384       VG_(snprintf)(msg, msg_size,
385                     "drd_post_thread_join joiner = %d, joinee = %d",
386                     drd_joiner, drd_joinee);
387       if (joiner)
388       {
389          char* vc;
390 
391          vc = DRD_(vc_aprint)(DRD_(thread_get_vc)(drd_joiner));
392          VG_(snprintf)(msg + VG_(strlen)(msg), msg_size - VG_(strlen)(msg),
393                        ", new vc: %s", vc);
394          VG_(free)(vc);
395       }
396       DRD_(trace_msg)("%pS", msg);
397       VG_(free)(msg);
398    }
399 
400    if (!  DRD_(get_check_stack_accesses)())
401    {
402       DRD_(finish_suppression)(DRD_(thread_get_stack_max)(drd_joinee)
403                                - DRD_(thread_get_stack_size)(drd_joinee),
404                                DRD_(thread_get_stack_max)(drd_joinee));
405    }
406    DRD_(clientobj_delete_thread)(drd_joinee);
407    DRD_(thread_delayed_delete)(drd_joinee);
408 }
409 
410 /**
411  * NPTL hack: NPTL allocates the 'struct pthread' on top of the stack,
412  * and accesses this data structure from multiple threads without locking.
413  * Any conflicting accesses in the range stack_startup..stack_max will be
414  * ignored.
415  */
DRD_(thread_set_stack_startup)416 void DRD_(thread_set_stack_startup)(const DrdThreadId tid,
417                                     const Addr stack_startup)
418 {
419    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
420              && tid != DRD_INVALID_THREADID);
421    tl_assert(DRD_(g_threadinfo)[tid].stack_min <= stack_startup);
422    tl_assert(stack_startup <= DRD_(g_threadinfo)[tid].stack_max);
423    DRD_(g_threadinfo)[tid].stack_startup = stack_startup;
424 }
425 
426 /** Return the stack pointer for the specified thread. */
DRD_(thread_get_stack_min)427 Addr DRD_(thread_get_stack_min)(const DrdThreadId tid)
428 {
429    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
430              && tid != DRD_INVALID_THREADID);
431    return DRD_(g_threadinfo)[tid].stack_min;
432 }
433 
434 /**
435  * Return the lowest value that was ever assigned to the stack pointer
436  * for the specified thread.
437  */
DRD_(thread_get_stack_min_min)438 Addr DRD_(thread_get_stack_min_min)(const DrdThreadId tid)
439 {
440    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
441              && tid != DRD_INVALID_THREADID);
442    return DRD_(g_threadinfo)[tid].stack_min_min;
443 }
444 
445 /** Return the top address for the stack of the specified thread. */
DRD_(thread_get_stack_max)446 Addr DRD_(thread_get_stack_max)(const DrdThreadId tid)
447 {
448    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
449              && tid != DRD_INVALID_THREADID);
450    return DRD_(g_threadinfo)[tid].stack_max;
451 }
452 
453 /** Return the maximum stack size for the specified thread. */
DRD_(thread_get_stack_size)454 SizeT DRD_(thread_get_stack_size)(const DrdThreadId tid)
455 {
456    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
457              && tid != DRD_INVALID_THREADID);
458    return DRD_(g_threadinfo)[tid].stack_size;
459 }
460 
DRD_(thread_get_on_alt_stack)461 Bool DRD_(thread_get_on_alt_stack)(const DrdThreadId tid)
462 {
463    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
464              && tid != DRD_INVALID_THREADID);
465    return DRD_(g_threadinfo)[tid].on_alt_stack;
466 }
467 
DRD_(thread_set_on_alt_stack)468 void DRD_(thread_set_on_alt_stack)(const DrdThreadId tid,
469                                    const Bool on_alt_stack)
470 {
471    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
472              && tid != DRD_INVALID_THREADID);
473    tl_assert(on_alt_stack == !!on_alt_stack);
474    DRD_(g_threadinfo)[tid].on_alt_stack = on_alt_stack;
475 }
476 
DRD_(thread_get_threads_on_alt_stack)477 Int DRD_(thread_get_threads_on_alt_stack)(void)
478 {
479    int i, n = 0;
480 
481    for (i = 1; i < DRD_N_THREADS; i++)
482       n += DRD_(g_threadinfo)[i].on_alt_stack;
483    return n;
484 }
485 
486 /**
487  * Clean up thread-specific data structures.
488  */
DRD_(thread_delete)489 void DRD_(thread_delete)(const DrdThreadId tid, const Bool detached)
490 {
491    Segment* sg;
492    Segment* sg_prev;
493 
494    tl_assert(DRD_(IsValidDrdThreadId)(tid));
495 
496    tl_assert(DRD_(g_threadinfo)[tid].synchr_nesting >= 0);
497    for (sg = DRD_(g_threadinfo)[tid].sg_last; sg; sg = sg_prev) {
498       sg_prev = sg->thr_prev;
499       sg->thr_next = NULL;
500       sg->thr_prev = NULL;
501       DRD_(sg_put)(sg);
502    }
503    DRD_(g_threadinfo)[tid].valid = False;
504    DRD_(g_threadinfo)[tid].vg_thread_exists = False;
505    DRD_(g_threadinfo)[tid].posix_thread_exists = False;
506    if (detached)
507       DRD_(g_threadinfo)[tid].detached_posix_thread = False;
508    else
509       tl_assert(!DRD_(g_threadinfo)[tid].detached_posix_thread);
510    DRD_(g_threadinfo)[tid].sg_first = NULL;
511    DRD_(g_threadinfo)[tid].sg_last = NULL;
512 
513    tl_assert(!DRD_(IsValidDrdThreadId)(tid));
514 }
515 
516 /**
517  * Called after a thread performed its last memory access and before
518  * thread_delete() is called. Note: thread_delete() is only called for
519  * joinable threads, not for detached threads.
520  */
DRD_(thread_finished)521 void DRD_(thread_finished)(const DrdThreadId tid)
522 {
523    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
524              && tid != DRD_INVALID_THREADID);
525 
526    DRD_(g_threadinfo)[tid].vg_thread_exists = False;
527 
528    if (DRD_(g_threadinfo)[tid].detached_posix_thread)
529    {
530       /*
531        * Once a detached thread has finished, its stack is deallocated and
532        * should no longer be taken into account when computing the conflict set.
533        */
534       DRD_(g_threadinfo)[tid].stack_min = DRD_(g_threadinfo)[tid].stack_max;
535 
536       /*
537        * For a detached thread, calling pthread_exit() invalidates the
538        * POSIX thread ID associated with the detached thread. For joinable
539        * POSIX threads however, the POSIX thread ID remains live after the
540        * pthread_exit() call until pthread_join() is called.
541        */
542       DRD_(g_threadinfo)[tid].posix_thread_exists = False;
543    }
544 }
545 
546 /** Called just after fork() in the child process. */
DRD_(drd_thread_atfork_child)547 void DRD_(drd_thread_atfork_child)(const DrdThreadId tid)
548 {
549    unsigned i;
550 
551    for (i = 1; i < DRD_N_THREADS; i++)
552    {
553       if (i == tid)
554 	 continue;
555       if (DRD_(IsValidDrdThreadId(i)))
556 	 DRD_(thread_delete)(i, True);
557       tl_assert(!DRD_(IsValidDrdThreadId(i)));
558    }
559 }
560 
561 /** Called just before pthread_cancel(). */
DRD_(thread_pre_cancel)562 void DRD_(thread_pre_cancel)(const DrdThreadId tid)
563 {
564    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
565              && tid != DRD_INVALID_THREADID);
566    tl_assert(DRD_(g_threadinfo)[tid].pt_threadid != INVALID_POSIX_THREADID);
567 
568    if (DRD_(thread_get_trace_fork_join)())
569       DRD_(trace_msg)("[%d] drd_thread_pre_cancel %d",
570                       DRD_(g_drd_running_tid), tid);
571 }
572 
573 /**
574  * Store the POSIX thread ID for the specified thread.
575  *
576  * @note This function can be called two times for the same thread -- see also
577  * the comment block preceding the pthread_create() wrapper in
578  * drd_pthread_intercepts.c.
579  */
DRD_(thread_set_pthreadid)580 void DRD_(thread_set_pthreadid)(const DrdThreadId tid, const PThreadId ptid)
581 {
582    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
583              && tid != DRD_INVALID_THREADID);
584    tl_assert(DRD_(g_threadinfo)[tid].pt_threadid == INVALID_POSIX_THREADID
585              || DRD_(g_threadinfo)[tid].pt_threadid == ptid);
586    tl_assert(ptid != INVALID_POSIX_THREADID);
587    DRD_(g_threadinfo)[tid].posix_thread_exists = True;
588    DRD_(g_threadinfo)[tid].pt_threadid         = ptid;
589 }
590 
591 /** Returns true for joinable threads and false for detached threads. */
DRD_(thread_get_joinable)592 Bool DRD_(thread_get_joinable)(const DrdThreadId tid)
593 {
594    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
595              && tid != DRD_INVALID_THREADID);
596    return ! DRD_(g_threadinfo)[tid].detached_posix_thread;
597 }
598 
599 /** Store the thread mode: joinable or detached. */
600 #if defined(VGP_mips32_linux)
601  /* There is a cse related issue in gcc for MIPS. Optimization level
602     has to be lowered, so cse related optimizations are not
603     included.*/
604  __attribute__((optimize("O1")))
605 #endif
DRD_(thread_set_joinable)606 void DRD_(thread_set_joinable)(const DrdThreadId tid, const Bool joinable)
607 {
608    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
609              && tid != DRD_INVALID_THREADID);
610    tl_assert(!! joinable == joinable);
611    tl_assert(DRD_(g_threadinfo)[tid].pt_threadid != INVALID_POSIX_THREADID);
612 
613    DRD_(g_threadinfo)[tid].detached_posix_thread = ! joinable;
614 }
615 
616 /** Tells DRD that the calling thread is about to enter pthread_create(). */
DRD_(thread_entering_pthread_create)617 void DRD_(thread_entering_pthread_create)(const DrdThreadId tid)
618 {
619    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
620              && tid != DRD_INVALID_THREADID);
621    tl_assert(DRD_(g_threadinfo)[tid].pt_threadid != INVALID_POSIX_THREADID);
622    tl_assert(DRD_(g_threadinfo)[tid].pthread_create_nesting_level >= 0);
623 
624    DRD_(g_threadinfo)[tid].pthread_create_nesting_level++;
625 }
626 
627 /** Tells DRD that the calling thread has left pthread_create(). */
DRD_(thread_left_pthread_create)628 void DRD_(thread_left_pthread_create)(const DrdThreadId tid)
629 {
630    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
631              && tid != DRD_INVALID_THREADID);
632    tl_assert(DRD_(g_threadinfo)[tid].pt_threadid != INVALID_POSIX_THREADID);
633    tl_assert(DRD_(g_threadinfo)[tid].pthread_create_nesting_level > 0);
634 
635    DRD_(g_threadinfo)[tid].pthread_create_nesting_level--;
636 }
637 
638 /** Obtain the thread number and the user-assigned thread name. */
DRD_(thread_get_name)639 const char* DRD_(thread_get_name)(const DrdThreadId tid)
640 {
641    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
642              && tid != DRD_INVALID_THREADID);
643 
644    return DRD_(g_threadinfo)[tid].name;
645 }
646 
647 /** Set the name of the specified thread. */
DRD_(thread_set_name)648 void DRD_(thread_set_name)(const DrdThreadId tid, const char* const name)
649 {
650    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
651              && tid != DRD_INVALID_THREADID);
652 
653    if (name == NULL || name[0] == 0)
654       VG_(snprintf)(DRD_(g_threadinfo)[tid].name,
655                     sizeof(DRD_(g_threadinfo)[tid].name),
656                     "Thread %d",
657                     tid);
658    else
659       VG_(snprintf)(DRD_(g_threadinfo)[tid].name,
660                     sizeof(DRD_(g_threadinfo)[tid].name),
661                     "Thread %d (%s)",
662                     tid, name);
663    DRD_(g_threadinfo)[tid].name[sizeof(DRD_(g_threadinfo)[tid].name) - 1] = 0;
664 }
665 
666 /**
667  * Update s_vg_running_tid, DRD_(g_drd_running_tid) and recalculate the
668  * conflict set.
669  */
DRD_(thread_set_vg_running_tid)670 void DRD_(thread_set_vg_running_tid)(const ThreadId vg_tid)
671 {
672    tl_assert(vg_tid != VG_INVALID_THREADID);
673 
674    if (vg_tid != s_vg_running_tid)
675    {
676       DRD_(thread_set_running_tid)(vg_tid,
677                                    DRD_(VgThreadIdToDrdThreadId)(vg_tid));
678    }
679 
680    tl_assert(s_vg_running_tid != VG_INVALID_THREADID);
681    tl_assert(DRD_(g_drd_running_tid) != DRD_INVALID_THREADID);
682 }
683 
684 /**
685  * Update s_vg_running_tid, DRD_(g_drd_running_tid) and recalculate the
686  * conflict set.
687  */
DRD_(thread_set_running_tid)688 void DRD_(thread_set_running_tid)(const ThreadId vg_tid,
689                                   const DrdThreadId drd_tid)
690 {
691    tl_assert(vg_tid != VG_INVALID_THREADID);
692    tl_assert(drd_tid != DRD_INVALID_THREADID);
693 
694    if (vg_tid != s_vg_running_tid)
695    {
696       if (s_trace_context_switches
697           && DRD_(g_drd_running_tid) != DRD_INVALID_THREADID)
698       {
699          VG_(message)(Vg_DebugMsg,
700                       "Context switch from thread %d to thread %d;"
701                       " segments: %llu\n",
702                       DRD_(g_drd_running_tid), drd_tid,
703                       DRD_(sg_get_segments_alive_count)());
704       }
705       s_vg_running_tid = vg_tid;
706       DRD_(g_drd_running_tid) = drd_tid;
707       thread_compute_conflict_set(&DRD_(g_conflict_set), drd_tid);
708       s_context_switch_count++;
709    }
710 
711    tl_assert(s_vg_running_tid != VG_INVALID_THREADID);
712    tl_assert(DRD_(g_drd_running_tid) != DRD_INVALID_THREADID);
713 }
714 
715 /**
716  * Increase the synchronization nesting counter. Must be called before the
717  * client calls a synchronization function.
718  */
DRD_(thread_enter_synchr)719 int DRD_(thread_enter_synchr)(const DrdThreadId tid)
720 {
721    tl_assert(DRD_(IsValidDrdThreadId)(tid));
722    return DRD_(g_threadinfo)[tid].synchr_nesting++;
723 }
724 
725 /**
726  * Decrease the synchronization nesting counter. Must be called after the
727  * client left a synchronization function.
728  */
DRD_(thread_leave_synchr)729 int DRD_(thread_leave_synchr)(const DrdThreadId tid)
730 {
731    tl_assert(DRD_(IsValidDrdThreadId)(tid));
732    tl_assert(DRD_(g_threadinfo)[tid].synchr_nesting >= 1);
733    return --DRD_(g_threadinfo)[tid].synchr_nesting;
734 }
735 
736 /** Returns the synchronization nesting counter. */
DRD_(thread_get_synchr_nesting_count)737 int DRD_(thread_get_synchr_nesting_count)(const DrdThreadId tid)
738 {
739    tl_assert(DRD_(IsValidDrdThreadId)(tid));
740    return DRD_(g_threadinfo)[tid].synchr_nesting;
741 }
742 
743 /** Append a new segment at the end of the segment list. */
744 static
thread_append_segment(const DrdThreadId tid,Segment * const sg)745 void thread_append_segment(const DrdThreadId tid, Segment* const sg)
746 {
747    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
748              && tid != DRD_INVALID_THREADID);
749 
750 #ifdef ENABLE_DRD_CONSISTENCY_CHECKS
751    tl_assert(DRD_(sane_ThreadInfo)(&DRD_(g_threadinfo)[tid]));
752 #endif
753 
754    // add at tail
755    sg->thr_prev = DRD_(g_threadinfo)[tid].sg_last;
756    sg->thr_next = NULL;
757    if (DRD_(g_threadinfo)[tid].sg_last)
758       DRD_(g_threadinfo)[tid].sg_last->thr_next = sg;
759    DRD_(g_threadinfo)[tid].sg_last = sg;
760    if (DRD_(g_threadinfo)[tid].sg_first == NULL)
761       DRD_(g_threadinfo)[tid].sg_first = sg;
762 
763 #ifdef ENABLE_DRD_CONSISTENCY_CHECKS
764    tl_assert(DRD_(sane_ThreadInfo)(&DRD_(g_threadinfo)[tid]));
765 #endif
766 }
767 
768 /**
769  * Remove a segment from the segment list of thread threadid, and free the
770  * associated memory.
771  */
772 static
thread_discard_segment(const DrdThreadId tid,Segment * const sg)773 void thread_discard_segment(const DrdThreadId tid, Segment* const sg)
774 {
775    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
776              && tid != DRD_INVALID_THREADID);
777 
778 #ifdef ENABLE_DRD_CONSISTENCY_CHECKS
779    tl_assert(DRD_(sane_ThreadInfo)(&DRD_(g_threadinfo)[tid]));
780 #endif
781 
782    if (sg->thr_prev)
783       sg->thr_prev->thr_next = sg->thr_next;
784    if (sg->thr_next)
785       sg->thr_next->thr_prev = sg->thr_prev;
786    if (sg == DRD_(g_threadinfo)[tid].sg_first)
787       DRD_(g_threadinfo)[tid].sg_first = sg->thr_next;
788    if (sg == DRD_(g_threadinfo)[tid].sg_last)
789       DRD_(g_threadinfo)[tid].sg_last = sg->thr_prev;
790    DRD_(sg_put)(sg);
791 
792 #ifdef ENABLE_DRD_CONSISTENCY_CHECKS
793    tl_assert(DRD_(sane_ThreadInfo)(&DRD_(g_threadinfo)[tid]));
794 #endif
795 }
796 
797 /**
798  * Returns a pointer to the vector clock of the most recent segment associated
799  * with thread 'tid'.
800  */
DRD_(thread_get_vc)801 VectorClock* DRD_(thread_get_vc)(const DrdThreadId tid)
802 {
803    Segment* latest_sg;
804 
805    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
806              && tid != DRD_INVALID_THREADID);
807    latest_sg = DRD_(g_threadinfo)[tid].sg_last;
808    tl_assert(latest_sg);
809    return &latest_sg->vc;
810 }
811 
812 /**
813  * Return the latest segment of thread 'tid' and increment its reference count.
814  */
DRD_(thread_get_latest_segment)815 void DRD_(thread_get_latest_segment)(Segment** sg, const DrdThreadId tid)
816 {
817    Segment* latest_sg;
818 
819    tl_assert(sg);
820    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
821              && tid != DRD_INVALID_THREADID);
822    latest_sg = DRD_(g_threadinfo)[tid].sg_last;
823    tl_assert(latest_sg);
824 
825    DRD_(sg_put)(*sg);
826    *sg = DRD_(sg_get)(latest_sg);
827 }
828 
829 /**
830  * Compute the minimum of all latest vector clocks of all threads
831  * (Michiel Ronsse calls this "clock snooping" in his papers about DIOTA).
832  *
833  * @param vc pointer to a vectorclock, holds result upon return.
834  */
DRD_(thread_compute_minimum_vc)835 static void DRD_(thread_compute_minimum_vc)(VectorClock* vc)
836 {
837    unsigned i;
838    Bool first;
839    Segment* latest_sg;
840 
841    first = True;
842    for (i = 0; i < DRD_N_THREADS; i++)
843    {
844       latest_sg = DRD_(g_threadinfo)[i].sg_last;
845       if (latest_sg) {
846          if (first)
847             DRD_(vc_assign)(vc, &latest_sg->vc);
848          else
849             DRD_(vc_min)(vc, &latest_sg->vc);
850          first = False;
851       }
852    }
853 }
854 
855 /**
856  * Compute the maximum of all latest vector clocks of all threads.
857  *
858  * @param vc pointer to a vectorclock, holds result upon return.
859  */
DRD_(thread_compute_maximum_vc)860 static void DRD_(thread_compute_maximum_vc)(VectorClock* vc)
861 {
862    unsigned i;
863    Bool first;
864    Segment* latest_sg;
865 
866    first = True;
867    for (i = 0; i < DRD_N_THREADS; i++)
868    {
869       latest_sg = DRD_(g_threadinfo)[i].sg_last;
870       if (latest_sg) {
871          if (first)
872             DRD_(vc_assign)(vc, &latest_sg->vc);
873          else
874             DRD_(vc_combine)(vc, &latest_sg->vc);
875          first = False;
876       }
877    }
878 }
879 
880 /**
881  * Discard all segments that have a defined order against the latest vector
882  * clock of all threads -- these segments can no longer be involved in a
883  * data race.
884  */
thread_discard_ordered_segments(void)885 static void thread_discard_ordered_segments(void)
886 {
887    unsigned i;
888    VectorClock thread_vc_min;
889 
890    s_discard_ordered_segments_count++;
891 
892    DRD_(vc_init)(&thread_vc_min, 0, 0);
893    DRD_(thread_compute_minimum_vc)(&thread_vc_min);
894    if (DRD_(sg_get_trace)())
895    {
896       char *vc_min, *vc_max;
897       VectorClock thread_vc_max;
898 
899       DRD_(vc_init)(&thread_vc_max, 0, 0);
900       DRD_(thread_compute_maximum_vc)(&thread_vc_max);
901       vc_min = DRD_(vc_aprint)(&thread_vc_min);
902       vc_max = DRD_(vc_aprint)(&thread_vc_max);
903       VG_(message)(Vg_DebugMsg,
904                    "Discarding ordered segments -- min vc is %s, max vc is %s\n",
905                    vc_min, vc_max);
906       VG_(free)(vc_min);
907       VG_(free)(vc_max);
908       DRD_(vc_cleanup)(&thread_vc_max);
909    }
910 
911    for (i = 0; i < DRD_N_THREADS; i++) {
912       Segment* sg;
913       Segment* sg_next;
914 
915       for (sg = DRD_(g_threadinfo)[i].sg_first;
916            sg && (sg_next = sg->thr_next)
917               && DRD_(vc_lte)(&sg->vc, &thread_vc_min);
918            sg = sg_next)
919       {
920          thread_discard_segment(i, sg);
921       }
922    }
923    DRD_(vc_cleanup)(&thread_vc_min);
924 }
925 
926 /**
927  * An implementation of the property 'equiv(sg1, sg2)' as defined in the paper
928  * by Mark Christiaens e.a. The property equiv(sg1, sg2) holds if and only if
929  * all segments in the set CS are ordered consistently against both sg1 and
930  * sg2. The set CS is defined as the set of segments that can immediately
931  * precede future segments via inter-thread synchronization operations. In
932  * DRD the set CS consists of the latest segment of each thread combined with
933  * all segments for which the reference count is strictly greater than one.
934  * The code below is an optimized version of the following:
935  *
936  * for (i = 0; i < DRD_N_THREADS; i++)
937  * {
938  *    Segment* sg;
939  *
940  *    for (sg = DRD_(g_threadinfo)[i].first; sg; sg = sg->next)
941  *    {
942  *       if (sg == DRD_(g_threadinfo)[i].last || DRD_(sg_get_refcnt)(sg) > 1)
943  *       {
944  *          if (   DRD_(vc_lte)(&sg1->vc, &sg->vc)
945  *              != DRD_(vc_lte)(&sg2->vc, &sg->vc)
946  *              || DRD_(vc_lte)(&sg->vc, &sg1->vc)
947  *              != DRD_(vc_lte)(&sg->vc, &sg2->vc))
948  *          {
949  *             return False;
950  *          }
951  *       }
952  *    }
953  * }
954  */
thread_consistent_segment_ordering(const DrdThreadId tid,Segment * const sg1,Segment * const sg2)955 static Bool thread_consistent_segment_ordering(const DrdThreadId tid,
956                                                Segment* const sg1,
957                                                Segment* const sg2)
958 {
959    unsigned i;
960 
961    tl_assert(sg1->thr_next);
962    tl_assert(sg2->thr_next);
963    tl_assert(sg1->thr_next == sg2);
964    tl_assert(DRD_(vc_lte)(&sg1->vc, &sg2->vc));
965 
966    for (i = 0; i < DRD_N_THREADS; i++)
967    {
968       Segment* sg;
969 
970       for (sg = DRD_(g_threadinfo)[i].sg_first; sg; sg = sg->thr_next) {
971          if (!sg->thr_next || DRD_(sg_get_refcnt)(sg) > 1) {
972             if (DRD_(vc_lte)(&sg2->vc, &sg->vc))
973                break;
974             if (DRD_(vc_lte)(&sg1->vc, &sg->vc))
975                return False;
976          }
977       }
978       for (sg = DRD_(g_threadinfo)[i].sg_last; sg; sg = sg->thr_prev) {
979          if (!sg->thr_next || DRD_(sg_get_refcnt)(sg) > 1) {
980             if (DRD_(vc_lte)(&sg->vc, &sg1->vc))
981                break;
982             if (DRD_(vc_lte)(&sg->vc, &sg2->vc))
983                return False;
984          }
985       }
986    }
987    return True;
988 }
989 
990 /**
991  * Merge all segments that may be merged without triggering false positives
992  * or discarding real data races. For the theoretical background of segment
993  * merging, see also the following paper: Mark Christiaens, Michiel Ronsse
994  * and Koen De Bosschere. Bounding the number of segment histories during
995  * data race detection. Parallel Computing archive, Volume 28, Issue 9,
996  * pp 1221-1238, September 2002. This paper contains a proof that merging
997  * consecutive segments for which the property equiv(s1,s2) holds can be
998  * merged without reducing the accuracy of datarace detection. Furthermore
999  * it is also proven that the total number of all segments will never grow
1000  * unbounded if all segments s1, s2 for which equiv(s1, s2) holds are merged
1001  * every time a new segment is created. The property equiv(s1, s2) is defined
1002  * as follows: equiv(s1, s2) <=> for all segments in the set CS, the vector
1003  * clocks of segments s and s1 are ordered in the same way as those of segments
1004  * s and s2. The set CS is defined as the set of existing segments s that have
1005  * the potential to conflict with not yet created segments, either because the
1006  * segment s is the latest segment of a thread or because it can become the
1007  * immediate predecessor of a new segment due to a synchronization operation.
1008  */
thread_merge_segments(void)1009 static void thread_merge_segments(void)
1010 {
1011    unsigned i;
1012 
1013    s_new_segments_since_last_merge = 0;
1014 
1015    for (i = 0; i < DRD_N_THREADS; i++)
1016    {
1017       Segment* sg;
1018 
1019 #ifdef ENABLE_DRD_CONSISTENCY_CHECKS
1020       tl_assert(DRD_(sane_ThreadInfo)(&DRD_(g_threadinfo)[i]));
1021 #endif
1022 
1023       for (sg = DRD_(g_threadinfo)[i].sg_first; sg; sg = sg->thr_next) {
1024          if (DRD_(sg_get_refcnt)(sg) == 1 && sg->thr_next) {
1025             Segment* const sg_next = sg->thr_next;
1026             if (DRD_(sg_get_refcnt)(sg_next) == 1
1027                 && sg_next->thr_next
1028                 && thread_consistent_segment_ordering(i, sg, sg_next))
1029             {
1030                /* Merge sg and sg_next into sg. */
1031                DRD_(sg_merge)(sg, sg_next);
1032                thread_discard_segment(i, sg_next);
1033             }
1034          }
1035       }
1036 
1037 #ifdef ENABLE_DRD_CONSISTENCY_CHECKS
1038       tl_assert(DRD_(sane_ThreadInfo)(&DRD_(g_threadinfo)[i]));
1039 #endif
1040    }
1041 }
1042 
1043 /**
1044  * Create a new segment for the specified thread, and discard any segments
1045  * that cannot cause races anymore.
1046  */
DRD_(thread_new_segment)1047 void DRD_(thread_new_segment)(const DrdThreadId tid)
1048 {
1049    Segment* last_sg;
1050    Segment* new_sg;
1051 
1052    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
1053              && tid != DRD_INVALID_THREADID);
1054    tl_assert(thread_conflict_set_up_to_date(DRD_(g_drd_running_tid)));
1055 
1056    last_sg = DRD_(g_threadinfo)[tid].sg_last;
1057    new_sg = DRD_(sg_new)(tid, tid);
1058    thread_append_segment(tid, new_sg);
1059    if (tid == DRD_(g_drd_running_tid) && last_sg)
1060    {
1061       DRD_(thread_update_conflict_set)(tid, &last_sg->vc);
1062       s_update_conflict_set_new_sg_count++;
1063    }
1064 
1065    tl_assert(thread_conflict_set_up_to_date(DRD_(g_drd_running_tid)));
1066 
1067    if (s_segment_merging
1068        && ++s_new_segments_since_last_merge >= s_segment_merge_interval)
1069    {
1070       thread_discard_ordered_segments();
1071       thread_merge_segments();
1072    }
1073 }
1074 
1075 /** Call this function after thread 'joiner' joined thread 'joinee'. */
DRD_(thread_combine_vc_join)1076 void DRD_(thread_combine_vc_join)(DrdThreadId joiner, DrdThreadId joinee)
1077 {
1078    tl_assert(joiner != joinee);
1079    tl_assert(0 <= (int)joiner && joiner < DRD_N_THREADS
1080              && joiner != DRD_INVALID_THREADID);
1081    tl_assert(0 <= (int)joinee && joinee < DRD_N_THREADS
1082              && joinee != DRD_INVALID_THREADID);
1083    tl_assert(DRD_(g_threadinfo)[joiner].sg_first);
1084    tl_assert(DRD_(g_threadinfo)[joiner].sg_last);
1085    tl_assert(DRD_(g_threadinfo)[joinee].sg_first);
1086    tl_assert(DRD_(g_threadinfo)[joinee].sg_last);
1087 
1088    if (DRD_(sg_get_trace)())
1089    {
1090       char *str1, *str2;
1091       str1 = DRD_(vc_aprint)(DRD_(thread_get_vc)(joiner));
1092       str2 = DRD_(vc_aprint)(DRD_(thread_get_vc)(joinee));
1093       VG_(message)(Vg_DebugMsg, "Before join: joiner %s, joinee %s\n",
1094                    str1, str2);
1095       VG_(free)(str1);
1096       VG_(free)(str2);
1097    }
1098    if (joiner == DRD_(g_drd_running_tid)) {
1099       VectorClock old_vc;
1100 
1101       DRD_(vc_copy)(&old_vc, DRD_(thread_get_vc)(joiner));
1102       DRD_(vc_combine)(DRD_(thread_get_vc)(joiner),
1103                        DRD_(thread_get_vc)(joinee));
1104       DRD_(thread_update_conflict_set)(joiner, &old_vc);
1105       s_update_conflict_set_join_count++;
1106       DRD_(vc_cleanup)(&old_vc);
1107    } else {
1108       DRD_(vc_combine)(DRD_(thread_get_vc)(joiner),
1109                        DRD_(thread_get_vc)(joinee));
1110    }
1111 
1112    thread_discard_ordered_segments();
1113 
1114    if (DRD_(sg_get_trace)()) {
1115       char* str;
1116 
1117       str = DRD_(vc_aprint)(DRD_(thread_get_vc)(joiner));
1118       VG_(message)(Vg_DebugMsg, "After join: %s\n", str);
1119       VG_(free)(str);
1120    }
1121 }
1122 
1123 /**
1124  * Update the vector clock of the last segment of thread tid with the
1125  * the vector clock of segment sg.
1126  */
thread_combine_vc_sync(DrdThreadId tid,const Segment * sg)1127 static void thread_combine_vc_sync(DrdThreadId tid, const Segment* sg)
1128 {
1129    const VectorClock* const vc = &sg->vc;
1130 
1131    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
1132              && tid != DRD_INVALID_THREADID);
1133    tl_assert(DRD_(g_threadinfo)[tid].sg_first);
1134    tl_assert(DRD_(g_threadinfo)[tid].sg_last);
1135    tl_assert(sg);
1136    tl_assert(vc);
1137 
1138    if (tid != sg->tid) {
1139       VectorClock old_vc;
1140 
1141       DRD_(vc_copy)(&old_vc, DRD_(thread_get_vc)(tid));
1142       DRD_(vc_combine)(DRD_(thread_get_vc)(tid), vc);
1143       if (DRD_(sg_get_trace)()) {
1144          char *str1, *str2;
1145          str1 = DRD_(vc_aprint)(&old_vc);
1146          str2 = DRD_(vc_aprint)(DRD_(thread_get_vc)(tid));
1147          VG_(message)(Vg_DebugMsg, "thread %d: vc %s -> %s\n", tid, str1, str2);
1148          VG_(free)(str1);
1149          VG_(free)(str2);
1150       }
1151 
1152       thread_discard_ordered_segments();
1153 
1154       DRD_(thread_update_conflict_set)(tid, &old_vc);
1155       s_update_conflict_set_sync_count++;
1156 
1157       DRD_(vc_cleanup)(&old_vc);
1158    } else {
1159       tl_assert(DRD_(vc_lte)(vc, DRD_(thread_get_vc)(tid)));
1160    }
1161 }
1162 
1163 /**
1164  * Create a new segment for thread tid and update the vector clock of the last
1165  * segment of this thread with the the vector clock of segment sg. Call this
1166  * function after thread tid had to wait because of thread synchronization
1167  * until the memory accesses in the segment sg finished.
1168  */
DRD_(thread_new_segment_and_combine_vc)1169 void DRD_(thread_new_segment_and_combine_vc)(DrdThreadId tid, const Segment* sg)
1170 {
1171    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
1172              && tid != DRD_INVALID_THREADID);
1173    tl_assert(thread_conflict_set_up_to_date(DRD_(g_drd_running_tid)));
1174    tl_assert(sg);
1175 
1176    thread_append_segment(tid, DRD_(sg_new)(tid, tid));
1177 
1178    thread_combine_vc_sync(tid, sg);
1179 
1180    if (s_segment_merging
1181        && ++s_new_segments_since_last_merge >= s_segment_merge_interval)
1182    {
1183       thread_discard_ordered_segments();
1184       thread_merge_segments();
1185    }
1186 }
1187 
1188 /**
1189  * Call this function whenever a thread is no longer using the memory
1190  * [ a1, a2 [, e.g. because of a call to free() or a stack pointer
1191  * increase.
1192  */
DRD_(thread_stop_using_mem)1193 void DRD_(thread_stop_using_mem)(const Addr a1, const Addr a2)
1194 {
1195    Segment* p;
1196 
1197    for (p = DRD_(g_sg_list); p; p = p->g_next)
1198       DRD_(bm_clear)(DRD_(sg_bm)(p), a1, a2);
1199 
1200    DRD_(bm_clear)(DRD_(g_conflict_set), a1, a2);
1201 }
1202 
1203 /** Specify whether memory loads should be recorded. */
DRD_(thread_set_record_loads)1204 void DRD_(thread_set_record_loads)(const DrdThreadId tid, const Bool enabled)
1205 {
1206    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
1207              && tid != DRD_INVALID_THREADID);
1208    tl_assert(enabled == !! enabled);
1209 
1210    DRD_(g_threadinfo)[tid].is_recording_loads = enabled;
1211 }
1212 
1213 /** Specify whether memory stores should be recorded. */
DRD_(thread_set_record_stores)1214 void DRD_(thread_set_record_stores)(const DrdThreadId tid, const Bool enabled)
1215 {
1216    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
1217              && tid != DRD_INVALID_THREADID);
1218    tl_assert(enabled == !! enabled);
1219 
1220    DRD_(g_threadinfo)[tid].is_recording_stores = enabled;
1221 }
1222 
1223 /**
1224  * Print the segment information for all threads.
1225  *
1226  * This function is only used for debugging purposes.
1227  */
DRD_(thread_print_all)1228 void DRD_(thread_print_all)(void)
1229 {
1230    unsigned i;
1231    Segment* p;
1232 
1233    for (i = 0; i < DRD_N_THREADS; i++)
1234    {
1235       p = DRD_(g_threadinfo)[i].sg_first;
1236       if (p) {
1237          VG_(printf)("**************\n"
1238                      "* thread %3d (%d/%d/%d/%d/0x%lx/%d) *\n"
1239                      "**************\n",
1240                      i,
1241                      DRD_(g_threadinfo)[i].valid,
1242                      DRD_(g_threadinfo)[i].vg_thread_exists,
1243                      DRD_(g_threadinfo)[i].vg_threadid,
1244                      DRD_(g_threadinfo)[i].posix_thread_exists,
1245                      DRD_(g_threadinfo)[i].pt_threadid,
1246                      DRD_(g_threadinfo)[i].detached_posix_thread);
1247          for ( ; p; p = p->thr_next)
1248             DRD_(sg_print)(p);
1249       }
1250    }
1251 }
1252 
1253 /** Show a call stack involved in a data race. */
show_call_stack(const DrdThreadId tid,ExeContext * const callstack)1254 static void show_call_stack(const DrdThreadId tid, ExeContext* const callstack)
1255 {
1256    const ThreadId vg_tid = DRD_(DrdThreadIdToVgThreadId)(tid);
1257 
1258    if (vg_tid != VG_INVALID_THREADID) {
1259       if (callstack)
1260          VG_(pp_ExeContext)(callstack);
1261       else
1262          VG_(get_and_pp_StackTrace)(vg_tid, VG_(clo_backtrace_size));
1263    } else {
1264       if (!VG_(clo_xml))
1265          VG_(message)(Vg_UserMsg,
1266                       "   (thread finished, call stack no longer available)\n");
1267    }
1268 }
1269 
1270 /** Print information about the segments involved in a data race. */
1271 static void
thread_report_conflicting_segments_segment(const DrdThreadId tid,const Addr addr,const SizeT size,const BmAccessTypeT access_type,const Segment * const p)1272 thread_report_conflicting_segments_segment(const DrdThreadId tid,
1273                                            const Addr addr,
1274                                            const SizeT size,
1275                                            const BmAccessTypeT access_type,
1276                                            const Segment* const p)
1277 {
1278    unsigned i;
1279 
1280    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
1281              && tid != DRD_INVALID_THREADID);
1282    tl_assert(p);
1283 
1284    for (i = 0; i < DRD_N_THREADS; i++) {
1285       if (i != tid) {
1286          Segment* q;
1287 
1288          for (q = DRD_(g_threadinfo)[i].sg_last; q; q = q->thr_prev) {
1289             /*
1290              * Since q iterates over the segments of thread i in order of
1291              * decreasing vector clocks, if q->vc <= p->vc, then
1292              * q->next->vc <= p->vc will also hold. Hence, break out of the
1293              * loop once this condition is met.
1294              */
1295             if (DRD_(vc_lte)(&q->vc, &p->vc))
1296                break;
1297             if (!DRD_(vc_lte)(&p->vc, &q->vc)) {
1298                if (DRD_(bm_has_conflict_with)(DRD_(sg_bm)(q), addr, addr + size,
1299                                               access_type)) {
1300                   Segment* q_next;
1301 
1302                   tl_assert(q->stacktrace);
1303                   if (VG_(clo_xml))
1304                      VG_(printf_xml)("  <other_segment_start>\n");
1305                   else
1306                      VG_(message)(Vg_UserMsg,
1307                                   "Other segment start (thread %d)\n", i);
1308                   show_call_stack(i, q->stacktrace);
1309                   if (VG_(clo_xml))
1310                      VG_(printf_xml)("  </other_segment_start>\n"
1311                                      "  <other_segment_end>\n");
1312                   else
1313                      VG_(message)(Vg_UserMsg,
1314                                   "Other segment end (thread %d)\n", i);
1315                   q_next = q->thr_next;
1316                   show_call_stack(i, q_next ? q_next->stacktrace : 0);
1317                   if (VG_(clo_xml))
1318                      VG_(printf_xml)("  </other_segment_end>\n");
1319                }
1320             }
1321          }
1322       }
1323    }
1324 }
1325 
1326 /** Print information about all segments involved in a data race. */
DRD_(thread_report_conflicting_segments)1327 void DRD_(thread_report_conflicting_segments)(const DrdThreadId tid,
1328                                               const Addr addr,
1329                                               const SizeT size,
1330                                               const BmAccessTypeT access_type)
1331 {
1332    Segment* p;
1333 
1334    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
1335              && tid != DRD_INVALID_THREADID);
1336 
1337    for (p = DRD_(g_threadinfo)[tid].sg_first; p; p = p->thr_next) {
1338       if (DRD_(bm_has)(DRD_(sg_bm)(p), addr, addr + size, access_type))
1339          thread_report_conflicting_segments_segment(tid, addr, size,
1340                                                     access_type, p);
1341    }
1342 }
1343 
1344 /**
1345  * Verify whether the conflict set for thread tid is up to date. Only perform
1346  * the check if the environment variable DRD_VERIFY_CONFLICT_SET has been set.
1347  */
thread_conflict_set_up_to_date(const DrdThreadId tid)1348 static Bool thread_conflict_set_up_to_date(const DrdThreadId tid)
1349 {
1350    static int do_verify_conflict_set = -1;
1351    Bool result;
1352    struct bitmap* computed_conflict_set = 0;
1353 
1354    if (do_verify_conflict_set < 0)
1355       do_verify_conflict_set = VG_(getenv)("DRD_VERIFY_CONFLICT_SET") != 0;
1356 
1357    if (do_verify_conflict_set == 0)
1358       return True;
1359 
1360    thread_compute_conflict_set(&computed_conflict_set, tid);
1361    result = DRD_(bm_equal)(DRD_(g_conflict_set), computed_conflict_set);
1362    if (! result)
1363    {
1364       VG_(printf)("actual conflict set:\n");
1365       DRD_(bm_print)(DRD_(g_conflict_set));
1366       VG_(printf)("\n");
1367       VG_(printf)("computed conflict set:\n");
1368       DRD_(bm_print)(computed_conflict_set);
1369       VG_(printf)("\n");
1370    }
1371    DRD_(bm_delete)(computed_conflict_set);
1372    return result;
1373 }
1374 
1375 /**
1376  * Compute the conflict set: a bitmap that represents the union of all memory
1377  * accesses of all segments that are unordered to the current segment of the
1378  * thread tid.
1379  */
thread_compute_conflict_set(struct bitmap ** conflict_set,const DrdThreadId tid)1380 static void thread_compute_conflict_set(struct bitmap** conflict_set,
1381                                         const DrdThreadId tid)
1382 {
1383    Segment* p;
1384 
1385    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
1386              && tid != DRD_INVALID_THREADID);
1387    tl_assert(tid == DRD_(g_drd_running_tid));
1388 
1389    s_compute_conflict_set_count++;
1390    s_conflict_set_bitmap_creation_count
1391       -= DRD_(bm_get_bitmap_creation_count)();
1392    s_conflict_set_bitmap2_creation_count
1393       -= DRD_(bm_get_bitmap2_creation_count)();
1394 
1395    if (*conflict_set) {
1396       DRD_(bm_cleanup)(*conflict_set);
1397       DRD_(bm_init)(*conflict_set);
1398    } else {
1399       *conflict_set = DRD_(bm_new)();
1400    }
1401 
1402    if (s_trace_conflict_set) {
1403       char* str;
1404 
1405       str = DRD_(vc_aprint)(DRD_(thread_get_vc)(tid));
1406       VG_(message)(Vg_DebugMsg,
1407                    "computing conflict set for thread %d with vc %s\n",
1408                    tid, str);
1409       VG_(free)(str);
1410    }
1411 
1412    p = DRD_(g_threadinfo)[tid].sg_last;
1413    {
1414       unsigned j;
1415 
1416       if (s_trace_conflict_set) {
1417          char* vc;
1418 
1419          vc = DRD_(vc_aprint)(&p->vc);
1420          VG_(message)(Vg_DebugMsg, "conflict set: thread [%d] at vc %s\n",
1421                       tid, vc);
1422          VG_(free)(vc);
1423       }
1424 
1425       for (j = 0; j < DRD_N_THREADS; j++) {
1426          if (j != tid && DRD_(IsValidDrdThreadId)(j)) {
1427             Segment* q;
1428 
1429             for (q = DRD_(g_threadinfo)[j].sg_last; q; q = q->thr_prev) {
1430                if (!DRD_(vc_lte)(&q->vc, &p->vc)
1431                    && !DRD_(vc_lte)(&p->vc, &q->vc)) {
1432                   if (s_trace_conflict_set) {
1433                      char* str;
1434 
1435                      str = DRD_(vc_aprint)(&q->vc);
1436                      VG_(message)(Vg_DebugMsg,
1437                                   "conflict set: [%d] merging segment %s\n",
1438                                   j, str);
1439                      VG_(free)(str);
1440                   }
1441                   DRD_(bm_merge2)(*conflict_set, DRD_(sg_bm)(q));
1442                } else {
1443                   if (s_trace_conflict_set) {
1444                      char* str;
1445 
1446                      str = DRD_(vc_aprint)(&q->vc);
1447                      VG_(message)(Vg_DebugMsg,
1448                                   "conflict set: [%d] ignoring segment %s\n",
1449                                   j, str);
1450                      VG_(free)(str);
1451                   }
1452                }
1453             }
1454          }
1455       }
1456    }
1457 
1458    s_conflict_set_bitmap_creation_count
1459       += DRD_(bm_get_bitmap_creation_count)();
1460    s_conflict_set_bitmap2_creation_count
1461       += DRD_(bm_get_bitmap2_creation_count)();
1462 
1463    if (s_trace_conflict_set_bm) {
1464       VG_(message)(Vg_DebugMsg, "[%d] new conflict set:\n", tid);
1465       DRD_(bm_print)(*conflict_set);
1466       VG_(message)(Vg_DebugMsg, "[%d] end of new conflict set.\n", tid);
1467    }
1468 }
1469 
1470 /**
1471  * Update the conflict set after the vector clock of thread tid has been
1472  * updated from old_vc to its current value, either because a new segment has
1473  * been created or because of a synchronization operation.
1474  */
DRD_(thread_update_conflict_set)1475 void DRD_(thread_update_conflict_set)(const DrdThreadId tid,
1476                                       const VectorClock* const old_vc)
1477 {
1478    const VectorClock* new_vc;
1479    Segment* p;
1480    unsigned j;
1481 
1482    tl_assert(0 <= (int)tid && tid < DRD_N_THREADS
1483              && tid != DRD_INVALID_THREADID);
1484    tl_assert(old_vc);
1485    tl_assert(tid == DRD_(g_drd_running_tid));
1486    tl_assert(DRD_(g_conflict_set));
1487 
1488    if (s_trace_conflict_set) {
1489       char* str;
1490 
1491       str = DRD_(vc_aprint)(DRD_(thread_get_vc)(tid));
1492       VG_(message)(Vg_DebugMsg,
1493                    "updating conflict set for thread %d with vc %s\n",
1494                    tid, str);
1495       VG_(free)(str);
1496    }
1497 
1498    new_vc = DRD_(thread_get_vc)(tid);
1499    tl_assert(DRD_(vc_lte)(old_vc, new_vc));
1500 
1501    DRD_(bm_unmark)(DRD_(g_conflict_set));
1502 
1503    for (j = 0; j < DRD_N_THREADS; j++)
1504    {
1505       Segment* q;
1506 
1507       if (j == tid || ! DRD_(IsValidDrdThreadId)(j))
1508          continue;
1509 
1510       for (q = DRD_(g_threadinfo)[j].sg_last;
1511            q && !DRD_(vc_lte)(&q->vc, new_vc);
1512            q = q->thr_prev) {
1513          const Bool included_in_old_conflict_set
1514             = !DRD_(vc_lte)(old_vc, &q->vc);
1515          const Bool included_in_new_conflict_set
1516             = !DRD_(vc_lte)(new_vc, &q->vc);
1517 
1518          if (UNLIKELY(s_trace_conflict_set)) {
1519             char* str;
1520 
1521             str = DRD_(vc_aprint)(&q->vc);
1522             VG_(message)(Vg_DebugMsg,
1523                          "conflict set: [%d] %s segment %s\n", j,
1524                          included_in_old_conflict_set
1525                          != included_in_new_conflict_set
1526                          ? "merging" : "ignoring", str);
1527             VG_(free)(str);
1528          }
1529          if (included_in_old_conflict_set != included_in_new_conflict_set)
1530             DRD_(bm_mark)(DRD_(g_conflict_set), DRD_(sg_bm)(q));
1531       }
1532 
1533       for ( ; q && !DRD_(vc_lte)(&q->vc, old_vc); q = q->thr_prev) {
1534          const Bool included_in_old_conflict_set
1535             = !DRD_(vc_lte)(old_vc, &q->vc);
1536          const Bool included_in_new_conflict_set
1537             = !DRD_(vc_lte)(&q->vc, new_vc)
1538             && !DRD_(vc_lte)(new_vc, &q->vc);
1539 
1540          if (UNLIKELY(s_trace_conflict_set)) {
1541             char* str;
1542 
1543             str = DRD_(vc_aprint)(&q->vc);
1544             VG_(message)(Vg_DebugMsg,
1545                          "conflict set: [%d] %s segment %s\n", j,
1546                          included_in_old_conflict_set
1547                          != included_in_new_conflict_set
1548                          ? "merging" : "ignoring", str);
1549             VG_(free)(str);
1550          }
1551          if (included_in_old_conflict_set != included_in_new_conflict_set)
1552             DRD_(bm_mark)(DRD_(g_conflict_set), DRD_(sg_bm)(q));
1553       }
1554    }
1555 
1556    DRD_(bm_clear_marked)(DRD_(g_conflict_set));
1557 
1558    p = DRD_(g_threadinfo)[tid].sg_last;
1559    for (j = 0; j < DRD_N_THREADS; j++) {
1560       if (j != tid && DRD_(IsValidDrdThreadId)(j)) {
1561          Segment* q;
1562          for (q = DRD_(g_threadinfo)[j].sg_last;
1563               q && !DRD_(vc_lte)(&q->vc, &p->vc);
1564               q = q->thr_prev) {
1565             if (!DRD_(vc_lte)(&p->vc, &q->vc))
1566                DRD_(bm_merge2_marked)(DRD_(g_conflict_set), DRD_(sg_bm)(q));
1567          }
1568       }
1569    }
1570 
1571    DRD_(bm_remove_cleared_marked)(DRD_(g_conflict_set));
1572 
1573    s_update_conflict_set_count++;
1574 
1575    if (s_trace_conflict_set_bm)
1576    {
1577       VG_(message)(Vg_DebugMsg, "[%d] updated conflict set:\n", tid);
1578       DRD_(bm_print)(DRD_(g_conflict_set));
1579       VG_(message)(Vg_DebugMsg, "[%d] end of updated conflict set.\n", tid);
1580    }
1581 
1582    tl_assert(thread_conflict_set_up_to_date(DRD_(g_drd_running_tid)));
1583 }
1584 
1585 /** Report the number of context switches performed. */
DRD_(thread_get_context_switch_count)1586 ULong DRD_(thread_get_context_switch_count)(void)
1587 {
1588    return s_context_switch_count;
1589 }
1590 
1591 /** Report the number of ordered segments that have been discarded. */
DRD_(thread_get_discard_ordered_segments_count)1592 ULong DRD_(thread_get_discard_ordered_segments_count)(void)
1593 {
1594    return s_discard_ordered_segments_count;
1595 }
1596 
1597 /** Return how many times the conflict set has been updated entirely. */
DRD_(thread_get_compute_conflict_set_count)1598 ULong DRD_(thread_get_compute_conflict_set_count)()
1599 {
1600    return s_compute_conflict_set_count;
1601 }
1602 
1603 /** Return how many times the conflict set has been updated partially. */
DRD_(thread_get_update_conflict_set_count)1604 ULong DRD_(thread_get_update_conflict_set_count)(void)
1605 {
1606    return s_update_conflict_set_count;
1607 }
1608 
1609 /**
1610  * Return how many times the conflict set has been updated partially
1611  * because a new segment has been created.
1612  */
DRD_(thread_get_update_conflict_set_new_sg_count)1613 ULong DRD_(thread_get_update_conflict_set_new_sg_count)(void)
1614 {
1615    return s_update_conflict_set_new_sg_count;
1616 }
1617 
1618 /**
1619  * Return how many times the conflict set has been updated partially
1620  * because of combining vector clocks due to synchronization operations
1621  * other than reader/writer lock or barrier operations.
1622  */
DRD_(thread_get_update_conflict_set_sync_count)1623 ULong DRD_(thread_get_update_conflict_set_sync_count)(void)
1624 {
1625    return s_update_conflict_set_sync_count;
1626 }
1627 
1628 /**
1629  * Return how many times the conflict set has been updated partially
1630  * because of thread joins.
1631  */
DRD_(thread_get_update_conflict_set_join_count)1632 ULong DRD_(thread_get_update_conflict_set_join_count)(void)
1633 {
1634    return s_update_conflict_set_join_count;
1635 }
1636 
1637 /**
1638  * Return the number of first-level bitmaps that have been created during
1639  * conflict set updates.
1640  */
DRD_(thread_get_conflict_set_bitmap_creation_count)1641 ULong DRD_(thread_get_conflict_set_bitmap_creation_count)(void)
1642 {
1643    return s_conflict_set_bitmap_creation_count;
1644 }
1645 
1646 /**
1647  * Return the number of second-level bitmaps that have been created during
1648  * conflict set updates.
1649  */
DRD_(thread_get_conflict_set_bitmap2_creation_count)1650 ULong DRD_(thread_get_conflict_set_bitmap2_creation_count)(void)
1651 {
1652    return s_conflict_set_bitmap2_creation_count;
1653 }
1654