File: cleanup_test.go

package info (click to toggle)
golang-github-mmcloughlin-avo 0.0~git20200523.4439b6b-6
  • links: PTS, VCS
  • area: main
  • in suites: bullseye
  • size: 8,304 kB
  • sloc: xml: 71,029; asm: 13,138; sh: 179; makefile: 35
file content (86 lines) | stat: -rw-r--r-- 2,237 bytes parent folder | download | duplicates (2)
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
package pass_test

import (
	"reflect"
	"testing"

	"github.com/mmcloughlin/avo/build"
	"github.com/mmcloughlin/avo/ir"
	"github.com/mmcloughlin/avo/operand"
	"github.com/mmcloughlin/avo/pass"
	"github.com/mmcloughlin/avo/reg"
)

func TestPruneSelfMoves(t *testing.T) {
	// Construct a function containing a self-move.
	ctx := build.NewContext()
	ctx.Function("add")
	ctx.MOVQ(operand.U64(1), reg.RAX)
	ctx.MOVQ(operand.U64(2), reg.RCX)
	ctx.MOVQ(reg.RAX, reg.RAX) // self move
	ctx.MOVQ(reg.RCX, reg.R8)
	ctx.ADDQ(reg.R8, reg.RAX)

	// Build the function without the pass and save the nodes.
	fn := BuildFunction(t, ctx)
	pre := append([]ir.Node{}, fn.Nodes...)

	// Apply the pass.
	if err := pass.PruneSelfMoves(fn); err != nil {
		t.Fatal(err)
	}

	// Confirm the self-move was removed and everything else was untouched.
	expect := []ir.Node{}
	for i, n := range pre {
		if i != 2 {
			expect = append(expect, n)
		}
	}

	if !reflect.DeepEqual(fn.Nodes, expect) {
		t.Fatal("unexpected result from self-move pruning")
	}
}

func TestPruneJumpToFollowingLabel(t *testing.T) {
	// Construct a function containing a jump to following.
	ctx := build.NewContext()
	ctx.Function("add")
	ctx.XORQ(reg.RAX, reg.RAX)
	ctx.JMP(operand.LabelRef("next"))
	ctx.Label("next")
	ctx.XORQ(reg.RAX, reg.RAX)

	// Build the function with the PruneJumpToFollowingLabel pass.
	fn := BuildFunction(t, ctx, pass.PruneJumpToFollowingLabel)

	// Confirm no JMP instruction remains.
	for _, i := range fn.Instructions() {
		if i.Opcode == "JMP" {
			t.Fatal("JMP instruction not removed")
		}
	}
}

func TestPruneDanglingLabels(t *testing.T) {
	// Construct a function containing an unreferenced label.
	ctx := build.NewContext()
	ctx.Function("add")
	ctx.XORQ(reg.RAX, reg.RAX)
	ctx.JMP(operand.LabelRef("referenced"))
	ctx.XORQ(reg.RAX, reg.RAX)
	ctx.Label("dangling")
	ctx.XORQ(reg.RAX, reg.RAX)
	ctx.Label("referenced")
	ctx.XORQ(reg.RAX, reg.RAX)

	// Build the function with the PruneDanglingLabels pass.
	fn := BuildFunction(t, ctx, pass.PruneDanglingLabels)

	// Confirm the only label remaining is "referenced".
	expect := []ir.Label{"referenced"}
	if !reflect.DeepEqual(expect, fn.Labels()) {
		t.Fatal("expected dangling label to be removed")
	}
}