blob: fe7a82cf728ba2e849a14f4d4fe7b0e6e6a889f1 [file] [log] [blame]
Russ Cox8c195bd2015-02-13 14:40:36 -05001// Copyright 2012 The Go Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
5package gc
6
7import (
8 "fmt"
9 "strings"
10)
11
12// The racewalk pass modifies the code tree for the function as follows:
13//
14// 1. It inserts a call to racefuncenter at the beginning of each function.
15// 2. It inserts a call to racefuncexit at the end of each function.
16// 3. It inserts a call to raceread before each memory read.
17// 4. It inserts a call to racewrite before each memory write.
18//
19// The rewriting is not yet complete. Certain nodes are not rewritten
20// but should be.
21
22// TODO(dvyukov): do not instrument initialization as writes:
23// a := make([]int, 10)
24
25// Do not instrument the following packages at all,
26// at best instrumentation would cause infinite recursion.
27var omit_pkgs = []string{"runtime", "runtime/race"}
28
29// Only insert racefuncenter/racefuncexit into the following packages.
30// Memory accesses in the packages are either uninteresting or will cause false positives.
31var noinst_pkgs = []string{"sync", "sync/atomic"}
32
33func ispkgin(pkgs []string) int {
34 var i int
35
36 if myimportpath != "" {
37 for i = 0; i < len(pkgs); i++ {
38 if myimportpath == pkgs[i] {
39 return 1
40 }
41 }
42 }
43
44 return 0
45}
46
47func isforkfunc(fn *Node) int {
48 // Special case for syscall.forkAndExecInChild.
49 // In the child, this function must not acquire any locks, because
50 // they might have been locked at the time of the fork. This means
51 // no rescheduling, no malloc calls, and no new stack segments.
52 // Race instrumentation does all of the above.
53 return bool2int(myimportpath != "" && myimportpath == "syscall" && fn.Nname.Sym.Name == "forkAndExecInChild")
54}
55
56func racewalk(fn *Node) {
57 var nd *Node
58 var nodpc *Node
59 var s string
60
61 if ispkgin(omit_pkgs) != 0 || isforkfunc(fn) != 0 {
62 return
63 }
64
65 if !(ispkgin(noinst_pkgs) != 0) {
66 racewalklist(fn.Nbody, nil)
67
68 // nothing interesting for race detector in fn->enter
69 racewalklist(fn.Exit, nil)
70 }
71
72 // nodpc is the PC of the caller as extracted by
73 // getcallerpc. We use -widthptr(FP) for x86.
74 // BUG: this will not work on arm.
75 nodpc = Nod(OXXX, nil, nil)
76
77 *nodpc = *nodfp
78 nodpc.Type = Types[TUINTPTR]
79 nodpc.Xoffset = int64(-Widthptr)
80 nd = mkcall("racefuncenter", nil, nil, nodpc)
81 fn.Enter = concat(list1(nd), fn.Enter)
82 nd = mkcall("racefuncexit", nil, nil)
83 fn.Exit = list(fn.Exit, nd)
84
85 if Debug['W'] != 0 {
86 s = fmt.Sprintf("after racewalk %v", Sconv(fn.Nname.Sym, 0))
87 dumplist(s, fn.Nbody)
88 s = fmt.Sprintf("enter %v", Sconv(fn.Nname.Sym, 0))
89 dumplist(s, fn.Enter)
90 s = fmt.Sprintf("exit %v", Sconv(fn.Nname.Sym, 0))
91 dumplist(s, fn.Exit)
92 }
93}
94
95func racewalklist(l *NodeList, init **NodeList) {
96 var instr *NodeList
97
98 for ; l != nil; l = l.Next {
99 instr = nil
100 racewalknode(&l.N, &instr, 0, 0)
101 if init == nil {
102 l.N.Ninit = concat(l.N.Ninit, instr)
103 } else {
104 *init = concat(*init, instr)
105 }
106 }
107}
108
109// walkexpr and walkstmt combined
110// walks the tree and adds calls to the
111// instrumentation code to top-level (statement) nodes' init
112func racewalknode(np **Node, init **NodeList, wr int, skip int) {
113 var n *Node
114 var n1 *Node
115 var l *NodeList
116 var fini *NodeList
117
118 n = *np
119
120 if n == nil {
121 return
122 }
123
124 if Debug['w'] > 1 {
125 Dump("racewalk-before", n)
126 }
127 setlineno(n)
128 if init == nil {
129 Fatal("racewalk: bad init list")
130 }
131 if init == &n.Ninit {
132 // If init == &n->ninit and n->ninit is non-nil,
133 // racewalknode might append it to itself.
134 // nil it out and handle it separately before putting it back.
135 l = n.Ninit
136
137 n.Ninit = nil
138 racewalklist(l, nil)
139 racewalknode(&n, &l, wr, skip) // recurse with nil n->ninit
140 appendinit(&n, l)
141 *np = n
142 return
143 }
144
145 racewalklist(n.Ninit, nil)
146
147 switch n.Op {
148 default:
149 Fatal("racewalk: unknown node type %v", Oconv(int(n.Op), 0))
150 fallthrough
151
152 case OAS,
153 OAS2FUNC:
154 racewalknode(&n.Left, init, 1, 0)
155 racewalknode(&n.Right, init, 0, 0)
156 goto ret
157
158 // can't matter
159 case OCFUNC,
160 OVARKILL:
161 goto ret
162
163 case OBLOCK:
164 if n.List == nil {
165 goto ret
166 }
167
168 switch n.List.N.Op {
169 // Blocks are used for multiple return function calls.
170 // x, y := f() becomes BLOCK{CALL f, AS x [SP+0], AS y [SP+n]}
171 // We don't want to instrument between the statements because it will
172 // smash the results.
173 case OCALLFUNC,
174 OCALLMETH,
175 OCALLINTER:
176 racewalknode(&n.List.N, &n.List.N.Ninit, 0, 0)
177
178 fini = nil
179 racewalklist(n.List.Next, &fini)
180 n.List = concat(n.List, fini)
181
182 // Ordinary block, for loop initialization or inlined bodies.
183 default:
184 racewalklist(n.List, nil)
185 }
186
187 goto ret
188
189 case ODEFER:
190 racewalknode(&n.Left, init, 0, 0)
191 goto ret
192
193 case OPROC:
194 racewalknode(&n.Left, init, 0, 0)
195 goto ret
196
197 case OCALLINTER:
198 racewalknode(&n.Left, init, 0, 0)
199 goto ret
200
201 // Instrument dst argument of runtime.writebarrier* calls
202 // as we do not instrument runtime code.
203 // typedslicecopy is instrumented in runtime.
204 case OCALLFUNC:
205 if n.Left.Sym != nil && n.Left.Sym.Pkg == Runtimepkg && (strings.HasPrefix(n.Left.Sym.Name, "writebarrier") || n.Left.Sym.Name == "typedmemmove") {
206 // Find the dst argument.
207 // The list can be reordered, so it's not necessary just the first or the second element.
208 for l = n.List; l != nil; l = l.Next {
209 if n.Left.Sym.Name == "typedmemmove" {
210 if l.N.Left.Xoffset == int64(Widthptr) {
211 break
212 }
213 } else {
214 if l.N.Left.Xoffset == 0 {
215 break
216 }
217 }
218 }
219
220 if l == nil {
221 Fatal("racewalk: writebarrier no arg")
222 }
223 if l.N.Right.Op != OADDR {
224 Fatal("racewalk: writebarrier bad arg")
225 }
226 callinstr(&l.N.Right.Left, init, 1, 0)
227 }
228
229 racewalknode(&n.Left, init, 0, 0)
230 goto ret
231
232 case ONOT,
233 OMINUS,
234 OPLUS,
235 OREAL,
236 OIMAG,
237 OCOM:
238 racewalknode(&n.Left, init, wr, 0)
239 goto ret
240
241 case ODOTINTER:
242 racewalknode(&n.Left, init, 0, 0)
243 goto ret
244
245 case ODOT:
246 racewalknode(&n.Left, init, 0, 1)
247 callinstr(&n, init, wr, skip)
248 goto ret
249
250 case ODOTPTR: // dst = (*x).f with implicit *; otherwise it's ODOT+OIND
251 racewalknode(&n.Left, init, 0, 0)
252
253 callinstr(&n, init, wr, skip)
254 goto ret
255
256 case OIND: // *p
257 racewalknode(&n.Left, init, 0, 0)
258
259 callinstr(&n, init, wr, skip)
260 goto ret
261
262 case OSPTR,
263 OLEN,
264 OCAP:
265 racewalknode(&n.Left, init, 0, 0)
266 if Istype(n.Left.Type, TMAP) != 0 {
267 n1 = Nod(OCONVNOP, n.Left, nil)
268 n1.Type = Ptrto(Types[TUINT8])
269 n1 = Nod(OIND, n1, nil)
270 typecheck(&n1, Erv)
271 callinstr(&n1, init, 0, skip)
272 }
273
274 goto ret
275
276 case OLSH,
277 ORSH,
278 OLROT,
279 OAND,
280 OANDNOT,
281 OOR,
282 OXOR,
283 OSUB,
284 OMUL,
285 OHMUL,
286 OEQ,
287 ONE,
288 OLT,
289 OLE,
290 OGE,
291 OGT,
292 OADD,
293 OCOMPLEX:
294 racewalknode(&n.Left, init, wr, 0)
295 racewalknode(&n.Right, init, wr, 0)
296 goto ret
297
298 case OANDAND,
299 OOROR:
300 racewalknode(&n.Left, init, wr, 0)
301
302 // walk has ensured the node has moved to a location where
303 // side effects are safe.
304 // n->right may not be executed,
305 // so instrumentation goes to n->right->ninit, not init.
306 racewalknode(&n.Right, &n.Right.Ninit, wr, 0)
307
308 goto ret
309
310 case ONAME:
311 callinstr(&n, init, wr, skip)
312 goto ret
313
314 case OCONV:
315 racewalknode(&n.Left, init, wr, 0)
316 goto ret
317
318 case OCONVNOP:
319 racewalknode(&n.Left, init, wr, 0)
320 goto ret
321
322 case ODIV,
323 OMOD:
324 racewalknode(&n.Left, init, wr, 0)
325 racewalknode(&n.Right, init, wr, 0)
326 goto ret
327
328 case OINDEX:
329 if !(Isfixedarray(n.Left.Type) != 0) {
330 racewalknode(&n.Left, init, 0, 0)
331 } else if !(islvalue(n.Left) != 0) {
332 // index of unaddressable array, like Map[k][i].
333 racewalknode(&n.Left, init, wr, 0)
334
335 racewalknode(&n.Right, init, 0, 0)
336 goto ret
337 }
338
339 racewalknode(&n.Right, init, 0, 0)
340 if n.Left.Type.Etype != TSTRING {
341 callinstr(&n, init, wr, skip)
342 }
343 goto ret
344
345 // Seems to only lead to double instrumentation.
346 //racewalknode(&n->left, init, 0, 0);
347 case OSLICE,
348 OSLICEARR,
349 OSLICE3,
350 OSLICE3ARR:
351 goto ret
352
353 case OADDR:
354 racewalknode(&n.Left, init, 0, 1)
355 goto ret
356
357 // n->left is Type* which is not interesting.
358 case OEFACE:
359 racewalknode(&n.Right, init, 0, 0)
360
361 goto ret
362
363 case OITAB:
364 racewalknode(&n.Left, init, 0, 0)
365 goto ret
366
367 // should not appear in AST by now
368 case OSEND,
369 ORECV,
370 OCLOSE,
371 ONEW,
372 OXCASE,
373 OXFALL,
374 OCASE,
375 OPANIC,
376 ORECOVER,
377 OCONVIFACE,
378 OCMPIFACE,
379 OMAKECHAN,
380 OMAKEMAP,
381 OMAKESLICE,
382 OCALL,
383 OCOPY,
384 OAPPEND,
385 ORUNESTR,
386 OARRAYBYTESTR,
387 OARRAYRUNESTR,
388 OSTRARRAYBYTE,
389 OSTRARRAYRUNE,
390 OINDEXMAP,
391 // lowered to call
392 OCMPSTR,
393 OADDSTR,
394 ODOTTYPE,
395 ODOTTYPE2,
396 OAS2DOTTYPE,
397 OCALLPART,
398 // lowered to PTRLIT
399 OCLOSURE, // lowered to PTRLIT
400 ORANGE, // lowered to ordinary for loop
401 OARRAYLIT, // lowered to assignments
402 OMAPLIT,
403 OSTRUCTLIT,
404 OAS2,
405 OAS2RECV,
406 OAS2MAPR,
407 OASOP:
408 Yyerror("racewalk: %v must be lowered by now", Oconv(int(n.Op), 0))
409
410 goto ret
411
412 // impossible nodes: only appear in backend.
413 case ORROTC,
414 OEXTEND:
415 Yyerror("racewalk: %v cannot exist now", Oconv(int(n.Op), 0))
416
417 goto ret
418
419 // just do generic traversal
420 case OFOR,
421 OIF,
422 OCALLMETH,
423 ORETURN,
424 ORETJMP,
425 OSWITCH,
426 OSELECT,
427 OEMPTY,
428 OBREAK,
429 OCONTINUE,
430 OFALL,
431 OGOTO,
432 OLABEL:
433 goto ret
434
435 // does not require instrumentation
436 case OPRINT, // don't bother instrumenting it
437 OPRINTN, // don't bother instrumenting it
438 OCHECKNIL, // always followed by a read.
439 OPARAM, // it appears only in fn->exit to copy heap params back
440 OCLOSUREVAR, // immutable pointer to captured variable
441 ODOTMETH, // either part of CALLMETH or CALLPART (lowered to PTRLIT)
442 OINDREG, // at this stage, only n(SP) nodes from nodarg
443 ODCL, // declarations (without value) cannot be races
444 ODCLCONST,
445 ODCLTYPE,
446 OTYPE,
447 ONONAME,
448 OLITERAL,
449 OSLICESTR,
450 // always preceded by bounds checking, avoid double instrumentation.
451 OTYPESW: // ignored by code generation, do not instrument.
452 goto ret
453 }
454
455ret:
456 if n.Op != OBLOCK { // OBLOCK is handled above in a special way.
457 racewalklist(n.List, init)
458 }
459 if n.Ntest != nil {
460 racewalknode(&n.Ntest, &n.Ntest.Ninit, 0, 0)
461 }
462 if n.Nincr != nil {
463 racewalknode(&n.Nincr, &n.Nincr.Ninit, 0, 0)
464 }
465 racewalklist(n.Nbody, nil)
466 racewalklist(n.Nelse, nil)
467 racewalklist(n.Rlist, nil)
468 *np = n
469}
470
471func isartificial(n *Node) int {
472 // compiler-emitted artificial things that we do not want to instrument,
473 // cant' possibly participate in a data race.
474 if n.Op == ONAME && n.Sym != nil && n.Sym.Name != "" {
475 if n.Sym.Name == "_" {
476 return 1
477 }
478
479 // autotmp's are always local
480 if strings.HasPrefix(n.Sym.Name, "autotmp_") {
481 return 1
482 }
483
484 // statictmp's are read-only
485 if strings.HasPrefix(n.Sym.Name, "statictmp_") {
486 return 1
487 }
488
489 // go.itab is accessed only by the compiler and runtime (assume safe)
490 if n.Sym.Pkg != nil && n.Sym.Pkg.Name != "" && n.Sym.Pkg.Name == "go.itab" {
491 return 1
492 }
493 }
494
495 return 0
496}
497
498func callinstr(np **Node, init **NodeList, wr int, skip int) int {
499 var name string
500 var f *Node
501 var b *Node
502 var n *Node
503 var t *Type
504 var class int
505 var hascalls int
506
507 n = *np
508
509 //print("callinstr for %+N [ %O ] etype=%E class=%d\n",
510 // n, n->op, n->type ? n->type->etype : -1, n->class);
511
512 if skip != 0 || n.Type == nil || n.Type.Etype >= TIDEAL {
513 return 0
514 }
515 t = n.Type
516 if isartificial(n) != 0 {
517 return 0
518 }
519
520 b = outervalue(n)
521
522 // it skips e.g. stores to ... parameter array
523 if isartificial(b) != 0 {
524 return 0
525 }
526 class = int(b.Class)
527
528 // BUG: we _may_ want to instrument PAUTO sometimes
529 // e.g. if we've got a local variable/method receiver
530 // that has got a pointer inside. Whether it points to
531 // the heap or not is impossible to know at compile time
532 if (class&PHEAP != 0) || class == PPARAMREF || class == PEXTERN || b.Op == OINDEX || b.Op == ODOTPTR || b.Op == OIND {
533 hascalls = 0
534 foreach(n, hascallspred, &hascalls)
535 if hascalls != 0 {
536 n = detachexpr(n, init)
537 *np = n
538 }
539
540 n = treecopy(n)
541 makeaddable(n)
542 if t.Etype == TSTRUCT || Isfixedarray(t) != 0 {
543 name = "racereadrange"
544 if wr != 0 {
545 name = "racewriterange"
546 }
547 f = mkcall(name, nil, init, uintptraddr(n), Nodintconst(t.Width))
548 } else {
549 name = "raceread"
550 if wr != 0 {
551 name = "racewrite"
552 }
553 f = mkcall(name, nil, init, uintptraddr(n))
554 }
555
556 *init = list(*init, f)
557 return 1
558 }
559
560 return 0
561}
562
563// makeaddable returns a node whose memory location is the
564// same as n, but which is addressable in the Go language
565// sense.
566// This is different from functions like cheapexpr that may make
567// a copy of their argument.
568func makeaddable(n *Node) {
569 // The arguments to uintptraddr technically have an address but
570 // may not be addressable in the Go sense: for example, in the case
571 // of T(v).Field where T is a struct type and v is
572 // an addressable value.
573 switch n.Op {
574 case OINDEX:
575 if Isfixedarray(n.Left.Type) != 0 {
576 makeaddable(n.Left)
577 }
578
579 // Turn T(v).Field into v.Field
580 case ODOT,
581 OXDOT:
582 if n.Left.Op == OCONVNOP {
583 n.Left = n.Left.Left
584 }
585 makeaddable(n.Left)
586
587 // nothing to do
588 case ODOTPTR:
589 fallthrough
590 default:
591 break
592 }
593}
594
595func uintptraddr(n *Node) *Node {
596 var r *Node
597
598 r = Nod(OADDR, n, nil)
599 r.Bounded = 1
600 r = conv(r, Types[TUNSAFEPTR])
601 r = conv(r, Types[TUINTPTR])
602 return r
603}
604
605func detachexpr(n *Node, init **NodeList) *Node {
606 var addr *Node
607 var as *Node
608 var ind *Node
609 var l *Node
610
611 addr = Nod(OADDR, n, nil)
612 l = temp(Ptrto(n.Type))
613 as = Nod(OAS, l, addr)
614 typecheck(&as, Etop)
615 walkexpr(&as, init)
616 *init = list(*init, as)
617 ind = Nod(OIND, l, nil)
618 typecheck(&ind, Erv)
619 walkexpr(&ind, init)
620 return ind
621}
622
623func foreachnode(n *Node, f func(*Node, interface{}), c interface{}) {
624 if n != nil {
625 f(n, c)
626 }
627}
628
629func foreachlist(l *NodeList, f func(*Node, interface{}), c interface{}) {
630 for ; l != nil; l = l.Next {
631 foreachnode(l.N, f, c)
632 }
633}
634
635func foreach(n *Node, f func(*Node, interface{}), c interface{}) {
636 foreachlist(n.Ninit, f, c)
637 foreachnode(n.Left, f, c)
638 foreachnode(n.Right, f, c)
639 foreachlist(n.List, f, c)
640 foreachnode(n.Ntest, f, c)
641 foreachnode(n.Nincr, f, c)
642 foreachlist(n.Nbody, f, c)
643 foreachlist(n.Nelse, f, c)
644 foreachlist(n.Rlist, f, c)
645}
646
647func hascallspred(n *Node, c interface{}) {
648 switch n.Op {
649 case OCALL,
650 OCALLFUNC,
651 OCALLMETH,
652 OCALLINTER:
653 (*c.(*int))++
654 }
655}
656
657// appendinit is like addinit in subr.c
658// but appends rather than prepends.
659func appendinit(np **Node, init *NodeList) {
660 var n *Node
661
662 if init == nil {
663 return
664 }
665
666 n = *np
667 switch n.Op {
668 // There may be multiple refs to this node;
669 // introduce OCONVNOP to hold init list.
670 case ONAME,
671 OLITERAL:
672 n = Nod(OCONVNOP, n, nil)
673
674 n.Type = n.Left.Type
675 n.Typecheck = 1
676 *np = n
677 }
678
679 n.Ninit = concat(n.Ninit, init)
680 n.Ullman = UINF
681}