• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 
2 /*--------------------------------------------------------------------*/
3 /*--- Error management for Helgrind.                               ---*/
4 /*---                                                  hg_errors.c ---*/
5 /*--------------------------------------------------------------------*/
6 
7 /*
8    This file is part of Helgrind, a Valgrind tool for detecting errors
9    in threaded programs.
10 
11    Copyright (C) 2007-2012 OpenWorks Ltd
12       info@open-works.co.uk
13 
14    This program is free software; you can redistribute it and/or
15    modify it under the terms of the GNU General Public License as
16    published by the Free Software Foundation; either version 2 of the
17    License, or (at your option) any later version.
18 
19    This program is distributed in the hope that it will be useful, but
20    WITHOUT ANY WARRANTY; without even the implied warranty of
21    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
22    General Public License for more details.
23 
24    You should have received a copy of the GNU General Public License
25    along with this program; if not, write to the Free Software
26    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
27    02111-1307, USA.
28 
29    The GNU General Public License is contained in the file COPYING.
30 */
31 
32 #include "pub_tool_basics.h"
33 #include "pub_tool_libcbase.h"
34 #include "pub_tool_libcassert.h"
35 #include "pub_tool_libcprint.h"
36 #include "pub_tool_execontext.h"
37 #include "pub_tool_errormgr.h"
38 #include "pub_tool_wordfm.h"
39 #include "pub_tool_xarray.h"
40 #include "pub_tool_debuginfo.h"
41 #include "pub_tool_threadstate.h"
42 #include "pub_tool_options.h"     // VG_(clo_xml)
43 
44 #include "hg_basics.h"
45 #include "hg_wordset.h"
46 #include "hg_lock_n_thread.h"
47 #include "libhb.h"
48 #include "hg_errors.h"            /* self */
49 
50 
51 /*----------------------------------------------------------------*/
52 /*--- Error management -- storage                              ---*/
53 /*----------------------------------------------------------------*/
54 
55 /* maps (by value) strings to a copy of them in ARENA_TOOL */
56 
57 static WordFM* string_table = NULL;
58 
59 ULong HG_(stats__string_table_queries) = 0;
60 
HG_(stats__string_table_get_map_size)61 ULong HG_(stats__string_table_get_map_size) ( void ) {
62    return string_table ? (ULong)VG_(sizeFM)(string_table) : 0;
63 }
64 
string_table_cmp(UWord s1,UWord s2)65 static Word string_table_cmp ( UWord s1, UWord s2 ) {
66    return (Word)VG_(strcmp)( (HChar*)s1, (HChar*)s2 );
67 }
68 
string_table_strdup(HChar * str)69 static HChar* string_table_strdup ( HChar* str ) {
70    HChar* copy = NULL;
71    HG_(stats__string_table_queries)++;
72    if (!str)
73       str = "(null)";
74    if (!string_table) {
75       string_table = VG_(newFM)( HG_(zalloc), "hg.sts.1",
76                                  HG_(free), string_table_cmp );
77       tl_assert(string_table);
78    }
79    if (VG_(lookupFM)( string_table,
80                       NULL, (Word*)&copy, (Word)str )) {
81       tl_assert(copy);
82       if (0) VG_(printf)("string_table_strdup: %p -> %p\n", str, copy );
83       return copy;
84    } else {
85       copy = HG_(strdup)("hg.sts.2", str);
86       tl_assert(copy);
87       VG_(addToFM)( string_table, (Word)copy, (Word)copy );
88       return copy;
89    }
90 }
91 
92 /* maps from Lock .unique fields to LockP*s */
93 
94 static WordFM* map_LockN_to_P = NULL;
95 
96 ULong HG_(stats__LockN_to_P_queries) = 0;
97 
HG_(stats__LockN_to_P_get_map_size)98 ULong HG_(stats__LockN_to_P_get_map_size) ( void ) {
99    return map_LockN_to_P ? (ULong)VG_(sizeFM)(map_LockN_to_P) : 0;
100 }
101 
lock_unique_cmp(UWord lk1W,UWord lk2W)102 static Word lock_unique_cmp ( UWord lk1W, UWord lk2W )
103 {
104    Lock* lk1 = (Lock*)lk1W;
105    Lock* lk2 = (Lock*)lk2W;
106    tl_assert( HG_(is_sane_LockNorP)(lk1) );
107    tl_assert( HG_(is_sane_LockNorP)(lk2) );
108    if (lk1->unique < lk2->unique) return -1;
109    if (lk1->unique > lk2->unique) return 1;
110    return 0;
111 }
112 
113 /* Given a normal Lock (LockN), convert it to a persistent Lock
114    (LockP).  In some cases the LockN could be invalid (if it's been
115    freed), so we enquire, in hg_main.c's admin_locks list, whether it
116    is in fact valid.  If allowed_to_be_invalid is True, then it's OK
117    for the LockN to be invalid, in which case Lock_INVALID is
118    returned.  In all other cases, we insist that the LockN is a valid
119    lock, and return its corresponding LockP.
120 
121    Why can LockNs sometimes be invalid?  Because they are harvested
122    from locksets that are attached to the OldRef info for conflicting
123    threads.  By the time we detect a race, the some of the elements of
124    the lockset may have been destroyed by the client, in which case
125    the corresponding Lock structures we maintain will have been freed.
126 
127    So we check that each LockN is a member of the admin_locks double
128    linked list of all Lock structures.  That stops us prodding around
129    in potentially freed-up Lock structures.  However, it's not quite a
130    proper check: if a new Lock has been reallocated at the same
131    address as one which was previously freed, we'll wind up copying
132    the new one as the basis for the LockP, which is completely bogus
133    because it is unrelated to the previous Lock that lived there.
134    Let's hope that doesn't happen too often.
135 */
mk_LockP_from_LockN(Lock * lkn,Bool allowed_to_be_invalid)136 static Lock* mk_LockP_from_LockN ( Lock* lkn,
137                                    Bool allowed_to_be_invalid )
138 {
139    Lock* lkp = NULL;
140    HG_(stats__LockN_to_P_queries)++;
141 
142    /* First off, let's do some sanity checks.  If
143       allowed_to_be_invalid is False, we _must_ be able to find 'lkn'
144       in admin_locks; else we must assert.  If it is True, it's OK for
145       it not to be findable, but in that case we must return
146       Lock_INVALID right away. */
147    Lock* lock_list = HG_(get_admin_locks)();
148    while (lock_list) {
149       if (lock_list == lkn)
150          break;
151       lock_list = lock_list->admin_next;
152    }
153    if (lock_list == NULL) {
154       /* We didn't find it.  That possibility has to be OK'd by the
155          caller. */
156       tl_assert(allowed_to_be_invalid);
157       return Lock_INVALID;
158    }
159 
160    /* So we must be looking at a valid LockN. */
161    tl_assert( HG_(is_sane_LockN)(lkn) );
162 
163    if (!map_LockN_to_P) {
164       map_LockN_to_P = VG_(newFM)( HG_(zalloc), "hg.mLPfLN.1",
165                                    HG_(free), lock_unique_cmp );
166       tl_assert(map_LockN_to_P);
167    }
168    if (!VG_(lookupFM)( map_LockN_to_P, NULL, (Word*)&lkp, (Word)lkn)) {
169       lkp = HG_(zalloc)( "hg.mLPfLN.2", sizeof(Lock) );
170       *lkp = *lkn;
171       lkp->admin_next = NULL;
172       lkp->admin_prev = NULL;
173       lkp->magic = LockP_MAGIC;
174       /* Forget about the bag of lock holders - don't copy that.
175          Also, acquired_at should be NULL whenever heldBy is, and vice
176          versa.  Also forget about the associated libhb synch object. */
177       lkp->heldW  = False;
178       lkp->heldBy = NULL;
179       lkp->acquired_at = NULL;
180       lkp->hbso = NULL;
181       VG_(addToFM)( map_LockN_to_P, (Word)lkp, (Word)lkp );
182    }
183    tl_assert( HG_(is_sane_LockP)(lkp) );
184    return lkp;
185 }
186 
187 /* Expand a WordSet of LockN*'s into a NULL-terminated vector of
188    LockP*'s.  Any LockN's that can't be converted into a LockP
189    (because they have been freed, see comment on mk_LockP_from_LockN)
190    are converted instead into the value Lock_INVALID.  Hence the
191    returned vector is a sequence: zero or more (valid LockP* or
192    LockN_INVALID), terminated by a NULL. */
193 static
enumerate_WordSet_into_LockP_vector(WordSetU * univ_lsets,WordSetID lockset,Bool allowed_to_be_invalid)194 Lock** enumerate_WordSet_into_LockP_vector( WordSetU* univ_lsets,
195                                             WordSetID lockset,
196                                             Bool allowed_to_be_invalid )
197 {
198    tl_assert(univ_lsets);
199    tl_assert( HG_(plausibleWS)(univ_lsets, lockset) );
200    UWord  nLocks = HG_(cardinalityWS)(univ_lsets, lockset);
201    Lock** lockPs = HG_(zalloc)( "hg.eWSiLPa",
202                                 (nLocks+1) * sizeof(Lock*) );
203    tl_assert(lockPs);
204    tl_assert(lockPs[nLocks] == NULL); /* pre-NULL terminated */
205    UWord* lockNs  = NULL;
206    UWord  nLockNs = 0;
207    if (nLocks > 0)  {
208       /* HG_(getPayloadWS) doesn't assign non-NULL to &lockNs if the
209          lockset is empty; hence the guarding "if".  Sigh. */
210       HG_(getPayloadWS)( &lockNs, &nLockNs, univ_lsets, lockset );
211       tl_assert(lockNs);
212    }
213    UWord i;
214    /* Convert to LockPs. */
215    for (i = 0; i < nLockNs; i++) {
216       lockPs[i] = mk_LockP_from_LockN( (Lock*)lockNs[i],
217                                        allowed_to_be_invalid );
218    }
219    return lockPs;
220 }
221 
222 /* Get the number of useful elements in a vector created by
223    enumerate_WordSet_into_LockP_vector.  Returns both the total number
224    of elements (not including the terminating NULL) and the number of
225    non-Lock_INVALID elements. */
count_LockP_vector(UWord * nLocks,UWord * nLocksValid,Lock ** vec)226 static void count_LockP_vector ( /*OUT*/UWord* nLocks,
227                                  /*OUT*/UWord* nLocksValid,
228                                  Lock** vec )
229 {
230    tl_assert(vec);
231    *nLocks = *nLocksValid = 0;
232    UWord n = 0;
233    while (vec[n]) {
234       (*nLocks)++;
235       if (vec[n] != Lock_INVALID)
236          (*nLocksValid)++;
237       n++;
238    }
239 }
240 
241 /* Find out whether 'lk' is in 'vec'. */
elem_LockP_vector(Lock ** vec,Lock * lk)242 static Bool elem_LockP_vector ( Lock** vec, Lock* lk )
243 {
244    tl_assert(vec);
245    tl_assert(lk);
246    UWord n = 0;
247    while (vec[n]) {
248       if (vec[n] == lk)
249          return True;
250       n++;
251    }
252    return False;
253 }
254 
255 
256 /* Errors:
257 
258       race: program counter
259             read or write
260             data size
261             previous state
262             current state
263 
264       FIXME: how does state printing interact with lockset gc?
265       Are the locksets in prev/curr state always valid?
266       Ditto question for the threadsets
267           ThreadSets - probably are always valid if Threads
268           are never thrown away.
269           LockSets - could at least print the lockset elements that
270           correspond to actual locks at the time of printing.  Hmm.
271 */
272 
273 /* Error kinds */
274 typedef
275    enum {
276       XE_Race=1101,      // race
277       XE_UnlockUnlocked, // unlocking a not-locked lock
278       XE_UnlockForeign,  // unlocking a lock held by some other thread
279       XE_UnlockBogus,    // unlocking an address not known to be a lock
280       XE_PthAPIerror,    // error from the POSIX pthreads API
281       XE_LockOrder,      // lock order error
282       XE_Misc            // misc other error (w/ string to describe it)
283    }
284    XErrorTag;
285 
286 /* Extra contexts for kinds */
287 typedef
288    struct  {
289       XErrorTag tag;
290       union {
291          struct {
292             Addr        data_addr;
293             Int         szB;
294             Bool        isWrite;
295             Thread*     thr;
296             Lock**      locksHeldW;
297             /* descr1/2 provide a description of stack/global locs */
298             XArray*     descr1; /* XArray* of HChar */
299             XArray*     descr2; /* XArray* of HChar */
300             /* halloc/haddr/hszB describe the addr if it is a heap block. */
301             ExeContext* hctxt;
302             Addr        haddr;
303             SizeT       hszB;
304             /* h1_* and h2_* provide some description of a previously
305                observed access with which we are conflicting. */
306             Thread*     h1_ct; /* non-NULL means h1 info present */
307             ExeContext* h1_ct_mbsegstartEC;
308             ExeContext* h1_ct_mbsegendEC;
309             Thread*     h2_ct; /* non-NULL means h2 info present */
310             ExeContext* h2_ct_accEC;
311             Int         h2_ct_accSzB;
312             Bool        h2_ct_accIsW;
313             Lock**      h2_ct_locksHeldW;
314          } Race;
315          struct {
316             Thread* thr;  /* doing the unlocking */
317             Lock*   lock; /* lock (that is already unlocked) */
318          } UnlockUnlocked;
319          struct {
320             Thread* thr;    /* doing the unlocking */
321             Thread* owner;  /* thread that actually holds the lock */
322             Lock*   lock;   /* lock (that is held by 'owner') */
323          } UnlockForeign;
324          struct {
325             Thread* thr;     /* doing the unlocking */
326             Addr    lock_ga; /* purported address of the lock */
327          } UnlockBogus;
328          struct {
329             Thread* thr;
330             HChar*  fnname; /* persistent, in tool-arena */
331             Word    err;    /* pth error code */
332             HChar*  errstr; /* persistent, in tool-arena */
333          } PthAPIerror;
334          struct {
335             Thread*     thr;
336             /* The first 4 fields describe the previously observed
337                (should-be) ordering. */
338             Addr        shouldbe_earlier_ga;
339             Addr        shouldbe_later_ga;
340             ExeContext* shouldbe_earlier_ec;
341             ExeContext* shouldbe_later_ec;
342             /* In principle we need to record two more stacks, from
343                this thread, when acquiring the locks in the "wrong"
344                order.  In fact the wallclock-later acquisition by this
345                thread is recorded in the main stack for this error.
346                So we only need a stack for the earlier acquisition by
347                this thread. */
348             ExeContext* actual_earlier_ec;
349          } LockOrder;
350          struct {
351             Thread*     thr;
352             HChar*      errstr; /* persistent, in tool-arena */
353             HChar*      auxstr; /* optional, persistent, in tool-arena */
354             ExeContext* auxctx; /* optional */
355          } Misc;
356       } XE;
357    }
358    XError;
359 
init_XError(XError * xe)360 static void init_XError ( XError* xe ) {
361    VG_(memset)(xe, 0, sizeof(*xe) );
362    xe->tag = XE_Race-1; /* bogus */
363 }
364 
365 
366 /* Extensions of suppressions */
367 typedef
368    enum {
369       XS_Race=1201, /* race */
370       XS_FreeMemLock,
371       XS_UnlockUnlocked,
372       XS_UnlockForeign,
373       XS_UnlockBogus,
374       XS_PthAPIerror,
375       XS_LockOrder,
376       XS_Misc
377    }
378    XSuppTag;
379 
380 
381 /* Updates the copy with address info if necessary. */
HG_(update_extra)382 UInt HG_(update_extra) ( Error* err )
383 {
384    XError* xe = (XError*)VG_(get_error_extra)(err);
385    tl_assert(xe);
386    //if (extra != NULL && Undescribed == extra->addrinfo.akind) {
387    //   describe_addr ( VG_(get_error_address)(err), &(extra->addrinfo) );
388    //}
389 
390    if (xe->tag == XE_Race) {
391 
392       /* Note the set of locks that the thread is (w-)holding.
393          Convert the WordSetID of LockN*'s into a NULL-terminated
394          vector of LockP*'s.  We don't expect to encounter any invalid
395          LockNs in this conversion. */
396       tl_assert(xe->XE.Race.thr);
397       xe->XE.Race.locksHeldW
398          = enumerate_WordSet_into_LockP_vector(
399               HG_(get_univ_lsets)(),
400               xe->XE.Race.thr->locksetW,
401               False/*!allowed_to_be_invalid*/
402            );
403 
404       /* See if we can come up with a source level description of the
405          raced-upon address.  This is potentially expensive, which is
406          why it's only done at the update_extra point, not when the
407          error is initially created. */
408       static Int xxx = 0;
409       xxx++;
410       if (0)
411          VG_(printf)("HG_(update_extra): "
412                      "%d conflicting-event queries\n", xxx);
413 
414       tl_assert(!xe->XE.Race.hctxt);
415       tl_assert(!xe->XE.Race.descr1);
416       tl_assert(!xe->XE.Race.descr2);
417 
418       /* First, see if it's in any heap block.  Unfortunately this
419          means a linear search through all allocated heap blocks.  The
420          assertion says that if it's detected as a heap block, then we
421          must have an allocation context for it, since all heap blocks
422          should have an allocation context. */
423       Bool is_heapblock
424          = HG_(mm_find_containing_block)(
425               &xe->XE.Race.hctxt, &xe->XE.Race.haddr, &xe->XE.Race.hszB,
426               xe->XE.Race.data_addr
427            );
428       tl_assert(is_heapblock == (xe->XE.Race.hctxt != NULL));
429 
430       if (!xe->XE.Race.hctxt) {
431          /* It's not in any heap block.  See if we can map it to a
432             stack or global symbol. */
433 
434          xe->XE.Race.descr1
435             = VG_(newXA)( HG_(zalloc), "hg.update_extra.Race.descr1",
436                           HG_(free), sizeof(HChar) );
437          xe->XE.Race.descr2
438             = VG_(newXA)( HG_(zalloc), "hg.update_extra.Race.descr2",
439                           HG_(free), sizeof(HChar) );
440 
441          (void) VG_(get_data_description)( xe->XE.Race.descr1,
442                                            xe->XE.Race.descr2,
443                                            xe->XE.Race.data_addr );
444 
445          /* If there's nothing in descr1/2, free it.  Why is it safe to
446             to VG_(indexXA) at zero here?  Because
447             VG_(get_data_description) guarantees to zero terminate
448             descr1/2 regardless of the outcome of the call.  So there's
449             always at least one element in each XA after the call.
450          */
451          if (0 == VG_(strlen)( VG_(indexXA)( xe->XE.Race.descr1, 0 ))) {
452             VG_(deleteXA)( xe->XE.Race.descr1 );
453             xe->XE.Race.descr1 = NULL;
454          }
455          if (0 == VG_(strlen)( VG_(indexXA)( xe->XE.Race.descr2, 0 ))) {
456             VG_(deleteXA)( xe->XE.Race.descr2 );
457             xe->XE.Race.descr2 = NULL;
458          }
459       }
460 
461       /* And poke around in the conflicting-event map, to see if we
462          can rustle up a plausible-looking conflicting memory access
463          to show. */
464       if (HG_(clo_history_level) >= 2) {
465          Thr*        thrp            = NULL;
466          ExeContext* wherep          = NULL;
467          Addr        acc_addr        = xe->XE.Race.data_addr;
468          Int         acc_szB         = xe->XE.Race.szB;
469          Thr*        acc_thr         = xe->XE.Race.thr->hbthr;
470          Bool        acc_isW         = xe->XE.Race.isWrite;
471          SizeT       conf_szB        = 0;
472          Bool        conf_isW        = False;
473          WordSetID   conf_locksHeldW = 0;
474          tl_assert(!xe->XE.Race.h2_ct_accEC);
475          tl_assert(!xe->XE.Race.h2_ct);
476          if (libhb_event_map_lookup(
477                 &wherep, &thrp, &conf_szB, &conf_isW, &conf_locksHeldW,
478                 acc_thr, acc_addr, acc_szB, acc_isW )) {
479             Thread* threadp;
480             tl_assert(wherep);
481             tl_assert(thrp);
482             threadp = libhb_get_Thr_hgthread( thrp );
483             tl_assert(threadp);
484             xe->XE.Race.h2_ct_accEC  = wherep;
485             xe->XE.Race.h2_ct        = threadp;
486             xe->XE.Race.h2_ct_accSzB = (Int)conf_szB;
487             xe->XE.Race.h2_ct_accIsW = conf_isW;
488             xe->XE.Race.h2_ct_locksHeldW
489                = enumerate_WordSet_into_LockP_vector(
490                     HG_(get_univ_lsets)(),
491                     conf_locksHeldW,
492                     True/*allowed_to_be_invalid*/
493                  );
494         }
495       }
496 
497       // both NULL or both non-NULL
498       tl_assert( (!!xe->XE.Race.h2_ct) == (!!xe->XE.Race.h2_ct_accEC) );
499    }
500 
501    return sizeof(XError);
502 }
503 
HG_(record_error_Race)504 void HG_(record_error_Race) ( Thread* thr,
505                               Addr data_addr, Int szB, Bool isWrite,
506                               Thread* h1_ct,
507                               ExeContext* h1_ct_segstart,
508                               ExeContext* h1_ct_mbsegendEC )
509 {
510    XError xe;
511    tl_assert( HG_(is_sane_Thread)(thr) );
512 
513 #  if defined(VGO_linux)
514    /* Skip any races on locations apparently in GOTPLT sections.  This
515       is said to be caused by ld.so poking PLT table entries (or
516       whatever) when it writes the resolved address of a dynamically
517       linked routine, into the table (or whatever) when it is called
518       for the first time. */
519    {
520      VgSectKind sect = VG_(DebugInfo_sect_kind)( NULL, 0, data_addr );
521      if (0) VG_(printf)("XXXXXXXXX RACE on %#lx %s\n",
522                         data_addr, VG_(pp_SectKind)(sect));
523      /* SectPLT is required on ???-linux */
524      if (sect == Vg_SectGOTPLT) return;
525      /* SectPLT is required on ppc32/64-linux */
526      if (sect == Vg_SectPLT) return;
527    }
528 #  endif
529 
530    init_XError(&xe);
531    xe.tag = XE_Race;
532    xe.XE.Race.data_addr   = data_addr;
533    xe.XE.Race.szB         = szB;
534    xe.XE.Race.isWrite     = isWrite;
535    xe.XE.Race.thr         = thr;
536    tl_assert(isWrite == False || isWrite == True);
537    tl_assert(szB == 8 || szB == 4 || szB == 2 || szB == 1);
538    /* Skip on the detailed description of the raced-on address at this
539       point; it's expensive.  Leave it for the update_extra function
540       if we ever make it that far. */
541    tl_assert(xe.XE.Race.descr1 == NULL);
542    tl_assert(xe.XE.Race.descr2 == NULL);
543    // FIXME: tid vs thr
544    // Skip on any of the conflicting-access info at this point.
545    // It's expensive to obtain, and this error is more likely than
546    // not to be discarded.  We'll fill these fields in in
547    // HG_(update_extra) just above, assuming the error ever makes
548    // it that far (unlikely).
549    xe.XE.Race.h2_ct_accSzB = 0;
550    xe.XE.Race.h2_ct_accIsW = False;
551    xe.XE.Race.h2_ct_accEC  = NULL;
552    xe.XE.Race.h2_ct        = NULL;
553    tl_assert( HG_(is_sane_ThreadId)(thr->coretid) );
554    tl_assert( thr->coretid != VG_INVALID_THREADID );
555 
556    xe.XE.Race.h1_ct              = h1_ct;
557    xe.XE.Race.h1_ct_mbsegstartEC = h1_ct_segstart;
558    xe.XE.Race.h1_ct_mbsegendEC   = h1_ct_mbsegendEC;
559 
560    VG_(maybe_record_error)( thr->coretid,
561                             XE_Race, data_addr, NULL, &xe );
562 }
563 
HG_(record_error_UnlockUnlocked)564 void HG_(record_error_UnlockUnlocked) ( Thread* thr, Lock* lk )
565 {
566    XError xe;
567    tl_assert( HG_(is_sane_Thread)(thr) );
568    tl_assert( HG_(is_sane_LockN)(lk) );
569    init_XError(&xe);
570    xe.tag = XE_UnlockUnlocked;
571    xe.XE.UnlockUnlocked.thr
572       = thr;
573    xe.XE.UnlockUnlocked.lock
574       = mk_LockP_from_LockN(lk, False/*!allowed_to_be_invalid*/);
575    // FIXME: tid vs thr
576    tl_assert( HG_(is_sane_ThreadId)(thr->coretid) );
577    tl_assert( thr->coretid != VG_INVALID_THREADID );
578    VG_(maybe_record_error)( thr->coretid,
579                             XE_UnlockUnlocked, 0, NULL, &xe );
580 }
581 
HG_(record_error_UnlockForeign)582 void HG_(record_error_UnlockForeign) ( Thread* thr,
583                                        Thread* owner, Lock* lk )
584 {
585    XError xe;
586    tl_assert( HG_(is_sane_Thread)(thr) );
587    tl_assert( HG_(is_sane_Thread)(owner) );
588    tl_assert( HG_(is_sane_LockN)(lk) );
589    init_XError(&xe);
590    xe.tag = XE_UnlockForeign;
591    xe.XE.UnlockForeign.thr   = thr;
592    xe.XE.UnlockForeign.owner = owner;
593    xe.XE.UnlockForeign.lock
594       = mk_LockP_from_LockN(lk, False/*!allowed_to_be_invalid*/);
595    // FIXME: tid vs thr
596    tl_assert( HG_(is_sane_ThreadId)(thr->coretid) );
597    tl_assert( thr->coretid != VG_INVALID_THREADID );
598    VG_(maybe_record_error)( thr->coretid,
599                             XE_UnlockForeign, 0, NULL, &xe );
600 }
601 
HG_(record_error_UnlockBogus)602 void HG_(record_error_UnlockBogus) ( Thread* thr, Addr lock_ga )
603 {
604    XError xe;
605    tl_assert( HG_(is_sane_Thread)(thr) );
606    init_XError(&xe);
607    xe.tag = XE_UnlockBogus;
608    xe.XE.UnlockBogus.thr     = thr;
609    xe.XE.UnlockBogus.lock_ga = lock_ga;
610    // FIXME: tid vs thr
611    tl_assert( HG_(is_sane_ThreadId)(thr->coretid) );
612    tl_assert( thr->coretid != VG_INVALID_THREADID );
613    VG_(maybe_record_error)( thr->coretid,
614                             XE_UnlockBogus, 0, NULL, &xe );
615 }
616 
HG_(record_error_LockOrder)617 void HG_(record_error_LockOrder)(
618         Thread*     thr,
619         Addr        shouldbe_earlier_ga,
620         Addr        shouldbe_later_ga,
621         ExeContext* shouldbe_earlier_ec,
622         ExeContext* shouldbe_later_ec,
623         ExeContext* actual_earlier_ec
624      )
625 {
626    XError xe;
627    tl_assert( HG_(is_sane_Thread)(thr) );
628    tl_assert(HG_(clo_track_lockorders));
629    init_XError(&xe);
630    xe.tag = XE_LockOrder;
631    xe.XE.LockOrder.thr       = thr;
632    xe.XE.LockOrder.shouldbe_earlier_ga = shouldbe_earlier_ga;
633    xe.XE.LockOrder.shouldbe_earlier_ec = shouldbe_earlier_ec;
634    xe.XE.LockOrder.shouldbe_later_ga   = shouldbe_later_ga;
635    xe.XE.LockOrder.shouldbe_later_ec   = shouldbe_later_ec;
636    xe.XE.LockOrder.actual_earlier_ec   = actual_earlier_ec;
637    // FIXME: tid vs thr
638    tl_assert( HG_(is_sane_ThreadId)(thr->coretid) );
639    tl_assert( thr->coretid != VG_INVALID_THREADID );
640    VG_(maybe_record_error)( thr->coretid,
641                             XE_LockOrder, 0, NULL, &xe );
642 }
643 
HG_(record_error_PthAPIerror)644 void HG_(record_error_PthAPIerror) ( Thread* thr, HChar* fnname,
645                                      Word err, HChar* errstr )
646 {
647    XError xe;
648    tl_assert( HG_(is_sane_Thread)(thr) );
649    tl_assert(fnname);
650    tl_assert(errstr);
651    init_XError(&xe);
652    xe.tag = XE_PthAPIerror;
653    xe.XE.PthAPIerror.thr    = thr;
654    xe.XE.PthAPIerror.fnname = string_table_strdup(fnname);
655    xe.XE.PthAPIerror.err    = err;
656    xe.XE.PthAPIerror.errstr = string_table_strdup(errstr);
657    // FIXME: tid vs thr
658    tl_assert( HG_(is_sane_ThreadId)(thr->coretid) );
659    tl_assert( thr->coretid != VG_INVALID_THREADID );
660    VG_(maybe_record_error)( thr->coretid,
661                             XE_PthAPIerror, 0, NULL, &xe );
662 }
663 
HG_(record_error_Misc_w_aux)664 void HG_(record_error_Misc_w_aux) ( Thread* thr, HChar* errstr,
665                                     HChar* auxstr, ExeContext* auxctx )
666 {
667    XError xe;
668    tl_assert( HG_(is_sane_Thread)(thr) );
669    tl_assert(errstr);
670    init_XError(&xe);
671    xe.tag = XE_Misc;
672    xe.XE.Misc.thr    = thr;
673    xe.XE.Misc.errstr = string_table_strdup(errstr);
674    xe.XE.Misc.auxstr = auxstr ? string_table_strdup(auxstr) : NULL;
675    xe.XE.Misc.auxctx = auxctx;
676    // FIXME: tid vs thr
677    tl_assert( HG_(is_sane_ThreadId)(thr->coretid) );
678    tl_assert( thr->coretid != VG_INVALID_THREADID );
679    VG_(maybe_record_error)( thr->coretid,
680                             XE_Misc, 0, NULL, &xe );
681 }
682 
HG_(record_error_Misc)683 void HG_(record_error_Misc) ( Thread* thr, HChar* errstr )
684 {
685    HG_(record_error_Misc_w_aux)(thr, errstr, NULL, NULL);
686 }
687 
HG_(eq_Error)688 Bool HG_(eq_Error) ( VgRes not_used, Error* e1, Error* e2 )
689 {
690    XError *xe1, *xe2;
691 
692    tl_assert(VG_(get_error_kind)(e1) == VG_(get_error_kind)(e2));
693 
694    xe1 = (XError*)VG_(get_error_extra)(e1);
695    xe2 = (XError*)VG_(get_error_extra)(e2);
696    tl_assert(xe1);
697    tl_assert(xe2);
698 
699    switch (VG_(get_error_kind)(e1)) {
700       case XE_Race:
701          return xe1->XE.Race.szB == xe2->XE.Race.szB
702                 && xe1->XE.Race.isWrite == xe2->XE.Race.isWrite
703                 && (HG_(clo_cmp_race_err_addrs)
704                        ? xe1->XE.Race.data_addr == xe2->XE.Race.data_addr
705                        : True);
706       case XE_UnlockUnlocked:
707          return xe1->XE.UnlockUnlocked.thr == xe2->XE.UnlockUnlocked.thr
708                 && xe1->XE.UnlockUnlocked.lock == xe2->XE.UnlockUnlocked.lock;
709       case XE_UnlockForeign:
710          return xe1->XE.UnlockForeign.thr == xe2->XE.UnlockForeign.thr
711                 && xe1->XE.UnlockForeign.owner == xe2->XE.UnlockForeign.owner
712                 && xe1->XE.UnlockForeign.lock == xe2->XE.UnlockForeign.lock;
713       case XE_UnlockBogus:
714          return xe1->XE.UnlockBogus.thr == xe2->XE.UnlockBogus.thr
715                 && xe1->XE.UnlockBogus.lock_ga == xe2->XE.UnlockBogus.lock_ga;
716       case XE_PthAPIerror:
717          return xe1->XE.PthAPIerror.thr == xe2->XE.PthAPIerror.thr
718                 && 0==VG_(strcmp)(xe1->XE.PthAPIerror.fnname,
719                                   xe2->XE.PthAPIerror.fnname)
720                 && xe1->XE.PthAPIerror.err == xe2->XE.PthAPIerror.err;
721       case XE_LockOrder:
722          return xe1->XE.LockOrder.thr == xe2->XE.LockOrder.thr;
723       case XE_Misc:
724          return xe1->XE.Misc.thr == xe2->XE.Misc.thr
725                 && 0==VG_(strcmp)(xe1->XE.Misc.errstr, xe2->XE.Misc.errstr);
726       default:
727          tl_assert(0);
728    }
729 
730    /*NOTREACHED*/
731    tl_assert(0);
732 }
733 
734 
735 /*----------------------------------------------------------------*/
736 /*--- Error management -- printing                             ---*/
737 /*----------------------------------------------------------------*/
738 
739 /* Do a printf-style operation on either the XML or normal output
740    channel, depending on the setting of VG_(clo_xml).
741 */
emit_WRK(HChar * format,va_list vargs)742 static void emit_WRK ( HChar* format, va_list vargs )
743 {
744    if (VG_(clo_xml)) {
745       VG_(vprintf_xml)(format, vargs);
746    } else {
747       VG_(vmessage)(Vg_UserMsg, format, vargs);
748    }
749 }
750 static void emit ( HChar* format, ... ) PRINTF_CHECK(1, 2);
emit(HChar * format,...)751 static void emit ( HChar* format, ... )
752 {
753    va_list vargs;
754    va_start(vargs, format);
755    emit_WRK(format, vargs);
756    va_end(vargs);
757 }
758 
759 
760 /* Announce (that is, print the point-of-creation) of 'thr'.  Only do
761    this once, as we only want to see these announcements once per
762    thread.  Returned Bool indicates whether or not an announcement was
763    made.
764 */
announce_one_thread(Thread * thr)765 static Bool announce_one_thread ( Thread* thr )
766 {
767    tl_assert(HG_(is_sane_Thread)(thr));
768    tl_assert(thr->errmsg_index >= 1);
769    if (thr->announced)
770       return False;
771 
772    if (VG_(clo_xml)) {
773 
774       VG_(printf_xml)("<announcethread>\n");
775       VG_(printf_xml)("  <hthreadid>%d</hthreadid>\n", thr->errmsg_index);
776       if (thr->errmsg_index == 1) {
777          tl_assert(thr->created_at == NULL);
778          VG_(printf_xml)("  <isrootthread></isrootthread>\n");
779       } else {
780          tl_assert(thr->created_at != NULL);
781          VG_(pp_ExeContext)( thr->created_at );
782       }
783       VG_(printf_xml)("</announcethread>\n\n");
784 
785    } else {
786 
787       VG_(umsg)("---Thread-Announcement----------"
788                 "--------------------------------" "\n");
789       VG_(umsg)("\n");
790 
791       if (thr->errmsg_index == 1) {
792          tl_assert(thr->created_at == NULL);
793          VG_(message)(Vg_UserMsg,
794                       "Thread #%d is the program's root thread\n",
795                        thr->errmsg_index);
796       } else {
797          tl_assert(thr->created_at != NULL);
798          VG_(message)(Vg_UserMsg, "Thread #%d was created\n",
799                                   thr->errmsg_index);
800          VG_(pp_ExeContext)( thr->created_at );
801       }
802       VG_(message)(Vg_UserMsg, "\n");
803 
804    }
805 
806    thr->announced = True;
807    return True;
808 }
809 
810 
811 /* Announce 'lk'. */
announce_LockP(Lock * lk)812 static void announce_LockP ( Lock* lk )
813 {
814    tl_assert(lk);
815    if (lk == Lock_INVALID)
816       return; /* Can't be announced -- we know nothing about it. */
817    tl_assert(lk->magic == LockP_MAGIC);
818    if (!lk->appeared_at)
819      return; /* There's nothing we can show */
820 
821    if (VG_(clo_xml)) {
822       /* fixme: add announcement */
823    } else {
824       VG_(umsg)( "Lock at %p was first observed\n",
825                  (void*)lk->guestaddr );
826       VG_(pp_ExeContext)( lk->appeared_at );
827       VG_(umsg)("\n");
828    }
829 }
830 
831 /* Announce (that is, print point-of-first-observation) for the
832    locks in 'lockvec' and, if non-NULL, 'lockvec2'. */
announce_combined_LockP_vecs(Lock ** lockvec,Lock ** lockvec2)833 static void announce_combined_LockP_vecs ( Lock** lockvec,
834                                            Lock** lockvec2 )
835 {
836    UWord i;
837    tl_assert(lockvec);
838    for (i = 0; lockvec[i]; i++) {
839       announce_LockP(lockvec[i]);
840    }
841    if (lockvec2) {
842       for (i = 0; lockvec2[i]; i++) {
843          Lock* lk = lockvec2[i];
844          if (!elem_LockP_vector(lockvec, lk))
845             announce_LockP(lk);
846       }
847    }
848 }
849 
850 
show_LockP_summary_textmode(Lock ** locks,HChar * pre)851 static void show_LockP_summary_textmode ( Lock** locks, HChar* pre )
852 {
853    tl_assert(locks);
854    UWord i;
855    UWord nLocks = 0, nLocksValid = 0;
856    count_LockP_vector(&nLocks, &nLocksValid, locks);
857    tl_assert(nLocksValid <= nLocks);
858 
859    if (nLocks == 0) {
860       VG_(umsg)( "%sLocks held: none", pre );
861    } else {
862       VG_(umsg)( "%sLocks held: %lu, at address%s ",
863                  pre, nLocks, nLocksValid == 1 ? "" : "es" );
864    }
865 
866    if (nLocks > 0) {
867       for (i = 0; i < nLocks; i++) {
868          if (locks[i] == Lock_INVALID)
869             continue;
870          VG_(umsg)( "%p", (void*)locks[i]->guestaddr);
871          if (locks[i+1] != NULL)
872             VG_(umsg)(" ");
873       }
874       if (nLocksValid < nLocks)
875          VG_(umsg)(" (and %lu that can't be shown)", nLocks - nLocksValid);
876    }
877    VG_(umsg)("\n");
878 }
879 
880 
881 /* This is the "this error is due to be printed shortly; so have a
882    look at it any print any preamble you want" function.  We use it to
883    announce any previously un-announced threads in the upcoming error
884    message.
885 */
HG_(before_pp_Error)886 void HG_(before_pp_Error) ( Error* err )
887 {
888    XError* xe;
889    tl_assert(err);
890    xe = (XError*)VG_(get_error_extra)(err);
891    tl_assert(xe);
892 
893    switch (VG_(get_error_kind)(err)) {
894       case XE_Misc:
895          announce_one_thread( xe->XE.Misc.thr );
896          break;
897       case XE_LockOrder:
898          announce_one_thread( xe->XE.LockOrder.thr );
899          break;
900       case XE_PthAPIerror:
901          announce_one_thread( xe->XE.PthAPIerror.thr );
902          break;
903       case XE_UnlockBogus:
904          announce_one_thread( xe->XE.UnlockBogus.thr );
905          break;
906       case XE_UnlockForeign:
907          announce_one_thread( xe->XE.UnlockForeign.thr );
908          announce_one_thread( xe->XE.UnlockForeign.owner );
909          break;
910       case XE_UnlockUnlocked:
911          announce_one_thread( xe->XE.UnlockUnlocked.thr );
912          break;
913       case XE_Race:
914          announce_one_thread( xe->XE.Race.thr );
915          if (xe->XE.Race.h2_ct)
916             announce_one_thread( xe->XE.Race.h2_ct );
917          if (xe->XE.Race.h1_ct)
918             announce_one_thread( xe->XE.Race.h1_ct );
919          break;
920       default:
921          tl_assert(0);
922    }
923 }
924 
HG_(pp_Error)925 void HG_(pp_Error) ( Error* err )
926 {
927    const Bool xml = VG_(clo_xml); /* a shorthand, that's all */
928 
929    if (!xml) {
930       VG_(umsg)("--------------------------------"
931                 "--------------------------------" "\n");
932       VG_(umsg)("\n");
933    }
934 
935    XError *xe = (XError*)VG_(get_error_extra)(err);
936    tl_assert(xe);
937 
938    if (xml)
939       emit( "  <kind>%s</kind>\n", HG_(get_error_name)(err));
940 
941    switch (VG_(get_error_kind)(err)) {
942 
943    case XE_Misc: {
944       tl_assert( HG_(is_sane_Thread)( xe->XE.Misc.thr ) );
945 
946       if (xml) {
947 
948          emit( "  <xwhat>\n" );
949          emit( "    <text>Thread #%d: %s</text>\n",
950                (Int)xe->XE.Misc.thr->errmsg_index,
951                xe->XE.Misc.errstr );
952          emit( "    <hthreadid>%d</hthreadid>\n",
953                (Int)xe->XE.Misc.thr->errmsg_index );
954          emit( "  </xwhat>\n" );
955          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
956          if (xe->XE.Misc.auxstr) {
957             emit("  <auxwhat>%s</auxwhat>\n", xe->XE.Misc.auxstr);
958             if (xe->XE.Misc.auxctx)
959                VG_(pp_ExeContext)( xe->XE.Misc.auxctx );
960          }
961 
962       } else {
963 
964          emit( "Thread #%d: %s\n",
965                (Int)xe->XE.Misc.thr->errmsg_index,
966                xe->XE.Misc.errstr );
967          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
968          if (xe->XE.Misc.auxstr) {
969             emit(" %s\n", xe->XE.Misc.auxstr);
970             if (xe->XE.Misc.auxctx)
971                VG_(pp_ExeContext)( xe->XE.Misc.auxctx );
972          }
973 
974       }
975       break;
976    }
977 
978    case XE_LockOrder: {
979       tl_assert( HG_(is_sane_Thread)( xe->XE.LockOrder.thr ) );
980 
981       if (xml) {
982 
983          emit( "  <xwhat>\n" );
984          emit( "    <text>Thread #%d: lock order \"%p before %p\" "
985                     "violated</text>\n",
986                (Int)xe->XE.LockOrder.thr->errmsg_index,
987                (void*)xe->XE.LockOrder.shouldbe_earlier_ga,
988                (void*)xe->XE.LockOrder.shouldbe_later_ga );
989          emit( "    <hthreadid>%d</hthreadid>\n",
990                (Int)xe->XE.LockOrder.thr->errmsg_index );
991          emit( "  </xwhat>\n" );
992          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
993          if (xe->XE.LockOrder.shouldbe_earlier_ec
994              && xe->XE.LockOrder.shouldbe_later_ec) {
995             emit( "  <auxwhat>Required order was established by "
996                   "acquisition of lock at %p</auxwhat>\n",
997                   (void*)xe->XE.LockOrder.shouldbe_earlier_ga );
998             VG_(pp_ExeContext)( xe->XE.LockOrder.shouldbe_earlier_ec );
999             emit( "  <auxwhat>followed by a later acquisition "
1000                   "of lock at %p</auxwhat>\n",
1001                   (void*)xe->XE.LockOrder.shouldbe_later_ga );
1002             VG_(pp_ExeContext)( xe->XE.LockOrder.shouldbe_later_ec );
1003          }
1004 
1005       } else {
1006 
1007          emit( "Thread #%d: lock order \"%p before %p\" violated\n",
1008                (Int)xe->XE.LockOrder.thr->errmsg_index,
1009                (void*)xe->XE.LockOrder.shouldbe_earlier_ga,
1010                (void*)xe->XE.LockOrder.shouldbe_later_ga );
1011          emit( "\n" );
1012          emit( "Observed (incorrect) order is: "
1013                "acquisition of lock at %p\n",
1014                (void*)xe->XE.LockOrder.shouldbe_later_ga);
1015          if (xe->XE.LockOrder.actual_earlier_ec) {
1016              VG_(pp_ExeContext)(xe->XE.LockOrder.actual_earlier_ec);
1017          } else {
1018             emit("   (stack unavailable)\n");
1019          }
1020          emit( "\n" );
1021          emit(" followed by a later acquisition of lock at %p\n",
1022               (void*)xe->XE.LockOrder.shouldbe_earlier_ga);
1023          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
1024          if (xe->XE.LockOrder.shouldbe_earlier_ec
1025              && xe->XE.LockOrder.shouldbe_later_ec) {
1026             emit("\n");
1027             emit( "Required order was established by "
1028                   "acquisition of lock at %p\n",
1029                   (void*)xe->XE.LockOrder.shouldbe_earlier_ga );
1030             VG_(pp_ExeContext)( xe->XE.LockOrder.shouldbe_earlier_ec );
1031             emit( "\n" );
1032             emit( " followed by a later acquisition of lock at %p\n",
1033                   (void*)xe->XE.LockOrder.shouldbe_later_ga );
1034             VG_(pp_ExeContext)( xe->XE.LockOrder.shouldbe_later_ec );
1035          }
1036 
1037       }
1038 
1039       break;
1040    }
1041 
1042    case XE_PthAPIerror: {
1043       tl_assert( HG_(is_sane_Thread)( xe->XE.PthAPIerror.thr ) );
1044 
1045       if (xml) {
1046 
1047          emit( "  <xwhat>\n" );
1048          emit(
1049             "    <text>Thread #%d's call to %pS failed</text>\n",
1050             (Int)xe->XE.PthAPIerror.thr->errmsg_index,
1051             xe->XE.PthAPIerror.fnname );
1052          emit( "    <hthreadid>%d</hthreadid>\n",
1053                (Int)xe->XE.PthAPIerror.thr->errmsg_index );
1054          emit( "  </xwhat>\n" );
1055          emit( "  <what>with error code %ld (%s)</what>\n",
1056                xe->XE.PthAPIerror.err, xe->XE.PthAPIerror.errstr );
1057          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
1058 
1059       } else {
1060 
1061          emit( "Thread #%d's call to %pS failed\n",
1062                       (Int)xe->XE.PthAPIerror.thr->errmsg_index,
1063                       xe->XE.PthAPIerror.fnname );
1064          emit( "   with error code %ld (%s)\n",
1065                xe->XE.PthAPIerror.err, xe->XE.PthAPIerror.errstr );
1066          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
1067 
1068       }
1069 
1070       break;
1071    }
1072 
1073    case XE_UnlockBogus: {
1074       tl_assert( HG_(is_sane_Thread)( xe->XE.UnlockBogus.thr ) );
1075 
1076       if (xml) {
1077 
1078          emit( "  <xwhat>\n" );
1079          emit( "    <text>Thread #%d unlocked an invalid "
1080                     "lock at %p</text>\n",
1081                (Int)xe->XE.UnlockBogus.thr->errmsg_index,
1082                (void*)xe->XE.UnlockBogus.lock_ga );
1083          emit( "    <hthreadid>%d</hthreadid>\n",
1084                (Int)xe->XE.UnlockBogus.thr->errmsg_index );
1085          emit( "  </xwhat>\n" );
1086          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
1087 
1088       } else {
1089 
1090          emit( "Thread #%d unlocked an invalid lock at %p\n",
1091                (Int)xe->XE.UnlockBogus.thr->errmsg_index,
1092                (void*)xe->XE.UnlockBogus.lock_ga );
1093          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
1094 
1095       }
1096 
1097       break;
1098    }
1099 
1100    case XE_UnlockForeign: {
1101       tl_assert( HG_(is_sane_LockP)( xe->XE.UnlockForeign.lock ) );
1102       tl_assert( HG_(is_sane_Thread)( xe->XE.UnlockForeign.owner ) );
1103       tl_assert( HG_(is_sane_Thread)( xe->XE.UnlockForeign.thr ) );
1104 
1105       if (xml) {
1106 
1107          emit( "  <xwhat>\n" );
1108          emit( "    <text>Thread #%d unlocked lock at %p "
1109                     "currently held by thread #%d</text>\n",
1110                (Int)xe->XE.UnlockForeign.thr->errmsg_index,
1111                (void*)xe->XE.UnlockForeign.lock->guestaddr,
1112                (Int)xe->XE.UnlockForeign.owner->errmsg_index );
1113          emit( "    <hthreadid>%d</hthreadid>\n",
1114                (Int)xe->XE.UnlockForeign.thr->errmsg_index );
1115          emit( "    <hthreadid>%d</hthreadid>\n",
1116                (Int)xe->XE.UnlockForeign.owner->errmsg_index );
1117          emit( "  </xwhat>\n" );
1118          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
1119 
1120          if (xe->XE.UnlockForeign.lock->appeared_at) {
1121             emit( "  <auxwhat>Lock at %p was first observed</auxwhat>\n",
1122                   (void*)xe->XE.UnlockForeign.lock->guestaddr );
1123             VG_(pp_ExeContext)( xe->XE.UnlockForeign.lock->appeared_at );
1124          }
1125 
1126       } else {
1127 
1128          emit( "Thread #%d unlocked lock at %p "
1129                "currently held by thread #%d\n",
1130                (Int)xe->XE.UnlockForeign.thr->errmsg_index,
1131                (void*)xe->XE.UnlockForeign.lock->guestaddr,
1132                (Int)xe->XE.UnlockForeign.owner->errmsg_index );
1133          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
1134          if (xe->XE.UnlockForeign.lock->appeared_at) {
1135             emit( "  Lock at %p was first observed\n",
1136                   (void*)xe->XE.UnlockForeign.lock->guestaddr );
1137             VG_(pp_ExeContext)( xe->XE.UnlockForeign.lock->appeared_at );
1138          }
1139 
1140       }
1141 
1142       break;
1143    }
1144 
1145    case XE_UnlockUnlocked: {
1146       tl_assert( HG_(is_sane_LockP)( xe->XE.UnlockUnlocked.lock ) );
1147       tl_assert( HG_(is_sane_Thread)( xe->XE.UnlockUnlocked.thr ) );
1148 
1149       if (xml) {
1150 
1151          emit( "  <xwhat>\n" );
1152          emit( "    <text>Thread #%d unlocked a "
1153                     "not-locked lock at %p</text>\n",
1154                (Int)xe->XE.UnlockUnlocked.thr->errmsg_index,
1155                (void*)xe->XE.UnlockUnlocked.lock->guestaddr );
1156          emit( "    <hthreadid>%d</hthreadid>\n",
1157                (Int)xe->XE.UnlockUnlocked.thr->errmsg_index );
1158          emit( "  </xwhat>\n" );
1159          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
1160          if (xe->XE.UnlockUnlocked.lock->appeared_at) {
1161             emit( "  <auxwhat>Lock at %p was first observed</auxwhat>\n",
1162                   (void*)xe->XE.UnlockUnlocked.lock->guestaddr );
1163             VG_(pp_ExeContext)( xe->XE.UnlockUnlocked.lock->appeared_at );
1164          }
1165 
1166       } else {
1167 
1168          emit( "Thread #%d unlocked a not-locked lock at %p\n",
1169                (Int)xe->XE.UnlockUnlocked.thr->errmsg_index,
1170                (void*)xe->XE.UnlockUnlocked.lock->guestaddr );
1171          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
1172          if (xe->XE.UnlockUnlocked.lock->appeared_at) {
1173             emit( "  Lock at %p was first observed\n",
1174                   (void*)xe->XE.UnlockUnlocked.lock->guestaddr );
1175             VG_(pp_ExeContext)( xe->XE.UnlockUnlocked.lock->appeared_at );
1176          }
1177 
1178       }
1179 
1180       break;
1181    }
1182 
1183    case XE_Race: {
1184       Addr      err_ga;
1185       HChar*    what;
1186       Int       szB;
1187       what      = xe->XE.Race.isWrite ? "write" : "read";
1188       szB       = xe->XE.Race.szB;
1189       err_ga = VG_(get_error_address)(err);
1190 
1191       tl_assert( HG_(is_sane_Thread)( xe->XE.Race.thr ));
1192       if (xe->XE.Race.h2_ct)
1193          tl_assert( HG_(is_sane_Thread)( xe->XE.Race.h2_ct ));
1194 
1195       if (xml) {
1196 
1197          /* ------ XML ------ */
1198          emit( "  <xwhat>\n" );
1199          emit( "    <text>Possible data race during %s of size %d "
1200                     "at %p by thread #%d</text>\n",
1201                what, szB, (void*)err_ga, (Int)xe->XE.Race.thr->errmsg_index );
1202          emit( "    <hthreadid>%d</hthreadid>\n",
1203                (Int)xe->XE.Race.thr->errmsg_index );
1204          emit( "  </xwhat>\n" );
1205          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
1206 
1207          if (xe->XE.Race.h2_ct) {
1208             tl_assert(xe->XE.Race.h2_ct_accEC); // assured by update_extra
1209             emit( "  <xauxwhat>\n");
1210             emit( "    <text>This conflicts with a previous %s of size %d "
1211                             "by thread #%d</text>\n",
1212                   xe->XE.Race.h2_ct_accIsW ? "write" : "read",
1213                   xe->XE.Race.h2_ct_accSzB,
1214                   xe->XE.Race.h2_ct->errmsg_index );
1215             emit( "    <hthreadid>%d</hthreadid>\n",
1216                   xe->XE.Race.h2_ct->errmsg_index);
1217             emit("  </xauxwhat>\n");
1218             VG_(pp_ExeContext)( xe->XE.Race.h2_ct_accEC );
1219          }
1220 
1221          if (xe->XE.Race.h1_ct) {
1222             emit( "  <xauxwhat>\n");
1223             emit( "    <text>This conflicts with a previous access "
1224                   "by thread #%d, after</text>\n",
1225                   xe->XE.Race.h1_ct->errmsg_index );
1226             emit( "    <hthreadid>%d</hthreadid>\n",
1227                   xe->XE.Race.h1_ct->errmsg_index );
1228             emit("  </xauxwhat>\n");
1229             if (xe->XE.Race.h1_ct_mbsegstartEC) {
1230                VG_(pp_ExeContext)( xe->XE.Race.h1_ct_mbsegstartEC );
1231             } else {
1232                emit( "  <auxwhat>(the start of the thread)</auxwhat>\n" );
1233             }
1234             emit( "  <auxwhat>but before</auxwhat>\n" );
1235             if (xe->XE.Race.h1_ct_mbsegendEC) {
1236                VG_(pp_ExeContext)( xe->XE.Race.h1_ct_mbsegendEC );
1237             } else {
1238                emit( "  <auxwhat>(the end of the the thread)</auxwhat>\n" );
1239             }
1240          }
1241 
1242       } else {
1243 
1244          /* ------ Text ------ */
1245          announce_combined_LockP_vecs( xe->XE.Race.locksHeldW,
1246                                        xe->XE.Race.h2_ct_locksHeldW );
1247 
1248          emit( "Possible data race during %s of size %d "
1249                "at %p by thread #%d\n",
1250                what, szB, (void*)err_ga, (Int)xe->XE.Race.thr->errmsg_index );
1251 
1252          tl_assert(xe->XE.Race.locksHeldW);
1253          show_LockP_summary_textmode( xe->XE.Race.locksHeldW, "" );
1254          VG_(pp_ExeContext)( VG_(get_error_where)(err) );
1255 
1256          if (xe->XE.Race.h2_ct) {
1257             tl_assert(xe->XE.Race.h2_ct_accEC); // assured by update_extra
1258             tl_assert(xe->XE.Race.h2_ct_locksHeldW);
1259             emit( "\n" );
1260             emit( "This conflicts with a previous %s of size %d "
1261                   "by thread #%d\n",
1262                   xe->XE.Race.h2_ct_accIsW ? "write" : "read",
1263                   xe->XE.Race.h2_ct_accSzB,
1264                   xe->XE.Race.h2_ct->errmsg_index );
1265             show_LockP_summary_textmode( xe->XE.Race.h2_ct_locksHeldW, "" );
1266             VG_(pp_ExeContext)( xe->XE.Race.h2_ct_accEC );
1267          }
1268 
1269          if (xe->XE.Race.h1_ct) {
1270             emit( " This conflicts with a previous access by thread #%d, "
1271                   "after\n",
1272                   xe->XE.Race.h1_ct->errmsg_index );
1273             if (xe->XE.Race.h1_ct_mbsegstartEC) {
1274                VG_(pp_ExeContext)( xe->XE.Race.h1_ct_mbsegstartEC );
1275             } else {
1276                emit( "   (the start of the thread)\n" );
1277             }
1278             emit( " but before\n" );
1279             if (xe->XE.Race.h1_ct_mbsegendEC) {
1280                VG_(pp_ExeContext)( xe->XE.Race.h1_ct_mbsegendEC );
1281             } else {
1282                emit( "   (the end of the the thread)\n" );
1283             }
1284          }
1285 
1286       }
1287 
1288       /* If we have a description of the address in terms of a heap
1289          block, show it. */
1290       if (xe->XE.Race.hctxt) {
1291          SizeT delta = err_ga - xe->XE.Race.haddr;
1292          if (xml) {
1293             emit("  <auxwhat>Address %p is %ld bytes inside a block "
1294                  "of size %ld alloc'd</auxwhat>\n", (void*)err_ga, delta,
1295                  xe->XE.Race.hszB);
1296             VG_(pp_ExeContext)( xe->XE.Race.hctxt );
1297          } else {
1298             emit("\n");
1299             emit("Address %p is %ld bytes inside a block "
1300                  "of size %ld alloc'd\n", (void*)err_ga, delta,
1301                  xe->XE.Race.hszB);
1302             VG_(pp_ExeContext)( xe->XE.Race.hctxt );
1303          }
1304       }
1305 
1306       /* If we have a better description of the address, show it.
1307          Note that in XML mode, it will already by nicely wrapped up
1308          in tags, either <auxwhat> or <xauxwhat>, so we can just emit
1309          it verbatim. */
1310       if (xml) {
1311          if (xe->XE.Race.descr1)
1312             emit( "  %s\n",
1313                   (HChar*)VG_(indexXA)( xe->XE.Race.descr1, 0 ) );
1314          if (xe->XE.Race.descr2)
1315             emit( "  %s\n",
1316                   (HChar*)VG_(indexXA)( xe->XE.Race.descr2, 0 ) );
1317       } else {
1318          if (xe->XE.Race.descr1 || xe->XE.Race.descr2)
1319             emit("\n");
1320          if (xe->XE.Race.descr1)
1321             emit( "%s\n",
1322                   (HChar*)VG_(indexXA)( xe->XE.Race.descr1, 0 ) );
1323          if (xe->XE.Race.descr2)
1324             emit( "%s\n",
1325                   (HChar*)VG_(indexXA)( xe->XE.Race.descr2, 0 ) );
1326       }
1327 
1328       break; /* case XE_Race */
1329    } /* case XE_Race */
1330 
1331    default:
1332       tl_assert(0);
1333    } /* switch (VG_(get_error_kind)(err)) */
1334 }
1335 
HG_(get_error_name)1336 Char* HG_(get_error_name) ( Error* err )
1337 {
1338    switch (VG_(get_error_kind)(err)) {
1339       case XE_Race:           return "Race";
1340       case XE_UnlockUnlocked: return "UnlockUnlocked";
1341       case XE_UnlockForeign:  return "UnlockForeign";
1342       case XE_UnlockBogus:    return "UnlockBogus";
1343       case XE_PthAPIerror:    return "PthAPIerror";
1344       case XE_LockOrder:      return "LockOrder";
1345       case XE_Misc:           return "Misc";
1346       default: tl_assert(0); /* fill in missing case */
1347    }
1348 }
1349 
HG_(recognised_suppression)1350 Bool HG_(recognised_suppression) ( Char* name, Supp *su )
1351 {
1352 #  define TRY(_name,_xskind)                   \
1353       if (0 == VG_(strcmp)(name, (_name))) {   \
1354          VG_(set_supp_kind)(su, (_xskind));    \
1355          return True;                          \
1356       }
1357    TRY("Race",           XS_Race);
1358    TRY("FreeMemLock",    XS_FreeMemLock);
1359    TRY("UnlockUnlocked", XS_UnlockUnlocked);
1360    TRY("UnlockForeign",  XS_UnlockForeign);
1361    TRY("UnlockBogus",    XS_UnlockBogus);
1362    TRY("PthAPIerror",    XS_PthAPIerror);
1363    TRY("LockOrder",      XS_LockOrder);
1364    TRY("Misc",           XS_Misc);
1365    return False;
1366 #  undef TRY
1367 }
1368 
HG_(read_extra_suppression_info)1369 Bool HG_(read_extra_suppression_info) ( Int fd, Char** bufpp, SizeT* nBufp,
1370                                         Supp* su )
1371 {
1372    /* do nothing -- no extra suppression info present.  Return True to
1373       indicate nothing bad happened. */
1374    return True;
1375 }
1376 
HG_(error_matches_suppression)1377 Bool HG_(error_matches_suppression) ( Error* err, Supp* su )
1378 {
1379    switch (VG_(get_supp_kind)(su)) {
1380    case XS_Race:           return VG_(get_error_kind)(err) == XE_Race;
1381    case XS_UnlockUnlocked: return VG_(get_error_kind)(err) == XE_UnlockUnlocked;
1382    case XS_UnlockForeign:  return VG_(get_error_kind)(err) == XE_UnlockForeign;
1383    case XS_UnlockBogus:    return VG_(get_error_kind)(err) == XE_UnlockBogus;
1384    case XS_PthAPIerror:    return VG_(get_error_kind)(err) == XE_PthAPIerror;
1385    case XS_LockOrder:      return VG_(get_error_kind)(err) == XE_LockOrder;
1386    case XS_Misc:           return VG_(get_error_kind)(err) == XE_Misc;
1387    //case XS_: return VG_(get_error_kind)(err) == XE_;
1388    default: tl_assert(0); /* fill in missing cases */
1389    }
1390 }
1391 
HG_(get_extra_suppression_info)1392 Bool HG_(get_extra_suppression_info) ( Error* err,
1393                                        /*OUT*/Char* buf, Int nBuf )
1394 {
1395    /* Do nothing */
1396    return False;
1397 }
1398 
1399 
1400 /*--------------------------------------------------------------------*/
1401 /*--- end                                              hg_errors.c ---*/
1402 /*--------------------------------------------------------------------*/
1403