blob: 9ded8bd6e663fe4bfdbbb74b37a94203e0534c21 [file] [log] [blame]
Keith Randall8d323602015-05-26 14:43:25 -07001// 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
5package ssa
6
Todd Nealdee1f272015-08-10 21:05:35 -05007import "testing"
Keith Randall8d323602015-05-26 14:43:25 -07008
9func TestDeadStore(t *testing.T) {
Josh Bleecher Snyder85e03292015-07-30 11:03:05 -070010 c := testConfig(t)
Todd Nealce977882015-08-11 19:31:53 -050011 elemType := &TypeImpl{Size_: 8, Name: "testtype"}
12 ptrType := &TypeImpl{Size_: 8, Ptr: true, Name: "testptr", Elem_: elemType} // dummy for testing
Keith Randall8d323602015-05-26 14:43:25 -070013 fun := Fun(c, "entry",
14 Bloc("entry",
Keith Randall16b1fce2016-01-31 11:39:39 -080015 Valu("start", OpInitMem, TypeMem, 0, nil),
Todd Neal929c2aa2015-06-25 18:03:50 -050016 Valu("sb", OpSB, TypeInvalid, 0, nil),
Todd Neal991036a2015-09-03 18:24:22 -050017 Valu("v", OpConstBool, TypeBool, 1, nil),
Todd Neal929c2aa2015-06-25 18:03:50 -050018 Valu("addr1", OpAddr, ptrType, 0, nil, "sb"),
19 Valu("addr2", OpAddr, ptrType, 0, nil, "sb"),
Todd Nealdee1f272015-08-10 21:05:35 -050020 Valu("addr3", OpAddr, ptrType, 0, nil, "sb"),
21 Valu("zero1", OpZero, TypeMem, 8, nil, "addr3", "start"),
Keith Randalld4cc51d2015-08-14 21:47:20 -070022 Valu("store1", OpStore, TypeMem, 1, nil, "addr1", "v", "zero1"),
23 Valu("store2", OpStore, TypeMem, 1, nil, "addr2", "v", "store1"),
24 Valu("store3", OpStore, TypeMem, 1, nil, "addr1", "v", "store2"),
25 Valu("store4", OpStore, TypeMem, 1, nil, "addr3", "v", "store3"),
Keith Randall8d323602015-05-26 14:43:25 -070026 Goto("exit")),
27 Bloc("exit",
28 Exit("store3")))
29
30 CheckFunc(fun.f)
31 dse(fun.f)
32 CheckFunc(fun.f)
33
Todd Nealdee1f272015-08-10 21:05:35 -050034 v1 := fun.values["store1"]
35 if v1.Op != OpCopy {
Keith Randall8d323602015-05-26 14:43:25 -070036 t.Errorf("dead store not removed")
37 }
Todd Nealdee1f272015-08-10 21:05:35 -050038
39 v2 := fun.values["zero1"]
40 if v2.Op != OpCopy {
41 t.Errorf("dead store (zero) not removed")
42 }
Keith Randall8d323602015-05-26 14:43:25 -070043}
44func TestDeadStorePhi(t *testing.T) {
45 // make sure we don't get into an infinite loop with phi values.
Josh Bleecher Snyder85e03292015-07-30 11:03:05 -070046 c := testConfig(t)
Keith Randall8d323602015-05-26 14:43:25 -070047 ptrType := &TypeImpl{Size_: 8, Ptr: true, Name: "testptr"} // dummy for testing
48 fun := Fun(c, "entry",
49 Bloc("entry",
Keith Randall16b1fce2016-01-31 11:39:39 -080050 Valu("start", OpInitMem, TypeMem, 0, nil),
Todd Neal929c2aa2015-06-25 18:03:50 -050051 Valu("sb", OpSB, TypeInvalid, 0, nil),
Todd Neal991036a2015-09-03 18:24:22 -050052 Valu("v", OpConstBool, TypeBool, 1, nil),
Todd Neal929c2aa2015-06-25 18:03:50 -050053 Valu("addr", OpAddr, ptrType, 0, nil, "sb"),
Keith Randall8d323602015-05-26 14:43:25 -070054 Goto("loop")),
55 Bloc("loop",
Keith Randall8f22b522015-06-11 21:29:25 -070056 Valu("phi", OpPhi, TypeMem, 0, nil, "start", "store"),
Keith Randalld4cc51d2015-08-14 21:47:20 -070057 Valu("store", OpStore, TypeMem, 1, nil, "addr", "v", "phi"),
Keith Randall8d323602015-05-26 14:43:25 -070058 If("v", "loop", "exit")),
59 Bloc("exit",
60 Exit("store")))
61
62 CheckFunc(fun.f)
63 dse(fun.f)
64 CheckFunc(fun.f)
65}
66
67func TestDeadStoreTypes(t *testing.T) {
68 // Make sure a narrow store can't shadow a wider one. We test an even
69 // stronger restriction, that one store can't shadow another unless the
70 // types of the address fields are identical (where identicalness is
71 // decided by the CSE pass).
Josh Bleecher Snyder85e03292015-07-30 11:03:05 -070072 c := testConfig(t)
Keith Randall8d323602015-05-26 14:43:25 -070073 t1 := &TypeImpl{Size_: 8, Ptr: true, Name: "t1"}
74 t2 := &TypeImpl{Size_: 4, Ptr: true, Name: "t2"}
75 fun := Fun(c, "entry",
76 Bloc("entry",
Keith Randall16b1fce2016-01-31 11:39:39 -080077 Valu("start", OpInitMem, TypeMem, 0, nil),
Todd Neal929c2aa2015-06-25 18:03:50 -050078 Valu("sb", OpSB, TypeInvalid, 0, nil),
Todd Neal991036a2015-09-03 18:24:22 -050079 Valu("v", OpConstBool, TypeBool, 1, nil),
Todd Neal929c2aa2015-06-25 18:03:50 -050080 Valu("addr1", OpAddr, t1, 0, nil, "sb"),
81 Valu("addr2", OpAddr, t2, 0, nil, "sb"),
Keith Randalld4cc51d2015-08-14 21:47:20 -070082 Valu("store1", OpStore, TypeMem, 1, nil, "addr1", "v", "start"),
83 Valu("store2", OpStore, TypeMem, 1, nil, "addr2", "v", "store1"),
Keith Randall8d323602015-05-26 14:43:25 -070084 Goto("exit")),
85 Bloc("exit",
86 Exit("store2")))
87
88 CheckFunc(fun.f)
89 cse(fun.f)
90 dse(fun.f)
91 CheckFunc(fun.f)
92
93 v := fun.values["store1"]
94 if v.Op == OpCopy {
95 t.Errorf("store %s incorrectly removed", v)
96 }
97}