mirror of
https://github.com/golang/go
synced 2024-11-21 18:14:42 -07:00
cmd/gc: implement len(array) / cap(array) rule
The spec is looser than the current implementation. The spec edit was made in CL 4444050 (May 2011) but I never implemented it. Fixes #3244. R=ken2 CC=golang-dev https://golang.org/cl/5785049
This commit is contained in:
parent
43d71e7d7d
commit
d4fb568e04
@ -190,6 +190,46 @@ typecheck(Node **np, int top)
|
|||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* does n contain a call or receive operation?
|
||||||
|
*/
|
||||||
|
static int callrecvlist(NodeList*);
|
||||||
|
|
||||||
|
static int
|
||||||
|
callrecv(Node *n)
|
||||||
|
{
|
||||||
|
if(n == nil)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
switch(n->op) {
|
||||||
|
case OCALL:
|
||||||
|
case OCALLMETH:
|
||||||
|
case OCALLINTER:
|
||||||
|
case OCALLFUNC:
|
||||||
|
case ORECV:
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return callrecv(n->left) ||
|
||||||
|
callrecv(n->right) ||
|
||||||
|
callrecv(n->ntest) ||
|
||||||
|
callrecv(n->nincr) ||
|
||||||
|
callrecvlist(n->ninit) ||
|
||||||
|
callrecvlist(n->nbody) ||
|
||||||
|
callrecvlist(n->nelse) ||
|
||||||
|
callrecvlist(n->list) ||
|
||||||
|
callrecvlist(n->rlist);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
callrecvlist(NodeList *l)
|
||||||
|
{
|
||||||
|
for(; l; l=l->next)
|
||||||
|
if(callrecv(l->n))
|
||||||
|
return 1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
typecheck1(Node **np, int top)
|
typecheck1(Node **np, int top)
|
||||||
{
|
{
|
||||||
@ -995,12 +1035,14 @@ reswitch:
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case TARRAY:
|
case TARRAY:
|
||||||
if(t->bound >= 0 && l->op == ONAME) {
|
if(t->bound < 0) // slice
|
||||||
r = nod(OXXX, N, N);
|
break;
|
||||||
nodconst(r, types[TINT], t->bound);
|
if(callrecv(l)) // has call or receive
|
||||||
r->orig = n;
|
break;
|
||||||
n = r;
|
r = nod(OXXX, N, N);
|
||||||
}
|
nodconst(r, types[TINT], t->bound);
|
||||||
|
r->orig = n;
|
||||||
|
n = r;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
n->type = types[TINT];
|
n->type = types[TINT];
|
||||||
|
77
test/const4.go
Normal file
77
test/const4.go
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
// run
|
||||||
|
|
||||||
|
// Copyright 2011 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.
|
||||||
|
|
||||||
|
// Test len constants and non-constants, http://golang.org/issue/3244.
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
var b struct {
|
||||||
|
a[10]int
|
||||||
|
}
|
||||||
|
|
||||||
|
var m map[string][20]int
|
||||||
|
|
||||||
|
var s [][30]int
|
||||||
|
|
||||||
|
const (
|
||||||
|
n1 = len(b.a)
|
||||||
|
n2 = len(m[""])
|
||||||
|
n3 = len(s[10])
|
||||||
|
)
|
||||||
|
|
||||||
|
// Non-constants (see also const5.go).
|
||||||
|
var (
|
||||||
|
n4 = len(f())
|
||||||
|
n5 = len(<-c)
|
||||||
|
n6 = cap(g())
|
||||||
|
n7 = cap(<-c1)
|
||||||
|
)
|
||||||
|
|
||||||
|
var calledF = false
|
||||||
|
|
||||||
|
func f() *[40]int {
|
||||||
|
calledF = true
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var c = func() chan *[50]int {
|
||||||
|
c := make(chan *[50]int, 2)
|
||||||
|
c <- nil
|
||||||
|
c <- new([50]int)
|
||||||
|
return c
|
||||||
|
}()
|
||||||
|
|
||||||
|
var calledG = false
|
||||||
|
|
||||||
|
func g() *[60]int {
|
||||||
|
calledG = true
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var c1 = func() chan *[70]int {
|
||||||
|
c := make(chan *[70]int, 2)
|
||||||
|
c <- nil
|
||||||
|
c <- new([70]int)
|
||||||
|
return c
|
||||||
|
}()
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
if n1 != 10 || n2 != 20 || n3 != 30 || n4 != 40 || n5 != 50 || n6 != 60 || n7 != 70 {
|
||||||
|
println("BUG:", n1, n2, n3, n4, n5, n6, n7)
|
||||||
|
}
|
||||||
|
if !calledF {
|
||||||
|
println("BUG: did not call f")
|
||||||
|
}
|
||||||
|
if <-c == nil {
|
||||||
|
println("BUG: did not receive from c")
|
||||||
|
}
|
||||||
|
if !calledG {
|
||||||
|
println("BUG: did not call g")
|
||||||
|
}
|
||||||
|
if <-c1 == nil {
|
||||||
|
println("BUG: did not receive from c1")
|
||||||
|
}
|
||||||
|
}
|
33
test/const5.go
Normal file
33
test/const5.go
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
// errorcheck
|
||||||
|
|
||||||
|
// Copyright 2011 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.
|
||||||
|
|
||||||
|
// Test that len non-constants are not constants, http://golang.org/issue/3244.
|
||||||
|
|
||||||
|
package p
|
||||||
|
|
||||||
|
var b struct {
|
||||||
|
a[10]int
|
||||||
|
}
|
||||||
|
|
||||||
|
var m map[string][20]int
|
||||||
|
|
||||||
|
var s [][30]int
|
||||||
|
|
||||||
|
func f() *[40]int
|
||||||
|
var c chan *[50]int
|
||||||
|
|
||||||
|
const (
|
||||||
|
n1 = len(b.a)
|
||||||
|
n2 = len(m[""])
|
||||||
|
n3 = len(s[10])
|
||||||
|
|
||||||
|
n4 = len(f()) // ERROR "must be constant"
|
||||||
|
n5 = len(<-c) // ERROR "must be constant"
|
||||||
|
|
||||||
|
n6 = cap(f()) // ERROR "must be constant"
|
||||||
|
n7 = cap(<-c) // ERROR "must be constant"
|
||||||
|
)
|
||||||
|
|
Loading…
Reference in New Issue
Block a user