Files
mercury/compiler/follow_code.m
Zoltan Somogyi fc36858e4f Fix a bug that prevented some of the Mercury programs used for the packrat
Estimated hours taken: 0.2
Branches: main

Fix a bug that prevented some of the Mercury programs used for the packrat
paper from compiling. This is the bug description from the new test case:

% This is a regression test. In versions of the compiler before 7 Aug 2007,
% it used to cause this compiler abort:
%
% Software Error: code_gen.m: Unexpected: nondet model in det/semidet context
%
% when generating code for the java_identifier predicate.
%
% The bug was that the follow_code pass would push a goal binding a variable
% into a semidet disjunction. This would cause the disjunction to have an
% output, which would cause the simplify pass to change its determinism,
% and from there the determinism of the entire procedure body, to nondet.
% Since java_identifier is supposed to be semidet, this causes the abort above.

compiler/follow_code.m:
	Fix the bug.

tests/valid/mercury_java_parser_follow_code_bug.m:
	Add the new test case.

tests/valid/Mmakefile:
	Enable the new test case.
2007-08-07 10:03:51 +00:00

389 lines
15 KiB
Mathematica

%-----------------------------------------------------------------------------%
% vim: ft=mercury ts=4 sw=4 et
%-----------------------------------------------------------------------------%
% Copyright (C) 1994-2007 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.
%-----------------------------------------------------------------------------%
%
% File: follow_code.m.
% Main author: conway.
% Extensive modifications by zs.
%
% The problem attacked by this module is that sometimes the code generator
% doesn't know where it should put the values of live variables at the end
% of a branched control structure. All branches must put each live variable
% into the same lval, so having each branch leave each live variable where it
% just happens to be is not an option. We currently just put all live variables
% into its own rN register or stack slot, but often is not where the variable
% happens to be at the end of any branch, nor is it where the variable is next
% needed.
%
% The idea used by this module to attack this problem is to try to ensure
% that the branched control structure is followed immediately either by a call
% or by the end of the procedure body, because both have clear rules about
% where every live variable must be. If a branched control structure is
% followed by builtin goals such as unifications, we push those goals into
% each branch.
%
%-----------------------------------------------------------------------------%
:- module ll_backend.follow_code.
:- interface.
:- import_module hlds.hlds_module.
:- import_module hlds.hlds_pred.
%-----------------------------------------------------------------------------%
:- pred move_follow_code_in_proc(pred_id::in, proc_id::in, pred_info::in,
proc_info::in, proc_info::out, module_info::in, module_info::out) is det.
%-----------------------------------------------------------------------------%
%-----------------------------------------------------------------------------%
:- implementation.
:- import_module check_hlds.mode_util.
:- import_module hlds.code_model.
:- import_module hlds.goal_util.
:- import_module hlds.hlds_goal.
:- import_module hlds.instmap.
:- import_module hlds.quantification.
:- import_module libs.compiler_util.
:- import_module libs.globals.
:- import_module libs.options.
:- import_module parse_tree.prog_data.
:- import_module bool.
:- import_module list.
:- import_module pair.
:- import_module set.
%-----------------------------------------------------------------------------%
move_follow_code_in_proc(_PredId, _ProcId, _PredInfo, !ProcInfo,
!ModuleInfo) :-
module_info_get_globals(!.ModuleInfo, Globals),
globals.lookup_bool_option(Globals, follow_code, FollowCode),
(
FollowCode = no
;
FollowCode = yes,
proc_info_get_goal(!.ProcInfo, Goal0),
proc_info_get_varset(!.ProcInfo, Varset0),
proc_info_get_vartypes(!.ProcInfo, VarTypes0),
proc_info_get_rtti_varmaps(!.ProcInfo, RttiVarMaps0),
(
move_follow_code_in_goal(Goal0, Goal1, no, Changed),
Changed = yes
->
% We need to fix up the goal_info by recalculating the nonlocal
% vars and the non-atomic instmap deltas.
proc_info_get_headvars(!.ProcInfo, HeadVars),
implicitly_quantify_clause_body(HeadVars, _Warnings, Goal1, Goal2,
Varset0, Varset, VarTypes0, VarTypes,
RttiVarMaps0, RttiVarMaps),
proc_info_get_initial_instmap(!.ProcInfo, !.ModuleInfo, InstMap0),
proc_info_get_inst_varset(!.ProcInfo, InstVarSet),
recompute_instmap_delta(no, Goal2, Goal, VarTypes, InstVarSet,
InstMap0, !ModuleInfo)
;
Goal = Goal0,
Varset = Varset0,
VarTypes = VarTypes0,
RttiVarMaps = RttiVarMaps0
),
proc_info_set_goal(Goal, !ProcInfo),
proc_info_set_varset(Varset, !ProcInfo),
proc_info_set_vartypes(VarTypes, !ProcInfo),
proc_info_set_rtti_varmaps(RttiVarMaps, !ProcInfo)
).
%-----------------------------------------------------------------------------%
%-----------------------------------------------------------------------------%
:- pred move_follow_code_in_goal(hlds_goal::in, hlds_goal::out,
bool::in, bool::out) is det.
move_follow_code_in_goal(Goal0, Goal, !Changed) :-
Goal0 = hlds_goal(GoalExpr0, GoalInfo),
(
GoalExpr0 = conj(ConjType, Goals0),
(
ConjType = plain_conj,
ConjPurity = goal_info_get_purity(GoalInfo),
move_follow_code_in_conj(Goals0, ConjPurity, Goals, !Changed)
;
ConjType = parallel_conj,
move_follow_code_in_independent_goals(Goals0, Goals, !Changed)
),
GoalExpr = conj(ConjType, Goals),
Goal = hlds_goal(GoalExpr, GoalInfo)
;
GoalExpr0 = disj(Goals0),
move_follow_code_in_independent_goals(Goals0, Goals, !Changed),
GoalExpr = disj(Goals),
Goal = hlds_goal(GoalExpr, GoalInfo)
;
GoalExpr0 = negation(SubGoal0),
move_follow_code_in_goal(SubGoal0, SubGoal, !Changed),
GoalExpr = negation(SubGoal),
Goal = hlds_goal(GoalExpr, GoalInfo)
;
GoalExpr0 = switch(Var, Det, Cases0),
move_follow_code_in_cases(Cases0, Cases, !Changed),
GoalExpr = switch(Var, Det, Cases),
Goal = hlds_goal(GoalExpr, GoalInfo)
;
GoalExpr0 = if_then_else(Vars, Cond0, Then0, Else0),
move_follow_code_in_goal(Cond0, Cond, !Changed),
move_follow_code_in_goal(Then0, Then, !Changed),
move_follow_code_in_goal(Else0, Else, !Changed),
GoalExpr = if_then_else(Vars, Cond, Then, Else),
Goal = hlds_goal(GoalExpr, GoalInfo)
;
GoalExpr0 = scope(Remove, SubGoal0),
move_follow_code_in_goal(SubGoal0, SubGoal, !Changed),
GoalExpr = scope(Remove, SubGoal),
Goal = hlds_goal(GoalExpr, GoalInfo)
;
( GoalExpr0 = generic_call(_, _, _, _)
; GoalExpr0 = plain_call(_, _, _, _, _, _)
; GoalExpr0 = unify(_, _, _, _, _)
; GoalExpr0 = call_foreign_proc(_, _, _, _, _, _, _)
),
Goal = Goal0
;
GoalExpr0 = shorthand(_),
% These should have been expanded out by now.
unexpected(this_file,
"move_follow_code_in_goal: unexpected shorthand")
).
%-----------------------------------------------------------------------------%
% move_follow_code_in_independent_goals is used both for disjunction and
% parallel conjunction.
%
:- pred move_follow_code_in_independent_goals(list(hlds_goal)::in,
list(hlds_goal)::out, bool::in, bool::out) is det.
move_follow_code_in_independent_goals([], [], !Changed).
move_follow_code_in_independent_goals([Goal0 | Goals0], [Goal | Goals],
!Changed) :-
move_follow_code_in_goal(Goal0, Goal, !Changed),
move_follow_code_in_independent_goals(Goals0, Goals, !Changed).
%-----------------------------------------------------------------------------%
:- pred move_follow_code_in_cases(list(case)::in, list(case)::out,
bool::in, bool::out) is det.
move_follow_code_in_cases([], [], !Changed).
move_follow_code_in_cases([Case0 | Cases0], [Case | Cases], !Changed) :-
Case0 = case(ConsId, Goal0),
move_follow_code_in_goal(Goal0, Goal, !Changed),
Case = case(ConsId, Goal),
move_follow_code_in_cases(Cases0, Cases, !Changed).
%-----------------------------------------------------------------------------%
% Find the first branched structure, and split the conj into those goals
% before and after it.
%
:- pred move_follow_code_in_conj(list(hlds_goal)::in, purity::in,
list(hlds_goal)::out, bool::in, bool::out) is det.
move_follow_code_in_conj(Goals0, ConjPurity, Goals, !Changed) :-
move_follow_code_in_conj_2(Goals0, ConjPurity, [], RevGoals, !Changed),
list.reverse(RevGoals, Goals).
:- pred move_follow_code_in_conj_2(list(hlds_goal)::in, purity::in,
list(hlds_goal)::in, list(hlds_goal)::out, bool::in, bool::out) is det.
move_follow_code_in_conj_2([], _ConjPurity, !RevPrevGoals, !Changed).
move_follow_code_in_conj_2([Goal0 | Goals0], ConjPurity, !RevPrevGoals,
!Changed) :-
(
Goal0 = hlds_goal(GoalExpr0, GoalInfo0),
goal_util.goal_is_branched(GoalExpr0),
move_follow_code_select(Goals0, FollowGoals, RestGoalsPrime,
ConjPurity, WorstPurity),
FollowGoals = [_ | _],
% Moving any goals that bind variables into a model_semi (or model_det)
% disjunction gives that disjunction some outputs, which means that it
% will become nondet.
(
(
GoalExpr0 = disj(_),
goal_info_get_code_model(GoalInfo0) \= model_non
)
=>
no_bind_vars(FollowGoals)
),
move_follow_code_move_goals(Goal0, FollowGoals, WorstPurity,
Goal1Prime)
->
!:Changed = yes,
Goal1 = Goal1Prime,
RestGoals = RestGoalsPrime
;
Goal1 = Goal0,
RestGoals = Goals0
),
move_follow_code_in_goal(Goal1, Goal, !Changed),
!:RevPrevGoals = [Goal | !.RevPrevGoals],
move_follow_code_in_conj_2(RestGoals, ConjPurity, !RevPrevGoals, !Changed).
:- pred no_bind_vars(list(hlds_goal)::in) is semidet.
no_bind_vars([]).
no_bind_vars([Goal | Goals]) :-
Goal = hlds_goal(_, GoalInfo),
InstMapDelta = goal_info_get_instmap_delta(GoalInfo),
instmap_delta_changed_vars(InstMapDelta, ChangedVars),
set.empty(ChangedVars),
no_bind_vars(Goals).
%-----------------------------------------------------------------------------%
% Split a list of goals into the prefix of builtins and the rest.
%
:- pred move_follow_code_select(list(hlds_goal)::in, list(hlds_goal)::out,
list(hlds_goal)::out, purity::in, purity::out) is det.
move_follow_code_select([], [], [], !Purity).
move_follow_code_select([Goal | Goals], FollowGoals, RestGoals, !Purity) :-
Goal = hlds_goal(GoalExpr, GoalInfo),
( move_follow_code_is_builtin(GoalExpr) ->
GoalPurity = goal_info_get_purity(GoalInfo),
!:Purity = worst_purity(!.Purity, GoalPurity),
move_follow_code_select(Goals, FollowGoals0, RestGoals, !Purity),
FollowGoals = [Goal | FollowGoals0]
;
FollowGoals = [],
RestGoals = [Goal | Goals]
).
%-----------------------------------------------------------------------------%
:- pred move_follow_code_move_goals(hlds_goal::in, list(hlds_goal)::in,
purity::in, hlds_goal::out) is semidet.
move_follow_code_move_goals(Goal0, FollowGoals, FollowPurity, Goal) :-
Goal0 = hlds_goal(GoalExpr0, GoalInfo0),
(
GoalExpr0 = switch(Var, Det, Cases0),
move_follow_code_move_goals_cases(Cases0, FollowGoals, FollowPurity,
Cases),
GoalExpr = switch(Var, Det, Cases)
;
GoalExpr0 = disj(Goals0),
move_follow_code_move_goals_disj(Goals0, FollowGoals, FollowPurity,
Goals),
GoalExpr = disj(Goals)
;
GoalExpr0 = if_then_else(Vars, Cond, Then0, Else0),
follow_code_conjoin_goal_and_goal_list(Then0, FollowGoals,
FollowPurity, Then),
follow_code_conjoin_goal_and_goal_list(Else0, FollowGoals,
FollowPurity, Else),
GoalExpr = if_then_else(Vars, Cond, Then, Else)
),
OldPurity = goal_info_get_purity(GoalInfo0),
NewPurity = worst_purity(OldPurity, FollowPurity),
goal_info_set_purity(NewPurity, GoalInfo0, GoalInfo),
Goal = hlds_goal(GoalExpr, GoalInfo).
%-----------------------------------------------------------------------------%
:- pred move_follow_code_move_goals_cases(list(case)::in, list(hlds_goal)::in,
purity::in, list(case)::out) is semidet.
move_follow_code_move_goals_cases([], _FollowGoals, _FollowPurity, []).
move_follow_code_move_goals_cases([Case0 | Cases0], FollowGoals, FollowPurity,
[Case | Cases]) :-
Case0 = case(Cons, Goal0),
follow_code_conjoin_goal_and_goal_list(Goal0, FollowGoals, FollowPurity,
Goal),
Case = case(Cons, Goal),
move_follow_code_move_goals_cases(Cases0, FollowGoals, FollowPurity,
Cases).
%-----------------------------------------------------------------------------%
:- pred move_follow_code_move_goals_disj(list(hlds_goal)::in,
list(hlds_goal)::in, purity::in, list(hlds_goal)::out) is semidet.
move_follow_code_move_goals_disj([], _FollowGoals, _FollowPurity, []).
move_follow_code_move_goals_disj([Goal0 | Goals0], FollowGoals, FollowPurity,
[Goal | Goals]) :-
follow_code_conjoin_goal_and_goal_list(Goal0, FollowGoals, FollowPurity,
Goal),
move_follow_code_move_goals_disj(Goals0, FollowGoals, FollowPurity, Goals).
%-----------------------------------------------------------------------------%
% Takes a goal and a list of goals, and conjoins them (with a potentially
% blank goal_info), checking that the determinism of the goal is not
% changed.
%
:- pred follow_code_conjoin_goal_and_goal_list(hlds_goal::in,
list(hlds_goal)::in, purity::in, hlds_goal::out) is semidet.
follow_code_conjoin_goal_and_goal_list(Goal0, FollowGoals, FollowPurity,
Goal) :-
Goal0 = hlds_goal(GoalExpr0, GoalInfo0),
Detism0 = goal_info_get_determinism(GoalInfo0),
determinism_components(Detism0, _CanFail0, MaxSolns0),
( MaxSolns0 = at_most_zero ->
Goal = Goal0
;
check_follow_code_detism(FollowGoals, Detism0),
( GoalExpr0 = conj(plain_conj, GoalList0) ->
list.append(GoalList0, FollowGoals, GoalList),
GoalExpr = conj(plain_conj, GoalList)
;
GoalExpr = conj(plain_conj, [Goal0 | FollowGoals])
),
OldPurity = goal_info_get_purity(GoalInfo0),
NewPurity = worst_purity(OldPurity, FollowPurity),
goal_info_set_purity(NewPurity, GoalInfo0, GoalInfo),
Goal = hlds_goal(GoalExpr, GoalInfo)
).
% This check is necessary to make sure that follow_code doesn't change
% the determinism of the goal.
%
:- pred check_follow_code_detism(list(hlds_goal)::in, determinism::in)
is semidet.
check_follow_code_detism([], _).
check_follow_code_detism([hlds_goal(_, GoalInfo) | Goals], Detism0) :-
Detism1 = goal_info_get_determinism(GoalInfo),
det_conjunction_detism(Detism0, Detism1, Detism0),
check_follow_code_detism(Goals, Detism0).
%-----------------------------------------------------------------------------%
:- pred move_follow_code_is_builtin(hlds_goal_expr::in) is semidet.
move_follow_code_is_builtin(GoalExpr) :-
(
GoalExpr = unify(_, _, _, Unification, _),
Unification \= complicated_unify(_, _, _)
;
GoalExpr = plain_call(_, _, _, inline_builtin, _, _)
).
%-----------------------------------------------------------------------------%
:- func this_file = string.
this_file = "follow_code.m".
%-----------------------------------------------------------------------------%
:- end_module follow_code.
%-----------------------------------------------------------------------------%