mirror of
https://github.com/Mercury-Language/mercury.git
synced 2026-04-15 01:13:30 +00:00
The code that checks whether a bound inst wrapped around
a list of bound_functors matched the ground inst did several things
in a suboptimal fashion.
- It looked up the definition of the type constructor of the relevant type
(the type of the variable the inst is for) more than once. (This was
not easily visible because the lookups were in different predicates.)
This diff factors these out, not for the immesurably small speedup,
but to make possible the fixes for the next two issues.
- To simplify the "is there a bound_functor for each constructor in the type"
check, it sorted the constructors of the type by name and arity. (Lists of
bound_functors are always sorted by name and arity.) Given that most
modules contain more than one bound inst for any given type constructor,
any sorting after the first was unnecessarily repeated work. This diff
therefore extends the representation of du types, which until now has
include only a list of the data constructors in the type definition
in definition order, with a list of those exact same data constructors
in name/arity order.
- Even if a list of bound_functors lists all the constructors of a type,
the bound inst containing them is not equivalent to ground if the inst
of some argument of some bound_inst is not equivalent to ground.
This means that we need to know the actual argument of each constructor.
The du type definition lists argument types that refer to the type
constructor's type parameters; we need the instances of these argument types
that apply to type of the variable at hand, which usually binds concrete
types to those type parameters.
We used to apply the type-parameter-to-actual-type substitution to
each argument of each data constructor in the type before we compared
the resulting filled-in data constructor descriptions against the list of
bound_functors. However, in cases where the comparison fails, the
substitution applications to arguments beyond the point of failure
are all wasted work. This diff therefore applies the substitution
only when its result is about to be needed.
This diff leads to a speedup of about 3.5% on tools/speedtest,
and about 38% (yes, more than a third) when compiling options.m.
compiler/hlds_data.m:
Add the new field to the representation of du types.
Add a utility predicate that helps construct that field, since it is
now needed by two modules (add_type.m and equiv_type_hlds.m).
Delete two functions that were used only by det_check_switch.m,
which this diff moves to that module (in modified form).
compiler/inst_match.m:
Implement the first and third changes listed above, and take advantage
of the second.
The old call to all_du_ctor_arg_types, which this diff replaces,
effectively lied about the list of constructors it returned,
by simply not returning any constructors containing existentially
quantified types, on the grounds that they "were not handled yet".
We now fail explicitly when we find any such constructors.
Perform the check for one-to-one match between bound_functors and
constructors with less argument passing.
compiler/det_check_switch.m:
Move the code deleted from hlds_data.m here, and simplify it,
taking advantage of the new field in du types.
compiler/Mercury.options:
Specify --optimize-constructor-last-call for det_check_switch.m
to optimize the updated moved code.
compiler/add_foreign_enum.m:
compiler/add_special_pred.m:
compiler/add_type.m:
compiler/check_typeclass.m:
compiler/code_info.m:
compiler/dead_proc_elim.m:
compiler/direct_arg_in_out.m:
compiler/du_type_layout.m:
compiler/equiv_type_hlds.m:
compiler/hlds_out_type_table.m:
compiler/inst_check.m:
compiler/intermod.m:
compiler/intermod_decide.m:
compiler/lookup_switch_util.m:
compiler/ml_type_gen.m:
compiler/ml_unify_gen_test.m:
compiler/ml_unify_gen_util.m:
compiler/mlds.m:
compiler/post_term_analysis.m:
compiler/recompilation.usage.m:
compiler/resolve_unify_functor.m:
compiler/simplify_goal_ite.m:
compiler/table_gen.m:
compiler/tag_switch_util.m:
compiler/term_norm.m:
compiler/type_ctor_info.m:
compiler/type_util.m:
compiler/typecheck_coerce.m:
compiler/unify_proc.m:
compiler/unused_imports.m:
compiler/xml_documentation.m:
Conform to the changes above. This mostly means handling
the new field in du types (usually by ignoring it).
3135 lines
134 KiB
Mathematica
3135 lines
134 KiB
Mathematica
%---------------------------------------------------------------------------%
|
|
% vim: ft=mercury ts=4 sw=4 et
|
|
%---------------------------------------------------------------------------%
|
|
% Copyright (C) 1996-2001, 2003-2012 The University of Melbourne.
|
|
% Copyright (C) 2013-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: check_typeclass.m.
|
|
% Author: dgj, mark.
|
|
%
|
|
% This module checks conformance of instance declarations to the typeclass
|
|
% declaration. It does this in three phases, consisting of seven passes.
|
|
%
|
|
% Phase 1 checks typeclass declarations. This phase consists of pass 1.
|
|
% Phase 2 checks instance declarations. This phase consists of passes 2 to 5.
|
|
% Phase 3 checks the constructs that depend on typeclass and instance
|
|
% constraints, which are predicate, function and type declarations.
|
|
% This phase consists of passes 6 and 7.
|
|
%
|
|
% (1) In check_for_cyclic_classes/4, we check for cycles in the typeclass
|
|
% hierarchy. A cycle occurs if we can start from any given typeclass
|
|
% declaration and follow the superclass constraints on classes to reach the
|
|
% same class that we started from. Only the class_id needs to be repeated;
|
|
% it doesn't need to have the parameters. Note that we follow the constraints
|
|
% on class declarations only, not those on instance declarations. While doing
|
|
% this, we fill in the fundeps_ancestors field in the class table.
|
|
%
|
|
% (2) In check_instance_declaration_types/4, we check that each type
|
|
% in the instance declaration is either a type with no arguments,
|
|
% or a polymorphic type whose arguments are all type variables.
|
|
% We also check that all of the types in exported instance declarations are
|
|
% in scope here. XXX The latter part should really be done earlier, but with
|
|
% the current implementation this is the most convenient spot.
|
|
%
|
|
% This step also checks that types in instance declarations are not abstract
|
|
% exported equivalence types defined in this module. Unfortunately, there is
|
|
% no way to check at compile time that it is not an abstract exported
|
|
% equivalence type defined in some *other* module.
|
|
%
|
|
% (3) In generate_instance_method_procs/6, we generate and add to the HLDS
|
|
% a new procedure for every method of every instance of every class.
|
|
% The types, modes and determinisms of these procedures are taken from
|
|
% the method's signature in the class declaration, and the procedure body
|
|
% is generated from the implementation provided by the instance declaration.
|
|
% When later semantic analysis passes of the compiler check these new
|
|
% procedures, they will be checking the type, mode and determinism correctness
|
|
% of the instance.
|
|
%
|
|
% For example, given the declarations:
|
|
%
|
|
% :- typeclass c(T) where [
|
|
% pred m(T::in, T::out) is semidet
|
|
% ].
|
|
%
|
|
% :- instance c(int) where [
|
|
% pred(m/2) is my_m
|
|
% ].
|
|
%
|
|
% we check the correctness of my_m/2 as an implementation of m/2
|
|
% by generating the following new predicate:
|
|
%
|
|
% :- pred 'implementation of m/2'(int::in, int::out) is semidet.
|
|
%
|
|
% 'implementation of m/2'(HeadVar_1, HeadVar_2) :-
|
|
% my_m(HeadVar_1, HeadVar_2).
|
|
%
|
|
% By generating the new pred, we check the instance method for type, mode,
|
|
% determinism and uniqueness correctness since the generated pred is checked
|
|
% in each of those passes too. At this point, we add instance method pred/proc
|
|
% ids to the instance table of the HLDS. We also check that there are no
|
|
% missing, duplicate or bogus methods.
|
|
%
|
|
% In this pass we also call check_instance_for_superclass_conformance/9,
|
|
% which checks that the given instance declaration satisfies all its
|
|
% superclass constraints. To do this, that predicate attempts to perform
|
|
% context reduction on the typeclass constraints, using the instance
|
|
% constraints as assumptions. At this point, we fill in the super class proofs.
|
|
%
|
|
% (4) In check_for_missing_concrete_instances/4, we check that each
|
|
% abstract instance has a corresponding concrete instance.
|
|
%
|
|
% (5) In check_functional_dependencies/4, all visible instances are
|
|
% checked for coverage and mutual consistency with respect to any functional
|
|
% dependencies. This doesn't necessarily catch all cases of inconsistent
|
|
% instances, however, since in general that cannot be done until link time.
|
|
% We try to catch as many cases as possible here, though, since we can give
|
|
% better error messages.
|
|
%
|
|
% (6) In check_typeclass_constraints_on_preds/4, we check typeclass constraints
|
|
% on predicate and function declarations for ambiguity, taking into
|
|
% consideration the information provided by functional dependencies.
|
|
% We also call check_constraint_quant/5 to check that all type variables
|
|
% in constraints are universally quantified, or that they are all
|
|
% existentially quantified. We don't support constraints where
|
|
% some of the type variables are universal and some are existential.
|
|
%
|
|
% (7) In check_typeclass_constraints_on_data_ctors/4, we check typeclass
|
|
% constraints on existentially typed data constructors for ambiguity,
|
|
% taking into consideration the information provided by functional
|
|
% dependencies.
|
|
%
|
|
%---------------------------------------------------------------------------%
|
|
|
|
:- module check_hlds.check_typeclass.
|
|
:- interface.
|
|
|
|
:- import_module hlds.
|
|
:- import_module hlds.hlds_module.
|
|
:- import_module hlds.make_hlds.
|
|
:- import_module hlds.make_hlds.qual_info.
|
|
:- import_module parse_tree.
|
|
:- import_module parse_tree.error_spec.
|
|
:- import_module parse_tree.prog_data.
|
|
|
|
:- import_module io.
|
|
:- import_module list.
|
|
|
|
:- pred check_typeclasses(io.text_output_stream::in,
|
|
module_info::in, module_info::out, qual_info::in, qual_info::out,
|
|
list(error_spec)::out) is det.
|
|
|
|
% XXX Exported to add_class.m.
|
|
% This export should be temporary, since the code that needs it
|
|
% should also be moved to this module.
|
|
%
|
|
:- pred constraints_are_identical(
|
|
list(tvar)::in, tvarset::in, list(prog_constraint)::in,
|
|
list(tvar)::in, tvarset::in, list(prog_constraint)::in) is semidet.
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%---------------------------------------------------------------------------%
|
|
|
|
:- implementation.
|
|
|
|
:- import_module check_hlds.typeclasses.
|
|
:- import_module hlds.hlds_class.
|
|
:- import_module hlds.hlds_data.
|
|
:- import_module hlds.hlds_error_util.
|
|
:- import_module hlds.hlds_markers.
|
|
:- import_module hlds.hlds_pred.
|
|
:- import_module hlds.make_hlds.instance_method_clauses.
|
|
:- import_module hlds.passes_aux.
|
|
:- import_module hlds.pred_name.
|
|
:- import_module hlds.pred_table.
|
|
:- import_module hlds.status.
|
|
:- import_module hlds.type_util.
|
|
:- import_module libs.
|
|
:- import_module libs.file_util.
|
|
:- 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.error_util.
|
|
:- import_module parse_tree.parse_tree_out_term.
|
|
:- import_module parse_tree.parse_tree_out_type.
|
|
:- import_module parse_tree.prog_parse_tree.
|
|
:- import_module parse_tree.prog_type.
|
|
:- import_module parse_tree.prog_type_scan.
|
|
:- import_module parse_tree.prog_type_subst.
|
|
:- import_module parse_tree.prog_type_unify.
|
|
:- import_module parse_tree.prog_util.
|
|
|
|
:- import_module assoc_list.
|
|
:- import_module bool.
|
|
:- import_module cord.
|
|
:- import_module int.
|
|
:- import_module map.
|
|
:- import_module maybe.
|
|
:- import_module one_or_more.
|
|
:- import_module one_or_more_map.
|
|
:- import_module pair.
|
|
:- import_module require.
|
|
:- import_module set.
|
|
:- import_module string.
|
|
:- import_module term.
|
|
:- import_module varset.
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%---------------------------------------------------------------------------%
|
|
|
|
check_typeclasses(ProgressStream, !ModuleInfo, !QualInfo, !:Specs) :-
|
|
module_info_get_globals(!.ModuleInfo, Globals),
|
|
globals.lookup_bool_option(Globals, verbose, Verbose),
|
|
|
|
% Pass 1.
|
|
trace [io(!IO)] (
|
|
maybe_write_string(ProgressStream, Verbose,
|
|
"% Checking for cyclic classes...\n", !IO)
|
|
),
|
|
check_for_cyclic_classes(!ModuleInfo, CycleSpecs),
|
|
|
|
% Pass 2.
|
|
trace [io(!IO)] (
|
|
maybe_write_string(ProgressStream, Verbose,
|
|
"% Checking instance declaration types...\n", !IO)
|
|
),
|
|
check_instance_declaration_types(!.ModuleInfo,
|
|
[], InstanceDeclErrorSpecs, [], InstanceDeclWarnSpecs),
|
|
|
|
!:Specs = CycleSpecs ++ InstanceDeclErrorSpecs ++ InstanceDeclWarnSpecs,
|
|
|
|
% If we encounter any errors while checking that the types in an
|
|
% instance declaration are valid, then don't attempt the remaining passes.
|
|
% Pass 3 cannot be run since the name mangling scheme we use
|
|
% to generate the names of the method wrapper predicates may abort
|
|
% if the types in an instance are not valid, e.g. if an instance head
|
|
% contains a type variable that is not wrapped inside a functor.
|
|
% Most of the other passes also depend upon information that is
|
|
% calculated during pass 3.
|
|
%
|
|
% XXX It would be better to just remove the invalid instances at this
|
|
% point and then continue on with the valid instances.
|
|
|
|
(
|
|
InstanceDeclErrorSpecs = [],
|
|
% Pass 3.
|
|
trace [io(!IO)] (
|
|
maybe_write_string(ProgressStream, Verbose,
|
|
"% Checking typeclass instances...\n", !IO)
|
|
),
|
|
generate_instance_method_procs(!ModuleInfo, !QualInfo, !Specs),
|
|
|
|
% Pass 4.
|
|
trace [io(!IO)] (
|
|
maybe_write_string(ProgressStream, Verbose,
|
|
"% Checking for missing concrete instances...\n", !IO)
|
|
),
|
|
check_for_missing_concrete_instances(!.ModuleInfo, !Specs),
|
|
|
|
% Pass 5.
|
|
trace [io(!IO)] (
|
|
maybe_write_string(ProgressStream, Verbose,
|
|
"% Checking functional dependencies on instances...\n", !IO)
|
|
),
|
|
check_functional_dependencies(!.ModuleInfo, !Specs),
|
|
|
|
% Pass 6.
|
|
trace [io(!IO)] (
|
|
maybe_write_string(ProgressStream, Verbose,
|
|
"% Checking typeclass constraints on predicates...\n", !IO)
|
|
),
|
|
check_typeclass_constraints_on_preds(ProgressStream, !.ModuleInfo,
|
|
!Specs),
|
|
|
|
% Pass 7.
|
|
trace [io(!IO)] (
|
|
maybe_write_string(ProgressStream, Verbose,
|
|
"% Checking typeclass constraints on data constructors...\n",
|
|
!IO)
|
|
),
|
|
check_typeclass_constraints_on_data_ctors(!.ModuleInfo, !Specs)
|
|
;
|
|
InstanceDeclErrorSpecs = [_ | _]
|
|
).
|
|
|
|
%---------------------------------------------------------------------------%
|
|
% Pass 1.
|
|
%---------------------------------------------------------------------------%
|
|
|
|
% Check for cyclic classes in the class table by traversing the class
|
|
% hierarchy for each class. While we are doing this, calculate the set
|
|
% of ancestors with functional dependencies for each class, and enter
|
|
% this information in the class table.
|
|
%
|
|
:- pred check_for_cyclic_classes(module_info::in, module_info::out,
|
|
list(error_spec)::out) is det.
|
|
|
|
check_for_cyclic_classes(!ModuleInfo, Specs) :-
|
|
module_info_get_class_table(!.ModuleInfo, ClassTable0),
|
|
ClassIds = map.keys(ClassTable0),
|
|
list.foldl3(find_class_cycles(class_path([])), ClassIds,
|
|
ClassTable0, ClassTable, set.init, _, [], Cycles),
|
|
list.foldl(report_cyclic_classes(ClassTable), Cycles, [], Specs),
|
|
module_info_set_class_table(ClassTable, !ModuleInfo).
|
|
|
|
:- type class_path
|
|
---> class_path(list(class_id)).
|
|
|
|
% find_class_cycles(Path, ClassId, !ClassTable, !Visited, !Cycles)
|
|
%
|
|
% Perform a depth first traversal of the class hierarchy, starting
|
|
% from ClassId. Path contains a list of nodes joining the current node
|
|
% to the root. When we reach a node that has already been visited,
|
|
% check whether there is a cycle in the Path.
|
|
%
|
|
:- pred find_class_cycles(class_path::in, class_id::in,
|
|
class_table::in, class_table::out, set(class_id)::in, set(class_id)::out,
|
|
list(class_path)::in, list(class_path)::out) is det.
|
|
|
|
find_class_cycles(Path, ClassId, !ClassTable, !Visited, !Cycles) :-
|
|
find_class_cycles_2(Path, ClassId, _, _, !ClassTable, !Visited, !Cycles).
|
|
|
|
% As above, but also return this class's parameters and the list
|
|
% of its ancestors with fundeps. This functionality is needed by
|
|
% find_class_cycles_3, with which this predicate is mutually recursive.
|
|
%
|
|
:- pred find_class_cycles_2(class_path::in, class_id::in,
|
|
list(tvar)::out, list(prog_constraint)::out,
|
|
class_table::in, class_table::out, set(class_id)::in, set(class_id)::out,
|
|
list(class_path)::in, list(class_path)::out) is det.
|
|
|
|
find_class_cycles_2(Path0, ClassId, ClassParamTVars, FunDepAncestors,
|
|
!ClassTable, !Visited, !Cycles) :-
|
|
map.lookup(!.ClassTable, ClassId, ClassDefn0),
|
|
ClassParamTVars = ClassDefn0 ^ classdefn_vars,
|
|
Kinds = ClassDefn0 ^ classdefn_kinds,
|
|
( if set.member(ClassId, !.Visited) then
|
|
( if
|
|
find_class_cycle(ClassId, Path0, class_path([ClassId]), Cycle)
|
|
then
|
|
!:Cycles = [Cycle | !.Cycles]
|
|
else
|
|
true
|
|
),
|
|
FunDepAncestors = ClassDefn0 ^ classdefn_fundep_ancestors
|
|
else
|
|
set.insert(ClassId, !Visited),
|
|
|
|
% If this class has fundeps, then include it in its own list of
|
|
% "ancestors with fundeps".
|
|
FunDeps = ClassDefn0 ^ classdefn_fundeps,
|
|
(
|
|
FunDeps = [],
|
|
FunDepAncestors0 = []
|
|
;
|
|
FunDeps = [_ | _],
|
|
ClassId = class_id(ClassName, _),
|
|
prog_type.var_list_to_type_list(Kinds, ClassParamTVars, ArgTypes),
|
|
FunDepAncestors0 = [constraint(ClassName, ArgTypes)]
|
|
),
|
|
Superclasses = ClassDefn0 ^ classdefn_supers,
|
|
Path0 = class_path(PathClassIds),
|
|
Path1 = class_path([ClassId | PathClassIds]),
|
|
list.foldl4(find_class_cycles_3(Path1), Superclasses,
|
|
!ClassTable, !Visited, !Cycles, FunDepAncestors0, FunDepAncestors),
|
|
ClassDefn = ClassDefn0 ^ classdefn_fundep_ancestors := FunDepAncestors,
|
|
map.det_update(ClassId, ClassDefn, !ClassTable)
|
|
).
|
|
|
|
% As we go, accumulate the ancestors from all the superclasses,
|
|
% with the class parameters bound to the corresponding arguments.
|
|
% Note that we don't need to merge varsets because typeclass
|
|
% parameters are guaranteed to be distinct variables.
|
|
%
|
|
:- pred find_class_cycles_3(class_path::in, prog_constraint::in,
|
|
class_table::in, class_table::out,
|
|
set(class_id)::in, set(class_id)::out,
|
|
list(class_path)::in, list(class_path)::out,
|
|
list(prog_constraint)::in, list(prog_constraint)::out) is det.
|
|
|
|
find_class_cycles_3(Path, Constraint, !ClassTable, !Visited, !Cycles,
|
|
!FunDepAncestors) :-
|
|
Constraint = constraint(ClassName, ArgTypes),
|
|
list.length(ArgTypes, Arity),
|
|
ClassId = class_id(ClassName, Arity),
|
|
find_class_cycles_2(Path, ClassId, ClassParamTVars, NewFunDepAncestors0,
|
|
!ClassTable, !Visited, !Cycles),
|
|
map.from_corresponding_lists(ClassParamTVars, ArgTypes, Binding),
|
|
apply_subst_to_prog_constraints(Binding,
|
|
NewFunDepAncestors0, NewFunDepAncestors),
|
|
!:FunDepAncestors = NewFunDepAncestors ++ !.FunDepAncestors.
|
|
|
|
% find_class_cycle(ClassId, PathRemaining0, PathSoFar, Cycle):
|
|
%
|
|
% Check if ClassId is present in PathRemaining0, and if so, then make
|
|
% a cycle out of the front part of the path up to the point where
|
|
% the ClassId is found. The part of the path checked so far is
|
|
% accumulated in PathSoFar.
|
|
%
|
|
:- pred find_class_cycle(class_id::in, class_path::in, class_path::in,
|
|
class_path::out) is semidet.
|
|
|
|
find_class_cycle(ClassId, PathRemaining0, PathSoFar0, Cycle) :-
|
|
PathRemaining0 = class_path([Head | Tail]),
|
|
PathSoFar0 = class_path(PathSoFarClassIds0),
|
|
PathSoFar1 = class_path([Head | PathSoFarClassIds0]),
|
|
( if ClassId = Head then
|
|
Cycle = PathSoFar1
|
|
else
|
|
PathRemaining1 = class_path(Tail),
|
|
find_class_cycle(ClassId, PathRemaining1, PathSoFar1, Cycle)
|
|
).
|
|
|
|
%---------------------------------------------------------------------------%
|
|
% Pass 2.
|
|
%---------------------------------------------------------------------------%
|
|
|
|
% In check_instance_declaration_types/4, we check that each type
|
|
% in the instance declaration must be either a type with no arguments,
|
|
% or a polymorphic type whose arguments are all type variables.
|
|
%
|
|
:- pred check_instance_declaration_types(module_info::in,
|
|
list(error_spec)::in, list(error_spec)::out,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_instance_declaration_types(ModuleInfo, !ErrorSpecs, !WarnSpecs) :-
|
|
module_info_get_instance_table(ModuleInfo, InstanceTable),
|
|
map.foldl2(check_instance_declaration_types_for_class(ModuleInfo),
|
|
InstanceTable, !ErrorSpecs, !WarnSpecs).
|
|
|
|
:- pred check_instance_declaration_types_for_class(module_info::in,
|
|
class_id::in, list(hlds_instance_defn)::in,
|
|
list(error_spec)::in, list(error_spec)::out,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_instance_declaration_types_for_class(ModuleInfo, ClassId,
|
|
InstanceDefns, !ErrorSpecs, !WarnSpecs) :-
|
|
list.foldl2(
|
|
is_instance_type_vector_valid(ModuleInfo, ClassId),
|
|
InstanceDefns, !ErrorSpecs, map.init, TooPrivateMap),
|
|
map.foldl(report_any_too_private_instance(ModuleInfo, ClassId),
|
|
TooPrivateMap, !WarnSpecs).
|
|
|
|
% This type contains the information we need from a hlds_instance_defn
|
|
% to implement the --warn-too-private-instances option for that instance.
|
|
% Note that we need this info from *all* instance declarations for given
|
|
% class_id/arg_vector combo. There will often be two such declarations:
|
|
% an abstract instance in the interface section, and a concrete instance
|
|
% in the implementation section.
|
|
%
|
|
% We need to know about all instance declarations because
|
|
% when we record a concrete instance declaration in the instance table,
|
|
% its status will NEVER say that the instance is exported, even if it is.
|
|
% This is because
|
|
%
|
|
% - the item_mercury_status of the concrete instance, which will occur
|
|
% in the implementation, will say that this instance declaration
|
|
% is not exported, and
|
|
%
|
|
% - we do NOT update this status even if an abstract version of the
|
|
% same instance declaration exists in the interface section.
|
|
%
|
|
% XXX We *should* update the status, and delete the abstract version.
|
|
%
|
|
:- type instance_too_private_info
|
|
---> instance_too_private_info(
|
|
instance_body,
|
|
instance_status,
|
|
list(prog_constraint),
|
|
prog_context
|
|
).
|
|
|
|
:- type instance_too_private_map ==
|
|
one_or_more_map(list(mer_type), instance_too_private_info).
|
|
|
|
:- pred is_instance_type_vector_valid(module_info::in,
|
|
class_id::in, hlds_instance_defn::in,
|
|
list(error_spec)::in, list(error_spec)::out,
|
|
instance_too_private_map::in, instance_too_private_map::out) is det.
|
|
|
|
is_instance_type_vector_valid(ModuleInfo, ClassId, InstanceDefn,
|
|
!ErrorSpecs, !TooPrivateMap) :-
|
|
OriginalTypes = InstanceDefn ^ instdefn_orig_types,
|
|
InstanceErrorSpecs0 = [],
|
|
list.foldl2(is_orig_type_non_eqv_type(ModuleInfo, ClassId, InstanceDefn),
|
|
OriginalTypes, 1, _, InstanceErrorSpecs0, InstanceErrorSpecs1),
|
|
Types = InstanceDefn ^ instdefn_types,
|
|
list.foldl2(is_valid_instance_type(ModuleInfo, ClassId, InstanceDefn),
|
|
Types, 1, _, InstanceErrorSpecs1, InstanceErrorSpecs),
|
|
(
|
|
InstanceErrorSpecs = [_ | _]
|
|
;
|
|
InstanceErrorSpecs = [],
|
|
InstanceBody = InstanceDefn ^ instdefn_body,
|
|
InstanceStatus = InstanceDefn ^ instdefn_status,
|
|
InstanceConstraints = InstanceDefn ^ instdefn_constraints,
|
|
InstanceContext = InstanceDefn ^ instdefn_context,
|
|
TooPrivateInfo = instance_too_private_info(InstanceBody,
|
|
InstanceStatus, InstanceConstraints, InstanceContext),
|
|
one_or_more_map.add(OriginalTypes, TooPrivateInfo, !TooPrivateMap)
|
|
),
|
|
!:ErrorSpecs = !.ErrorSpecs ++ InstanceErrorSpecs.
|
|
|
|
% The only check we carry out on the original form of instance types
|
|
% is that they are not equivalence types. All other checks we carry out
|
|
% are done on the equivalence-expanded version of these types, by
|
|
% is_valid_instance_type below.
|
|
%
|
|
:- pred is_orig_type_non_eqv_type(module_info::in,
|
|
class_id::in, hlds_instance_defn::in, mer_type::in,
|
|
int::in, int::out, list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
is_orig_type_non_eqv_type(ModuleInfo, ClassId, InstanceDefn, Type,
|
|
ArgNum, ArgNum+1, !Specs) :-
|
|
(
|
|
Type = defined_type(_TypeCtorName, _, _),
|
|
( if type_to_type_defn(ModuleInfo, Type, TypeDefn) then
|
|
get_type_defn_body(TypeDefn, TypeBody),
|
|
(
|
|
TypeBody = hlds_eqv_type(_),
|
|
get_type_defn_status(TypeDefn, TypeDefnStatus),
|
|
( if
|
|
TypeDefnStatus = type_status(status_abstract_exported),
|
|
% If the instance definition is itself abstract exported,
|
|
% we want to generate only one error message, instead of
|
|
% two error messages, one for the abstract and one for the
|
|
% concrete instance definition.
|
|
InstanceDefn ^ instdefn_body = instance_body_concrete(_)
|
|
then
|
|
report_eqv_type_in_abstract_exported_instance(ClassId,
|
|
InstanceDefn, ArgNum, Type, !Specs)
|
|
else
|
|
true
|
|
)
|
|
;
|
|
( TypeBody = hlds_du_type(_)
|
|
; TypeBody = hlds_foreign_type(_)
|
|
; TypeBody = hlds_solver_type(_)
|
|
; TypeBody = hlds_abstract_type(_)
|
|
)
|
|
)
|
|
else
|
|
% The type is either a builtin type or a type variable.
|
|
true
|
|
)
|
|
;
|
|
( Type = builtin_type(_)
|
|
; Type = higher_order_type(_, _, _, _)
|
|
; Type = apply_n_type(_, _, _)
|
|
; Type = type_variable(_, _)
|
|
; Type = tuple_type(_, _)
|
|
)
|
|
;
|
|
Type = kinded_type(_, _),
|
|
unexpected($pred, "kinded_type")
|
|
).
|
|
|
|
% Each of these types in the instance declaration must be
|
|
% a type constructor applied to zero or more type variables.
|
|
% We used to require those type variables to be distinct,
|
|
% but we lifted that requirement some time ago.
|
|
%
|
|
:- pred is_valid_instance_type(module_info::in,
|
|
class_id::in, hlds_instance_defn::in, mer_type::in,
|
|
int::in, int::out, list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
is_valid_instance_type(ModuleInfo, ClassId, InstanceDefn, Type,
|
|
ArgNum, ArgNum+1, !Specs) :-
|
|
(
|
|
Type = builtin_type(_)
|
|
;
|
|
(
|
|
Type = higher_order_type(_, _, _, _),
|
|
KindPieces = [words("is a")] ++
|
|
color_as_incorrect([words("higher order type;")])
|
|
;
|
|
Type = apply_n_type(_, _, _),
|
|
KindPieces = [words("is an")] ++
|
|
color_as_incorrect([words("apply/N type;")])
|
|
;
|
|
Type = type_variable(_, _),
|
|
KindPieces = [words("is a")] ++
|
|
color_as_incorrect([words("type variable;")])
|
|
),
|
|
TVarSet = InstanceDefn ^ instdefn_tvarset,
|
|
TypeStr = mercury_type_to_string(TVarSet, print_name_only, Type),
|
|
EndPieces =
|
|
[words("the"), nth_fixed(ArgNum), words("instance type")] ++
|
|
color_as_subject([quote(TypeStr)]) ++
|
|
KindPieces ++
|
|
[words("it should be a")] ++
|
|
color_as_correct([words("type constructor"),
|
|
words("applied to zero or more type variables.")]) ++
|
|
[nl],
|
|
report_bad_type_in_instance(ClassId, InstanceDefn, EndPieces,
|
|
badly_formed, !Specs)
|
|
;
|
|
( Type = defined_type(TypeCtorSymName, ArgTypes, _)
|
|
; Type = tuple_type(ArgTypes, _), TypeCtorSymName = unqualified("{}")
|
|
),
|
|
find_non_type_variables(ArgTypes, 1, NonTVarArgs),
|
|
(
|
|
NonTVarArgs = []
|
|
;
|
|
NonTVarArgs = [HeadNonTVarArg | TailNonTVarArgs],
|
|
TypeCtor = type_ctor(TypeCtorSymName, list.length(ArgTypes)),
|
|
report_badly_formed_type_in_instance(ClassId, InstanceDefn,
|
|
TypeCtor, ArgNum, HeadNonTVarArg, TailNonTVarArgs, !Specs)
|
|
),
|
|
% For defined types, report an error if the type_ctor is defined
|
|
% to be equivalence type that for some reason was not expanded out
|
|
% by the equiv_type pass.
|
|
% For tuple types, there is no check to be made, but there is no
|
|
% point in trying avoid this call for tuple types, since the switch
|
|
% inside is_orig_type_non_eqv_type will do that just as fast.
|
|
is_orig_type_non_eqv_type(ModuleInfo, ClassId, InstanceDefn, Type,
|
|
ArgNum, _, !Specs)
|
|
;
|
|
Type = kinded_type(_, _),
|
|
unexpected($pred, "kinded_type")
|
|
).
|
|
|
|
:- pred find_non_type_variables(list(mer_type)::in, int::in,
|
|
assoc_list(int, mer_type)::out) is det.
|
|
|
|
find_non_type_variables([], _, []).
|
|
find_non_type_variables([ArgType | ArgTypes], ArgNum, NonTVarArgs) :-
|
|
find_non_type_variables(ArgTypes, ArgNum + 1, TailNonTVarArgs),
|
|
(
|
|
ArgType = type_variable(_, _),
|
|
NonTVarArgs = TailNonTVarArgs
|
|
;
|
|
( ArgType = defined_type(_, _, _)
|
|
; ArgType = builtin_type(_)
|
|
; ArgType = higher_order_type(_, _, _, _)
|
|
; ArgType = tuple_type(_, _)
|
|
; ArgType = apply_n_type(_, _, _)
|
|
; ArgType = kinded_type(_, _)
|
|
),
|
|
NonTVarArgs = [ArgNum - ArgType | TailNonTVarArgs]
|
|
).
|
|
|
|
%---------------------%
|
|
|
|
:- pred report_any_too_private_instance(module_info::in, class_id::in,
|
|
list(mer_type)::in, one_or_more(instance_too_private_info)::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_any_too_private_instance(ModuleInfo, ClassId,
|
|
OrigTypes, OoMTooPrivateInfos, !WarnSpecs) :-
|
|
module_info_get_class_table(ModuleInfo, ClassTable),
|
|
map.lookup(ClassTable, ClassId, ClassDefn),
|
|
ClassStatus = ClassDefn ^ classdefn_status,
|
|
OoMTooPrivateInfos = one_or_more(HeadTooPrivateInfo, TailTooPrivateInfos),
|
|
is_any_instance_defn_exported(HeadTooPrivateInfo, TailTooPrivateInfos,
|
|
InstanceIsExported, no, MaybeLocalConcretePair),
|
|
( if
|
|
InstanceIsExported = no,
|
|
% If the typeclass is private, it is reasonable to keep its instances
|
|
% private, regardless of what type_ctors they refer to.
|
|
typeclass_status_is_exported(ClassStatus) = yes,
|
|
% If the instance is either defined in another module or is only
|
|
% abstract, then do not even try to generate warnings about it.
|
|
MaybeLocalConcretePair = yes(LocalConcretePair)
|
|
then
|
|
LocalConcretePair = {InstanceConstraints, LocalConcreteContext},
|
|
list.foldl2(acc_classes_type_ctors_in_constraint, InstanceConstraints,
|
|
set.init, ConstraintClassIds, set.init, TypeCtorsInTypes1),
|
|
list.foldl(acc_type_ctors_in_type, OrigTypes,
|
|
TypeCtorsInTypes1, TypeCtorsInTypes),
|
|
set.filter(type_ctor_is_private(ModuleInfo), TypeCtorsInTypes,
|
|
PrivateTypeCtorsInTypes, NonPrivateTypeCtorsInTypes),
|
|
set.filter(class_is_private(ModuleInfo), ConstraintClassIds,
|
|
PrivateClassIds, _NonPrivateClassIds),
|
|
( if
|
|
set.is_empty(PrivateTypeCtorsInTypes),
|
|
set.is_non_empty(NonPrivateTypeCtorsInTypes),
|
|
set.is_empty(PrivateClassIds)
|
|
then
|
|
report_unnecessarily_private_instance(ClassId,
|
|
ConstraintClassIds, TypeCtorsInTypes, LocalConcreteContext,
|
|
!WarnSpecs)
|
|
else
|
|
true
|
|
)
|
|
else
|
|
true
|
|
).
|
|
|
|
:- pred is_any_instance_defn_exported(instance_too_private_info::in,
|
|
list(instance_too_private_info)::in, bool::out,
|
|
maybe({list(prog_constraint), prog_context})::in,
|
|
maybe({list(prog_constraint), prog_context})::out) is det.
|
|
|
|
is_any_instance_defn_exported(HeadTooPrivateInfo, TailTooPrivateInfos,
|
|
IsExported, !MaybeLocalConcretePair) :-
|
|
HeadTooPrivateInfo = instance_too_private_info(HeadBody, HeadStatus,
|
|
HeadConstraints, HeadContext),
|
|
( if
|
|
instance_status_defined_in_this_module(HeadStatus) = yes,
|
|
HeadBody = instance_body_concrete(_)
|
|
then
|
|
!:MaybeLocalConcretePair = yes({HeadConstraints, HeadContext})
|
|
else
|
|
true
|
|
),
|
|
(
|
|
TailTooPrivateInfos = [],
|
|
TailIsExported = no
|
|
;
|
|
TailTooPrivateInfos =
|
|
[HeadTailTooPrivateInfo | TailTailTooPrivateInfos],
|
|
is_any_instance_defn_exported(HeadTailTooPrivateInfo,
|
|
TailTailTooPrivateInfos, TailIsExported,
|
|
!MaybeLocalConcretePair)
|
|
),
|
|
HeadExported = instance_status_is_exported(HeadStatus),
|
|
(
|
|
HeadExported = yes,
|
|
IsExported = yes
|
|
;
|
|
HeadExported = no,
|
|
IsExported = TailIsExported
|
|
).
|
|
|
|
:- pred acc_classes_type_ctors_in_constraint(prog_constraint::in,
|
|
set(class_id)::in, set(class_id)::out,
|
|
set(type_ctor)::in, set(type_ctor)::out) is det.
|
|
|
|
acc_classes_type_ctors_in_constraint(Constraint,
|
|
!ClassNameSet, !TypeCtorSet) :-
|
|
Constraint = constraint(ClassName, ArgTypes),
|
|
list.length(ArgTypes, Arity),
|
|
set.insert(class_id(ClassName, Arity), !ClassNameSet),
|
|
list.foldl(acc_type_ctors_in_type, ArgTypes, !TypeCtorSet).
|
|
|
|
:- pred acc_type_ctors_in_type(mer_type::in,
|
|
set(type_ctor)::in, set(type_ctor)::out) is det.
|
|
|
|
acc_type_ctors_in_type(Type, !TypeCtorSet) :-
|
|
(
|
|
( Type = builtin_type(_)
|
|
; Type = type_variable(_, _)
|
|
)
|
|
;
|
|
Type = defined_type(SymName, ArgTypes, _Kind),
|
|
list.length(ArgTypes, Arity),
|
|
TypeCtor = type_ctor(SymName, Arity),
|
|
set.insert(TypeCtor, !TypeCtorSet),
|
|
list.foldl(acc_type_ctors_in_type, ArgTypes, !TypeCtorSet)
|
|
;
|
|
( Type = tuple_type(ArgTypes, _Kind)
|
|
; Type = higher_order_type(_, ArgTypes, _, _)
|
|
; Type = apply_n_type(_, ArgTypes, _)
|
|
),
|
|
list.foldl(acc_type_ctors_in_type, ArgTypes, !TypeCtorSet)
|
|
;
|
|
Type = kinded_type(SubType, _Kind),
|
|
acc_type_ctors_in_type(SubType, !TypeCtorSet)
|
|
).
|
|
|
|
:- pred type_ctor_is_private(module_info::in, type_ctor::in) is semidet.
|
|
|
|
type_ctor_is_private(ModuleInfo, TypeCtor) :-
|
|
module_info_get_type_table(ModuleInfo, TypeTable),
|
|
lookup_type_ctor_defn(TypeTable, TypeCtor, TypeDefn),
|
|
get_type_defn_status(TypeDefn, TypeStatus),
|
|
DefinedHere = type_status_defined_in_this_module(TypeStatus),
|
|
% A type_ctor is private if either ...
|
|
(
|
|
DefinedHere = yes,
|
|
% ... it is defined in this module, and not exported, ...
|
|
type_status_is_exported(TypeStatus) = no
|
|
;
|
|
DefinedHere = no,
|
|
% ... or it is defined in a submodule nested inside this module,
|
|
% and that submodule is not visible from outside this module.
|
|
TypeCtor = type_ctor(TypeCtorSymName, _Arity),
|
|
TypeCtorSymName = qualified(TypeCtorModuleName, _),
|
|
module_name_is_private_submodule(ModuleInfo, TypeCtorModuleName)
|
|
).
|
|
|
|
:- pred class_is_private(module_info::in, class_id::in) is semidet.
|
|
|
|
class_is_private(ModuleInfo, ClassId) :-
|
|
module_info_get_class_table(ModuleInfo, ClassTable),
|
|
map.lookup(ClassTable, ClassId, ClassDefn),
|
|
ClassStatus = ClassDefn ^ classdefn_status,
|
|
DefinedHere = typeclass_status_defined_in_this_module(ClassStatus),
|
|
% A class is private if either ...
|
|
(
|
|
DefinedHere = yes,
|
|
% ... it is defined in this module, and not exported, ...
|
|
typeclass_status_is_exported(ClassStatus) = no
|
|
;
|
|
DefinedHere = no,
|
|
% ... or it is defined in a submodule nested inside this module,
|
|
% and that submodule is not visible from outside this module.
|
|
ClassId = class_id(ClassSymName, _Arity),
|
|
ClassSymName = qualified(ClassModuleName, _),
|
|
module_name_is_private_submodule(ModuleInfo, ClassModuleName)
|
|
).
|
|
|
|
% module_name_is_private_submodule(ModuleInfo, QueryModuleName):
|
|
%
|
|
% Is QueryModuleName the name of a private submodule of the module
|
|
% represented by ModuleInfo?
|
|
%
|
|
:- pred module_name_is_private_submodule(module_info::in, module_name::in)
|
|
is semidet.
|
|
|
|
module_name_is_private_submodule(ModuleInfo, QueryModuleName) :-
|
|
module_info_get_name(ModuleInfo, ModuleName),
|
|
% XXX Should the code that tests whether QueryModuleName is a submodule
|
|
% of ModuleName be in mdbcomp/sym_name.m? is_same_module_or_submodule
|
|
% is already there, but it does not return LeftOverComponents.
|
|
ModuleNameComponents = sym_name_to_list(ModuleName),
|
|
QueryModuleNameComponents = sym_name_to_list(QueryModuleName),
|
|
remove_prefix(ModuleNameComponents, QueryModuleNameComponents,
|
|
LeftOverComponents),
|
|
LeftOverComponents = [HeadSubComponent | _TailSubComponents],
|
|
HeadSubModuleName = qualified(ModuleName, HeadSubComponent),
|
|
% What we want to do here is
|
|
% - to fail if any component in [HeadSubComponent | TailSubComponents]
|
|
% is in the interface section of ita parent, and conversely
|
|
% - to succeed only if all of them are in the implementation section.
|
|
% Unfortunately, while we can use IncludeMap to tell whether
|
|
% a *direct* submodule of ModuleName is visible outside ModuleName,
|
|
% we do not have the include_module_maps of ModuleName's submodules,
|
|
% and so we cannot tell whether an *indirect* submodule of ModuleName
|
|
% that is included in the interface section of ModuleName,
|
|
% is visible outside ModuleName. We have to guess. We always guess
|
|
% that such indirect submodules are visible outside ModuleName,
|
|
% because we prefer to miss generating a valid warning in this
|
|
% rare circumstance over generating an invalid warning.
|
|
module_info_get_include_module_map(ModuleInfo, IncludeMap),
|
|
map.lookup(IncludeMap, HeadSubModuleName, HeadSubIncludeInfo),
|
|
HeadSubIncludeInfo = include_module_info(HeadSubSection, _),
|
|
HeadSubSection = ms_implementation.
|
|
|
|
:- pred report_unnecessarily_private_instance(class_id::in,
|
|
set(class_id)::in, set(type_ctor)::in,
|
|
prog_context::in, list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_unnecessarily_private_instance(ClassId, ConstraintClassIdSet,
|
|
TypeCtorsSet, Context, !Specs) :-
|
|
set.to_sorted_list(ConstraintClassIdSet, ClassIds),
|
|
set.to_sorted_list(TypeCtorsSet, TypeCtors),
|
|
ClassIdPieces = list.map((func(CI) = qual_class_id(CI)), ClassIds),
|
|
TypeCtorPieces = list.map((func(TC) = qual_type_ctor(TC)), TypeCtors),
|
|
(
|
|
ClassIdPieces = [],
|
|
ClassIdsPieces = []
|
|
;
|
|
ClassIdPieces = [ClassIdPiece],
|
|
ClassIdsPieces =
|
|
[(if TypeCtorPieces = [] then words("and") else suffix(",")),
|
|
words("the typeclass listed in the instance constraints"),
|
|
words("of this instance declaration for it, namely"),
|
|
ClassIdPiece, suffix(","), words("and its argument types,")]
|
|
;
|
|
ClassIdPieces = [_, _ | _],
|
|
ClassIdsPieces =
|
|
[(if TypeCtorPieces = [] then words("and") else suffix(",")),
|
|
words("the typeclasses listed in the instance constraints"),
|
|
words("of this instance declaration for it, namely")] ++
|
|
piece_list_to_pieces("and", ClassIdPieces) ++ [suffix(","),
|
|
words("and their argument types")]
|
|
),
|
|
(
|
|
TypeCtorPieces = [],
|
|
TypeCtorsPieces = [words("is")]
|
|
;
|
|
TypeCtorPieces = [TypeCtorPiece],
|
|
TypeCtorsPieces =
|
|
[words("and the type constructor in the argument vector"),
|
|
words("of this instance declaration for it, namely"),
|
|
TypeCtorPiece, suffix(",")]
|
|
;
|
|
TypeCtorPieces = [_, _ | _],
|
|
TypeCtorsPieces =
|
|
[words("and all the type constructors in the argument vector"),
|
|
words("of this instance declaration for it, namely")] ++
|
|
piece_list_to_pieces("and", TypeCtorPieces) ++ [suffix(",")]
|
|
),
|
|
( if ClassIdPieces = [], TypeCtorPieces = [] then
|
|
IsArePiece = words("is")
|
|
else
|
|
IsArePiece = words("are all")
|
|
),
|
|
Pieces = [words("Warning: the type class"), qual_class_id(ClassId)] ++
|
|
ClassIdsPieces ++ TypeCtorsPieces ++ [IsArePiece,
|
|
words("visible outside this module, which means that"),
|
|
words("this instance can be relevant outside this module.")] ++
|
|
color_as_subject([words("Keeping it private")]) ++
|
|
[words("to this module is therefore")] ++
|
|
color_as_incorrect([words("likely to be a mistake.")]) ++ [nl],
|
|
Spec = spec($pred, severity_warning(warn_too_private_instances),
|
|
phase_pt2h, Context, Pieces),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
%---------------------------------------------------------------------------%
|
|
% Pass 3.
|
|
%---------------------------------------------------------------------------%
|
|
|
|
:- pred generate_instance_method_procs(module_info::in, module_info::out,
|
|
qual_info::in, qual_info::out,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
generate_instance_method_procs(!ModuleInfo, !QualInfo, !Specs) :-
|
|
module_info_get_class_table(!.ModuleInfo, ClassTable),
|
|
module_info_get_instance_table(!.ModuleInfo, InstanceTable0),
|
|
map.map_foldl3(generate_instance_method_procs_for_class(ClassTable),
|
|
InstanceTable0, InstanceTable, !ModuleInfo, !QualInfo, [], NewSpecs),
|
|
module_info_get_globals(!.ModuleInfo, Globals),
|
|
Errors = contains_errors(Globals, NewSpecs),
|
|
(
|
|
Errors = no,
|
|
module_info_set_instance_table(InstanceTable, !ModuleInfo)
|
|
;
|
|
Errors = yes
|
|
),
|
|
!:Specs = NewSpecs ++ !.Specs.
|
|
|
|
% Generate procedures for all methods in all the instances
|
|
% of the given class.
|
|
%
|
|
:- pred generate_instance_method_procs_for_class(class_table::in, class_id::in,
|
|
list(hlds_instance_defn)::in, list(hlds_instance_defn)::out,
|
|
module_info::in, module_info::out, qual_info::in, qual_info::out,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
generate_instance_method_procs_for_class(ClassTable, ClassId,
|
|
InstanceDefns0, InstanceDefns, !ModuleInfo, !QualInfo, !Specs) :-
|
|
map.lookup(ClassTable, ClassId, ClassDefn),
|
|
ClassDefn = hlds_class_defn(TypeClassStatus, ClassTVarSet, _Kinds,
|
|
ClassParamTVars, SuperClasses, _FunDeps, _Ancestors,
|
|
Interface, ClassMethodInfos, ClassContext, MaybeBadDefn),
|
|
( if
|
|
typeclass_status_defined_in_this_module(TypeClassStatus) = yes,
|
|
Interface = class_interface_abstract
|
|
then
|
|
(
|
|
MaybeBadDefn = has_no_bad_class_defn,
|
|
Pieces = [words("Error: typeclass")] ++
|
|
color_as_subject([unqual_class_id(ClassId)]) ++
|
|
[words("has a declaration, but it")] ++
|
|
color_as_incorrect([words("has no definition.")]) ++
|
|
[nl],
|
|
Spec = spec($pred, severity_error, phase_type_check,
|
|
ClassContext, Pieces),
|
|
!:Specs = [Spec | !.Specs]
|
|
;
|
|
MaybeBadDefn = has_bad_class_defn
|
|
% If the class had a definition that was not added to the HLDS
|
|
% because it had an error in it, reporting that the class
|
|
% has *no* definition would be misleading.
|
|
),
|
|
InstanceDefns = InstanceDefns0
|
|
else
|
|
list.map_foldl3(
|
|
generate_instance_method_procs_for_class_instance(ClassId,
|
|
ClassTVarSet, ClassParamTVars, SuperClasses,
|
|
Interface, ClassMethodInfos),
|
|
InstanceDefns0, InstanceDefns,
|
|
!ModuleInfo, !QualInfo, !Specs)
|
|
).
|
|
|
|
%---------------------%
|
|
|
|
% Generate a procedure for each method of the given instance
|
|
% of the given class.
|
|
%
|
|
:- pred generate_instance_method_procs_for_class_instance(class_id::in,
|
|
tvarset::in, list(tvar)::in, list(prog_constraint)::in,
|
|
class_interface::in, list(method_info)::in,
|
|
hlds_instance_defn::in, hlds_instance_defn::out,
|
|
module_info::in, module_info::out, qual_info::in, qual_info::out,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
generate_instance_method_procs_for_class_instance(ClassId, ClassTVarSet,
|
|
ClassParamTVars, SuperClasses, ClassInterface, ClassMethodInfos,
|
|
!InstanceDefn, !ModuleInfo, !QualInfo, !Specs):-
|
|
% Generate the procedure from the instance body, if there is one.
|
|
!.InstanceDefn = hlds_instance_defn(_, _, _, _, _, _, _, _,
|
|
InstanceBody, _, InstanceContext),
|
|
(
|
|
InstanceBody = instance_body_abstract
|
|
;
|
|
InstanceBody = instance_body_concrete(InstanceMethods),
|
|
(
|
|
ClassInterface = class_interface_abstract,
|
|
Pieces = [words("Error:")] ++
|
|
color_as_incorrect([words("instance declaration"),
|
|
words("for abstract typeclass")]) ++
|
|
color_as_subject([unqual_class_id(ClassId), suffix(".")]) ++
|
|
[nl],
|
|
Spec = spec($pred, severity_error, phase_type_check,
|
|
InstanceContext, Pieces),
|
|
!:Specs = [Spec | !.Specs]
|
|
;
|
|
ClassInterface = class_interface_concrete(_),
|
|
generate_instance_method_procs_for_concrete_instance(ClassId,
|
|
ClassParamTVars, ClassMethodInfos, InstanceMethods,
|
|
!InstanceDefn, !ModuleInfo, !QualInfo, !Specs)
|
|
)
|
|
),
|
|
% Check that the superclass constraints are satisfied for the types
|
|
% in this instance declaration.
|
|
check_instance_for_superclass_conformance(!.ModuleInfo, ClassId,
|
|
ClassTVarSet, ClassParamTVars, SuperClasses, !InstanceDefn, !Specs).
|
|
|
|
:- pred generate_instance_method_procs_for_concrete_instance(class_id::in,
|
|
list(tvar)::in, list(method_info)::in, list(instance_method)::in,
|
|
hlds_instance_defn::in, hlds_instance_defn::out,
|
|
module_info::in, module_info::out, qual_info::in, qual_info::out,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
generate_instance_method_procs_for_concrete_instance(ClassId, ClassParamTVars,
|
|
ClassMethodInfos, InstanceMethods,
|
|
!InstanceDefn, !ModuleInfo, !QualInfo, !Specs) :-
|
|
build_instance_method_map(ClassId, !.InstanceDefn, InstanceMethods,
|
|
map.init, InstanceMethodMap, !Specs),
|
|
generate_instance_method_procs_for_preds(ClassId, ClassParamTVars,
|
|
!.InstanceDefn, ClassMethodInfos,
|
|
InstanceMethodMap, LeftOverInstanceMethodMap,
|
|
cord.init, InstanceMethodsCord, cord.init, InstanceMethodInfosCord,
|
|
!ModuleInfo, !QualInfo, !Specs),
|
|
!InstanceDefn ^ instdefn_maybe_method_infos :=
|
|
yes(cord.list(InstanceMethodInfosCord)),
|
|
!InstanceDefn ^ instdefn_body :=
|
|
instance_body_concrete(cord.list(InstanceMethodsCord)),
|
|
|
|
map.values(LeftOverInstanceMethodMap, UnknownInstanceMethods),
|
|
(
|
|
UnknownInstanceMethods = []
|
|
;
|
|
UnknownInstanceMethods =
|
|
[HeadUnknownInstanceMethod | TailUnknownInstanceMethods],
|
|
report_unknown_instance_methods(ClassId, !.InstanceDefn,
|
|
HeadUnknownInstanceMethod, TailUnknownInstanceMethods, !Specs)
|
|
).
|
|
|
|
%---------------------%
|
|
|
|
:- pred build_instance_method_map(class_id::in, hlds_instance_defn::in,
|
|
list(instance_method)::in,
|
|
map(pred_pf_name_arity, instance_method)::in,
|
|
map(pred_pf_name_arity, instance_method)::out,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
build_instance_method_map(_ClassId, _InstanceDefn, [],
|
|
!InstanceMethodMap, !Specs).
|
|
build_instance_method_map(ClassId, InstanceDefn, [Method | Methods],
|
|
!InstanceMethodMap, !Specs) :-
|
|
Method = instance_method(MethodName, NewDefn, NewContext),
|
|
( if map.search(!.InstanceMethodMap, MethodName, OldMethod) then
|
|
OldMethod = instance_method(_OldMethodName, OldDefn, OldContext),
|
|
( if
|
|
OldDefn = instance_proc_def_clauses(OldClausesCord),
|
|
NewDefn = instance_proc_def_clauses(NewClausesCord)
|
|
then
|
|
NewClauses = cord.list(NewClausesCord),
|
|
(
|
|
NewClauses = [],
|
|
unexpected($pred, "NewClauses = []")
|
|
;
|
|
NewClauses = [NewClause]
|
|
;
|
|
NewClauses = [_, _ | _],
|
|
unexpected($pred, "NewClauses = [_, _, | _]")
|
|
),
|
|
cord.snoc(NewClause, OldClausesCord, UpdatedClausesCord),
|
|
UpdatedDefn = instance_proc_def_clauses(UpdatedClausesCord),
|
|
UpdatedMethod =
|
|
instance_method(MethodName, UpdatedDefn, OldContext),
|
|
map.det_update(MethodName, UpdatedMethod, !InstanceMethodMap)
|
|
else
|
|
report_duplicate_method_defn(ClassId, InstanceDefn,
|
|
MethodName, OldContext, NewContext, !Specs)
|
|
)
|
|
else
|
|
map.det_insert(MethodName, Method, !InstanceMethodMap)
|
|
),
|
|
build_instance_method_map(ClassId, InstanceDefn, Methods,
|
|
!InstanceMethodMap, !Specs).
|
|
|
|
%---------------------------------------------------------------------------%
|
|
|
|
% Check every method in one instance of one class.
|
|
%
|
|
% Our caller invokes us with the method_info list of the class declaration,
|
|
% which consists of
|
|
%
|
|
% - the method_infos of all the procedures of the first method
|
|
% - the method_infos of all the procedures of the second method
|
|
% - and so on.
|
|
%
|
|
% So each method consists of a sequence of one or more method_infos.
|
|
%
|
|
:- pred generate_instance_method_procs_for_preds(class_id::in, list(tvar)::in,
|
|
hlds_instance_defn::in, list(method_info)::in,
|
|
map(pred_pf_name_arity, instance_method)::in,
|
|
map(pred_pf_name_arity, instance_method)::out,
|
|
cord(instance_method)::in, cord(instance_method)::out,
|
|
cord(method_info)::in, cord(method_info)::out,
|
|
module_info::in, module_info::out, qual_info::in, qual_info::out,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
generate_instance_method_procs_for_preds(_ClassId, _ClassVars, _InstanceDefn,
|
|
[], !InstanceMethodMap, !InstanceMethodsCord, !InstanceMethodInfosCord,
|
|
!ModuleInfo, !QualInfo, !Specs).
|
|
generate_instance_method_procs_for_preds(ClassId, ClassVars, InstanceDefn,
|
|
[HeadClassMethodInfo | TailClassMethodInfos],
|
|
!InstanceMethodMap, !InstanceMethodsCord, !InstanceMethodInfosCord,
|
|
!ModuleInfo, !QualInfo, !Specs) :-
|
|
HeadClassMethodInfo = method_info(_MethodNum, MethodName,
|
|
OrigClassPredProcId, _CurClassPredProcId),
|
|
OrigClassPredProcId = proc(ClassPredId, _),
|
|
get_other_class_method_procs(MethodName, TailClassMethodInfos,
|
|
TailCurClassMethodInfos, OtherClassMethodInfos),
|
|
( if map.remove(MethodName, InstanceMethod, !InstanceMethodMap) then
|
|
ClassMethodInfos = [HeadClassMethodInfo | TailCurClassMethodInfos],
|
|
generate_instance_method_pred_and_procs(ClassId, ClassVars,
|
|
ClassPredId, InstanceDefn, InstanceMethod,
|
|
ClassMethodInfos, InstanceMethodInfos,
|
|
!ModuleInfo, !QualInfo, !Specs),
|
|
cord.snoc(InstanceMethod, !InstanceMethodsCord),
|
|
!:InstanceMethodInfosCord = !.InstanceMethodInfosCord ++
|
|
cord.from_list(InstanceMethodInfos)
|
|
else
|
|
report_undefined_method(ClassId, InstanceDefn, MethodName, !Specs)
|
|
),
|
|
generate_instance_method_procs_for_preds(ClassId, ClassVars, InstanceDefn,
|
|
OtherClassMethodInfos,
|
|
!InstanceMethodMap, !InstanceMethodsCord, !InstanceMethodInfosCord,
|
|
!ModuleInfo, !QualInfo, !Specs).
|
|
|
|
%---------------------%
|
|
|
|
:- pred get_other_class_method_procs(pred_pf_name_arity::in,
|
|
list(method_info)::in,
|
|
list(method_info)::out, list(method_info)::out) is det.
|
|
|
|
get_other_class_method_procs(_ThisMethodName, [], [], []).
|
|
get_other_class_method_procs(CurMethodName,
|
|
MethodInfos @ [HeadMethodInfo | TailMethodInfos],
|
|
CurMethodInfos, OtherMethodInfos) :-
|
|
( if HeadMethodInfo ^ method_pred_name = CurMethodName then
|
|
get_other_class_method_procs(CurMethodName, TailMethodInfos,
|
|
TailCurMethodInfos, OtherMethodInfos),
|
|
CurMethodInfos = [HeadMethodInfo | TailCurMethodInfos]
|
|
else
|
|
CurMethodInfos = [],
|
|
OtherMethodInfos = MethodInfos
|
|
).
|
|
|
|
%---------------------------------------------------------------------------%
|
|
|
|
:- pred generate_instance_method_pred_and_procs(class_id::in,
|
|
list(tvar)::in, pred_id::in, hlds_instance_defn::in,
|
|
instance_method::in, list(method_info)::in, list(method_info)::out,
|
|
module_info::in, module_info::out, qual_info::in, qual_info::out,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
generate_instance_method_pred_and_procs(ClassId, ClassVars, ClassPredId,
|
|
InstanceDefn, InstanceMethod, ClassMethodInfos, InstanceMethodInfos,
|
|
!ModuleInfo, !QualInfo, !Specs) :-
|
|
module_info_pred_info(!.ModuleInfo, ClassPredId, ClassPredInfo),
|
|
pred_info_get_arg_types(ClassPredInfo, TVarSet0, ExistQVars0, ArgTypes0),
|
|
pred_info_get_class_context(ClassPredInfo, ClassContext0),
|
|
pred_info_get_markers(ClassPredInfo, Markers0),
|
|
% The first constraint in the class context of a class method
|
|
% is always the constraint for the class of which it is a member.
|
|
% Seeing that we are checking an instance declaration, we don't check
|
|
% that constraint... the instance declaration itself satisfies it!
|
|
ClassContext0 = univ_exist_constraints(UnivCs0, ExistCs),
|
|
(
|
|
UnivCs0 = [_ | UnivCs]
|
|
;
|
|
UnivCs0 = [],
|
|
unexpected($pred, "no constraint on class method")
|
|
),
|
|
ClassMethodClassContext0 = univ_exist_constraints(UnivCs, ExistCs),
|
|
ClassPredModule = pred_info_module(ClassPredInfo),
|
|
ClassPredMethodName = pred_info_name(ClassPredInfo),
|
|
ClassPredMethodSymName = qualified(ClassPredModule, ClassPredMethodName),
|
|
% The pred_or_func, pred_form_arity and user_arity of the method
|
|
% are the same in the class and in the instance.
|
|
PredOrFunc = pred_info_is_pred_or_func(ClassPredInfo),
|
|
PredFormArity = pred_info_pred_form_arity(ClassPredInfo),
|
|
user_arity_pred_form_arity(PredOrFunc, UserArity, PredFormArity),
|
|
|
|
% Work out the name of the predicate that we will generate
|
|
% to check this instance method.
|
|
make_instance_method_pred_name(ClassId, ClassPredMethodSymName,
|
|
UserArity, InstanceTypes0, InstancePredName),
|
|
|
|
InstanceDefn = hlds_instance_defn(InstanceModuleName, InstanceStatus0,
|
|
InstanceTVarSet, _OriginalTypes, InstanceTypes0,
|
|
InstanceConstraints0, _MaybeSubsumedContext,_InstanceProofs,
|
|
_InstanceBody, _MaybeMethodInfos, _Context),
|
|
UnsubstArgTypes = ArgTypes0,
|
|
|
|
InstanceMethod = instance_method(_, InstanceProcDefn, Context),
|
|
|
|
% Rename the instance variables apart from the class variables.
|
|
tvarset_merge_renaming(TVarSet0, InstanceTVarSet, TVarSet1, Renaming),
|
|
apply_renaming_to_types(Renaming, InstanceTypes0, InstanceTypes1),
|
|
apply_renaming_to_prog_constraints(Renaming,
|
|
InstanceConstraints0, InstanceConstraints1),
|
|
|
|
% Work out what the type variables are bound to for this
|
|
% instance, and update the class types appropriately.
|
|
map.from_corresponding_lists(ClassVars, InstanceTypes1, TypeSubst),
|
|
apply_subst_to_types(TypeSubst, ArgTypes0, ArgTypes1),
|
|
apply_subst_to_univ_exist_constraints(TypeSubst, ClassMethodClassContext0,
|
|
ClassMethodClassContext1),
|
|
|
|
% Calculate which type variables we need to keep. This includes all
|
|
% type variables appearing in the arguments, the class method context and
|
|
% the instance constraints. (Type variables in the existq_tvars must
|
|
% occur either in the argument types or in the class method context;
|
|
% type variables in the instance types must appear in the arguments.)
|
|
type_vars_in_types(ArgTypes1, ArgTVars),
|
|
univ_exist_constraints_get_tvars(ClassMethodClassContext1,
|
|
MethodContextTVars),
|
|
constraint_list_get_tvars(InstanceConstraints1, InstanceTVars),
|
|
VarsToKeep0 = ArgTVars ++ MethodContextTVars ++ InstanceTVars,
|
|
list.sort_and_remove_dups(VarsToKeep0, VarsToKeep),
|
|
|
|
% Project away the unwanted type variables.
|
|
varset.squash(TVarSet1, VarsToKeep, TVarSet2, SquashSubst),
|
|
apply_renaming_to_types(SquashSubst, ArgTypes1, ArgTypes),
|
|
apply_renaming_to_univ_exist_constraints(SquashSubst,
|
|
ClassMethodClassContext1, ClassMethodClassContext),
|
|
apply_partial_map_to_list(SquashSubst, ExistQVars0, ExistQVars),
|
|
apply_renaming_to_types(SquashSubst, InstanceTypes1, InstanceTypes),
|
|
apply_renaming_to_prog_constraints(SquashSubst,
|
|
InstanceConstraints1, InstanceConstraints),
|
|
|
|
% Add the constraints from the instance declaration to the constraints
|
|
% from the class method. This allows an instance method to have constraints
|
|
% on it which are not part of the instance declaration as a whole.
|
|
ClassMethodClassContext =
|
|
univ_exist_constraints(UnivConstraints1, ExistConstraints),
|
|
UnivConstraints = InstanceConstraints ++ UnivConstraints1,
|
|
ClassContext =
|
|
univ_exist_constraints(UnivConstraints, ExistConstraints),
|
|
|
|
% Introduce a new predicate which calls the implementation
|
|
% given in the instance declaration.
|
|
map.init(Proofs),
|
|
map.init(ConstraintMap),
|
|
remove_marker(marker_class_method, Markers0, Markers1),
|
|
add_marker(marker_class_instance_method, Markers1, Markers2),
|
|
(
|
|
InstanceProcDefn = instance_proc_def_name(_),
|
|
% For instance methods which are defined using the named syntax
|
|
% (e.g. "pred(...) is ...") rather than using clauses, we record
|
|
% this additional marker, whose only effect is that we generate
|
|
% slightly different error messages for such predicates.
|
|
add_marker(marker_named_class_instance_method, Markers2, Markers)
|
|
;
|
|
InstanceProcDefn = instance_proc_def_clauses(_),
|
|
Markers = Markers2
|
|
),
|
|
|
|
IsImported = instance_status_is_imported(InstanceStatus0),
|
|
(
|
|
IsImported = yes,
|
|
InstanceStatus = instance_status(status_opt_imported)
|
|
;
|
|
IsImported = no,
|
|
InstanceStatus = InstanceStatus0
|
|
),
|
|
|
|
produce_instance_method_clauses(InstanceProcDefn, PredOrFunc, ArgTypes,
|
|
Markers, Context, InstanceStatus, ClausesInfo,
|
|
TVarSet2, TVarSet, !ModuleInfo, !QualInfo, !Specs),
|
|
|
|
% Fill in some information in the pred_info which is used by polymorphism
|
|
% to make sure the type-infos and typeclass-infos are added in the correct
|
|
% order.
|
|
InstanceMethodConstraints = instance_method_constraints(ClassId,
|
|
InstanceTypes, InstanceConstraints, ClassMethodClassContext),
|
|
ClassMethodPFSymNameArity = pred_pf_name_arity(PredOrFunc,
|
|
ClassPredMethodSymName, UserArity),
|
|
PredOrigin = origin_user(
|
|
user_made_instance_method(ClassMethodPFSymNameArity,
|
|
InstanceMethodConstraints)),
|
|
map.init(VarNameRemap),
|
|
% XXX STATUS
|
|
InstanceStatus = instance_status(OldImportStatus),
|
|
PredStatus = pred_status(OldImportStatus),
|
|
CurUserDecl = maybe.no,
|
|
GoalType = goal_not_for_promise(np_goal_type_none),
|
|
pred_info_init(PredOrFunc, InstanceModuleName, InstancePredName,
|
|
PredFormArity, Context, PredOrigin, PredStatus, CurUserDecl, GoalType,
|
|
Markers, ArgTypes, TVarSet, ExistQVars, ClassContext, Proofs,
|
|
ConstraintMap, ClausesInfo, VarNameRemap, InstancePredInfo0),
|
|
pred_info_set_clauses_info(ClausesInfo,
|
|
InstancePredInfo0, InstancePredInfo1),
|
|
pred_info_set_instance_method_arg_types(UnsubstArgTypes,
|
|
InstancePredInfo1, InstancePredInfo2),
|
|
|
|
% We first insert the incomplete predicate InstancePredInfo2 into
|
|
% !ModuleInfo in order to get InstancePredId, the pred_id of the
|
|
% new predicate. We need this pred_id in order to compute the contents
|
|
% of the InstanceMethodInfos.
|
|
module_info_get_predicate_table(!.ModuleInfo, PredTable1),
|
|
predicate_table_insert(InstancePredInfo2, InstancePredId,
|
|
PredTable1, PredTable),
|
|
module_info_set_predicate_table(PredTable, !ModuleInfo),
|
|
|
|
% Add procs with the expected modes and determinisms.
|
|
pred_info_get_proc_table(ClassPredInfo, ClassProcTable),
|
|
list.map_foldl(
|
|
add_instance_method_proc(!.ModuleInfo, ClassPredId, ClassProcTable,
|
|
InstancePredId, Context),
|
|
ClassMethodInfos, InstanceMethodInfos,
|
|
InstancePredInfo2, InstancePredInfo),
|
|
% Replace the incomplete InstancePredInfo2.
|
|
module_info_set_pred_info(InstancePredId, InstancePredInfo, !ModuleInfo).
|
|
|
|
:- pred add_instance_method_proc(module_info::in, pred_id::in,
|
|
map(proc_id, proc_info)::in, pred_id::in, prog_context::in,
|
|
method_info::in, method_info::out, pred_info::in, pred_info::out) is det.
|
|
|
|
add_instance_method_proc(ModuleInfo, ClassPredId, ClassProcTable,
|
|
InstancePredId, Context, ClassMethodInfo, InstanceMethodInfo,
|
|
!InstancePredInfo) :-
|
|
ClassMethodInfo = method_info(MethodNum, MethodName,
|
|
ClassOrigPredProcId, _ClassCurPredProcId),
|
|
ClassOrigPredProcId = proc(ClassOrigPredId, ClassOrigProcId),
|
|
expect(unify(ClassOrigPredId, ClassPredId), $pred,
|
|
"ClassOrigPredId != ClassPredId"),
|
|
map.lookup(ClassProcTable, ClassOrigProcId, ClassProcInfo),
|
|
proc_info_get_inst_varset(ClassProcInfo, InstVarSet),
|
|
proc_info_get_argmodes(ClassProcInfo, Modes),
|
|
% If the determinism declaration on the method was omitted,
|
|
% then make_hlds will have already issued an error message,
|
|
% so don't complain here.
|
|
proc_info_get_declared_determinism(ClassProcInfo, MaybeDetism),
|
|
% Before the simplification pass, HasParallelConj
|
|
% is not meaningful.
|
|
HasParallelConj = has_no_parallel_conj,
|
|
add_new_proc(ModuleInfo, Context, item_no_seq_num, InstVarSet,
|
|
Modes, yes(Modes), no, detism_decl_implicit, MaybeDetism,
|
|
address_is_taken, HasParallelConj, !InstancePredInfo, InstanceProcId),
|
|
InstanceOrigPredProcId = proc(InstancePredId, InstanceProcId),
|
|
InstanceMethodInfo = method_info(MethodNum, MethodName,
|
|
InstanceOrigPredProcId, InstanceOrigPredProcId).
|
|
|
|
%---------------------------------------------------------------------------%
|
|
|
|
% Check that the superclass constraints are satisfied for the
|
|
% types in this instance declaration.
|
|
%
|
|
:- pred check_instance_for_superclass_conformance(module_info::in,
|
|
class_id::in, tvarset::in, list(tvar)::in, list(prog_constraint)::in,
|
|
hlds_instance_defn::in, hlds_instance_defn::out,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_instance_for_superclass_conformance(ModuleInfo, ClassId, ClassTVarSet,
|
|
ClassVars0, ProgSuperClasses0, InstanceDefn0, InstanceDefn, !Specs) :-
|
|
InstanceDefn0 = hlds_instance_defn(ModuleName, Status,
|
|
InstanceTVarSet0, OriginalTypes, InstanceTypes,
|
|
InstanceProgConstraints, MaybeSubsumedContext, Proofs0,
|
|
Body, Interface, Context),
|
|
tvarset_merge_renaming(InstanceTVarSet0, ClassTVarSet, InstanceTVarSet1,
|
|
Renaming),
|
|
|
|
% Make the constraints in terms of the instance variables.
|
|
apply_renaming_to_prog_constraints(Renaming,
|
|
ProgSuperClasses0, ProgSuperClasses),
|
|
|
|
% Now handle the class variables.
|
|
apply_renaming_to_tvars(Renaming, ClassVars0, ClassVars),
|
|
|
|
% Calculate the bindings.
|
|
map.from_corresponding_lists(ClassVars, InstanceTypes, TypeSubst),
|
|
|
|
module_info_get_class_table(ModuleInfo, ClassTable),
|
|
module_info_get_instance_table(ModuleInfo, InstanceTable),
|
|
|
|
% Build a suitable constraint context for checking the instance.
|
|
% To do this, we assume any constraints on the instance declaration
|
|
% (that is, treat them as universal constraints on a predicate) and try
|
|
% to prove the constraints on the class declaration (that is, treat them
|
|
% as existential constraints on a predicate).
|
|
%
|
|
% We don't bother assigning ids to these constraints, since the resulting
|
|
% constraint map is not used anyway.
|
|
%
|
|
init_hlds_constraint_list(ProgSuperClasses, SuperClasses),
|
|
init_hlds_constraint_list(InstanceProgConstraints, InstanceConstraints),
|
|
make_hlds_constraint_db(ClassTable, InstanceTVarSet1, SuperClasses,
|
|
InstanceConstraints, ConstraintDb0),
|
|
|
|
% Try to reduce the superclass constraints, using the declared instance
|
|
% constraints and the usual context reduction rules.
|
|
map.init(ConstraintMap0),
|
|
typeclasses.reduce_context_by_rule_application(ClassTable, InstanceTable,
|
|
ClassVars, TypeSubst, _, InstanceTVarSet1, InstanceTVarSet2,
|
|
Proofs0, Proofs1, ConstraintMap0, _, ConstraintDb0, ConstraintDb),
|
|
UnprovenConstraints = ConstraintDb ^ hcd_unproven,
|
|
|
|
(
|
|
UnprovenConstraints = [],
|
|
InstanceDefn = hlds_instance_defn(ModuleName, Status,
|
|
InstanceTVarSet2, OriginalTypes, InstanceTypes,
|
|
InstanceProgConstraints, MaybeSubsumedContext, Proofs1,
|
|
Body, Interface, Context)
|
|
;
|
|
UnprovenConstraints = [_ | _],
|
|
report_unsatistfied_superclass_constraint(ClassId,
|
|
InstanceDefn0, ClassTVarSet, UnprovenConstraints, !Specs),
|
|
InstanceDefn = InstanceDefn0
|
|
).
|
|
|
|
%---------------------------------------------------------------------------%
|
|
% Pass 4.
|
|
%---------------------------------------------------------------------------%
|
|
|
|
:- type type_vector_instances_map == map(type_vector, type_vector_instances).
|
|
:- type type_vector == list(mer_type).
|
|
:- type type_vector_instances
|
|
---> type_vector_instances(
|
|
local_abstracts :: list(hlds_instance_defn),
|
|
local_concretes :: list(hlds_instance_defn),
|
|
nonlocal_abstracts :: list(hlds_instance_defn),
|
|
nonlocal_concretes :: list(hlds_instance_defn)
|
|
).
|
|
|
|
% Check that every abstract instance in the module has a
|
|
% corresponding concrete instance in the implementation.
|
|
%
|
|
% XXX That was the original purpose of this predicate. Now it performs
|
|
% several other checks:
|
|
%
|
|
% - It reports errors for duplicate concrete instance declarations
|
|
% in both the current module, and in (the visible parts of)
|
|
% other modules.
|
|
%
|
|
% - It reports errors for concrete local instance declarations
|
|
% that duplicate concrete instances from other modules.
|
|
%
|
|
% - It reports warnings for duplicate abstract instance declarations
|
|
% in the current module.
|
|
%
|
|
% - It compares local concrete instance declarations with their
|
|
% abstract versions (if any), and report an error if they specify
|
|
% different constraints.
|
|
%
|
|
% The checks done by the rest of this module on instances should
|
|
% also be moved here, both to avoid redundant traversals of the
|
|
% instance table, and because we should be able to generate
|
|
% better messages using the knowledge we gather here.
|
|
%
|
|
% XXX This predicate should be renamed once that is done.
|
|
% Likewise for many of the predicates in its call tree.
|
|
%
|
|
:- pred check_for_missing_concrete_instances(module_info::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_for_missing_concrete_instances(ModuleInfo, !Specs) :-
|
|
module_info_get_instance_table(ModuleInfo, InstanceTable),
|
|
map.foldl(check_for_missing_concrete_instances_in_class,
|
|
InstanceTable, !Specs).
|
|
|
|
:- pred check_for_missing_concrete_instances_in_class(class_id::in,
|
|
list(hlds_instance_defn)::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_for_missing_concrete_instances_in_class(ClassId, Instances, !Specs) :-
|
|
build_type_vector_instances_map(Instances, map.init, VectorInstancesMap),
|
|
map.to_sorted_assoc_list(VectorInstancesMap, VectorInstancesAL),
|
|
list.map_foldl(
|
|
check_for_missing_concrete_instances_in_class_and_vector(ClassId),
|
|
VectorInstancesAL, PickedInstances, !Specs),
|
|
% XXX PickedInstances should replace Instances in ClassId's entry
|
|
% in the instance table.
|
|
check_for_overlapping_nonidentical_instances(ClassId,
|
|
PickedInstances, !Specs).
|
|
|
|
:- pred check_for_overlapping_nonidentical_instances(class_id::in,
|
|
list(hlds_instance_defn)::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_for_overlapping_nonidentical_instances(_, [], !Specs).
|
|
check_for_overlapping_nonidentical_instances(ClassId,
|
|
[HeadInstanceDefn | TailInstanceDefns], !Specs) :-
|
|
list.foldl(
|
|
check_for_overlapping_nonidentical_instance(ClassId, HeadInstanceDefn),
|
|
TailInstanceDefns, !Specs),
|
|
check_for_overlapping_nonidentical_instances(ClassId,
|
|
TailInstanceDefns, !Specs).
|
|
|
|
:- pred check_for_overlapping_nonidentical_instance(class_id::in,
|
|
hlds_instance_defn::in, hlds_instance_defn::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_for_overlapping_nonidentical_instance(ClassId,
|
|
InstanceDefnA, InstanceDefnB, !Specs) :-
|
|
InstanceDefnA = hlds_instance_defn(_, _, TVarSetA, _, TypesA,
|
|
_, _, _, _, _, ContextA),
|
|
InstanceDefnB = hlds_instance_defn(_, _, TVarSetB, _, TypesB,
|
|
_, _, _, _, _, ContextB),
|
|
tvarset_merge_renaming(TVarSetA, TVarSetB, _MergedTVarSetAB, RenamingAB),
|
|
tvarset_merge_renaming(TVarSetB, TVarSetB, _MergedTVarSetBA, RenamingBA),
|
|
( if
|
|
(
|
|
apply_renaming_to_types(RenamingAB, TypesB, TypesBR),
|
|
type_list_subsumes(TypesA, TypesBR, _)
|
|
;
|
|
apply_renaming_to_types(RenamingBA, TypesA, TypesAR),
|
|
type_list_subsumes(TypesB, TypesAR, _)
|
|
)
|
|
then
|
|
report_overlapping_instances(ClassId, ContextA, ContextB, !Specs)
|
|
else
|
|
true
|
|
).
|
|
|
|
%---------------------%
|
|
|
|
:- pred build_type_vector_instances_map(list(hlds_instance_defn)::in,
|
|
type_vector_instances_map::in, type_vector_instances_map::out) is det.
|
|
|
|
build_type_vector_instances_map([], !VectorInstancesMap).
|
|
build_type_vector_instances_map([InstanceDefn | InstanceDefns],
|
|
!VectorInstancesMap) :-
|
|
% Two instance declarations can talk about the same instance of a class
|
|
% even if they look different. Consider these two instances:
|
|
%
|
|
% :- instance c1(t1(A, B), t2(A), t3(B)) where ...
|
|
% :- instance c1(t1(X, Y), t2(X), t3(Y)) where ...
|
|
%
|
|
% They look different due to the differences in variable names,
|
|
% but types contain variable *numbers*, not variable *names*, and
|
|
% these numbers are allocated sequentially in order of first appearance.
|
|
% Therefore if two types unify, then they are structurally identical,
|
|
% and may differ only in variable names. Two instance declarations
|
|
% talk about the same instance of a class if their type vectors are
|
|
% identical.
|
|
TypeVector = InstanceDefn ^ instdefn_types,
|
|
InstanceStatus = InstanceDefn ^ instdefn_status,
|
|
IsImported = instance_status_is_imported(InstanceStatus),
|
|
( if map.search(!.VectorInstancesMap, TypeVector, VectorInstances0) then
|
|
categorize_and_add_instance_defn(IsImported, InstanceDefn,
|
|
VectorInstances0, VectorInstances),
|
|
map.det_update(TypeVector, VectorInstances, !VectorInstancesMap)
|
|
else
|
|
VectorInstances0 = type_vector_instances([], [], [], []),
|
|
categorize_and_add_instance_defn(IsImported, InstanceDefn,
|
|
VectorInstances0, VectorInstances),
|
|
map.det_insert(TypeVector, VectorInstances, !VectorInstancesMap)
|
|
),
|
|
build_type_vector_instances_map(InstanceDefns, !VectorInstancesMap).
|
|
|
|
:- pred categorize_and_add_instance_defn(bool::in, hlds_instance_defn::in,
|
|
type_vector_instances::in, type_vector_instances::out) is det.
|
|
|
|
categorize_and_add_instance_defn(IsImported, InstanceDefn,
|
|
VectorInstances0, VectorInstances) :-
|
|
Body = InstanceDefn ^ instdefn_body,
|
|
(
|
|
IsImported = no,
|
|
(
|
|
Body = instance_body_abstract,
|
|
LocalAbstracts0 = VectorInstances0 ^ local_abstracts,
|
|
VectorInstances = VectorInstances0 ^ local_abstracts
|
|
:= [InstanceDefn | LocalAbstracts0]
|
|
;
|
|
Body = instance_body_concrete(_),
|
|
LocalConcretes0 = VectorInstances0 ^ local_concretes,
|
|
VectorInstances = VectorInstances0 ^ local_concretes
|
|
:= [InstanceDefn | LocalConcretes0]
|
|
)
|
|
;
|
|
IsImported = yes,
|
|
(
|
|
Body = instance_body_abstract,
|
|
NonLocalAbstracts0 = VectorInstances0 ^ nonlocal_abstracts,
|
|
VectorInstances = VectorInstances0 ^ nonlocal_abstracts
|
|
:= [InstanceDefn | NonLocalAbstracts0]
|
|
;
|
|
Body = instance_body_concrete(_),
|
|
NonLocalConcretes0 = VectorInstances0 ^ nonlocal_concretes,
|
|
VectorInstances = VectorInstances0 ^ nonlocal_concretes
|
|
:= [InstanceDefn | NonLocalConcretes0]
|
|
)
|
|
).
|
|
|
|
%---------------------%
|
|
|
|
% XXX See the comment on our ancestor check_for_missing_concrete_instances.
|
|
%
|
|
:- pred check_for_missing_concrete_instances_in_class_and_vector(class_id::in,
|
|
pair(type_vector, type_vector_instances)::in, hlds_instance_defn::out,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_for_missing_concrete_instances_in_class_and_vector(ClassId,
|
|
_TypeVector - VectorInstances, PickedInstance, !Specs) :-
|
|
% XXX We should use _TypeVector to implement the check
|
|
% now done by check_instance_declaration_types_for_instance.
|
|
%
|
|
% XXX We should also use it to implement --warn-unnecessarily-private-
|
|
% instance, which (as proposed by Julien on 2022 mar 8) would warn about
|
|
% non-exported instances whose type vectors contain only non-private
|
|
% type constructors.
|
|
|
|
% We pick a single instance for this type vector in the instance table
|
|
% using a multi-stage tournament. Most, though not all, instances
|
|
% that are knocked out of the tournament get an error message generated
|
|
% for them.
|
|
|
|
% The first stage partitions the instance definitions
|
|
% based on two properties: local/nonlocal, and abstract/concrete.
|
|
% It then picks the (lexically) first in each category,
|
|
% and reports all the others as duplicates.
|
|
VectorInstances = type_vector_instances(LocalAbstracts0, LocalConcretes0,
|
|
NonLocalAbstracts0, NonLocalConcretes0),
|
|
list.sort(compare_instance_defns_by_context,
|
|
LocalAbstracts0, LocalAbstracts),
|
|
list.sort(compare_instance_defns_by_context,
|
|
LocalConcretes0, LocalConcretes),
|
|
list.sort(compare_instance_defns_by_context,
|
|
NonLocalAbstracts0, NonLocalAbstracts),
|
|
list.sort(compare_instance_defns_by_context,
|
|
NonLocalConcretes0, NonLocalConcretes),
|
|
report_any_duplicate_instance_defns_in_category(ClassId,
|
|
severity_error, "Error", "concrete",
|
|
LocalConcretes, MaybeLocalConcrete, [], LocalConcreteSpecs),
|
|
report_any_duplicate_instance_defns_in_category(ClassId,
|
|
severity_error, "Error", "imported concrete",
|
|
NonLocalConcretes, MaybeNonLocalConcrete, [], NonLocalConcreteSpecs),
|
|
report_any_duplicate_instance_defns_in_category(ClassId,
|
|
severity_warning(warn_duplicate_abstract_instances),
|
|
"Warning", "abstract",
|
|
LocalAbstracts, MaybeLocalAbstract, !Specs),
|
|
% intermod.m can put an abstract instance declaration into a .opt file
|
|
% even when an interface file contains that same abstract instance.
|
|
% We still want to pick a single abstract nonlocal abstract instance,
|
|
% but we don't want to report duplicates.
|
|
report_any_duplicate_instance_defns_in_category(ClassId,
|
|
severity_warning(warn_duplicate_abstract_instances),
|
|
"Warning", "imported abstract",
|
|
NonLocalAbstracts, MaybeNonLocalAbstract, !.Specs, _),
|
|
!:Specs = LocalConcreteSpecs ++ NonLocalConcreteSpecs ++ !.Specs,
|
|
|
|
% If there is no concrete instance definition, then there is no
|
|
% constraint set to check any abstract instance definitions against/
|
|
% On the other hand, if there was more than one concrete definition,
|
|
% then an error message about an abstract instance not matching
|
|
% the picked concrete instance's constraints may be misleading,
|
|
% since the abstract instance's constraints may match a *non*-picked
|
|
% concrete instance's constraints.
|
|
( if
|
|
MaybeLocalConcrete = yes(LocalConcrete0),
|
|
LocalConcreteSpecs = []
|
|
then
|
|
list.foldl(
|
|
check_that_instance_constraints_match(ClassId, LocalConcrete0),
|
|
LocalAbstracts, !Specs)
|
|
else
|
|
true
|
|
),
|
|
( if
|
|
MaybeNonLocalConcrete = yes(NonLocalConcrete0),
|
|
NonLocalConcreteSpecs = []
|
|
then
|
|
list.foldl(
|
|
check_that_instance_constraints_match(ClassId, NonLocalConcrete0),
|
|
NonLocalAbstracts, !Specs)
|
|
else
|
|
true
|
|
),
|
|
|
|
% The second stage of the tournament picks one local instance
|
|
% and one nonlocal instance, with concrete trumping abstract.
|
|
(
|
|
MaybeLocalConcrete = no,
|
|
(
|
|
MaybeLocalAbstract = no,
|
|
MaybeLocal = no
|
|
;
|
|
MaybeLocalAbstract = yes(LocalAbstract),
|
|
report_abstract_instance_without_concrete(ClassId,
|
|
LocalAbstract, !Specs),
|
|
MaybeLocal = MaybeLocalAbstract
|
|
)
|
|
;
|
|
MaybeLocalConcrete = yes(LocalConcrete1),
|
|
(
|
|
MaybeLocalAbstract = no,
|
|
MaybeLocal = MaybeLocalConcrete
|
|
;
|
|
MaybeLocalAbstract = yes(LocalAbstract),
|
|
LocalConcrete = LocalConcrete1 ^ instdefn_subsumed_ctxt :=
|
|
yes(LocalAbstract ^ instdefn_context),
|
|
MaybeLocal = yes(LocalConcrete)
|
|
)
|
|
),
|
|
(
|
|
MaybeNonLocalConcrete = no,
|
|
% It is not just ok but expected for the concrete version of a
|
|
% nonlocal abstract instance declaration to be invisible to us.
|
|
% Concrete instance declarations are *never* supposed to occur
|
|
% in in .int* files (we make instance declaration abstract
|
|
% before putting them into .int* files), but concrete instances
|
|
% may appear in .opt files.
|
|
MaybeNonLocal = MaybeNonLocalAbstract
|
|
;
|
|
MaybeNonLocalConcrete = yes(_),
|
|
MaybeNonLocal = MaybeNonLocalConcrete
|
|
),
|
|
|
|
% The third and last stage of the tournament picks the winner,
|
|
% with local trumping nonlocal.
|
|
(
|
|
MaybeLocal = no,
|
|
(
|
|
MaybeNonLocal = no,
|
|
unexpected($pred, "no instance left to pick")
|
|
;
|
|
MaybeNonLocal = yes(NonLocal),
|
|
PickedInstance = NonLocal
|
|
)
|
|
;
|
|
MaybeLocal = yes(Local),
|
|
PickedInstance = Local,
|
|
(
|
|
MaybeNonLocal = no
|
|
;
|
|
MaybeNonLocal = yes(NonLocal),
|
|
report_local_vs_nonlocal_clash(ClassId, Local, NonLocal, !Specs)
|
|
)
|
|
).
|
|
|
|
:- pred compare_instance_defns_by_context(
|
|
hlds_instance_defn::in, hlds_instance_defn::in, comparison_result::out)
|
|
is det.
|
|
|
|
compare_instance_defns_by_context(InstanceDefnA, InstanceDefnB, Result) :-
|
|
ContextA = InstanceDefnA ^ instdefn_context,
|
|
ContextB = InstanceDefnB ^ instdefn_context,
|
|
compare(Result, ContextA, ContextB).
|
|
|
|
%---------------------%
|
|
|
|
:- pred check_that_instance_constraints_match(class_id::in,
|
|
hlds_instance_defn::in, hlds_instance_defn::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_that_instance_constraints_match(ClassId,
|
|
ConcreteInstanceDefn, AbstractInstanceDefn, !Specs) :-
|
|
type_vars_in_types(ConcreteInstanceDefn ^ instdefn_types, ConcreteTVars),
|
|
type_vars_in_types(AbstractInstanceDefn ^ instdefn_types, AbstractTVars),
|
|
ConcreteTVarSet = ConcreteInstanceDefn ^ instdefn_tvarset,
|
|
AbstractTVarSet = AbstractInstanceDefn ^ instdefn_tvarset,
|
|
ConcreteConstraints = ConcreteInstanceDefn ^ instdefn_constraints,
|
|
AbstractConstraints = AbstractInstanceDefn ^ instdefn_constraints,
|
|
( if
|
|
constraints_are_identical(
|
|
ConcreteTVars, ConcreteTVarSet, ConcreteConstraints,
|
|
AbstractTVars, AbstractTVarSet, AbstractConstraints)
|
|
then
|
|
true
|
|
else
|
|
report_abstract_concrete_constraints_mismatch(ClassId,
|
|
AbstractInstanceDefn, ConcreteInstanceDefn, !Specs)
|
|
).
|
|
|
|
constraints_are_identical(OldVars0, OldVarSet, OldConstraints0,
|
|
Vars, VarSet, Constraints) :-
|
|
tvarset_merge_renaming(VarSet, OldVarSet, _, Renaming),
|
|
apply_renaming_to_prog_constraints(Renaming,
|
|
OldConstraints0, OldConstraints1),
|
|
apply_renaming_to_tvars(Renaming, OldVars0, OldVars),
|
|
|
|
map.from_corresponding_lists(OldVars, Vars, VarRenaming),
|
|
apply_renaming_to_prog_constraints(VarRenaming,
|
|
OldConstraints1, OldConstraints),
|
|
OldConstraints = Constraints.
|
|
|
|
%---------------------------------------------------------------------------%
|
|
% Pass 5.
|
|
%---------------------------------------------------------------------------%
|
|
|
|
% Check that all instances are range restricted with respect to the
|
|
% functional dependencies. This means that, for each functional dependency,
|
|
% the set of tvars in the range arguments must be a subset of the set
|
|
% of tvars in the domain arguments. (Note that with the requirement of
|
|
% distinct variables as arguments, this implies that all range arguments
|
|
% must be ground. However, this code should work even if that requirement
|
|
% is lifted in the future.)
|
|
%
|
|
% Also, check that all pairs of visible instances are mutually consistent
|
|
% with respect to the functional dependencies. This is true iff the most
|
|
% general unifier of corresponding domain arguments (if it exists) is
|
|
% also a unifier of the corresponding range arguments.
|
|
%
|
|
:- pred check_functional_dependencies(module_info::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_functional_dependencies(ModuleInfo, !Specs) :-
|
|
module_info_get_instance_table(ModuleInfo, InstanceTable),
|
|
map.keys(InstanceTable, ClassIds),
|
|
list.foldl(check_fundeps_for_class(ModuleInfo), ClassIds, !Specs).
|
|
|
|
:- pred check_fundeps_for_class(module_info::in, class_id::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_fundeps_for_class(ModuleInfo, ClassId, !Specs) :-
|
|
module_info_get_class_table(ModuleInfo, ClassTable),
|
|
map.lookup(ClassTable, ClassId, ClassDefn),
|
|
module_info_get_instance_table(ModuleInfo, InstanceTable),
|
|
map.lookup(InstanceTable, ClassId, InstanceDefns),
|
|
FunDeps = ClassDefn ^ classdefn_fundeps,
|
|
check_coverage_for_instance_defns(ModuleInfo, ClassId, InstanceDefns,
|
|
FunDeps, !Specs),
|
|
module_info_get_globals(ModuleInfo, Globals),
|
|
% Abstract definitions will always overlap with concrete definitions,
|
|
% so we filter out the abstract definitions for this module. If
|
|
% --intermodule-optimization is enabled then we strip out the imported
|
|
% abstract definitions for all modules, since we will have the concrete
|
|
% definitions for imported instances from the .opt files. If it is not
|
|
% enabled, then we keep the abstract definitions for imported instances
|
|
% since doing so may allow us to detect errors.
|
|
globals.lookup_bool_option(Globals, intermodule_optimization, IntermodOpt),
|
|
(
|
|
IntermodOpt = yes,
|
|
list.filter(is_concrete_instance_defn, InstanceDefns,
|
|
ConcreteInstanceDefns)
|
|
;
|
|
IntermodOpt = no,
|
|
list.filter(is_concrete_or_imported_instance_defn, InstanceDefns,
|
|
ConcreteInstanceDefns)
|
|
),
|
|
check_consistency(ClassId, ClassDefn, ConcreteInstanceDefns, FunDeps,
|
|
!Specs).
|
|
|
|
:- pred is_concrete_instance_defn(hlds_instance_defn::in) is semidet.
|
|
|
|
is_concrete_instance_defn(InstanceDefn) :-
|
|
InstanceDefn ^ instdefn_body = instance_body_concrete(_).
|
|
|
|
:- pred is_concrete_or_imported_instance_defn(hlds_instance_defn::in)
|
|
is semidet.
|
|
|
|
is_concrete_or_imported_instance_defn(InstanceDefn) :-
|
|
(
|
|
is_concrete_instance_defn(InstanceDefn)
|
|
;
|
|
instance_status_is_imported(InstanceDefn ^ instdefn_status) = yes
|
|
).
|
|
|
|
:- pred check_coverage_for_instance_defns(module_info::in, class_id::in,
|
|
list(hlds_instance_defn)::in, list(hlds_class_fundep)::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_coverage_for_instance_defns(_, _, [], _, !Specs).
|
|
check_coverage_for_instance_defns(ModuleInfo, ClassId,
|
|
[InstanceDefn | InstanceDefns], FunDeps, !Specs) :-
|
|
list.foldl(
|
|
check_coverage_for_instance_defn(ModuleInfo, ClassId, InstanceDefn),
|
|
FunDeps, !Specs),
|
|
check_coverage_for_instance_defns(ModuleInfo, ClassId,
|
|
InstanceDefns, FunDeps, !Specs).
|
|
|
|
:- pred check_coverage_for_instance_defn(module_info::in, class_id::in,
|
|
hlds_instance_defn::in, hlds_class_fundep::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_coverage_for_instance_defn(ModuleInfo, ClassId, InstanceDefn, FunDep,
|
|
!Specs) :-
|
|
TVarSet = InstanceDefn ^ instdefn_tvarset,
|
|
Types = InstanceDefn ^ instdefn_types,
|
|
FunDep = fundep(Domain, Range),
|
|
DomainTypes = restrict_list_elements(Domain, Types),
|
|
RangeTypes = restrict_list_elements(Range, Types),
|
|
type_vars_in_types(DomainTypes, DomainTVars),
|
|
type_vars_in_types(RangeTypes, RangeTVars),
|
|
Constraints = InstanceDefn ^ instdefn_constraints,
|
|
get_unbound_tvars(ModuleInfo, TVarSet, DomainTVars, RangeTVars,
|
|
Constraints, UnboundVars),
|
|
(
|
|
UnboundVars = []
|
|
;
|
|
UnboundVars = [_ | _],
|
|
report_coverage_error(ClassId, InstanceDefn, UnboundVars, !Specs)
|
|
).
|
|
|
|
% Check the consistency of each (unordered) pair of instances.
|
|
%
|
|
:- pred check_consistency(class_id::in, hlds_class_defn::in,
|
|
list(hlds_instance_defn)::in, list(hlds_class_fundep)::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_consistency(_, _, [], _, !Specs).
|
|
check_consistency(ClassId, ClassDefn, [Instance | Instances], FunDeps,
|
|
!Specs) :-
|
|
list.foldl(check_consistency_pair(ClassId, ClassDefn, FunDeps, Instance),
|
|
Instances, !Specs),
|
|
check_consistency(ClassId, ClassDefn, Instances, FunDeps,
|
|
!Specs).
|
|
|
|
:- pred check_consistency_pair(class_id::in, hlds_class_defn::in,
|
|
list(hlds_class_fundep)::in,
|
|
hlds_instance_defn::in, hlds_instance_defn::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_consistency_pair(ClassId, ClassDefn, FunDeps, InstanceA, InstanceB,
|
|
!Specs) :-
|
|
% If both instances are imported from the same module, then we do not
|
|
% need to check the consistency, since this would have been checked
|
|
% when compiling that module.
|
|
( if
|
|
InstanceA ^ instdefn_module = InstanceB ^ instdefn_module,
|
|
instance_status_is_imported(InstanceA ^ instdefn_status) = yes
|
|
then
|
|
true
|
|
else
|
|
list.foldl(
|
|
check_consistency_pair_2(ClassId, ClassDefn, InstanceA, InstanceB),
|
|
FunDeps, !Specs)
|
|
).
|
|
|
|
:- pred check_consistency_pair_2(class_id::in, hlds_class_defn::in,
|
|
hlds_instance_defn::in, hlds_instance_defn::in, hlds_class_fundep::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_consistency_pair_2(ClassId, ClassDefn, InstanceA, InstanceB, FunDep,
|
|
!Specs) :-
|
|
TVarSetA = InstanceA ^ instdefn_tvarset,
|
|
TVarSetB = InstanceB ^ instdefn_tvarset,
|
|
tvarset_merge_renaming(TVarSetA, TVarSetB, _, Renaming),
|
|
|
|
TypesA = InstanceA ^ instdefn_types,
|
|
TypesB0 = InstanceB ^ instdefn_types,
|
|
apply_renaming_to_types(Renaming, TypesB0, TypesB),
|
|
|
|
FunDep = fundep(Domain, Range),
|
|
DomainA = restrict_list_elements(Domain, TypesA),
|
|
DomainB = restrict_list_elements(Domain, TypesB),
|
|
|
|
( if type_unify_list(DomainA, DomainB, [], map.init, Subst) then
|
|
RangeA0 = restrict_list_elements(Range, TypesA),
|
|
RangeB0 = restrict_list_elements(Range, TypesB),
|
|
apply_rec_subst_to_types(Subst, RangeA0, RangeA),
|
|
apply_rec_subst_to_types(Subst, RangeB0, RangeB),
|
|
( if RangeA = RangeB then
|
|
true
|
|
else
|
|
report_consistency_error(ClassId, ClassDefn,
|
|
InstanceA, InstanceB, FunDep, !Specs)
|
|
)
|
|
else
|
|
true
|
|
).
|
|
|
|
%---------------------------------------------------------------------------%
|
|
% Pass 6.
|
|
%---------------------------------------------------------------------------%
|
|
|
|
% Look for pred or func declarations for which the type variables in
|
|
% the constraints are not all determined by the type variables in the type
|
|
% and the functional dependencies.
|
|
%
|
|
:- pred check_typeclass_constraints_on_preds(io.text_output_stream::in,
|
|
module_info::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_typeclass_constraints_on_preds(ProgressStream, ModuleInfo, !Specs) :-
|
|
module_info_get_valid_pred_ids(ModuleInfo, PredIds),
|
|
list.foldl(check_typeclass_constraints_on_pred(ProgressStream, ModuleInfo),
|
|
PredIds, !Specs).
|
|
|
|
:- pred check_typeclass_constraints_on_pred(io.text_output_stream::in,
|
|
module_info::in, pred_id::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_typeclass_constraints_on_pred(ProgressStream, ModuleInfo, PredId,
|
|
!Specs) :-
|
|
module_info_pred_info(ModuleInfo, PredId, PredInfo),
|
|
pred_info_get_class_context(PredInfo, Constraints),
|
|
Constraints = univ_exist_constraints(UnivConstraints, ExistConstraints),
|
|
( if
|
|
( UnivConstraints = [_ | _]
|
|
; ExistConstraints = [_ | _]
|
|
)
|
|
then
|
|
% We are checking only the forms of the constraints, so for now,
|
|
% the distinction between universally and existentially quantified
|
|
% constraints does not matter.
|
|
AllConstraints = UnivConstraints ++ ExistConstraints,
|
|
trace [io(!IO)] (
|
|
maybe_write_pred_progress_message(ProgressStream, ModuleInfo,
|
|
"Checking typeclass constraints on", PredId, !IO)
|
|
),
|
|
|
|
% Check that all class ids mentioned in the constraints
|
|
% actually exist.
|
|
%
|
|
% Code in the type checker does map.lookups of class_ids
|
|
% in the class table, and these will abort the compiler
|
|
% if invoked on a class id that does not exist in the class table.
|
|
% I (zs) have tried to change that code to do a map.search instead,
|
|
% and report the nonexistence of the class id then, but that turned
|
|
% out to be quite complicated, mostly because the code that does
|
|
% the lookup is called from several different places, each of which
|
|
% would need to handle this error in its own way. It seems simpler
|
|
% to look for and report such errors before typechecking even begins.
|
|
%
|
|
% Normally, module qualification, operating on the parse tree
|
|
% before the HLDS is even constructed, will find and report such
|
|
% errors. However, it does such checks *only* on the code of
|
|
% the curent module; it does *not* check the information read in
|
|
% from .int* files. Any references to nonexistent typeclasses
|
|
% in other modules, that this module imports, will be reported
|
|
% when that other module is compiled, and *could* be reported
|
|
% when the other module's interface file is generated, but for now,
|
|
% the latter is inhibited by the fact that halt_at_invalid_interface
|
|
% can't really be set to "yes" just yet. This allows such errors
|
|
% to slip into .int* files silently. The code here exists
|
|
% to catch and report them. And if this phase reports any errors,
|
|
% the compiler will stop before the typecheck phase.
|
|
|
|
module_info_get_class_table(ModuleInfo, ClassTable),
|
|
find_bad_class_ids_in_constraints(ClassTable, AllConstraints,
|
|
set.init, BadClassSNAsSet),
|
|
set.to_sorted_list(BadClassSNAsSet, BadClassSNAs),
|
|
(
|
|
BadClassSNAs = [HeadBadClassSNA | TailBadClassSNAs],
|
|
% In the presence of any BadClassSNAs, the map.lookups done
|
|
% in the class table by code indirectly invoked by
|
|
% get_unbound_tvars could cause a compiler abort.
|
|
%
|
|
% Any ambiguity errors can be reported *after* the programmer
|
|
% fixes the references to nonexistent typeclasses.
|
|
report_bad_class_ids_in_pred_decl(ModuleInfo, PredInfo,
|
|
HeadBadClassSNA, TailBadClassSNAs, !Specs)
|
|
;
|
|
BadClassSNAs = [],
|
|
pred_info_get_status(PredInfo, Status),
|
|
NeedsAmbiguityCheck = pred_needs_ambiguity_check(Status),
|
|
(
|
|
NeedsAmbiguityCheck = no
|
|
;
|
|
NeedsAmbiguityCheck = yes,
|
|
check_pred_type_ambiguities(ModuleInfo, PredInfo, !Specs),
|
|
check_constraint_quant(PredInfo, !Specs)
|
|
)
|
|
)
|
|
else
|
|
true
|
|
).
|
|
|
|
%---------------------%
|
|
|
|
:- func pred_needs_ambiguity_check(pred_status) = bool.
|
|
|
|
pred_needs_ambiguity_check(pred_status(status_imported(_))) = no.
|
|
pred_needs_ambiguity_check(pred_status(status_external(_))) = yes.
|
|
pred_needs_ambiguity_check(pred_status(status_abstract_imported)) = no.
|
|
pred_needs_ambiguity_check(pred_status(status_pseudo_imported)) = no.
|
|
pred_needs_ambiguity_check(pred_status(status_opt_imported)) = no.
|
|
pred_needs_ambiguity_check(pred_status(status_exported)) = yes.
|
|
pred_needs_ambiguity_check(pred_status(status_opt_exported)) = yes.
|
|
pred_needs_ambiguity_check(pred_status(status_abstract_exported)) = yes.
|
|
pred_needs_ambiguity_check(pred_status(status_pseudo_exported)) = yes.
|
|
pred_needs_ambiguity_check(pred_status(status_exported_to_submodules)) = yes.
|
|
pred_needs_ambiguity_check(pred_status(status_local)) = yes.
|
|
|
|
:- pred check_pred_type_ambiguities(module_info::in, pred_info::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_pred_type_ambiguities(ModuleInfo, PredInfo, !Specs) :-
|
|
pred_info_get_typevarset(PredInfo, TVarSet),
|
|
pred_info_get_arg_types(PredInfo, ArgTypes),
|
|
pred_info_get_class_context(PredInfo, Constraints),
|
|
type_vars_in_types(ArgTypes, ArgTVars),
|
|
univ_exist_constraints_get_tvars(Constraints, ConstrainedTVars),
|
|
Constraints = univ_exist_constraints(UnivCs, ExistCs),
|
|
get_unbound_tvars(ModuleInfo, TVarSet, ArgTVars, ConstrainedTVars,
|
|
UnivCs ++ ExistCs, UnboundTVars),
|
|
(
|
|
UnboundTVars = []
|
|
;
|
|
UnboundTVars = [_ | _],
|
|
report_unbound_tvars_in_pred_context(PredInfo, UnboundTVars, !Specs)
|
|
).
|
|
|
|
%---------------------%
|
|
|
|
% Check that all types appearing in universal (existential) constraints are
|
|
% universally (existentially) quantified.
|
|
%
|
|
:- pred check_constraint_quant(pred_info::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_constraint_quant(PredInfo, !Specs) :-
|
|
pred_info_get_exist_quant_tvars(PredInfo, ExistQVars),
|
|
pred_info_get_class_context(PredInfo, Constraints),
|
|
Constraints = univ_exist_constraints(UnivCs, ExistCs),
|
|
constraint_list_get_tvars(UnivCs, UnivTVars),
|
|
set.list_to_set(ExistQVars, ExistQVarsSet),
|
|
set.list_to_set(UnivTVars, UnivTVarsSet),
|
|
set.intersect(ExistQVarsSet, UnivTVarsSet, BadUnivTVarsSet),
|
|
maybe_report_badly_quantified_vars(PredInfo, universal_constraint,
|
|
set.to_sorted_list(BadUnivTVarsSet), !Specs),
|
|
constraint_list_get_tvars(ExistCs, ExistTVars),
|
|
list.delete_elems(ExistTVars, ExistQVars, BadExistTVars),
|
|
maybe_report_badly_quantified_vars(PredInfo, existential_constraint,
|
|
BadExistTVars, !Specs).
|
|
|
|
:- pred maybe_report_badly_quantified_vars(pred_info::in, quant_error_type::in,
|
|
list(tvar)::in, list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
maybe_report_badly_quantified_vars(PredInfo, QuantErrorType, TVars, !Specs) :-
|
|
(
|
|
TVars = []
|
|
;
|
|
TVars = [_ | _],
|
|
report_badly_quantified_vars(PredInfo, QuantErrorType, TVars, !Specs)
|
|
).
|
|
|
|
%---------------------------------------------------------------------------%
|
|
% Pass 7.
|
|
%---------------------------------------------------------------------------%
|
|
|
|
% Look for data constructors for which the existential type variables
|
|
% in the constraints are not all determined by the type variables
|
|
% in the constructor arguments and the functional dependencies.
|
|
%
|
|
:- pred check_typeclass_constraints_on_data_ctors(module_info::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_typeclass_constraints_on_data_ctors(ModuleInfo, !Specs) :-
|
|
module_info_get_type_table(ModuleInfo, TypeTable),
|
|
get_all_type_ctor_defns(TypeTable, TypeCtorsDefns),
|
|
list.foldl(check_typeclass_constraints_on_type_data_ctors(ModuleInfo),
|
|
TypeCtorsDefns, !Specs).
|
|
|
|
:- pred check_typeclass_constraints_on_type_data_ctors(module_info::in,
|
|
pair(type_ctor, hlds_type_defn)::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_typeclass_constraints_on_type_data_ctors(ModuleInfo, TypeCtor - TypeDefn,
|
|
!Specs) :-
|
|
get_type_defn_body(TypeDefn, Body),
|
|
(
|
|
Body = hlds_du_type(type_body_du(Ctors, _, _, _, _, _)),
|
|
list.foldl(
|
|
check_typeclass_constraints_on_data_ctor(ModuleInfo, TypeCtor,
|
|
TypeDefn),
|
|
one_or_more_to_list(Ctors), !Specs)
|
|
;
|
|
( Body = hlds_eqv_type(_)
|
|
; Body = hlds_foreign_type(_)
|
|
; Body = hlds_solver_type(_)
|
|
; Body = hlds_abstract_type(_)
|
|
)
|
|
).
|
|
|
|
:- pred check_typeclass_constraints_on_data_ctor(module_info::in,
|
|
type_ctor::in, hlds_type_defn::in, constructor::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
check_typeclass_constraints_on_data_ctor(ModuleInfo, TypeCtor, TypeDefn,
|
|
Ctor, !Specs) :-
|
|
Ctor = ctor(_, MaybeExistConstraints, _, CtorArgs, _, _),
|
|
(
|
|
MaybeExistConstraints = no_exist_constraints
|
|
;
|
|
MaybeExistConstraints = exist_constraints(ExistConstraints),
|
|
ExistConstraints = cons_exist_constraints(ExistQVars, Constraints,
|
|
_UnconstrainedQVars, _ConstrainedQVars),
|
|
|
|
module_info_get_class_table(ModuleInfo, ClassTable),
|
|
find_bad_class_ids_in_constraints(ClassTable, Constraints,
|
|
set.init, BadClassSNAsSet),
|
|
set.to_sorted_list(BadClassSNAsSet, BadClassSNAs),
|
|
(
|
|
BadClassSNAs = [HeadBadClassSNA | TailBadClassSNAs],
|
|
report_bad_class_ids_in_data_ctor(TypeCtor, TypeDefn,
|
|
HeadBadClassSNA, TailBadClassSNAs, !Specs)
|
|
;
|
|
BadClassSNAs = [],
|
|
% In the presence of any BadClassSNAs, the map.lookups done
|
|
% in the class table by code indirectly invoked by
|
|
% get_unbound_tvars could cause a compiler abort.
|
|
%
|
|
% Any ambiguity errors can be reported *after* the programmer
|
|
% fixes the references to nonexistent typeclasses.
|
|
get_ctor_arg_types(CtorArgs, ArgTypes),
|
|
type_vars_in_types(ArgTypes, ArgTVars),
|
|
list.filter((pred(V::in) is semidet :- list.member(V, ExistQVars)),
|
|
ArgTVars, ExistQArgTVars),
|
|
% Sanity check.
|
|
list.filter(list.contains(ExistQVars), ArgTVars, ExistQArgTVarsB),
|
|
expect(unify(ExistQArgTVars, ExistQArgTVarsB), $pred,
|
|
"ExistQArgTVars != ExistQArgTVarsB"),
|
|
constraint_list_get_tvars(Constraints, ConstrainedTVars),
|
|
get_type_defn_tvarset(TypeDefn, TVarSet),
|
|
get_unbound_tvars(ModuleInfo, TVarSet,
|
|
ExistQArgTVars, ConstrainedTVars, Constraints, UnboundTVars),
|
|
(
|
|
UnboundTVars = []
|
|
;
|
|
UnboundTVars = [_ | _],
|
|
report_unbound_tvars_in_ctor_context(UnboundTVars,
|
|
TypeCtor, TypeDefn, !Specs)
|
|
)
|
|
)
|
|
).
|
|
|
|
%---------------------------------------------------------------------------%
|
|
|
|
:- pred find_bad_class_ids_in_constraints(class_table::in,
|
|
list(prog_constraint)::in,
|
|
set(class_id)::in, set(class_id)::out) is det.
|
|
|
|
find_bad_class_ids_in_constraints(_, [], !BadClassIds).
|
|
find_bad_class_ids_in_constraints(ClassTable, [C | Cs], !BadClassIds) :-
|
|
find_bad_class_ids_in_constraint(ClassTable, C, !BadClassIds),
|
|
find_bad_class_ids_in_constraints(ClassTable, Cs, !BadClassIds).
|
|
|
|
:- pred find_bad_class_ids_in_constraint(class_table::in,
|
|
prog_constraint::in,
|
|
set(class_id)::in, set(class_id)::out) is det.
|
|
|
|
find_bad_class_ids_in_constraint(ClassTable, C, !BadClassIds) :-
|
|
C = constraint(ClassSymName, ArgTypes),
|
|
list.length(ArgTypes, Arity),
|
|
ClassId = class_id(ClassSymName, Arity),
|
|
( if map.search(ClassTable, ClassId, _) then
|
|
true
|
|
else
|
|
set.insert(ClassId, !BadClassIds)
|
|
).
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%---------------------------------------------------------------------------%
|
|
%
|
|
% Utility predicates.
|
|
%
|
|
|
|
:- pred get_unbound_tvars(module_info::in, tvarset::in, list(tvar)::in,
|
|
list(tvar)::in, list(prog_constraint)::in, list(tvar)::out) is det.
|
|
|
|
get_unbound_tvars(ModuleInfo, TVarSet, RootTVars, AllTVars, Constraints,
|
|
UnboundTVars) :-
|
|
module_info_get_class_table(ModuleInfo, ClassTable),
|
|
InducedFunDeps = compute_induced_fundeps(ClassTable, TVarSet, Constraints),
|
|
FunDepsClosure =
|
|
compute_fundeps_closure(InducedFunDeps, list_to_set(RootTVars)),
|
|
UnboundTVarsSet = set.difference(list_to_set(AllTVars), FunDepsClosure),
|
|
UnboundTVars = set.to_sorted_list(UnboundTVarsSet).
|
|
|
|
:- type induced_fundep
|
|
---> induced_fundep(
|
|
domain :: set(tvar),
|
|
range :: set(tvar)
|
|
).
|
|
|
|
:- func compute_induced_fundeps(class_table, tvarset, list(prog_constraint))
|
|
= list(induced_fundep).
|
|
|
|
compute_induced_fundeps(ClassTable, TVarSet, Constraints) = FunDeps :-
|
|
list.foldl(acc_induced_fundeps_for_constraint(ClassTable, TVarSet),
|
|
Constraints, [], FunDeps).
|
|
|
|
:- pred acc_induced_fundeps_for_constraint(class_table::in,
|
|
tvarset::in, prog_constraint::in,
|
|
list(induced_fundep)::in, list(induced_fundep)::out) is det.
|
|
|
|
acc_induced_fundeps_for_constraint(ClassTable, TVarSet, Constraint,
|
|
!FunDeps) :-
|
|
Constraint = constraint(Name, Args),
|
|
Arity = length(Args),
|
|
ClassDefn = map.lookup(ClassTable, class_id(Name, Arity)),
|
|
% The ancestors includes all superclasses of Constraint which have
|
|
% functional dependencies on them (possibly including Constraint itself).
|
|
ClassAncestors = ClassDefn ^ classdefn_fundep_ancestors,
|
|
(
|
|
% Optimize the common case.
|
|
ClassAncestors = []
|
|
;
|
|
ClassAncestors = [_ | _],
|
|
ClassTVarSet = ClassDefn ^ classdefn_tvarset,
|
|
ClassParams = ClassDefn ^ classdefn_vars,
|
|
|
|
% We can ignore the resulting tvarset, since any new variables
|
|
% will become bound when the arguments are bound. (This follows
|
|
% from the fact that constraints on class declarations can only use
|
|
% variables that appear in the head of the declaration.)
|
|
|
|
tvarset_merge_renaming(TVarSet, ClassTVarSet, _, Renaming),
|
|
apply_renaming_to_prog_constraints(Renaming,
|
|
ClassAncestors, RenamedAncestors),
|
|
apply_renaming_to_tvars(Renaming, ClassParams, RenamedParams),
|
|
map.from_corresponding_lists(RenamedParams, Args, Subst),
|
|
apply_subst_to_prog_constraints(Subst, RenamedAncestors, Ancestors),
|
|
list.foldl(induced_fundeps_3(ClassTable), Ancestors, !FunDeps)
|
|
).
|
|
|
|
:- pred induced_fundeps_3(class_table::in, prog_constraint::in,
|
|
list(induced_fundep)::in, list(induced_fundep)::out) is det.
|
|
|
|
induced_fundeps_3(ClassTable, Constraint, !FunDeps) :-
|
|
Constraint = constraint(ClassName, ClassArgs),
|
|
list.length(ClassArgs, ClassArity),
|
|
map.lookup(ClassTable, class_id(ClassName, ClassArity), ClassDefn),
|
|
ClassFunDeps = ClassDefn ^ classdefn_fundeps,
|
|
list.foldl(add_induced_fundep(ClassArgs), ClassFunDeps, !FunDeps).
|
|
|
|
:- pred add_induced_fundep(list(mer_type)::in, hlds_class_fundep::in,
|
|
list(induced_fundep)::in, list(induced_fundep)::out) is det.
|
|
|
|
add_induced_fundep(Args, fundep(Domain0, Range0), !FunDeps) :-
|
|
set.fold(induced_vars(Args), Domain0, set.init, Domain),
|
|
set.fold(induced_vars(Args), Range0, set.init, Range),
|
|
!:FunDeps = [induced_fundep(Domain, Range) | !.FunDeps].
|
|
|
|
:- pred induced_vars(list(mer_type)::in, int::in,
|
|
set(tvar)::in, set(tvar)::out) is det.
|
|
|
|
induced_vars(ArgTypes, ArgNum, !TVars) :-
|
|
ArgType = list.det_index1(ArgTypes, ArgNum),
|
|
type_vars_in_type(ArgType, ArgTVars),
|
|
set.list_to_set(ArgTVars, NewTVars),
|
|
set.union(NewTVars, !TVars).
|
|
|
|
:- func compute_fundeps_closure(list(induced_fundep), set(tvar)) = set(tvar).
|
|
|
|
compute_fundeps_closure(FunDeps, TVars0) = TVars :-
|
|
fundeps_closure_loop(FunDeps, TVars0, set.init, TVars).
|
|
|
|
:- pred fundeps_closure_loop(list(induced_fundep)::in, set(tvar)::in,
|
|
set(tvar)::in, set(tvar)::out) is det.
|
|
|
|
fundeps_closure_loop(FunDeps0, NewVars0, Result0, Result) :-
|
|
( if set.is_empty(NewVars0) then
|
|
Result = Result0
|
|
else
|
|
set.union(Result0, NewVars0, Result1),
|
|
list.map(remove_vars(NewVars0), FunDeps0, FunDeps1),
|
|
list.foldl2(collect_determined_vars, FunDeps1, [], FunDeps,
|
|
set.init, NewVars),
|
|
fundeps_closure_loop(FunDeps, NewVars, Result1, Result)
|
|
).
|
|
|
|
:- pred remove_vars(set(tvar)::in,
|
|
induced_fundep::in, induced_fundep::out) is det.
|
|
|
|
remove_vars(Vars, InducedFunDep0, InducedFunDep) :-
|
|
InducedFunDep0 = induced_fundep(Domain0, Range0),
|
|
set.difference(Domain0, Vars, Domain),
|
|
set.difference(Range0, Vars, Range),
|
|
InducedFunDep = induced_fundep(Domain, Range).
|
|
|
|
:- pred collect_determined_vars(induced_fundep::in,
|
|
list(induced_fundep)::in, list(induced_fundep)::out,
|
|
set(tvar)::in, set(tvar)::out) is det.
|
|
|
|
collect_determined_vars(FunDep, !FunDeps, !Vars) :-
|
|
FunDep = induced_fundep(Domain, Range),
|
|
( if set.is_empty(Domain) then
|
|
set.union(Range, !Vars)
|
|
else
|
|
!:FunDeps = [FunDep | !.FunDeps]
|
|
).
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%---------------------------------------------------------------------------%
|
|
%
|
|
% Error reporting.
|
|
%
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%
|
|
% Error reports from pass 1.
|
|
%
|
|
|
|
% The error message for cyclic classes is intended to look like this:
|
|
%
|
|
% module.m:NNN: Error: cyclic superclass relation detected:
|
|
% module.m:NNN: `foo/N'
|
|
% module.m:NNN: <= `bar/N'
|
|
% module.m:NNN: <= `baz/N'
|
|
% module.m:NNN: <= `foo/N'
|
|
%
|
|
:- pred report_cyclic_classes(class_table::in, class_path::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_cyclic_classes(ClassTable, ClassPath, !Specs) :-
|
|
ClassPath = class_path(ClassPathClassIds),
|
|
(
|
|
ClassPathClassIds = [],
|
|
unexpected($pred, "empty cycle found.")
|
|
;
|
|
ClassPathClassIds = [HeadClassId | TailClassIds],
|
|
Context = map.lookup(ClassTable, HeadClassId) ^ classdefn_context,
|
|
StartPieces =
|
|
[words("Error:")] ++
|
|
color_as_incorrect([words("cyclic superclass relation")]) ++
|
|
[words("detected:"), nl],
|
|
FirstLine = [qual_class_id(HeadClassId), nl],
|
|
list.foldl(add_path_element, TailClassIds, cord.init, LaterLinesCord),
|
|
CycleLines = FirstLine ++ cord.list(LaterLinesCord),
|
|
Pieces = StartPieces ++ color_as_subject(CycleLines),
|
|
Spec = spec($pred, severity_error, phase_type_check, Context, Pieces),
|
|
!:Specs = [Spec | !.Specs]
|
|
).
|
|
|
|
:- pred add_path_element(class_id::in,
|
|
cord(format_piece)::in, cord(format_piece)::out) is det.
|
|
|
|
add_path_element(ClassId, !LaterLines) :-
|
|
Line = [words("<="), qual_class_id(ClassId), nl],
|
|
cord.snoc_list(Line, !LaterLines).
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%
|
|
% Error reports from pass 2.
|
|
%
|
|
|
|
:- pred report_badly_formed_type_in_instance(class_id::in,
|
|
hlds_instance_defn::in, type_ctor::in, int::in,
|
|
pair(int, mer_type)::in, assoc_list(int, mer_type)::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_badly_formed_type_in_instance(ClassId, InstanceDefn, TypeCtor, ArgNum,
|
|
HeadNonTVarArg, TailNonTVarArgs, !Specs) :-
|
|
TVarSet = InstanceDefn ^ instdefn_tvarset,
|
|
NonTVarArgs = [HeadNonTVarArg | TailNonTVarArgs],
|
|
NotTVar = words(choose_number(NonTVarArgs,
|
|
"is not a type variable,", "are not type variables")),
|
|
IsAre = words(choose_number(NonTVarArgs, "This is", "These are")),
|
|
NonTVarArgPieces = non_tvar_args_to_pieces(TVarSet,
|
|
HeadNonTVarArg, TailNonTVarArgs),
|
|
EndPieces = [words("in the"), nth_fixed(ArgNum), words("instance type,"),
|
|
words(choose_number(NonTVarArgs, "one", "some")),
|
|
words("of the arguments of the type constructor"),
|
|
unqual_type_ctor(TypeCtor)] ++
|
|
color_as_incorrect([NotTVar]) ++
|
|
color_as_correct([words("but should be.")]) ++
|
|
[IsAre | NonTVarArgPieces] ++ [nl],
|
|
report_bad_type_in_instance(ClassId, InstanceDefn, EndPieces,
|
|
badly_formed, !Specs).
|
|
|
|
:- func non_tvar_args_to_pieces(tvarset,
|
|
pair(int, mer_type), assoc_list(int, mer_type)) = list(format_piece).
|
|
|
|
non_tvar_args_to_pieces(TVarSet, HeadArgNumType, TailArgNumTypes) = Pieces :-
|
|
HeadArgNumType = ArgNum - ArgType,
|
|
TypeStr = mercury_type_to_string(TVarSet, print_name_only, ArgType),
|
|
TheNthArgPieces = [words("the"), nth_fixed(ArgNum), words("argument,")],
|
|
TypePieces = [quote(TypeStr)],
|
|
(
|
|
TailArgNumTypes = [],
|
|
Pieces = TheNthArgPieces ++
|
|
color_as_subject(TypePieces ++ [suffix(".")])
|
|
;
|
|
TailArgNumTypes = [HeadTailArgNumType | TailTailArgNumTypes],
|
|
(
|
|
TailTailArgNumTypes = [],
|
|
MaybeComma = [],
|
|
MaybeAnd = [words("and")]
|
|
;
|
|
TailTailArgNumTypes = [_ | _],
|
|
MaybeComma = [suffix(",")],
|
|
MaybeAnd = []
|
|
),
|
|
HeadTypePieces = TheNthArgPieces ++
|
|
color_as_subject(TypePieces ++ MaybeComma),
|
|
TailTypePieces = non_tvar_args_to_pieces(TVarSet,
|
|
HeadTailArgNumType, TailTailArgNumTypes),
|
|
Pieces = HeadTypePieces ++ MaybeAnd ++ TailTypePieces
|
|
).
|
|
|
|
:- pred report_eqv_type_in_abstract_exported_instance(class_id::in,
|
|
hlds_instance_defn::in, int::in, mer_type::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_eqv_type_in_abstract_exported_instance(ClassId, InstanceDefn, ArgNum,
|
|
Type, !Specs) :-
|
|
TVarSet = InstanceDefn ^ instdefn_tvarset,
|
|
TypeStr = mercury_type_to_string(TVarSet, print_name_only, Type),
|
|
EndPieces = [words("the"), nth_fixed(ArgNum), words("instance type,")] ++
|
|
color_as_subject([quote(TypeStr), suffix(",")]) ++
|
|
[words("is an")] ++
|
|
color_as_incorrect([words("abstract exported equivalence type.")]) ++
|
|
[nl],
|
|
% XXX Should we add this explanatory text? If so, should it be
|
|
% in a verbose-only component? Should we add it to reference manual?
|
|
% [words("Mercury does not allow this, because allowing it"),
|
|
% words("would mean that the code inside the module"),
|
|
% words("(which can see the equivalence)"),
|
|
% words("and the code outside the module"),
|
|
% words("(which cannot)"),
|
|
% words("would disagree about the identity of the type.")]
|
|
report_bad_type_in_instance(ClassId, InstanceDefn, EndPieces,
|
|
abstract_exported_eqv, !Specs).
|
|
|
|
:- type bad_instance_type_kind
|
|
---> badly_formed
|
|
; abstract_exported_eqv.
|
|
|
|
:- pred report_bad_type_in_instance(class_id::in, hlds_instance_defn::in,
|
|
list(format_piece)::in, bad_instance_type_kind::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_bad_type_in_instance(ClassId, InstanceDefn, EndPieces, Kind, !Specs) :-
|
|
(
|
|
Kind = badly_formed,
|
|
% We are generating the error message because the type is badly formed
|
|
% as expanded. The unexpanded version may be correctly formed.
|
|
WhichTypes = cur_types
|
|
;
|
|
Kind = abstract_exported_eqv,
|
|
% Messages about the expanded type being an equivalence type
|
|
% would not make sense.
|
|
WhichTypes = orig_types
|
|
),
|
|
PrefixPieces = in_instance_decl_pieces(WhichTypes, ClassId, InstanceDefn),
|
|
Pieces = PrefixPieces ++ [words("error:")] ++ EndPieces,
|
|
Context = InstanceDefn ^ instdefn_context,
|
|
Spec = spec($pred, severity_error, phase_type_check, Context, Pieces),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%
|
|
% Error reports from pass 3.
|
|
%
|
|
|
|
% Duplicate method definition error.
|
|
%
|
|
:- pred report_duplicate_method_defn(class_id::in, hlds_instance_defn::in,
|
|
pred_pf_name_arity::in, prog_context::in, prog_context::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_duplicate_method_defn(ClassId, InstanceDefn, MethodName,
|
|
FirstContext, LaterContext, !Specs) :-
|
|
PrefixPieces = in_instance_decl_pieces(cur_types, ClassId, InstanceDefn),
|
|
PFMethodNamePieces = pf_method_name_pieces(MethodName),
|
|
HeaderPieces = PrefixPieces ++ [words("error: this instance has")] ++
|
|
color_as_incorrect([words("multiple implementations")]) ++
|
|
[words("of")] ++
|
|
color_as_subject(PFMethodNamePieces ++ [suffix(".")]) ++
|
|
[nl],
|
|
HeaderMsg = msg(InstanceDefn ^ instdefn_context, HeaderPieces),
|
|
FirstPieces = [words("First definition appears here."), nl],
|
|
FirstMsg = msg(FirstContext, FirstPieces),
|
|
LaterPieces = [words("Later definition appears here."), nl],
|
|
LaterMsg = msg(LaterContext, LaterPieces),
|
|
Spec = error_spec($pred, severity_error, phase_type_check,
|
|
[HeaderMsg, FirstMsg, LaterMsg]),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
%---------------------------------------------------------------------------%
|
|
|
|
:- pred report_undefined_method(class_id::in, hlds_instance_defn::in,
|
|
pred_pf_name_arity::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_undefined_method(ClassId, InstanceDefn, MethodName, !Specs) :-
|
|
PrefixPieces = in_instance_decl_pieces(cur_types, ClassId, InstanceDefn),
|
|
PFMethodNamePieces = pf_method_name_pieces(MethodName),
|
|
Pieces = PrefixPieces ++ [words("error: this instance has")] ++
|
|
color_as_incorrect([words("no implementation")]) ++
|
|
[words("for")] ++
|
|
color_as_subject(PFMethodNamePieces ++ [suffix(".")]) ++
|
|
[nl],
|
|
Context = InstanceDefn ^ instdefn_context,
|
|
Spec = spec($pred, severity_error, phase_type_check, Context, Pieces),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
%---------------------------------------------------------------------------%
|
|
|
|
:- pred report_unknown_instance_methods(class_id::in, hlds_instance_defn::in,
|
|
instance_method::in, list(instance_method)::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_unknown_instance_methods(ClassId, InstanceDefn,
|
|
HeadMethod, TailMethods, !Specs) :-
|
|
PrefixPieces = in_instance_decl_pieces(cur_types, ClassId, InstanceDefn),
|
|
(
|
|
TailMethods = [],
|
|
HeadMethod = instance_method(MethodName, _Defn, HeadMethodContext),
|
|
% If we have a context for the specific incorrect method, use it.
|
|
SelectedContext = HeadMethodContext,
|
|
MethodName = pred_pf_name_arity(PredOrFunc, MethodSymName, UserArity),
|
|
UserArity = user_arity(UserArityInt),
|
|
SNA = sym_name_arity(MethodSymName, UserArityInt),
|
|
Pieces = PrefixPieces ++ [words("error: the type class")] ++
|
|
color_as_incorrect([words("has no"), p_or_f(PredOrFunc),
|
|
words("method")]) ++
|
|
[words("named")] ++
|
|
color_as_subject([unqual_sym_name_arity(SNA), suffix(".")]) ++
|
|
[nl]
|
|
;
|
|
TailMethods = [_ | _],
|
|
SelectedContext = InstanceDefn ^ instdefn_context,
|
|
MethodPieces =
|
|
list.map(method_name_pieces, [HeadMethod | TailMethods]),
|
|
Pieces = PrefixPieces ++ [words("error: the type class")] ++
|
|
color_as_incorrect([words("has none of these methods:")]) ++
|
|
[nl_indent_delta(1)] ++
|
|
pieces_list_to_color_line_pieces(color_subject, [suffix(".")],
|
|
MethodPieces) ++
|
|
[nl_indent_delta(-1)]
|
|
),
|
|
Spec = spec($pred, severity_error, phase_type_check,
|
|
SelectedContext, Pieces),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
%---------------------%
|
|
|
|
:- pred report_unsatistfied_superclass_constraint(class_id::in,
|
|
hlds_instance_defn::in, tvarset::in, list(hlds_constraint)::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_unsatistfied_superclass_constraint(ClassId, InstanceDefn, ClassTVarSet,
|
|
UnprovenConstraints, !Specs) :-
|
|
PrefixPieces = in_instance_decl_pieces(cur_types, ClassId, InstanceDefn),
|
|
list.map(constraint_to_pieces(ClassTVarSet),
|
|
UnprovenConstraints, ConstraintPieceLists),
|
|
Pieces = PrefixPieces ++
|
|
[words("error: the following superclass"),
|
|
words(choose_number(UnprovenConstraints,
|
|
"constraint is", "constraints are"))] ++
|
|
color_as_incorrect([words("not satisfied:")]) ++
|
|
[nl_indent_delta(1)] ++
|
|
pieces_list_to_color_line_pieces(color_subject, [suffix(".")],
|
|
ConstraintPieceLists) ++
|
|
[nl_indent_delta(-1)],
|
|
Context = InstanceDefn ^ instdefn_context,
|
|
Spec = spec($pred, severity_error, phase_type_check, Context, Pieces),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
:- pred constraint_to_pieces(tvarset::in, hlds_constraint::in,
|
|
list(format_piece)::out) is det.
|
|
|
|
constraint_to_pieces(TVarSet, Constraint, Pieces) :-
|
|
retrieve_prog_constraint(Constraint, ProgConstraint),
|
|
ConstraintStr = mercury_constraint_to_string(TVarSet, print_name_only,
|
|
ProgConstraint),
|
|
Pieces = [quote(ConstraintStr)].
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%
|
|
% Error reports from pass 4.
|
|
%
|
|
|
|
:- pred report_overlapping_instances(class_id::in,
|
|
prog_context::in, prog_context::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_overlapping_instances(ClassId, ContextA, ContextB, !Specs) :-
|
|
PiecesA = [words("Error:")] ++
|
|
color_as_incorrect([words("overlapping instance declarations")]) ++
|
|
[words("for class")] ++
|
|
color_as_subject([qual_class_id(ClassId), suffix(".")]) ++
|
|
[nl, words("One instance declaration is here, ..."), nl],
|
|
MsgA = msg(ContextA, PiecesA),
|
|
PiecesB = [words("... and the other is here."), nl],
|
|
MsgB = msg(ContextB, PiecesB),
|
|
Spec = error_spec($pred, severity_error, phase_type_check, [MsgA, MsgB]),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
:- pred report_any_duplicate_instance_defns_in_category(class_id::in,
|
|
spec_severity::in, string::in, string::in,
|
|
list(hlds_instance_defn)::in, maybe(hlds_instance_defn)::out,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_any_duplicate_instance_defns_in_category(ClassId, Severity,
|
|
SeverityWord, Category, InstanceDefns, MaybeInstanceDefn, !Specs) :-
|
|
(
|
|
InstanceDefns = [],
|
|
MaybeInstanceDefn = no
|
|
;
|
|
InstanceDefns = [FirstInstanceDefn | LaterInstanceDefns],
|
|
MaybeInstanceDefn = yes(FirstInstanceDefn),
|
|
FirstContext = FirstInstanceDefn ^ instdefn_context,
|
|
list.foldl(
|
|
report_duplicate_instance_defn(ClassId, Severity, SeverityWord,
|
|
Category, FirstContext),
|
|
LaterInstanceDefns, !Specs)
|
|
).
|
|
|
|
:- pred report_duplicate_instance_defn(class_id::in, spec_severity::in,
|
|
string::in, string::in, prog_context::in, hlds_instance_defn::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_duplicate_instance_defn(ClassId, Severity, SeverityWord, Category,
|
|
FirstContext, LaterInstanceDefn, !Specs) :-
|
|
LaterContext = LaterInstanceDefn ^ instdefn_context,
|
|
LaterPieces = [words(SeverityWord), suffix(":")] ++
|
|
color_as_incorrect([words("duplicate"), words(Category),
|
|
words("instance declaration")]) ++
|
|
[words("for class")] ++
|
|
color_as_subject([qual_class_id(ClassId), suffix(".")]) ++ [nl],
|
|
LaterMsg = msg(LaterContext, LaterPieces),
|
|
FirstPieces = [words("Previous instance declaration was here."), nl],
|
|
FirstMsg = msg(FirstContext, FirstPieces),
|
|
Spec = error_spec($pred, Severity, phase_type_check, [LaterMsg, FirstMsg]),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
:- pred report_abstract_concrete_constraints_mismatch(class_id::in,
|
|
hlds_instance_defn::in, hlds_instance_defn::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_abstract_concrete_constraints_mismatch(ClassId,
|
|
AbstractInstanceDefn, ConcreteInstanceDefn, !Specs) :-
|
|
ConcreteContext = ConcreteInstanceDefn ^ instdefn_context,
|
|
AbstractContext = AbstractInstanceDefn ^ instdefn_context,
|
|
AbstractPieces = [words("Error: the instance constraints on this")] ++
|
|
color_as_subject([words("abstract instance declaration"),
|
|
words("for class"), qual_class_id(ClassId)]) ++
|
|
color_as_incorrect([words("do not match")]) ++
|
|
[words("the instance constraints on the corresponding"),
|
|
words("concrete instance declaration."), nl],
|
|
AbstractMsg = msg(AbstractContext, AbstractPieces),
|
|
ConcretePieces = [words("The corresponding"),
|
|
words("concrete instance declaration is here."), nl],
|
|
ConcreteMsg = msg(ConcreteContext, ConcretePieces),
|
|
Spec = error_spec($pred, severity_error, phase_type_check,
|
|
[AbstractMsg, ConcreteMsg]),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
:- pred report_abstract_instance_without_concrete(class_id::in,
|
|
hlds_instance_defn::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_abstract_instance_without_concrete(ClassId, InstanceDefn, !Specs) :-
|
|
ClassId = class_id(ClassName, _),
|
|
ClassNameString = sym_name_to_string(ClassName),
|
|
Types = InstanceDefn ^ instdefn_types,
|
|
TVarSet = InstanceDefn ^ instdefn_tvarset,
|
|
TypesStr = mercury_types_to_string(TVarSet, print_name_only, Types),
|
|
string.format("%s(%s)", [s(ClassNameString), s(TypesStr)], InstanceName),
|
|
Pieces = [words("Error: this")] ++
|
|
color_as_subject([words("abstract instance declaration"),
|
|
words("for"), quote(InstanceName)]) ++
|
|
color_as_incorrect([words("has no corresponding"),
|
|
words("concrete instance declaration")]) ++
|
|
[words("in the implementation section."), nl],
|
|
Context = InstanceDefn ^ instdefn_context,
|
|
Spec = spec($pred, severity_error, phase_type_check,
|
|
Context, Pieces),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
:- pred report_local_vs_nonlocal_clash(class_id::in,
|
|
hlds_instance_defn::in, hlds_instance_defn::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_local_vs_nonlocal_clash(ClassId, LocalInstance, NonLocalInstance,
|
|
!Specs) :-
|
|
InstanceName = instance_name(orig_types, print_all_types, keep_all,
|
|
set_default_func, ClassId, LocalInstance),
|
|
% XXX Should we mention any constraints on the instance declaration?
|
|
LocalPieces = [words("Error: this")] ++
|
|
color_as_subject([words("instance declaration for"),
|
|
quote(InstanceName)]) ++
|
|
color_as_incorrect([words("clashes")]) ++
|
|
[words("with an instance declaration in another module."), nl],
|
|
LocalContext = LocalInstance ^ instdefn_context,
|
|
LocalMsg = msg(LocalContext, LocalPieces),
|
|
NonLocalPieces = [words("The other instance declaration is here."), nl],
|
|
NonLocalContext = NonLocalInstance ^ instdefn_context,
|
|
NonLocalMsg = msg(NonLocalContext, NonLocalPieces),
|
|
Spec = error_spec($pred, severity_error, phase_type_check,
|
|
[LocalMsg, NonLocalMsg]),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%
|
|
% Error reports from pass 5.
|
|
%
|
|
|
|
% The coverage error message is intended to look like this:
|
|
%
|
|
% long_module_name:001: In instance for typeclass `long_class/2':
|
|
% long_module_name:001: functional dependency not satisfied: type
|
|
% long_module_name:001: variables T1, T2 and T3 occur in the range of a
|
|
% long_module_name:001: functional dependency, but are not determined
|
|
% long_module_name:001: by the domain.
|
|
%
|
|
:- pred report_coverage_error(class_id::in, hlds_instance_defn::in,
|
|
list(tvar)::in, list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_coverage_error(ClassId, InstanceDefn, Vars, !Specs) :-
|
|
TVarSet = InstanceDefn ^ instdefn_tvarset,
|
|
TypeVars = choose_number(Vars, "type variable", "type variables"),
|
|
VarPieces = list.map(var_to_quote_piece(TVarSet), Vars),
|
|
VarsPieces = piece_list_to_color_pieces(color_subject, "and", [],
|
|
VarPieces),
|
|
Pieces = [words("In instance for typeclass"),
|
|
unqual_class_id(ClassId), suffix(":"), nl,
|
|
words("error: functional dependency not satisfied:"),
|
|
words(TypeVars)] ++
|
|
VarsPieces ++
|
|
[words(choose_number(Vars, "occurs", "occur")),
|
|
words("in the range of the functional dependency, but")] ++
|
|
color_as_incorrect([words(choose_number(Vars, "is", "are")),
|
|
words("not determined by the domain.")]) ++ [nl],
|
|
Context = InstanceDefn ^ instdefn_context,
|
|
Spec = spec($pred, severity_error, phase_type_check, Context, Pieces),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
:- pred report_consistency_error(class_id::in, hlds_class_defn::in,
|
|
hlds_instance_defn::in, hlds_instance_defn::in, hlds_class_fundep::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_consistency_error(ClassId, ClassDefn, InstanceA, InstanceB, FunDep,
|
|
!Specs) :-
|
|
Params = ClassDefn ^ classdefn_vars,
|
|
TVarSet = ClassDefn ^ classdefn_tvarset,
|
|
ContextA = InstanceA ^ instdefn_context,
|
|
ContextB = InstanceB ^ instdefn_context,
|
|
|
|
FunDep = fundep(Domain, Range),
|
|
DomainParams = restrict_list_elements(Domain, Params),
|
|
RangeParams = restrict_list_elements(Range, Params),
|
|
Domains = mercury_vars_to_name_only_vs(TVarSet, DomainParams),
|
|
Ranges = mercury_vars_to_name_only_vs(TVarSet, RangeParams),
|
|
|
|
% XXX This should give the specific details of the inconsistency.
|
|
PiecesA = [words("Error:")] ++
|
|
color_as_incorrect([words("inconsistent instance declaration")]) ++
|
|
[words("for typeclass")] ++
|
|
color_as_subject([qual_class_id(ClassId)]) ++
|
|
[words("with functional dependency"),
|
|
quote("(" ++ Domains ++ " -> " ++ Ranges ++ ")"), suffix("."), nl],
|
|
PiecesB = [words("Here is the conflicting instance."), nl],
|
|
|
|
MsgA = msg(ContextA, PiecesA),
|
|
MsgB = error_msg(yes(ContextB), always_treat_as_first, 0u,
|
|
[always(PiecesB)]),
|
|
Spec = error_spec($pred, severity_error, phase_type_check, [MsgA, MsgB]),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%
|
|
% Error reports from pass 6.
|
|
%
|
|
|
|
% The error messages for ambiguous types are intended to look like this:
|
|
%
|
|
% long_module_name:001: In declaration for function `long_function/2':
|
|
% long_module_name:001: error in type class constraints: type variables
|
|
% long_module_name:001: T1, T2 and T3 occur in the constraints, but are
|
|
% long_module_name:001: not determined by the function's argument or
|
|
% long_module_name:001: result types.
|
|
%
|
|
% long_module_name:002: In declaration for predicate `long_predicate/3':
|
|
% long_module_name:002: error in type class constraints: type variable
|
|
% long_module_name:002: T occurs in the constraints, but is not
|
|
% long_module_name:002: determined by the predicate's argument types.
|
|
%
|
|
% long_module_name:002: In declaration for type `long_type/3':
|
|
% long_module_name:002: error in type class constraints: type variable
|
|
% long_module_name:002: T occurs in the constraints, but is not
|
|
% long_module_name:002: determined by the constructor's argument types.
|
|
%
|
|
:- pred report_unbound_tvars_in_pred_context(pred_info::in, list(tvar)::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_unbound_tvars_in_pred_context(PredInfo, Vars, !Specs) :-
|
|
pred_info_get_context(PredInfo, Context),
|
|
pred_info_get_arg_types(PredInfo, TVarSet, _, ArgTypes),
|
|
PredName = pred_info_name(PredInfo),
|
|
Module = pred_info_module(PredInfo),
|
|
SymName = qualified(Module, PredName),
|
|
PredFormArity = arg_list_arity(ArgTypes),
|
|
PredOrFunc = pred_info_is_pred_or_func(PredInfo),
|
|
|
|
|
|
PFSymNameArity = pf_sym_name_arity(PredOrFunc, SymName, PredFormArity),
|
|
TypeVars = choose_number(Vars, "type variable", "type variables"),
|
|
VarPieces = list.map(var_to_quote_piece(TVarSet), Vars),
|
|
VarsPieces = piece_list_to_color_pieces(color_subject, "and", [],
|
|
VarPieces),
|
|
Pieces0 = [words("In declaration for"),
|
|
unqual_pf_sym_name_pred_form_arity(PFSymNameArity), suffix(":"), nl,
|
|
words("error in type class constraints:"), words(TypeVars)] ++
|
|
VarsPieces ++
|
|
[words(choose_number(Vars, "occurs", "occur")),
|
|
words("in the constraints, but")] ++
|
|
color_as_incorrect([words(choose_number(Vars, "is", "are")),
|
|
words("not determined")]) ++
|
|
[words("by the")],
|
|
(
|
|
PredOrFunc = pf_predicate,
|
|
Pieces = Pieces0 ++ [words("predicate's argument types."), nl]
|
|
;
|
|
PredOrFunc = pf_function,
|
|
Pieces = Pieces0 ++ [words("function's argument or result types."), nl]
|
|
),
|
|
Msg = simple_msg(Context,
|
|
[always(Pieces),
|
|
verbose_only(verbose_once, unbound_tvars_explanation_pieces)]),
|
|
Spec = error_spec($pred, severity_error, phase_type_check, [Msg]),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
:- pred report_bad_class_ids_in_pred_decl(module_info::in, pred_info::in,
|
|
class_id::in, list(class_id)::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_bad_class_ids_in_pred_decl(ModuleInfo, PredInfo,
|
|
HeadBadClassId, TailBadClassIds, !Specs) :-
|
|
pred_info_get_context(PredInfo, Context),
|
|
PredOrFunc = pred_info_is_pred_or_func(PredInfo),
|
|
PredName = pred_info_name(PredInfo),
|
|
PredModuleName = pred_info_module(PredInfo),
|
|
module_info_get_name(ModuleInfo, CurModuleName),
|
|
% Module qualify the name of the predicate we are complaining about
|
|
% *only* if it is not from the current module.
|
|
( if CurModuleName = PredModuleName then
|
|
PredSymName = unqualified(PredName)
|
|
else
|
|
PredSymName = qualified(PredModuleName, PredName)
|
|
),
|
|
pred_info_get_arg_types(PredInfo, _TVarSet, _, ArgTypes),
|
|
PredFormArity = arg_list_arity(ArgTypes),
|
|
PFSymNameArity = pf_sym_name_arity(PredOrFunc, PredSymName, PredFormArity),
|
|
StartPieces = [words("In declaration for"),
|
|
unqual_pf_sym_name_pred_form_arity(PFSymNameArity), suffix(":"), nl],
|
|
Pieces = StartPieces ++
|
|
error_classes_do_not_exist_pieces(HeadBadClassId, TailBadClassIds),
|
|
Spec = spec($pred, severity_error, phase_type_check, Context, Pieces),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
:- type quant_error_type
|
|
---> universal_constraint
|
|
; existential_constraint.
|
|
|
|
:- pred report_badly_quantified_vars(pred_info::in, quant_error_type::in,
|
|
list(tvar)::in, list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_badly_quantified_vars(PredInfo, QuantErrorType, TVars, !Specs) :-
|
|
pred_info_get_typevarset(PredInfo, TVarSet),
|
|
pred_info_get_context(PredInfo, Context),
|
|
InDeclaration = [words("In declaration of")] ++
|
|
describe_one_pred_info_name(no, should_module_qualify, [], PredInfo) ++
|
|
[suffix(":"), nl],
|
|
TypeVariables =
|
|
[words(choose_number(TVars, "type variable", "type variables"))],
|
|
TVarPieces = list.map(var_to_quote_piece(TVarSet), TVars),
|
|
TVarsPieces = piece_list_to_color_pieces(color_subject, "and", [],
|
|
TVarPieces),
|
|
Are = words(choose_number(TVars, "is", "are")),
|
|
(
|
|
QuantErrorType = universal_constraint,
|
|
BlahConstrained = words("universally constrained"),
|
|
BlahQuantified = words("existentially quantified")
|
|
;
|
|
QuantErrorType = existential_constraint,
|
|
BlahConstrained = words("existentially constrained"),
|
|
BlahQuantified = words("universally quantified")
|
|
),
|
|
Pieces = InDeclaration ++ [words("error:")] ++
|
|
TypeVariables ++ TVarsPieces ++ [Are] ++
|
|
color_as_inconsistent([BlahConstrained, suffix(",")]) ++
|
|
[words("but"), Are] ++
|
|
color_as_inconsistent([BlahQuantified, suffix(".")]) ++ [nl],
|
|
Spec = spec($pred, severity_error, phase_type_check, Context, Pieces),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%
|
|
% Error reports from pass 7.
|
|
%
|
|
|
|
:- pred report_unbound_tvars_in_ctor_context(list(tvar)::in, type_ctor::in,
|
|
hlds_type_defn::in, list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_unbound_tvars_in_ctor_context(Vars, TypeCtor, TypeDefn, !Specs) :-
|
|
get_type_defn_context(TypeDefn, Context),
|
|
get_type_defn_tvarset(TypeDefn, TVarSet),
|
|
TypeVars = choose_number(Vars, "type variable", "type variables"),
|
|
VarPieces = list.map(var_to_quote_piece(TVarSet), Vars),
|
|
VarsPieces = piece_list_to_color_pieces(color_subject, "and", [],
|
|
VarPieces),
|
|
Pieces = [words("In declaration for type"),
|
|
unqual_type_ctor(TypeCtor), suffix(":"), nl,
|
|
words("error in type class constraints:"), words(TypeVars)] ++
|
|
VarsPieces ++
|
|
[words(choose_number(Vars, "occurs", "occur")),
|
|
words("in the constraints, but")] ++
|
|
color_as_incorrect([words(choose_number(Vars, "is", "are")),
|
|
words("not determined")]) ++
|
|
[words("by the constructor's argument types."), nl],
|
|
Msg = simple_msg(Context,
|
|
[always(Pieces),
|
|
verbose_only(verbose_once, unbound_tvars_explanation_pieces)]),
|
|
Spec = error_spec($pred, severity_error, phase_type_check, [Msg]),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
:- pred report_bad_class_ids_in_data_ctor(type_ctor::in,
|
|
hlds_type_defn::in, class_id::in, list(class_id)::in,
|
|
list(error_spec)::in, list(error_spec)::out) is det.
|
|
|
|
report_bad_class_ids_in_data_ctor(TypeCtor, TypeDefn,
|
|
HeadBadClassId, TailBadClassIds, !Specs) :-
|
|
get_type_defn_context(TypeDefn, Context),
|
|
StartPieces = [words("In declaration for type"), qual_type_ctor(TypeCtor),
|
|
suffix(":"), nl],
|
|
Pieces = StartPieces ++
|
|
error_classes_do_not_exist_pieces(HeadBadClassId, TailBadClassIds),
|
|
Spec = spec($pred, severity_error, phase_type_check,
|
|
Context, Pieces),
|
|
!:Specs = [Spec | !.Specs].
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%
|
|
% Utility predicates for error reporting.
|
|
%
|
|
|
|
:- func unbound_tvars_explanation_pieces = list(format_piece).
|
|
|
|
unbound_tvars_explanation_pieces =
|
|
[words("All types occurring in typeclass constraints"),
|
|
words("must be fully determined."),
|
|
words("A type is fully determined if one of the"),
|
|
words("following holds:"),
|
|
nl,
|
|
words("1) All type variables occurring in the type"),
|
|
words("are determined."),
|
|
nl,
|
|
words("2) The type occurs in a constraint argument,"),
|
|
words("that argument is in the range of some"),
|
|
words("functional dependency for that class, and"),
|
|
words("the types in all of the domain arguments for"),
|
|
words("that functional dependency are fully determined."),
|
|
nl,
|
|
words("A type variable is determined if one of the"),
|
|
words("following holds:"),
|
|
nl,
|
|
words("1) The type variable occurs in the argument"),
|
|
words("types of the predicate, function, or"),
|
|
words("constructor which is constrained."),
|
|
nl,
|
|
words("2) The type variable occurs in a type which"),
|
|
words("is fully determined."),
|
|
nl,
|
|
words("See the ""Functional dependencies"" section"),
|
|
words("of the reference manual for details."), nl].
|
|
|
|
:- func error_classes_do_not_exist_pieces(class_id, list(class_id)) =
|
|
list(format_piece).
|
|
|
|
error_classes_do_not_exist_pieces(HeadClassId, TailClassIds) = Pieces :-
|
|
WrapQualClassId = (func(ClassId) = qual_class_id(ClassId)),
|
|
QualHeadClassId = WrapQualClassId(HeadClassId),
|
|
(
|
|
TailClassIds = [],
|
|
Pieces = [words("error: the type class")] ++
|
|
color_as_subject([QualHeadClassId]) ++
|
|
color_as_incorrect([words("does not exist.")]) ++ [nl]
|
|
;
|
|
TailClassIds = [_ | _],
|
|
QualTailClassIds = list.map(WrapQualClassId, TailClassIds),
|
|
QualClassIds = [QualHeadClassId | QualTailClassIds],
|
|
Pieces = [words("error: the type classes")] ++
|
|
piece_list_to_color_pieces(color_subject, "and", [],
|
|
QualClassIds) ++
|
|
color_as_incorrect([words("do not exist.")]) ++ [nl]
|
|
).
|
|
|
|
%---------------------%
|
|
|
|
:- func in_instance_decl_pieces(which_types, class_id, hlds_instance_defn)
|
|
= list(format_piece).
|
|
|
|
in_instance_decl_pieces(WhichTypes, ClassId, InstanceDefn) = Pieces :-
|
|
% We are printing types only to make the context more easily recognizable.
|
|
% The disambiguating power of module qualifiers is not needed in that role,
|
|
% and their only contribution to the error message would be clutter.
|
|
%
|
|
% Likewise, we could mention the constraints (if any) on the
|
|
% instance declaration, but it would also be very likely to be clutter.
|
|
InstanceName = instance_name(WhichTypes, print_few_types, delete_all,
|
|
set_default_func, ClassId, InstanceDefn),
|
|
Pieces = [words("In instance declaration for"),
|
|
words_quote(InstanceName), suffix(":"), nl].
|
|
|
|
:- type which_types
|
|
---> orig_types
|
|
; cur_types.
|
|
|
|
:- type type_limit
|
|
---> print_all_types
|
|
; print_few_types.
|
|
|
|
:- type module_quals
|
|
---> keep_all
|
|
; delete_builtin
|
|
; delete_all.
|
|
|
|
:- func instance_name(which_types, type_limit, module_quals,
|
|
maybe_set_default_func, class_id, hlds_instance_defn) = string.
|
|
|
|
instance_name(WhichTypes, Limit, Quals, SetDefaultFunc, ClassId, InstanceDefn)
|
|
= InstanceName :-
|
|
ClassId = class_id(ClassName, _),
|
|
ClassNameStr = unqualify_name(ClassName),
|
|
(
|
|
WhichTypes = orig_types,
|
|
Types0 = InstanceDefn ^ instdefn_orig_types
|
|
;
|
|
WhichTypes = cur_types,
|
|
Types0 = InstanceDefn ^ instdefn_types
|
|
),
|
|
list.length(Types0, NumTypes0),
|
|
( if
|
|
Limit = print_few_types,
|
|
NumTypes0 > 5
|
|
then
|
|
string.format("%s/%d", [s(ClassNameStr), i(NumTypes0)], InstanceName)
|
|
else
|
|
(
|
|
Quals = keep_all,
|
|
Types = Types0
|
|
;
|
|
Quals = delete_builtin,
|
|
strip_module_names_from_type_list(strip_builtin_module_name,
|
|
SetDefaultFunc, Types0, Types)
|
|
;
|
|
Quals = delete_all,
|
|
strip_module_names_from_type_list(strip_all_module_names,
|
|
SetDefaultFunc, Types0, Types)
|
|
),
|
|
TVarSet = InstanceDefn ^ instdefn_tvarset,
|
|
TypesStr = mercury_types_to_string(TVarSet, print_name_only, Types),
|
|
string.format("%s(%s)", [s(ClassNameStr), s(TypesStr)], InstanceName)
|
|
).
|
|
|
|
%---------------------%
|
|
|
|
:- func pf_method_name_pieces(pred_pf_name_arity) = list(format_piece).
|
|
|
|
pf_method_name_pieces(MethodName) = Pieces :-
|
|
MethodName = pred_pf_name_arity(PredOrFunc, MethodSymName, UserArity),
|
|
UserArity = user_arity(UserArityInt),
|
|
SNA = sym_name_arity(MethodSymName, UserArityInt),
|
|
Pieces = [p_or_f(PredOrFunc), words("method"), unqual_sym_name_arity(SNA)].
|
|
|
|
:- func method_name_pieces(instance_method) = list(format_piece).
|
|
|
|
method_name_pieces(Method) = Pieces :-
|
|
Method = instance_method(PFSymNameArity, _, _),
|
|
Pieces = [unqual_pf_sym_name_user_arity(PFSymNameArity)].
|
|
|
|
%---------------------------------------------------------------------------%
|
|
:- end_module check_hlds.check_typeclass.
|
|
%---------------------------------------------------------------------------%
|