Files
mercury/compiler/check_typeclass.m
Zoltan Somogyi e2a8a8cbfa Break up mercury_to_mercury.m.
compiler/mercury_to_mercury.m:
    Delete this module, and replace it with ...

compiler/parse_tree_out_cons_id.m:
compiler/parse_tree_out_sym_name.m:
compiler/parse_tree_out_type.m:
compiler/parse_tree_out_misc.m:
    ... these four modules. The first three write out the entities
    in their names: cons_ids, sym_names, and types. The fourth contains
    the rest of the old mercury_to_mercury.m, plus a few predicates
    moved there from prog_out.m that deal with indentation.

compiler/parse_tree.m:
    Include the four new modules, and stop including the deleted module.

compiler/notes/compiler_design.html:
    Document the new modules.

compiler/prog_out.m:
    Delete the code moved to parse_tree_out_misc.m.

compiler/*.m:
    Adjust the imports as needed. Most modules need only one, maybe two
    of mercury_to_mercury's four successor modules.
2023-04-06 15:32:48 +10:00

2755 lines
117 KiB
Mathematica

%---------------------------------------------------------------------------%
% vim: ft=mercury ts=4 sw=4 et
%---------------------------------------------------------------------------%
% Copyright (C) 1996-2001, 2003-2012 The University of Melbourne.
% This file may only be copied under the terms of the GNU General
% Public License - see the file COPYING in the Mercury distribution.
%---------------------------------------------------------------------------%
%
% File: 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.type_util.
:- import_module check_hlds.typeclasses.
:- import_module hlds.add_pred.
:- import_module hlds.hlds_class.
:- import_module hlds.hlds_data.
:- import_module hlds.hlds_error_util.
:- 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 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_type.
:- import_module parse_tree.prog_type_subst.
:- 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 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, [], InstanceDeclSpecs),
!:Specs = CycleSpecs ++ InstanceDeclSpecs,
% 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.
(
InstanceDeclSpecs = [],
% 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(!.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)
;
InstanceDeclSpecs = [_ | _]
).
%---------------------------------------------------------------------------%
% 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) :-
ClassDefn0 = map.lookup(!.ClassTable, ClassId),
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_constraint_list(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 distinct type variables.
%
:- pred check_instance_declaration_types(module_info::in, module_info::out,
list(error_spec)::in, list(error_spec)::out) is det.
check_instance_declaration_types(!ModuleInfo, !Specs) :-
module_info_get_instance_table(!.ModuleInfo, InstanceTable),
map.foldl(check_instance_declaration_types_for_class(!.ModuleInfo),
InstanceTable, !Specs).
:- 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) is det.
check_instance_declaration_types_for_class(ModuleInfo, ClassId,
InstanceDefns, !Specs) :-
list.map(
is_instance_type_vector_valid(ModuleInfo, ClassId),
InstanceDefns, InstanceSpecLists),
list.condense(InstanceSpecLists, ClassInstanceSpecs),
!:Specs = ClassInstanceSpecs ++ !.Specs.
:- pred is_instance_type_vector_valid(module_info::in,
class_id::in, hlds_instance_defn::in, list(error_spec)::out) is det.
is_instance_type_vector_valid(ModuleInfo, ClassId, InstanceDefn, !:Specs) :-
OriginalTypes = InstanceDefn ^ instdefn_orig_types,
!:Specs = [],
list.foldl2(is_orig_type_non_eqv_type(ModuleInfo, ClassId, InstanceDefn),
OriginalTypes, 1, _, !Specs),
Types = InstanceDefn ^ instdefn_types,
list.foldl2(is_valid_instance_type(ModuleInfo, ClassId, InstanceDefn),
Types, 1, _, !Specs).
% 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(_, _, _, _, _),
KindPiece = words("is a higher order type;")
;
Type = apply_n_type(_, _, _),
KindPiece = words("is an apply/N type;")
;
Type = type_variable(_, _),
KindPiece = words("is a 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"),
quote(TypeStr), KindPiece, words("it should be"),
words("a type constructor applied to zero or more"),
words("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 = [_ | _],
TypeCtor = type_ctor(TypeCtorSymName, list.length(ArgTypes)),
report_badly_formed_type_in_instance(ClassId, InstanceDefn,
TypeCtor, ArgNum, NonTVarArgs, !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]
).
%---------------------------------------------------------------------------%
% 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: no definition for typeclass"),
unqual_class_id(ClassId), suffix("."), nl],
Spec = simplest_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: instance declaration for abstract typeclass"),
unqual_class_id(ClassId), suffix("."), nl],
Spec = simplest_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 = constraints(UnivCs0, ExistCs),
(
UnivCs0 = [_ | UnivCs]
;
UnivCs0 = [],
unexpected($pred, "no constraint on class method")
),
ClassMethodClassContext0 = 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_variable_renaming_to_type_list(Renaming, InstanceTypes0,
InstanceTypes1),
apply_variable_renaming_to_prog_constraint_list(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_type_list(TypeSubst, ArgTypes0, ArgTypes1),
apply_subst_to_prog_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),
prog_constraints_get_tvars(ClassMethodClassContext1, MethodContextTVars),
constraint_list_get_tvars(InstanceConstraints1, InstanceTVars),
list.condense([ArgTVars, MethodContextTVars, InstanceTVars], VarsToKeep0),
list.sort_and_remove_dups(VarsToKeep0, VarsToKeep),
% Project away the unwanted type variables.
varset.squash(TVarSet1, VarsToKeep, TVarSet2, SquashSubst),
apply_variable_renaming_to_type_list(SquashSubst, ArgTypes1, ArgTypes),
apply_variable_renaming_to_prog_constraints(SquashSubst,
ClassMethodClassContext1, ClassMethodClassContext),
apply_partial_map_to_list(SquashSubst, ExistQVars0, ExistQVars),
apply_variable_renaming_to_type_list(SquashSubst, InstanceTypes1,
InstanceTypes),
apply_variable_renaming_to_prog_constraint_list(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 = constraints(UnivConstraints1, ExistConstraints),
list.append(InstanceConstraints, UnivConstraints1, UnivConstraints),
ClassContext = 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, PredInfo0),
pred_info_set_clauses_info(ClausesInfo, PredInfo0, PredInfo1),
pred_info_set_instance_method_arg_types(UnsubstArgTypes,
PredInfo1, PredInfo2),
% We first insert the incomplete predicate PredInfo2 into !ModuleInfo,
% in order to get InstancePredId, the pred_id of the new predicate,
% which we need to compute the contents of the InstanceMethodInfos.
module_info_get_predicate_table(!.ModuleInfo, PredTable1),
predicate_table_insert(PredInfo2, 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),
AddProc =
( pred(ClassMI::in, InstanceMI::out,
OldPredInfo::in, NewPredInfo::out) is det :-
ClassMI = 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),
ItemNumber = item_no_seq_num,
% Before the simplification pass, HasParallelConj
% is not meaningful.
HasParallelConj = has_no_parallel_conj,
add_new_proc(!.ModuleInfo, Context, ItemNumber,
InstVarSet, Modes, yes(Modes), no, detism_decl_implicit,
MaybeDetism, address_is_taken, HasParallelConj,
OldPredInfo, NewPredInfo, InstanceProcId),
InstanceOrigPredProcId = proc(InstancePredId, InstanceProcId),
InstanceMI = method_info(MethodNum, MethodName,
InstanceOrigPredProcId, InstanceOrigPredProcId)
),
list.map_foldl(AddProc, ClassMethodInfos, InstanceMethodInfos,
PredInfo2, PredInfo),
% Replace the incomplete PredInfo2.
module_info_set_pred_info(InstancePredId, PredInfo, !ModuleInfo).
%---------------------------------------------------------------------------%
% 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_variable_renaming_to_prog_constraint_list(Renaming,
ProgSuperClasses0, ProgSuperClasses),
% Now handle the class variables.
apply_variable_renaming_to_tvar_list(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_constraints(ClassTable, InstanceTVarSet1, SuperClasses,
InstanceConstraints, Constraints0),
% 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, _, Constraints0, Constraints),
UnprovenConstraints = Constraints ^ hcs_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_variable_renaming_to_type_list(RenamingAB, TypesB, TypesBR),
type_list_subsumes(TypesA, TypesBR, _)
;
apply_variable_renaming_to_type_list(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, "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, "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_variable_renaming_to_prog_constraint_list(Renaming, OldConstraints0,
OldConstraints1),
apply_variable_renaming_to_tvar_list(Renaming, OldVars0, OldVars),
map.from_corresponding_lists(OldVars, Vars, VarRenaming),
apply_variable_renaming_to_prog_constraint_list(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, hlds_class_fundeps::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, hlds_class_fundeps::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,
hlds_class_fundeps::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_variable_renaming_to_type_list(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_type_list(Subst, RangeA0, RangeA),
apply_rec_subst_to_type_list(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(module_info::in,
list(error_spec)::in, list(error_spec)::out) is det.
check_typeclass_constraints_on_preds(ModuleInfo, !Specs) :-
module_info_get_valid_pred_ids(ModuleInfo, PredIds),
list.foldl(check_typeclass_constraints_on_pred(ModuleInfo),
PredIds, !Specs).
:- pred check_typeclass_constraints_on_pred(module_info::in, pred_id::in,
list(error_spec)::in, list(error_spec)::out) is det.
check_typeclass_constraints_on_pred(ModuleInfo, PredId, !Specs) :-
module_info_pred_info(ModuleInfo, PredId, PredInfo),
pred_info_get_class_context(PredInfo, Constraints),
Constraints = 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)] (
get_progress_output_stream(ModuleInfo, ProgressStream, !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),
prog_constraints_get_tvars(Constraints, ConstrainedTVars),
Constraints = 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 = constraints(UnivCs, ExistCs),
prog_type.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),
prog_type.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_fundeps == list(induced_fundep).
:- 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_variable_renaming_to_prog_constraint_list(Renaming,
ClassAncestors, RenamedAncestors),
apply_variable_renaming_to_tvar_list(Renaming, ClassParams,
RenamedParams),
map.from_corresponding_lists(RenamedParams, Args, Subst),
apply_subst_to_prog_constraint_list(Subst, RenamedAncestors,
Ancestors),
list.foldl(induced_fundeps_3(ClassTable), Ancestors, !FunDeps)
).
:- pred induced_fundeps_3(class_table::in, prog_constraint::in,
induced_fundeps::in, induced_fundeps::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: cyclic superclass relation detected:"), nl,
qual_class_id(HeadClassId), nl],
list.foldl(add_path_element, TailClassIds, cord.init, LaterLinesCord),
Pieces = StartPieces ++ cord.list(LaterLinesCord),
Spec = simplest_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],
!:LaterLines = !.LaterLines ++ cord.from_list(Line).
%---------------------------------------------------------------------------%
%
% Error reports from pass 2.
%
:- pred report_badly_formed_type_in_instance(class_id::in,
hlds_instance_defn::in, type_ctor::in, int::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,
NonTVarArgs, !Specs) :-
TVarSet = InstanceDefn ^ instdefn_tvarset,
NonTVarArgPieceLists =
list.map(non_tvar_arg_to_pieces(TVarSet), NonTVarArgs),
NonTVarArgPieces = component_lists_to_pieces("and", NonTVarArgPieceLists),
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),
words(choose_number(NonTVarArgs,
"is not a type variable, but should be. This is",
"are not type variables, but should be. These are"))
| NonTVarArgPieces] ++ [suffix("."), nl],
report_bad_type_in_instance(ClassId, InstanceDefn, EndPieces,
badly_formed, !Specs).
:- func non_tvar_arg_to_pieces(tvarset, pair(int, mer_type))
= list(format_piece).
non_tvar_arg_to_pieces(TVarSet, ArgNum - ArgType) = Pieces :-
TypeStr = mercury_type_to_string(TVarSet, print_name_only, ArgType),
Pieces = [words("the"), nth_fixed(ArgNum), words("argument,"),
quote(TypeStr)].
:- 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"),
quote(TypeStr),
words("is an abstract exported equivalence type."), nl],
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),
InstanceContext = InstanceDefn ^ instdefn_context,
(
Kind = abstract_exported_eqv,
HeadingMsg = simple_msg(InstanceContext,
[always(PrefixPieces), always(EndPieces)])
;
Kind = badly_formed,
VerbosePieces =
[words("(Every type in an instance declaration must consist of"),
words("a type constructor applied to zero or more type variables"),
words("as arguments.)"), nl],
HeadingMsg = simple_msg(InstanceContext,
[always(PrefixPieces), always(EndPieces),
verbose_only(verbose_once, VerbosePieces)])
),
Spec = error_spec($pred, severity_error, phase_type_check, [HeadingMsg]),
!: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("multiple implementations of") | PFMethodNamePieces] ++
[suffix("."), nl],
HeaderMsg = simplest_msg(InstanceDefn ^ instdefn_context, HeaderPieces),
FirstPieces = [words("First definition appears here."), nl],
FirstMsg = simplest_msg(FirstContext, FirstPieces),
LaterPieces = [words("Later definition appears here."), nl],
LaterMsg = simplest_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("no implementation for") | PFMethodNamePieces] ++
[suffix("."), nl],
Spec = simplest_spec($pred, severity_error, phase_type_check,
InstanceDefn ^ instdefn_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("the type class has no"),
p_or_f(PredOrFunc), words("method named"),
unqual_sym_name_arity(SNA), suffix("."), nl]
;
TailMethods = [_ | _],
SelectedContext = InstanceDefn ^ instdefn_context,
MethodPieces =
list.map(method_name_pieces, [HeadMethod | TailMethods]),
Pieces = PrefixPieces ++
[words("the type class has none of these methods:"),
nl_indent_delta(1)] ++
% XXX ARITY We could separate last two MethodPieces with ", or".
component_list_to_line_pieces(MethodPieces,
[suffix("."), nl_indent_delta(-1)])
),
Spec = simplest_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),
constraint_list_to_string(ClassTVarSet, UnprovenConstraints,
ConstraintsStr),
Pieces = PrefixPieces ++
[words("the following superclass"),
words(choose_number(UnprovenConstraints,
"constraint is", "constraints are")),
words("not satisfied:"), nl,
words(ConstraintsStr), suffix("."), nl],
Context = InstanceDefn ^ instdefn_context,
Spec = simplest_spec($pred, severity_error, phase_type_check,
Context, Pieces),
!:Specs = [Spec | !.Specs].
:- pred constraint_list_to_string(tvarset::in, list(hlds_constraint)::in,
string::out) is det.
constraint_list_to_string(_, [], "").
constraint_list_to_string(TVarSet, [C | Cs], String) :-
retrieve_prog_constraint(C, P),
PString = mercury_constraint_to_string(TVarSet, print_name_only, P),
constraint_list_to_comma_strings(TVarSet, Cs, TailStrings),
Strings = ["`", PString, "'" | TailStrings],
String = string.append_list(Strings).
:- pred constraint_list_to_comma_strings(tvarset::in,
list(hlds_constraint)::in, list(string)::out) is det.
constraint_list_to_comma_strings(_TVarSet, [], []).
constraint_list_to_comma_strings(TVarSet, [C | Cs], Strings) :-
retrieve_prog_constraint(C, P),
PString = mercury_constraint_to_string(TVarSet, print_name_only, P),
constraint_list_to_comma_strings(TVarSet, Cs, TailStrings),
Strings = [", `", PString, "'" | TailStrings].
%---------------------------------------------------------------------------%
%
% 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: overlapping instance declarations"),
words("for class"), qual_class_id(ClassId), suffix("."), nl,
words("One instance declaration is here, ..."), nl],
MsgA = simplest_msg(ContextA, PiecesA),
PiecesB = [words("... and the other is here."), nl],
MsgB = simplest_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,
error_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, error_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(":"),
words("duplicate"), words(Category), words("instance declaration"),
words("for class"), qual_class_id(ClassId), suffix("."), nl],
LaterMsg = simplest_msg(LaterContext, LaterPieces),
FirstPieces = [words("Previous instance declaration was here."), nl],
FirstMsg = error_msg(yes(FirstContext), always_treat_as_first, 0,
[always(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"),
words("on this abstract instance declaration"),
words("for class"), qual_class_id(ClassId),
words("do not match the instance constraints"),
words("on the corresponding concrete instance declaration."), nl],
AbstractMsg = simplest_msg(AbstractContext, AbstractPieces),
ConcretePieces = [words("The corresponding"),
words("concrete instance declaration is here."), nl],
ConcreteMsg = simplest_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_type_list_to_string(TVarSet, Types),
string.format("%s(%s)", [s(ClassNameString), s(TypesStr)], InstanceName),
Pieces = [words("Error: this abstract instance declaration"),
words("for"), quote(InstanceName),
words("has no corresponding concrete instance declaration"),
words("in the implementation section."), nl],
Context = InstanceDefn ^ instdefn_context,
Spec = simplest_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,
ClassId, LocalInstance),
% XXX Should we mention any constraints on the instance declaration?
LocalPieces = [words("Error: this instance declaration"),
words("for"), quote(InstanceName), words("clashes with"),
words("an instance declaration in another module."), nl],
LocalContext = LocalInstance ^ instdefn_context,
LocalMsg = simplest_msg(LocalContext, LocalPieces),
NonLocalPieces = [words("The other instance declaration is here."), nl],
NonLocalContext = NonLocalInstance ^ instdefn_context,
NonLocalMsg = simplest_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,
VarsStrs = list.map(mercury_var_to_name_only_vs(TVarSet), Vars),
Pieces = [words("In instance for typeclass"),
unqual_class_id(ClassId), suffix(":"), nl,
words("functional dependency not satisfied:"),
words(choose_number(Vars, "type variable", "type variables"))]
++ list_to_quoted_pieces(VarsStrs) ++
[words(choose_number(Vars, "occurs", "occur")),
words("in the range of the functional dependency, but"),
words(choose_number(Vars, "is", "are")),
words("not determined by the domain."), nl],
Context = InstanceDefn ^ instdefn_context,
Spec = simplest_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),
PiecesA = [words("Inconsistent instance declaration for typeclass"),
qual_class_id(ClassId), words("with functional dependency"),
quote("(" ++ Domains ++ " -> " ++ Ranges ++ ")"), suffix("."), nl],
PiecesB = [words("Here is the conflicting instance."), nl],
MsgA = simplest_msg(ContextA, PiecesA),
MsgB = error_msg(yes(ContextB), always_treat_as_first, 0,
[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),
VarsStrs = list.map(mercury_var_to_name_only_vs(TVarSet), Vars),
PFSymNameArity = pf_sym_name_arity(PredOrFunc, SymName, PredFormArity),
Pieces0 = [words("In declaration for"),
unqual_pf_sym_name_pred_form_arity(PFSymNameArity), suffix(":"), nl,
words("error in type class constraints:"),
words(choose_number(Vars, "type variable", "type variables"))]
++ list_to_quoted_pieces(VarsStrs) ++
[words(choose_number(Vars, "occurs", "occur")),
words("in the constraints, but"),
words(choose_number(Vars, "is", "are")),
words("not determined 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 = simplest_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(should_module_qualify, PredInfo) ++
[suffix(":"), nl],
TypeVariables = [words("type variable"),
suffix(choose_number(TVars, "", "s"))],
TVarsStrs = list.map(mercury_var_to_name_only_vs(TVarSet), TVars),
TVarsPart = list_to_quoted_pieces(TVarsStrs),
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 ++ TypeVariables ++ TVarsPart ++
[Are, BlahConstrained, suffix(","), words("but"), Are,
BlahQuantified, suffix("."), nl],
Spec = simplest_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),
VarsStrs = list.map(mercury_var_to_name_only_vs(TVarSet), Vars),
Pieces = [words("In declaration for type"), qual_type_ctor(TypeCtor),
suffix(":"), nl,
words("error in type class constraints:"),
words(choose_number(Vars, "type variable", "type variables"))]
++ list_to_quoted_pieces(VarsStrs) ++
[words(choose_number(Vars, "occurs", "occur")),
words("in the constraints, but"),
words(choose_number(Vars, "is", "are")),
words("not determined 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 = simplest_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"), QualHeadClassId,
words("does not exist."), nl]
;
TailClassIds = [_ | _],
QualTailClassIds = list.map(WrapQualClassId, TailClassIds),
QualClassIds = [QualHeadClassId | QualTailClassIds],
Pieces = [words("error: the type classes")] ++
component_list_to_pieces("and", QualClassIds) ++
[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,
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,
class_id, hlds_instance_defn) = string.
instance_name(WhichTypes, Limit, Quals, 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,
Types0, Types)
;
Quals = delete_all,
strip_module_names_from_type_list(strip_all_module_names,
Types0, Types)
),
TVarSet = InstanceDefn ^ instdefn_tvarset,
TypesStr = mercury_type_list_to_string(TVarSet, 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.
%---------------------------------------------------------------------------%