171 lines
4.5 KiB
Go
171 lines
4.5 KiB
Go
package backend
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"github.com/tetratelabs/wazero/internal/engine/wazevo/backend/regalloc"
|
|
"github.com/tetratelabs/wazero/internal/engine/wazevo/ssa"
|
|
)
|
|
|
|
type (
|
|
// FunctionABI represents the ABI information for a function which corresponds to a ssa.Signature.
|
|
FunctionABI struct {
|
|
Initialized bool
|
|
|
|
Args, Rets []ABIArg
|
|
ArgStackSize, RetStackSize int64
|
|
|
|
ArgIntRealRegs byte
|
|
ArgFloatRealRegs byte
|
|
RetIntRealRegs byte
|
|
RetFloatRealRegs byte
|
|
}
|
|
|
|
// ABIArg represents either argument or return value's location.
|
|
ABIArg struct {
|
|
// Index is the index of the argument.
|
|
Index int
|
|
// Kind is the kind of the argument.
|
|
Kind ABIArgKind
|
|
// Reg is valid if Kind == ABIArgKindReg.
|
|
// This VReg must be based on RealReg.
|
|
Reg regalloc.VReg
|
|
// Offset is valid if Kind == ABIArgKindStack.
|
|
// This is the offset from the beginning of either arg or ret stack slot.
|
|
Offset int64
|
|
// Type is the type of the argument.
|
|
Type ssa.Type
|
|
}
|
|
|
|
// ABIArgKind is the kind of ABI argument.
|
|
ABIArgKind byte
|
|
)
|
|
|
|
const (
|
|
// ABIArgKindReg represents an argument passed in a register.
|
|
ABIArgKindReg = iota
|
|
// ABIArgKindStack represents an argument passed in the stack.
|
|
ABIArgKindStack
|
|
)
|
|
|
|
// String implements fmt.Stringer.
|
|
func (a *ABIArg) String() string {
|
|
return fmt.Sprintf("args[%d]: %s", a.Index, a.Kind)
|
|
}
|
|
|
|
// String implements fmt.Stringer.
|
|
func (a ABIArgKind) String() string {
|
|
switch a {
|
|
case ABIArgKindReg:
|
|
return "reg"
|
|
case ABIArgKindStack:
|
|
return "stack"
|
|
default:
|
|
panic("BUG")
|
|
}
|
|
}
|
|
|
|
// Init initializes the abiImpl for the given signature.
|
|
func (a *FunctionABI) Init(sig *ssa.Signature, argResultInts, argResultFloats []regalloc.RealReg) {
|
|
if len(a.Rets) < len(sig.Results) {
|
|
a.Rets = make([]ABIArg, len(sig.Results))
|
|
}
|
|
a.Rets = a.Rets[:len(sig.Results)]
|
|
a.RetStackSize = a.setABIArgs(a.Rets, sig.Results, argResultInts, argResultFloats)
|
|
if argsNum := len(sig.Params); len(a.Args) < argsNum {
|
|
a.Args = make([]ABIArg, argsNum)
|
|
}
|
|
a.Args = a.Args[:len(sig.Params)]
|
|
a.ArgStackSize = a.setABIArgs(a.Args, sig.Params, argResultInts, argResultFloats)
|
|
|
|
// Gather the real registers usages in arg/return.
|
|
a.ArgIntRealRegs, a.ArgFloatRealRegs = 0, 0
|
|
a.RetIntRealRegs, a.RetFloatRealRegs = 0, 0
|
|
for i := range a.Rets {
|
|
r := &a.Rets[i]
|
|
if r.Kind == ABIArgKindReg {
|
|
if r.Type.IsInt() {
|
|
a.RetIntRealRegs++
|
|
} else {
|
|
a.RetFloatRealRegs++
|
|
}
|
|
}
|
|
}
|
|
for i := range a.Args {
|
|
arg := &a.Args[i]
|
|
if arg.Kind == ABIArgKindReg {
|
|
if arg.Type.IsInt() {
|
|
a.ArgIntRealRegs++
|
|
} else {
|
|
a.ArgFloatRealRegs++
|
|
}
|
|
}
|
|
}
|
|
|
|
a.Initialized = true
|
|
}
|
|
|
|
// setABIArgs sets the ABI arguments in the given slice. This assumes that len(s) >= len(types)
|
|
// where if len(s) > len(types), the last elements of s is for the multi-return slot.
|
|
func (a *FunctionABI) setABIArgs(s []ABIArg, types []ssa.Type, ints, floats []regalloc.RealReg) (stackSize int64) {
|
|
il, fl := len(ints), len(floats)
|
|
|
|
var stackOffset int64
|
|
intParamIndex, floatParamIndex := 0, 0
|
|
for i, typ := range types {
|
|
arg := &s[i]
|
|
arg.Index = i
|
|
arg.Type = typ
|
|
if typ.IsInt() {
|
|
if intParamIndex >= il {
|
|
arg.Kind = ABIArgKindStack
|
|
const slotSize = 8 // Align 8 bytes.
|
|
arg.Offset = stackOffset
|
|
stackOffset += slotSize
|
|
} else {
|
|
arg.Kind = ABIArgKindReg
|
|
arg.Reg = regalloc.FromRealReg(ints[intParamIndex], regalloc.RegTypeInt)
|
|
intParamIndex++
|
|
}
|
|
} else {
|
|
if floatParamIndex >= fl {
|
|
arg.Kind = ABIArgKindStack
|
|
slotSize := int64(8) // Align at least 8 bytes.
|
|
if typ.Bits() == 128 { // Vector.
|
|
slotSize = 16
|
|
}
|
|
arg.Offset = stackOffset
|
|
stackOffset += slotSize
|
|
} else {
|
|
arg.Kind = ABIArgKindReg
|
|
arg.Reg = regalloc.FromRealReg(floats[floatParamIndex], regalloc.RegTypeFloat)
|
|
floatParamIndex++
|
|
}
|
|
}
|
|
}
|
|
return stackOffset
|
|
}
|
|
|
|
func (a *FunctionABI) AlignedArgResultStackSlotSize() uint32 {
|
|
stackSlotSize := a.RetStackSize + a.ArgStackSize
|
|
// Align stackSlotSize to 16 bytes.
|
|
stackSlotSize = (stackSlotSize + 15) &^ 15
|
|
// Check overflow 32-bit.
|
|
if stackSlotSize > 0xFFFFFFFF {
|
|
panic("ABI stack slot size overflow")
|
|
}
|
|
return uint32(stackSlotSize)
|
|
}
|
|
|
|
func (a *FunctionABI) ABIInfoAsUint64() uint64 {
|
|
return uint64(a.ArgIntRealRegs)<<56 |
|
|
uint64(a.ArgFloatRealRegs)<<48 |
|
|
uint64(a.RetIntRealRegs)<<40 |
|
|
uint64(a.RetFloatRealRegs)<<32 |
|
|
uint64(a.AlignedArgResultStackSlotSize())
|
|
}
|
|
|
|
func ABIInfoFromUint64(info uint64) (argIntRealRegs, argFloatRealRegs, retIntRealRegs, retFloatRealRegs byte, stackSlotSize uint32) {
|
|
return byte(info >> 56), byte(info >> 48), byte(info >> 40), byte(info >> 32), uint32(info)
|
|
}
|