]> Cypherpunks repositories - gostls13.git/commitdiff
cmd/compile: fix ordering for short-circuiting ops
authorKeith Randall <khr@google.com>
Wed, 6 Mar 2019 01:42:48 +0000 (17:42 -0800)
committerKeith Randall <khr@golang.org>
Wed, 6 Mar 2019 20:04:07 +0000 (20:04 +0000)
Make sure the side effects inside short-circuited operations (&& and ||)
happen correctly.

Before this CL, we attached the side effects to the node itself using
exprInPlace. That caused other side effects in sibling expressions
to get reordered with respect to the short circuit side effect.

Instead, rewrite a && b like:

r := a
if r {
  r = b
}

That code we can keep correctly ordered with respect to other
side-effects extracted from part of a big expression.

exprInPlace seems generally unsafe. But this was the only case where
exprInPlace is called not at the top level of an expression, so I
don't think the other uses can actually trigger an issue (there can't
be a sibling expression). TODO: maybe those cases don't need "in
place", and we can retire that function generally.

This CL needed a small tweak to the SSA generation of OIF so that the
short circuit optimization still triggers. The short circuit optimization
looks for triangle but not diamonds, so don't bother allocating a block
if it will be empty.

Go 1 benchmarks are in the noise.

Fixes #30566

Change-Id: I19c04296bea63cbd6ad05f87a63b005029123610
Reviewed-on: https://go-review.googlesource.com/c/go/+/165617
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: David Chase <drchase@google.com>
src/cmd/compile/internal/gc/order.go
src/cmd/compile/internal/gc/ssa.go
test/checkbce.go
test/fixedbugs/issue30566a.go [new file with mode: 0644]
test/fixedbugs/issue30566b.go [new file with mode: 0644]
test/live.go

index 4848a02bb6a8c4af01444f3db95ebb23da48c669..0098242c799e2a88e690e81458e0645ebb40d9f7 100644 (file)
@@ -1130,14 +1130,40 @@ func (o *Order) expr(n, lhs *Node) *Node {
                }
 
        case OANDAND, OOROR:
-               mark := o.markTemp()
-               n.Left = o.expr(n.Left, nil)
+               // ... = LHS && RHS
+               //
+               // var r bool
+               // r = LHS
+               // if r {       // or !r, for OROR
+               //     r = RHS
+               // }
+               // ... = r
+
+               r := o.newTemp(n.Type, false)
+
+               // Evaluate left-hand side.
+               lhs := o.expr(n.Left, nil)
+               o.out = append(o.out, typecheck(nod(OAS, r, lhs), ctxStmt))
+
+               // Evaluate right-hand side, save generated code.
+               saveout := o.out
+               o.out = nil
+               t := o.markTemp()
+               rhs := o.expr(n.Right, nil)
+               o.out = append(o.out, typecheck(nod(OAS, r, rhs), ctxStmt))
+               o.cleanTemp(t)
+               gen := o.out
+               o.out = saveout
 
-               // Clean temporaries from first branch at beginning of second.
-               // Leave them on the stack so that they can be killed in the outer
-               // context in case the short circuit is taken.
-               n.Right = addinit(n.Right, o.cleanTempNoPop(mark))
-               n.Right = o.exprInPlace(n.Right)
+               // If left-hand side doesn't cause a short-circuit, issue right-hand side.
+               nif := nod(OIF, r, nil)
+               if n.Op == OANDAND {
+                       nif.Nbody.Set(gen)
+               } else {
+                       nif.Rlist.Set(gen)
+               }
+               o.out = append(o.out, nif)
+               n = r
 
        case OCALLFUNC,
                OCALLINTER,
index 95904edd6a063d9a06ec07ea64a97526411d1153..e03988dac26de2e925596e808831fd01cfb189d5 100644 (file)
@@ -994,26 +994,32 @@ func (s *state) stmt(n *Node) {
                s.assign(n.Left, r, deref, skip)
 
        case OIF:
-               bThen := s.f.NewBlock(ssa.BlockPlain)
                bEnd := s.f.NewBlock(ssa.BlockPlain)
-               var bElse *ssa.Block
                var likely int8
                if n.Likely() {
                        likely = 1
                }
+               var bThen *ssa.Block
+               if n.Nbody.Len() != 0 {
+                       bThen = s.f.NewBlock(ssa.BlockPlain)
+               } else {
+                       bThen = bEnd
+               }
+               var bElse *ssa.Block
                if n.Rlist.Len() != 0 {
                        bElse = s.f.NewBlock(ssa.BlockPlain)
-                       s.condBranch(n.Left, bThen, bElse, likely)
                } else {
-                       s.condBranch(n.Left, bThen, bEnd, likely)
+                       bElse = bEnd
                }
+               s.condBranch(n.Left, bThen, bElse, likely)
 
-               s.startBlock(bThen)
-               s.stmtList(n.Nbody)
-               if b := s.endBlock(); b != nil {
-                       b.AddEdgeTo(bEnd)
+               if n.Nbody.Len() != 0 {
+                       s.startBlock(bThen)
+                       s.stmtList(n.Nbody)
+                       if b := s.endBlock(); b != nil {
+                               b.AddEdgeTo(bEnd)
+                       }
                }
-
                if n.Rlist.Len() != 0 {
                        s.startBlock(bElse)
                        s.stmtList(n.Rlist)
index a8f060aa72eb6002fbefd35ab57284f192e25bbf..6a126099bc89accd5f2d3519664648188c8bfdf1 100644 (file)
@@ -33,9 +33,7 @@ func f1(a [256]int, i int) {
 
        if 4 <= i && i < len(a) {
                useInt(a[i])
-               useInt(a[i-1]) // ERROR "Found IsInBounds$"
-               // TODO: 'if 4 <= i && i < len(a)' gets rewritten to 'if uint(i - 4) < 256 - 4',
-               // which the bounds checker cannot yet use to infer that the next line doesn't need a bounds check.
+               useInt(a[i-1])
                useInt(a[i-4])
        }
 }
diff --git a/test/fixedbugs/issue30566a.go b/test/fixedbugs/issue30566a.go
new file mode 100644 (file)
index 0000000..5d736cc
--- /dev/null
@@ -0,0 +1,23 @@
+// run
+
+// Copyright 2019 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 "fmt"
+
+//go:noinline
+func ident(s string) string { return s }
+
+func returnSecond(x bool, s string) string { return s }
+
+func identWrapper(s string) string { return ident(s) }
+
+func main() {
+       got := returnSecond((false || identWrapper("bad") != ""), ident("good"))
+       if got != "good" {
+               panic(fmt.Sprintf("wanted \"good\", got \"%s\"", got))
+       }
+}
diff --git a/test/fixedbugs/issue30566b.go b/test/fixedbugs/issue30566b.go
new file mode 100644 (file)
index 0000000..92e0644
--- /dev/null
@@ -0,0 +1,27 @@
+// run
+
+// Copyright 2019 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 (
+       "bytes"
+       "fmt"
+)
+
+func main() {
+       _, _ = false || g(1), g(2)
+       if !bytes.Equal(x, []byte{1, 2}) {
+               panic(fmt.Sprintf("wanted [1,2], got %v", x))
+       }
+}
+
+var x []byte
+
+//go:noinline
+func g(b byte) bool {
+       x = append(x, b)
+       return false
+}
index a508947afc5a838e78b4208cc9540c9b20d7cd31..e7134eca0cede8dfafb050ce434197569eb21d7e 100644 (file)
@@ -572,7 +572,7 @@ func f36() {
 func f37() {
        if (m33[byteptr()] == 0 || // ERROR "stack object .autotmp_[0-9]+ interface \{\}"
                m33[byteptr()] == 0) && // ERROR "stack object .autotmp_[0-9]+ interface \{\}"
-               m33[byteptr()] == 0 { // ERROR "stack object .autotmp_[0-9]+ interface \{\}"
+               m33[byteptr()] == 0 {
                printnl()
                return
        }
@@ -697,9 +697,10 @@ func f41(p, q *int) (r *int) { // ERROR "live at entry to f41: p q$"
 
 func f42() {
        var p, q, r int
-       f43([]*int{&p,&q,&r}) // ERROR "stack object .autotmp_[0-9]+ \[3\]\*int$"
-       f43([]*int{&p,&r,&q})
-       f43([]*int{&q,&p,&r})
+       f43([]*int{&p, &q, &r}) // ERROR "stack object .autotmp_[0-9]+ \[3\]\*int$"
+       f43([]*int{&p, &r, &q})
+       f43([]*int{&q, &p, &r})
 }
+
 //go:noescape
 func f43(a []*int)