Files
mercury/compiler/add_foreign_proc.m
Zoltan Somogyi 5193012242 Fix a fourth batch of C# invalid test fails.
compiler/add_foreign_proc.m:
    Don't module qualify the named of predicates/functions
    defined by foreign_procs when generating an error message
    about that foreign_proc, since

    - you can add foreign_procs only for local predicates/functions, and
    - even if you one for a NON-local predicate/function, the context will
      tell you exactly which foreign_proc is being complained about.

    Also, improve the wording of the error message.

tests/invalid/instances_pc.instances_pc_helper_1.err_exp2:
tests/invalid/instances_pc.m:
    Add a new expected output file for bootchecks in C# and Java grades,
    and document the new expected output file.

tests/invalid/pragma_c_code_dup_var.{err_exp2,err_exp3}:
tests/invalid/pragma_c_code_dup_var.{m,err_exp}:
    Add two new expected output files for bootchecks in C# and Java grades.
    Note the role of each expected output file, and apply the resulting
    line number changes to the expected output file for C.

tests/invalid_make_int/instance_no_type.{m,int_err_exp}:
    Move this test case from tests/invalid to tests/invalid_make_int,
    since the compiler now diagnoses the bug it tests for
    during the creation of the .int file.

tests/invalid_make_int/pragma_export_int.{m,err_exp}:
    Move half of the pragma_export test case here from tests/invalid,
    since the compiler now diagnoses the bug it tests for
    during the creation of the .int file.

tests/invalid_nodepend/pragma_export.{m,err_exp}:
    Move the other half of the pragma_export test case here from tests/invalid.
    With mmc --make, the diagnosable-at-.int-file-creation-time errors
    prevent the compiler from ever getting to the errors that are not
    diagnosable at that time, so delete the code that gives rise to those
    diagnosable-at-that-time errors from this version of the test case..

tests/invalid/Mmakefile:
    Delete the tests moved to other test directories.

tests/invalid_make_int/Mercury.options:
tests/invalid_make_int/Mmakefile:
tests/invalid_nodepend/Mmakefile:
    Add the tests moved to these directories from tests/invalid.
2023-10-26 15:18:41 +11:00

731 lines
34 KiB
Mathematica

%-----------------------------------------------------------------------------%
% vim: ft=mercury ts=4 sw=4 et
%-----------------------------------------------------------------------------%
% Copyright (C) 2015 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.
%-----------------------------------------------------------------------------%
:- module hlds.make_hlds.add_foreign_proc.
:- interface.
:- import_module hlds.hlds_module.
:- import_module hlds.make_hlds.make_hlds_types.
:- import_module hlds.status.
:- import_module parse_tree.
:- import_module parse_tree.error_spec.
:- import_module parse_tree.prog_item.
:- import_module io.
:- import_module list.
:- pred add_foreign_procs(io.text_output_stream::in,
ims_list(item_foreign_proc_info)::in, module_info::in, module_info::out,
list(error_spec)::in, list(error_spec)::out) is det.
:- pred add_foreign_proc(io.text_output_stream::in, item_mercury_status::in,
pred_status::in, item_foreign_proc_info::in,
module_info::in, module_info::out,
list(error_spec)::in, list(error_spec)::out) is det.
%-----------------------------------------------------------------------------%
:- implementation.
:- import_module hlds.add_pred.
:- import_module hlds.hlds_args.
:- import_module hlds.hlds_clauses.
:- import_module hlds.hlds_code_util.
:- import_module hlds.hlds_goal.
:- import_module hlds.hlds_pred.
:- import_module hlds.hlds_rtti.
:- import_module hlds.make_hlds.make_hlds_warn.
:- import_module hlds.pred_name.
:- import_module hlds.pred_table.
:- import_module hlds.quantification.
:- import_module libs.
:- import_module libs.globals.
:- import_module libs.options.
:- import_module mdbcomp.
:- import_module mdbcomp.prim_data.
:- import_module mdbcomp.sym_name.
:- import_module parse_tree.parse_tree_out_misc.
:- import_module parse_tree.parse_tree_out_sym_name.
:- import_module parse_tree.parse_tree_out_term.
:- import_module parse_tree.prog_ctgc.
:- import_module parse_tree.prog_data.
:- import_module parse_tree.prog_data_foreign.
:- import_module parse_tree.prog_foreign.
:- import_module parse_tree.prog_util.
:- import_module parse_tree.vartypes.
:- import_module bag.
:- import_module bool.
:- import_module int.
:- import_module map.
:- import_module maybe.
:- import_module pair.
:- import_module require.
:- import_module string.
%-----------------------------------------------------------------------------%
add_foreign_procs(_, [], !ModuleInfo, !Specs).
add_foreign_procs(ProgressStream, [ImsSubList | ImsSubLists],
!ModuleInfo, !Specs) :-
ImsSubList = ims_sub_list(ItemMercuryStatus, PragmaFPInfos),
item_mercury_status_to_pred_status(ItemMercuryStatus, PredStatus),
list.foldl2(
add_foreign_proc(ProgressStream, ItemMercuryStatus, PredStatus),
PragmaFPInfos, !ModuleInfo, !Specs),
add_foreign_procs(ProgressStream, ImsSubLists, !ModuleInfo, !Specs).
%-----------------------------------------------------------------------------%
add_foreign_proc(ProgressStream, ItemMercurystatus, PredStatus, FPInfo,
!ModuleInfo, !Specs) :-
FPInfo = item_foreign_proc_info(Attributes0, PredSymName, PredOrFunc,
PragmaVars, ProgVarSet, _InstVarset, PragmaImpl, Context, SeqNum),
(
PredSymName = qualified(PredModuleName, PredName)
;
PredSymName = unqualified(_),
unexpected($pred, "unexpected PredSymName")
),
PredFormArity = arg_list_arity(PragmaVars),
PFSymNameArity = pf_sym_name_arity(PredOrFunc, PredSymName, PredFormArity),
module_info_get_globals(!.ModuleInfo, Globals),
globals.lookup_bool_option(Globals, very_verbose, VeryVerbose),
(
VeryVerbose = yes,
trace [io(!IO)] (
IdStr = pf_sym_name_pred_form_arity_to_string(PFSymNameArity),
io.format(ProgressStream,
"%% Processing `:- pragma foreign_proc' for %s...\n",
[s(IdStr)], !IO)
)
;
VeryVerbose = no
),
% Lookup the pred declaration in the predicate table.
% If it is not there, print an error message and insert
% a dummy declaration for the predicate.
module_info_get_predicate_table(!.ModuleInfo, PredTable0),
predicate_table_lookup_pf_m_n_a(PredTable0, is_fully_qualified,
PredOrFunc, PredModuleName, PredName, PredFormArity, PredIds),
(
PredIds = [],
user_arity_pred_form_arity(PredOrFunc, UserArity, PredFormArity),
Origin = origin_user(user_made_pred(PredOrFunc,
PredSymName, UserArity)),
add_implicit_pred_decl_report_error(PredOrFunc, PredModuleName,
PredName, PredFormArity, PredStatus, is_not_a_class_method,
Context, Origin,
[pragma_decl("foreign_proc"), words("declaration")],
PredId, !ModuleInfo, !Specs)
;
PredIds = [PredId]
;
PredIds = [PredId, _ | _],
% Any attempt to define more than one pred with the same PredOrFunc,
% PredSymName and Arity should have been caught earlier, and an error
% message generated. We continue so that we can try to find more
% errors.
AmbiPieces = [words("Error: ambiguous predicate name"),
qual_pf_sym_name_pred_form_arity(PFSymNameArity), words("in"),
quote("pragma foreign_proc"), suffix("."), nl],
AmbiSpec = simplest_spec($pred, severity_error,
phase_parse_tree_to_hlds, Context, AmbiPieces),
!:Specs = [AmbiSpec | !.Specs]
),
% Lookup the pred_info for this pred, add the pragma to the proc_info
% in the proc_table in the pred_info, and save the pred_info.
some [!PredInfo] (
module_info_pred_info(!.ModuleInfo, PredId, !:PredInfo),
% status_opt_imported preds are initially tagged as status_imported
% and are tagged as status_opt_imported only if/when we see a clause
% (including a `foreign_proc' clause) for them.
( if PredStatus = pred_status(status_opt_imported) then
pred_info_set_status(pred_status(status_opt_imported), !PredInfo)
else
true
),
% Record the existence of this "clause".
pred_info_get_clauses_info(!.PredInfo, ClausesInfo0),
ItemNumbers0 = ClausesInfo0 ^ cli_item_numbers,
add_clause_item_number(SeqNum, Context, item_is_foreign_proc,
ItemNumbers0, ItemNumbers),
ClausesInfo1 = ClausesInfo0 ^ cli_item_numbers := ItemNumbers,
pred_info_set_clauses_info(ClausesInfo1, !PredInfo),
module_info_set_pred_info(PredId, !.PredInfo, !ModuleInfo),
% Replace any maybe_thread_safe foreign_proc attributes with
% the actual thread safety attributes which we get from the
% `--maybe-thread-safe' option.
globals.get_maybe_thread_safe(Globals, MaybeThreadSafe),
ThreadSafe = get_thread_safe(Attributes0),
(
ThreadSafe = proc_maybe_thread_safe,
(
MaybeThreadSafe = yes,
set_thread_safe(proc_thread_safe, Attributes0, Attributes)
;
MaybeThreadSafe = no,
set_thread_safe(proc_not_thread_safe, Attributes0, Attributes)
)
;
( ThreadSafe = proc_thread_safe
; ThreadSafe = proc_not_thread_safe
),
Attributes = Attributes0
),
CurrentBackend = lookup_current_backend(Globals),
globals.get_backend_foreign_languages(Globals, BackendForeignLangs),
PragmaForeignLanguage = get_foreign_language(Attributes),
MaybeForSpecificBackend = get_for_specific_backend(Attributes),
( if
% Don't allow definitions, whether clauses or foreign_procs,
% for imported predicates/functions. Nota that this applies to
% *plain imported( predicates/function, not the *opt-imported*
% ones.
pred_info_is_imported(!.PredInfo)
then
Pieces = [words("Error:"), pragma_decl("foreign_proc"),
words("declaration for imported"),
qual_pf_sym_name_pred_form_arity(PFSymNameArity),
suffix("."), nl],
Spec = simplest_spec($pred, severity_error,
phase_parse_tree_to_hlds, Context, Pieces),
!:Specs = [Spec | !.Specs]
else if
( if
not list.member(PragmaForeignLanguage, BackendForeignLangs)
then
RejectCause =
wrong_lang(PragmaForeignLanguage, BackendForeignLangs)
else if
MaybeForSpecificBackend = yes(SpecificBackend),
SpecificBackend \= CurrentBackend
then
RejectCause = right_lang_wrong_backend
else
fail
)
then
% This foreign_proc is for the wrong language or the wrong backend.
% This is ok when coming from a source file, which is allowed
% to define a predicate by different foreign_procs for different
% target languages and/or backends. This is not ok when coming
% from a .opt file, which (if everything is set up properly)
% we should be reading only if it has been generated for the
% current grade.
( if ItemMercurystatus =
item_defined_in_other_module(item_import_opt_int)
then
report_bad_foreign_proc_in_dot_opt_file(RejectCause, Context,
!Specs)
else
true
),
% XXX The next two calls seem redundant, and in most cases,
% they are. However, deleting them results in the failure
% of the hard_coded/foreign_type2 test case, with this message:
% In clause for function `x'/1:
% error: undefined symbol `foreign_type2.x'/1.
% There are `:- pragma foreign_type' declarations for type
% `foreign_type2.coord'/1, so it is treated as an abstract
% type in all predicates and functions which are not
% implemented for those foreign types.
% The reason for this is the fact that convert_cons_defn
% in typecheck.m checks whether the goal type is
% goal_not_for_promise(np_goal_type_clause_and_foreign),
% and it is these two calls that set up the "_and_foreign"
% part of that.
%
% XXX Originally, we executed these two calls only in
% the wrong_lang case, and this was enough for fixing the
% foreign_type2 test case. Whether we need them in the
% right_lang_wrong_backend case is a guess, which would need
% a dedicated test case to test. However, since backend-specific
% eternal pragmas are intended only for implementors,
% this is not an urgent matter.
pred_info_update_goal_type(np_goal_type_foreign, !PredInfo),
module_info_set_pred_info(PredId, !.PredInfo, !ModuleInfo)
else
% Add the pragma declaration to the proc_info for this procedure.
pred_info_get_proc_table(!.PredInfo, Procs),
map.to_assoc_list(Procs, ExistingProcs),
pragma_get_modes(PragmaVars, Modes),
( if
% The inst variables for the foreign_proc declaration
% and predmode declarations are from different varsets.
% We cannot just unify the argument modes directly because
% the representation of the inst variables may be different.
% Instead we need to allow for a renaming between the
% inst variables in the argument modes of the foreign_proc
% and those of the predmode declaration.
%
% XXX We should probably also check that each pair in
% the renaming has the same name.
get_procedure_matching_declmodes_with_renaming(!.ModuleInfo,
ExistingProcs, Modes, ProcId)
then
pred_info_get_arg_types(!.PredInfo, ArgTypes),
pred_info_get_purity(!.PredInfo, Purity),
pred_info_get_markers(!.PredInfo, Markers),
clauses_info_add_foreign_proc(PredOrFunc,
PredModuleName, PredName, PredId, ProcId,
ProgVarSet, PragmaVars, ArgTypes,
Purity, Attributes, Markers, Context, PragmaImpl,
ClausesInfo1, ClausesInfo, !ModuleInfo, !Specs),
pred_info_set_clauses_info(ClausesInfo, !PredInfo),
pred_info_update_goal_type(np_goal_type_foreign, !PredInfo),
module_info_set_pred_info(PredId, !.PredInfo, !ModuleInfo),
pragma_get_var_infos(PragmaVars, ArgInfos),
ArgNameModes = list.map(
foreign_arg_name_mode_box_project_maybe_name_mode,
ArgInfos),
warn_singletons_in_pragma_foreign_proc(!.ModuleInfo,
PragmaImpl, PragmaForeignLanguage, ArgNameModes, Context,
PFSymNameArity, PredId, ProcId, !Specs)
else
Pieces = [words("Error:"),
pragma_decl("foreign_proc"), words("declaration"),
words("for undeclared mode of"),
qual_pf_sym_name_pred_form_arity(PFSymNameArity),
suffix("."), nl],
Spec = simplest_spec($pred, severity_error,
phase_parse_tree_to_hlds, Context, Pieces),
!:Specs = [Spec | !.Specs]
)
)
).
:- type reject_cause
---> wrong_lang(foreign_language, list(foreign_language))
; right_lang_wrong_backend.
:- pred report_bad_foreign_proc_in_dot_opt_file(reject_cause::in,
prog_context::in, list(error_spec)::in, list(error_spec)::out) is det.
report_bad_foreign_proc_in_dot_opt_file(RejectCause, Context, !Specs) :-
(
RejectCause = wrong_lang(PragmaLang, BackendForeignLangs),
PragmaLangStr = foreign_language_string(PragmaLang),
FrontPieces = [words("Error:"), pragma_decl("foreign_proc"),
words("declaration in a .opt file for a foreign language,"),
words(PragmaLangStr), suffix(",")],
(
BackendForeignLangs = [],
unexpected($pred, "BackendForeignLangs = []")
;
BackendForeignLangs = [BackendForeignLang],
BackendLangStr = foreign_language_string(BackendForeignLang),
MainPieces = FrontPieces ++ [words("which differs from"),
words("the only language supported by the current backend,"),
words("which is"), words(BackendLangStr), suffix("."), nl]
;
BackendForeignLangs = [_, _ | _],
BackendLangStrs =
list.map(foreign_language_string, BackendForeignLangs),
BackendLangsStr = list_to_pieces(BackendLangStrs),
MainPieces = FrontPieces ++ [words("which is not one of the"),
words("languages supported by the current backend,"),
words("which are")] ++ BackendLangsStr ++ [suffix("."), nl]
)
;
RejectCause = right_lang_wrong_backend,
MainPieces = [words("Error:"), pragma_decl("foreign_proc"),
words("declaration in a .opt file"),
words("whose backend attribute states that"),
words("it is not for the current grade."), nl]
),
Pieces = MainPieces ++ [words("This indicates that the .opt file"),
words("was generated for a different grade."),
words("You will need to rebuild this file"),
words("for the current grade."), nl],
Spec = simplest_spec($pred, severity_error, phase_parse_tree_to_hlds,
Context, Pieces),
!:Specs = [Spec | !.Specs].
% Add the pragma_foreign_proc goal to the clauses_info for this procedure.
% To do so, we must also insert unifications between the variables in the
% pragma foreign_proc declaration and the head vars of the pred. Also
% return the hlds_goal.
%
:- pred clauses_info_add_foreign_proc(pred_or_func::in,
module_name::in, string::in, pred_id::in, proc_id::in,
prog_varset::in, list(pragma_var)::in, list(mer_type)::in,
purity::in, foreign_proc_attributes::in, pred_markers::in,
prog_context::in, pragma_foreign_proc_impl::in,
clauses_info::in, clauses_info::out, module_info::in, module_info::out,
list(error_spec)::in, list(error_spec)::out) is det.
clauses_info_add_foreign_proc(PredOrFunc, PredModuleName, PredName,
PredId, ProcId, VarSet, PragmaVars, OrigArgTypes,
Purity, Attributes0, Markers, Context, PragmaImpl0,
!ClausesInfo, !ModuleInfo, !Specs) :-
module_info_pred_info(!.ModuleInfo, PredId, PredInfo),
( if pred_info_is_builtin(PredInfo) then
% When bootstrapping a change that defines a builtin using
% normal Mercury code, we need to disable the generation
% of the error message, and just ignore the definition.
module_info_get_globals(!.ModuleInfo, Globals),
globals.lookup_bool_option(Globals, allow_defn_of_builtins,
AllowDefnOfBuiltin),
(
AllowDefnOfBuiltin = no,
Pieces = [words("Error: foreign_proc for builtin."), nl],
Spec = simplest_spec($pred, severity_error,
phase_parse_tree_to_hlds, Context, Pieces),
!:Specs = [Spec | !.Specs]
;
AllowDefnOfBuiltin = yes
)
else
AllProcIds = pred_info_all_procids(PredInfo),
clauses_info_do_add_foreign_proc(PredOrFunc,
PredModuleName, PredName, PredId, ProcId, AllProcIds,
VarSet, PragmaVars, OrigArgTypes, Purity, Attributes0, Markers,
Context, PragmaImpl0, !ClausesInfo, !ModuleInfo, !Specs)
).
:- pred clauses_info_do_add_foreign_proc(pred_or_func::in,
module_name::in, string::in, pred_id::in, proc_id::in, list(proc_id)::in,
prog_varset::in, list(pragma_var)::in, list(mer_type)::in,
purity::in, foreign_proc_attributes::in, pred_markers::in,
prog_context::in, pragma_foreign_proc_impl::in,
clauses_info::in, clauses_info::out, module_info::in, module_info::out,
list(error_spec)::in, list(error_spec)::out) is det.
clauses_info_do_add_foreign_proc(PredOrFunc, PredModuleName, PredName,
PredId, ProcId, AllProcIds, PVarSet, PragmaVars, OrigArgTypes,
Purity, Attributes0, Markers, Context, PragmaImpl,
!ClausesInfo, !ModuleInfo, !Specs) :-
% Our caller should have already added this foreign_proc to ItemNumbers.
!.ClausesInfo = clauses_info(VarSet0, ExplicitVarTypes,
VarTable, RttiVarMaps, TVarNameMap, HeadVars, ClausesRep0,
ItemNumbers, _HasForeignClauses, HadSyntaxError),
get_clause_list_for_replacement(ClausesRep0, Clauses0),
% Currently we can override Mercury clauses with a foreign_proc right here,
% which means that semantic analysis never sees those Mercury clauses.
% Any errors in them thus do get picked not when they first arise, but
% only when the code gets compiled for a target that requires their use.
% XXX We should retain and check the Mercury clauses, and override them
% with a more specific foreign language implementation only after semantic
% analysis.
module_info_get_globals(!.ModuleInfo, Globals),
globals.get_target(Globals, Target),
NewLang = get_foreign_language(Attributes0),
PredFormArity = arg_list_arity(OrigArgTypes),
add_foreign_proc_update_existing_clauses(Globals, PredOrFunc,
PredModuleName, PredName, PredFormArity, Context, Target, NewLang,
AllProcIds, ProcId, Overridden, Clauses0, Clauses1, !Specs),
% We used have this code here, but as of 2022 feb 15, and almost certainly
% for a long, long time before that, it effectively did nothing.
% % If the foreign language is not one of the backend languages, we will
% % have to generate an interface to it in a backend language.
% globals.get_backend_foreign_languages(Globals, BackendForeignLanguages),
% foreign.extrude_pragma_implementation(
% PredOrFunc, PredModuleName, PredName, PragmaVars, Context, !ModuleInfo,
% Attributes0, Attributes1, PragmaImpl0, PragmaImpl),
% Check for arguments occurring more than once.
pragma_get_vars_and_var_infos(PragmaVars, ArgVars, ArgInfos),
bag.init(ArgVarBag0),
bag.insert_list(ArgVars, ArgVarBag0, ArgVarBag),
bag.to_assoc_list(ArgVarBag, ArgVarBagAssocList),
list.filter_map(
( pred(ArgPair::in, Var::out) is semidet :-
ArgPair = Var - Occurrences,
Occurrences > 1
), ArgVarBagAssocList, MultiplyOccurringArgVars),
(
MultiplyOccurringArgVars = [_ | _],
user_arity_pred_form_arity(PredOrFunc, UserArity, PredFormArity),
PredSymName = qualified(PredModuleName, PredName),
PFSymNameArity =
pred_pf_name_arity(PredOrFunc, PredSymName, UserArity),
Pieces1 = [words("In"), pragma_decl("foreign_proc"),
words("declaration for"),
unqual_pf_sym_name_user_arity(PFSymNameArity), suffix(":"), nl],
(
MultiplyOccurringArgVars = [MultiplyOccurringArgVar],
BadVarStr = mercury_var_to_name_only_vs(PVarSet,
MultiplyOccurringArgVar),
Pieces2 = [words("error: variable"), quote(BadVarStr),
words("occurs more than once in the argument list."), nl]
;
MultiplyOccurringArgVars = [_, _ | _],
BadVarsStr = mercury_vars_to_name_only_vs(PVarSet,
MultiplyOccurringArgVars),
Pieces2 = [words("error: variables"), quote(BadVarsStr),
words("each occur more than once in the argument list."), nl]
),
Spec = simplest_spec($pred, severity_error, phase_parse_tree_to_hlds,
Context, Pieces1 ++ Pieces2),
!:Specs = [Spec | !.Specs]
;
MultiplyOccurringArgVars = [],
% Build the foreign_proc.
%
% Check that the purity of a predicate/function declaration agrees
% with the (promised) purity of the foreign proc. We do not perform
% this check if there is a promise_{pure,semipure} pragma for the
% predicate/function, since in that case they will differ anyway.
( if
( check_marker(Markers, marker_promised_pure)
; check_marker(Markers, marker_promised_semipure)
)
then
true
else
ForeignAttributePurity = get_purity(Attributes0),
( if ForeignAttributePurity = Purity then
true
else
PredSymName = qualified(PredModuleName, PredName),
user_arity_pred_form_arity(PredOrFunc, UserArity,
PredFormArity),
PFSymNameArity =
pred_pf_name_arity(PredOrFunc, PredSymName, UserArity),
purity_name(ForeignAttributePurity, ForeignAttributePurityStr),
purity_name(Purity, PurityStr),
Pieces = [words("Error: foreign clause for"),
unqual_pf_sym_name_user_arity(PFSymNameArity),
words("has purity"), words(ForeignAttributePurityStr),
words("but that"), p_or_f(PredOrFunc),
words("has been declared"), words(PurityStr),
suffix("."), nl],
Spec = simplest_spec($pred, severity_error,
phase_parse_tree_to_hlds, Context, Pieces),
!:Specs = [Spec | !.Specs]
)
),
(
Overridden = overridden_by_old_foreign_proc
;
Overridden = not_overridden_by_old_foreign_proc,
% Put the purity in the goal_info in case this foreign code is
% inlined.
goal_info_init_context_purity(Context, Purity, GoalInfo),
% XXX ARGVEC - the foreign_args field in the hlds_goal_expr type
% should also be a an proc_arg_vector rather than a list.
HeadVarList = proc_arg_vector_to_list(HeadVars),
make_foreign_args(HeadVarList, ArgInfos,
OrigArgTypes, ForeignArgs),
% Perform some renaming in any user annotated sharing information.
maybe_rename_user_annotated_sharing_information(Globals,
ArgVars, HeadVarList, OrigArgTypes, Attributes0, Attributes),
ExtraArgs = [],
MaybeTraceRuntimeCond = no,
GoalExpr = call_foreign_proc(Attributes, PredId, ProcId,
ForeignArgs, ExtraArgs, MaybeTraceRuntimeCond, PragmaImpl),
HldsGoal0 = hlds_goal(GoalExpr, GoalInfo),
% Foreign_procs cannot contain explicit variable type annotations.
init_vartypes(EmptyExplicitVarTypes),
rtti_varmaps_init(EmptyRttiVarmaps),
implicitly_quantify_clause_body_general_vs(ord_nl_maybe_lambda,
do_not_keep_quant_vars, HeadVarList, _Warnings,
HldsGoal0, HldsGoal, VarSet0, VarSet,
EmptyExplicitVarTypes, _, EmptyRttiVarmaps, _),
Clause = clause(selected_modes([ProcId]), HldsGoal,
impl_lang_foreign(NewLang), Context, []),
Clauses = [Clause | Clauses1],
set_clause_list(Clauses, ClausesRep),
HasForeignClauses = some_foreign_lang_clauses,
!:ClausesInfo = clauses_info(VarSet, ExplicitVarTypes,
VarTable, RttiVarMaps, TVarNameMap, HeadVars, ClausesRep,
ItemNumbers, HasForeignClauses, HadSyntaxError)
)
).
% Rename any user annotated structure sharing information from the
% variables (incl. type variables) in terms of which that information
% is expressed, to the formal variables in terms of which the clause
% is expressed.
%
:- pred maybe_rename_user_annotated_sharing_information(globals::in,
list(prog_var)::in, list(prog_var)::in, list(mer_type)::in,
foreign_proc_attributes::in, foreign_proc_attributes::out)
is det.
maybe_rename_user_annotated_sharing_information(Globals,
ActualHeadVars, FormalHeadVars, FormalTypes, !Attributes):-
globals.lookup_bool_option(Globals, structure_sharing_analysis,
SharingAnalysis),
(
SharingAnalysis = no
;
SharingAnalysis = yes,
rename_user_annotated_sharing(ActualHeadVars, FormalHeadVars,
FormalTypes, get_user_annotated_sharing(!.Attributes),
FormalUserSharing),
set_user_annotated_sharing(FormalUserSharing, !Attributes)
).
:- type overridden_by_old_foreign_proc
---> overridden_by_old_foreign_proc
; not_overridden_by_old_foreign_proc.
:- pred add_foreign_proc_update_existing_clauses(globals::in, pred_or_func::in,
module_name::in, string::in, pred_form_arity::in, prog_context::in,
compilation_target::in, foreign_language::in,
list(proc_id)::in, proc_id::in, overridden_by_old_foreign_proc::out,
list(clause)::in, list(clause)::out,
list(error_spec)::in, list(error_spec)::out) is det.
add_foreign_proc_update_existing_clauses(Globals, PredOrFunc,
PredModuleName, PredName, PredFormArity, NewContext, Target, NewLang,
AllProcIds, NewClauseProcId, Overridden, Clauses0, Clauses, !Specs) :-
(
Clauses0 = [],
Clauses = [],
Overridden = not_overridden_by_old_foreign_proc
;
Clauses0 = [FirstClause0 | LaterClauses0],
add_foreign_proc_update_existing_clauses(Globals, PredOrFunc,
PredModuleName, PredName, PredFormArity, NewContext, Target,
NewLang, AllProcIds, NewClauseProcId, LaterOverridden,
LaterClauses0, LaterClauses, !Specs),
FirstClause0 = clause(ApplProcIds0, Body, FirstClauseLang,
FirstClauseContext, StateVarWarnings),
(
FirstClauseLang = impl_lang_mercury,
(
ApplProcIds0 = all_modes,
ProcIds0 = AllProcIds
;
ApplProcIds0 = selected_modes(ProcIds0)
;
( ApplProcIds0 = unify_in_in_modes
; ApplProcIds0 = unify_non_in_in_modes
),
unexpected($pred, "unify mode for user defined predicate")
),
( if list.delete_first(ProcIds0, NewClauseProcId, ProcIds) then
(
ProcIds = [],
% This clause is totally overridden by the new
% foreign_proc, so delete it.
Clauses = LaterClauses
;
ProcIds = [_ | _],
% This clause is overridden by the new foreign_proc only
% in some modes, so mark it as being applicable only in the
% remaining modes.
FirstClause = clause(selected_modes(ProcIds), Body,
FirstClauseLang, FirstClauseContext, StateVarWarnings),
Clauses = [FirstClause | LaterClauses]
)
else
% This clause is not applicable to the mode of the new
% foreign_proc, so leave it alone.
Clauses = [FirstClause0 | LaterClauses]
),
% A Mercury clause can never take precedence over a foreign_proc.
Overridden = LaterOverridden
;
FirstClauseLang = impl_lang_foreign(OldLang),
(
ApplProcIds0 = all_modes,
unexpected($pred, "all_modes")
;
ApplProcIds0 = selected_modes(ProcIds0)
;
( ApplProcIds0 = unify_in_in_modes
; ApplProcIds0 = unify_non_in_in_modes
),
unexpected($pred, "unify modes")
),
( if list.delete_first(ProcIds0, NewClauseProcId, ProcIds) then
PreferNewForeignLang = prefer_foreign_language(Globals, Target,
OldLang, NewLang),
(
PreferNewForeignLang = yes,
(
ProcIds = [],
% The language of the new foreign_proc is preferred
% to the language of the old foreign_proc,
% so we should replace the old foreign_proc.
Clauses = LaterClauses,
Overridden = LaterOverridden
;
ProcIds = [_ | _],
% The language of the new foreign_proc is preferred
% to the language of the old foreign_proc,
% but the old foreign_proc is still applicable
% in some modes, so we keep it in those modes.
%
% XXX This should not happen.
FirstClause = clause(selected_modes(ProcIds), Body,
FirstClauseLang, FirstClauseContext,
StateVarWarnings),
Clauses = [FirstClause | LaterClauses],
Overridden = LaterOverridden
),
% Any later clause that overrides the new foreign_proc
% should have overridden this old foreign_proc as well.
expect(
unify(LaterOverridden,
not_overridden_by_old_foreign_proc),
$pred, "inconsistent old foreign_procs")
;
PreferNewForeignLang = no,
% We prefer the old foreign_proc to the new one,
% so keep the old one and tell our caller to ignore
% the new one.
Clauses = [FirstClause0 | LaterClauses],
Overridden = overridden_by_old_foreign_proc,
% However, if the old and the new foreign_procs are
% in the same language, then we emit an error message
% as well.
% XXX This won't detect multiple clauses in languages
% that are not supported by this backend, since we filter
% out foreign_procs in such languages way before we get
% here.
( if OldLang = NewLang then
PredSymName = qualified(PredModuleName, PredName),
PFSymNameArity = pf_sym_name_arity(PredOrFunc,
PredSymName, PredFormArity),
OldLangStr = foreign_language_string(OldLang),
PiecesA = [words("Error: duplicate"),
pragma_decl("foreign_proc"), words("declaration"),
words("for this mode of"),
unqual_pf_sym_name_pred_form_arity(PFSymNameArity),
words("in"), words(OldLangStr), suffix("."), nl],
PiecesB = [words("The first one was here."), nl],
MsgA = simplest_msg(NewContext, PiecesA),
MsgB = error_msg(yes(FirstClauseContext),
always_treat_as_first, 0, [always(PiecesB)]),
Spec = error_spec($pred, severity_error,
phase_parse_tree_to_hlds, [MsgA, MsgB]),
!:Specs = [Spec | !.Specs]
else
true
)
)
else
% This old foreign_proc is not overridden by the new one,
% so leave it alone.
Clauses = [FirstClause0 | LaterClauses],
Overridden = LaterOverridden
)
)
).
%----------------------------------------------------------------------------%
:- end_module hlds.make_hlds.add_foreign_proc.
%----------------------------------------------------------------------------%