mirror of https://github.com/golang/go.git
cmd/compile: fix comma-ok assignments for non-boolean ok
Passes toolstash -cmp. Fixes #16870. Change-Id: I70dc3bbb3cd3031826e5a54b96ba1ea603c282d1 Reviewed-on: https://go-review.googlesource.com/27910 Run-TryBot: Matthew Dempsky <mdempsky@google.com> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Josh Bleecher Snyder <josharian@gmail.com>
This commit is contained in:
parent
150de948ee
commit
fb273fc3a3
|
|
@ -573,16 +573,29 @@ func orderstmt(n *Node, order *Order) {
|
||||||
|
|
||||||
orderexprlist(n.List, order)
|
orderexprlist(n.List, order)
|
||||||
n.Rlist.First().Left = orderexpr(n.Rlist.First().Left, order, nil) // i in i.(T)
|
n.Rlist.First().Left = orderexpr(n.Rlist.First().Left, order, nil) // i in i.(T)
|
||||||
if isblank(n.List.First()) {
|
|
||||||
order.out = append(order.out, n)
|
var tmp1, tmp2 *Node
|
||||||
} else {
|
if !isblank(n.List.First()) {
|
||||||
typ := n.Rlist.First().Type
|
typ := n.Rlist.First().Type
|
||||||
tmp1 := ordertemp(typ, order, haspointers(typ))
|
tmp1 = ordertemp(typ, order, haspointers(typ))
|
||||||
order.out = append(order.out, n)
|
}
|
||||||
|
if !isblank(n.List.Second()) && !n.List.Second().Type.IsBoolean() {
|
||||||
|
tmp2 = ordertemp(Types[TBOOL], order, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
order.out = append(order.out, n)
|
||||||
|
|
||||||
|
if tmp1 != nil {
|
||||||
r := Nod(OAS, n.List.First(), tmp1)
|
r := Nod(OAS, n.List.First(), tmp1)
|
||||||
r = typecheck(r, Etop)
|
r = typecheck(r, Etop)
|
||||||
ordermapassign(r, order)
|
ordermapassign(r, order)
|
||||||
n.List.Set([]*Node{tmp1, n.List.Second()})
|
n.List.SetIndex(0, tmp1)
|
||||||
|
}
|
||||||
|
if tmp2 != nil {
|
||||||
|
r := okas(n.List.Second(), tmp2)
|
||||||
|
r = typecheck(r, Etop)
|
||||||
|
ordermapassign(r, order)
|
||||||
|
n.List.SetIndex(1, tmp2)
|
||||||
}
|
}
|
||||||
|
|
||||||
cleantemp(t, order)
|
cleantemp(t, order)
|
||||||
|
|
@ -596,17 +609,12 @@ func orderstmt(n *Node, order *Order) {
|
||||||
n.Rlist.First().Left = orderexpr(n.Rlist.First().Left, order, nil) // arg to recv
|
n.Rlist.First().Left = orderexpr(n.Rlist.First().Left, order, nil) // arg to recv
|
||||||
ch := n.Rlist.First().Left.Type
|
ch := n.Rlist.First().Left.Type
|
||||||
tmp1 := ordertemp(ch.Elem(), order, haspointers(ch.Elem()))
|
tmp1 := ordertemp(ch.Elem(), order, haspointers(ch.Elem()))
|
||||||
var tmp2 *Node
|
tmp2 := ordertemp(Types[TBOOL], order, false)
|
||||||
if !isblank(n.List.Second()) {
|
|
||||||
tmp2 = ordertemp(n.List.Second().Type, order, false)
|
|
||||||
} else {
|
|
||||||
tmp2 = ordertemp(Types[TBOOL], order, false)
|
|
||||||
}
|
|
||||||
order.out = append(order.out, n)
|
order.out = append(order.out, n)
|
||||||
r := Nod(OAS, n.List.First(), tmp1)
|
r := Nod(OAS, n.List.First(), tmp1)
|
||||||
r = typecheck(r, Etop)
|
r = typecheck(r, Etop)
|
||||||
ordermapassign(r, order)
|
ordermapassign(r, order)
|
||||||
r = Nod(OAS, n.List.Second(), tmp2)
|
r = okas(n.List.Second(), tmp2)
|
||||||
r = typecheck(r, Etop)
|
r = typecheck(r, Etop)
|
||||||
ordermapassign(r, order)
|
ordermapassign(r, order)
|
||||||
n.List.Set([]*Node{tmp1, tmp2})
|
n.List.Set([]*Node{tmp1, tmp2})
|
||||||
|
|
@ -882,8 +890,8 @@ func orderstmt(n *Node, order *Order) {
|
||||||
n2.Ninit.Append(tmp2)
|
n2.Ninit.Append(tmp2)
|
||||||
}
|
}
|
||||||
|
|
||||||
r.List.Set1(ordertemp(tmp1.Type, order, false))
|
r.List.Set1(ordertemp(Types[TBOOL], order, false))
|
||||||
tmp2 = Nod(OAS, tmp1, r.List.First())
|
tmp2 = okas(tmp1, r.List.First())
|
||||||
tmp2 = typecheck(tmp2, Etop)
|
tmp2 = typecheck(tmp2, Etop)
|
||||||
n2.Ninit.Append(tmp2)
|
n2.Ninit.Append(tmp2)
|
||||||
}
|
}
|
||||||
|
|
@ -1206,3 +1214,12 @@ func orderexpr(n *Node, order *Order, lhs *Node) *Node {
|
||||||
lineno = lno
|
lineno = lno
|
||||||
return n
|
return n
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// okas creates and returns an assignment of val to ok,
|
||||||
|
// including an explicit conversion if necessary.
|
||||||
|
func okas(ok, val *Node) *Node {
|
||||||
|
if !isblank(ok) {
|
||||||
|
val = conv(val, ok.Type)
|
||||||
|
}
|
||||||
|
return Nod(OAS, ok, val)
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -842,8 +842,9 @@ opswitch:
|
||||||
}
|
}
|
||||||
n1.Etype = 1 // addr does not escape
|
n1.Etype = 1 // addr does not escape
|
||||||
fn := chanfn("chanrecv2", 2, r.Left.Type)
|
fn := chanfn("chanrecv2", 2, r.Left.Type)
|
||||||
r = mkcall1(fn, n.List.Second().Type, init, typename(r.Left.Type), r.Left, n1)
|
ok := n.List.Second()
|
||||||
n = Nod(OAS, n.List.Second(), r)
|
call := mkcall1(fn, ok.Type, init, typename(r.Left.Type), r.Left, n1)
|
||||||
|
n = Nod(OAS, ok, call)
|
||||||
n = typecheck(n, Etop)
|
n = typecheck(n, Etop)
|
||||||
|
|
||||||
// a,b = m[i];
|
// a,b = m[i];
|
||||||
|
|
@ -898,8 +899,8 @@ opswitch:
|
||||||
// mapaccess2* returns a typed bool, but due to spec changes,
|
// mapaccess2* returns a typed bool, but due to spec changes,
|
||||||
// the boolean result of i.(T) is now untyped so we make it the
|
// the boolean result of i.(T) is now untyped so we make it the
|
||||||
// same type as the variable on the lhs.
|
// same type as the variable on the lhs.
|
||||||
if !isblank(n.List.Second()) {
|
if ok := n.List.Second(); !isblank(ok) && ok.Type.IsBoolean() {
|
||||||
r.Type.Field(1).Type = n.List.Second().Type
|
r.Type.Field(1).Type = ok.Type
|
||||||
}
|
}
|
||||||
n.Rlist.Set1(r)
|
n.Rlist.Set1(r)
|
||||||
n.Op = OAS2FUNC
|
n.Op = OAS2FUNC
|
||||||
|
|
@ -933,6 +934,7 @@ opswitch:
|
||||||
|
|
||||||
case OAS2DOTTYPE:
|
case OAS2DOTTYPE:
|
||||||
e := n.Rlist.First() // i.(T)
|
e := n.Rlist.First() // i.(T)
|
||||||
|
|
||||||
// TODO(rsc): The Isfat is for consistency with componentgen and orderexpr.
|
// TODO(rsc): The Isfat is for consistency with componentgen and orderexpr.
|
||||||
// It needs to be removed in all three places.
|
// It needs to be removed in all three places.
|
||||||
// That would allow inlining x.(struct{*int}) the same as x.(*int).
|
// That would allow inlining x.(struct{*int}) the same as x.(*int).
|
||||||
|
|
@ -957,6 +959,9 @@ opswitch:
|
||||||
if !isblank(ok) {
|
if !isblank(ok) {
|
||||||
oktype = ok.Type
|
oktype = ok.Type
|
||||||
}
|
}
|
||||||
|
if !oktype.IsBoolean() {
|
||||||
|
Fatalf("orderstmt broken: got %L, want boolean", oktype)
|
||||||
|
}
|
||||||
|
|
||||||
fromKind := from.Type.iet()
|
fromKind := from.Type.iet()
|
||||||
toKind := t.iet()
|
toKind := t.iet()
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,140 @@
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
"reflect"
|
||||||
|
)
|
||||||
|
|
||||||
|
func test(got, want interface{}) {
|
||||||
|
if !reflect.DeepEqual(got, want) {
|
||||||
|
log.Fatalf("got %v, want %v", got, want)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
var i int
|
||||||
|
var ip *int
|
||||||
|
var ok interface{}
|
||||||
|
|
||||||
|
// Channel receives.
|
||||||
|
c := make(chan int, 1)
|
||||||
|
c2 := make(chan int)
|
||||||
|
|
||||||
|
c <- 42
|
||||||
|
i, ok = <-c
|
||||||
|
test(i, 42)
|
||||||
|
test(ok, true)
|
||||||
|
|
||||||
|
c <- 42
|
||||||
|
_, ok = <-c
|
||||||
|
test(ok, true)
|
||||||
|
|
||||||
|
c <- 42
|
||||||
|
select {
|
||||||
|
case i, ok = <-c:
|
||||||
|
test(i, 42)
|
||||||
|
test(ok, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
c <- 42
|
||||||
|
select {
|
||||||
|
case _, ok = <-c:
|
||||||
|
test(ok, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
c <- 42
|
||||||
|
select {
|
||||||
|
case i, ok = <-c:
|
||||||
|
test(i, 42)
|
||||||
|
test(ok, true)
|
||||||
|
default:
|
||||||
|
log.Fatal("bad select")
|
||||||
|
}
|
||||||
|
|
||||||
|
c <- 42
|
||||||
|
select {
|
||||||
|
case _, ok = <-c:
|
||||||
|
test(ok, true)
|
||||||
|
default:
|
||||||
|
log.Fatal("bad select")
|
||||||
|
}
|
||||||
|
|
||||||
|
c <- 42
|
||||||
|
select {
|
||||||
|
case i, ok = <-c:
|
||||||
|
test(i, 42)
|
||||||
|
test(ok, true)
|
||||||
|
case <-c2:
|
||||||
|
log.Fatal("bad select")
|
||||||
|
}
|
||||||
|
|
||||||
|
c <- 42
|
||||||
|
select {
|
||||||
|
case _, ok = <-c:
|
||||||
|
test(ok, true)
|
||||||
|
case <-c2:
|
||||||
|
log.Fatal("bad select")
|
||||||
|
}
|
||||||
|
|
||||||
|
close(c)
|
||||||
|
i, ok = <-c
|
||||||
|
test(i, 0)
|
||||||
|
test(ok, false)
|
||||||
|
|
||||||
|
_, ok = <-c
|
||||||
|
test(ok, false)
|
||||||
|
|
||||||
|
// Map indexing.
|
||||||
|
m := make(map[int]int)
|
||||||
|
|
||||||
|
i, ok = m[0]
|
||||||
|
test(i, 0)
|
||||||
|
test(ok, false)
|
||||||
|
|
||||||
|
_, ok = m[0]
|
||||||
|
test(ok, false)
|
||||||
|
|
||||||
|
m[0] = 42
|
||||||
|
i, ok = m[0]
|
||||||
|
test(i, 42)
|
||||||
|
test(ok, true)
|
||||||
|
|
||||||
|
_, ok = m[0]
|
||||||
|
test(ok, true)
|
||||||
|
|
||||||
|
// Type assertions.
|
||||||
|
var u interface{}
|
||||||
|
|
||||||
|
i, ok = u.(int)
|
||||||
|
test(i, 0)
|
||||||
|
test(ok, false)
|
||||||
|
|
||||||
|
ip, ok = u.(*int)
|
||||||
|
test(ip, (*int)(nil))
|
||||||
|
test(ok, false)
|
||||||
|
|
||||||
|
_, ok = u.(int)
|
||||||
|
test(ok, false)
|
||||||
|
|
||||||
|
u = 42
|
||||||
|
i, ok = u.(int)
|
||||||
|
test(i, 42)
|
||||||
|
test(ok, true)
|
||||||
|
|
||||||
|
_, ok = u.(int)
|
||||||
|
test(ok, true)
|
||||||
|
|
||||||
|
u = &i
|
||||||
|
ip, ok = u.(*int)
|
||||||
|
test(ip, &i)
|
||||||
|
test(ok, true)
|
||||||
|
|
||||||
|
_, ok = u.(*int)
|
||||||
|
test(ok, true)
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue