1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
| // 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
| }
|
|