cmd/compile: deduplicate OpArg's across types

For in-register arguments, it must have only a single copy of it
present in the function. If there are multiple copies, it confuses
the register allocator, as they are in the same register.

Change-Id: I55cb06746f08aa7c9168026d0f411bce0a9f93f4
Reviewed-on: https://go-review.googlesource.com/c/go/+/306330
Trust: Cherry Zhang <cherryyz@google.com>
Run-TryBot: Cherry Zhang <cherryyz@google.com>
TryBot-Result: Go Bot <gobot@golang.org>
Reviewed-by: David Chase <drchase@google.com>
This commit is contained in:
Cherry Zhang 2021-03-31 12:41:20 -04:00
parent 4acefa07b1
commit 5d6581d747
2 changed files with 84 additions and 1 deletions

View File

@ -1359,7 +1359,42 @@ func expandCalls(f *Func) {
}
}
// Step 5: elide any copies introduced.
// Step 5: dedup OpArgXXXReg values. Mostly it is already dedup'd by commonArgs,
// but there are cases that we have same OpArgXXXReg values with different types.
// E.g. string is sometimes decomposed as { *int8, int }, sometimes as { unsafe.Pointer, uintptr }.
// (Can we avoid that?)
var IArg, FArg [32]*Value
for _, v := range f.Entry.Values {
switch v.Op {
case OpArgIntReg:
i := v.AuxInt
if w := IArg[i]; w != nil {
if w.Type.Width != v.Type.Width {
f.Fatalf("incompatible OpArgIntReg [%d]: %v and %v", i, v, w)
}
if w.Type.IsUnsafePtr() && !v.Type.IsUnsafePtr() {
// Update unsafe.Pointer type if we know the actual pointer type.
w.Type = v.Type
}
// TODO: don't dedup pointer and scalar? Rewrite to OpConvert? Can it happen?
v.copyOf(w)
} else {
IArg[i] = v
}
case OpArgFloatReg:
i := v.AuxInt
if w := FArg[i]; w != nil {
if w.Type.Width != v.Type.Width {
f.Fatalf("incompatible OpArgFloatReg [%d]: %v and %v", i, v, w)
}
v.copyOf(w)
} else {
FArg[i] = v
}
}
}
// Step 6: elide any copies introduced.
for _, b := range f.Blocks {
for _, v := range b.Values {
for i, a := range v.Args {

View File

@ -0,0 +1,48 @@
// run
// Copyright 2021 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
const p0exp = "foo"
const p1exp = 10101
const p2exp = 3030303
const p3exp = 505050505
const p4exp = 70707070707
//go:noinline
//go:registerparams
func callee(p0 string, p1 uint64, p2 uint64, p3 uint64, p4 uint64) {
if p0 != p0exp {
panic("bad p0")
}
if p1 != p1exp {
panic("bad p1")
}
if p2 != p2exp {
panic("bad p2")
}
if p3 != p3exp {
panic("bad p3")
}
if p4 != p4exp {
panic("bad p4")
}
defer func(p0 string, p2 uint64) {
if p0 != p0exp {
panic("defer bad p0")
}
if p1 != p1exp {
panic("defer bad p1")
}
if p2 != p2exp {
panic("defer bad p2")
}
}(p0, p2)
}
func main() {
callee(p0exp, p1exp, p2exp, p3exp, p4exp)
}