1
0
mirror of https://github.com/golang/go synced 2024-11-11 19:21:37 -07:00

cmd/gc: fix capturing by value for range statements

Kindly detected by race builders by failing TestRaceRange.
ORANGE typecheck does not increment decldepth around body.

Change-Id: I0df5f310cb3370a904c94d9647a9cf0f15729075
Reviewed-on: https://go-review.googlesource.com/3507
Reviewed-by: Russ Cox <rsc@golang.org>
This commit is contained in:
Dmitry Vyukov 2015-01-29 18:33:19 +03:00
parent 8e2423a67d
commit eaa872009d
3 changed files with 118 additions and 51 deletions

View File

@ -18,14 +18,25 @@ typecheckrange(Node *n)
Node *v1, *v2;
NodeList *ll;
// Typechecking order is important here:
// 0. first typecheck range expression (slice/map/chan),
// it is evaluated only once and so logically it is not part of the loop.
// 1. typcheck produced values,
// this part can declare new vars and so it must be typechecked before body,
// because body can contain a closure that captures the vars.
// 2. decldepth++ to denote loop body.
// 3. typecheck body.
// 4. decldepth--.
typecheck(&n->right, Erv);
if((t = n->right->type) == T)
goto out;
// delicate little dance. see typecheckas2
for(ll=n->list; ll; ll=ll->next)
if(ll->n->defn != n)
typecheck(&ll->n, Erv | Easgn);
typecheck(&n->right, Erv);
if((t = n->right->type) == T)
goto out;
if(isptr[t->etype] && isfixedarray(t->type))
t = t->type;
n->type = t;
@ -106,7 +117,9 @@ out:
if(ll->n->typecheck == 0)
typecheck(&ll->n, Erv | Easgn);
decldepth++;
typechecklist(n->nbody, Etop);
decldepth--;
}
void

View File

@ -2828,7 +2828,8 @@ checkassign(Node *stmt, Node *n)
{
Node *r, *l;
if(n->defn != stmt) {
// Variables declared in ORANGE are assigned on every iteration.
if(n->defn != stmt || stmt->op == ORANGE) {
r = outervalue(n);
for(l = n; l != r; l = l->left) {
l->assigned = 1;

View File

@ -10,56 +10,109 @@
package main
func main() {
type X struct {
v int
}
var x X
func() {
x.v++
}()
if x.v != 1 {
panic("x.v != 1")
}
type Y struct {
X
}
var y Y
func() {
y.v = 1
}()
if y.v != 1 {
panic("y.v != 1")
}
type Z struct {
a [3]byte
}
var z Z
func() {
i := 0
for z.a[1] = 1; i < 10; i++ {
{
type X struct {
v int
}
}()
if z.a[1] != 1 {
panic("z.a[1] != 1")
}
w := 0
tmp := 0
f := func() {
if w != 1 {
panic("w != 1")
}
}
func() {
tmp = w // force capture of w, but do not write to it yet
_ = tmp
var x X
func() {
x.v++
}()
if x.v != 1 {
panic("x.v != 1")
}
type Y struct {
X
}
var y Y
func() {
y.v = 1
}()
if y.v != 1 {
panic("y.v != 1")
}
}
{
type Z struct {
a [3]byte
}
var z Z
func() {
i := 0
for z.a[1] = 1; i < 10; i++ {
}
}()
if z.a[1] != 1 {
panic("z.a[1] != 1")
}
}
{
w := 0
tmp := 0
f := func() {
if w != 1 {
panic("w != 1")
}
}
func() {
tmp = w // force capture of w, but do not write to it yet
_ = tmp
func() {
w++ // write in a nested closure
func() {
w++ // write in a nested closure
}()
}()
}()
}()
f()
f()
}
{
var g func() int
for i := range [2]int{} {
if i == 0 {
g = func() int {
return i // test that we capture by ref here, i is mutated on every interation
}
}
}
if g() != 1 {
panic("g() != 1")
}
}
{
var g func() int
q := 0
for range [2]int{} {
q++
g = func() int {
return q // test that we capture by ref here
// q++ must on a different decldepth than q declaration
}
}
if g() != 2 {
panic("g() != 2")
}
}
{
var g func() int
var a [2]int
q := 0
for a[func() int {
q++
return 0
}()] = range [2]int{} {
g = func() int {
return q // test that we capture by ref here
// q++ must on a different decldepth than q declaration
}
}
if g() != 2 {
panic("g() != 2")
}
}
}