Files
mercury/compiler/switch_gen.m
Zoltan Somogyi 3dd0f2e03b Act on all remaining warnings about unused state vars.
compiler/add_heap_ops.m:
compiler/check_import_accessibility.m:
compiler/comp_unit_interface.m:
compiler/convert_import_use.m:
compiler/deforest.m:
compiler/dep_par_conj.m:
compiler/distance_granularity.m:
compiler/equiv_type.m:
compiler/generate_dep_d_files.m:
compiler/generate_mmakefile_fragments.m:
compiler/get_dependencies.m:
compiler/grab_modules.m:
compiler/higher_order.specialize_unify_compare.m:
compiler/jumpopt.m:
compiler/layout_out.m:
compiler/lco.m:
compiler/live_vars.m:
compiler/liveness.m:
compiler/llds_out_file.m:
compiler/make.build.m:
compiler/make.get_module_dep_info.m:
compiler/make.library_install.m:
compiler/make.module_target.m:
compiler/make.program_target.m:
compiler/make.track_flags.m:
compiler/make_hlds_passes.m:
compiler/make_module_file_names.m:
compiler/mercury_compile_front_end.m:
compiler/mercury_compile_llds_back_end.m:
compiler/mercury_compile_main.m:
compiler/mercury_compile_middle_passes.m:
compiler/ml_call_gen.m:
compiler/ml_closure_gen.m:
compiler/ml_code_gen.m:
compiler/ml_code_util.m:
compiler/ml_disj_gen.m:
compiler/ml_elim_nested.m:
compiler/ml_foreign_proc_gen.m:
compiler/ml_lookup_switch.m:
compiler/ml_switch_gen.m:
compiler/ml_unify_gen_deconstruct.m:
compiler/ml_unify_gen_test.m:
compiler/mlds_to_c_file.m:
compiler/mlds_to_target_util.m:
compiler/module_cmds.m:
compiler/opt_deps_spec.m:
compiler/optimize.m:
compiler/parse_dcg_goal.m:
compiler/parse_goal.m:
compiler/parse_item.m:
compiler/parse_module.m:
compiler/parse_string_format.m:
compiler/proc_gen.m:
compiler/prop_mode_constraints.m:
compiler/rbmm.points_to_analysis.m:
compiler/rbmm.region_analysis.m:
compiler/rbmm.region_transformation.m:
compiler/simplify_goal_disj.m:
compiler/ssdebug.m:
compiler/stack_opt.m:
compiler/string_switch.m:
compiler/switch_gen.m:
compiler/term_constr_build.m:
compiler/trace_gen.m:
compiler/tupling.m:
compiler/untupling.m:
compiler/write_deps_file.m:
deep_profiler/autopar_calc_overlap.m:
deep_profiler/autopar_find_best_par.m:
deep_profiler/html_format.m:
deep_profiler/startup.m:
profiler/mercury_profile.m:
profiler/propagate.m:
    Act on the new warnings. In a few cases, conform to the changes
    resulting from acting on the warnings in other modules.

browser/Mercury.options:
compiler/Mercury.options:
library/Mercury.options:
mdbcomp/Mercury.options:
ssdb/Mercury.options:
    Specify options for disabling the new warnings for modules
    where we (probably) won't want them.

configure.ac:
    Require the installed compiler to understand the options that
    we now reference in the Mercury.options files above.

tests/debugger/tailrec1.exp:
    Expect variable names for the middle versions of state vars
    using the new naming scheme.

tests/invalid/Mercury.options:
    Fix references to obsolete test names.

tests/warnings/Mercury.options:
    Avoid a test failure with intermodule optimization.
2025-05-19 00:33:06 +10:00

659 lines
28 KiB
Mathematica

%---------------------------------------------------------------------------%
% vim: ft=mercury ts=4 sw=4 et
%---------------------------------------------------------------------------%
% Copyright (C) 1994-2012 The University of Melbourne.
% Copyright (C) 2013-2018, 2024-2025 The Mercury team.
% This file may only be copied under the terms of the GNU General
% Public License - see the file COPYING in the Mercury distribution.
%---------------------------------------------------------------------------%
%
% File: switch_gen.m.
% Authors: conway, fjh, zs.
%
% This module determines how we should generate code for a switch, primarily
% by deciding what sort of indexing, if any, we should use.
% NOTE The code here is quite similar to the code in ml_switch_gen.m,
% which does the same thing for the MLDS back-end. Any changes here
% probably also require similar changes there.
%
% The following describes the different forms of indexing that we can use.
%
% 1 For switches on atomic data types (int, char, enums), we can use two
% smart indexing strategies.
%
% a) If all the switch arms contain only construction unifications of
% constants, then we generate a dense lookup table (an array) in which
% we look up the values of the output variables.
% Implemented by lookup_switch.m.
%
% b) If the cases are not sparse, we use a computed_goto.
% Implemented by dense_switch.m.
%
% 2 For switches on strings, we can use four smart indexing strategies,
% which are the possible combinations of two possible implementations
% of each of two aspects of the switch.
%
% The first aspect is the implementation of the lookup.
%
% a) One basic implementation approach is the use of a hash table with
% open addressing. Since the contents of the hash table is fixed,
% the open addressing can select buckets that are not the home bucket
% of any string in the table. And if we know that no two strings in
% the table share the same home address, we can dispense with open
% addressing altogether.
%
% b) The other basic implementation approach is the use of binary search.
% We generate a table containing all the strings in the switch cases in
% order, and search it using binary search.
%
% The second aspect is whether we use a lookup table. If all the switch arms
% contain only construction unifications of constants, then we extend each
% row in either the hash table or the binary search table with extra columns
% containing the values of the output variables.
%
% All these indexing strategies are implemented by string_switch.m, with
% some help from utility predicates in lookup_switch.m.
%
% 3 For switches on discriminated union types, we generate code that does
% indexing first on the primary tag, and then on the secondary tag (if
% the primary tag is shared between several function symbols). The
% indexing code for switches on both primary and secondary tags can be
% in the form of a try-me-else chain, a try chain, a dense jump table
% or a binary search.
% Implemented by tag_switch.m.
%
% XXX We should implement lookup switches on secondary tags, and (if the
% switched-on type does not use any secondary tags) on primary tags as well.
%
% 4 For switches on floats, we could generate code that does binary search.
% However, this is not yet implemented.
%
% If we cannot apply any of the above smart indexing strategies, or if the
% --smart-indexing option was disabled, then this module just generates
% a chain of if-then-elses.
%
%---------------------------------------------------------------------------%
:- module ll_backend.switch_gen.
:- interface.
:- import_module hlds.
:- import_module hlds.code_model.
:- import_module hlds.hlds_goal.
:- import_module ll_backend.code_info.
:- import_module ll_backend.code_loc_dep.
:- import_module ll_backend.llds.
:- import_module parse_tree.
:- import_module parse_tree.prog_data.
:- import_module list.
%---------------------------------------------------------------------------%
:- pred generate_switch(code_model::in, prog_var::in, can_fail::in,
list(case)::in, hlds_goal_info::in, llds_code::out,
code_info::in, code_info::out, code_loc_dep::in, code_loc_dep::out) is det.
%---------------------------------------------------------------------------%
:- implementation.
:- import_module backend_libs.
:- import_module backend_libs.lookup_switch_util.
:- import_module backend_libs.switch_util.
:- import_module hlds.goal_form.
:- import_module hlds.hlds_data.
:- import_module hlds.hlds_llds.
:- import_module hlds.hlds_module.
:- import_module hlds.hlds_out.
:- import_module hlds.hlds_out.hlds_out_goal.
:- import_module libs.
:- import_module libs.globals.
:- import_module libs.optimization_options.
:- import_module ll_backend.code_gen.
:- import_module ll_backend.dense_switch.
:- import_module ll_backend.lookup_switch.
:- import_module ll_backend.string_switch.
:- import_module ll_backend.tag_switch.
:- import_module ll_backend.trace_gen.
:- import_module ll_backend.unify_gen_test.
:- import_module parse_tree.prog_type.
:- import_module parse_tree.var_table.
:- import_module assoc_list.
:- import_module cord.
:- import_module int.
:- import_module maybe.
:- import_module pair.
:- import_module string.
%---------------------------------------------------------------------------%
generate_switch(CodeModel, SwitchVar, CanFail, Cases, GoalInfo, Code,
!CI, !CLD) :-
% Choose which method to use to generate the switch.
% CanFail says whether the switch covers all cases.
goal_info_get_store_map(GoalInfo, StoreMap),
get_next_label(EndLabel, !CI),
get_module_info(!.CI, ModuleInfo),
get_var_table(!.CI, VarTable),
lookup_var_entry(VarTable, SwitchVar, SwitchVarEntry),
SwitchVarType = SwitchVarEntry ^ vte_type,
SwitchVarName = var_entry_name(SwitchVar, SwitchVarEntry),
tag_cases(ModuleInfo, SwitchVarType, Cases, TaggedCases0,
MaybeIntSwitchInfo),
list.sort_and_remove_dups(TaggedCases0, TaggedCases),
produce_variable(SwitchVar, SwitchVarCode, SwitchVarRval, !CLD),
find_switch_category(ModuleInfo, SwitchVarType, SwitchCategory),
(
( SwitchCategory = ite_chain_switch
; SwitchCategory = float_switch
),
order_and_generate_cases(SwitchVarRval, SwitchVarType, SwitchVarName,
TaggedCases, CodeModel, CanFail, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, !.CLD)
;
( SwitchCategory = int_max_32_switch
; SwitchCategory = int_64_switch
),
module_info_get_globals(ModuleInfo, Globals),
generate_int_switch(ModuleInfo, Globals, SwitchVarRval, SwitchVarType,
SwitchVarName, TaggedCases, MaybeIntSwitchInfo,
CodeModel, CanFail, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, !.CLD)
;
SwitchCategory = string_switch,
module_info_get_globals(ModuleInfo, Globals),
generate_string_switch(Globals, SwitchVarRval, SwitchVarType,
SwitchVarName, TaggedCases, CodeModel, CanFail, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, !.CLD)
;
SwitchCategory = tag_switch,
num_cons_ids_in_tagged_cases(TaggedCases, NumConsIds, NumArms),
module_info_get_globals(ModuleInfo, Globals),
globals.get_opt_tuple(Globals, OptTuple),
TagSize = OptTuple ^ ot_tag_switch_size,
( if NumConsIds >= TagSize, NumArms > 1 then
generate_tag_switch(SwitchVarRval, SwitchVarType, SwitchVarName,
TaggedCases, CodeModel, CanFail, GoalInfo, EndLabel,
no, MaybeEnd, SwitchCode, !CI, !.CLD)
else
order_and_generate_cases(SwitchVarRval, SwitchVarType,
SwitchVarName, TaggedCases, CodeModel, CanFail, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, !.CLD)
)
),
Code = SwitchVarCode ++ SwitchCode,
after_all_branches(StoreMap, MaybeEnd, !.CI, !:CLD).
:- pred generate_int_switch(module_info::in, globals::in,
rval::in, mer_type::in, string::in, list(tagged_case)::in,
maybe_int_switch_info::in, code_model::in, can_fail::in,
hlds_goal_info::in, label::in, branch_end::out, llds_code::out,
code_info::in, code_info::out, code_loc_dep::in) is det.
generate_int_switch(ModuleInfo, Globals,
SwitchVarRval, SwitchVarType, SwitchVarName,
TaggedCases, MaybeIntSwitchInfo, CodeModel, CanFail, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, CLD) :-
num_cons_ids_in_tagged_cases(TaggedCases, NumConsIds, NumArms),
( if
MaybeIntSwitchInfo = int_switch(IntSwitchInfo),
IntSwitchInfo = int_switch_info(LowerLimit, UpperLimit, NumValues),
% Since lookup switches rely on static ground terms to work
% efficiently, there is no point in using a lookup switch
% if static ground terms are not enabled. Well, actually,
% it is possible that they might be a win in some
% circumstances, but it would take a pretty complex heuristic
% to get it right, so, lets just use a simple one - no static
% ground terms, no lookup switch.
globals.get_opt_tuple(Globals, OptTuple),
OptTuple ^ ot_use_static_ground_cells = use_static_ground_cells,
% Lookup switches do not generate trace events.
get_maybe_trace_info(!.CI, MaybeTraceInfo),
MaybeTraceInfo = no,
LookupSize = OptTuple ^ ot_lookup_switch_size,
NumConsIds >= LookupSize,
NumArms > 1,
ReqDensity = OptTuple ^ ot_lookup_switch_req_density,
filter_out_failing_cases_if_needed(CodeModel,
TaggedCases, FilteredTaggedCases,
CanFail, FilteredCanFail),
find_int_lookup_switch_params(ModuleInfo, SwitchVarType,
FilteredCanFail, LowerLimit, UpperLimit, NumValues,
ReqDensity, NeedBitVecCheck, NeedRangeCheck,
FirstVal, LastVal),
is_lookup_switch(get_int_tag, FilteredTaggedCases, GoalInfo,
!.CI, CLD, MaybeLookupSwitchInfo),
MaybeLookupSwitchInfo = yes(LookupSwitchInfo)
then
% We update MaybeEnd1 to MaybeEnd to account for the possible
% reservation of temp slots for model_non switches.
generate_int_lookup_switch(SwitchVarRval, LookupSwitchInfo,
EndLabel, FirstVal, LastVal, NeedBitVecCheck, NeedRangeCheck,
MaybeEnd, SwitchCode, !:CI)
else if
MaybeIntSwitchInfo = int_switch(IntSwitchInfo),
IntSwitchInfo = int_switch_info(LowerLimit, UpperLimit, NumValues),
globals.get_opt_tuple(Globals, OptTuple),
DenseSize = OptTuple ^ ot_dense_switch_size,
NumConsIds >= DenseSize,
NumArms > 1,
ReqDensity = OptTuple ^ ot_dense_switch_req_density,
tagged_case_list_is_dense_switch(!.CI, SwitchVarType,
TaggedCases, LowerLimit, UpperLimit, NumValues,
ReqDensity, CanFail, DenseSwitchInfo)
then
generate_dense_switch(TaggedCases, SwitchVarRval,
SwitchVarName, CodeModel, GoalInfo, DenseSwitchInfo,
EndLabel, no, MaybeEnd, SwitchCode, !CI, CLD)
else
order_and_generate_cases(SwitchVarRval, SwitchVarType, SwitchVarName,
TaggedCases, CodeModel, CanFail, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, CLD)
).
:- pred generate_string_switch(globals::in, rval::in, mer_type::in, string::in,
list(tagged_case)::in, code_model::in, can_fail::in,
hlds_goal_info::in, label::in, branch_end::out, llds_code::out,
code_info::in, code_info::out, code_loc_dep::in) is det.
generate_string_switch(Globals, SwitchVarRval, SwitchVarType, SwitchVarName,
TaggedCases, CodeModel, CanFail, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, CLD) :-
filter_out_failing_cases_if_needed(CodeModel,
TaggedCases, FilteredTaggedCases, CanFail, FilteredCanFail),
num_cons_ids_in_tagged_cases(FilteredTaggedCases, NumConsIds, NumArms),
% If we don't have at least two arms, an if-then-else chain
% (which will contain at most one if-then-else) will be faster than
% any other method for implementing the "switch".
( if NumArms < 2 then
order_and_generate_cases(SwitchVarRval, SwitchVarType, SwitchVarName,
TaggedCases, CodeModel, CanFail, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, CLD)
else
generate_smart_string_switch(Globals, SwitchVarRval, SwitchVarType,
SwitchVarName, TaggedCases, FilteredTaggedCases,
CodeModel, CanFail, FilteredCanFail, NumConsIds, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, CLD)
).
:- pred generate_smart_string_switch(globals::in, rval::in, mer_type::in,
string::in, list(tagged_case)::in, list(tagged_case)::in,
code_model::in, can_fail::in, can_fail::in, int::in, hlds_goal_info::in,
label::in, branch_end::out, llds_code::out,
code_info::in, code_info::out, code_loc_dep::in) is det.
generate_smart_string_switch(Globals,
SwitchVarRval, SwitchVarType, SwitchVarName,
TaggedCases, FilteredTaggedCases,
CodeModel, CanFail, FilteredCanFail, NumConsIds, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, CLD) :-
is_lookup_switch(get_string_tag, FilteredTaggedCases, GoalInfo,
!.CI, CLD, MaybeLookupSwitchInfo),
globals.get_opt_tuple(Globals, OptTuple),
( if
% For now, string trie switches have been implemented only for C,
% and the code assumes that the host and the target use the same
% string encoding. This effectively requires that both the host and
% the target compile to C.
%
% Unlike ml_switch_gen.m, we don't need to check whether
% we are targeting C.
internal_string_encoding = utf8, % host is C
StringTrieSwitchSize = OptTuple ^ ot_string_trie_switch_size,
NumConsIds >= StringTrieSwitchSize
then
(
MaybeLookupSwitchInfo = yes(LookupSwitchInfo),
generate_string_trie_lookup_switch(SwitchVarRval,
LookupSwitchInfo, FilteredCanFail,
EndLabel, MaybeEnd, SwitchCode, !:CI)
;
MaybeLookupSwitchInfo = no,
generate_string_trie_jump_switch(SwitchVarRval, SwitchVarName,
TaggedCases, CodeModel, CanFail, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, CLD)
)
else if
StringHashSwitchSize = OptTuple ^ ot_string_hash_switch_size,
NumConsIds >= StringHashSwitchSize
then
(
MaybeLookupSwitchInfo = yes(LookupSwitchInfo),
% We update MaybeEnd to account for the possible reservation
% of temp slots for nondet switches.
generate_string_hash_lookup_switch(SwitchVarRval,
LookupSwitchInfo, FilteredCanFail,
EndLabel, MaybeEnd, SwitchCode, !:CI)
;
MaybeLookupSwitchInfo = no,
generate_string_hash_jump_switch(SwitchVarRval, SwitchVarName,
TaggedCases, CodeModel, CanFail, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, CLD)
)
else if
StringBinarySwitchSize = OptTuple ^ ot_string_binary_switch_size,
NumConsIds >= StringBinarySwitchSize
then
(
MaybeLookupSwitchInfo = yes(LookupSwitchInfo),
% We update MaybeEnd to account for the possible reservation
% of temp slots for nondet switches.
generate_string_binary_lookup_switch(SwitchVarRval,
LookupSwitchInfo, FilteredCanFail, EndLabel,
MaybeEnd, SwitchCode, !:CI)
;
MaybeLookupSwitchInfo = no,
generate_string_binary_jump_switch(SwitchVarRval, SwitchVarName,
TaggedCases, CodeModel, CanFail, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, CLD)
)
else
order_and_generate_cases(SwitchVarRval, SwitchVarType, SwitchVarName,
TaggedCases, CodeModel, CanFail, GoalInfo,
EndLabel, MaybeEnd, SwitchCode, !CI, CLD)
).
%---------------------------------------------------------------------------%
%---------------------------------------------------------------------------%
% Generate a switch as a chain of if-then-elses.
%
% To generate a case for a switch, we generate code to do a tag-test,
% and fall through to the next case in the event of failure.
%
% Each case except the last consists of
%
% - a tag test, jumping to the next case if it fails;
% - the goal for that case;
% - code to move variables to where the store map says they ought to be;
% - a branch to the end of the switch.
%
% For the last case, if the switch covers all cases that can occur,
% we don't need to generate the tag test, and we never need to generate
% the branch to the end of the switch.
%
% After the last case, we put the end-of-switch label which other cases
% branch to after their case goals.
%
% In the important special case of a det switch with two cases,
% we try to find out which case will be executed more frequently,
% and put that one first. This minimizes the number of pipeline
% breaks caused by taken branches.
%
:- pred order_and_generate_cases(rval::in, mer_type::in, string::in,
list(tagged_case)::in, code_model::in, can_fail::in, hlds_goal_info::in,
label::in, branch_end::out, llds_code::out,
code_info::in, code_info::out, code_loc_dep::in) is det.
order_and_generate_cases(VarRval, VarType, VarName, TaggedCases,
CodeModel, CanFail, GoalInfo, EndLabel, MaybeEnd, Code, !CI, CLD) :-
order_cases(TaggedCases, OrderedTaggedCases, CodeModel, CanFail, !.CI),
type_to_ctor_det(VarType, TypeCtor),
get_module_info(!.CI, ModuleInfo),
module_info_get_type_table(ModuleInfo, TypeTable),
( if search_type_ctor_defn(TypeTable, TypeCtor, TypeDefn) then
get_type_defn_body(TypeDefn, TypeBody),
CheaperTagTest = get_maybe_cheaper_tag_test(TypeBody)
else
CheaperTagTest = no_cheaper_tag_test
),
remember_position(CLD, BranchStart),
generate_if_then_else_chain_cases(CheaperTagTest,
VarRval, VarType, VarName, OrderedTaggedCases,
CodeModel, CanFail, GoalInfo, BranchStart,
EndLabel, no, MaybeEnd, Code, !CI).
:- pred order_cases(list(tagged_case)::in, list(tagged_case)::out,
code_model::in, can_fail::in, code_info::in) is det.
order_cases(Cases0, Cases, CodeModel, CanFail, CI) :-
% We do ordering here based on three considerations.
%
% - We try to put cases that can succeed before ones that cannot, since
% cases that cannot succeed clearly won't be executed frequently.
%
% - If the recursion structure of the predicate is sufficiently simple that
% we can make a good guess at which case will be executed more
% frequently, we try to put the frequent case first.
%
% - We try to put cheap-to-execute tests first; for arms with more than one
% cons_id, we sum the costs of their tests. The main aim of this is to
% reduce the average cost at runtime. For cannot_fail switches, putting
% the most expensive-to-test case last has the additional benefit that
% we don't ever need to execute that test, since the failure of all the
% previous ones guarantees that it could not fail. This should be
% especially useful for switches in which many cons_ids share a single
% arm.
%
% Each consideration is implemented by its own predicate, which calls the
% predicate of the next consideration to decide ties. The predicates for
% the three considerations are
%
% - order_cases (this predicate),
% - order_recursive_cases,
% - order_tag_test_cost
%
% respectively.
separate_cannot_succeed_cases(Cases0, CanSucceedCases, CannotSucceedCases),
(
CannotSucceedCases = [],
order_recursive_cases(Cases0, Cases, CodeModel, CanFail, CI)
;
CannotSucceedCases = [_ | _],
% There is no point in calling order_recursive_cases in this situation.
Cases = CanSucceedCases ++ CannotSucceedCases
).
:- pred separate_cannot_succeed_cases(list(tagged_case)::in,
list(tagged_case)::out, list(tagged_case)::out) is det.
separate_cannot_succeed_cases([], [], []).
separate_cannot_succeed_cases([Case | Cases],
CanSucceedCases, CannotSucceedCases) :-
separate_cannot_succeed_cases(Cases,
CanSucceedCases1, CannotSucceedCases1),
Case = tagged_case(_, _, _, Goal),
Goal = hlds_goal(_, GoalInfo),
Detism = goal_info_get_determinism(GoalInfo),
determinism_components(Detism, _CanFail, SolnCount),
(
( SolnCount = at_most_one
; SolnCount = at_most_many_cc
; SolnCount = at_most_many
),
CanSucceedCases = [Case | CanSucceedCases1],
CannotSucceedCases = CannotSucceedCases1
;
SolnCount = at_most_zero,
CanSucceedCases = CanSucceedCases1,
CannotSucceedCases = [Case | CannotSucceedCases1]
).
%---------------------------------------------------------------------------%
:- pred order_recursive_cases(list(tagged_case)::in, list(tagged_case)::out,
code_model::in, can_fail::in, code_info::in) is det.
order_recursive_cases(Cases0, Cases, CodeModel, CanFail, CI) :-
( if
CodeModel = model_det,
CanFail = cannot_fail,
Cases0 = [Case1, Case2],
Case1 = tagged_case(_, _, _, Goal1),
Case2 = tagged_case(_, _, _, Goal2)
then
get_module_info(CI, ModuleInfo),
module_info_get_globals(ModuleInfo, Globals),
get_pred_id(CI, PredId),
get_proc_id(CI, ProcId),
count_recursive_calls(Goal1, PredId, ProcId, Min1, Max1),
count_recursive_calls(Goal2, PredId, ProcId, Min2, Max2),
( if
( if
Max1 = 0, % Goal1 is a base case
Min2 = 1 % Goal2 is probably singly recursive
then
BaseCase = Case1,
SingleRecCase = Case2
else if
Max2 = 0, % Goal2 is a base case
Min1 = 1 % Goal1 is probably singly recursive
then
BaseCase = Case2,
SingleRecCase = Case1
else
fail
)
then
globals.get_opt_tuple(Globals, OptTuple),
SingleRecBaseFirst = OptTuple ^ ot_put_base_first_single_rec,
(
SingleRecBaseFirst = put_base_first_single_rec,
Cases = [SingleRecCase, BaseCase]
;
SingleRecBaseFirst = do_not_put_base_first_single_rec,
Cases = [BaseCase, SingleRecCase]
)
else if
( if
Max1 = 0, % Goal1 is a base case
Min2 > 1 % Goal2 is at least doubly recursive
then
BaseCase = Case1,
MultiRecCase = Case2
else if
Max2 = 0, % Goal2 is a base case
Min1 > 1 % Goal1 is at least doubly recursive
then
BaseCase = Case2,
MultiRecCase = Case1
else
fail
)
then
globals.get_opt_tuple(Globals, OptTuple),
MultiRecBaseFirst = OptTuple ^ ot_put_base_first_multi_rec,
(
MultiRecBaseFirst = put_base_first_multi_rec,
Cases = [BaseCase, MultiRecCase]
;
MultiRecBaseFirst = do_not_put_base_first_multi_rec,
Cases = [MultiRecCase, BaseCase]
)
else
order_tag_test_cost(Cases0, Cases)
)
else
order_tag_test_cost(Cases0, Cases)
).
%---------------------------------------------------------------------------%
:- pred order_tag_test_cost(list(tagged_case)::in, list(tagged_case)::out)
is det.
order_tag_test_cost(Cases0, Cases) :-
CostedCases = list.map(estimate_cost_of_case_test, Cases0),
list.sort(CostedCases, SortedCostedCases),
assoc_list.values(SortedCostedCases, Cases).
:- func estimate_cost_of_case_test(tagged_case) = pair(int, tagged_case).
estimate_cost_of_case_test(TaggedCase) = Cost - TaggedCase :-
TaggedCase = tagged_case(MainTaggedConsId, OtherTaggedConsIds, _, _),
MainTag = project_tagged_cons_id_tag(MainTaggedConsId),
MainCost = estimate_switch_tag_test_cost(MainTag),
OtherTags = list.map(project_tagged_cons_id_tag, OtherTaggedConsIds),
OtherCosts = list.map(estimate_switch_tag_test_cost, OtherTags),
Cost = list.foldl(int.plus, [MainCost | OtherCosts], 0).
%---------------------------------------------------------------------------%
:- pred generate_if_then_else_chain_cases(maybe_cheaper_tag_test::in,
rval::in, mer_type::in, string::in, list(tagged_case)::in,
code_model::in, can_fail::in, hlds_goal_info::in,
position_info::in, label::in, branch_end::in, branch_end::out,
llds_code::out, code_info::in, code_info::out) is det.
generate_if_then_else_chain_cases(CheaperTagTest, VarRval, VarType, VarName,
Cases, CodeModel, CanFail, SwitchGoalInfo,
BranchStart, EndLabel, !MaybeEnd, Code, !CI) :-
(
Cases = [HeadCase | TailCases],
HeadCase = tagged_case(MainTaggedConsId, OtherTaggedConsIds, _, Goal),
goal_info_get_store_map(SwitchGoalInfo, StoreMap),
( if
( TailCases = [_ | _]
; CanFail = can_fail
)
then
generate_test_var_has_one_tagged_cons_id(VarRval, VarName,
MainTaggedConsId, OtherTaggedConsIds, CheaperTagTest,
branch_on_failure, NextLabel, TestCode, !CI),
ElseCode = from_list([
llds_instr(goto(code_label(EndLabel)),
"skip to the end of the switch on " ++ VarName),
llds_instr(label(NextLabel), "next case")
])
else
% When debugging code generator output, we need a way to tell
% which case's code is next. We normally hang this comment
% on the test, but in this case there is no test.
project_cons_name_and_tag(MainTaggedConsId, MainConsName, _),
list.map2(project_cons_name_and_tag, OtherTaggedConsIds,
OtherConsNames, _),
Comment = case_comment(VarName, MainConsName, OtherConsNames),
TestCode = singleton(
llds_instr(comment(Comment), "")
),
ElseCode = empty
),
some [!CLD] (
reset_to_position(BranchStart, !.CI, !:CLD),
maybe_generate_internal_event_code(Goal, SwitchGoalInfo, TraceCode,
!CI, !CLD),
generate_goal(CodeModel, Goal, GoalCode, !CI, !CLD),
generate_branch_end(StoreMap, !MaybeEnd, SaveCode, !.CLD)
),
HeadCaseCode = TestCode ++ TraceCode ++ GoalCode ++ SaveCode ++
ElseCode,
generate_if_then_else_chain_cases(CheaperTagTest,
VarRval, VarType, VarName, TailCases,
CodeModel, CanFail, SwitchGoalInfo,
BranchStart, EndLabel, !MaybeEnd, TailCasesCode, !CI),
Code = HeadCaseCode ++ TailCasesCode
;
Cases = [],
(
CanFail = can_fail,
% At the end of a locally semidet switch, we fail because we came
% across a tag which was not covered by one of the cases. It is
% followed by the end of switch label to which the cases branch.
some [!CLD] (
reset_to_position(BranchStart, !.CI, !:CLD),
generate_failure(FailCode, !.CI, !.CLD)
)
;
CanFail = cannot_fail,
FailCode = empty
),
EndCode = singleton(
llds_instr(label(EndLabel),
"end of the switch on " ++ VarName)
),
Code = FailCode ++ EndCode
).
%---------------------------------------------------------------------------%
:- end_module ll_backend.switch_gen.
%---------------------------------------------------------------------------%