File: delay_slot.m

package info (click to toggle)
mercury 0.9-1
  • links: PTS
  • area: main
  • in suites: potato
  • size: 18,488 kB
  • ctags: 9,800
  • sloc: objc: 146,680; ansic: 51,418; sh: 6,436; lisp: 1,567; cpp: 1,040; perl: 854; makefile: 450; asm: 232; awk: 203; exp: 32; fortran: 3; csh: 1
file content (81 lines) | stat: -rw-r--r-- 2,839 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
%-----------------------------------------------------------------------------%
% Copyright (C) 1997-1998 The University of Melbourne.
% This file may only be copied under the terms of the GNU General
% Public License - see the file COPYING in the Mercury distribution.
%-----------------------------------------------------------------------------%
%
% Main author: zs.
%
% This module attempts to fill the delay slots of branch instructions.
% Since it assumes the existence of one delay slot after every branch,
% this module should not be invoked if the architecture we are targeting
% has no delay slots on its branch instructions.
%
% Since we are generating C code, not assembler, we cannot fill slots directly.
% However, we can let the C compiler know that the instruction immediately
% after a condition branch instruction can be safely executed even if the
% branch is not taken. We can do so by moving the instruction before the
% conditional branch. This actually requires the instruction to be executed
% in both continuations, therefore we want to do this only if there is no
% chance that the C compiler would be able to fill the delay slot any other
% way.
%
% The only code pattern that we recognize as being both safe and advantageous
% to transform the code in order to fill a delay slot is this pattern,
% produced by frameopt:
%
%	L1:
%		if (cond) goto L2
%		incr_sp(N)
%		stackvar(N) = succip
%		...
%
% We transform this code into:
%
%	L1:
%		stackvar(0) = succip
%		if (cond) goto L2
%		incr_sp(N)
%		...
%
% The initial store intro stackvar(0) is into the first word above the
% current detstack top. After the incr_sp is executed (if it ever is), this
% word will become the bottom stack slot of the new frame, i.e. stackvar(N).

%-----------------------------------------------------------------------------%

:- module delay_slot.

:- interface.

:- import_module llds, list.

	% Delay the construction of det stack frames as long as possible,
	% in order to avoid the construction in as many cases as possible.

:- pred fill_branch_delay_slot(list(instruction)::in, list(instruction)::out)
	is det.

%-----------------------------------------------------------------------------%

:- implementation.

:- import_module string, std_util.

fill_branch_delay_slot([], []).
fill_branch_delay_slot([Instr0 | Instrs0], Instrs) :-
	(
		Instr0 = label(_) - _,
		Instrs0 = [Instr1, Instr2, Instr3 | Tail0],
		Instr1 = if_val(_, _) - _,
		Instr2 = incr_sp(Size, _) - _,
		Instr3 = assign(stackvar(Size), lval(succip)) - C2
	->
		fill_branch_delay_slot(Tail0, Tail1),
		string__append(C2, " (early save in delay slot)", NewC2),
		EarlySave = assign(stackvar(0), lval(succip)) - NewC2,
		Instrs = [Instr0, EarlySave, Instr1, Instr2 | Tail1]
	;
		fill_branch_delay_slot(Instrs0, Instrs1),
		Instrs = [Instr0 | Instrs1]
	).