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