mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
This reduces the API surface of Type slightly (for #25056), but also makes it more consistent with the reflect and go/types APIs. Passes toolstash-check. Change-Id: Ief9a8eb461ae6e88895f347e2a1b7b8a62423222 Reviewed-on: https://go-review.googlesource.com/109138 Run-TryBot: Matthew Dempsky <mdempsky@google.com> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Brad Fitzpatrick <bradfitz@golang.org>
536 lines
12 KiB
Go
536 lines
12 KiB
Go
// Copyright 2009 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.
|
|
|
|
package gc
|
|
|
|
import (
|
|
"cmd/compile/internal/types"
|
|
"cmd/internal/objabi"
|
|
"cmd/internal/sys"
|
|
"unicode/utf8"
|
|
)
|
|
|
|
// range
|
|
func typecheckrange(n *Node) {
|
|
// 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--.
|
|
typecheckrangeExpr(n)
|
|
|
|
// second half of dance, the first half being typecheckrangeExpr
|
|
n.SetTypecheck(1)
|
|
ls := n.List.Slice()
|
|
for i1, n1 := range ls {
|
|
if n1.Typecheck() == 0 {
|
|
ls[i1] = typecheck(ls[i1], Erv|Easgn)
|
|
}
|
|
}
|
|
|
|
decldepth++
|
|
typecheckslice(n.Nbody.Slice(), Etop)
|
|
decldepth--
|
|
}
|
|
|
|
func typecheckrangeExpr(n *Node) {
|
|
n.Right = typecheck(n.Right, Erv)
|
|
|
|
t := n.Right.Type
|
|
if t == nil {
|
|
return
|
|
}
|
|
// delicate little dance. see typecheckas2
|
|
ls := n.List.Slice()
|
|
for i1, n1 := range ls {
|
|
if n1.Name == nil || n1.Name.Defn != n {
|
|
ls[i1] = typecheck(ls[i1], Erv|Easgn)
|
|
}
|
|
}
|
|
|
|
if t.IsPtr() && t.Elem().IsArray() {
|
|
t = t.Elem()
|
|
}
|
|
n.Type = t
|
|
|
|
var t1, t2 *types.Type
|
|
toomany := false
|
|
switch t.Etype {
|
|
default:
|
|
yyerrorl(n.Pos, "cannot range over %L", n.Right)
|
|
return
|
|
|
|
case TARRAY, TSLICE:
|
|
t1 = types.Types[TINT]
|
|
t2 = t.Elem()
|
|
|
|
case TMAP:
|
|
t1 = t.Key()
|
|
t2 = t.Elem()
|
|
|
|
case TCHAN:
|
|
if !t.ChanDir().CanRecv() {
|
|
yyerrorl(n.Pos, "invalid operation: range %v (receive from send-only type %v)", n.Right, n.Right.Type)
|
|
return
|
|
}
|
|
|
|
t1 = t.Elem()
|
|
t2 = nil
|
|
if n.List.Len() == 2 {
|
|
toomany = true
|
|
}
|
|
|
|
case TSTRING:
|
|
t1 = types.Types[TINT]
|
|
t2 = types.Runetype
|
|
}
|
|
|
|
if n.List.Len() > 2 || toomany {
|
|
yyerrorl(n.Pos, "too many variables in range")
|
|
}
|
|
|
|
var v1, v2 *Node
|
|
if n.List.Len() != 0 {
|
|
v1 = n.List.First()
|
|
}
|
|
if n.List.Len() > 1 {
|
|
v2 = n.List.Second()
|
|
}
|
|
|
|
// this is not only a optimization but also a requirement in the spec.
|
|
// "if the second iteration variable is the blank identifier, the range
|
|
// clause is equivalent to the same clause with only the first variable
|
|
// present."
|
|
if v2.isBlank() {
|
|
if v1 != nil {
|
|
n.List.Set1(v1)
|
|
}
|
|
v2 = nil
|
|
}
|
|
|
|
var why string
|
|
if v1 != nil {
|
|
if v1.Name != nil && v1.Name.Defn == n {
|
|
v1.Type = t1
|
|
} else if v1.Type != nil && assignop(t1, v1.Type, &why) == 0 {
|
|
yyerrorl(n.Pos, "cannot assign type %v to %L in range%s", t1, v1, why)
|
|
}
|
|
checkassign(n, v1)
|
|
}
|
|
|
|
if v2 != nil {
|
|
if v2.Name != nil && v2.Name.Defn == n {
|
|
v2.Type = t2
|
|
} else if v2.Type != nil && assignop(t2, v2.Type, &why) == 0 {
|
|
yyerrorl(n.Pos, "cannot assign type %v to %L in range%s", t2, v2, why)
|
|
}
|
|
checkassign(n, v2)
|
|
}
|
|
}
|
|
|
|
func cheapComputableIndex(width int64) bool {
|
|
switch thearch.LinkArch.Family {
|
|
// MIPS does not have R+R addressing
|
|
// Arm64 may lack ability to generate this code in our assembler,
|
|
// but the architecture supports it.
|
|
case sys.PPC64, sys.S390X:
|
|
return width == 1
|
|
case sys.AMD64, sys.I386, sys.ARM64, sys.ARM:
|
|
switch width {
|
|
case 1, 2, 4, 8:
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// walkrange transforms various forms of ORANGE into
|
|
// simpler forms. The result must be assigned back to n.
|
|
// Node n may also be modified in place, and may also be
|
|
// the returned node.
|
|
func walkrange(n *Node) *Node {
|
|
// variable name conventions:
|
|
// ohv1, hv1, hv2: hidden (old) val 1, 2
|
|
// ha, hit: hidden aggregate, iterator
|
|
// hn, hp: hidden len, pointer
|
|
// hb: hidden bool
|
|
// a, v1, v2: not hidden aggregate, val 1, 2
|
|
|
|
t := n.Type
|
|
|
|
a := n.Right
|
|
lno := setlineno(a)
|
|
n.Right = nil
|
|
|
|
var v1, v2 *Node
|
|
l := n.List.Len()
|
|
if l > 0 {
|
|
v1 = n.List.First()
|
|
}
|
|
|
|
if l > 1 {
|
|
v2 = n.List.Second()
|
|
}
|
|
|
|
if v2.isBlank() {
|
|
v2 = nil
|
|
}
|
|
|
|
if v1.isBlank() && v2 == nil {
|
|
v1 = nil
|
|
}
|
|
|
|
if v1 == nil && v2 != nil {
|
|
Fatalf("walkrange: v2 != nil while v1 == nil")
|
|
}
|
|
|
|
// n.List has no meaning anymore, clear it
|
|
// to avoid erroneous processing by racewalk.
|
|
n.List.Set(nil)
|
|
|
|
var ifGuard *Node
|
|
|
|
translatedLoopOp := OFOR
|
|
|
|
var body []*Node
|
|
var init []*Node
|
|
switch t.Etype {
|
|
default:
|
|
Fatalf("walkrange")
|
|
|
|
case TARRAY, TSLICE:
|
|
if memclrrange(n, v1, v2, a) {
|
|
lineno = lno
|
|
return n
|
|
}
|
|
|
|
// orderstmt arranged for a copy of the array/slice variable if needed.
|
|
ha := a
|
|
|
|
hv1 := temp(types.Types[TINT])
|
|
hn := temp(types.Types[TINT])
|
|
|
|
init = append(init, nod(OAS, hv1, nil))
|
|
init = append(init, nod(OAS, hn, nod(OLEN, ha, nil)))
|
|
|
|
n.Left = nod(OLT, hv1, hn)
|
|
n.Right = nod(OAS, hv1, nod(OADD, hv1, nodintconst(1)))
|
|
|
|
// for range ha { body }
|
|
if v1 == nil {
|
|
break
|
|
}
|
|
|
|
// for v1 := range ha { body }
|
|
if v2 == nil {
|
|
body = []*Node{nod(OAS, v1, hv1)}
|
|
break
|
|
}
|
|
|
|
// for v1, v2 := range ha { body }
|
|
if cheapComputableIndex(n.Type.Elem().Width) {
|
|
// v1, v2 = hv1, ha[hv1]
|
|
tmp := nod(OINDEX, ha, hv1)
|
|
tmp.SetBounded(true)
|
|
// Use OAS2 to correctly handle assignments
|
|
// of the form "v1, a[v1] := range".
|
|
a := nod(OAS2, nil, nil)
|
|
a.List.Set2(v1, v2)
|
|
a.Rlist.Set2(hv1, tmp)
|
|
body = []*Node{a}
|
|
break
|
|
}
|
|
|
|
if objabi.Preemptibleloops_enabled != 0 {
|
|
// Doing this transformation makes a bounds check removal less trivial; see #20711
|
|
// TODO enhance the preemption check insertion so that this transformation is not necessary.
|
|
ifGuard = nod(OIF, nil, nil)
|
|
ifGuard.Left = nod(OLT, hv1, hn)
|
|
translatedLoopOp = OFORUNTIL
|
|
}
|
|
|
|
hp := temp(types.NewPtr(n.Type.Elem()))
|
|
tmp := nod(OINDEX, ha, nodintconst(0))
|
|
tmp.SetBounded(true)
|
|
init = append(init, nod(OAS, hp, nod(OADDR, tmp, nil)))
|
|
|
|
// Use OAS2 to correctly handle assignments
|
|
// of the form "v1, a[v1] := range".
|
|
a := nod(OAS2, nil, nil)
|
|
a.List.Set2(v1, v2)
|
|
a.Rlist.Set2(hv1, nod(OIND, hp, nil))
|
|
body = append(body, a)
|
|
|
|
// Advance pointer as part of increment.
|
|
// We used to advance the pointer before executing the loop body,
|
|
// but doing so would make the pointer point past the end of the
|
|
// array during the final iteration, possibly causing another unrelated
|
|
// piece of memory not to be garbage collected until the loop finished.
|
|
// Advancing during the increment ensures that the pointer p only points
|
|
// pass the end of the array during the final "p++; i++; if(i >= len(x)) break;",
|
|
// after which p is dead, so it cannot confuse the collector.
|
|
tmp = nod(OADD, hp, nodintconst(t.Elem().Width))
|
|
|
|
tmp.Type = hp.Type
|
|
tmp.SetTypecheck(1)
|
|
tmp.Right.Type = types.Types[types.Tptr]
|
|
tmp.Right.SetTypecheck(1)
|
|
a = nod(OAS, hp, tmp)
|
|
a = typecheck(a, Etop)
|
|
n.Right.Ninit.Set1(a)
|
|
|
|
case TMAP:
|
|
// orderstmt allocated the iterator for us.
|
|
// we only use a once, so no copy needed.
|
|
ha := a
|
|
|
|
hit := prealloc[n]
|
|
th := hit.Type
|
|
n.Left = nil
|
|
keysym := th.Field(0).Sym // depends on layout of iterator struct. See reflect.go:hiter
|
|
valsym := th.Field(1).Sym // ditto
|
|
|
|
fn := syslook("mapiterinit")
|
|
|
|
fn = substArgTypes(fn, t.Key(), t.Elem(), th)
|
|
init = append(init, mkcall1(fn, nil, nil, typename(t), ha, nod(OADDR, hit, nil)))
|
|
n.Left = nod(ONE, nodSym(ODOT, hit, keysym), nodnil())
|
|
|
|
fn = syslook("mapiternext")
|
|
fn = substArgTypes(fn, th)
|
|
n.Right = mkcall1(fn, nil, nil, nod(OADDR, hit, nil))
|
|
|
|
key := nodSym(ODOT, hit, keysym)
|
|
key = nod(OIND, key, nil)
|
|
if v1 == nil {
|
|
body = nil
|
|
} else if v2 == nil {
|
|
body = []*Node{nod(OAS, v1, key)}
|
|
} else {
|
|
val := nodSym(ODOT, hit, valsym)
|
|
val = nod(OIND, val, nil)
|
|
a := nod(OAS2, nil, nil)
|
|
a.List.Set2(v1, v2)
|
|
a.Rlist.Set2(key, val)
|
|
body = []*Node{a}
|
|
}
|
|
|
|
case TCHAN:
|
|
// orderstmt arranged for a copy of the channel variable.
|
|
ha := a
|
|
|
|
n.Left = nil
|
|
|
|
hv1 := temp(t.Elem())
|
|
hv1.SetTypecheck(1)
|
|
if types.Haspointers(t.Elem()) {
|
|
init = append(init, nod(OAS, hv1, nil))
|
|
}
|
|
hb := temp(types.Types[TBOOL])
|
|
|
|
n.Left = nod(ONE, hb, nodbool(false))
|
|
a := nod(OAS2RECV, nil, nil)
|
|
a.SetTypecheck(1)
|
|
a.List.Set2(hv1, hb)
|
|
a.Rlist.Set1(nod(ORECV, ha, nil))
|
|
n.Left.Ninit.Set1(a)
|
|
if v1 == nil {
|
|
body = nil
|
|
} else {
|
|
body = []*Node{nod(OAS, v1, hv1)}
|
|
}
|
|
// Zero hv1. This prevents hv1 from being the sole, inaccessible
|
|
// reference to an otherwise GC-able value during the next channel receive.
|
|
// See issue 15281.
|
|
body = append(body, nod(OAS, hv1, nil))
|
|
|
|
case TSTRING:
|
|
// Transform string range statements like "for v1, v2 = range a" into
|
|
//
|
|
// ha := a
|
|
// for hv1 := 0; hv1 < len(ha); {
|
|
// hv1t := hv1
|
|
// hv2 := rune(ha[hv1])
|
|
// if hv2 < utf8.RuneSelf {
|
|
// hv1++
|
|
// } else {
|
|
// hv2, hv1 = decoderune(ha, hv1)
|
|
// }
|
|
// v1, v2 = hv1t, hv2
|
|
// // original body
|
|
// }
|
|
|
|
// orderstmt arranged for a copy of the string variable.
|
|
ha := a
|
|
|
|
hv1 := temp(types.Types[TINT])
|
|
hv1t := temp(types.Types[TINT])
|
|
hv2 := temp(types.Runetype)
|
|
|
|
// hv1 := 0
|
|
init = append(init, nod(OAS, hv1, nil))
|
|
|
|
// hv1 < len(ha)
|
|
n.Left = nod(OLT, hv1, nod(OLEN, ha, nil))
|
|
|
|
if v1 != nil {
|
|
// hv1t = hv1
|
|
body = append(body, nod(OAS, hv1t, hv1))
|
|
}
|
|
|
|
// hv2 := rune(ha[hv1])
|
|
nind := nod(OINDEX, ha, hv1)
|
|
nind.SetBounded(true)
|
|
body = append(body, nod(OAS, hv2, conv(nind, types.Runetype)))
|
|
|
|
// if hv2 < utf8.RuneSelf
|
|
nif := nod(OIF, nil, nil)
|
|
nif.Left = nod(OLT, hv2, nodintconst(utf8.RuneSelf))
|
|
|
|
// hv1++
|
|
nif.Nbody.Set1(nod(OAS, hv1, nod(OADD, hv1, nodintconst(1))))
|
|
|
|
// } else {
|
|
eif := nod(OAS2, nil, nil)
|
|
nif.Rlist.Set1(eif)
|
|
|
|
// hv2, hv1 = decoderune(ha, hv1)
|
|
eif.List.Set2(hv2, hv1)
|
|
fn := syslook("decoderune")
|
|
eif.Rlist.Set1(mkcall1(fn, fn.Type.Results(), nil, ha, hv1))
|
|
|
|
body = append(body, nif)
|
|
|
|
if v1 != nil {
|
|
if v2 != nil {
|
|
// v1, v2 = hv1t, hv2
|
|
a := nod(OAS2, nil, nil)
|
|
a.List.Set2(v1, v2)
|
|
a.Rlist.Set2(hv1t, hv2)
|
|
body = append(body, a)
|
|
} else {
|
|
// v1 = hv1t
|
|
body = append(body, nod(OAS, v1, hv1t))
|
|
}
|
|
}
|
|
}
|
|
|
|
n.Op = translatedLoopOp
|
|
typecheckslice(init, Etop)
|
|
|
|
if ifGuard != nil {
|
|
ifGuard.Ninit.Append(init...)
|
|
ifGuard = typecheck(ifGuard, Etop)
|
|
} else {
|
|
n.Ninit.Append(init...)
|
|
}
|
|
|
|
typecheckslice(n.Left.Ninit.Slice(), Etop)
|
|
|
|
n.Left = typecheck(n.Left, Erv)
|
|
n.Left = defaultlit(n.Left, nil)
|
|
n.Right = typecheck(n.Right, Etop)
|
|
typecheckslice(body, Etop)
|
|
n.Nbody.Prepend(body...)
|
|
|
|
if ifGuard != nil {
|
|
ifGuard.Nbody.Set1(n)
|
|
n = ifGuard
|
|
}
|
|
|
|
n = walkstmt(n)
|
|
|
|
lineno = lno
|
|
return n
|
|
}
|
|
|
|
// Lower n into runtime·memclr if possible, for
|
|
// fast zeroing of slices and arrays (issue 5373).
|
|
// Look for instances of
|
|
//
|
|
// for i := range a {
|
|
// a[i] = zero
|
|
// }
|
|
//
|
|
// in which the evaluation of a is side-effect-free.
|
|
//
|
|
// Parameters are as in walkrange: "for v1, v2 = range a".
|
|
func memclrrange(n, v1, v2, a *Node) bool {
|
|
if Debug['N'] != 0 || instrumenting {
|
|
return false
|
|
}
|
|
if v1 == nil || v2 != nil {
|
|
return false
|
|
}
|
|
if n.Nbody.Len() == 0 || n.Nbody.First() == nil || n.Nbody.Len() > 1 {
|
|
return false
|
|
}
|
|
stmt := n.Nbody.First() // only stmt in body
|
|
if stmt.Op != OAS || stmt.Left.Op != OINDEX {
|
|
return false
|
|
}
|
|
if !samesafeexpr(stmt.Left.Left, a) || !samesafeexpr(stmt.Left.Right, v1) {
|
|
return false
|
|
}
|
|
elemsize := n.Type.Elem().Width
|
|
if elemsize <= 0 || !isZero(stmt.Right) {
|
|
return false
|
|
}
|
|
|
|
// Convert to
|
|
// if len(a) != 0 {
|
|
// hp = &a[0]
|
|
// hn = len(a)*sizeof(elem(a))
|
|
// memclr{NoHeap,Has}Pointers(hp, hn)
|
|
// i = len(a) - 1
|
|
// }
|
|
n.Op = OIF
|
|
|
|
n.Nbody.Set(nil)
|
|
n.Left = nod(ONE, nod(OLEN, a, nil), nodintconst(0))
|
|
|
|
// hp = &a[0]
|
|
hp := temp(types.Types[TUNSAFEPTR])
|
|
|
|
tmp := nod(OINDEX, a, nodintconst(0))
|
|
tmp.SetBounded(true)
|
|
tmp = nod(OADDR, tmp, nil)
|
|
tmp = nod(OCONVNOP, tmp, nil)
|
|
tmp.Type = types.Types[TUNSAFEPTR]
|
|
n.Nbody.Append(nod(OAS, hp, tmp))
|
|
|
|
// hn = len(a) * sizeof(elem(a))
|
|
hn := temp(types.Types[TUINTPTR])
|
|
|
|
tmp = nod(OLEN, a, nil)
|
|
tmp = nod(OMUL, tmp, nodintconst(elemsize))
|
|
tmp = conv(tmp, types.Types[TUINTPTR])
|
|
n.Nbody.Append(nod(OAS, hn, tmp))
|
|
|
|
var fn *Node
|
|
if types.Haspointers(a.Type.Elem()) {
|
|
// memclrHasPointers(hp, hn)
|
|
fn = mkcall("memclrHasPointers", nil, nil, hp, hn)
|
|
} else {
|
|
// memclrNoHeapPointers(hp, hn)
|
|
fn = mkcall("memclrNoHeapPointers", nil, nil, hp, hn)
|
|
}
|
|
|
|
n.Nbody.Append(fn)
|
|
|
|
// i = len(a) - 1
|
|
v1 = nod(OAS, v1, nod(OSUB, nod(OLEN, a, nil), nodintconst(1)))
|
|
|
|
n.Nbody.Append(v1)
|
|
|
|
n.Left = typecheck(n.Left, Erv)
|
|
n.Left = defaultlit(n.Left, nil)
|
|
typecheckslice(n.Nbody.Slice(), Etop)
|
|
n = walkstmt(n)
|
|
return true
|
|
}
|