| // 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 |
| } |