mirror of
https://github.com/golang/go.git
synced 2025-05-05 15:43:04 +00:00
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:
parent
4acefa07b1
commit
5d6581d747
@ -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 {
|
||||
|
48
test/abi/defer_aggregate.go
Normal file
48
test/abi/defer_aggregate.go
Normal 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)
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user