1
2
3
4
5 package escape
6
7 import (
8 "cmd/compile/internal/base"
9 "cmd/compile/internal/ir"
10 )
11
12
13
14 func (e *escape) addr(n ir.Node) hole {
15 if n == nil || ir.IsBlank(n) {
16
17 return e.discardHole()
18 }
19
20 k := e.heapHole()
21
22 switch n.Op() {
23 default:
24 base.Fatalf("unexpected addr: %v", n)
25 case ir.ONAME:
26 n := n.(*ir.Name)
27 if n.Class == ir.PEXTERN {
28 break
29 }
30 k = e.oldLoc(n).asHole()
31 case ir.OLINKSYMOFFSET:
32 break
33 case ir.ODOT:
34 n := n.(*ir.SelectorExpr)
35 k = e.addr(n.X)
36 case ir.OINDEX:
37 n := n.(*ir.IndexExpr)
38 e.discard(n.Index)
39 if n.X.Type().IsArray() {
40 k = e.addr(n.X)
41 } else {
42 e.mutate(n.X)
43 }
44 case ir.ODEREF:
45 n := n.(*ir.StarExpr)
46 e.mutate(n.X)
47 case ir.ODOTPTR:
48 n := n.(*ir.SelectorExpr)
49 e.mutate(n.X)
50 case ir.OINDEXMAP:
51 n := n.(*ir.IndexExpr)
52 e.discard(n.X)
53 e.assignHeap(n.Index, "key of map put", n)
54 }
55
56 return k
57 }
58
59 func (e *escape) mutate(n ir.Node) {
60 e.expr(e.mutatorHole(), n)
61 }
62
63 func (e *escape) addrs(l ir.Nodes) []hole {
64 var ks []hole
65 for _, n := range l {
66 ks = append(ks, e.addr(n))
67 }
68 return ks
69 }
70
71 func (e *escape) assignHeap(src ir.Node, why string, where ir.Node) {
72 e.expr(e.heapHole().note(where, why), src)
73 }
74
75
76 func (e *escape) assignList(dsts, srcs []ir.Node, why string, where ir.Node) {
77 ks := e.addrs(dsts)
78 for i, k := range ks {
79 var src ir.Node
80 if i < len(srcs) {
81 src = srcs[i]
82 }
83
84 if dst := dsts[i]; dst != nil {
85
86
87 if dst.Op() == ir.ODOTPTR && ir.IsReflectHeaderDataField(dst) {
88 e.unsafeValue(e.heapHole().note(where, why), src)
89 continue
90 }
91
92
93 if src != nil && isSelfAssign(dst, src) {
94 if base.Flag.LowerM != 0 {
95 base.WarnfAt(where.Pos(), "%v ignoring self-assignment in %v", e.curfn, where)
96 }
97 k = e.discardHole()
98 }
99 }
100
101 e.expr(k.note(where, why), src)
102 }
103
104 e.reassigned(ks, where)
105 }
106
107
108
109
110 func (e *escape) reassigned(ks []hole, where ir.Node) {
111 if as, ok := where.(*ir.AssignStmt); ok && as.Op() == ir.OAS && as.Y == nil {
112 if dst, ok := as.X.(*ir.Name); ok && dst.Op() == ir.ONAME && dst.Defn == nil {
113
114
115
116 return
117 }
118 }
119
120 for _, k := range ks {
121 loc := k.dst
122
123 if n, ok := loc.n.(*ir.Name); ok && n.Defn == where && where.Op() != ir.ORANGE {
124 continue
125 }
126 loc.reassigned = true
127 }
128 }
129
View as plain text