mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
[This CL is part of a sequence implementing the proposal #51082. The design doc is at https://go.dev/s/godocfmt-design.] Run the updated gofmt, which reformats doc comments, on the main repository. Vendored files are excluded. For #51082. Change-Id: I7332f099b60f716295fb34719c98c04eb1a85407 Reviewed-on: https://go-review.googlesource.com/c/go/+/384268 Reviewed-by: Jonathan Amsterdam <jba@google.com> Reviewed-by: Ian Lance Taylor <iant@golang.org>
107 lines
4 KiB
Go
107 lines
4 KiB
Go
// Copyright 2015 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 ssa
|
|
|
|
import (
|
|
"cmd/compile/internal/types"
|
|
"testing"
|
|
)
|
|
|
|
func TestShiftConstAMD64(t *testing.T) {
|
|
c := testConfig(t)
|
|
fun := makeConstShiftFunc(c, 18, OpLsh64x64, c.config.Types.UInt64)
|
|
checkOpcodeCounts(t, fun.f, map[Op]int{OpAMD64SHLQconst: 1, OpAMD64CMPQconst: 0, OpAMD64ANDQconst: 0})
|
|
|
|
fun = makeConstShiftFunc(c, 66, OpLsh64x64, c.config.Types.UInt64)
|
|
checkOpcodeCounts(t, fun.f, map[Op]int{OpAMD64SHLQconst: 0, OpAMD64CMPQconst: 0, OpAMD64ANDQconst: 0})
|
|
|
|
fun = makeConstShiftFunc(c, 18, OpRsh64Ux64, c.config.Types.UInt64)
|
|
checkOpcodeCounts(t, fun.f, map[Op]int{OpAMD64SHRQconst: 1, OpAMD64CMPQconst: 0, OpAMD64ANDQconst: 0})
|
|
|
|
fun = makeConstShiftFunc(c, 66, OpRsh64Ux64, c.config.Types.UInt64)
|
|
checkOpcodeCounts(t, fun.f, map[Op]int{OpAMD64SHRQconst: 0, OpAMD64CMPQconst: 0, OpAMD64ANDQconst: 0})
|
|
|
|
fun = makeConstShiftFunc(c, 18, OpRsh64x64, c.config.Types.Int64)
|
|
checkOpcodeCounts(t, fun.f, map[Op]int{OpAMD64SARQconst: 1, OpAMD64CMPQconst: 0})
|
|
|
|
fun = makeConstShiftFunc(c, 66, OpRsh64x64, c.config.Types.Int64)
|
|
checkOpcodeCounts(t, fun.f, map[Op]int{OpAMD64SARQconst: 1, OpAMD64CMPQconst: 0})
|
|
}
|
|
|
|
func makeConstShiftFunc(c *Conf, amount int64, op Op, typ *types.Type) fun {
|
|
ptyp := c.config.Types.BytePtr
|
|
fun := c.Fun("entry",
|
|
Bloc("entry",
|
|
Valu("mem", OpInitMem, types.TypeMem, 0, nil),
|
|
Valu("SP", OpSP, c.config.Types.Uintptr, 0, nil),
|
|
Valu("argptr", OpOffPtr, ptyp, 8, nil, "SP"),
|
|
Valu("resptr", OpOffPtr, ptyp, 16, nil, "SP"),
|
|
Valu("load", OpLoad, typ, 0, nil, "argptr", "mem"),
|
|
Valu("c", OpConst64, c.config.Types.UInt64, amount, nil),
|
|
Valu("shift", op, typ, 0, nil, "load", "c"),
|
|
Valu("store", OpStore, types.TypeMem, 0, c.config.Types.UInt64, "resptr", "shift", "mem"),
|
|
Exit("store")))
|
|
Compile(fun.f)
|
|
return fun
|
|
}
|
|
|
|
func TestShiftToExtensionAMD64(t *testing.T) {
|
|
c := testConfig(t)
|
|
// Test that eligible pairs of constant shifts are converted to extensions.
|
|
// For example:
|
|
// (uint64(x) << 32) >> 32 -> uint64(uint32(x))
|
|
ops := map[Op]int{
|
|
OpAMD64SHLQconst: 0, OpAMD64SHLLconst: 0,
|
|
OpAMD64SHRQconst: 0, OpAMD64SHRLconst: 0,
|
|
OpAMD64SARQconst: 0, OpAMD64SARLconst: 0,
|
|
}
|
|
tests := [...]struct {
|
|
amount int64
|
|
left, right Op
|
|
typ *types.Type
|
|
}{
|
|
// unsigned
|
|
{56, OpLsh64x64, OpRsh64Ux64, c.config.Types.UInt64},
|
|
{48, OpLsh64x64, OpRsh64Ux64, c.config.Types.UInt64},
|
|
{32, OpLsh64x64, OpRsh64Ux64, c.config.Types.UInt64},
|
|
{24, OpLsh32x64, OpRsh32Ux64, c.config.Types.UInt32},
|
|
{16, OpLsh32x64, OpRsh32Ux64, c.config.Types.UInt32},
|
|
{8, OpLsh16x64, OpRsh16Ux64, c.config.Types.UInt16},
|
|
// signed
|
|
{56, OpLsh64x64, OpRsh64x64, c.config.Types.Int64},
|
|
{48, OpLsh64x64, OpRsh64x64, c.config.Types.Int64},
|
|
{32, OpLsh64x64, OpRsh64x64, c.config.Types.Int64},
|
|
{24, OpLsh32x64, OpRsh32x64, c.config.Types.Int32},
|
|
{16, OpLsh32x64, OpRsh32x64, c.config.Types.Int32},
|
|
{8, OpLsh16x64, OpRsh16x64, c.config.Types.Int16},
|
|
}
|
|
for _, tc := range tests {
|
|
fun := makeShiftExtensionFunc(c, tc.amount, tc.left, tc.right, tc.typ)
|
|
checkOpcodeCounts(t, fun.f, ops)
|
|
}
|
|
}
|
|
|
|
// makeShiftExtensionFunc generates a function containing:
|
|
//
|
|
// (rshift (lshift (Const64 [amount])) (Const64 [amount]))
|
|
//
|
|
// This may be equivalent to a sign or zero extension.
|
|
func makeShiftExtensionFunc(c *Conf, amount int64, lshift, rshift Op, typ *types.Type) fun {
|
|
ptyp := c.config.Types.BytePtr
|
|
fun := c.Fun("entry",
|
|
Bloc("entry",
|
|
Valu("mem", OpInitMem, types.TypeMem, 0, nil),
|
|
Valu("SP", OpSP, c.config.Types.Uintptr, 0, nil),
|
|
Valu("argptr", OpOffPtr, ptyp, 8, nil, "SP"),
|
|
Valu("resptr", OpOffPtr, ptyp, 16, nil, "SP"),
|
|
Valu("load", OpLoad, typ, 0, nil, "argptr", "mem"),
|
|
Valu("c", OpConst64, c.config.Types.UInt64, amount, nil),
|
|
Valu("lshift", lshift, typ, 0, nil, "load", "c"),
|
|
Valu("rshift", rshift, typ, 0, nil, "lshift", "c"),
|
|
Valu("store", OpStore, types.TypeMem, 0, c.config.Types.UInt64, "resptr", "rshift", "mem"),
|
|
Exit("store")))
|
|
Compile(fun.f)
|
|
return fun
|
|
}
|