github.com/go-asm/go@v1.21.1-0.20240213172139-40c5ead50c48/cmd/compile/test/testdata/loadstore_test.go (about) 1 // Copyright 2015 The Go Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 // Tests load/store ordering 6 7 package main 8 9 import "testing" 10 11 // testLoadStoreOrder tests for reordering of stores/loads. 12 func testLoadStoreOrder(t *testing.T) { 13 z := uint32(1000) 14 if testLoadStoreOrder_ssa(&z, 100) == 0 { 15 t.Errorf("testLoadStoreOrder failed") 16 } 17 } 18 19 //go:noinline 20 func testLoadStoreOrder_ssa(z *uint32, prec uint) int { 21 old := *z // load 22 *z = uint32(prec) // store 23 if *z < old { // load 24 return 1 25 } 26 return 0 27 } 28 29 func testStoreSize(t *testing.T) { 30 a := [4]uint16{11, 22, 33, 44} 31 testStoreSize_ssa(&a[0], &a[2], 77) 32 want := [4]uint16{77, 22, 33, 44} 33 if a != want { 34 t.Errorf("testStoreSize failed. want = %d, got = %d", want, a) 35 } 36 } 37 38 //go:noinline 39 func testStoreSize_ssa(p *uint16, q *uint16, v uint32) { 40 // Test to make sure that (Store ptr (Trunc32to16 val) mem) 41 // does not end up as a 32-bit store. It must stay a 16 bit store 42 // even when Trunc32to16 is rewritten to be a nop. 43 // To ensure that we get rewrite the Trunc32to16 before 44 // we rewrite the Store, we force the truncate into an 45 // earlier basic block by using it on both branches. 46 w := uint16(v) 47 if p != nil { 48 *p = w 49 } else { 50 *q = w 51 } 52 } 53 54 //go:noinline 55 func testExtStore_ssa(p *byte, b bool) int { 56 x := *p 57 *p = 7 58 if b { 59 return int(x) 60 } 61 return 0 62 } 63 64 func testExtStore(t *testing.T) { 65 const start = 8 66 var b byte = start 67 if got := testExtStore_ssa(&b, true); got != start { 68 t.Errorf("testExtStore failed. want = %d, got = %d", start, got) 69 } 70 } 71 72 var b int 73 74 // testDeadStorePanic_ssa ensures that we don't optimize away stores 75 // that could be read by after recover(). Modeled after fixedbugs/issue1304. 76 // 77 //go:noinline 78 func testDeadStorePanic_ssa(a int) (r int) { 79 defer func() { 80 recover() 81 r = a 82 }() 83 a = 2 // store 84 b := a - a // optimized to zero 85 c := 4 86 a = c / b // store, but panics 87 a = 3 // store 88 r = a 89 return 90 } 91 92 func testDeadStorePanic(t *testing.T) { 93 if want, got := 2, testDeadStorePanic_ssa(1); want != got { 94 t.Errorf("testDeadStorePanic failed. want = %d, got = %d", want, got) 95 } 96 } 97 98 //go:noinline 99 func loadHitStore8(x int8, p *int8) int32 { 100 x *= x // try to trash high bits (arch-dependent) 101 *p = x // store 102 return int32(*p) // load and cast 103 } 104 105 //go:noinline 106 func loadHitStoreU8(x uint8, p *uint8) uint32 { 107 x *= x // try to trash high bits (arch-dependent) 108 *p = x // store 109 return uint32(*p) // load and cast 110 } 111 112 //go:noinline 113 func loadHitStore16(x int16, p *int16) int32 { 114 x *= x // try to trash high bits (arch-dependent) 115 *p = x // store 116 return int32(*p) // load and cast 117 } 118 119 //go:noinline 120 func loadHitStoreU16(x uint16, p *uint16) uint32 { 121 x *= x // try to trash high bits (arch-dependent) 122 *p = x // store 123 return uint32(*p) // load and cast 124 } 125 126 //go:noinline 127 func loadHitStore32(x int32, p *int32) int64 { 128 x *= x // try to trash high bits (arch-dependent) 129 *p = x // store 130 return int64(*p) // load and cast 131 } 132 133 //go:noinline 134 func loadHitStoreU32(x uint32, p *uint32) uint64 { 135 x *= x // try to trash high bits (arch-dependent) 136 *p = x // store 137 return uint64(*p) // load and cast 138 } 139 140 func testLoadHitStore(t *testing.T) { 141 // Test that sign/zero extensions are kept when a load-hit-store 142 // is replaced by a register-register move. 143 { 144 var in int8 = (1 << 6) + 1 145 var p int8 146 got := loadHitStore8(in, &p) 147 want := int32(in * in) 148 if got != want { 149 t.Errorf("testLoadHitStore (int8) failed. want = %d, got = %d", want, got) 150 } 151 } 152 { 153 var in uint8 = (1 << 6) + 1 154 var p uint8 155 got := loadHitStoreU8(in, &p) 156 want := uint32(in * in) 157 if got != want { 158 t.Errorf("testLoadHitStore (uint8) failed. want = %d, got = %d", want, got) 159 } 160 } 161 { 162 var in int16 = (1 << 10) + 1 163 var p int16 164 got := loadHitStore16(in, &p) 165 want := int32(in * in) 166 if got != want { 167 t.Errorf("testLoadHitStore (int16) failed. want = %d, got = %d", want, got) 168 } 169 } 170 { 171 var in uint16 = (1 << 10) + 1 172 var p uint16 173 got := loadHitStoreU16(in, &p) 174 want := uint32(in * in) 175 if got != want { 176 t.Errorf("testLoadHitStore (uint16) failed. want = %d, got = %d", want, got) 177 } 178 } 179 { 180 var in int32 = (1 << 30) + 1 181 var p int32 182 got := loadHitStore32(in, &p) 183 want := int64(in * in) 184 if got != want { 185 t.Errorf("testLoadHitStore (int32) failed. want = %d, got = %d", want, got) 186 } 187 } 188 { 189 var in uint32 = (1 << 30) + 1 190 var p uint32 191 got := loadHitStoreU32(in, &p) 192 want := uint64(in * in) 193 if got != want { 194 t.Errorf("testLoadHitStore (uint32) failed. want = %d, got = %d", want, got) 195 } 196 } 197 } 198 199 func TestLoadStore(t *testing.T) { 200 testLoadStoreOrder(t) 201 testStoreSize(t) 202 testExtStore(t) 203 testDeadStorePanic(t) 204 testLoadHitStore(t) 205 }