%% -*- erlang-indent-level: 2 -*-
%%
%% Licensed under the Apache License, Version 2.0 (the "License");
%% you may not use this file except in compliance with the License.
%% You may obtain a copy of the License at
%%
%% http://www.apache.org/licenses/LICENSE-2.0
%%
%% Unless required by applicable law or agreed to in writing, software
%% distributed under the License is distributed on an "AS IS" BASIS,
%% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
%% See the License for the specific language governing permissions and
%% limitations under the License.
-module(hipe_sparc_cfg).
-export([init/1,
labels/1, start_label/1,
succ/2,
map_bbs/2, fold_bbs/3,
bb/2, bb_add/3]).
-export([postorder/1, reverse_postorder/1]).
-export([linearise/1]).
-export([params/1]).
-export([arity/1]). % for linear scan
-export([redirect_jmp/3, branch_preds/1]).
-define(SPARC_CFG, true). % needed for cfg.inc
-include("../main/hipe.hrl").
-include("hipe_sparc.hrl").
-include("../flow/cfg.hrl").
-include("../flow/cfg.inc").
%%----------------------------------------------------------------------------
%% CFG interface to SPARC
%%----------------------------------------------------------------------------
init(Defun) ->
Code = hipe_sparc:defun_code(Defun),
StartLab = hipe_sparc:label_label(hd(Code)),
Data = hipe_sparc:defun_data(Defun),
IsClosure = hipe_sparc:defun_is_closure(Defun),
Name = hipe_sparc:defun_mfa(Defun),
IsLeaf = hipe_sparc:defun_is_leaf(Defun),
Formals = hipe_sparc:defun_formals(Defun),
CFG = mk_empty_cfg(Name, StartLab, Data, IsClosure, IsLeaf, Formals),
take_bbs(Code, CFG).
is_branch(I) ->
case I of
#bp{'cond'='a'} -> true;
%% not br
#call_tail{} -> true;
#jmp{} -> true;
%% not jmpl
#pseudo_bp{} -> true;
%% #pseudo_br{} -> true;
#pseudo_call{} -> true;
#pseudo_ret{} -> true;
#pseudo_tailcall{} -> true;
_ -> false
end.
branch_successors(Branch) ->
case Branch of
#bp{'cond'='a',label=Label} -> [Label];
#call_tail{} -> [];
#jmp{labels=Labels} -> Labels;
#pseudo_bp{true_label=TrueLab,false_label=FalseLab} -> [FalseLab,TrueLab];
%% #pseudo_br{true_label=TrueLab,false_label=FalseLab} -> [FalseLab,TrueLab];
#pseudo_call{contlab=ContLab, sdesc=#sparc_sdesc{exnlab=ExnLab}} ->
case ExnLab of
[] -> [ContLab];
_ -> [ContLab,ExnLab]
end;
#pseudo_ret{} -> [];
#pseudo_tailcall{} -> []
end.
branch_preds(Branch) ->
case Branch of
#jmp{labels=Labels} ->
Prob = 1.0/length(Labels),
[{L, Prob} || L <- Labels];
#pseudo_bp{true_label=TrueLab,false_label=FalseLab,pred=Pred} ->
[{FalseLab, 1.0-Pred}, {TrueLab, Pred}];
#pseudo_call{contlab=ContLab, sdesc=#sparc_sdesc{exnlab=[]}} ->
%% A function can still cause an exception, even if we won't catch it
[{ContLab, 1.0-hipe_bb_weights:call_exn_pred()}];
#pseudo_call{contlab=ContLab, sdesc=#sparc_sdesc{exnlab=ExnLab}} ->
CallExnPred = hipe_bb_weights:call_exn_pred(),
[{ContLab, 1.0-CallExnPred}, {ExnLab, CallExnPred}];
_ ->
case branch_successors(Branch) of
[] -> [];
[Single] -> [{Single, 1.0}]
end
end.
-ifdef(REMOVE_TRIVIAL_BBS_NEEDED).
fails_to(_Instr) -> [].
-endif.
redirect_jmp(I, Old, New) ->
case I of
#bp{'cond'='a',label=Label} ->
if Old =:= Label -> I#bp{label=New};
true -> I
end;
#pseudo_bp{true_label=TrueLab, false_label=FalseLab} ->
I1 = if Old =:= TrueLab -> I#pseudo_bp{true_label=New};
true -> I
end,
if Old =:= FalseLab -> I1#pseudo_bp{false_label=New};
true -> I1
end;
#pseudo_call{contlab=ContLab0, sdesc=SDesc0} ->
SDesc = case SDesc0 of
#sparc_sdesc{exnlab=Old} -> SDesc0#sparc_sdesc{exnlab=New};
#sparc_sdesc{exnlab=_} -> SDesc0
end,
ContLab = if Old =:= ContLab0 -> New;
true -> ContLab0
end,
I#pseudo_call{sdesc=SDesc, contlab=ContLab}
end.
mk_goto(Label) ->
hipe_sparc:mk_b_label(Label).
is_label(I) ->
hipe_sparc:is_label(I).
label_name(Label) ->
hipe_sparc:label_label(Label).
mk_label(Name) ->
hipe_sparc:mk_label(Name).
linearise(CFG) -> % -> defun, not insn list
MFA = function(CFG),
Formals = params(CFG),
Code = linearize_cfg(CFG),
Data = data(CFG),
VarRange = hipe_gensym:var_range(sparc),
LabelRange = hipe_gensym:label_range(sparc),
IsClosure = is_closure(CFG),
IsLeaf = is_leaf(CFG),
hipe_sparc:mk_defun(MFA, Formals, IsClosure, IsLeaf,
Code, Data, VarRange, LabelRange).
arity(CFG) ->
{_M, _F, A} = function(CFG),
A.