| // run | |
| // Copyright 2016 The Go Authors. All rights reserved. | |
| // Use of this source code is governed by a BSD-style | |
| // license that can be found in the LICENSE file. | |
| // Check that we do loads exactly once. The SSA backend | |
| // once tried to do the load in f twice, once sign extended | |
| // and once zero extended. This can cause problems in | |
| // racy code, particularly sync/mutex. | |
| package main | |
| func f(p *byte) bool { | |
| x := *p | |
| a := int64(int8(x)) | |
| b := int64(uint8(x)) | |
| return a == b | |
| } | |
| func main() { | |
| var x byte | |
| const N = 1000000 | |
| c := make(chan struct{}) | |
| go func() { | |
| for i := 0; i < N; i++ { | |
| x = 1 | |
| } | |
| c <- struct{}{} | |
| }() | |
| go func() { | |
| for i := 0; i < N; i++ { | |
| x = 2 | |
| } | |
| c <- struct{}{} | |
| }() | |
| for i := 0; i < N; i++ { | |
| if !f(&x) { | |
| panic("non-atomic load!") | |
| } | |
| } | |
| <-c | |
| <-c | |
| } | |