mirror of
https://github.com/Mercury-Language/mercury.git
synced 2025-12-15 22:03:26 +00:00
compiler/*: Add copyright messages. Change all occurences of *.nl in comments to *.m. compiler/mercury_compile.pp: Change the output to the .dep files to use *.m rather than *.nl. (NOTE: this means that `mmake' will not work any more if you call your files *.nl!!!)
369 lines
11 KiB
Mathematica
369 lines
11 KiB
Mathematica
%-----------------------------------------------------------------------------%
|
|
% Copyright (C) 1995 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.
|
|
%-----------------------------------------------------------------------------%
|
|
|
|
% vn_livemap.m - module to build up a map that gives the set of live lvals
|
|
% at each label.
|
|
|
|
% Author: zs.
|
|
|
|
%-----------------------------------------------------------------------------%
|
|
|
|
:- module vn_livemap.
|
|
|
|
:- interface.
|
|
|
|
:- import_module vn_type, llds, list, std_util.
|
|
|
|
% Build up a map of what lvals are live at each label.
|
|
% This step must be iterated in the presence of backward
|
|
% branches, which at the moment are generated by middle
|
|
% recursion and the construction of closures.
|
|
|
|
:- pred vn__repeat_build_livemap(list(instruction), bool, livemap).
|
|
:- mode vn__repeat_build_livemap(in, out, out) is det.
|
|
|
|
:- implementation.
|
|
|
|
:- import_module vn_util, opt_util.
|
|
:- import_module require, map, bintree_set.
|
|
|
|
%-----------------------------------------------------------------------------%
|
|
%-----------------------------------------------------------------------------%
|
|
|
|
vn__repeat_build_livemap(Backinstrs, Ccode, Livemap) :-
|
|
map__init(Livemap0),
|
|
vn__repeat_build_livemap_2(Backinstrs, Livemap0, Ccode, Livemap).
|
|
|
|
:- pred vn__repeat_build_livemap_2(list(instruction), livemap, bool, livemap).
|
|
:- mode vn__repeat_build_livemap_2(in, di, out, uo) is det.
|
|
|
|
vn__repeat_build_livemap_2(Backinstrs, Livemap0, Ccode, Livemap) :-
|
|
bintree_set__init(Livevals0),
|
|
vn__build_livemap(Backinstrs, Livevals0, no, Ccode1,
|
|
Livemap0, Livemap1),
|
|
( Ccode1 = yes ->
|
|
Ccode = yes,
|
|
Livemap = Livemap1
|
|
; vn__equal_livemaps(Livemap0, Livemap1) ->
|
|
Ccode = no,
|
|
Livemap = Livemap1
|
|
;
|
|
vn__repeat_build_livemap_2(Backinstrs, Livemap1, Ccode, Livemap)
|
|
).
|
|
|
|
:- pred vn__equal_livemaps(livemap, livemap).
|
|
:- mode vn__equal_livemaps(in, in) is semidet.
|
|
|
|
vn__equal_livemaps(Livemap1, Livemap2) :-
|
|
map__keys(Livemap1, Labels),
|
|
vn__equal_livemaps_keys(Labels, Livemap1, Livemap2).
|
|
|
|
:- pred vn__equal_livemaps_keys(list(label), livemap, livemap).
|
|
:- mode vn__equal_livemaps_keys(in, in, in) is semidet.
|
|
|
|
vn__equal_livemaps_keys([], _Livemap1, _Livemap2).
|
|
vn__equal_livemaps_keys([Label | Labels], Livemap1, Livemap2) :-
|
|
map__lookup(Livemap1, Label, Liveset1),
|
|
map__lookup(Livemap2, Label, Liveset2),
|
|
bintree_set__equal(Liveset1, Liveset2),
|
|
vn__equal_livemaps_keys(Labels, Livemap1, Livemap2).
|
|
|
|
%-----------------------------------------------------------------------------%
|
|
%-----------------------------------------------------------------------------%
|
|
|
|
% Build up a map of what lvals are live at each label.
|
|
% The input instruction sequence is reversed.
|
|
|
|
:- pred vn__build_livemap(list(instruction), lvalset, bool, bool,
|
|
livemap, livemap).
|
|
:- mode vn__build_livemap(in, in, in, out, di, uo) is det.
|
|
|
|
vn__build_livemap([], _, Ccode, Ccode, Livemap, Livemap).
|
|
vn__build_livemap([Instr|Moreinstrs], Livevals0, Ccode0, Ccode,
|
|
Livemap0, Livemap) :-
|
|
Instr = Uinstr - _Comment,
|
|
(
|
|
Uinstr = comment(_),
|
|
Livemap1 = Livemap0,
|
|
Livevals2 = Livevals0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
Uinstr = livevals(_),
|
|
error("livevals found in backward scan in vn__build_livemap")
|
|
;
|
|
Uinstr = block(_, _),
|
|
error("block found in backward scan in vn__build_livemap")
|
|
;
|
|
Uinstr = assign(Lval, Rval),
|
|
|
|
% Make dead the variable assigned, but make any variables
|
|
% needed to access it live. Make the variables in the assigned
|
|
% expression live as well.
|
|
% The deletion has to be done first. If the assigned-to lval
|
|
% appears on the right hand side as well as the left, then we
|
|
% want make_live to put it back into the liveval set.
|
|
|
|
bintree_set__delete(Livevals0, Lval, Livevals1),
|
|
vn__lval_access_rval(Lval, Rvals),
|
|
vn__make_live([Rval | Rvals], Livevals1, Livevals2),
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
Uinstr = call(_, _, _, _),
|
|
opt_util__skip_comments(Moreinstrs, Moreinstrs1),
|
|
(
|
|
Moreinstrs1 = [Nextinstr | Evenmoreinstrs],
|
|
Nextinstr = Nextuinstr - Nextcomment,
|
|
Nextuinstr = livevals(Livevals1)
|
|
->
|
|
vn__filter_livevals(Livevals1, Livevals2),
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Evenmoreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
error("call not preceded by livevals")
|
|
)
|
|
;
|
|
Uinstr = call_closure(_, _, _),
|
|
opt_util__skip_comments(Moreinstrs, Moreinstrs1),
|
|
(
|
|
Moreinstrs1 = [Nextinstr | Evenmoreinstrs],
|
|
Nextinstr = Nextuinstr - Nextcomment,
|
|
Nextuinstr = livevals(Livevals1)
|
|
->
|
|
vn__filter_livevals(Livevals1, Livevals2),
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Evenmoreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
error("call_closure not preceded by livevals")
|
|
)
|
|
;
|
|
Uinstr = mkframe(_, _, _),
|
|
Livemap1 = Livemap0,
|
|
Livevals2 = Livevals0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
Uinstr = modframe(_),
|
|
Livemap1 = Livemap0,
|
|
Livevals2 = Livevals0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
Uinstr = label(Label),
|
|
map__det_insert(Livemap0, Label, Livevals0, Livemap1),
|
|
Livevals2 = Livevals0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
Uinstr = goto(CodeAddr, _),
|
|
opt_util__skip_comments(Moreinstrs, Moreinstrs1),
|
|
opt_util__livevals_addr(CodeAddr, LivevalsNeeded),
|
|
( LivevalsNeeded = yes ->
|
|
(
|
|
Moreinstrs1 = [Nextinstr | Evenmoreinstrs],
|
|
Nextinstr = Nextuinstr - Nextcomment,
|
|
Nextuinstr = livevals(Livevals1)
|
|
->
|
|
vn__filter_livevals(Livevals1, Livevals2),
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Evenmoreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
error("tailcall not preceded by livevals")
|
|
)
|
|
; CodeAddr = label(Label) ->
|
|
bintree_set__init(Livevals1),
|
|
vn__insert_label_livevals([Label], Livemap0,
|
|
Livevals1, Livevals2),
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
error("unknown label type in vn__build_livemap")
|
|
% Livevals2 = Livevals0,
|
|
% Livemap1 = Livemap0,
|
|
% Moreinstrs2 = Moreinstrs,
|
|
% Ccode1 = Ccode0
|
|
)
|
|
;
|
|
Uinstr = computed_goto(_, Labels),
|
|
bintree_set__init(Livevals1),
|
|
vn__insert_label_livevals(Labels, Livemap0,
|
|
Livevals1, Livevals2),
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
Uinstr = c_code(_),
|
|
Livemap1 = Livemap0,
|
|
Livevals2 = Livevals0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = yes
|
|
;
|
|
Uinstr = if_val(Rval, CodeAddr),
|
|
opt_util__skip_comments(Moreinstrs, Moreinstrs1),
|
|
(
|
|
Moreinstrs1 = [Nextinstr | Evenmoreinstrs],
|
|
Nextinstr = Nextuinstr - Nextcomment,
|
|
Nextuinstr = livevals(Livevals1)
|
|
->
|
|
% This if_val was put here by middle_rec.
|
|
vn__filter_livevals(Livevals1, Livevals2),
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Evenmoreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
vn__make_live([Rval], Livevals0, Livevals1),
|
|
( CodeAddr = label(Label) ->
|
|
vn__insert_label_livevals([Label], Livemap0,
|
|
Livevals1, Livevals2)
|
|
;
|
|
Livevals2 = Livevals1
|
|
),
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
)
|
|
;
|
|
Uinstr = incr_hp(Lval, _, Rval),
|
|
|
|
% Make dead the variable assigned, but make any variables
|
|
% needed to access it live. Make the variables in the size
|
|
% expression live as well.
|
|
% The use of the size expression occurs after the assignment
|
|
% to lval, but the two should never have any variables in
|
|
% common. This is why doing the deletion first works.
|
|
|
|
bintree_set__delete(Livevals0, Lval, Livevals1),
|
|
vn__lval_access_rval(Lval, Rvals),
|
|
vn__make_live([Rval | Rvals], Livevals1, Livevals2),
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
Uinstr = mark_hp(Lval),
|
|
vn__lval_access_rval(Lval, Rvals),
|
|
vn__make_live(Rvals, Livevals0, Livevals2),
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
Uinstr = restore_hp(Rval),
|
|
vn__make_live([Rval], Livevals0, Livevals2),
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
Uinstr = incr_sp(_),
|
|
Livevals2 = Livevals0,
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
;
|
|
Uinstr = decr_sp(_),
|
|
Livevals2 = Livevals0,
|
|
Livemap1 = Livemap0,
|
|
Moreinstrs2 = Moreinstrs,
|
|
Ccode1 = Ccode0
|
|
),
|
|
vn__build_livemap(Moreinstrs2, Livevals2, Ccode1, Ccode,
|
|
Livemap1, Livemap).
|
|
|
|
%-----------------------------------------------------------------------------%
|
|
%-----------------------------------------------------------------------------%
|
|
|
|
% Set all lvals found in this rval to live, with the exception of
|
|
% fields, since they are treated specially (the later stages consider
|
|
% them to be live even if they are not explicitly in the live set).
|
|
|
|
:- pred vn__make_live(list(rval), lvalset, lvalset).
|
|
:- mode vn__make_live(in, di, uo) is det.
|
|
|
|
vn__make_live([], Livevals, Livevals).
|
|
vn__make_live([Rval | Rvals], Livevals0, Livevals) :-
|
|
(
|
|
Rval = lval(Lval),
|
|
( Lval = field(_, Rval1, Rval2) ->
|
|
vn__make_live([Rval1, Rval2], Livevals0, Livevals1)
|
|
;
|
|
bintree_set__insert(Livevals0, Lval, Livevals1)
|
|
)
|
|
;
|
|
Rval = create(_, _, _),
|
|
Livevals1 = Livevals0
|
|
;
|
|
Rval = mkword(_, Rval1),
|
|
vn__make_live([Rval1], Livevals0, Livevals1)
|
|
;
|
|
Rval = const(_),
|
|
Livevals1 = Livevals0
|
|
;
|
|
Rval = unop(_, Rval1),
|
|
vn__make_live([Rval1], Livevals0, Livevals1)
|
|
;
|
|
Rval = binop(_, Rval1, Rval2),
|
|
vn__make_live([Rval1, Rval2], Livevals0, Livevals1)
|
|
;
|
|
Rval = var(_),
|
|
error("var rval should not propagate to value_number")
|
|
),
|
|
vn__make_live(Rvals, Livevals1, Livevals).
|
|
|
|
%-----------------------------------------------------------------------------%
|
|
%-----------------------------------------------------------------------------%
|
|
|
|
:- pred vn__filter_livevals(lvalset, lvalset).
|
|
:- mode vn__filter_livevals(in, out) is det.
|
|
|
|
vn__filter_livevals(Livevals0, Livevals) :-
|
|
bintree_set__to_sorted_list(Livevals0, Livelist),
|
|
bintree_set__init(Livevals1),
|
|
vn__insert_proper_livevals(Livelist, Livevals1, Livevals).
|
|
|
|
:- pred vn__insert_label_livevals(list(label), livemap, lvalset, lvalset).
|
|
:- mode vn__insert_label_livevals(in, in, di, uo) is det.
|
|
|
|
vn__insert_label_livevals([], _, Livevals, Livevals).
|
|
vn__insert_label_livevals([Label | Labels], Livemap, Livevals0, Livevals) :-
|
|
( map__search(Livemap, Label, LabelLivevals) ->
|
|
bintree_set__to_sorted_list(LabelLivevals, Livelist),
|
|
vn__insert_proper_livevals(Livelist, Livevals0, Livevals1)
|
|
;
|
|
Livevals1 = Livevals0
|
|
),
|
|
vn__insert_label_livevals(Labels, Livemap, Livevals1, Livevals).
|
|
|
|
:- pred vn__insert_proper_livevals(list(lval), lvalset, lvalset).
|
|
:- mode vn__insert_proper_livevals(in, di, uo) is det.
|
|
|
|
vn__insert_proper_livevals([], Livevals, Livevals).
|
|
vn__insert_proper_livevals([Live | Livelist], Livevals0, Livevals) :-
|
|
vn__insert_proper_liveval(Live, Livevals0, Livevals1),
|
|
vn__insert_proper_livevals(Livelist, Livevals1, Livevals).
|
|
|
|
% Make sure that we insert general register and stack references only.
|
|
|
|
:- pred vn__insert_proper_liveval(lval, lvalset, lvalset).
|
|
:- mode vn__insert_proper_liveval(in, di, uo) is det.
|
|
|
|
vn__insert_proper_liveval(Live, Livevals0, Livevals) :-
|
|
( Live = reg(_) ->
|
|
bintree_set__insert(Livevals0, Live, Livevals)
|
|
; Live = stackvar(_) ->
|
|
bintree_set__insert(Livevals0, Live, Livevals)
|
|
; Live = framevar(_) ->
|
|
bintree_set__insert(Livevals0, Live, Livevals)
|
|
;
|
|
Livevals = Livevals0
|
|
).
|
|
|
|
%-----------------------------------------------------------------------------%
|
|
%-----------------------------------------------------------------------------%
|