blob: 55ba81e764e08cdb4dbbf859d55019090dbf4915 [file] [log] [blame]
Russ Cox9406f682015-04-17 00:34:18 -04001// errorcheck -0 -l -d=wb
2
Emmanuel Odeke53fd5222016-04-10 14:32:26 -07003// Copyright 2015 The Go Authors. All rights reserved.
Russ Cox9406f682015-04-17 00:34:18 -04004// Use of this source code is governed by a BSD-style
5// license that can be found in the LICENSE file.
6
7// Test where write barriers are and are not emitted.
8
9package p
10
11import "unsafe"
12
13func f(x **byte, y *byte) {
Josh Bleecher Snydere5c93582017-04-27 13:15:24 -070014 *x = y // no barrier (dead store)
Russ Cox9406f682015-04-17 00:34:18 -040015
16 z := y // no barrier
17 *x = z // ERROR "write barrier"
18}
19
20func f1(x *[]byte, y []byte) {
Josh Bleecher Snydere5c93582017-04-27 13:15:24 -070021 *x = y // no barrier (dead store)
Russ Cox9406f682015-04-17 00:34:18 -040022
23 z := y // no barrier
24 *x = z // ERROR "write barrier"
25}
26
27func f1a(x *[]byte, y *[]byte) {
28 *x = *y // ERROR "write barrier"
29
30 z := *y // no barrier
Russ Cox85520472015-05-06 12:34:30 -040031 *x = z // ERROR "write barrier"
Russ Cox9406f682015-04-17 00:34:18 -040032}
33
34func f2(x *interface{}, y interface{}) {
Josh Bleecher Snydere5c93582017-04-27 13:15:24 -070035 *x = y // no barrier (dead store)
Russ Cox9406f682015-04-17 00:34:18 -040036
37 z := y // no barrier
38 *x = z // ERROR "write barrier"
39}
40
41func f2a(x *interface{}, y *interface{}) {
Josh Bleecher Snydere5c93582017-04-27 13:15:24 -070042 *x = *y // no barrier (dead store)
Russ Cox9406f682015-04-17 00:34:18 -040043
44 z := y // no barrier
45 *x = z // ERROR "write barrier"
46}
47
48func f3(x *string, y string) {
Josh Bleecher Snydere5c93582017-04-27 13:15:24 -070049 *x = y // no barrier (dead store)
Russ Cox9406f682015-04-17 00:34:18 -040050
51 z := y // no barrier
52 *x = z // ERROR "write barrier"
53}
54
55func f3a(x *string, y *string) {
56 *x = *y // ERROR "write barrier"
57
58 z := *y // no barrier
Russ Cox85520472015-05-06 12:34:30 -040059 *x = z // ERROR "write barrier"
Russ Cox9406f682015-04-17 00:34:18 -040060}
61
62func f4(x *[2]string, y [2]string) {
63 *x = y // ERROR "write barrier"
64
65 z := y // no barrier
66 *x = z // ERROR "write barrier"
67}
68
69func f4a(x *[2]string, y *[2]string) {
70 *x = *y // ERROR "write barrier"
71
72 z := *y // no barrier
Russ Cox85520472015-05-06 12:34:30 -040073 *x = z // ERROR "write barrier"
Russ Cox9406f682015-04-17 00:34:18 -040074}
75
76type T struct {
77 X *int
78 Y int
79 M map[int]int
80}
81
82func f5(t, u *T) {
83 t.X = &u.Y // ERROR "write barrier"
84}
85
86func f6(t *T) {
87 t.M = map[int]int{1: 2} // ERROR "write barrier"
88}
89
90func f7(x, y *int) []*int {
91 var z [3]*int
92 i := 0
93 z[i] = x // ERROR "write barrier"
94 i++
95 z[i] = y // ERROR "write barrier"
96 i++
97 return z[:i]
98}
99
100func f9(x *interface{}, v *byte) {
101 *x = v // ERROR "write barrier"
102}
103
104func f10(x *byte, f func(interface{})) {
105 f(x)
106}
107
108func f11(x *unsafe.Pointer, y unsafe.Pointer) {
109 *x = unsafe.Pointer(uintptr(y) + 1) // ERROR "write barrier"
110}
Russ Cox85520472015-05-06 12:34:30 -0400111
112func f12(x []*int, y *int) []*int {
113 // write barrier for storing y in x's underlying array
114 x = append(x, y) // ERROR "write barrier"
115 return x
116}
117
118func f12a(x []int, y int) []int {
119 // y not a pointer, so no write barriers in this function
120 x = append(x, y)
121 return x
122}
123
124func f13(x []int, y *[]int) {
125 *y = append(x, 1) // ERROR "write barrier"
126}
127
128func f14(y *[]int) {
129 *y = append(*y, 1) // ERROR "write barrier"
130}
Russ Cox366ba522015-05-18 16:54:59 -0400131
132type T1 struct {
133 X *int
134}
135
136func f15(x []T1, y T1) []T1 {
137 return append(x, y) // ERROR "write barrier"
138}
139
140type T8 struct {
141 X [8]*int
142}
143
144func f16(x []T8, y T8) []T8 {
145 return append(x, y) // ERROR "write barrier"
146}
Keith Randalle3033fc2016-02-12 10:07:36 -0800147
148func t1(i interface{}) **int {
149 // From issue 14306, make sure we have write barriers in a type switch
150 // where the assigned variable escapes.
151 switch x := i.(type) { // ERROR "write barrier"
152 case *int:
153 return &x
154 }
155 switch y := i.(type) { // no write barrier here
156 case **int:
157 return y
158 }
159 return nil
160}
Austin Clements3e54ca92016-03-16 18:22:58 -0400161
162type T17 struct {
163 f func(*T17)
164}
165
166func f17(x *T17) {
Austin Clementsc39918a2016-10-18 10:26:28 -0400167 // Originally from golang.org/issue/13901, but the hybrid
168 // barrier requires both to have barriers.
169 x.f = f17 // ERROR "write barrier"
Austin Clements3e54ca92016-03-16 18:22:58 -0400170 x.f = func(y *T17) { *y = *x } // ERROR "write barrier"
171}
Keith Randall15ed37d2016-03-16 21:51:17 -0700172
173type T18 struct {
174 a []int
175 s string
176}
177
178func f18(p *T18, x *[]int) {
179 p.a = p.a[:5] // no barrier
Keith Randalld4663e12016-03-21 10:22:03 -0700180 *x = (*x)[0:5] // no barrier
181 p.a = p.a[3:5] // ERROR "write barrier"
182 p.a = p.a[1:2:3] // ERROR "write barrier"
183 p.s = p.s[8:9] // ERROR "write barrier"
184 *x = (*x)[3:5] // ERROR "write barrier"
Keith Randall15ed37d2016-03-16 21:51:17 -0700185}
Keith Randall934c3592016-04-23 22:59:01 -0700186
187func f19(x, y *int, i int) int {
188 // Constructing a temporary slice on the stack should not
189 // require any write barriers. See issue 14263.
190 a := []*int{x, y} // no barrier
191 return *a[i]
192}
193
194func f20(x, y *int, i int) []*int {
195 // ... but if that temporary slice escapes, then the
196 // write barriers are necessary.
197 a := []*int{x, y} // ERROR "write barrier"
198 return a
199}
Austin Clementsb49b71a2016-03-18 11:27:59 -0400200
201var x21 *int
202var y21 struct {
203 x *int
204}
205var z21 int
206
Josh Bleecher Snydere5c93582017-04-27 13:15:24 -0700207// f21x: Global -> heap pointer updates must have write barriers.
208func f21a(x *int) {
209 x21 = x // ERROR "write barrier"
210 y21.x = x // ERROR "write barrier"
211}
212
213func f21b(x *int) {
214 x21 = &z21 // ERROR "write barrier"
215 y21.x = &z21 // ERROR "write barrier"
216}
217
218func f21c(x *int) {
Austin Clementsb49b71a2016-03-18 11:27:59 -0400219 y21 = struct{ x *int }{x} // ERROR "write barrier"
220}
Cherry Zhangf6aec882016-10-13 06:57:00 -0400221
222func f22(x *int) (y *int) {
223 // pointer write on stack should have no write barrier.
224 // this is a case that the frontend failed to eliminate.
225 p := &y
226 *p = x // no barrier
227 return
228}
Cherry Zhang160914e2017-02-06 14:24:16 -0500229
230type T23 struct {
231 p *int
232 a int
233}
234
235var t23 T23
236var i23 int
237
Josh Bleecher Snydere5c93582017-04-27 13:15:24 -0700238// f23x: zeroing global needs write barrier for the hybrid barrier.
239func f23a() {
Cherry Zhang160914e2017-02-06 14:24:16 -0500240 t23 = T23{} // ERROR "write barrier"
Josh Bleecher Snyderee69c212017-05-10 12:48:17 -0700241}
242
243func f23b() {
Cherry Zhang160914e2017-02-06 14:24:16 -0500244 // also test partial assignments
Josh Bleecher Snydere5c93582017-04-27 13:15:24 -0700245 t23 = T23{a: 1} // ERROR "write barrier"
246}
247
Josh Bleecher Snyderee69c212017-05-10 12:48:17 -0700248func f23c() {
Josh Bleecher Snydere5c93582017-04-27 13:15:24 -0700249 t23 = T23{} // no barrier (dead store)
250 // also test partial assignments
Cherry Zhang160914e2017-02-06 14:24:16 -0500251 t23 = T23{p: &i23} // ERROR "write barrier"
252}