]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
Update the concepts implementation to conform to C++20.
authorJason Merrill <jason@gcc.gnu.org>
Wed, 9 Oct 2019 17:20:32 +0000 (13:20 -0400)
committerJason Merrill <jason@gcc.gnu.org>
Wed, 9 Oct 2019 17:20:32 +0000 (13:20 -0400)
gcc/c-family/
* c-cppbuiltin.c (c_cpp_builtins): Use new feature test values for
concepts when -std=c++2a. Bump __cpp_concepts to 201907.
* c.opt: Add -Wconcepts-ts.
* c-opts.c (c_common_post_options): Warn when -fconcepts is used
with -std=c++2a. Disable warning for -fconcepts in C++20 mode.
(set_std_cxx2a): Enable concepts by default.
gcc/cp/
* call.c (build_new_function_call): Don't evaluate concepts here.
(constraint_failure): Don't record the template.
(print_z_candidate): Don't extract the template.
* class.c (add_method): When overloading, hide ineligible special
member fns.
(check_methods): Set TYPE_HAS_COMPLEX_* here.
* constexpr.c (cxx_eval_constant_expression): Evaluate concepts.
(maybe_initialize_fundef_copies_table): Remove.
(get_fundef_copy): Use hash_map_safe_get_or_insert.
(clear_cv_and_fold_caches): Clear the satisfaction cache.
* constraint.cc (known_non_bool_p): New.
(parsing_constraint_expression_sentinel): Renamed from
expanding_constraint_sentinel.
(check_constraint_operands): New.
(check_constraint_atom): New.
(finish_constraint_binary_op): New.
(finish_constraint_or_expr): Likewise.
(finish_constraint_and_expr): Likewise.
(finish_constraint_primary_expr): Likewise.
(combine_constraint_expressions): New.
(finish_requires_expr): Add location parm.
(get_concept_definition): Return the initializer of concept definitions.
(get_template_head_requirements): New.
(get_trailing_function_requirements): New.
(deduce_constrained_parameter): Check if the identifier or template-id
is a concept definition.
(resolve_concept_definition_check): Removed.
(resolve_variable_concept_check): Removed.
(resolve_concept_check): New.
(resolve_constraint_check): Handle concept definitions.
converting arguments.
(function_concept_check_p): Removed.
(variable_concept_check_p): Removed.
(unpack_concept_check): New.
(get_concept_check_template): New.
(build_call_check): Moved and renamed to build_function_check.
(build_concept_check_arguments): make static.
(build_function_check): Always do overload resolution
in order to force conversion of template arguments (i.e., actually
check that the use of a concept is valid).
(build_standard_check): Renamed from build_real_concept_check.
(build_real_concept_check): Build checks for C++2a concepts by
(build_wildcard_concept_check): New.
(build_concept_check): Use build_real_concept_check. New overload.
(build_constraints): Save expressions, not normalized constraints.
(build_concept_id): New. Pass tf_warning_or_error.
(build_type_constraint): New.
(finish_type_constraints): New.
(associate_classtype_constraints): Also add constraints to union
types. Note the original declaration in errors. Don't return
error_mark_node in order to avoid an assertion later.
(push_down_pack_expansion): Remove.
(finish_shorthand_constraint): Make fold expressions, not naked
parameter packs. Always apply the constraint to each template argument.
(check_introduction_list): New. Fail if not enough
names are introduced.
(finish_template_introduction): Don't normalize constraints. Pass
tsubst flags. Check for insufficient introductions.
(placeholder_extract_concept_and_args): Handle the template-id case.
Unpack function concept checks correctly.
(tsubst_simple_requirement): Return errors if they occur. Don't
process as a template.
(tsubst_type_requirement): Likewise.
(type_deducible_p): New. Wrap the input expression in parens for the
purpose of deduction.
(expression_convertible_t): New.
(tsubst_compound_requirement): Use new deduction, conversion predicates.
(tsubst_nested_requirement): Return errors if they occur. Don't
process as a template. Instantiate and evaluate the nested requirement.
(tsubst_valid_expression_requirement): New.
(tsubst_simple_requirement): Use tsubst_valid_expression_requirement.
(tsubst_compound_requirement): Use tsubst_valid_expression_requirement.
(check_constaint_variables): New.
(tsubst_constraint_variables): Check that type substitutions are valid.
(tsubst_requires_expr): Likewise. Produce new requires-exprs during
template substitution. Copy the previous local specialization stack,
so references to non-local parameters can be found. Use cp_unevaluated.
(tsubst_constraint): New. Don't evaluate concept checks.
(subst_info): New.
(norm_info): New. Used to build a normalization tree for concept check
diagnostics.
(debug_parameter_mapping): New.
(debug_argument_list): New.
(expand_concept): Removed.
(normalize_logical_operation): Pass subst_info through call.
(normalize_pack_expansion): Remove.
(normalize_simple_requirement): Removed
(normalize_type_requirement): Removed
(normalize_compound_requirement): Removed
(normalize_nested_requirement): Removed
(normalize_requirement): Removed
(normalize_requirements): Removed
(normalize_requires_expression): Removed
(normalize_variable_concept_check): Removed.
(normalize_function_concept_check): Removed.
(normalize_concept_check): Merged all normalize_*_check here.
Substitute through written template arguments before normalizing the
definition. Only substitute the innermost template arguments.
(check_for_logical_overloads): Delete.
(map_arguments): New. Associate template parameters with arguments.
(build_parameter_mapping): New. Extract used parameters.
(normalize_expression): Rewrite.
(normalize_conjunction): Removed
(normalize_disjunction): Removed
(normalize_predicate_constraint): Removed
(normalize_parameterized_constraint): Removed
(normalized_map): New variable.
(get_normalized_constraints): New entry point for normalization.
Establishes a timer.
(get_normalized_constraints_from_info): New.
(get_normalized_constraints_from_decl): New. Turn on template processing
prior to normalization. Handle inheriting ctors. Build the
normalization arguments from the full set of template parameters of the
most general template. This guarantees that we have no concrete arguments
in the parameter mapping (e.g., from template members of class
templates). Cache normalizations.
(normalize_concept_definition): New. Cache normalizations.
(normalize_template_requirements): New.
(normalize_nontemplate_requirements): New.
(normalize_constraint_expression): New.
(tsubst_parameter_mapping): New.
(get_mapped_args): New.
(parameter_mapping_equivalent_p): New. Use template_args_equal.
(atomic_constraints_identical_p): New.
(hash_atomic_constraint): New.
(satisfying_constraint_p): New. Guard against recursive evaluation of
constraints during satisfaction.
(satisfy_conjunction): New.
(satisfy_disjunction): New.
(sat_entry): New class for hashing satisfaction results.
(sat_hasher): New hash traits.
(sat_cache): New.
(get_satisfaction): New. Returns cached satisfaction result.
(save_satisfaction): New. Caches a satisfaction result.
(clear_satisfaction_cache): New.
(satisfaction_cache): New. Helps manage satisfaction cache requests.
(decl_satisfied_cache): New.
(satisfy_atom): New.
(satisfy_constraint_r): New.
(satisfy_constraint): Use new satisfaction algorithm.
(evaluate_concept_check): New.
(evaluate_concept): Removed.
(evaluate_function_concept): Removed.
(evaluate_variable_concept): Removed.
(satisfy_constraint_expression): New.
(constraint_expression_satisfied_p): New.
(constraints_satisfied_p): Use strip_inheriting_ctors. Use
push_/pop_access_scope.
(more_constrained): Normalize before calling out to subsumption. Allow
classes as arguments.
(strictly_subsumes): Allow non-templates as arguments. Accept a new
template argument.
(weakly_subsumes): New.
(at_least_as_constrained): Removed.
(diagnose_other_expression): Removed.
(diagnose_predicate_constraint): Removed.
(diagnose_pack_expansion): Removed.
(diagnose_check_constraint): Removed.
(diagnose_logical_constraint): Removed.
(diagnose_expression_constraint): Removed.
(diagnose_type_constraint): Removed.
(diagnose_implicit_conversion_constraint): Removed.
(diagnose_argument_deduction_constraint): Removed.
(diagnose_exception_constraint): Removed.
(diagnose_parameterized_constraint): Removed.
(diagnose_argument_deduction_constraint): Removed.
(diagnose_argument_deduction_constraint): Removed.
(diagnose_argument_deduction_constraint): Removed.
(diagnose_trait_expr): New.
(diagnose_requires_expr): New.
(diagnose_atomic_constraint): New.
(diagnose_valid_expression) Stop wrongly diagnosing valid expressions.
Don't substitute as if in template decls. This causes substitution
to generate expressions that aren't suitable for use with the noexcept
routines.
(diagnose_valid_type) Likewise.
(diagnose_compound_requirement) Actually emit diagnostics for
the causes of errors.Call force_paren_expr_uneval.
(diagnose_declaration_constraints): Turn on template processing to
suppress certain analyses.
* cp-objcp-common.c (cp_common_init_ts): Make concepts typed.
(cp_get_debug_type): Use hash_map_safe_*.
* cp-tree.h: New function declarations for semantic actions, other
facilities. Remove declaration no longer used or needed. Remove
unused _CONSTR macros.
(LANG_DECL_HAS_MIN): Add CONCEPT_DECL.
(template_info_decl_check): Factor macro check into an inline function.
(DECL_TEMPLATE_INFO): Use new check facility.
(finish_concept_definition): New. Don't invalid concept declarations
with invalid initializers.
(find_template_parameters): New.
(concept_definition_p): New.
(concept_check_p): New.
(variable_concept_check_p): New.
(force_paren_expr_uneval): New.
(ovl_iterator::using_p): A USING_DECL by itself was also
introduced by a using-declaration.
(struct tree_template_info): Use tree_base instead of
tree_common. Add tmpl and args fields.
(TI_TEMPLATE, TI_ARGS): Adjust.
(DECLTYPE_FOR_INIT_CAPTURE): Remove.
(CONSTR_CHECK, CONSTR_INFO, CONSTR_EXPR, CONSTR_CONTEXT): New.
(ATOMIC_CONSTR_MAP, TRAIT_EXPR_LOCATION): New.
(struct tree_trait_expr): Add locus field.
(enum tsubst_flags): Add tf_norm as a hint to generate normalization
context when diagnosing constraint failure.
* cp-tree.def: Remove unused _CONSTR nodes and rename PRED_CONSTR
to ATOMIC_CONSTR.
(CONCEPT_DECL): New.
* cxx-pretty-print.c: Remove constraint printing code.
(pp_cxx_concept_definition): New.
(pp_cxx_template_declaration): Print concept definitions.
(pp_cxx_check_constraint): Update printing for concept definitions.
(pp_cxx_nested_name_specifier): Fix a weird
case where we're printing '::::' for concepts.
(simple_type_specifier): Print requirements for placeholder types.
(pp_cxx_constrained_type_spec): Print the associated requirements of
a placeholder type.
(pp_cxx_compound_requirement): Add space before the '->'.
(pp_cxx_parameter_mapping): Print the parameter mapping.
(pp_cxx_atomic_constraint): Use the function above.
* decl.c (redeclaration_error_message): New error for concepts.
(grokdeclarator): Check for and disallow decltype(auto) in parameter
declarations.
(grokfndecl): Don't normalize constraints. Add check for constraints
on declaration.
(grokvardecl): Don't normalize constraints.
(grok_special_member_properties): Don't set TYPE_HAS_COMPLEX_*.
(function_requirements_equivalent_p): New. Compare trailing
requires clauses. Compare combined constraints in pre-C++20 mode.
(decls_match): Compare trailing requires clauses. Compare template
heads for function templates. Remove old constraint comparison.
Simplify comparison of functions, function templates.
(duplicate_function_template_decls): New. Refactor a nasty if
condition into a single predicate.
(require_deduced_type): Don't complain if we already complained about
deduction failure.
(finish_function): Perform auto deduction to ensure that constraints
are checked even when functions contain no return statements. Only do
auto deduction if we haven't previously seen any return statements.
This prevents multiple diagnostics of the same error.
(store_decomp_type): Remove.
(cp_finish_decomp): Use hash_map_safe_put.
* error.c: Remove constraint printing code.
(dump_decl): Dump concept definitions. Handle wildcard declarations.
(dump_template_decl): Likewise.
(dump_type): Print associated requirements for placeholder
types.
(rebuild_concept_check): New.
(maybe_print_single_constraint_context): New.
(maybe_print_constraint_context): Recursively print nested contexts.
* init.c (get_nsdmi): Use hash_map_safe_*.
* lambda.c (maybe_add_lambda_conv_op): Bail if deduction failed.
(add_capture): Copy parameter packs from init.
(lambda_capture_field_type): Always use auto for init-capture.
* logic.cc: Completely rewrite.
(constraint_hash): New.
(clause/ctor): Save atoms in the hash table.
(replace): Save atoms during replacement.
(insert): Save atoms during insertion.
(contains): Only search the hash table for containment.
(clause): Keep a hash of atomic constraints.
(clause::clause): Explicitly copy the hash table when copying.
(disjunction_p, conjunction_p, atomic_p, dnf_size, cnf_size): New.
(diagnose_constraint_size): New.
(subsumes_constraints_nonnull): Compare the sizes of normalized formula
to determine the cheapest decomposition.
* name-lookup.c (diagnose_name_conflict): Diagnose name issues with
concepts.
(matching_fn_p): Check constraints.
(push_class_level_binding_1): Move overloaded functions case down,
accept FUNCTION_DECL as target_decl.
* parser.c (enum required_token): New required token for auto.
(make_location): Add overload taking lexer as last parm.
(cp_parser_required_error): Diagnose missing auto.
(cp_parser_diagnose_ungrouped_constraint_plain): New.
(cp_parser_diagnose_ungrouped_constraint_plain): New.
(cp_parser_constraint_primary_expression): New. Tentatively parse the
primary expression. If that fails tentatively parse a lower
precedence expression in order to diagnose the error.
(cp_parser_check_non_logical_constraint): New. Performs a trial
parse of the right-hand-side of non-logical operators in order to
generate good diagnostics.
(cp_parser_constraint_logical_and_expression): New.
(cp_parser_constraint_logical_or_expression): New.
(cp_parser_requires_clause_expression): New.
(cp_parser_requires_clause): Renamed to cp_parser_constraint_expression.
(cp_parser_requires_clause_opt): Parse the requires-clause differently
in -fconcepts and -std=c++2a modes.
(cp_parser_requirement_list): Rename to cp_parser_requirement_seq.
Rewrite so that semicolons are parsed
along with requirements, not the sequence.
(cp_parser_simple_requirement): Expect a semicolon at end.
(cp_parser_compound_requirement): Expect a semicolon at end. Only
allow trailing-return-type with -fconcepts-ts.
(cp_parser_nested_requirement): Expect a semicolon at end. Parse
constraint-expressions.
(cp_parser_concept_definition): New. Don't fail parsing the concept
definition if the initializer is ill-formed. Don't declare the concept
before parsing the initializer.
(cp_parser_constraint_expression): Declare earlier.
(cp_parser_type_requirement): Current scope is not valid.
(cp_parser_requires_expression): Commit to the tentative parse.
(cp_parser_decl_specifier_seq): Warn when concept appears to be used
as a decl-specifier.
(cp_parser_template_declaration_after_parameters): Parse concept
definitions.
(cp_parser_template_id): Don't try to resolve a concept template-id yet.
(cp_parser_template_id_expr): Resolve it as a concept check.
(cp_parser_decl_specifier_seq): Warn on 'concept bool'.
(cp_parser_type_parameter): Combine expressions not
constraints.
(cp_parser_explicit_template_declaration): Combine expressions not
constraints.
(cp_parser_maybe_concept_name): Removed.
(cp_parser_simple_type_specifier): Handle an error condition of
a bad constrained type specifier. Expect auto or decltype after
a concept name. Also handle the case where we have a template-id
as a concept check.
(cp_parser_template_introduction): Diagnose errors on invalid
introductions. Give up if it doesn't start with a concept name.
Pedwarn if not -fconcepts-ts.
(synthesize_implicit_template_parm): Don't do consistent binding.
Use a new flag for constrained parameters. Combine expressions,
not constraints. Fail if we get a placeholder in block scope.
Placeholders that do not constrain types are not allowed in parameter
declarations, so don't handle them.
(cp_parser_placeholder_type_specifier): New. Implement parsing of
placeholder type specifiers following a concept name or partial
concept check. Disallow decltype(auto) parameters.
(cp_parser_nested_name_specifier_opt): If the token is already
CPP_NESTED_NAME_SPECIFIER, leave it alone.
(cp_parser_id_expression, cp_parser_unqualified_id): Call
cp_parser_template_id_expr.
(cp_parser_placeholder_type_specifier): Add tentative parm.  Don't
expect a WILDCARD_DECL.
(cp_parser_trait_expr): Pass trait_loc down.
(cp_parser_postfix_expression): Do set location of dependent member
call.
* pt.c (finish_concept_definition): New.
(push_template_decl_real): Handle concept definitions.
(start_concept_definition): Let push_template_decl_real handle the
creation of the template.
(get_constraints): Return null if the table hasn't been initialized.
(tsubst_copy_and_build): Build template-id expressions for concept
checks.
[TRAIT_EXPR]: Pass trait_loc down.
(lookup_template_class_1): Add the template name to the constraint
failure diagnostic.
(lookup_and_finish_template_variable): Build concept checks
with the correct arguments.
(tsubst_function_decl): Don't substitute through constraints.
Always associate constraints with functions.
(template_parm_level_and_index): Make non-static.
(for_each_template_parm_r): Handle requires expressions.
(keep_template_parm): New.
(find_template_parameters): New.
(more_specialized_fn): Change how winners and losers are chosen.
(make_constrained_auto): Don't normalize constraints.
(template_parameters_equivalent_p): New. Compare template
parameters. Add a comparison for implicitly vs. explicitly declared
parameters.
(template_parameter_lists_equivalent_p): New. Compare template
parameter lists.
(template_requirements_equivalent_p): New.
(template_heads_equivalent_p): New. Compare template heads.
(template_parameter_constraints_equivalent_p): New.
(is_compatible_template_arg): Use weakly_subsumes.
(maybe_new_partial_specialization): Use new constraint comparison
for finding specializations.
(process_partial_specialization): Pass main template as argument.
(more_specialized_partial_spec): Don't immediately return when
detecting a winner.
(make_constrained_auto): Handle concept definitions.
(do_auto_deduction): Update auto deduction for new concept model.
Extract the function concept correctly; rename constr to check to
reflect the kind of node.
(tsubst): Adjust wildcard argument during substitution.
[DECLTYPE_TYPE]: Remove init-capture handling.
(tsubst_copy_and_build): Build concept checks, not template ids.
Defer checks of function concepts. Handle concepts before variable
templates. Handle calls to function concepts explicitly.
(coerce_template_parms): Use concept_definition_p. Handle a deduction
error where a potentially empty pack can be supplied after the last
parameter of a concept.
(finish_template_variable): Don't process concepts here.
(instantiation_dependent_r): Use concept_check_p.
(tsubst_template_args): Make non-static.
(make_constrained_placeholder_type): New. Refactored from
make_constrained_auto.
(make_constrained_auto) Use make_constrained_placeholder_type.
(make_constrained_decltype_auto) New.
(tsubst_function_parms): New.
(value_dependent_expression_p) [TEMPLATE_ID_EXPR]: Use
concept_definition_p.
(push_access_scope, pop_access_scope): No longer static.
(tsubst_template_parm): Substitute TEMPLATE_PARM_CONSTRAINTS.
(tsubst_friend_function): Use tsubst_constraint. Use generic_targs_for.
(get_underlying_template) Use generic_targs_for.
(uses_parameter_packs): Return tree.
(gen_elem_of_pack_expansion_instantiation): Don't push
local_specialization_stack.
(prepend_one_capture): New.
(tsubst_lambda_expr): Use prepend_one_capture. Don't touch
local_specializations.
(template_parms_level_to_args): No longer static.
(add_outermost_template_args): Likewise.
(find_template_parameter_info): New. Provide context for finding
template parameters.
(keep_template_parm): Don't keep parameters declared at depth levels
greater than those of the template parameters of the source declaration.
Don't propagate cv-qualified types. Return 0, so we find all template
parameters, not the just first.
(any_template_parm_r): New. Handle cases that are mishandled by
for_each_template_parm_r.
(generic_targs_for): Factor out of coerce_template_args_for_ttp.
(tsubst_argument_pack): Factor out of tsubst_template_args.
(constraint_sat_entry): Removed.
(constraint_sat_hasher): Removed.
(concept_spec_entry): Removed.
(concept_spec_hasher): Removed.
(constraint_memos): Removed.
(concept_memos): Removed.
(lookup_constraint_satisfaction): Removed.
(memoize_constraint_satisfaction): Removed.
(lookup_concept_satisfaction): Removed.
(memoize_concept_satisfaction): Removed.
(concept_expansions): Removed.
(get_concept_expansion): Removed.
(save_concept_expansion): Removed.
(init_constraint_processing): Remove initialization of non-existing
resources.
(find_template_requirement): New. Search for the sub-requirement
within the associated constraints.
(convert_generic_types_to_packs): Also transform the associated
constraint and update the current template requirements.
(store_defaulted_ttp, lookup_defaulted_ttp): Remove.
(add_defaults_to_ttp): Use hash_map_safe_*.
* semantics.c (finish_call_expr): Diagnose calls to concepts.
Handle concept checks explicitly.
(finish_id_expression): Evaluate variable concepts as part of
id-expression processing. Don't treat variable concepts as variables,
and don't process function concepts as plain id-expressions.
(force_paren_expr): Add even_uneval parm.
(finish_trait_expr): Add location parm.
* tree.c (special_memfn_p): New.
(cp_expr_location): Handle TRAIT_EXPR.
* typeck.c (check_return_expr): Actually use the diagnostic kind
when performing return-type deduction.
* typeck2.c (build_functional_cast): Don't rely on the location of
'auto'.
gcc/testsuite/
* lib/target-supports.exp (check_effective_target_concepts): Check
for std=c++2a.
gcc/
* doc/invoke.texi: Document -fconcepts-ts.

From-SVN: r276764

326 files changed:
gcc/ChangeLog
gcc/c-family/ChangeLog
gcc/c-family/c-cppbuiltin.c
gcc/c-family/c-opts.c
gcc/c-family/c.opt
gcc/cp/ChangeLog
gcc/cp/call.c
gcc/cp/class.c
gcc/cp/config-lang.in
gcc/cp/constexpr.c
gcc/cp/constraint.cc
gcc/cp/cp-objcp-common.c
gcc/cp/cp-tree.def
gcc/cp/cp-tree.h
gcc/cp/cxx-pretty-print.c
gcc/cp/decl.c
gcc/cp/error.c
gcc/cp/lambda.c
gcc/cp/logic.cc
gcc/cp/name-lookup.c
gcc/cp/parser.c
gcc/cp/pt.c
gcc/cp/search.c
gcc/cp/semantics.c
gcc/cp/typeck.c
gcc/cp/typeck2.c
gcc/doc/invoke.texi
gcc/testsuite/ChangeLog
gcc/testsuite/g++.dg/concepts/alias1.C [deleted file]
gcc/testsuite/g++.dg/concepts/alias2.C [deleted file]
gcc/testsuite/g++.dg/concepts/alias3.C [deleted file]
gcc/testsuite/g++.dg/concepts/alias4.C [deleted file]
gcc/testsuite/g++.dg/concepts/auto1.C
gcc/testsuite/g++.dg/concepts/auto3.C
gcc/testsuite/g++.dg/concepts/auto4.C
gcc/testsuite/g++.dg/concepts/class-deduction1.C
gcc/testsuite/g++.dg/concepts/class.C [deleted file]
gcc/testsuite/g++.dg/concepts/class1.C [deleted file]
gcc/testsuite/g++.dg/concepts/class2.C [deleted file]
gcc/testsuite/g++.dg/concepts/class3.C [deleted file]
gcc/testsuite/g++.dg/concepts/class4.C [deleted file]
gcc/testsuite/g++.dg/concepts/class5.C
gcc/testsuite/g++.dg/concepts/class6.C
gcc/testsuite/g++.dg/concepts/debug1.C
gcc/testsuite/g++.dg/concepts/decl-diagnose.C
gcc/testsuite/g++.dg/concepts/deduction-constraint1.C
gcc/testsuite/g++.dg/concepts/diagnostic1.C
gcc/testsuite/g++.dg/concepts/disjunction1.C [deleted file]
gcc/testsuite/g++.dg/concepts/dr1430.C
gcc/testsuite/g++.dg/concepts/equiv.C
gcc/testsuite/g++.dg/concepts/equiv2.C
gcc/testsuite/g++.dg/concepts/explicit-inst4.C [deleted file]
gcc/testsuite/g++.dg/concepts/explicit-spec3.C [deleted file]
gcc/testsuite/g++.dg/concepts/expression.C
gcc/testsuite/g++.dg/concepts/expression2.C
gcc/testsuite/g++.dg/concepts/expression3.C
gcc/testsuite/g++.dg/concepts/feature-macro.C [deleted file]
gcc/testsuite/g++.dg/concepts/fn-concept1.C
gcc/testsuite/g++.dg/concepts/fn-concept2.C
gcc/testsuite/g++.dg/concepts/fn-generic-member-ool.C
gcc/testsuite/g++.dg/concepts/fn1.C
gcc/testsuite/g++.dg/concepts/fn10.C
gcc/testsuite/g++.dg/concepts/fn2.C
gcc/testsuite/g++.dg/concepts/fn3.C
gcc/testsuite/g++.dg/concepts/fn4.C
gcc/testsuite/g++.dg/concepts/fn5.C
gcc/testsuite/g++.dg/concepts/fn6.C
gcc/testsuite/g++.dg/concepts/fn7.C
gcc/testsuite/g++.dg/concepts/fn8.C
gcc/testsuite/g++.dg/concepts/fn9.C
gcc/testsuite/g++.dg/concepts/generic-fn-err.C
gcc/testsuite/g++.dg/concepts/generic-fn.C
gcc/testsuite/g++.dg/concepts/iconv1.C [deleted file]
gcc/testsuite/g++.dg/concepts/inherit-ctor1.C
gcc/testsuite/g++.dg/concepts/inherit-ctor3.C
gcc/testsuite/g++.dg/concepts/inherit-ctor4.C [deleted file]
gcc/testsuite/g++.dg/concepts/intro1.C
gcc/testsuite/g++.dg/concepts/intro2.C
gcc/testsuite/g++.dg/concepts/intro3.C
gcc/testsuite/g++.dg/concepts/intro4.C
gcc/testsuite/g++.dg/concepts/intro5.C
gcc/testsuite/g++.dg/concepts/intro6.C
gcc/testsuite/g++.dg/concepts/intro7.C
gcc/testsuite/g++.dg/concepts/locations1.C
gcc/testsuite/g++.dg/concepts/memtmpl1.C [deleted file]
gcc/testsuite/g++.dg/concepts/partial-concept-id1.C
gcc/testsuite/g++.dg/concepts/partial-concept-id2.C
gcc/testsuite/g++.dg/concepts/partial-spec5.C
gcc/testsuite/g++.dg/concepts/placeholder1.C [deleted file]
gcc/testsuite/g++.dg/concepts/placeholder2.C
gcc/testsuite/g++.dg/concepts/placeholder3.C
gcc/testsuite/g++.dg/concepts/placeholder4.C
gcc/testsuite/g++.dg/concepts/placeholder5.C
gcc/testsuite/g++.dg/concepts/placeholder6.C
gcc/testsuite/g++.dg/concepts/pr65634.C
gcc/testsuite/g++.dg/concepts/pr65636.C
gcc/testsuite/g++.dg/concepts/pr65681.C
gcc/testsuite/g++.dg/concepts/pr65848.C
gcc/testsuite/g++.dg/concepts/pr67249.C
gcc/testsuite/g++.dg/concepts/pr67544.C
gcc/testsuite/g++.dg/concepts/pr67595.C
gcc/testsuite/g++.dg/concepts/pr67655.C
gcc/testsuite/g++.dg/concepts/pr68434.C
gcc/testsuite/g++.dg/concepts/pr71127.C
gcc/testsuite/g++.dg/concepts/pr71128.C
gcc/testsuite/g++.dg/concepts/pr71131.C
gcc/testsuite/g++.dg/concepts/pr71385.C
gcc/testsuite/g++.dg/concepts/pr84330.C
gcc/testsuite/g++.dg/concepts/pr85065.C
gcc/testsuite/g++.dg/concepts/req-neg1.C [deleted file]
gcc/testsuite/g++.dg/concepts/req1.C [deleted file]
gcc/testsuite/g++.dg/concepts/req10.C [deleted file]
gcc/testsuite/g++.dg/concepts/req11.C [deleted file]
gcc/testsuite/g++.dg/concepts/req12.C [deleted file]
gcc/testsuite/g++.dg/concepts/req13.C [deleted file]
gcc/testsuite/g++.dg/concepts/req16.C [deleted file]
gcc/testsuite/g++.dg/concepts/req18.C [deleted file]
gcc/testsuite/g++.dg/concepts/req19.C [deleted file]
gcc/testsuite/g++.dg/concepts/req2.C [deleted file]
gcc/testsuite/g++.dg/concepts/req20.C [deleted file]
gcc/testsuite/g++.dg/concepts/req3.C [deleted file]
gcc/testsuite/g++.dg/concepts/req4.C [deleted file]
gcc/testsuite/g++.dg/concepts/req5.C [deleted file]
gcc/testsuite/g++.dg/concepts/req6.C [deleted file]
gcc/testsuite/g++.dg/concepts/req7.C [deleted file]
gcc/testsuite/g++.dg/concepts/req8.C [deleted file]
gcc/testsuite/g++.dg/concepts/req9.C [deleted file]
gcc/testsuite/g++.dg/concepts/template-parm1.C [deleted file]
gcc/testsuite/g++.dg/concepts/template-parm10.C [deleted file]
gcc/testsuite/g++.dg/concepts/template-parm11.C
gcc/testsuite/g++.dg/concepts/template-parm12.C
gcc/testsuite/g++.dg/concepts/template-parm2.C
gcc/testsuite/g++.dg/concepts/template-parm3.C
gcc/testsuite/g++.dg/concepts/template-parm4.C
gcc/testsuite/g++.dg/concepts/template-parm5.C [deleted file]
gcc/testsuite/g++.dg/concepts/template-parm6.C [deleted file]
gcc/testsuite/g++.dg/concepts/template-parm7.C [deleted file]
gcc/testsuite/g++.dg/concepts/template-parm9.C [deleted file]
gcc/testsuite/g++.dg/concepts/template-template-parm1.C
gcc/testsuite/g++.dg/concepts/var-concept1.C
gcc/testsuite/g++.dg/concepts/var-concept2.C
gcc/testsuite/g++.dg/concepts/var-concept3.C
gcc/testsuite/g++.dg/concepts/var-concept4.C
gcc/testsuite/g++.dg/concepts/var-concept5.C
gcc/testsuite/g++.dg/concepts/var-concept6.C
gcc/testsuite/g++.dg/concepts/var-concept7.C
gcc/testsuite/g++.dg/concepts/var-templ1.C
gcc/testsuite/g++.dg/concepts/var-templ2.C
gcc/testsuite/g++.dg/concepts/var-templ3.C
gcc/testsuite/g++.dg/concepts/variadic1.C
gcc/testsuite/g++.dg/concepts/variadic2.C
gcc/testsuite/g++.dg/concepts/variadic3.C
gcc/testsuite/g++.dg/concepts/variadic4.C
gcc/testsuite/g++.dg/cpp0x/auto52.C
gcc/testsuite/g++.dg/cpp0x/lambda/lambda-err2.C
gcc/testsuite/g++.dg/cpp2a/concepts-access1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-alias.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-alias2.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-class.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-cmath.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-complete1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-constrained-parm.C [moved from gcc/testsuite/g++.dg/concepts/constrained-parm.C with 82% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-decltype.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-defarg1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-dep1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-dr1430.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-explicit-inst1.C [moved from gcc/testsuite/g++.dg/concepts/explicit-inst1.C with 56% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-explicit-inst2.C [moved from gcc/testsuite/g++.dg/concepts/explicit-inst2.C with 72% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-explicit-inst3.C [moved from gcc/testsuite/g++.dg/concepts/explicit-inst3.C with 51% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-explicit-inst4.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec1.C [moved from gcc/testsuite/g++.dg/concepts/explicit-spec1.C with 76% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec2.C [moved from gcc/testsuite/g++.dg/concepts/explicit-spec2.C with 54% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec3.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec4.C [moved from gcc/testsuite/g++.dg/concepts/explicit-spec4.C with 70% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec5.C [moved from gcc/testsuite/g++.dg/concepts/explicit-spec5.C with 53% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec6.C [moved from gcc/testsuite/g++.dg/concepts/explicit-spec6.C with 84% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-feature-macro.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-fn1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-fn2.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-fn3.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-fnparm1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-friend1.C [moved from gcc/testsuite/g++.dg/concepts/friend1.C with 71% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-friend2.C [moved from gcc/testsuite/g++.dg/concepts/friend2.C with 66% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-friend3.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-iconv1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-inherit-ctor2.C [moved from gcc/testsuite/g++.dg/concepts/inherit-ctor2.C with 60% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-inherit-ctor4.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-inherit-ctor5.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-lambda1.C [moved from gcc/testsuite/g++.dg/concepts/lambda1.C with 56% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-locations1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-member-concept.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-memfun-err.C [moved from gcc/testsuite/g++.dg/concepts/memfun-err.C with 54% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-memfun.C [moved from gcc/testsuite/g++.dg/concepts/memfun.C with 56% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-memtmpl1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-memtmpl2.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-nested1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-noexcept1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-p1141.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec.C [moved from gcc/testsuite/g++.dg/concepts/partial-spec.C with 84% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec2.C [moved from gcc/testsuite/g++.dg/concepts/partial-spec2.C with 67% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec3.C [moved from gcc/testsuite/g++.dg/concepts/partial-spec3.C with 82% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec4.C [moved from gcc/testsuite/g++.dg/concepts/partial-spec4.C with 87% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec5.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec6.C [moved from gcc/testsuite/g++.dg/concepts/partial-spec6.C with 72% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-placeholder1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr58500.C [moved from gcc/testsuite/g++.dg/concepts/pr58500.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr58534.C [moved from gcc/testsuite/g++.dg/concepts/pr58534.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr58535.C [moved from gcc/testsuite/g++.dg/concepts/pr58535.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr58536.C [moved from gcc/testsuite/g++.dg/concepts/pr58536.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr58548.C [moved from gcc/testsuite/g++.dg/concepts/pr58548.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr58549.C [moved from gcc/testsuite/g++.dg/concepts/pr58549.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr59200.C [moved from gcc/testsuite/g++.dg/concepts/regress/alias-decl-42.C with 75% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr60052.C [moved from gcc/testsuite/g++.dg/concepts/pr60052.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr60053.C [moved from gcc/testsuite/g++.dg/concepts/pr60053.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr60064.C [moved from gcc/testsuite/g++.dg/concepts/pr60064.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr60065.C [moved from gcc/testsuite/g++.dg/concepts/pr60065.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr60377.C [moved from gcc/testsuite/g++.dg/concepts/pr60377.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr60390.C [moved from gcc/testsuite/g++.dg/concepts/pr60390.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr60391.C [moved from gcc/testsuite/g++.dg/concepts/pr60391.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr60573.C [moved from gcc/testsuite/g++.dg/concepts/pr60573.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr65552.C [moved from gcc/testsuite/g++.dg/concepts/pr65552.C with 63% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr65575.C [moved from gcc/testsuite/g++.dg/concepts/pr65575.C with 63% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr65634.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr65636.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr65848.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr65854.C [moved from gcc/testsuite/g++.dg/concepts/pr65854.C with 70% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr66091.C [moved from gcc/testsuite/g++.dg/concepts/pr66091.C with 75% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr66844.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr66962.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67070.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67147.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67148.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67178.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67210.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67217.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-2.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-3.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-4.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-5.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67319.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67427.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67654.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67658.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67684.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67685.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67692.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67697.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67719.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67774.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67825.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67860.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67862.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr67969.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr68093-1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr68093-2.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr68372.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr68434.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr68683.C [moved from gcc/testsuite/g++.dg/concepts/pr68683.C with 66% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr68812.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr69235.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr71368.C [moved from gcc/testsuite/g++.dg/concepts/pr71368.C with 61% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr71385.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr71965.C [moved from gcc/testsuite/g++.dg/concepts/pr71965.C with 77% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr72415.C [moved from gcc/testsuite/g++.dg/concepts/memfun2.C with 94% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr78752.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr79759.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr80471.C [moved from gcc/testsuite/g++.dg/concepts/pr80471.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr80746.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr80773.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr82507.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr82740.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr84140.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr84551.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr84979-2.C [moved from gcc/testsuite/g++.dg/concepts/pr84979-2.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr84979-3.C [moved from gcc/testsuite/g++.dg/concepts/pr84979-3.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr84979.C [moved from gcc/testsuite/g++.dg/concepts/pr84979.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr84980.C [moved from gcc/testsuite/g++.dg/concepts/pr84980.C with 65% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr85265.C [moved from gcc/testsuite/g++.dg/concepts/pr85265.C with 70% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr85706.C [moved from gcc/testsuite/g++.dg/concepts/class-deduction2.C with 100% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-pr85808.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr86269.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-pr87441.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires10.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires11.C [moved from gcc/testsuite/g++.dg/concepts/req17.C with 56% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-requires12.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires13.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires14.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires15.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires16.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires17.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires2.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires3.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires4.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires5.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires6.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires7.C [moved from gcc/testsuite/g++.dg/concepts/req14.C with 52% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-requires8.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-requires9.C [moved from gcc/testsuite/g++.dg/concepts/req15.C with 81% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-sfinae1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-template-parm1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-template-parm10.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-template-parm2.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-template-parm5.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-template-parm6.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-template-parm8.C [moved from gcc/testsuite/g++.dg/concepts/template-parm8.C with 50% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-template-parm9.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-traits1.C [moved from gcc/testsuite/g++.dg/concepts/traits1.C with 60% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-traits2.C [moved from gcc/testsuite/g++.dg/concepts/traits2.C with 59% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-ts1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-ts2.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-ts3.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-ts4.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-ts5.C [moved from gcc/testsuite/g++.dg/concepts/member-concept.C with 78% similarity]
gcc/testsuite/g++.dg/cpp2a/concepts-ts6.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts-using1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts2.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts3.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/concepts4.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp2a/cond-triv2.C [new file with mode: 0644]
gcc/testsuite/lib/g++-dg.exp
gcc/testsuite/lib/target-supports.exp
gcc/timevar.def

index 330e13b23829282526f6448d22fe730d48828655..a02d2110c1ac6f6bdd9fca2bd20d43dd70bf2205 100644 (file)
@@ -1,3 +1,7 @@
+2019-10-08  Jason Merrill  <jason@redhat.com>
+
+       * doc/invoke.texi: Document -fconcepts-ts.
+
 2019-10-09  Richard Biener  <rguenther@suse.de>
 
        * tree-vect-loop.c (vect_is_simple_reduction): Simplify and
index 93077ff7c0e550175dc06b72e6679f6c8cb4236e..3012f8b66fc69c0204c50fac2b840496866967ca 100644 (file)
@@ -1,3 +1,17 @@
+2019-10-08  Andrew Sutton  <asutton@lock3software.com>
+           Jason Merrill  <jason@redhat.com>
+
+       Update the concepts implementation to conform to the C++20
+       specification, improve compile times, and generally clean up
+       the implementation.
+
+       * c-cppbuiltin.c (c_cpp_builtins): Use new feature test values for
+       concepts when -std=c++2a. Bump __cpp_concepts to 201907.
+       * c.opt: Add -Wconcepts-ts.
+       * c-opts.c (c_common_post_options): Warn when -fconcepts is used
+       with -std=c++2a. Disable warning for -fconcepts in C++20 mode.
+       (set_std_cxx2a): Enable concepts by default.
+
 2019-10-08  Joseph Myers  <joseph@codesourcery.com>
 
        * c-opts.c (c_common_post_options): Set
index 9e0ce428127e78dab9ebd7468525feece6341f58..d9941e74e2ee08c8732380d4423c70753784988a 100644 (file)
@@ -992,7 +992,13 @@ c_cpp_builtins (cpp_reader *pfile)
          cpp_define (pfile, "__cpp_constexpr_dynamic_alloc=201907");
        }
       if (flag_concepts)
-       cpp_define (pfile, "__cpp_concepts=201507");
+        {
+          if (cxx_dialect >= cxx2a)
+            /* FIXME: Update this to the value required by the IS.  */
+            cpp_define (pfile, "__cpp_concepts=201907");
+          else
+            cpp_define (pfile, "__cpp_concepts=201507");
+        }
       if (flag_tm)
        /* Use a value smaller than the 201505 specified in
           the TS, since we don't yet support atomic_cancel.  */
index 4ad24bd3ea04b87b6f676ad1bb085d4dba0d4ee4..0fffe60b1409c93c491a0f1e29a609e5ab73f40f 100644 (file)
@@ -1034,6 +1034,16 @@ c_common_post_options (const char **pfilename)
   if (warn_return_type == -1 && c_dialect_cxx ())
     warn_return_type = 1;
 
+  /* C++2a is the final version of concepts. We still use -fconcepts
+     to know when concepts are enabled. Note that -fconcepts-ts can
+     be used to include additional features, although modified to
+     work with the standard.  */
+  if (cxx_dialect >= cxx2a)
+    flag_concepts = 1;
+  else if (flag_concepts)
+    /* For -std=c++17 -fconcepts, imply -fconcepts-ts.  */
+    flag_concepts_ts = 1;
+
   if (num_in_fnames > 1)
     error ("too many filenames given; type %<%s %s%> for usage",
           progname, "--help");
@@ -1713,6 +1723,7 @@ set_std_cxx2a (int iso)
   flag_isoc94 = 1;
   flag_isoc99 = 1;
   flag_isoc11 = 1;
+  /* C++2a includes concepts. */
   cxx_dialect = cxx2a;
   lang_hooks.name = "GNU C++17"; /* Pretend C++17 until standardization.  */
 }
index 88bbe2e20850ef22b7720dbd03cfe0b6e38b101a..8f6867b855e31eea7bb354ab1ae052dd5d4ad634 100644 (file)
@@ -1427,6 +1427,10 @@ fconcepts
 C++ ObjC++ Var(flag_concepts)
 Enable support for C++ concepts.
 
+fconcepts-ts
+C++ ObjC++ Var(flag_concepts_ts) Init(0)
+Enable certain features present in the Concepts TS.
+
 fcond-mismatch
 C ObjC C++ ObjC++
 Allow the arguments of the '?' operator to have different types.
index d75ea5dc44386c52a27d603e08c2fae28370af98..e633823ebaebd86b349925dba29153232919d75f 100644 (file)
@@ -1,3 +1,461 @@
+2019-10-08  Andrew Sutton  <asutton@lock3software.com>
+           Jason Merrill  <jason@redhat.com>
+
+       Update the concepts implementation to conform to the C++20
+       specification, improve compile times, and generally clean up
+       the implementation.
+       * call.c (build_new_function_call): Don't evaluate concepts here.
+       (constraint_failure): Don't record the template.
+       (print_z_candidate): Don't extract the template.
+       * class.c (add_method): When overloading, hide ineligible special
+       member fns.
+       (check_methods): Set TYPE_HAS_COMPLEX_* here.
+       * constexpr.c (cxx_eval_constant_expression): Evaluate concepts.
+       (maybe_initialize_fundef_copies_table): Remove.
+       (get_fundef_copy): Use hash_map_safe_get_or_insert.
+       (clear_cv_and_fold_caches): Clear the satisfaction cache.
+       * constraint.cc (known_non_bool_p): New.
+       (parsing_constraint_expression_sentinel): Renamed from
+       expanding_constraint_sentinel.
+       (check_constraint_operands): New.
+       (check_constraint_atom): New.
+       (finish_constraint_binary_op): New.
+       (finish_constraint_or_expr): Likewise.
+       (finish_constraint_and_expr): Likewise.
+       (finish_constraint_primary_expr): Likewise.
+       (combine_constraint_expressions): New.
+       (finish_requires_expr): Add location parm.
+       (get_concept_definition): Return the initializer of concept definitions.
+       (get_template_head_requirements): New.
+       (get_trailing_function_requirements): New.
+       (deduce_constrained_parameter): Check if the identifier or template-id
+       is a concept definition.
+       (resolve_concept_definition_check): Removed.
+       (resolve_variable_concept_check): Removed.
+       (resolve_concept_check): New.
+       (resolve_constraint_check): Handle concept definitions.
+       converting arguments.
+       (function_concept_check_p): Removed.
+       (variable_concept_check_p): Removed.
+       (unpack_concept_check): New.
+       (get_concept_check_template): New.
+       (build_call_check): Moved and renamed to build_function_check.
+       (build_concept_check_arguments): make static.
+       (build_function_check): Always do overload resolution
+       in order to force conversion of template arguments (i.e., actually
+       check that the use of a concept is valid).
+       (build_standard_check): Renamed from build_real_concept_check.
+       (build_real_concept_check): Build checks for C++2a concepts by
+       (build_wildcard_concept_check): New.
+       (build_concept_check): Use build_real_concept_check. New overload.
+       (build_constraints): Save expressions, not normalized constraints.
+       (build_concept_id): New. Pass tf_warning_or_error.
+       (build_type_constraint): New.
+       (finish_type_constraints): New.
+       (associate_classtype_constraints): Also add constraints to union
+       types. Note the original declaration in errors. Don't return
+       error_mark_node in order to avoid an assertion later.
+       (push_down_pack_expansion): Remove.
+       (finish_shorthand_constraint): Make fold expressions, not naked
+       parameter packs. Always apply the constraint to each template argument.
+       (check_introduction_list): New. Fail if not enough
+       names are introduced.
+       (finish_template_introduction): Don't normalize constraints. Pass
+       tsubst flags. Check for insufficient introductions.
+       (placeholder_extract_concept_and_args): Handle the template-id case.
+       Unpack function concept checks correctly.
+       (tsubst_simple_requirement): Return errors if they occur. Don't
+       process as a template.
+       (tsubst_type_requirement): Likewise.
+       (type_deducible_p): New. Wrap the input expression in parens for the
+       purpose of deduction.
+       (expression_convertible_t): New.
+       (tsubst_compound_requirement): Use new deduction, conversion predicates.
+       (tsubst_nested_requirement): Return errors if they occur. Don't
+       process as a template. Instantiate and evaluate the nested requirement.
+       (tsubst_valid_expression_requirement): New.
+       (tsubst_simple_requirement): Use tsubst_valid_expression_requirement.
+       (tsubst_compound_requirement): Use tsubst_valid_expression_requirement.
+       (check_constaint_variables): New.
+       (tsubst_constraint_variables): Check that type substitutions are valid.
+       (tsubst_requires_expr): Likewise. Produce new requires-exprs during
+       template substitution. Copy the previous local specialization stack,
+       so references to non-local parameters can be found. Use cp_unevaluated.
+       (tsubst_constraint): New. Don't evaluate concept checks.
+       (subst_info): New.
+       (norm_info): New. Used to build a normalization tree for concept check
+       diagnostics.
+       (debug_parameter_mapping): New.
+       (debug_argument_list): New.
+       (expand_concept): Removed.
+       (normalize_logical_operation): Pass subst_info through call.
+       (normalize_pack_expansion): Remove.
+       (normalize_simple_requirement): Removed
+       (normalize_type_requirement): Removed
+       (normalize_compound_requirement): Removed
+       (normalize_nested_requirement): Removed
+       (normalize_requirement): Removed
+       (normalize_requirements): Removed
+       (normalize_requires_expression): Removed
+       (normalize_variable_concept_check): Removed.
+       (normalize_function_concept_check): Removed.
+       (normalize_concept_check): Merged all normalize_*_check here.
+       Substitute through written template arguments before normalizing the
+       definition. Only substitute the innermost template arguments.
+       (check_for_logical_overloads): Delete.
+       (map_arguments): New. Associate template parameters with arguments.
+       (build_parameter_mapping): New. Extract used parameters.
+       (normalize_expression): Rewrite.
+       (normalize_conjunction): Removed
+       (normalize_disjunction): Removed
+       (normalize_predicate_constraint): Removed
+       (normalize_parameterized_constraint): Removed
+       (normalized_map): New variable.
+       (get_normalized_constraints): New entry point for normalization.
+       Establishes a timer.
+       (get_normalized_constraints_from_info): New.
+       (get_normalized_constraints_from_decl): New. Turn on template processing
+       prior to normalization. Handle inheriting ctors. Build the
+       normalization arguments from the full set of template parameters of the
+       most general template. This guarantees that we have no concrete arguments
+       in the parameter mapping (e.g., from template members of class
+       templates). Cache normalizations.
+       (normalize_concept_definition): New. Cache normalizations.
+       (normalize_template_requirements): New.
+       (normalize_nontemplate_requirements): New.
+       (normalize_constraint_expression): New.
+       (tsubst_parameter_mapping): New.
+       (get_mapped_args): New.
+       (parameter_mapping_equivalent_p): New. Use template_args_equal.
+       (atomic_constraints_identical_p): New.
+       (hash_atomic_constraint): New.
+       (satisfying_constraint_p): New. Guard against recursive evaluation of
+       constraints during satisfaction.
+       (satisfy_conjunction): New.
+       (satisfy_disjunction): New.
+       (sat_entry): New class for hashing satisfaction results.
+       (sat_hasher): New hash traits.
+       (sat_cache): New.
+       (get_satisfaction): New. Returns cached satisfaction result.
+       (save_satisfaction): New. Caches a satisfaction result.
+       (clear_satisfaction_cache): New.
+       (satisfaction_cache): New. Helps manage satisfaction cache requests.
+       (decl_satisfied_cache): New.
+       (satisfy_atom): New.
+       (satisfy_constraint_r): New.
+       (satisfy_constraint): Use new satisfaction algorithm.
+       (evaluate_concept_check): New.
+       (evaluate_concept): Removed.
+       (evaluate_function_concept): Removed.
+       (evaluate_variable_concept): Removed.
+       (satisfy_constraint_expression): New.
+       (constraint_expression_satisfied_p): New.
+       (constraints_satisfied_p): Use strip_inheriting_ctors. Use
+       push_/pop_access_scope.
+       (more_constrained): Normalize before calling out to subsumption. Allow
+       classes as arguments.
+       (strictly_subsumes): Allow non-templates as arguments. Accept a new
+       template argument.
+       (weakly_subsumes): New.
+       (at_least_as_constrained): Removed.
+       (diagnose_other_expression): Removed.
+       (diagnose_predicate_constraint): Removed.
+       (diagnose_pack_expansion): Removed.
+       (diagnose_check_constraint): Removed.
+       (diagnose_logical_constraint): Removed.
+       (diagnose_expression_constraint): Removed.
+       (diagnose_type_constraint): Removed.
+       (diagnose_implicit_conversion_constraint): Removed.
+       (diagnose_argument_deduction_constraint): Removed.
+       (diagnose_exception_constraint): Removed.
+       (diagnose_parameterized_constraint): Removed.
+       (diagnose_argument_deduction_constraint): Removed.
+       (diagnose_argument_deduction_constraint): Removed.
+       (diagnose_argument_deduction_constraint): Removed.
+       (diagnose_trait_expr): New.
+       (diagnose_requires_expr): New.
+       (diagnose_atomic_constraint): New.
+       (diagnose_valid_expression) Stop wrongly diagnosing valid expressions.
+       Don't substitute as if in template decls. This causes substitution
+       to generate expressions that aren't suitable for use with the noexcept
+       routines.
+       (diagnose_valid_type) Likewise.
+       (diagnose_compound_requirement) Actually emit diagnostics for
+       the causes of errors.Call force_paren_expr_uneval.
+       (diagnose_declaration_constraints): Turn on template processing to
+       suppress certain analyses.
+       * cp-objcp-common.c (cp_common_init_ts): Make concepts typed.
+       (cp_get_debug_type): Use hash_map_safe_*.
+       * cp-tree.h: New function declarations for semantic actions, other
+       facilities. Remove declaration no longer used or needed. Remove
+       unused _CONSTR macros.
+       (LANG_DECL_HAS_MIN): Add CONCEPT_DECL.
+       (template_info_decl_check): Factor macro check into an inline function.
+       (DECL_TEMPLATE_INFO): Use new check facility.
+       (finish_concept_definition): New. Don't invalid concept declarations
+       with invalid initializers.
+       (find_template_parameters): New.
+       (concept_definition_p): New.
+       (concept_check_p): New.
+       (variable_concept_check_p): New.
+       (force_paren_expr_uneval): New.
+       (ovl_iterator::using_p): A USING_DECL by itself was also
+       introduced by a using-declaration.
+       (struct tree_template_info): Use tree_base instead of
+       tree_common. Add tmpl and args fields.
+       (TI_TEMPLATE, TI_ARGS): Adjust.
+       (DECLTYPE_FOR_INIT_CAPTURE): Remove.
+       (CONSTR_CHECK, CONSTR_INFO, CONSTR_EXPR, CONSTR_CONTEXT): New.
+       (ATOMIC_CONSTR_MAP, TRAIT_EXPR_LOCATION): New.
+       (struct tree_trait_expr): Add locus field.
+       (enum tsubst_flags): Add tf_norm as a hint to generate normalization
+       context when diagnosing constraint failure.
+       * cp-tree.def: Remove unused _CONSTR nodes and rename PRED_CONSTR
+       to ATOMIC_CONSTR.
+       (CONCEPT_DECL): New.
+       * cxx-pretty-print.c: Remove constraint printing code.
+       (pp_cxx_concept_definition): New.
+       (pp_cxx_template_declaration): Print concept definitions.
+       (pp_cxx_check_constraint): Update printing for concept definitions.
+       (pp_cxx_nested_name_specifier): Fix a weird
+       case where we're printing '::::' for concepts.
+       (simple_type_specifier): Print requirements for placeholder types.
+       (pp_cxx_constrained_type_spec): Print the associated requirements of
+       a placeholder type.
+       (pp_cxx_compound_requirement): Add space before the '->'.
+       (pp_cxx_parameter_mapping): Print the parameter mapping.
+       (pp_cxx_atomic_constraint): Use the function above.
+       * decl.c (redeclaration_error_message): New error for concepts.
+       (grokdeclarator): Check for and disallow decltype(auto) in parameter
+       declarations.
+       (grokfndecl): Don't normalize constraints. Add check for constraints
+       on declaration.
+       (grokvardecl): Don't normalize constraints.
+       (grok_special_member_properties): Don't set TYPE_HAS_COMPLEX_*.
+       (function_requirements_equivalent_p): New. Compare trailing
+       requires clauses. Compare combined constraints in pre-C++20 mode.
+       (decls_match): Compare trailing requires clauses. Compare template
+       heads for function templates. Remove old constraint comparison.
+       Simplify comparison of functions, function templates.
+       (duplicate_function_template_decls): New. Refactor a nasty if
+       condition into a single predicate.
+       (require_deduced_type): Don't complain if we already complained about
+       deduction failure.
+       (finish_function): Perform auto deduction to ensure that constraints
+       are checked even when functions contain no return statements. Only do
+       auto deduction if we haven't previously seen any return statements.
+       This prevents multiple diagnostics of the same error.
+       (store_decomp_type): Remove.
+       (cp_finish_decomp): Use hash_map_safe_put.
+       * error.c: Remove constraint printing code.
+       (dump_decl): Dump concept definitions. Handle wildcard declarations.
+       (dump_template_decl): Likewise.
+       (dump_type): Print associated requirements for placeholder
+       types.
+       (rebuild_concept_check): New.
+       (maybe_print_single_constraint_context): New.
+       (maybe_print_constraint_context): Recursively print nested contexts.
+       * init.c (get_nsdmi): Use hash_map_safe_*.
+       * lambda.c (maybe_add_lambda_conv_op): Bail if deduction failed.
+       (add_capture): Copy parameter packs from init.
+       (lambda_capture_field_type): Always use auto for init-capture.
+       * logic.cc: Completely rewrite.
+       (constraint_hash): New.
+       (clause/ctor): Save atoms in the hash table.
+       (replace): Save atoms during replacement.
+       (insert): Save atoms during insertion.
+       (contains): Only search the hash table for containment.
+       (clause): Keep a hash of atomic constraints.
+       (clause::clause): Explicitly copy the hash table when copying.
+       (disjunction_p, conjunction_p, atomic_p, dnf_size, cnf_size): New.
+       (diagnose_constraint_size): New.
+       (subsumes_constraints_nonnull): Compare the sizes of normalized formula
+       to determine the cheapest decomposition.
+       * name-lookup.c (diagnose_name_conflict): Diagnose name issues with
+       concepts.
+       (matching_fn_p): Check constraints.
+       (push_class_level_binding_1): Move overloaded functions case down,
+       accept FUNCTION_DECL as target_decl.
+       * parser.c (enum required_token): New required token for auto.
+       (make_location): Add overload taking lexer as last parm.
+       (cp_parser_required_error): Diagnose missing auto.
+       (cp_parser_diagnose_ungrouped_constraint_plain): New.
+       (cp_parser_diagnose_ungrouped_constraint_plain): New.
+       (cp_parser_constraint_primary_expression): New. Tentatively parse the
+       primary expression. If that fails tentatively parse a lower
+       precedence expression in order to diagnose the error.
+       (cp_parser_check_non_logical_constraint): New. Performs a trial
+       parse of the right-hand-side of non-logical operators in order to
+       generate good diagnostics.
+       (cp_parser_constraint_logical_and_expression): New.
+       (cp_parser_constraint_logical_or_expression): New.
+       (cp_parser_requires_clause_expression): New.
+       (cp_parser_requires_clause): Renamed to cp_parser_constraint_expression.
+       (cp_parser_requires_clause_opt): Parse the requires-clause differently
+       in -fconcepts and -std=c++2a modes.
+       (cp_parser_requirement_list): Rename to cp_parser_requirement_seq.
+       Rewrite so that semicolons are parsed
+       along with requirements, not the sequence.
+       (cp_parser_simple_requirement): Expect a semicolon at end.
+       (cp_parser_compound_requirement): Expect a semicolon at end. Only
+       allow trailing-return-type with -fconcepts-ts.
+       (cp_parser_nested_requirement): Expect a semicolon at end. Parse
+       constraint-expressions.
+       (cp_parser_concept_definition): New. Don't fail parsing the concept
+       definition if the initializer is ill-formed. Don't declare the concept
+       before parsing the initializer.
+       (cp_parser_constraint_expression): Declare earlier.
+       (cp_parser_type_requirement): Current scope is not valid.
+       (cp_parser_requires_expression): Commit to the tentative parse.
+       (cp_parser_decl_specifier_seq): Warn when concept appears to be used
+       as a decl-specifier.
+       (cp_parser_template_declaration_after_parameters): Parse concept
+       definitions.
+       (cp_parser_template_id): Don't try to resolve a concept template-id yet.
+       (cp_parser_template_id_expr): Resolve it as a concept check.
+       (cp_parser_decl_specifier_seq): Warn on 'concept bool'.
+       (cp_parser_type_parameter): Combine expressions not
+       constraints.
+       (cp_parser_explicit_template_declaration): Combine expressions not
+       constraints.
+       (cp_parser_maybe_concept_name): Removed.
+       (cp_parser_simple_type_specifier): Handle an error condition of
+       a bad constrained type specifier. Expect auto or decltype after
+       a concept name. Also handle the case where we have a template-id
+       as a concept check.
+       (cp_parser_template_introduction): Diagnose errors on invalid
+       introductions. Give up if it doesn't start with a concept name.
+       Pedwarn if not -fconcepts-ts.
+       (synthesize_implicit_template_parm): Don't do consistent binding.
+       Use a new flag for constrained parameters. Combine expressions,
+       not constraints. Fail if we get a placeholder in block scope.
+       Placeholders that do not constrain types are not allowed in parameter
+       declarations, so don't handle them.
+       (cp_parser_placeholder_type_specifier): New. Implement parsing of
+       placeholder type specifiers following a concept name or partial
+       concept check. Disallow decltype(auto) parameters.
+       (cp_parser_nested_name_specifier_opt): If the token is already
+       CPP_NESTED_NAME_SPECIFIER, leave it alone.
+       (cp_parser_id_expression, cp_parser_unqualified_id): Call
+       cp_parser_template_id_expr.
+       (cp_parser_placeholder_type_specifier): Add tentative parm.  Don't
+       expect a WILDCARD_DECL.
+       (cp_parser_trait_expr): Pass trait_loc down.
+       (cp_parser_postfix_expression): Do set location of dependent member
+       call.
+       * pt.c (finish_concept_definition): New.
+       (push_template_decl_real): Handle concept definitions.
+       (start_concept_definition): Let push_template_decl_real handle the
+       creation of the template.
+       (get_constraints): Return null if the table hasn't been initialized.
+       (tsubst_copy_and_build): Build template-id expressions for concept
+       checks.
+       [TRAIT_EXPR]: Pass trait_loc down.
+       (lookup_template_class_1): Add the template name to the constraint
+       failure diagnostic.
+       (lookup_and_finish_template_variable): Build concept checks
+       with the correct arguments.
+       (tsubst_function_decl): Don't substitute through constraints.
+       Always associate constraints with functions.
+       (template_parm_level_and_index): Make non-static.
+       (for_each_template_parm_r): Handle requires expressions.
+       (keep_template_parm): New.
+       (find_template_parameters): New.
+       (more_specialized_fn): Change how winners and losers are chosen.
+       (make_constrained_auto): Don't normalize constraints.
+       (template_parameters_equivalent_p): New. Compare template
+       parameters. Add a comparison for implicitly vs. explicitly declared
+       parameters.
+       (template_parameter_lists_equivalent_p): New. Compare template
+       parameter lists.
+       (template_requirements_equivalent_p): New.
+       (template_heads_equivalent_p): New. Compare template heads.
+       (template_parameter_constraints_equivalent_p): New.
+       (is_compatible_template_arg): Use weakly_subsumes.
+       (maybe_new_partial_specialization): Use new constraint comparison
+       for finding specializations.
+       (process_partial_specialization): Pass main template as argument.
+       (more_specialized_partial_spec): Don't immediately return when
+       detecting a winner.
+       (make_constrained_auto): Handle concept definitions.
+       (do_auto_deduction): Update auto deduction for new concept model.
+       Extract the function concept correctly; rename constr to check to
+       reflect the kind of node.
+       (tsubst): Adjust wildcard argument during substitution.
+       [DECLTYPE_TYPE]: Remove init-capture handling.
+       (tsubst_copy_and_build): Build concept checks, not template ids.
+       Defer checks of function concepts. Handle concepts before variable
+       templates. Handle calls to function concepts explicitly.
+       (coerce_template_parms): Use concept_definition_p. Handle a deduction
+       error where a potentially empty pack can be supplied after the last
+       parameter of a concept.
+       (finish_template_variable): Don't process concepts here.
+       (instantiation_dependent_r): Use concept_check_p.
+       (tsubst_template_args): Make non-static.
+       (make_constrained_placeholder_type): New. Refactored from
+       make_constrained_auto.
+       (make_constrained_auto) Use make_constrained_placeholder_type.
+       (make_constrained_decltype_auto) New.
+       (tsubst_function_parms): New.
+       (value_dependent_expression_p) [TEMPLATE_ID_EXPR]: Use
+       concept_definition_p.
+       (push_access_scope, pop_access_scope): No longer static.
+       (tsubst_template_parm): Substitute TEMPLATE_PARM_CONSTRAINTS.
+       (tsubst_friend_function): Use tsubst_constraint. Use generic_targs_for.
+       (get_underlying_template) Use generic_targs_for.
+       (uses_parameter_packs): Return tree.
+       (gen_elem_of_pack_expansion_instantiation): Don't push
+       local_specialization_stack.
+       (prepend_one_capture): New.
+       (tsubst_lambda_expr): Use prepend_one_capture.  Don't touch
+       local_specializations.
+       (template_parms_level_to_args): No longer static.
+       (add_outermost_template_args): Likewise.
+       (find_template_parameter_info): New. Provide context for finding
+       template parameters.
+       (keep_template_parm): Don't keep parameters declared at depth levels
+       greater than those of the template parameters of the source declaration.
+       Don't propagate cv-qualified types. Return 0, so we find all template
+       parameters, not the just first.
+       (any_template_parm_r): New. Handle cases that are mishandled by
+       for_each_template_parm_r.
+       (generic_targs_for): Factor out of coerce_template_args_for_ttp.
+       (tsubst_argument_pack): Factor out of tsubst_template_args.
+       (constraint_sat_entry): Removed.
+       (constraint_sat_hasher): Removed.
+       (concept_spec_entry): Removed.
+       (concept_spec_hasher): Removed.
+       (constraint_memos): Removed.
+       (concept_memos): Removed.
+       (lookup_constraint_satisfaction): Removed.
+       (memoize_constraint_satisfaction): Removed.
+       (lookup_concept_satisfaction): Removed.
+       (memoize_concept_satisfaction): Removed.
+       (concept_expansions): Removed.
+       (get_concept_expansion): Removed.
+       (save_concept_expansion): Removed.
+       (init_constraint_processing): Remove initialization of non-existing
+       resources.
+       (find_template_requirement): New. Search for the sub-requirement
+       within the associated constraints.
+       (convert_generic_types_to_packs): Also transform the associated
+       constraint and update the current template requirements.
+       (store_defaulted_ttp, lookup_defaulted_ttp): Remove.
+       (add_defaults_to_ttp): Use hash_map_safe_*.
+       * semantics.c (finish_call_expr): Diagnose calls to concepts.
+       Handle concept checks explicitly.
+       (finish_id_expression): Evaluate variable concepts as part of
+       id-expression processing. Don't treat variable concepts as variables,
+       and don't process function concepts as plain id-expressions.
+       (force_paren_expr): Add even_uneval parm.
+       (finish_trait_expr): Add location parm.
+       * tree.c (special_memfn_p): New.
+       (cp_expr_location): Handle TRAIT_EXPR.
+       * typeck.c (check_return_expr): Actually use the diagnostic kind
+       when performing return-type deduction.
+       * typeck2.c (build_functional_cast): Don't rely on the location of
+       'auto'.
+
 2019-10-09  Paolo Carlini  <paolo.carlini@oracle.com>
 
        * decl.c (grok_ctor_properties): Use DECL_SOURCE_LOCATION.
index 4ed424f3a6c8ce1a0ddab8851987cca6c23b3d54..d36564d6527f0776999b846ba4beefb9e15da78f 100644 (file)
@@ -717,24 +717,12 @@ inherited_ctor_rejection (void)
   return r;
 }
 
-// Build a constraint failure record, saving information into the
-// template_instantiation field of the rejection. If FN is not a template
-// declaration, the TMPL member is the FN declaration and TARGS is empty.
+/* Build a constraint failure record.  */
 
 static struct rejection_reason *
-constraint_failure (tree fn)
+constraint_failure (void)
 {
   struct rejection_reason *r = alloc_rejection (rr_constraint_failure);
-  if (tree ti = DECL_TEMPLATE_INFO (fn))
-    {
-      r->u.template_instantiation.tmpl = TI_TEMPLATE (ti);
-      r->u.template_instantiation.targs = TI_ARGS (ti);
-    }
-  else
-    {
-      r->u.template_instantiation.tmpl = fn;
-      r->u.template_instantiation.targs = NULL_TREE;
-    }
   return r;
 }
 
@@ -2251,10 +2239,9 @@ add_function_candidate (struct z_candidate **candidates,
 
   /* Second, for a function to be viable, its constraints must be
      satisfied. */
-  if (flag_concepts && viable
-      && !constraints_satisfied_p (fn))
+  if (flag_concepts && viable && !constraints_satisfied_p (fn))
     {
-      reason = constraint_failure (fn);
+      reason = constraint_failure ();
       viable = false;
     }
 
@@ -3729,11 +3716,7 @@ print_z_candidate (location_t loc, const char *msgstr,
                  "class type is invalid");
          break;
        case rr_constraint_failure:
-         {
-           tree tmpl = r->u.template_instantiation.tmpl;
-           tree args = r->u.template_instantiation.targs;
-           diagnose_constraints (cloc, tmpl, args);
-         }
+         diagnose_constraints (cloc, fn, NULL_TREE);
          break;
        case rr_inherited_ctor:
          inform (cloc, "  an inherited constructor is not a candidate for "
@@ -4532,25 +4515,7 @@ build_new_function_call (tree fn, vec<tree, va_gc> **args,
          through flags so that later we can use it to decide whether to warn
          about peculiar null pointer conversion.  */
       if (TREE_CODE (fn) == TEMPLATE_ID_EXPR)
-        {
-          /* If overload resolution selects a specialization of a
-             function concept for non-dependent template arguments,
-             the expression is true if the constraints are satisfied
-             and false otherwise.
-
-             NOTE: This is an extension of Concepts Lite TS that
-             allows constraints to be used in expressions. */
-          if (flag_concepts && !processing_template_decl)
-            {
-              tree tmpl = DECL_TI_TEMPLATE (cand->fn);
-              tree targs = DECL_TI_ARGS (cand->fn);
-              tree decl = DECL_TEMPLATE_RESULT (tmpl);
-              if (DECL_DECLARED_CONCEPT_P (decl))
-                return evaluate_function_concept (decl, targs);
-            }
-
-          flags |= LOOKUP_EXPLICIT_TMPL_ARGS;
-        }
+        flags |= LOOKUP_EXPLICIT_TMPL_ARGS;
 
       result = build_over_call (cand, flags, complain);
     }
@@ -10828,8 +10793,11 @@ joust (struct z_candidate *cand1, struct z_candidate *cand2, bool warn,
        return winner;
     }
 
-  // C++ Concepts
-  // or, if not that, F1 is more constrained than F2.
+  /* Concepts: ... or, if not that, F1 is more constrained than F2.
+
+     FIXME: For function templates with no winner, this subsumption may
+     be computed a separate time.  This needs to be validated, and if
+     so, the redundant check removed.  */
   if (flag_concepts && DECL_P (cand1->fn) && DECL_P (cand2->fn))
     {
       winner = more_constrained (cand1->fn, cand2->fn);
index 4abcfaf4c1d2fa8dc92634ab4e4bb5d6dc4b1488..b6afdc487e70dd9ead93db9322af6a8c425a244a 100644 (file)
@@ -7189,6 +7189,9 @@ finish_struct_1 (tree t)
   /* Finish debugging output for this type.  */
   rest_of_type_compilation (t, ! LOCAL_CLASS_P (t));
 
+  /* Recalculate satisfaction that might depend on completeness.  */
+  clear_satisfaction_cache ();
+
   if (TYPE_TRANSPARENT_AGGR (t))
     {
       tree field = first_field (t);
index 6e5f04677d69efce9eaea242c3c9ed44fc7e1379..adfe1b2415d6b2e372622401c5b991c7f772188e 100644 (file)
@@ -39,6 +39,7 @@ gtfiles="\
 \$(srcdir)/c-family/c-common.c \$(srcdir)/c-family/c-format.c \
 \$(srcdir)/c-family/c-cppbuiltin.c \$(srcdir)/c-family/c-pragma.c \
 \$(srcdir)/cp/call.c \$(srcdir)/cp/class.c \$(srcdir)/cp/constexpr.c \
+\$(srcdir)/cp/constraint.cc \
 \$(srcdir)/cp/cp-gimplify.c \
 \$(srcdir)/cp/cp-lang.c \$(srcdir)/cp/cp-objcp-common.c \
 \$(srcdir)/cp/decl.c \$(srcdir)/cp/decl2.c \
index f50cc04c2932b3a3256a9080b582d5dffbb00139..ea502343e40176262bfaa1c177038c891d1de02c 100644 (file)
@@ -5523,7 +5523,7 @@ cxx_eval_constant_expression (const constexpr_ctx *ctx, tree t,
          '!requires (T t) { ... }' which is not transformed into
          a constraint.  */
       if (!processing_template_decl)
-        return evaluate_constraint_expression (t, NULL_TREE);
+        return satisfy_constraint_expression (t);
       else
         *non_constant_p = true;
       return t;
@@ -5539,6 +5539,20 @@ cxx_eval_constant_expression (const constexpr_ctx *ctx, tree t,
       r = void_node;
       break;
 
+    case TEMPLATE_ID_EXPR:
+      {
+        /* We can evaluate template-id that refers to a concept only if
+          the template arguments are non-dependent.  */
+       if (!concept_definition_p (TREE_OPERAND (t, 0)))
+         internal_error ("unexpected template-id %qE", t);
+
+       if (!processing_template_decl)
+         return satisfy_constraint_expression (t);
+       else
+         *non_constant_p = true;
+       return t;
+      }
+
     case ASM_EXPR:
       if (!ctx->quiet)
        inline_asm_in_constexpr_error (cp_expr_loc_or_input_loc (t));
@@ -5980,13 +5994,15 @@ clear_cv_cache (void)
     cv_cache->empty ();
 }
 
-/* Dispose of the whole CV_CACHE and FOLD_CACHE.  */
+/* Dispose of the whole CV_CACHE, FOLD_CACHE, and satisfaction caches.  */
 
 void
-clear_cv_and_fold_caches (void)
+clear_cv_and_fold_caches (bool sat /*= true*/)
 {
   clear_cv_cache ();
   clear_fold_cache ();
+  if (sat)
+    clear_satisfaction_cache ();
 }
 
 /* Internal function handling expressions in templates for
index c7a172cce4ddb2f913f8462863447afe8dd0646b..db4a81858f65ea24340b51a64bcefa88faf51221 100644 (file)
@@ -46,89 +46,189 @@ along with GCC; see the file COPYING3.  If not see
 #include "toplev.h"
 #include "type-utils.h"
 
+static tree satisfaction_value (tree t);
+
+/* When we're parsing or substuting a constraint expression, we have slightly
+   different expression semantics.  In particular, we don't want to reduce a
+   concept-id to a satisfaction value.  */
+
+processing_constraint_expression_sentinel::
+processing_constraint_expression_sentinel ()
+{
+  ++scope_chain->x_processing_constraint;
+}
+
+processing_constraint_expression_sentinel::
+~processing_constraint_expression_sentinel ()
+{
+  --scope_chain->x_processing_constraint;
+}
+
+bool
+processing_constraint_expression_p ()
+{
+  return scope_chain->x_processing_constraint != 0;
+}
+
 /*---------------------------------------------------------------------------
-                       Operations on constraints
+                      Constraint expressions
 ---------------------------------------------------------------------------*/
 
-/* Returns true if C is a constraint tree code. Note that ERROR_MARK
-   is a valid constraint.  */
+/* Information provided to substitution.  */
+
+struct subst_info
+{
+  subst_info (tsubst_flags_t cmp, tree in)
+    : complain (cmp), in_decl (in)
+  { }
+
+  /* True if we should not diagnose errors.  */
+  bool quiet() const
+  {
+    return complain == tf_none;
+  }
+
+  /* True if we should diagnose errors.  */
+  bool noisy() const
+  {
+    return !quiet ();
+  }
+
+  tsubst_flags_t complain;
+  tree in_decl;
+};
+
+/* True if T is known to be some type other than bool. Note that this
+   is false for dependent types and errors.  */
 
 static inline bool
-constraint_p (tree_code c)
+known_non_bool_p (tree t)
 {
-  return ((PRED_CONSTR <= c && c <= DISJ_CONSTR)
-          || c == EXPR_PACK_EXPANSION
-          || c == ERROR_MARK);
+  return (t && !WILDCARD_TYPE_P (t) && TREE_CODE (t) != BOOLEAN_TYPE);
 }
 
-/* Returns true if T is a constraint. Note that error_mark_node
-   is a valid constraint.  */
+static bool
+check_constraint_atom (cp_expr expr)
+{
+  if (known_non_bool_p (TREE_TYPE (expr)))
+    {
+      error_at (expr.get_location (),
+               "constraint expression does not have type %<bool%>");
+      return false;
+    }
 
-bool
-constraint_p (tree t)
+  /* Check that we're using function concepts correctly.  */
+  if (concept_check_p (expr))
+    {
+      tree id = unpack_concept_check (expr);
+      tree tmpl = TREE_OPERAND (id, 0);
+      if (OVL_P (tmpl) && TREE_CODE (expr) == TEMPLATE_ID_EXPR)
+        {
+         error_at (EXPR_LOC_OR_LOC (expr, input_location),
+                   "function concept must be called");
+         return false;
+       }
+    }
+
+  return true;
+}
+
+static bool
+check_constraint_operands (location_t, cp_expr lhs, cp_expr rhs)
 {
-  return constraint_p (TREE_CODE (t));
+  return check_constraint_atom (lhs) && check_constraint_atom (rhs);
 }
 
-/* Returns the conjunction of two constraints A and B. Note that
-   conjoining a non-null constraint with NULL_TREE is an identity
-   operation. That is, for non-null A,
+/* Validate the semantic properties of the constraint expression.  */
+
+static cp_expr
+finish_constraint_binary_op (location_t loc,
+                            tree_code code,
+                            cp_expr lhs,
+                            cp_expr rhs)
+{
+  gcc_assert (processing_constraint_expression_p ());
+  if (lhs == error_mark_node || rhs == error_mark_node)
+    return error_mark_node;
+  if (!check_constraint_operands (loc, lhs, rhs))
+    return error_mark_node;
+  tree overload;
+  tree expr = build_x_binary_op (loc, code,
+                                lhs, TREE_CODE (lhs),
+                                rhs, TREE_CODE (rhs),
+                                &overload, tf_none);
+  /* When either operand is dependent, the overload set may be non-empty.  */
+  if (expr == error_mark_node)
+    return error_mark_node;
+  SET_EXPR_LOCATION (expr, loc);
+  return expr;
+}
 
-      conjoin_constraints(a, NULL_TREE) == a
+cp_expr
+finish_constraint_or_expr (location_t loc, cp_expr lhs, cp_expr rhs)
+{
+  return finish_constraint_binary_op (loc, TRUTH_ORIF_EXPR, lhs, rhs);
+}
 
-   and
+cp_expr
+finish_constraint_and_expr (location_t loc, cp_expr lhs, cp_expr rhs)
+{
+  return finish_constraint_binary_op (loc, TRUTH_ANDIF_EXPR, lhs, rhs);
+}
 
-      conjoin_constraints (NULL_TREE, a) == a
+cp_expr
+finish_constraint_primary_expr (cp_expr expr)
+{
+  if (expr == error_mark_node)
+    return error_mark_node;
+  if (!check_constraint_atom (expr))
+    return cp_expr (error_mark_node, expr.get_location ());
+  return expr;
+}
 
-   If both A and B are NULL_TREE, the result is also NULL_TREE. */
+/* Combine two constraint-expressions with a logical-and.  */
 
 tree
-conjoin_constraints (tree a, tree b)
-{
-  gcc_assert (a ? constraint_p (a) : true);
-  gcc_assert (b ? constraint_p (b) : true);
-  if (a)
-    return b ? build_nt (CONJ_CONSTR, a, b) : a;
-  else if (b)
-    return b;
-  else
-    return NULL_TREE;
+combine_constraint_expressions (tree lhs, tree rhs)
+{
+  processing_constraint_expression_sentinel pce;
+  if (!lhs)
+    return rhs;
+  if (!rhs)
+    return lhs;
+  return finish_constraint_and_expr (input_location, lhs, rhs);
 }
 
-/* Transform the vector of expressions in the T into a conjunction
-   of requirements. T must be a TREE_VEC. */
+/* Extract the template-id from a concept check. For standard and variable
+   checks, this is simply T. For function concept checks, this is the
+   called function.  */
 
 tree
-conjoin_constraints (tree t)
+unpack_concept_check (tree t)
 {
-  gcc_assert (TREE_CODE (t) == TREE_VEC);
-  tree r = NULL_TREE;
-  for (int i = 0; i < TREE_VEC_LENGTH (t); ++i)
-    r = conjoin_constraints (r, TREE_VEC_ELT (t, i));
-  return r;
+  gcc_assert (concept_check_p (t));
+
+  if (TREE_CODE (t) == CALL_EXPR)
+    t = CALL_EXPR_FN (t);
+
+  gcc_assert (TREE_CODE (t) == TEMPLATE_ID_EXPR);
+  return t;
 }
 
-/* Returns true if T is a call expression to a function
-   concept. */
+/* Extract the TEMPLATE_DECL from a concept check.  */
 
-bool
-function_concept_check_p (tree t)
+tree
+get_concept_check_template (tree t)
 {
-  gcc_assert (TREE_CODE (t) == CALL_EXPR);
-  tree fn = CALL_EXPR_FN (t);
-  if (fn != NULL_TREE
-      && TREE_CODE (fn) == TEMPLATE_ID_EXPR)
-    {
-      tree f1 = OVL_FIRST (TREE_OPERAND (fn, 0));
-      if (TREE_CODE (f1) == TEMPLATE_DECL
-         && DECL_DECLARED_CONCEPT_P (DECL_TEMPLATE_RESULT (f1)))
-        return true;
-    }
-  return false;
+  tree id = unpack_concept_check (t);
+  tree tmpl = TREE_OPERAND (id, 0);
+  if (OVL_P (tmpl))
+    tmpl = OVL_FIRST (tmpl);
+  return tmpl;
 }
 
-/* Returns true if any of the arguments in the template
-   argument list is a wildcard or wildcard pack.  */
+/* Returns true if any of the arguments in the template argument list is
+   a wildcard or wildcard pack.  */
 
 bool
 contains_wildcard_p (tree args)
@@ -142,38 +242,6 @@ contains_wildcard_p (tree args)
   return false;
 }
 
-/* Build a new call expression, but don't actually generate a
-   new function call. We just want the tree, not the semantics.  */
-
-inline tree
-build_call_check (tree id)
-{
-  ++processing_template_decl;
-  vec<tree, va_gc> *fargs = make_tree_vector();
-  tree call = finish_call_expr (id, &fargs, false, false, tf_none);
-  release_tree_vector (fargs);
-  --processing_template_decl;
-  return call;
-}
-
-/* Build an expression that will check a variable concept. If any
-   argument contains a wildcard, don't try to finish the variable
-   template because we can't substitute into a non-existent
-   declaration.  */
-
-tree
-build_variable_check (tree id)
-{
-  gcc_assert (TREE_CODE (id) == TEMPLATE_ID_EXPR);
-  if (contains_wildcard_p (TREE_OPERAND (id, 1)))
-    return id;
-
-  ++processing_template_decl;
-  tree var = finish_template_variable (id);
-  --processing_template_decl;
-  return var;
-}
-
 /*---------------------------------------------------------------------------
                     Resolution of qualified concept names
 ---------------------------------------------------------------------------*/
@@ -200,13 +268,12 @@ build_variable_check (tree id)
    the complete set of arguments substituted into the parameter list.  */
 
 static tree
-resolve_constraint_check (tree ovl, tree args)
+resolve_function_concept_overload (tree ovl, tree args)
 {
   int nerrs = 0;
   tree cands = NULL_TREE;
   for (lkp_iterator iter (ovl); iter; ++iter)
     {
-      // Get the next template overload.
       tree tmpl = *iter;
       if (TREE_CODE (tmpl) != TEMPLATE_DECL)
         continue;
@@ -252,7 +319,7 @@ resolve_constraint_check (tree ovl, tree args)
    does not denote a constraint check, return NULL.  */
 
 tree
-resolve_constraint_check (tree call)
+resolve_function_concept_check (tree call)
 {
   gcc_assert (TREE_CODE (call) == CALL_EXPR);
 
@@ -279,37 +346,40 @@ resolve_constraint_check (tree call)
     }
 
   tree args = TREE_OPERAND (target, 1);
-  return resolve_constraint_check (ovl, args);
+  return resolve_function_concept_overload (ovl, args);
 }
 
-/* Returns a pair containing the checked variable concept
-   and its associated prototype parameter.  The result
-   is a TREE_LIST whose TREE_VALUE is the variable concept
-   and whose TREE_PURPOSE is the prototype parameter.  */
+/* Returns a pair containing the checked concept and its associated
+   prototype parameter. The result is a TREE_LIST whose TREE_VALUE
+   is the concept (non-template) and whose TREE_PURPOSE contains
+   the converted template arguments, including the deduced prototype
+   parameter (in position 0). */
 
 tree
-resolve_variable_concept_check (tree id)
+resolve_concept_check (tree check)
 {
+  gcc_assert (concept_check_p (check));
+  tree id = unpack_concept_check (check);
   tree tmpl = TREE_OPERAND (id, 0);
-  tree args = TREE_OPERAND (id, 1);
 
-  if (!variable_concept_p (tmpl))
-    return NULL_TREE;
+  /* If this is an overloaded function concept, perform overload
+     resolution (this only happens when deducing prototype parameters
+     and template introductions).  */
+  if (TREE_CODE (tmpl) == OVERLOAD)
+    {
+      if (OVL_CHAIN (tmpl))
+       return resolve_function_concept_check (check);
+      tmpl = OVL_FIRST (tmpl);
+    }
 
-  /* Make sure that we have the right parameters before
-     assuming that it works.  Note that failing to deduce
-     will result in diagnostics.  */
+  tree args = TREE_OPERAND (id, 1);
   tree parms = INNERMOST_TEMPLATE_PARMS (DECL_TEMPLATE_PARMS (tmpl));
   ++processing_template_decl;
   tree result = coerce_template_parms (parms, args, tmpl);
   --processing_template_decl;
-  if (result != error_mark_node)
-    {
-      tree decl = DECL_TEMPLATE_RESULT (tmpl);
-      return build_tree_list (result, decl);
-    }
-  else
+  if (result == error_mark_node)
     return error_mark_node;
+  return build_tree_list (result, DECL_TEMPLATE_RESULT (tmpl));
 }
 
 /* Given a call expression or template-id expression to a concept EXPR
@@ -321,14 +391,7 @@ resolve_variable_concept_check (tree id)
 bool
 deduce_constrained_parameter (tree expr, tree& check, tree& proto)
 {
-  tree info = NULL_TREE;
-  if (TREE_CODE (expr) == TEMPLATE_ID_EXPR)
-    info = resolve_variable_concept_check (expr);
-  else if (TREE_CODE (expr) == CALL_EXPR)
-    info = resolve_constraint_check (expr);
-  else
-    gcc_unreachable ();
-
+  tree info = resolve_concept_check (expr);
   if (info && info != error_mark_node)
     {
       check = TREE_VALUE (info);
@@ -338,6 +401,7 @@ deduce_constrained_parameter (tree expr, tree& check, tree& proto)
       proto = TREE_TYPE (arg);
       return true;
     }
+
   check = proto = NULL_TREE;
   return false;
 }
@@ -346,127 +410,39 @@ deduce_constrained_parameter (tree expr, tree& check, tree& proto)
    deduce the concept being checked and return the template arguments.
    Returns NULL_TREE if deduction fails.  */
 static tree
-deduce_concept_introduction (tree expr)
+deduce_concept_introduction (tree check)
 {
-  tree info = NULL_TREE;
-  if (TREE_CODE (expr) == TEMPLATE_ID_EXPR)
-    info = resolve_variable_concept_check (expr);
-  else if (TREE_CODE (expr) == CALL_EXPR)
-    info = resolve_constraint_check (expr);
-  else
-    gcc_unreachable ();
-
+  tree info = resolve_concept_check (check);
   if (info && info != error_mark_node)
     return TREE_PURPOSE (info);
   return NULL_TREE;
 }
 
-namespace {
-
-/*---------------------------------------------------------------------------
-                       Constraint implication learning
----------------------------------------------------------------------------*/
-
-/* The implication context determines how we memoize concept checks.
-   Given two checks C1 and C2, the direction of implication depends
-   on whether we are learning implications of a conjunction or disjunction.
-   For example:
-
-      template<typename T> concept bool C = ...;
-      template<typenaem T> concept bool D = C<T> && true;
-
-   From this, we can learn that D<T> implies C<T>. We cannot learn,
-   without further testing, that C<T> does not imply D<T>. If, for
-   example, C<T> were defined as true, then these constraints would
-   be logically equivalent.
-
-   In rare cases, we may start with a logical equivalence. For example:
-
-      template<typename T> concept bool C = ...;
-      template<typename T> concept bool D = C<T>;
+/* Build a constrained placeholder type where SPEC is a type-constraint.
+   SPEC can be anything were concept_definition_p is true.
 
-   Here, we learn that C<T> implies D<T> and vice versa.   */
-
-enum implication_context
-{
-  conjunction_cxt, /* C1 implies C2. */
-  disjunction_cxt, /* C2 implies C1. */
-  equivalence_cxt  /* C1 implies C2, C2 implies C1. */
-};
-
-void learn_implications(tree, tree, implication_context);
-
-void
-learn_implication (tree parent, tree child, implication_context cxt)
-{
-  switch (cxt)
-    {
-      case conjunction_cxt:
-        save_subsumption_result (parent, child, true);
-        break;
-      case disjunction_cxt:
-        save_subsumption_result (child, parent, true);
-        break;
-      case equivalence_cxt:
-        save_subsumption_result (parent, child, true);
-        save_subsumption_result (child, parent, true);
-        break;
-    }
-}
+   If DECLTYPE_P is true, then the placeholder is decltype(auto).
 
-void
-learn_logical_operation (tree parent, tree constr, implication_context cxt)
-{
-  learn_implications (parent, TREE_OPERAND (constr, 0), cxt);
-  learn_implications (parent, TREE_OPERAND (constr, 1), cxt);
-}
+   Returns a pair whose FIRST is the concept being checked and whose
+   SECOND is the prototype parameter.  */
 
-void
-learn_implications (tree parent, tree constr, implication_context cxt)
+tree_pair
+finish_type_constraints (tree spec, tree args, tsubst_flags_t complain)
 {
-  switch (TREE_CODE (constr))
-    {
-      case CHECK_CONSTR:
-        return learn_implication (parent, constr, cxt);
-
-      case CONJ_CONSTR:
-        if (cxt == disjunction_cxt)
-          return;
-        return learn_logical_operation (parent, constr, cxt);
+  gcc_assert (concept_definition_p (spec));
 
-      case DISJ_CONSTR:
-        if (cxt == conjunction_cxt)
-          return;
-        return learn_logical_operation (parent, constr, cxt);
-
-      default:
-        break;
-    }
-}
-
-/* Quickly scan the top-level constraints of CONSTR to learn and
-   cache logical relations between concepts.  The search does not
-   include conjunctions of disjunctions or vice versa.  */
-
-void
-learn_implications (tree tmpl, tree args, tree constr)
-{
-  /* Don't memoize relations between non-dependent arguemnts. It's not
-     helpful. */
-  if (!uses_template_parms (args))
-    return;
+  /* Build an initial concept check.  */
+  tree check = build_type_constraint (spec, args, complain);
+  if (check == error_mark_node)
+    return std::make_pair (error_mark_node, NULL_TREE);
 
-  /* Build a check constraint for the purpose of caching. */
-  tree parent = build_nt (CHECK_CONSTR, tmpl, args);
+  /* Extract the concept and prototype parameter from the check. */
+  tree con;
+  tree proto;
+  if (!deduce_constrained_parameter (check, con, proto))
+    return std::make_pair (error_mark_node, NULL_TREE);
 
-  /* Start learning based on the kind of the top-level contraint. */
-  if (TREE_CODE (constr) == CONJ_CONSTR)
-    return learn_logical_operation (parent, constr, conjunction_cxt);
-  else if (TREE_CODE (constr) == DISJ_CONSTR)
-    return learn_logical_operation (parent, constr, disjunction_cxt);
-  else if (TREE_CODE (constr) == CHECK_CONSTR)
-    /* This is the rare concept alias case. */
-    return learn_implication (parent, constr, equivalence_cxt);
+  return std::make_pair (con, proto);
 }
 
 /*---------------------------------------------------------------------------
@@ -475,7 +451,7 @@ learn_implications (tree tmpl, tree args, tree constr)
 
 /* Returns the expression of a function concept. */
 
-tree
+static tree
 get_returned_expression (tree fn)
 {
   /* Extract the body of the function minus the return expression.  */
@@ -492,594 +468,468 @@ get_returned_expression (tree fn)
 
 /* Returns the initializer of a variable concept. */
 
-tree
+static tree
 get_variable_initializer (tree var)
 {
   tree init = DECL_INITIAL (var);
   if (!init)
     return error_mark_node;
+  if (BRACE_ENCLOSED_INITIALIZER_P (init)
+      && CONSTRUCTOR_NELTS (init) == 1)
+    init = CONSTRUCTOR_ELT (init, 0)->value;
   return init;
 }
 
 /* Returns the definition of a variable or function concept.  */
 
-tree
+static tree
 get_concept_definition (tree decl)
 {
+  if (TREE_CODE (decl) == OVERLOAD)
+    decl = OVL_FIRST (decl);
+
+  if (TREE_CODE (decl) == TEMPLATE_DECL)
+    decl = DECL_TEMPLATE_RESULT (decl);
+
+  if (TREE_CODE (decl) == CONCEPT_DECL)
+    return DECL_INITIAL (decl);
   if (VAR_P (decl))
     return get_variable_initializer (decl);
-  else if (TREE_CODE (decl) == FUNCTION_DECL)
+  if (TREE_CODE (decl) == FUNCTION_DECL)
     return get_returned_expression (decl);
   gcc_unreachable ();
 }
 
-int expansion_level = 0;
+/*---------------------------------------------------------------------------
+                     Normalization of expressions
+
+This set of functions will transform an expression into a constraint
+in a sequence of steps.
+---------------------------------------------------------------------------*/
 
-class expanding_concept_sentinel
+void
+debug_parameter_mapping (tree map)
 {
-public:
-  expanding_concept_sentinel ()
-  {
-    ++expansion_level;
-  }
+  for (tree p = map; p; p = TREE_CHAIN (p))
+    {
+      tree parm = TREE_VALUE (p);
+      tree arg = TREE_PURPOSE (p);
+      if (TYPE_P (parm))
+       verbatim ("MAP %qD TO %qT", TEMPLATE_TYPE_DECL (parm), arg);
+      else
+       verbatim ("MAP %qD TO %qE", TEMPLATE_PARM_DECL (parm), arg);
+      // debug_tree (parm);
+      // debug_tree (arg);
+    }
+}
 
-  ~expanding_concept_sentinel()
-  {
-    --expansion_level;
-  }
-};
+void
+debug_argument_list (tree args)
+{
+  for (int i = 0; i < TREE_VEC_LENGTH (args); ++i)
+    {
+      tree arg = TREE_VEC_ELT (args, i);
+      if (TYPE_P (arg))
+       verbatim ("ARG %qT", arg);
+      else
+       verbatim ("ARG %qE", arg);
+    }
+}
 
+/* Associate each parameter in PARMS with its corresponding template
+   argument in ARGS.  */
 
-} /* namespace */
+static tree
+map_arguments (tree parms, tree args)
+{
+  for (tree p = parms; p; p = TREE_CHAIN (p))
+    {
+      int level;
+      int index;
+      template_parm_level_and_index (TREE_VALUE (p), &level, &index);
+      TREE_PURPOSE (p) = TMPL_ARG (args, level, index);
+    }
+  return parms;
+}
 
-/* Returns true when a concept is being expanded.  */
+/* Build the parameter mapping for EXPR using ARGS.  */
 
-bool
-expanding_concept()
+static tree
+build_parameter_mapping (tree expr, tree args, tree decl)
 {
-  return expansion_level > 0;
+  int depth = 0;
+  if (decl)
+    {
+      gcc_assert (TREE_CODE (decl) == TEMPLATE_DECL);
+      tree parms = DECL_TEMPLATE_PARMS (decl);
+      depth = TREE_INT_CST_LOW (TREE_PURPOSE (parms));
+    }
+  tree parms = find_template_parameters (expr, depth);
+  tree map = map_arguments (parms, args);
+  return map;
 }
 
-/* Expand a concept declaration (not a template) and its arguments to
-   a constraint defined by the concept's initializer or definition.  */
+/* True if the parameter mappings of two atomic constraints are equivalent.  */
 
-tree
-expand_concept (tree decl, tree args)
+static bool
+parameter_mapping_equivalent_p (tree t1, tree t2)
 {
-  expanding_concept_sentinel sentinel;
+  tree map1 = ATOMIC_CONSTR_MAP (t1);
+  tree map2 = ATOMIC_CONSTR_MAP (t2);
+  while (map1 && map2)
+    {
+      tree arg1 = TREE_PURPOSE (map1);
+      tree arg2 = TREE_PURPOSE (map2);
+      if (!template_args_equal (arg1, arg2))
+        return false;
+      map1 = TREE_CHAIN (map1);
+      map2 = TREE_CHAIN (map2);
+    }
+  return true;
+}
 
-  if (TREE_CODE (decl) == TEMPLATE_DECL)
-    decl = DECL_TEMPLATE_RESULT (decl);
-  tree tmpl = DECL_TI_TEMPLATE (decl);
+/* Provides additional context for normalization.  */
 
-  /* Check for a previous specialization. */
-  if (tree spec = get_concept_expansion (tmpl, args))
-    return spec;
+struct norm_info : subst_info
+{
+  norm_info(tsubst_flags_t complain)
+    : subst_info (tf_warning_or_error | complain, NULL_TREE),
+      context()
+  {}
 
-  /* Substitute the arguments to form a new definition expression.  */
-  tree def = get_concept_definition (decl);
+  /* Construct a top-level context for DECL.  */
 
-  ++processing_template_decl;
-  tree result = tsubst_expr (def, args, tf_none, NULL_TREE, true);
-  --processing_template_decl;
-  if (result == error_mark_node)
-    return error_mark_node;
+  norm_info (tree in_decl, tsubst_flags_t complain)
+    : subst_info (tf_warning_or_error | complain, in_decl),
+      context (make_context (in_decl))
+  {}
 
-  /* And lastly, normalize it, check for implications, and save
-     the specialization for later.  */
-  tree norm = normalize_expression (result);
-  learn_implications (tmpl, args, norm);
-  return save_concept_expansion (tmpl, args, norm);
-}
+  bool generate_diagnostics() const
+  {
+    return complain & tf_norm;
+  }
 
+  tree make_context(tree in_decl)
+  {
+    if (generate_diagnostics ())
+      return build_tree_list (NULL_TREE, in_decl);
+    return NULL_TREE;
+  }
 
-/*---------------------------------------------------------------------------
-                Stepwise normalization of expressions
+  void update_context(tree expr, tree args)
+  {
+    if (generate_diagnostics ())
+      {
+       tree map = build_parameter_mapping (expr, args, in_decl);
+       context = tree_cons (map, expr, context);
+      }
+    in_decl = get_concept_check_template (expr);
+  }
 
-This set of functions will transform an expression into a constraint
-in a sequence of steps. Normalization does not not look into concept
-definitions.
----------------------------------------------------------------------------*/
+  /* Provides information about the source of a constraint. This is a
+     TREE_LIST whose VALUE is either a concept check or a constrained
+     declaration. The PURPOSE, for concept checks is a parameter mapping
+     for that check.  */
 
-/* Transform a logical-or or logical-and expression into either
-   a conjunction or disjunction. */
+  tree context;
+};
 
-tree
-normalize_logical_operation (tree t, tree_code c)
-{
-  tree t0 = normalize_expression (TREE_OPERAND (t, 0));
-  tree t1 = normalize_expression (TREE_OPERAND (t, 1));
-  return build_nt (c, t0, t1);
-}
+static tree normalize_expression (tree, tree, norm_info);
 
-/* A simple requirement T introduces an expression constraint
-   for its expression. */
+/* Transform a logical-or or logical-and expression into either
+   a conjunction or disjunction. */
 
-inline tree
-normalize_simple_requirement (tree t)
+static tree
+normalize_logical_operation (tree t, tree args, tree_code c, norm_info info)
 {
-  return build_nt (EXPR_CONSTR, TREE_OPERAND (t, 0));
-}
+  tree t0 = normalize_expression (TREE_OPERAND (t, 0), args, info);
+  tree t1 = normalize_expression (TREE_OPERAND (t, 1), args, info);
 
-/* A type requirement T introduce a type constraint for its type.  */
+  /* Build a new info object for the constraint.  */
+  tree ci = info.generate_diagnostics()
+    ? build_tree_list (t, info.context)
+    : NULL_TREE;
 
-inline tree
-normalize_type_requirement (tree t)
-{
-  return build_nt (TYPE_CONSTR, TREE_OPERAND (t, 0));
+  return build2 (c, ci, t0, t1);
 }
 
-/* A compound requirement T introduces a conjunction of constraints
-   depending on its form.  The conjunction always includes an
-   expression constraint for the expression of the requirement.
-   If a trailing return type was specified, the conjunction includes
-   either an implicit conversion constraint or an argument deduction
-   constraint.  If the noexcept specifier is present, the conjunction
-   includes an exception constraint.  */
-
-tree
-normalize_compound_requirement (tree t)
+static tree
+normalize_concept_check (tree check, tree args, norm_info info)
 {
-  tree expr = TREE_OPERAND (t, 0);
-  tree constr = build_nt (EXPR_CONSTR, TREE_OPERAND (t, 0));
-
-  /* If a type is given, append an implicit conversion or
-     argument deduction constraint.  */
-  if (tree type = TREE_OPERAND (t, 1))
-    {
-      tree type_constr;
-      /* TODO: We should be extracting a list of auto nodes
-         from type_uses_auto, not a single node */
-      if (tree placeholder = type_uses_auto (type))
-        type_constr = build_nt (DEDUCT_CONSTR, expr, type, placeholder);
-      else
-        type_constr = build_nt (ICONV_CONSTR, expr, type);
-      constr = conjoin_constraints (constr, type_constr);
-    }
+  tree id = unpack_concept_check (check);
+  tree tmpl = TREE_OPERAND (id, 0);
+  tree targs = TREE_OPERAND (id, 1);
 
-  /* If noexcept is present, append an exception constraint. */
-  if (COMPOUND_REQ_NOEXCEPT_P (t))
+  /* A function concept is wrapped in an overload.  */
+  if (TREE_CODE (tmpl) == OVERLOAD)
     {
-      tree except = build_nt (EXCEPT_CONSTR, expr);
-      constr = conjoin_constraints (constr, except);
+      /* TODO: Can we diagnose this error during parsing?  */
+      if (TREE_CODE (check) == TEMPLATE_ID_EXPR)
+       error_at (EXPR_LOC_OR_LOC (check, input_location),
+                 "function concept must be called");
+      tmpl = OVL_FIRST (tmpl);
     }
 
-  return constr;
-}
+  /* Substitute through the arguments of the concept check. */
+  targs = tsubst_template_args (targs, args, info.complain, info.in_decl);
+  if (targs == error_mark_node)
+    return error_mark_node;
 
-/* A nested requirement T introduces a conjunction of constraints
-   corresponding to its constraint-expression.
+  /* Build the substitution for the concept definition.  */
+  tree parms = TREE_VALUE (DECL_TEMPLATE_PARMS (tmpl));
+  /* Turn on template processing; coercing non-type template arguments
+     will automatically assume they're non-dependent.  */
+  ++processing_template_decl;
+  tree subst = coerce_template_parms (parms, targs, tmpl);
+  --processing_template_decl;
+  if (subst == error_mark_node)
+    return error_mark_node;
 
-   If the result of transforming T is error_mark_node, the resulting
-   constraint is a predicate constraint whose operand is also
-   error_mark_node. This preserves the constraint structure, but
-   will guarantee that the constraint is never satisfied.  */
+  /* The concept may have been ill-formed.  */
+  tree def = get_concept_definition (DECL_TEMPLATE_RESULT (tmpl));
+  if (def == error_mark_node)
+    return error_mark_node;
 
-inline tree
-normalize_nested_requirement (tree t)
-{
-  return normalize_expression (TREE_OPERAND (t, 0));
+  info.update_context (check, args);
+  return normalize_expression (def, subst, info);
 }
 
-/* Transform a requirement T into one or more constraints.  */
+/* The normal form of an atom depends on the expression. The normal
+   form of a function call to a function concept is a check constraint
+   for that concept. The normal form of a reference to a variable
+   concept is a check constraint for that concept. Otherwise, the
+   constraint is a predicate constraint.  */
 
-tree
-normalize_requirement (tree t)
+static tree
+normalize_atom (tree t, tree args, norm_info info)
 {
-  switch (TREE_CODE (t))
-    {
-    case SIMPLE_REQ:
-      return normalize_simple_requirement (t);
+  /* Concept checks are not atomic.  */
+  if (concept_check_p (t))
+    return normalize_concept_check (t, args, info);
 
-    case TYPE_REQ:
-      return normalize_type_requirement (t);
+  /* Build the parameter mapping for the atom.  */
+  tree map = build_parameter_mapping (t, args, info.in_decl);
 
-    case COMPOUND_REQ:
-      return normalize_compound_requirement (t);
+  /* Build a new info object for the atom.  */
+  tree ci = build_tree_list (t, info.context);
 
-    case NESTED_REQ:
-      return normalize_nested_requirement (t);
+  return build1 (ATOMIC_CONSTR, ci, map);
+}
+
+/* Returns the normal form of an expression. */
 
+static tree
+normalize_expression (tree t, tree args, norm_info info)
+{
+  if (!t)
+    return NULL_TREE;
+
+  if (t == error_mark_node)
+    return error_mark_node;
+
+  switch (TREE_CODE (t))
+    {
+    case TRUTH_ANDIF_EXPR:
+      return normalize_logical_operation (t, args, CONJ_CONSTR, info);
+    case TRUTH_ORIF_EXPR:
+      return normalize_logical_operation (t, args, DISJ_CONSTR, info);
     default:
-      gcc_unreachable ();
+      return normalize_atom (t, args, info);
     }
-  return error_mark_node;
 }
 
-/* Transform a sequence of requirements into a conjunction of
-   constraints. */
+/* Cache of the normalized form of constraints.  Marked as deletable because it
+   can all be recalculated.  */
+static GTY((deletable)) hash_map<tree,tree> *normalized_map;
 
-tree
-normalize_requirements (tree t)
+static tree
+get_normalized_constraints (tree t, tree args, norm_info info)
 {
-  tree result = NULL_TREE;
-  for (; t; t = TREE_CHAIN (t))
-    {
-      tree constr = normalize_requirement (TREE_VALUE (t));
-      result = conjoin_constraints (result, constr);
-    }
-  return result;
-}
-
-/* The normal form of a requires-expression is a parameterized
-   constraint having the same parameters and a conjunction of
-   constraints representing the normal form of requirements.  */
-
-tree
-normalize_requires_expression (tree t)
-{
-  tree operand = normalize_requirements (TREE_OPERAND (t, 1));
-  if (tree parms = TREE_OPERAND (t, 0))
-    return build_nt (PARM_CONSTR, parms, operand);
-  else
-    return operand;
+  auto_timevar time (TV_CONSTRAINT_NORM);
+  return normalize_expression (t, args, info);
 }
 
-/* For a template-id referring to a variable concept, returns
-   a check constraint. Otherwise, returns a predicate constraint. */
+/* Returns the normalized constraints from a constraint-info object
+   or NULL_TREE if the constraints are null. ARGS provide the initial
+   arguments for normalization and IN_DECL provides the declaration
+   to which the constraints belong.  */
 
-tree
-normalize_template_id_expression (tree t)
+static tree
+get_normalized_constraints_from_info (tree ci, tree args, tree in_decl,
+                                     bool diag = false)
 {
-  if (tree info = resolve_variable_concept_check (t))
-    {
-      if (info == error_mark_node)
-        {
-          /* We get this when the template arguments don't match
-             the variable concept. */
-          error ("invalid reference to concept %qE", t);
-          return error_mark_node;
-        }
-
-      tree decl = TREE_VALUE (info);
-      tree args = TREE_PURPOSE (info);
-      return build_nt (CHECK_CONSTR, decl, args);
-    }
-
-  /* Check that we didn't refer to a function concept like a variable.  */
-  tree fn = OVL_FIRST (TREE_OPERAND (t, 0));
-  if (TREE_CODE (fn) == TEMPLATE_DECL
-      && DECL_DECLARED_CONCEPT_P (DECL_TEMPLATE_RESULT (fn)))
-    {
-      error_at (location_of (t),
-               "invalid reference to function concept %qD", fn);
-      return error_mark_node;
-    }
-
-  return build_nt (PRED_CONSTR, t);
-}
-
-/* For a call expression to a function concept, returns a check
-   constraint. Otherwise, returns a predicate constraint. */
+  if (ci == NULL_TREE)
+    return NULL_TREE;
 
-tree
-normalize_call_expression (tree t)
-{
-  /* Try to resolve this function call as a concept.  If not, then
-     it can be returned as a predicate constraint.  */
-  tree check = resolve_constraint_check (t);
-  if (!check)
-    return build_nt (PRED_CONSTR, t);
-  if (check == error_mark_node)
-    {
-      /* TODO: Improve diagnostics. We could report why the reference
-         is invalid. */
-      error ("invalid reference to concept %qE", t);
-      return error_mark_node;
-    }
+  /* Substitution errors during normalization are fatal.  */
+  ++processing_template_decl;
+  norm_info info (in_decl, diag ? tf_norm : tf_none);
+  tree t = get_normalized_constraints (CI_ASSOCIATED_CONSTRAINTS (ci),
+                                      args, info);
+  --processing_template_decl;
 
-  tree fn = TREE_VALUE (check);
-  tree args = TREE_PURPOSE (check);
-  return build_nt (CHECK_CONSTR, fn, args);
+  return t;
 }
 
-/* If T is a call to an overloaded && or || operator, diagnose that
-   as a non-SFINAEable error.  Returns true if an error is emitted.
-
-   TODO: It would be better to diagnose this at the point of definition,
-   if possible. Perhaps we should immediately do a first-pass normalization
-   of a concept definition to catch obvious non-dependent errors like
-   this.  */
+/* Returns the normalized constraints for the declaration D.  */
 
-bool
-check_for_logical_overloads (tree t)
+static tree
+get_normalized_constraints_from_decl (tree d, bool diag = false)
 {
-  if (TREE_CODE (t) != CALL_EXPR)
-    return false;
+  tree tmpl;
+  tree decl;
 
-  tree fn = CALL_EXPR_FN (t);
+  /* For inherited constructors, consider the original declaration;
+     it has the correct template information attached. */
+  d = strip_inheriting_ctors (d);
 
-  /* For member calls, try extracting the function from the
-     component ref.  */
-  if (TREE_CODE (fn) == COMPONENT_REF)
+  if (TREE_CODE (d) == TEMPLATE_DECL)
     {
-      fn = TREE_OPERAND (fn, 1);
-      if (TREE_CODE (fn) == BASELINK)
-        fn = BASELINK_FUNCTIONS (fn);
+      tmpl = d;
+      decl = DECL_TEMPLATE_RESULT (tmpl);
     }
-
-  if (TREE_CODE (fn) != FUNCTION_DECL)
-    return false;
-
-  if (DECL_OVERLOADED_OPERATOR_P (fn))
+  else
     {
-      location_t loc = cp_expr_loc_or_input_loc (t);
-      error_at (loc, "constraint %qE, uses overloaded operator", t);
-      return true;
+      if (tree ti = DECL_TEMPLATE_INFO (d))
+       tmpl = TI_TEMPLATE (ti);
+      else
+       tmpl = NULL_TREE;
+      decl = d;
     }
 
-  return false;
-}
+  /* Get the most general template for the declaration, and compute
+     arguments from that. This ensures that the arguments used for
+     normalization are always template parameters and not arguments
+     used for outer specializations.  For example:
 
-/* The normal form of an atom depends on the expression. The normal
-   form of a function call to a function concept is a check constraint
-   for that concept. The normal form of a reference to a variable
-   concept is a check constraint for that concept. Otherwise, the
-   constraint is a predicate constraint.  */
+        template<typename T>
+        struct S {
+         template<typename U> requires C<T, U> void f(U);
+        };
 
-tree
-normalize_atom (tree t)
-{
-  /* We can get constraints pushed down through pack expansions, so
-     just return them. */
-  if (constraint_p (t))
-    return t;
+        S<int>::f(0);
 
-  tree type = TREE_TYPE (t);
-  if (!type || type_unknown_p (t) || TREE_CODE (type) == TEMPLATE_TYPE_PARM)
-    ;
-  else if (!dependent_type_p (type))
-    {
-      if (check_for_logical_overloads (t))
-        return error_mark_node;
+     When we normalize the requirements for S<int>::f, we want the
+     arguments to be {T, U}, not {int, U}. One reason for this is that
+     accepting the latter causes the template parameter level of U
+     to be reduced in a way that makes it overly difficult substitute
+     concrete arguments (i.e., eventually {int, int} during satisfaction.  */
+  if (tmpl)
+  {
+    if (DECL_LANG_SPECIFIC(tmpl) && !DECL_TEMPLATE_SPECIALIZATION (tmpl))
+      tmpl = most_general_template (tmpl);
+  }
 
-      type = cv_unqualified (type);
-      if (!same_type_p (type, boolean_type_node))
-       {
-         error ("predicate constraint %q+E does not have type %<bool%>", t);
-         return error_mark_node;
-       }
-    }
+  /* If we're not diagnosing errors, use cached constraints, if any.  */
+  if (!diag)
+    if (tree *p = hash_map_safe_get (normalized_map, tmpl))
+      return *p;
 
-  if (TREE_CODE (t) == TEMPLATE_ID_EXPR)
-    return normalize_template_id_expression (t);
-  if (TREE_CODE (t) == CALL_EXPR)
-    return normalize_call_expression (t);
-  return build_nt (PRED_CONSTR, t);
-}
+  tree args = generic_targs_for (tmpl);
+  tree ci = get_constraints (decl);
+  tree norm = get_normalized_constraints_from_info (ci, args, tmpl, diag);
 
-/* Push down the pack expansion EXP into the leaves of the constraint PAT.  */
+  if (!diag)
+    hash_map_safe_put<hm_ggc> (normalized_map, tmpl, norm);
 
-tree
-push_down_pack_expansion (tree exp, tree pat)
-{
-  switch (TREE_CODE (pat))
-    {
-    case CONJ_CONSTR:
-    case DISJ_CONSTR:
-      {
-       pat = copy_node (pat);
-       TREE_OPERAND (pat, 0)
-         = push_down_pack_expansion (exp, TREE_OPERAND (pat, 0));
-       TREE_OPERAND (pat, 1)
-         = push_down_pack_expansion (exp, TREE_OPERAND (pat, 1));
-       return pat;
-      }
-    default:
-      {
-       exp = copy_node (exp);
-       SET_PACK_EXPANSION_PATTERN (exp, pat);
-       return exp;
-      }
-    }
+  return norm;
 }
 
-/* Transform a pack expansion into a constraint.  First we transform the
-   pattern of the pack expansion, then we push the pack expansion down into the
-   leaves of the constraint so that partial ordering will work.  */
+/* Returns the normal form of TMPL's definition.  */
 
-tree
-normalize_pack_expansion (tree t)
-{
-  tree pat = normalize_expression (PACK_EXPANSION_PATTERN (t));
-  return push_down_pack_expansion (t, pat);
+static tree
+normalize_concept_definition (tree tmpl)
+{
+  if (tree *p = hash_map_safe_get (normalized_map, tmpl))
+    return *p;
+  gcc_assert (concept_definition_p (tmpl));
+  if (OVL_P (tmpl))
+    tmpl = OVL_FIRST (tmpl);
+  gcc_assert (TREE_CODE (tmpl) == TEMPLATE_DECL);
+  tree args = generic_targs_for (tmpl);
+  tree def = get_concept_definition (DECL_TEMPLATE_RESULT (tmpl));
+  ++processing_template_decl;
+  norm_info info (tmpl, tf_none);
+  tree norm = get_normalized_constraints (def, args, info);
+  --processing_template_decl;
+  hash_map_safe_put<hm_ggc> (normalized_map, tmpl, norm);
+  return norm;
 }
 
-/* Transform an expression into a constraint.  */
-
-tree
-normalize_any_expression (tree t)
-{
-  switch (TREE_CODE (t))
-    {
-    case TRUTH_ANDIF_EXPR:
-      return normalize_logical_operation (t, CONJ_CONSTR);
-
-    case TRUTH_ORIF_EXPR:
-      return normalize_logical_operation (t, DISJ_CONSTR);
-
-    case REQUIRES_EXPR:
-      return normalize_requires_expression (t);
-
-    case BIND_EXPR:
-      return normalize_expression (BIND_EXPR_BODY (t));
-
-    case EXPR_PACK_EXPANSION:
-      return normalize_pack_expansion (t);
-
-    default:
-      /* All other constraints are atomic. */
-      return normalize_atom (t);
-    }
-}
+/* Returns the normal form of TMPL's requirements.  */
 
-/* Transform a statement into an expression.  */
-tree
-normalize_any_statement (tree t)
+static tree
+normalize_template_requirements (tree tmpl, bool diag = false)
 {
-  switch (TREE_CODE (t))
-    {
-    case RETURN_EXPR:
-      return normalize_expression (TREE_OPERAND (t, 0));
-    default:
-      gcc_unreachable ();
-    }
-  return error_mark_node;
+  return get_normalized_constraints_from_decl (tmpl, diag);
 }
 
-/* Reduction rules for the declaration T.  */
+/* Returns the normal form of TMPL's requirements.  */
 
-tree
-normalize_any_declaration (tree t)
+static tree
+normalize_nontemplate_requirements (tree decl, bool diag = false)
 {
-  switch (TREE_CODE (t))
-    {
-    case VAR_DECL:
-      return normalize_atom (t);
-    default:
-      gcc_unreachable ();
-    }
-  return error_mark_node;
+  return get_normalized_constraints_from_decl (decl, diag);
 }
 
-/* Returns the normal form of a constraint expression. */
+/* Normalize an EXPR as a constraint.  */
 
-tree
-normalize_expression (tree t)
+static tree
+normalize_constraint_expression (tree expr, bool diag = false)
 {
-  if (!t)
-    return NULL_TREE;
-
-  if (t == error_mark_node)
-    return error_mark_node;
+  if (!expr || expr == error_mark_node)
+    return expr;
 
-  switch (TREE_CODE_CLASS (TREE_CODE (t)))
+  /* For concept checks, use the supplied template arguments as those used
+     for normalization. Otherwise, there are no template arguments.  */
+  tree args;
+  if (concept_check_p (expr))
     {
-    case tcc_unary:
-    case tcc_binary:
-    case tcc_expression:
-    case tcc_vl_exp:
-      return normalize_any_expression (t);
-
-    case tcc_statement:
-      return normalize_any_statement (t);
-
-    case tcc_declaration:
-      return normalize_any_declaration (t);
-
-    case tcc_exceptional:
-    case tcc_constant:
-    case tcc_reference:
-    case tcc_comparison:
-      /* These are all atomic predicate constraints. */
-      return normalize_atom (t);
-
-    default:
-      /* Unhandled node kind. */
-      gcc_unreachable ();
+      tree id = unpack_concept_check (expr);
+      args = TREE_OPERAND (id, 1);
     }
-  return error_mark_node;
-}
-
-
-/*---------------------------------------------------------------------------
-                        Constraint normalization
----------------------------------------------------------------------------*/
-
-tree normalize_constraint (tree);
-
-/* The normal form of the disjunction T0 /\ T1 is the conjunction
-   of the normal form of T0 and the normal form of T1.  */
-
-inline tree
-normalize_conjunction (tree t)
-{
-  tree t0 = normalize_constraint (TREE_OPERAND (t, 0));
-  tree t1 = normalize_constraint (TREE_OPERAND (t, 1));
-  return build_nt (CONJ_CONSTR, t0, t1);
-}
-
-/* The normal form of the disjunction T0 \/ T1 is the disjunction
-   of the normal form of T0 and the normal form of T1.  */
-
-inline tree
-normalize_disjunction (tree t)
-{
-  tree t0 = normalize_constraint (TREE_OPERAND (t, 0));
-  tree t1 = normalize_constraint (TREE_OPERAND (t, 1));
-  return build_nt (DISJ_CONSTR, t0, t1);
-}
-
-/* A predicate constraint is normalized in two stages.  First all
-   references specializations of concepts are replaced by their
-   substituted definitions.  Then, the resulting expression is
-   transformed into a constraint by transforming && expressions
-   into conjunctions and || into disjunctions.  */
+  else
+    args = NULL_TREE;
 
-tree
-normalize_predicate_constraint (tree t)
-{
   ++processing_template_decl;
-  tree expr = PRED_CONSTR_EXPR (t);
-  tree constr = normalize_expression (expr);
+  norm_info info (diag ? tf_norm : tf_none);
+  tree norm = get_normalized_constraints (expr, args, info);
   --processing_template_decl;
-  return constr;
+  return norm;
 }
 
-/* The normal form of a parameterized constraint is the normal
-   form of its operand.  */
+/* 17.4.1.2p2. Two constraints are identical if they are formed
+   from the same expression and the targets of the parameter mapping
+   are equivalent.  */
 
-tree
-normalize_parameterized_constraint (tree t)
+bool
+atomic_constraints_identical_p (tree t1, tree t2)
 {
-  tree parms = PARM_CONSTR_PARMS (t);
-  tree operand = normalize_constraint (PARM_CONSTR_OPERAND (t));
-  return build_nt (PARM_CONSTR, parms, operand);
-}
+  if (ATOMIC_CONSTR_EXPR (t1) != ATOMIC_CONSTR_EXPR (t2))
+    return false;
 
-/* Normalize the constraint T by reducing it so that it is
-   comprised of only conjunctions and disjunctions of atomic
-   constraints.  */
+  if (!parameter_mapping_equivalent_p (t1, t2))
+    return false;
 
-tree
-normalize_constraint (tree t)
-{
-  if (!t)
-    return NULL_TREE;
+  return true;
+}
 
-  if (t == error_mark_node)
-    return t;
+hashval_t
+hash_atomic_constraint (tree t)
+{
+  /* Hash the identity of the expression.  */
+  hashval_t val = htab_hash_pointer (ATOMIC_CONSTR_EXPR (t));
 
-  switch (TREE_CODE (t))
+  /* Hash the targets of the parameter map.  */
+  tree p = ATOMIC_CONSTR_MAP (t);
+  while (p)
     {
-      case CONJ_CONSTR:
-        return normalize_conjunction (t);
-
-      case DISJ_CONSTR:
-        return normalize_disjunction (t);
-
-      case PRED_CONSTR:
-        return normalize_predicate_constraint (t);
-
-      case PARM_CONSTR:
-        return normalize_parameterized_constraint (t);
-
-      case EXPR_CONSTR:
-      case TYPE_CONSTR:
-      case ICONV_CONSTR:
-      case DEDUCT_CONSTR:
-      case EXCEPT_CONSTR:
-        /* These constraints are defined to be atomic. */
-        return t;
-
-      default:
-        /* CONSTR was not a constraint. */
-        gcc_unreachable();
+      val = iterative_hash_template_arg (TREE_PURPOSE (p), val);
+      p = TREE_CHAIN (p);
     }
-  return error_mark_node;
-}
-
 
+  return val;
+}
 
 // -------------------------------------------------------------------------- //
 // Constraint Semantic Processing
@@ -1093,7 +943,7 @@ current_template_constraints (void)
 {
   if (!current_template_parms)
     return NULL_TREE;
-  tree tmpl_constr = TEMPLATE_PARM_CONSTRAINTS (current_template_parms);
+  tree tmpl_constr = TEMPLATE_PARMS_CONSTRAINTS (current_template_parms);
   return build_constraints (tmpl_constr, NULL_TREE);
 }
 
@@ -1104,7 +954,7 @@ current_template_constraints (void)
 tree
 associate_classtype_constraints (tree type)
 {
-  if (!type || type == error_mark_node || TREE_CODE (type) != RECORD_TYPE)
+  if (!type || type == error_mark_node || !CLASS_TYPE_P (type))
     return type;
 
   /* An explicit class template specialization has no template parameters.  */
@@ -1124,9 +974,11 @@ associate_classtype_constraints (tree type)
         {
           if (!equivalent_constraints (ci, orig_ci))
             {
-              // FIXME: Improve diagnostics.
-              error ("%qT does not match any declaration", type);
-              return error_mark_node;
+             error ("%qT does not match original declaration", type);
+             tree tmpl = CLASSTYPE_TI_TEMPLATE (type);
+             location_t loc = DECL_SOURCE_LOCATION (tmpl);
+             inform (loc, "original template declaration here");
+             /* Fall through, so that we define the type anyway.  */
             }
           return type;
         }
@@ -1135,17 +987,14 @@ associate_classtype_constraints (tree type)
   return type;
 }
 
-namespace {
+/* Create an empty constraint info block.  */
 
-// Create an empty constraint info block.
-inline tree_constraint_info*
+static inline tree_constraint_info*
 build_constraint_info ()
 {
   return (tree_constraint_info *)make_node (CONSTRAINT_INFO);
 }
 
-} // namespace
-
 /* Build a constraint-info object that contains the associated constraints
    of a declaration.  This also includes the declaration's template
    requirements (TREQS) and any trailing requirements for a function
@@ -1155,27 +1004,46 @@ build_constraint_info ()
    this returns NULL_TREE, indicating an unconstrained declaration.  */
 
 tree
-build_constraints (tree tmpl_reqs, tree decl_reqs)
+build_constraints (tree tr, tree dr)
 {
-  gcc_assert (tmpl_reqs ? constraint_p (tmpl_reqs) : true);
-  gcc_assert (decl_reqs ? constraint_p (decl_reqs) : true);
-
-  if (!tmpl_reqs && !decl_reqs)
+  if (!tr && !dr)
     return NULL_TREE;
 
   tree_constraint_info* ci = build_constraint_info ();
-  ci->template_reqs = tmpl_reqs;
-  ci->declarator_reqs = decl_reqs;
-  ci->associated_constr = conjoin_constraints (tmpl_reqs, decl_reqs);
+  ci->template_reqs = tr;
+  ci->declarator_reqs = dr;
+  ci->associated_constr = combine_constraint_expressions (tr, dr);
 
   return (tree)ci;
 }
 
-namespace {
+/* Returns the template-head requires clause for the template
+   declaration T or NULL_TREE if none.  */
+
+tree
+get_template_head_requirements (tree t)
+{
+  tree ci = get_constraints (t);
+  if (!ci)
+    return NULL_TREE;
+  return CI_TEMPLATE_REQS (ci);
+}
+
+/* Returns the trailing requires clause of the declarator of
+   a template declaration T or NULL_TREE if none.  */
+
+tree
+get_trailing_function_requirements (tree t)
+{
+  tree ci = get_constraints (t);
+  if (!ci)
+    return NULL_TREE;
+  return CI_DECLARATOR_REQS (ci);
+}
 
 /* Construct a sequence of template arguments by prepending
    ARG to REST. Either ARG or REST may be null. */
-tree
+static tree
 build_concept_check_arguments (tree arg, tree rest)
 {
   gcc_assert (rest ? TREE_CODE (rest) == TREE_VEC : true);
@@ -1199,27 +1067,155 @@ build_concept_check_arguments (tree arg, tree rest)
   return args;
 }
 
-} // namespace
+/* Builds an id-expression of the form `C<Args...>()` where C is a function
+   concept.  */
+
+static tree
+build_function_check (tree tmpl, tree args, tsubst_flags_t /*complain*/)
+{
+  if (TREE_CODE (tmpl) == TEMPLATE_DECL)
+    {
+      /* If we just got a template, wrap it in an overload so it looks like any
+        other template-id. */
+      tmpl = ovl_make (tmpl);
+      TREE_TYPE (tmpl) = boolean_type_node;
+    }
+
+  /* Perform function concept resolution now so we always have a single
+     function of the overload set (even if we started with only one; the
+     resolution function converts template arguments). Note that we still
+     wrap this in an overload set so we don't upset other parts of the
+     compiler that expect template-ids referring to function concepts
+     to have an overload set.  */
+  tree info = resolve_function_concept_overload (tmpl, args);
+  if (info == error_mark_node)
+    return error_mark_node;
+  if (!info)
+    {
+      error ("no matching concepts for %qE", tmpl);
+      return error_mark_node;
+    }
+  args = TREE_PURPOSE (info);
+  tmpl = DECL_TI_TEMPLATE (TREE_VALUE (info));
+
+  /* Rebuild the singleton overload set; mark the type bool.  */
+  tmpl = ovl_make (tmpl, NULL_TREE);
+  TREE_TYPE (tmpl) = boolean_type_node;
+
+  /* Build the id-expression around the overload set.  */
+  tree id = build2 (TEMPLATE_ID_EXPR, boolean_type_node, tmpl, args);
+
+  /* Finally, build the call expression around the overload.  */
+  ++processing_template_decl;
+  vec<tree, va_gc> *fargs = make_tree_vector ();
+  tree call = build_min_nt_call_vec (id, fargs);
+  release_tree_vector (fargs);
+  --processing_template_decl;
+
+  return call;
+}
+
+/* Builds an id-expression of the form `C<Args...>` where C is a variable
+   concept.  */
+
+static tree
+build_variable_check (tree tmpl, tree args, tsubst_flags_t complain)
+{
+  gcc_assert (variable_concept_p (tmpl));
+  gcc_assert (TREE_CODE (tmpl) == TEMPLATE_DECL);
+  tree parms = INNERMOST_TEMPLATE_PARMS (DECL_TEMPLATE_PARMS (tmpl));
+  args = coerce_template_parms (parms, args, tmpl, complain);
+  if (args == error_mark_node)
+    return error_mark_node;
+  return build2 (TEMPLATE_ID_EXPR, boolean_type_node, tmpl, args);
+}
+
+/* Builds an id-expression of the form `C<Args...>` where C is a standard
+   concept.  */
+
+static tree
+build_standard_check (tree tmpl, tree args, tsubst_flags_t complain)
+{
+  gcc_assert (standard_concept_p (tmpl));
+  gcc_assert (TREE_CODE (tmpl) == TEMPLATE_DECL);
+  tree parms = INNERMOST_TEMPLATE_PARMS (DECL_TEMPLATE_PARMS (tmpl));
+  args = coerce_template_parms (parms, args, tmpl, complain);
+  if (args == error_mark_node)
+    return error_mark_node;
+  return build2 (TEMPLATE_ID_EXPR, boolean_type_node, tmpl, args);
+}
+
+/* Construct an expression that checks TARGET using ARGS.  */
 
-/* Construct an expression that checks the concept given by
-   TARGET. The TARGET must be:
+tree
+build_concept_check (tree target, tree args, tsubst_flags_t complain)
+{
+  return build_concept_check (target, NULL_TREE, args, complain);
+}
 
-   - an OVERLOAD referring to one or more function concepts
-   - a BASELINK referring to an overload set of the above, or
-   - a TEMPLTATE_DECL referring to a variable concept.
+/* Construct an expression that checks the concept given by DECL. If
+   concept_definition_p (DECL) is false, this returns null.  */
 
-   ARG and REST are the explicit template arguments for the
-   eventual concept check. */
 tree
-build_concept_check (tree target, tree arg, tree rest)
+build_concept_check (tree decl, tree arg, tree rest, tsubst_flags_t complain)
 {
+  if (arg == NULL_TREE && rest == NULL_TREE)
+    {
+      tree id = build_nt (TEMPLATE_ID_EXPR, decl, rest);
+      error ("invalid use concept %qE", id);
+      return error_mark_node;
+    }
+
   tree args = build_concept_check_arguments (arg, rest);
-  if (variable_template_p (target))
-    return build_variable_check (lookup_template_variable (target, args));
-  else
-    return build_call_check (lookup_template_function (target, args));
+
+  if (standard_concept_p (decl))
+    return build_standard_check (decl, args, complain);
+  if (variable_concept_p (decl))
+    return build_variable_check (decl, args, complain);
+  if (function_concept_p (decl))
+    return build_function_check (decl, args, complain);
+
+  return error_mark_node;
+}
+
+/* Build a template-id that can participate in a concept check.  */
+
+static tree
+build_concept_id (tree decl, tree args)
+{
+  tree check = build_concept_check (decl, args, tf_warning_or_error);
+  if (check == error_mark_node)
+    return error_mark_node;
+  return unpack_concept_check (check);
+}
+
+/* Build a template-id that can participate in a concept check, preserving
+   the source location of the original template-id.  */
+
+tree
+build_concept_id (tree expr)
+{
+  gcc_assert (TREE_CODE (expr) == TEMPLATE_ID_EXPR);
+  tree id = build_concept_id (TREE_OPERAND (expr, 0), TREE_OPERAND (expr, 1));
+  protected_set_expr_location (id, cp_expr_location (expr));
+  return id;
 }
 
+/* Build as template-id with a placeholder that can be used as a
+   type constraint.
+
+   Note that this will diagnose errors if the initial concept check
+   cannot be built.  */
+
+tree
+build_type_constraint (tree decl, tree args, tsubst_flags_t complain)
+{
+  tree wildcard = build_nt (WILDCARD_DECL);
+  tree check = build_concept_check (decl, wildcard, args, complain);
+  if (check == error_mark_node)
+    return error_mark_node;
+  return unpack_concept_check (check);
+}
 
 /* Returns a TYPE_DECL that contains sufficient information to
    build a template parameter of the same kind as PROTO and
@@ -1240,17 +1236,13 @@ build_constrained_parameter (tree cnc, tree proto, tree args)
   return decl;
 }
 
-/* Create a constraint expression for the given DECL that
-   evaluates the requirements specified by CONSTR, a TYPE_DECL
-   that contains all the information necessary to build the
-   requirements (see finish_concept_name for the layout of
-   that TYPE_DECL).
-
-   Note that the constraints are neither reduced nor decomposed.
-   That is done only after the requires clause has been parsed
-   (or not).
+/* Create a constraint expression for the given DECL that evaluates the
+   requirements specified by CONSTR, a TYPE_DECL that contains all the
+   information necessary to build the requirements (see finish_concept_name
+   for the layout of that TYPE_DECL).
 
-   This will always return a CHECK_CONSTR. */
+   Note that the constraints are neither reduced nor decomposed. That is
+   done only after the requires clause has been parsed (or not).  */
 tree
 finish_shorthand_constraint (tree decl, tree constr)
 {
@@ -1265,41 +1257,46 @@ finish_shorthand_constraint (tree decl, tree constr)
   tree con = CONSTRAINED_PARM_CONCEPT (constr);
   tree args = CONSTRAINED_PARM_EXTRA_ARGS (constr);
 
-  /* If the parameter declaration is variadic, but the concept
-     is not then we need to apply the concept to every element
-     in the pack.  */
-  bool is_proto_pack = template_parameter_pack_p (proto);
-  bool is_decl_pack = template_parameter_pack_p (decl);
-  bool apply_to_all_p = is_decl_pack && !is_proto_pack;
+  /* The TS lets use shorthand to constrain a pack of arguments, but the
+     standard does not.
+
+     For the TS, consider:
+
+       template<C... Ts> struct s;
+
+     If C is variadic (and because Ts is a pack), we associate the
+     constraint C<Ts...>. In all other cases, we associate
+     the constraint (C<Ts> && ...).
+
+     The standard behavior cannot be overridden by -fconcepts-ts.  */
+  bool variadic_concept_p = template_parameter_pack_p (proto);
+  bool declared_pack_p = template_parameter_pack_p (decl);
+  bool apply_to_each_p = (cxx_dialect >= cxx2a) ? true : !variadic_concept_p;
 
   /* Get the argument and overload used for the requirement
      and adjust it if we're going to expand later.  */
   tree arg = template_parm_to_arg (build_tree_list (NULL_TREE, decl));
-  if (apply_to_all_p)
+  if (apply_to_each_p && declared_pack_p)
     arg = PACK_EXPANSION_PATTERN (TREE_VEC_ELT (ARGUMENT_PACK_ARGS (arg), 0));
 
-  /* Build the concept check. If it the constraint needs to be
-     applied to all elements of the parameter pack, then make
-     the constraint an expansion. */
+  /* Build the concept constraint-expression.  */
   tree tmpl = DECL_TI_TEMPLATE (con);
-  tree check = VAR_P (con) ? tmpl : ovl_make (tmpl);
-  check = build_concept_check (check, arg, args);
+  tree check = tmpl;
+  if (TREE_CODE (con) == FUNCTION_DECL)
+    check = ovl_make (tmpl);
+  check = build_concept_check (check, arg, args, tf_warning_or_error);
 
-  /* Make the check a pack expansion if needed.
+  /* Make the check a fold-expression if needed.  */
+  if (apply_to_each_p && declared_pack_p)
+    check = finish_left_unary_fold_expr (check, TRUTH_ANDIF_EXPR);
 
-     FIXME: We should be making a fold expression. */
-  if (apply_to_all_p)
-    {
-      check = make_pack_expansion (check);
-      TREE_TYPE (check) = boolean_type_node;
-    }
-
-  return normalize_expression (check);
+  return check;
 }
 
 /* Returns a conjunction of shorthand requirements for the template
    parameter list PARMS. Note that the requirements are stored in
    the TYPE of each tree node. */
+
 tree
 get_shorthand_constraints (tree parms)
 {
@@ -1309,94 +1306,219 @@ get_shorthand_constraints (tree parms)
     {
       tree parm = TREE_VEC_ELT (parms, i);
       tree constr = TEMPLATE_PARM_CONSTRAINTS (parm);
-      result = conjoin_constraints (result, constr);
+      result = combine_constraint_expressions (result, constr);
     }
   return result;
 }
 
-// Returns and chains a new parameter for PARAMETER_LIST which will conform
-// to the prototype given by SRC_PARM.  The new parameter will have its
-// identifier and location set according to IDENT and PARM_LOC respectively.
+/* Get the deduced wildcard from a DEDUCED placeholder.  If the deduced
+   wildcard is a pack, return the first argument of that pack.  */
+
 static tree
-process_introduction_parm (tree parameter_list, tree src_parm)
+get_deduced_wildcard (tree wildcard)
 {
-  // If we have a pack, we should have a single pack argument which is the
-  // placeholder we want to look at.
-  bool is_parameter_pack = ARGUMENT_PACK_P (src_parm);
-  if (is_parameter_pack)
-    src_parm = TREE_VEC_ELT (ARGUMENT_PACK_ARGS (src_parm), 0);
+  if (ARGUMENT_PACK_P (wildcard))
+    wildcard = TREE_VEC_ELT (ARGUMENT_PACK_ARGS (wildcard), 0);
+  gcc_assert (TREE_CODE (wildcard) == WILDCARD_DECL);
+  return wildcard;
+}
 
-  // At this point we should have a wildcard, but we want to
-  // grab the associated decl from it.  Also grab the stored
-  // identifier and location that should be chained to it in
-  // a PARM_DECL.
-  gcc_assert (TREE_CODE (src_parm) == WILDCARD_DECL);
+/* Returns the prototype parameter for the nth deduced wildcard.  */
 
-  tree ident = DECL_NAME (src_parm);
-  location_t parm_loc = DECL_SOURCE_LOCATION (src_parm);
+static tree
+get_introduction_prototype (tree wildcards, int index)
+{
+  return TREE_TYPE (get_deduced_wildcard (TREE_VEC_ELT (wildcards, index)));
+}
 
-  // If we expect a pack and the deduced template is not a pack, or if the
-  // template is using a pack and we didn't declare a pack, throw an error.
-  if (is_parameter_pack != WILDCARD_PACK_P (src_parm))
-    {
-      error_at (parm_loc, "cannot match pack for introduced parameter");
-      tree err_parm = build_tree_list (error_mark_node, error_mark_node);
-      return chainon (parameter_list, err_parm);
-    }
+/* Introduce a type template parameter.  */
 
-  src_parm = TREE_TYPE (src_parm);
+static tree
+introduce_type_template_parameter (tree wildcard, bool& non_type_p)
+{
+  non_type_p = false;
+  return finish_template_type_parm (class_type_node, DECL_NAME (wildcard));
+}
+
+/* Introduce a template template parameter.  */
+
+static tree
+introduce_template_template_parameter (tree wildcard, bool& non_type_p)
+{
+  non_type_p = false;
+  begin_template_parm_list ();
+  current_template_parms = DECL_TEMPLATE_PARMS (TREE_TYPE (wildcard));
+  end_template_parm_list ();
+  return finish_template_template_parm (class_type_node, DECL_NAME (wildcard));
+}
+
+/* Introduce a template non-type parameter.  */
+
+static tree
+introduce_nontype_template_parameter (tree wildcard, bool& non_type_p)
+{
+  non_type_p = true;
+  tree parm = copy_decl (TREE_TYPE (wildcard));
+  DECL_NAME (parm) = DECL_NAME (wildcard);
+  return parm;
+}
+
+/* Introduce a single template parameter.  */
+
+static tree
+build_introduced_template_parameter (tree wildcard, bool& non_type_p)
+{
+  tree proto = TREE_TYPE (wildcard);
 
   tree parm;
-  bool is_non_type;
-  if (TREE_CODE (src_parm) == TYPE_DECL)
+  if (TREE_CODE (proto) == TYPE_DECL)
+    parm = introduce_type_template_parameter (wildcard, non_type_p);
+  else if (TREE_CODE (proto) == TEMPLATE_DECL)
+    parm = introduce_template_template_parameter (wildcard, non_type_p);
+  else
+    parm = introduce_nontype_template_parameter (wildcard, non_type_p);
+
+  /* Wrap in a TREE_LIST for process_template_parm. Note that introduced
+     parameters do not retain the defaults from the source parameter.  */
+  return build_tree_list (NULL_TREE, parm);
+}
+
+/* Introduce a single template parameter.  */
+
+static tree
+introduce_template_parameter (tree parms, tree wildcard)
+{
+  gcc_assert (!ARGUMENT_PACK_P (wildcard));
+  tree proto = TREE_TYPE (wildcard);
+  location_t loc = DECL_SOURCE_LOCATION (wildcard);
+
+  /* Diagnose the case where we have C{...Args}.  */
+  if (WILDCARD_PACK_P (wildcard))
     {
-      is_non_type = false;
-      parm = finish_template_type_parm (class_type_node, ident);
+      tree id = DECL_NAME (wildcard);
+      error_at (loc, "%qE cannot be introduced with an ellipsis %<...%>", id);
+      inform (DECL_SOURCE_LOCATION (proto), "prototype declared here");
     }
-  else if (TREE_CODE (src_parm) == TEMPLATE_DECL)
+
+  bool non_type_p;
+  tree parm = build_introduced_template_parameter (wildcard, non_type_p);
+  return process_template_parm (parms, loc, parm, non_type_p, false);
+}
+
+/* Introduce a template parameter pack.  */
+
+static tree
+introduce_template_parameter_pack (tree parms, tree wildcard)
+{
+  bool non_type_p;
+  tree parm = build_introduced_template_parameter (wildcard, non_type_p);
+  location_t loc = DECL_SOURCE_LOCATION (wildcard);
+  return process_template_parm (parms, loc, parm, non_type_p, true);
+}
+
+/* Introduce the nth template parameter.  */
+
+static tree
+introduce_template_parameter (tree parms, tree wildcards, int& index)
+{
+  tree deduced = TREE_VEC_ELT (wildcards, index++);
+  return introduce_template_parameter (parms, deduced);
+}
+
+/* Introduce either a template parameter pack or a list of template
+   parameters.  */
+
+static tree
+introduce_template_parameters (tree parms, tree wildcards, int& index)
+{
+  /* If the prototype was a parameter, we better have deduced an
+     argument pack, and that argument must be the last deduced value
+     in the wildcard vector.  */
+  tree deduced = TREE_VEC_ELT (wildcards, index++);
+  gcc_assert (ARGUMENT_PACK_P (deduced));
+  gcc_assert (index == TREE_VEC_LENGTH (wildcards));
+
+  /* Introduce each element in the pack.  */
+  tree args = ARGUMENT_PACK_ARGS (deduced);
+  for (int i = 0; i < TREE_VEC_LENGTH (args); ++i)
     {
-      is_non_type = false;
-      begin_template_parm_list ();
-      current_template_parms = DECL_TEMPLATE_PARMS (src_parm);
-      end_template_parm_list ();
-      parm = finish_template_template_parm (class_type_node, ident);
+      tree arg = TREE_VEC_ELT (args, i);
+      if (WILDCARD_PACK_P (arg))
+       parms = introduce_template_parameter_pack (parms, arg);
+      else
+       parms = introduce_template_parameter (parms, arg);
     }
+
+  return parms;
+}
+
+/* Builds the template parameter list PARMS by chaining introduced
+   parameters from the WILDCARD vector.  INDEX is the position of
+   the current parameter.  */
+
+static tree
+process_introduction_parms (tree parms, tree wildcards, int& index)
+{
+  tree proto = get_introduction_prototype (wildcards, index);
+  if (template_parameter_pack_p (proto))
+    return introduce_template_parameters (parms, wildcards, index);
   else
-    {
-      is_non_type = true;
+    return introduce_template_parameter (parms, wildcards, index);
+}
 
-      // Since we don't have a declarator, so we can copy the source
-      // parameter and change the name and eventually the location.
-      parm = copy_decl (src_parm);
-      DECL_NAME (parm) = ident;
-    }
+/* Ensure that all template parameters have been introduced for the concept
+   named in CHECK.  If not, emit a diagnostic.
 
-  // Wrap in a TREE_LIST for process_template_parm.  Introductions do not
-  // retain the defaults from the source template.
-  parm = build_tree_list (NULL_TREE, parm);
+   Note that implicitly introducing a parameter with a default argument
+     creates a case where a parameter is declared, but unnamed, making
+     it unusable in the definition.  */
 
-  return process_template_parm (parameter_list, parm_loc, parm,
-                                is_non_type, is_parameter_pack);
+static bool
+check_introduction_list (tree intros, tree check)
+{
+  check = unpack_concept_check (check);
+  tree tmpl = TREE_OPERAND (check, 0);
+  if (OVL_P (tmpl))
+    tmpl = OVL_FIRST (tmpl);
+
+  tree parms = DECL_INNERMOST_TEMPLATE_PARMS (tmpl);
+  if (TREE_VEC_LENGTH (intros) < TREE_VEC_LENGTH (parms))
+    {
+      error_at (input_location, "all template parameters of %qD must "
+                               "be introduced", tmpl);
+      return false;
+    }
+
+   return true;
 }
 
-/* Associates a constraint check to the current template based
-   on the introduction parameters.  INTRO_LIST must be a TREE_VEC
-   of WILDCARD_DECLs containing a chained PARM_DECL which
-   contains the identifier as well as the source location.
-   TMPL_DECL is the decl for the concept being used.  If we
-   take a concept, C, this will form a check in the form of
-   C<INTRO_LIST> filling in any extra arguments needed by the
-   defaults deduced.
+/* Associates a constraint check to the current template based on the
+   introduction parameters.  INTRO_LIST must be a TREE_VEC of WILDCARD_DECLs
+   containing a chained PARM_DECL which contains the identifier as well as
+   the source location. TMPL_DECL is the decl for the concept being used.
+   If we take a concept, C, this will form a check in the form of
+   C<INTRO_LIST> filling in any extra arguments needed by the defaults
+   deduced.
+
+   Returns NULL_TREE if no concept could be matched and error_mark_node if
+   an error occurred when matching.  */
 
-   Returns NULL_TREE if no concept could be matched and
-   error_mark_node if an error occurred when matching.  */
 tree
-finish_template_introduction (tree tmpl_decl, tree intro_list)
+finish_template_introduction (tree tmpl_decl,
+                             tree intro_list,
+                             location_t intro_loc)
 {
-  /* Deduce the concept check.  */
-  tree expr = build_concept_check (tmpl_decl, NULL_TREE, intro_list);
+  /* Build a concept check to deduce the actual parameters.  */
+  tree expr = build_concept_check (tmpl_decl, intro_list, tf_none);
   if (expr == error_mark_node)
-    return NULL_TREE;
+    {
+      error_at (intro_loc, "cannot deduce template parameters from "
+                          "introduction list");
+      return error_mark_node;
+    }
+
+  if (!check_introduction_list (intro_list, expr))
+    return error_mark_node;
 
   tree parms = deduce_concept_introduction (expr);
   if (!parms)
@@ -1406,9 +1528,15 @@ finish_template_introduction (tree tmpl_decl, tree intro_list)
   tree parm_list = NULL_TREE;
   begin_template_parm_list ();
   int nargs = MIN (TREE_VEC_LENGTH (parms), TREE_VEC_LENGTH (intro_list));
-  for (int n = 0; n < nargs; ++n)
-    parm_list = process_introduction_parm (parm_list, TREE_VEC_ELT (parms, n));
+  for (int n = 0; n < nargs; )
+    parm_list = process_introduction_parms (parm_list, parms, n);
   parm_list = end_template_parm_list (parm_list);
+
+  /* Update the number of arguments to reflect the number of deduced
+     template parameter introductions.  */
+  nargs = TREE_VEC_LENGTH (parm_list);
+
+  /* Determine if any errors occurred during matching.  */
   for (int i = 0; i < TREE_VEC_LENGTH (parm_list); ++i)
     if (TREE_VALUE (TREE_VEC_ELT (parm_list, i)) == error_mark_node)
       {
@@ -1417,7 +1545,7 @@ finish_template_introduction (tree tmpl_decl, tree intro_list)
       }
 
   /* Build a concept check for our constraint.  */
-  tree check_args = make_tree_vec (TREE_VEC_LENGTH (parms));
+  tree check_args = make_tree_vec (nargs);
   int n = 0;
   for (; n < TREE_VEC_LENGTH (parm_list); ++n)
     {
@@ -1431,22 +1559,33 @@ finish_template_introduction (tree tmpl_decl, tree intro_list)
   for (; n < TREE_VEC_LENGTH (parms); ++n)
     TREE_VEC_ELT (check_args, n) = TREE_VEC_ELT (parms, n);
 
-  /* Associate the constraint. */
-  tree check = build_concept_check (tmpl_decl, NULL_TREE, check_args);
-  tree constr = normalize_expression (check);
-  TEMPLATE_PARMS_CONSTRAINTS (current_template_parms) = constr;
+  /* Associate the constraint.  */
+  tree check = build_concept_check (tmpl_decl,
+                                   check_args,
+                                   tf_warning_or_error);
+  TEMPLATE_PARMS_CONSTRAINTS (current_template_parms) = check;
 
   return parm_list;
 }
 
 
-/* Given the predicate constraint T from a constrained-type-specifier, extract
+/* Given the concept check T from a constrained-type-specifier, extract
    its TMPL and ARGS.  FIXME why do we need two different forms of
    constrained-type-specifier?  */
 
 void
 placeholder_extract_concept_and_args (tree t, tree &tmpl, tree &args)
 {
+  if (concept_check_p (t))
+    {
+      t = unpack_concept_check (t);
+      tmpl = TREE_OPERAND (t, 0);
+      if (TREE_CODE (tmpl) == OVERLOAD)
+        tmpl = OVL_FIRST (tmpl);
+      args = TREE_OPERAND (t, 1);
+      return;
+    }
+
   if (TREE_CODE (t) == TYPE_DECL)
     {
       /* A constrained parameter.  Build a constraint check
@@ -1457,20 +1596,10 @@ placeholder_extract_concept_and_args (tree t, tree &tmpl, tree &args)
       placeholder_extract_concept_and_args (check, tmpl, args);
       return;
     }
-
-  if (TREE_CODE (t) == CHECK_CONSTR)
-    {
-      tree decl = CHECK_CONSTR_CONCEPT (t);
-      tmpl = DECL_TI_TEMPLATE (decl);
-      args = CHECK_CONSTR_ARGS (t);
-      return;
-    }
-
-    gcc_unreachable ();
 }
 
 /* Returns true iff the placeholders C1 and C2 are equivalent.  C1
-   and C2 can be either CHECK_CONSTR or TEMPLATE_TYPE_PARM.  */
+   and C2 can be either TEMPLATE_TYPE_PARM or template-ids.  */
 
 bool
 equivalent_placeholder_constraints (tree c1, tree c2)
@@ -1515,7 +1644,7 @@ equivalent_placeholder_constraints (tree c1, tree c2)
   return true;
 }
 
-/* Return a hash value for the placeholder PRED_CONSTR C.  */
+/* Return a hash value for the placeholder ATOMIC_CONSTR C.  */
 
 hashval_t
 hash_placeholder_constraint (tree c)
@@ -1532,178 +1661,215 @@ hash_placeholder_constraint (tree c)
   return val;
 }
 
-/*---------------------------------------------------------------------------
-                        Constraint substitution
----------------------------------------------------------------------------*/
+/* Substitute through the simple requirement.  */
 
-/* The following functions implement substitution rules for constraints.
-   Substitution without checking constraints happens only in the
-   instantiation of class templates. For example:
+static tree
+tsubst_valid_expression_requirement (tree t, tree args, subst_info info)
+{
+  return tsubst_expr (t, args, info.complain, info.in_decl, false);
+}
 
-      template<C1 T> struct S {
-        void f(T) requires C2<T>;
-        void g(T) requires T::value;
-      };
 
-      S<int> s; // error instantiating S<int>::g(T)
+/* Substitute through the simple requirement.  */
 
-   When we instantiate S, we substitute into its member declarations,
-   including their constraints. However, those constraints are not
-   checked. Substituting int into C2<T> yields C2<int>, and substituting
-   into T::value yields a substitution failure, making the program
-   ill-formed.
+static tree
+tsubst_simple_requirement (tree t, tree args, subst_info info)
+{
+  tree t0 = TREE_OPERAND (t, 0);
+  tree expr = tsubst_valid_expression_requirement (t0, args, info);
+  if (expr == error_mark_node)
+    return error_mark_node;
+  return finish_simple_requirement (EXPR_LOCATION (t), expr);
+}
 
-   Note that we only ever substitute into the associated constraints
-   of a declaration. That is, substitution is defined only for predicate
-   constraints and conjunctions. */
+/* Substitute through the type requirement.  */
 
-/* Substitute into the predicate constraints. Returns error_mark_node
-   if the substitution into the expression fails. */
-tree
-tsubst_predicate_constraint (tree t, tree args,
-                             tsubst_flags_t complain, tree in_decl)
+static tree
+tsubst_type_requirement (tree t, tree args, subst_info info)
 {
-  tree expr = PRED_CONSTR_EXPR (t);
-  ++processing_template_decl;
-  tree result = tsubst_expr (expr, args, complain, in_decl, false);
-  --processing_template_decl;
-  return build_nt (PRED_CONSTR, result);
+  tree t0 = TREE_OPERAND (t, 0);
+  tree type = tsubst (t0, args, info.complain, info.in_decl);
+  if (type == error_mark_node)
+    return error_mark_node;
+  return finish_type_requirement (EXPR_LOCATION (t), type);
 }
 
-/* Substitute into a check constraint. */
+/* True if TYPE can be deduced from EXPR.
 
-tree
-tsubst_check_constraint (tree t, tree args,
-                         tsubst_flags_t complain, tree in_decl)
+   FIXME: C++20 compound requirement constraints should be normalized and then
+   satisfied rather than substituted.  */
+
+static bool
+type_deducible_p (tree expr, tree type, tree placeholder, tree args,
+                  subst_info info)
 {
-  tree decl = CHECK_CONSTR_CONCEPT (t);
-  tree tmpl = DECL_TI_TEMPLATE (decl);
-  tree targs = CHECK_CONSTR_ARGS (t);
+  /* Make sure deduction is performed against ( EXPR ), so that
+     references are preserved in the result.  */
+  expr = force_paren_expr_uneval (expr);
 
-  /* Substitute through by building an template-id expression
-     and then substituting into that. */
-  tree expr = build_nt (TEMPLATE_ID_EXPR, tmpl, targs);
-  ++processing_template_decl;
-  tree result = tsubst_expr (expr, args, complain, in_decl, false);
-  --processing_template_decl;
+  /* Replace the constraints with the instantiated constraints. This
+     substitutes args into any template parameters in the trailing
+     result type.  */
+  tree saved_constr = PLACEHOLDER_TYPE_CONSTRAINTS (placeholder);
+  PLACEHOLDER_TYPE_CONSTRAINTS (placeholder)
+    = tsubst_constraint (saved_constr,
+                        args,
+                        info.complain | tf_partial,
+                        info.in_decl);
 
-  if (result == error_mark_node)
-    return error_mark_node;
+  /* Temporarily unlink the canonical type.  */
+  tree saved_type = TYPE_CANONICAL (placeholder);
+  TYPE_CANONICAL (placeholder) = NULL_TREE;
 
-  /* Extract the results and rebuild the check constraint. */
-  decl = DECL_TEMPLATE_RESULT (TREE_OPERAND (result, 0));
-  args = TREE_OPERAND (result, 1);
+  tree deduced_type
+    = do_auto_deduction (type,
+                        expr,
+                        placeholder,
+                        info.complain,
+                        adc_requirement);
+
+  PLACEHOLDER_TYPE_CONSTRAINTS (placeholder) = saved_constr;
+  TYPE_CANONICAL (placeholder) = saved_type;
+
+  if (deduced_type == error_mark_node)
+    return false;
 
-  return build_nt (CHECK_CONSTR, decl, args);
+  return true;
 }
 
-/* Substitute into the conjunction of constraints. Returns
-   error_mark_node if substitution into either operand fails. */
+/* True if EXPR can not be converted to TYPE.  */
 
-tree
-tsubst_logical_operator (tree t, tree args,
-                        tsubst_flags_t complain, tree in_decl)
+static bool
+expression_convertible_p (tree expr, tree type, subst_info info)
 {
-  tree t0 = TREE_OPERAND (t, 0);
-  tree r0 = tsubst_constraint (t0, args, complain, in_decl);
-  if (r0 == error_mark_node)
-    return error_mark_node;
-  tree t1 = TREE_OPERAND (t, 1);
-  tree r1 = tsubst_constraint (t1, args, complain, in_decl);
-  if (r1 == error_mark_node)
-    return error_mark_node;
-  return build_nt (TREE_CODE (t), r0, r1);
+  tree conv =
+    perform_direct_initialization_if_possible (type, expr, false,
+                                              info.complain);
+  if (conv == error_mark_node)
+    return false;
+  if (conv == NULL_TREE)
+    {
+      if (info.complain & tf_error)
+        {
+          location_t loc = EXPR_LOC_OR_LOC (expr, input_location);
+          error_at (loc, "cannot convert %qE to %qT", expr, type);
+        }
+      return false;
+    }
+  return true;
 }
 
-namespace {
 
-/* Substitute ARGS into the expression constraint T.  */
+/* Substitute through the compound requirement.  */
 
-tree
-tsubst_expr_constr (tree t, tree args, tsubst_flags_t complain, tree in_decl)
+static tree
+tsubst_compound_requirement (tree t, tree args, subst_info info)
 {
-  cp_unevaluated guard;
-  tree expr = EXPR_CONSTR_EXPR (t);
-  tree ret = tsubst_expr (expr, args, complain, in_decl, false);
-  if (ret == error_mark_node)
+  tree t0 = TREE_OPERAND (t, 0);
+  tree t1 = TREE_OPERAND (t, 1);
+  tree expr = tsubst_valid_expression_requirement (t0, args, info);
+  if (expr == error_mark_node)
     return error_mark_node;
-  return build_nt (EXPR_CONSTR, ret);
-}
 
-/* Substitute ARGS into the type constraint T.  */
+  /* Check the noexcept condition.  */
+  bool noexcept_p = COMPOUND_REQ_NOEXCEPT_P (t);
+  if (noexcept_p && !expr_noexcept_p (expr, tf_none))
+    return error_mark_node;
 
-tree
-tsubst_type_constr (tree t, tree args, tsubst_flags_t complain, tree in_decl)
-{
-  tree type = TYPE_CONSTR_TYPE (t);
-  tree ret = tsubst (type, args, complain, in_decl);
-  if (ret == error_mark_node)
+  /* Substitute through the type expression, if any.  */
+  tree type = tsubst (t1, args, info.complain, info.in_decl);
+  if (type == error_mark_node)
     return error_mark_node;
-  return build_nt (TYPE_CONSTR, ret);
-}
 
-/* Substitute ARGS into the implicit conversion constraint T.  */
+  /* Check expression against the result type.  */
+  if (type)
+    {
+      if (tree placeholder = type_uses_auto (type))
+       {
+         if (!type_deducible_p (expr, type, placeholder, args, info))
+           return error_mark_node;
+       }
+      else if (!expression_convertible_p (expr, type, info))
+       return error_mark_node;
+    }
 
-tree
-tsubst_implicit_conversion_constr (tree t, tree args, tsubst_flags_t complain,
-                                   tree in_decl)
-{
-  cp_unevaluated guard;
-  tree expr = ICONV_CONSTR_EXPR (t);
-  tree type = ICONV_CONSTR_TYPE (t);
-  tree new_expr = tsubst_expr (expr, args, complain, in_decl, false);
-  if (new_expr == error_mark_node)
-    return error_mark_node;
-  tree new_type = tsubst (type, args, complain, in_decl);
-  if (new_type == error_mark_node)
-    return error_mark_node;
-  return build_nt (ICONV_CONSTR, new_expr, new_type);
+  return finish_compound_requirement (EXPR_LOCATION (t),
+                                     expr, type, noexcept_p);
 }
 
-/* Substitute ARGS into the argument deduction constraint T.  */
-
-tree
-tsubst_argument_deduction_constr (tree t, tree args, tsubst_flags_t complain,
-                                  tree in_decl)
-{
-  cp_unevaluated guard;
-  tree expr = DEDUCT_CONSTR_EXPR (t);
-  tree pattern = DEDUCT_CONSTR_PATTERN (t);
-  tree autos = DEDUCT_CONSTR_PLACEHOLDER(t);
-  tree new_expr = tsubst_expr (expr, args, complain, in_decl, false);
-  if (new_expr == error_mark_node)
-    return error_mark_node;
-  /* It seems like substituting through the pattern will not affect the
-     placeholders.  We should (?) be able to reuse the existing list
-     without any problems.  If not, then we probably want to create a
-     new list of placeholders and then instantiate the pattern using
-     those.  */
-  tree new_pattern = tsubst (pattern, args, complain, in_decl);
-  if (new_pattern == error_mark_node)
+static tree
+tsubst_nested_requirement (tree t, tree args, subst_info info)
+{
+  tree t0 = TREE_OPERAND (t, 0);
+  tree expr = tsubst_expr (t0, args, info.complain, info.in_decl, false);
+  if (expr == error_mark_node)
     return error_mark_node;
-  return build_nt (DEDUCT_CONSTR, new_expr, new_pattern, autos);
+
+  /* Ensure that concrete results are satisfied.  */
+  if (!uses_template_parms (args))
+    {
+      /* FIXME satisfy_constraint_expression (t0, args, info) */
+
+      /* [17.4.1.2] ... lvalue-to-value conversion is performed as necessary,
+         and EXPR shall be a constant expression of type bool.  */
+      tree result = force_rvalue (expr, tf_error);
+      if (result == error_mark_node)
+        return error_mark_node;
+
+      /* FIXME: The expression must have boolean type.  */
+      if (cv_unqualified (TREE_TYPE (result)) != boolean_type_node)
+        return error_mark_node;
+
+      /* Compute the value of the expression.  */
+      result = satisfaction_value (cxx_constant_value (result));
+      if (result == error_mark_node || result == boolean_false_node)
+        return error_mark_node;
+    }
+
+  return finish_nested_requirement (EXPR_LOCATION (t), expr);
 }
 
-/* Substitute ARGS into the exception constraint T.  */
+/* Substitute ARGS into the requirement T.  */
 
-tree
-tsubst_exception_constr (tree t, tree args, tsubst_flags_t complain,
-                        tree in_decl)
+static tree
+tsubst_requirement (tree t, tree args, subst_info info)
 {
-  cp_unevaluated guard;
-  tree expr = EXCEPT_CONSTR_EXPR (t);
-  tree ret = tsubst_expr (expr, args, complain, in_decl, false);
-  if (ret == error_mark_node)
-    return error_mark_node;
-  return build_nt (EXCEPT_CONSTR, ret);
+  iloc_sentinel loc_s (cp_expr_location (t));
+  switch (TREE_CODE (t))
+    {
+    case SIMPLE_REQ:
+      return tsubst_simple_requirement (t, args, info);
+    case TYPE_REQ:
+      return tsubst_type_requirement (t, args, info);
+    case COMPOUND_REQ:
+      return tsubst_compound_requirement (t, args, info);
+    case NESTED_REQ:
+      return tsubst_nested_requirement (t, args, info);
+    default:
+      break;
+    }
+  gcc_unreachable ();
 }
 
-/* A subroutine of tsubst_constraint_variables. Register local
-   specializations for each of parameter in PARMS and its
-   corresponding substituted constraint variable in VARS.
-   Returns VARS. */
+/* Substitute ARGS into the list of requirements T. Note that
+   substitution failures here result in ill-formed programs. */
 
-tree
+static tree
+tsubst_requirement_body (tree t, tree args, subst_info info)
+{
+  tree result = NULL_TREE;
+  while (t)
+    {
+      tree req = tsubst_requirement (TREE_VALUE (t), args, info);
+      if (req == error_mark_node)
+       return error_mark_node;
+      result = tree_cons (NULL_TREE, req, result);
+      t = TREE_CHAIN (t);
+    }
+  return nreverse (result);
+}
+
+static tree
 declare_constraint_vars (tree parms, tree vars)
 {
   tree s = vars;
@@ -1723,6 +1889,24 @@ declare_constraint_vars (tree parms, tree vars)
   return vars;
 }
 
+/* Substitute through as if checking function parameter types. This
+   will diagnose common parameter type errors.  Returns error_mark_node
+   if an error occurred.  */
+
+static tree
+check_constaint_variables (tree t, tree args, subst_info info)
+{
+  tree types = NULL_TREE;
+  tree p = t;
+  while (p && !VOID_TYPE_P (p))
+    {
+      types = tree_cons (NULL_TREE, TREE_TYPE (p), types);
+      p = TREE_CHAIN (p);
+    }
+  types = chainon (nreverse (types), void_list_node);
+  return tsubst_function_parms (types, args, info.complain, info.in_decl);
+}
+
 /* A subroutine of tsubst_parameterized_constraint. Substitute ARGS
    into the parameter list T, producing a sequence of constraint
    variables, declared in the current scope.
@@ -1731,171 +1915,81 @@ declare_constraint_vars (tree parms, tree vars)
    prior to calling this function since this substitution will
    declare the substituted parameters. */
 
-tree
-tsubst_constraint_variables (tree t, tree args,
-                             tsubst_flags_t complain, tree in_decl)
+static tree
+tsubst_constraint_variables (tree t, tree args, subst_info info)
 {
+  /* Perform a trial substitution to check for type errors.  */
+  tree parms = check_constaint_variables (t, args, info);
+  if (parms == error_mark_node)
+    return error_mark_node;
+
   /* Clear cp_unevaluated_operand across tsubst so that we get a proper chain
      of PARM_DECLs.  */
   int saved_unevaluated_operand = cp_unevaluated_operand;
   cp_unevaluated_operand = 0;
-  tree vars = tsubst (t, args, complain, in_decl);
+  tree vars = tsubst (t, args, info.complain, info.in_decl);
   cp_unevaluated_operand = saved_unevaluated_operand;
   if (vars == error_mark_node)
     return error_mark_node;
   return declare_constraint_vars (t, vars);
 }
 
-/* Substitute ARGS into the parameterized constraint T.  */
+/* Substitute ARGS into the requires-expression T. [8.4.7]p6. The
+   substitution of template arguments into a requires-expression
+   may result in the formation of invalid types or expressions
+   in its requirements ... In such cases, the expression evaluates
+   to false; it does not cause the program to be ill-formed.
 
-tree
-tsubst_parameterized_constraint (tree t, tree args,
-                                tsubst_flags_t complain, tree in_decl)
-{
-  local_specialization_stack stack;
-  tree vars = tsubst_constraint_variables (PARM_CONSTR_PARMS (t),
-                                          args, complain, in_decl);
-  if (vars == error_mark_node)
-    return error_mark_node;
-  tree expr = tsubst_constraint (PARM_CONSTR_OPERAND (t), args,
-                                complain, in_decl);
-  if (expr == error_mark_node)
-    return error_mark_node;
-  return build_nt (PARM_CONSTR, vars, expr);
-}
-
-/* Substitute ARGS into the simple requirement T. Note that
-   substitution may result in an ill-formed expression without
-   causing the program to be ill-formed. In such cases, the
-   requirement wraps an error_mark_node. */
+   However, there are cases where substitution must produce a
+   new requires-expression, that is not a template constraint.
+   For example:
 
-inline tree
-tsubst_simple_requirement (tree t, tree args,
-                           tsubst_flags_t complain, tree in_decl)
-{
-  ++processing_template_decl;
-  tree expr = tsubst_expr (TREE_OPERAND (t, 0), args, complain, in_decl, false);
-  --processing_template_decl;
-  return finish_simple_requirement (expr);
-}
+        template<typename T>
+        class X {
+          template<typename U>
+          static constexpr bool var = requires (U u) { T::fn(u); };
+        };
 
-/* Substitute ARGS into the type requirement T. Note that
-   substitution may result in an ill-formed type without
-   causing the program to be ill-formed. In such cases, the
-   requirement wraps an error_mark_node. */
-
-inline tree
-tsubst_type_requirement (tree t, tree args,
-                         tsubst_flags_t complain, tree in_decl)
-{
-  ++processing_template_decl;
-  tree type = tsubst (TREE_OPERAND (t, 0), args, complain, in_decl);
-  --processing_template_decl;
-  return finish_type_requirement (type);
-}
-
-/* Substitute args into the compound requirement T. If substituting
-   into either the expression or the type fails, the corresponding
-   operands in the resulting node will be error_mark_node. This
-   preserves a requirement for the purpose of partial ordering, but
-   it will never be satisfied. */
+   In the instantiation of X<Y> (assuming Y defines fn), then the
+   instantiated requires-expression would include Y::fn(u). If any
+   substitution in the requires-expression fails, we can immediately
+   fold the expression to false, as would be the case e.g., when
+   instantiation X<int>.  */
 
 tree
-tsubst_compound_requirement (tree t, tree args,
-                             tsubst_flags_t complain, tree in_decl)
+tsubst_requires_expr (tree t, tree args,
+                     tsubst_flags_t complain, tree in_decl)
 {
-  ++processing_template_decl;
-  tree expr = tsubst_expr (TREE_OPERAND (t, 0), args, complain, in_decl, false);
-  tree type = tsubst (TREE_OPERAND (t, 1), args, complain, in_decl);
-  --processing_template_decl;
-  bool noexcept_p = COMPOUND_REQ_NOEXCEPT_P (t);
-  return finish_compound_requirement (expr, type, noexcept_p);
-}
+  local_specialization_stack stack (lss_copy);
 
-/* Substitute ARGS into the nested requirement T. */
+  subst_info info (complain, in_decl);
 
-tree
-tsubst_nested_requirement (tree t, tree args,
-                           tsubst_flags_t complain, tree in_decl)
-{
-  ++processing_template_decl;
-  tree expr = tsubst_expr (TREE_OPERAND (t, 0), args, complain, in_decl, false);
-  --processing_template_decl;
-  return finish_nested_requirement (expr);
-}
-
-/* Substitute ARGS into the requirement T.  */
-
-inline tree
-tsubst_requirement (tree t, tree args, tsubst_flags_t complain, tree in_decl)
-{
-  switch (TREE_CODE (t))
-    {
-    case SIMPLE_REQ:
-      return tsubst_simple_requirement (t, args, complain, in_decl);
-    case TYPE_REQ:
-      return tsubst_type_requirement (t, args, complain, in_decl);
-    case COMPOUND_REQ:
-      return tsubst_compound_requirement (t, args, complain, in_decl);
-    case NESTED_REQ:
-      return tsubst_nested_requirement (t, args, complain, in_decl);
-    default:
-      gcc_unreachable ();
-    }
-  return error_mark_node;
-}
-
-/* Substitute ARGS into the list of requirements T. Note that
-   substitution failures here result in ill-formed programs. */
-
-tree
-tsubst_requirement_body (tree t, tree args,
-                         tsubst_flags_t complain, tree in_decl)
-{
-  tree r = NULL_TREE;
-  while (t)
-    {
-      tree e = tsubst_requirement (TREE_VALUE (t), args, complain, in_decl);
-      if (e == error_mark_node)
-        return error_mark_node;
-      r = tree_cons (NULL_TREE, e, r);
-      t = TREE_CHAIN (t);
-    }
-  /* Ensure that the order of constraints is the same as the original.  */
-  return nreverse (r);
-}
-
-} /* namespace */
-
-/* Substitute ARGS into the requires expression T. Note that this
-   results in the re-declaration of local parameters when
-   substituting through the parameter list. If either substitution
-   fails, the program is ill-formed. */
-
-tree
-tsubst_requires_expr (tree t, tree args,
-                      tsubst_flags_t complain, tree in_decl)
-{
-  local_specialization_stack stack;
+  /* A requires-expression is an unevaluated context.  */
+  cp_unevaluated u;
 
   tree parms = TREE_OPERAND (t, 0);
   if (parms)
     {
-      parms = tsubst_constraint_variables (parms, args, complain, in_decl);
+      parms = tsubst_constraint_variables (parms, args, info);
       if (parms == error_mark_node)
-        return error_mark_node;
+       return boolean_false_node;
     }
 
   tree reqs = TREE_OPERAND (t, 1);
-  reqs = tsubst_requirement_body (reqs, args, complain, in_decl);
+  reqs = tsubst_requirement_body (reqs, args, info);
   if (reqs == error_mark_node)
-    return error_mark_node;
+    return boolean_false_node;
+
+  /* In certain cases, produce a new requires-expression.
+     Otherwise the value of the expression is true.  */
+  if (processing_template_decl && uses_template_parms (args))
+    return finish_requires_expr (cp_expr_location (t), parms, reqs);
 
-  return finish_requires_expr (parms, reqs);
+  return boolean_true_node;
 }
 
 /* Substitute ARGS into the constraint information CI, producing a new
-   constraint record. */
+   constraint record.  */
 
 tree
 tsubst_constraint_info (tree t, tree args,
@@ -1904,378 +1998,418 @@ tsubst_constraint_info (tree t, tree args,
   if (!t || t == error_mark_node || !check_constraint_info (t))
     return NULL_TREE;
 
-  tree tmpl_constr = NULL_TREE;
-  if (tree r = CI_TEMPLATE_REQS (t))
-    tmpl_constr = tsubst_constraint (r, args, complain, in_decl);
+  tree tr = tsubst_constraint (CI_TEMPLATE_REQS (t), args, complain, in_decl);
+  tree dr = tsubst_constraint (CI_DECLARATOR_REQS (t), args, complain, in_decl);
+  return build_constraints (tr, dr);
+}
 
-  tree decl_constr = NULL_TREE;
-  if (tree r = CI_DECLARATOR_REQS (t))
-    decl_constr = tsubst_constraint (r, args, complain, in_decl);
+/* Substitute through a parameter mapping, in order to get the actual
+   arguments used to instantiate an atomic constraint.  This may fail
+   if the substitution into arguments produces something ill-formed.  */
 
-  return build_constraints (tmpl_constr, decl_constr);
-}
+static tree
+tsubst_parameter_mapping (tree map, tree args, subst_info info)
+{
+  if (!map)
+    return NULL_TREE;
+
+  tsubst_flags_t complain = info.complain;
+  tree in_decl = info.in_decl;
 
-/* Substitute ARGS into the constraint T. */
+  tree result = NULL_TREE;
+  for (tree p = map; p; p = TREE_CHAIN (p))
+    {
+      if (p == error_mark_node)
+        return error_mark_node;
+      tree parm = TREE_VALUE (p);
+      tree arg = TREE_PURPOSE (p);
+      tree new_arg = NULL_TREE;
+      if (TYPE_P (arg))
+        {
+          /* If a template parameter is declared with a placeholder, we can
+             get those in the argument list if decltype is applied to the
+             placeholder. For example:
+
+               template<auto T>
+                 requires C<decltype(T)>
+               void f() { }
+
+            The normalized argument for C will be an auto type, so we'll
+             need to deduce the actual argument from the corresponding
+             initializer (whatever argument is provided for T), and use
+             that result in the instantiated parameter mapping.  */
+          if (tree auto_node = type_uses_auto (arg))
+            {
+              int level;
+              int index;
+             template_parm_level_and_index (parm, &level, &index);
+             tree init = TMPL_ARG (args, level, index);
+              new_arg = do_auto_deduction (arg, init, auto_node,
+                                          complain, adc_variable_type,
+                                          make_tree_vec (0));
+            }
+        }
+      else if (ARGUMENT_PACK_P (arg))
+       new_arg = tsubst_argument_pack (arg, args, complain, in_decl);
+      if (!new_arg)
+       new_arg = tsubst_template_arg (arg, args, complain, in_decl);
+      if (new_arg == error_mark_node)
+       return error_mark_node;
+
+      result = tree_cons (new_arg, parm, result);
+    }
+  return nreverse (result);
+}
 
 tree
-tsubst_constraint (tree t, tree args, tsubst_flags_t complain, tree in_decl)
+tsubst_parameter_mapping (tree map, tree args, tsubst_flags_t complain, tree in_decl)
 {
-  if (t == NULL_TREE || t == error_mark_node)
-    return t;
-  switch (TREE_CODE (t))
-  {
-  case PRED_CONSTR:
-    return tsubst_predicate_constraint (t, args, complain, in_decl);
-  case CHECK_CONSTR:
-    return tsubst_check_constraint (t, args, complain, in_decl);
-  case CONJ_CONSTR:
-  case DISJ_CONSTR:
-    return tsubst_logical_operator (t, args, complain, in_decl);
-  case PARM_CONSTR:
-    return tsubst_parameterized_constraint (t, args, complain, in_decl);
-  case EXPR_CONSTR:
-    return tsubst_expr_constr (t, args, complain, in_decl);
-  case TYPE_CONSTR:
-    return tsubst_type_constr (t, args, complain, in_decl);
-  case ICONV_CONSTR:
-    return tsubst_implicit_conversion_constr (t, args, complain, in_decl);
-  case DEDUCT_CONSTR:
-    return tsubst_argument_deduction_constr (t, args, complain, in_decl);
-  case EXCEPT_CONSTR:
-    return tsubst_exception_constr (t, args, complain, in_decl);
-  default:
-    gcc_unreachable ();
-  }
-  return error_mark_node;
+  return tsubst_parameter_mapping (map, args, subst_info (complain, in_decl));
 }
 
 /*---------------------------------------------------------------------------
                         Constraint satisfaction
 ---------------------------------------------------------------------------*/
 
-/* The following functions determine if a constraint, when
-   substituting template arguments, is satisfied. For convenience,
-   satisfaction reduces a constraint to either true or false (and
-   nothing else). */
-
-namespace {
-
-tree satisfy_constraint_1 (tree, tree, tsubst_flags_t, tree);
+/* Hash functions for satisfaction entries.  */
 
-/* Check the constraint pack expansion.  */
-
-tree
-satisfy_pack_expansion (tree t, tree args,
-                      tsubst_flags_t complain, tree in_decl)
+struct GTY((for_user)) sat_entry
 {
-  /* Get the vector of satisfaction results.
-     gen_elem_of_pack_expansion_instantiation will check that each element of
-     the expansion is satisfied.  */
-  tree exprs = tsubst_pack_expansion (t, args, complain, in_decl);
+  tree constr;
+  tree args;
+  tree result;
+};
 
-  if (exprs == error_mark_node)
-    return boolean_false_node;
+struct sat_hasher : ggc_ptr_hash<sat_entry>
+{
+  static hashval_t hash (sat_entry *e)
+  {
+    hashval_t value = hash_atomic_constraint (e->constr);
+    return iterative_hash_template_arg (e->args, value);
+  }
 
-  /* TODO: It might be better to normalize each expanded term
-     and evaluate them separately. That would provide better
-     opportunities for diagnostics.  */
-  for (int i = 0; i < TREE_VEC_LENGTH (exprs); ++i)
-    if (TREE_VEC_ELT (exprs, i) != boolean_true_node)
-      return boolean_false_node;
-  return boolean_true_node;
-}
+  static bool equal (sat_entry *e1, sat_entry *e2)
+  {
+    if (!atomic_constraints_identical_p (e1->constr, e2->constr))
+      return false;
+    return template_args_equal (e1->args, e2->args);
+  }
+};
 
-/* A predicate constraint is satisfied if its expression evaluates
-   to true. If substitution into that node fails, the constraint
-   is not satisfied ([temp.constr.pred]).
+/* Cache the result of satisfy_atom.  */
+static GTY((deletable)) hash_table<sat_hasher> *sat_cache;
 
-   Note that a predicate constraint is a constraint expression
-   of type bool. If neither of those are true, the program is
-   ill-formed; they are not SFINAE'able errors. */
+/* Cache the result of constraints_satisfied_p.  */
+static GTY((deletable)) hash_map<tree,bool> *decl_satisfied_cache;
 
-tree
-satisfy_predicate_constraint (tree t, tree args,
-                              tsubst_flags_t complain, tree in_decl)
+static tree
+get_satisfaction (tree constr, tree args)
 {
-  tree expr = TREE_OPERAND (t, 0);
-
-  /* We should never have a naked pack expansion in a predicate constraint.  */
-  gcc_assert (TREE_CODE (expr) != EXPR_PACK_EXPANSION);
-
-  /* If substitution into the expression fails, the constraint
-     is not satisfied.  */
-  expr = tsubst_expr (expr, args, complain, in_decl, false);
-  if (expr == error_mark_node)
-    return boolean_false_node;
+  if (!sat_cache)
+    return NULL_TREE;
+  sat_entry elt = { constr, args, NULL_TREE };
+  sat_entry* found = sat_cache->find (&elt);
+  if (found)
+    return found->result;
+  else
+    return NULL_TREE;
+}
 
-  /* A predicate constraint shall have type bool. In some
-     cases, substitution gives us const-qualified bool, which
-     is also acceptable.  */
-  tree type = cv_unqualified (TREE_TYPE (expr));
-  if (!same_type_p (type, boolean_type_node))
-    {
-      error_at (cp_expr_loc_or_input_loc (expr),
-                "constraint %qE does not have type %qT",
-                expr, boolean_type_node);
-      return boolean_false_node;
-    }
+static void
+save_satisfaction (tree constr, tree args, tree result)
+{
+  if (!sat_cache)
+    sat_cache = hash_table<sat_hasher>::create_ggc (31);
+  sat_entry elt = {constr, args, result};
+  sat_entry** slot = sat_cache->find_slot (&elt, INSERT);
+  sat_entry* entry = ggc_alloc<sat_entry> ();
+  *entry = elt;
+  *slot = entry;
+}
 
-  return cxx_constant_value (expr);
+void
+clear_satisfaction_cache ()
+{
+  if (sat_cache)
+    sat_cache->empty ();
+  if (decl_satisfied_cache)
+    decl_satisfied_cache->empty ();
 }
 
-/* A concept check constraint like C<CARGS> is satisfied if substituting ARGS
-   into CARGS succeeds and C is satisfied for the resulting arguments.  */
+/* A tool to help manage satisfaction caching in satisfy_constraint_r.
+   Note the cache is only used when not diagnosing errors.  */
 
-tree
-satisfy_check_constraint (tree t, tree args,
-                          tsubst_flags_t complain, tree in_decl)
+struct satisfaction_cache
 {
-  tree decl = CHECK_CONSTR_CONCEPT (t);
-  tree tmpl = DECL_TI_TEMPLATE (decl);
-  tree cargs = CHECK_CONSTR_ARGS (t);
+  satisfaction_cache (tree constr, tree args, tsubst_flags_t complain)
+    : constr(constr), args(args), complain(complain)
+  { }
 
-  /* Instantiate the concept check arguments.  */
-  tree targs = tsubst (cargs, args, tf_none, NULL_TREE);
-  if (targs == error_mark_node)
-    return boolean_false_node;
+  tree get ()
+  {
+    if (complain == tf_none)
+      return get_satisfaction (constr, args);
+    return NULL_TREE;
+  }
 
-  /* Search for a previous value.  */
-  if (tree prev = lookup_concept_satisfaction (tmpl, targs))
-    return prev;
+  tree save (tree result)
+  {
+    if (complain == tf_none)
+      save_satisfaction (constr, args, result);
+    return result;
+  }
 
-  /* Expand the concept; failure here implies non-satisfaction.  */
-  tree def = expand_concept (decl, targs);
-  if (def == error_mark_node)
-    return memoize_concept_satisfaction (tmpl, args, boolean_false_node);
+  tree constr;
+  tree args;
+  tsubst_flags_t complain;
+};
 
-  /* Recursively satisfy the constraint.  */
-  tree result = satisfy_constraint_1 (def, targs, complain, in_decl);
-  return memoize_concept_satisfaction (tmpl, targs, result);
-}
+static int satisfying_constraint = 0;
 
-/* Check an expression constraint. The constraint is satisfied if
-   substitution succeeds ([temp.constr.expr]).
+/* Returns true if we are currently satisfying a constraint.
 
-   Note that the expression is unevaluated. */
+   This is used to guard against recursive calls to evaluate_concept_check
+   during template argument substitution.
 
-tree
-satisfy_expression_constraint (tree t, tree args,
-                               tsubst_flags_t complain, tree in_decl)
-{
-  cp_unevaluated guard;
-  deferring_access_check_sentinel deferring;
+   TODO: Do we need this now that we fully normalize prior to evaluation?
+   I think not. */
 
-  tree expr = EXPR_CONSTR_EXPR (t);
-  tree check = tsubst_expr (expr, args, complain, in_decl, false);
-  if (check == error_mark_node)
-    return boolean_false_node;
-  if (!perform_deferred_access_checks (tf_none))
-    return boolean_false_node;
-  return boolean_true_node;
+bool
+satisfying_constraint_p ()
+{
+  return satisfying_constraint;
 }
 
-/* Check a type constraint. The constraint is satisfied if
-   substitution succeeds. */
+/* Substitute ARGS into constraint-expression T during instantiation of
+   a member of a class template.  */
 
-inline tree
-satisfy_type_constraint (tree t, tree args,
-                         tsubst_flags_t complain, tree in_decl)
+tree
+tsubst_constraint (tree t, tree args, tsubst_flags_t complain, tree in_decl)
 {
-  deferring_access_check_sentinel deferring;
-  tree type = TYPE_CONSTR_TYPE (t);
-  gcc_assert (TYPE_P (type) || type == error_mark_node);
-  tree check = tsubst (type, args, complain, in_decl);
-  if (error_operand_p (check))
-    return boolean_false_node;
-  if (!perform_deferred_access_checks (complain))
-    return boolean_false_node;
-  return boolean_true_node;
+  /* We also don't want to evaluate concept-checks when substituting the
+     constraint-expressions of a declaration.  */
+  processing_constraint_expression_sentinel s;
+  tree expr = tsubst_expr (t, args, complain, in_decl, false);
+  return expr;
 }
 
-/* Check an implicit conversion constraint.  */
+static tree satisfy_constraint_r (tree, tree, subst_info info);
 
-tree
-satisfy_implicit_conversion_constraint (tree t, tree args,
-                                        tsubst_flags_t complain, tree in_decl)
-{
-  /* Don't tsubst as if we're processing a template. If we try
-     to we can end up generating template-like expressions
-     (e.g., modop-exprs) that aren't properly typed.  */
-  tree expr =
-    tsubst_expr (ICONV_CONSTR_EXPR (t), args, complain, in_decl, false);
-  if (expr == error_mark_node)
-    return boolean_false_node;
+/* Compute the satisfaction of a conjunction.  */
 
-  /* Get the transformed target type.  */
-  tree type = tsubst (ICONV_CONSTR_TYPE (t), args, complain, in_decl);
-  if (type == error_mark_node)
-    return boolean_false_node;
-
-  /* Attempt the conversion as a direct initialization
-     of the form TYPE <unspecified> = EXPR.  */
-  tree conv =
-    perform_direct_initialization_if_possible (type, expr, false, complain);
-  if (conv == NULL_TREE || conv == error_mark_node)
-    return boolean_false_node;
-  else
-    return boolean_true_node;
+static tree
+satisfy_conjunction (tree t, tree args, subst_info info)
+{
+  tree lhs = satisfy_constraint_r (TREE_OPERAND (t, 0), args, info);
+  if (lhs == error_mark_node || lhs == boolean_false_node)
+    return lhs;
+  return satisfy_constraint_r (TREE_OPERAND (t, 1), args, info);
 }
 
-/* Check an argument deduction constraint. */
+/* Compute the satisfaction of a disjunction.  */
 
-tree
-satisfy_argument_deduction_constraint (tree t, tree args,
-                                       tsubst_flags_t complain, tree in_decl)
+static tree
+satisfy_disjunction (tree t, tree args, subst_info info)
 {
-  /* Substitute through the expression. */
-  tree expr = DEDUCT_CONSTR_EXPR (t);
-  tree init = tsubst_expr (expr, args, complain, in_decl, false);
-  if (expr == error_mark_node)
-    return boolean_false_node;
+  /* Evaluate the operands quietly.  */
+  subst_info quiet (tf_none, NULL_TREE);
 
-  /* Perform auto or decltype(auto) deduction to get the result. */
-  tree pattern = DEDUCT_CONSTR_PATTERN (t);
-  tree placeholder = DEDUCT_CONSTR_PLACEHOLDER (t);
-  tree constr = PLACEHOLDER_TYPE_CONSTRAINTS (placeholder);
-  tree type_canonical = TYPE_CANONICAL (placeholder);
-  PLACEHOLDER_TYPE_CONSTRAINTS (placeholder)
-    = tsubst_constraint (constr, args, complain|tf_partial, in_decl);
-  TYPE_CANONICAL (placeholder) = NULL_TREE;
-  tree type = do_auto_deduction (pattern, init, placeholder,
-                                 complain, adc_requirement);
-  PLACEHOLDER_TYPE_CONSTRAINTS (placeholder) = constr;
-  TYPE_CANONICAL (placeholder) = type_canonical;
-  if (type == error_mark_node)
-    return boolean_false_node;
+  /* Register the constraint for diagnostics, if needed.  */
+  diagnosing_failed_constraint failure (t, args, info.noisy ());
 
-  return boolean_true_node;
+  tree lhs = satisfy_constraint_r (TREE_OPERAND (t, 0), args, quiet);
+  if (lhs == boolean_true_node)
+    return boolean_true_node;
+  tree rhs = satisfy_constraint_r (TREE_OPERAND (t, 1), args, quiet);
+  if (rhs != boolean_true_node && info.noisy ())
+    {
+      location_t loc = cp_expr_location (CONSTR_EXPR (t));
+      inform (loc, "neither operand of the disjunction is satisfied");
+      /* TODO: Replay the LHS and RHS to find failures in both branches.  */
+      // satisfy_constraint_r (TREE_OPERAND (t, 0), args, info);
+      // satisfy_constraint_r (TREE_OPERAND (t, 1), args, info);
+    }
+  return rhs;
 }
 
-/* Check an exception constraint. An exception constraint for an
-   expression e is satisfied when noexcept(e) is true. */
+/* Ensures that T is a truth value and not (accidentally, as sometimes
+   happens) an integer value.  */
 
 tree
-satisfy_exception_constraint (tree t, tree args,
-                              tsubst_flags_t complain, tree in_decl)
+satisfaction_value (tree t)
 {
-  tree expr = EXCEPT_CONSTR_EXPR (t);
-  tree check = tsubst_expr (expr, args, complain, in_decl, false);
-  if (check == error_mark_node)
-    return boolean_false_node;
-
-  if (expr_noexcept_p (check, complain))
+  if (t == error_mark_node)
+    return t;
+  if (t == boolean_true_node || t == integer_one_node)
     return boolean_true_node;
-  else
+  if (t == boolean_false_node || t == integer_zero_node)
     return boolean_false_node;
+
+  /* Anything else should be invalid.  */
+  gcc_assert (false);
 }
 
-/* Check a parameterized constraint. */
+/* Build a new template argument list with template arguments corresponding
+   to the parameters used in an atomic constraint.  */
 
 tree
-satisfy_parameterized_constraint (tree t, tree args,
-                                  tsubst_flags_t complain, tree in_decl)
+get_mapped_args (tree map)
 {
-  local_specialization_stack stack;
-  tree parms = PARM_CONSTR_PARMS (t);
-  tree vars = tsubst_constraint_variables (parms, args, complain, in_decl);
-  if (vars == error_mark_node)
-    return boolean_false_node;
-  tree constr = PARM_CONSTR_OPERAND (t);
-  return satisfy_constraint_1 (constr, args, complain, in_decl);
-}
-
-/* Check that the conjunction of constraints is satisfied. Note
-   that if left operand is not satisfied, the right operand
-   is not checked.
+  /* If there's no map, then there are no arguments.  */
+  if (!map)
+    return NULL_TREE;
 
-   FIXME: Check that this wouldn't result in a user-defined
-   operator. Note that this error is partially diagnosed in
-   satisfy_predicate_constraint. It would be nice to diagnose
-   the overload, but I don't think it's strictly necessary.  */
+  /* Find the mapped parameter with the highest level.  */
+  int count = 0;
+  for (tree p = map; p; p = TREE_CHAIN (p))
+    {
+      int level;
+      int index;
+      template_parm_level_and_index (TREE_VALUE (p), &level, &index);
+      if (level > count)
+        count = level;
+    }
 
-tree
-satisfy_conjunction (tree t, tree args, tsubst_flags_t complain, tree in_decl)
-{
-  tree t0 = satisfy_constraint_1 (TREE_OPERAND (t, 0), args, complain, in_decl);
-  if (t0 == boolean_false_node)
-    return boolean_false_node;
-  return satisfy_constraint_1 (TREE_OPERAND (t, 1), args, complain, in_decl);
-}
+  /* Place each argument at its corresponding position in the argument
+     list. Note that the list will be sparse (not all arguments supplied),
+     but instantiation is guaranteed to only use the parameters in the
+     mapping, so null arguments would never be used.  */
+  auto_vec< auto_vec<tree> > lists (count);
+  lists.quick_grow_cleared (count);
+  for (tree p = map; p; p = TREE_CHAIN (p))
+    {
+      int level;
+      int index;
+      template_parm_level_and_index (TREE_VALUE (p), &level, &index);
+
+      /* Insert the argument into its corresponding position.  */
+      auto_vec<tree> &list = lists[level - 1];
+      if (index >= (int)list.length ())
+       list.safe_grow_cleared (index + 1);
+      list[index] = TREE_PURPOSE (p);
+    }
 
-/* Check that the disjunction of constraints is satisfied. Note
-   that if the left operand is satisfied, the right operand is not
-   checked.  */
+  /* Build the actual argument list.  */
+  tree args = make_tree_vec (lists.length ());
+  for (unsigned i = 0; i != lists.length (); ++i)
+    {
+      auto_vec<tree> &list = lists[i];
+      tree level = make_tree_vec (list.length ());
+      for (unsigned j = 0; j < list.length(); ++j)
+       TREE_VEC_ELT (level, j) = list[j];
+      SET_TMPL_ARGS_LEVEL (args, i + 1, level);
+    }
+  SET_NON_DEFAULT_TEMPLATE_ARGS_COUNT (args, 0);
 
-tree
-satisfy_disjunction (tree t, tree args, tsubst_flags_t complain, tree in_decl)
-{
-  tree t0 = satisfy_constraint_1 (TREE_OPERAND (t, 0), args, complain, in_decl);
-  if (t0 == boolean_true_node)
-    return boolean_true_node;
-  return satisfy_constraint_1 (TREE_OPERAND (t, 1), args, complain, in_decl);
+  return args;
 }
 
-/* Dispatch to an appropriate satisfaction routine depending on the
-   tree code of T.  */
+static void diagnose_atomic_constraint (tree, tree, subst_info);
 
-tree
-satisfy_constraint_1 (tree t, tree args, tsubst_flags_t complain, tree in_decl)
-{
-  gcc_assert (!processing_template_decl);
+/* Compute the satisfaction of an atomic constraint.  */
 
-  if (!t)
-    return boolean_false_node;
+static tree
+satisfy_atom (tree t, tree args, subst_info info)
+{
+  satisfaction_cache cache (t, args, info.complain);
+  if (tree r = cache.get ())
+    return r;
 
-  if (t == error_mark_node)
-    return boolean_false_node;
+  /* Perform substitution quietly.  */
+  subst_info quiet (tf_none, NULL_TREE);
 
-  switch (TREE_CODE (t))
-  {
-  case PRED_CONSTR:
-    return satisfy_predicate_constraint (t, args, complain, in_decl);
+  /* In case there is a diagnostic, we want to establish the context
+     prior to printing errors.  If no errors occur, this context is
+     removed before returning.  */
+  diagnosing_failed_constraint failure (t, args, info.noisy ());
 
-  case CHECK_CONSTR:
-    return satisfy_check_constraint (t, args, complain, in_decl);
+  /* Instantiate the parameter mapping, so that we map directly to
+     the arguments provided to the instantiation.  */
+  tree map = tsubst_parameter_mapping (ATOMIC_CONSTR_MAP (t), args, quiet);
+  if (map == error_mark_node)
+    {
+      /* If instantiation of the parameter mapping fails, the program
+         is ill-formed.  */
+      if (info.noisy())
+       tsubst_parameter_mapping (ATOMIC_CONSTR_MAP (t), args, info);
+      return cache.save (boolean_false_node);
+    }
 
-  case EXPR_CONSTR:
-    return satisfy_expression_constraint (t, args, complain, in_decl);
+  /* Rebuild the argument vector from the parameter mapping.  */
+  args = get_mapped_args (map);
 
-  case TYPE_CONSTR:
-    return satisfy_type_constraint (t, args, complain, in_decl);
+  /* Apply the parameter mapping (i.e., just substitute).  */
+  tree expr = ATOMIC_CONSTR_EXPR (t);
+  tree result = tsubst_expr (expr, args, quiet.complain, quiet.in_decl, false);
+  if (result == error_mark_node)
+    {
+      /* If substitution results in an invalid type or expression, the constraint
+        is not satisfied. Replay the substitution.  */
+      if (info.noisy ())
+       tsubst_expr (expr, args, info.complain, info.in_decl, false);
+      return cache.save (boolean_false_node);
+    }
 
-  case ICONV_CONSTR:
-    return satisfy_implicit_conversion_constraint (t, args, complain, in_decl);
+  location_t loc = cp_expr_location (expr);
 
-  case DEDUCT_CONSTR:
-    return satisfy_argument_deduction_constraint (t, args, complain, in_decl);
+  /* [17.4.1.2] ... lvalue-to-value conversion is performed as necessary,
+     and EXPR shall be a constant expression of type bool.  */
+  result = force_rvalue (result, tf_error);
+  if (result == error_mark_node)
+    {
+      if (info.noisy ())
+        inform (loc, "cannot convert constraint to rvalue");
+      return cache.save (error_mark_node);
+    }
+  if (!same_type_p (TREE_TYPE (result), boolean_type_node))
+    {
+      if (info.noisy ())
+       inform (loc, "constraint does not have type %<bool%>");
+      return cache.save (error_mark_node);
+    }
 
-  case EXCEPT_CONSTR:
-    return satisfy_exception_constraint (t, args, complain, in_decl);
+  /* Compute the value of the constraint.  */
+  result = satisfaction_value (cxx_constant_value (result));
+  if (result == boolean_false_node && info.noisy ())
+    diagnose_atomic_constraint (t, args, info);
 
-  case PARM_CONSTR:
-    return satisfy_parameterized_constraint (t, args, complain, in_decl);
+  return cache.save (result);
+}
 
-  case CONJ_CONSTR:
-    return satisfy_conjunction (t, args, complain, in_decl);
+/* Determine if the normalized constraint T is satisfied.
+   Returns boolean_true_node if the expression/constraint is
+   satisfied, boolean_false_node if not, and error_mark_node
+   if the there was an error evaluating the constraint.
 
-  case DISJ_CONSTR:
-    return satisfy_disjunction (t, args, complain, in_decl);
+   The parameter mapping of atomic constraints is simply the
+   set of template arguments that will be substituted into
+   the expression, regardless of template parameters appearing
+   withing. Whether a template argument is used in the atomic
+   constraint only matters for subsumption.  */
 
-  case EXPR_PACK_EXPANSION:
-    return satisfy_pack_expansion (t, args, complain, in_decl);
+static tree
+satisfy_constraint_r (tree t, tree args, subst_info info)
+{
+  if (t == error_mark_node)
+    return error_mark_node;
 
-  default:
-    gcc_unreachable ();
-  }
-  return boolean_false_node;
+  switch (TREE_CODE (t))
+    {
+    case CONJ_CONSTR:
+      return satisfy_conjunction (t, args, info);
+    case DISJ_CONSTR:
+      return satisfy_disjunction (t, args, info);
+    case ATOMIC_CONSTR:
+      return satisfy_atom (t, args, info);
+    default:
+      gcc_unreachable ();
+    }
 }
 
-/* Check that the constraint is satisfied, according to the rules
-   for that constraint. Note that each satisfy_* function returns
-   true or false, depending on whether it is satisfied or not.  */
+/* Check that the normalized constraint T is satisfied for ARGS.  */
 
-tree
-satisfy_constraint (tree t, tree args)
+static tree
+satisfy_constraint (tree t, tree args, subst_info info)
 {
   auto_timevar time (TV_CONSTRAINT_SAT);
 
@@ -2283,146 +2417,199 @@ satisfy_constraint (tree t, tree args)
      to non-dependent terms, so we want to ensure full checking here.  */
   processing_template_decl_sentinel proc (true);
 
-  /* Avoid early exit in tsubst and tsubst_copy from null args; since earlier
-     substitution was done with processing_template_decl forced on, there will
-     be expressions that still need semantic processing, possibly buried in
-     decltype or a template argument.  */
+  /* We need to check access during satisfaction.  */
+  deferring_access_check_sentinel acs (dk_no_deferred);
+
+  /* Avoid early exit in tsubst and tsubst_copy from null args.  */
   if (args == NULL_TREE)
     args = make_tree_vec (1);
 
-  return satisfy_constraint_1 (t, args, tf_none, NULL_TREE);
+  return satisfy_constraint_r (t, args, info);
 }
 
-/* Check the associated constraints in CI against the given
-   ARGS, returning true when the constraints are satisfied
-   and false otherwise.  */
+/* Check the normalized constraints T against ARGS, returning a satisfaction
+   value (either true, false, or error).  */
 
-tree
-satisfy_associated_constraints (tree ci, tree args)
+static tree
+satisfy_associated_constraints (tree t, tree args, subst_info info)
 {
   /* If there are no constraints then this is trivially satisfied. */
-  if (!ci)
+  if (!t)
     return boolean_true_node;
 
   /* If any arguments depend on template parameters, we can't
-     check constraints. */
+     check constraints. Pretend they're satisfied for now.  */
   if (args && uses_template_parms (args))
     return boolean_true_node;
 
-  /* Check if we've seen a previous result. */
-  if (tree prev = lookup_constraint_satisfaction (ci, args))
-    return prev;
-
-  /* Actually test for satisfaction. */
-  tree result = satisfy_constraint (CI_ASSOCIATED_CONSTRAINTS (ci), args);
-  return memoize_constraint_satisfaction (ci, args, result);
+  return satisfy_constraint (t, args, info);
 }
 
-} /* namespace */
+/* Evaluate EXPR as a constraint expression using ARGS, returning a
+   satisfaction value. */
 
-/* Evaluate the given constraint, returning boolean_true_node
-   if the constraint is satisfied and boolean_false_node
-   otherwise. */
-
-tree
-evaluate_constraints (tree constr, tree args)
+static tree
+satisfy_constraint_expression (tree expr, tree args, subst_info info)
 {
-  gcc_assert (constraint_p (constr));
-  return satisfy_constraint (constr, args);
+  /* Normalize the expression before satisfaction testing.  */
+  tree norm;
+  if (args == NULL_TREE && concept_check_p (expr))
+    {
+      tree id = unpack_concept_check (expr);
+      args = TREE_OPERAND (id, 1);
+      tree tmpl = get_concept_check_template (id);
+      norm = normalize_concept_definition (tmpl);
+    }
+  else
+    norm = normalize_constraint_expression (expr);
+  return satisfy_constraint (norm, args, info);
 }
 
-/* Evaluate the function concept FN by substituting its own args
-   into its definition and evaluating that as the result. Returns
-   boolean_true_node if the constraints are satisfied and
-   boolean_false_node otherwise.  */
+/* Used to evaluate concept checks and requires-expressions during
+   constant expression evaluation.  */
 
 tree
-evaluate_function_concept (tree fn, tree args)
+satisfy_constraint_expression (tree expr)
 {
-  tree constr = build_nt (CHECK_CONSTR, fn, args);
-  return satisfy_constraint (constr, args);
+  subst_info info (tf_none, NULL_TREE);
+  return satisfy_constraint_expression (expr, NULL_TREE, info);
 }
 
-/* Evaluate the variable concept VAR by substituting its own args into
-   its initializer and checking the resulting constraint. Returns
-   boolean_true_node if the constraints are satisfied and
-   boolean_false_node otherwise.  */
+/* True if T is satisfied for ARGS.  */
 
-tree
-evaluate_variable_concept (tree var, tree args)
+static bool
+constraint_expression_satisfied_p (tree t, tree args, subst_info info)
 {
-  tree constr = build_nt (CHECK_CONSTR, var, args);
-  return satisfy_constraint (constr, args);
+  tree r = satisfy_constraint_expression (t, args, info);
+  return r == boolean_true_node;
 }
 
-/* Evaluate the given expression as if it were a predicate
-   constraint. Returns boolean_true_node if the constraint
-   is satisfied and boolean_false_node otherwise. */
-
-tree
-evaluate_constraint_expression (tree expr, tree args)
+static bool
+constraints_satisfied_p (tree t, subst_info info)
 {
-  tree constr = normalize_expression (expr);
-  return satisfy_constraint (constr, args);
-}
+  if (!DECL_P (t))
+    return constraint_expression_satisfied_p (t, NULL_TREE, info);
 
-/* Returns true if the DECL's constraints are satisfied.
-   This is used in cases where a declaration is formed but
-   before it is used (e.g., overload resolution). */
+  /* For inherited constructors, consider the original declaration;
+     it has the correct template information attached. */
+  if (flag_new_inheriting_ctors)
+    t = strip_inheriting_ctors (t);
+
+  /* Update the declaration for diagnostics.  */
+  info.in_decl = t;
+
+  if (info.quiet ())
+    if (bool *p = hash_map_safe_get (decl_satisfied_cache, t))
+      return *p;
 
-bool
-constraints_satisfied_p (tree decl)
-{
   /* Get the constraints to check for satisfaction. This depends
      on whether we're looking at a template specialization or not. */
-  tree ci;
+  tree norm = NULL_TREE;
   tree args = NULL_TREE;
-  if (tree ti = DECL_TEMPLATE_INFO (decl))
+  tree ti = DECL_TEMPLATE_INFO (t);
+  if (ti)
     {
       tree tmpl = TI_TEMPLATE (ti);
-      ci = get_constraints (tmpl);
-      int depth = TMPL_PARMS_DEPTH (DECL_TEMPLATE_PARMS (tmpl));
-      args = get_innermost_template_args (TI_ARGS (ti), depth);
+      norm = normalize_template_requirements (tmpl, info.noisy ());
+
+      /* The initial parameter mapping is the complete set of
+         template arguments substituted into the declaration.  */
+      args = TI_ARGS (ti);
     }
   else
     {
-      ci = get_constraints (decl);
+      /* These should be empty until we allow constraints on non-templates.  */
+      norm = normalize_nontemplate_requirements (t, info.noisy ());
+    }
+
+  bool r = true;
+  if (norm)
+    {
+      push_access_scope (t);
+      tree eval = satisfy_associated_constraints (norm, args, info);
+      pop_access_scope (t);
+      r = (eval == boolean_true_node);
     }
 
-  tree eval = satisfy_associated_constraints (ci, args);
-  return eval == boolean_true_node;
+  if (info.quiet ())
+    hash_map_safe_put<hm_ggc> (decl_satisfied_cache, t, r);
+
+  return r;
 }
 
-/* Returns true if the constraints are satisfied by ARGS.
-   Here, T can be either a constraint or a constrained
-   declaration.  */
+/* Returns true if the T's constraints are satisfied, of if T is an expression,
+   if T is satisfied. This is used in cases where the arguments can be
+   determined from the declaration or expression.
+
+   Note that T is typically a template specialization.  */
 
 bool
-constraints_satisfied_p (tree t, tree args)
+constraints_satisfied_p (tree t)
 {
-  tree eval;
-  if (constraint_p (t))
-    eval = evaluate_constraints (t, args);
-  else
-    eval = satisfy_associated_constraints (get_constraints (t), args);
-  return eval == boolean_true_node;
+  subst_info info (tf_none, NULL_TREE);
+  return constraints_satisfied_p (t, info);
 }
 
-namespace
+/* Returns true if the expression or constrained declaration T is
+   satisfied by ARGS.  In this case, we don't have a specialization
+   where we can cache the results (e.g., alias templates).  */
+
+static bool
+constraints_satisfied_p (tree t, tree args, subst_info info)
 {
+  if (!DECL_P (t))
+    return constraint_expression_satisfied_p (t, args, info);
+
+  /* Update the declaration for diagnostics.  */
+  info.in_decl = t;
+
+  gcc_assert (TREE_CODE (t) == TEMPLATE_DECL);
+  if (tree norm = normalize_template_requirements (t, info.noisy ()))
+    {
+      tree pattern = DECL_TEMPLATE_RESULT (t);
+      push_access_scope (pattern);
+      tree eval = satisfy_associated_constraints (norm, args, info);
+      pop_access_scope (pattern);
+      return eval == boolean_true_node;
+    }
 
-/* Normalize EXPR and determine if the resulting constraint is
-   satisfied by ARGS. Returns true if and only if the constraint
-   is satisfied.  This is used extensively by diagnostics to
-   determine causes for failure.  */
+  return true;
+}
 
-inline bool
-constraint_expression_satisfied_p (tree expr, tree args)
+bool
+constraints_satisfied_p (tree t, tree args)
 {
-  return evaluate_constraint_expression (expr, args) == boolean_true_node;
+  subst_info info (tf_none, NULL);
+  return constraints_satisfied_p (t, args, info);
 }
 
-} /* namespace */
+/* Evaluate a concept check of the form C<ARGS>, returning either TRUE
+   or FALSE. If ARGS contains any template parameters, this returns the
+   check. If satisfaction yields a hard error, diagnose the error.  */
+
+tree
+evaluate_concept_check (tree check, tsubst_flags_t complain)
+{
+  /* FIXME we ought to be able to pass complain into subst_info rather
+     than repeat satisfaction, but currently that will complain about
+     non-satisfaction as well as errors.  */
+  if (check == error_mark_node)
+    return error_mark_node;
+
+  gcc_assert (concept_check_p (check));
+
+  subst_info info (tf_none, NULL_TREE);
+  tree result = satisfy_constraint_expression (check, NULL_TREE, info);
+  if (result == error_mark_node && (complain & tf_error))
+    {
+      location_t loc = cp_expr_loc_or_input_loc (check);
+      error_at (loc, "concept satisfaction failed");
+      info.complain = complain;
+      satisfy_constraint_expression (check, NULL_TREE, info);
+    }
+
+  return result;
+}
 
 /*---------------------------------------------------------------------------
                 Semantic analysis of requires-expressions
@@ -2430,8 +2617,9 @@ constraint_expression_satisfied_p (tree expr, tree args)
 
 /* Finish a requires expression for the given PARMS (possibly
    null) and the non-empty sequence of requirements.  */
+
 tree
-finish_requires_expr (tree parms, tree reqs)
+finish_requires_expr (location_t loc, tree parms, tree reqs)
 {
   /* Modify the declared parameters by removing their context
      so they don't refer to the enclosing scope and explicitly
@@ -2446,41 +2634,53 @@ finish_requires_expr (tree parms, tree reqs)
   tree r = build_min (REQUIRES_EXPR, boolean_type_node, parms, reqs);
   TREE_SIDE_EFFECTS (r) = false;
   TREE_CONSTANT (r) = true;
+  SET_EXPR_LOCATION (r, loc);
   return r;
 }
 
-/* Construct a requirement for the validity of EXPR. */
+/* Construct a requirement for the validity of EXPR.   */
+
 tree
-finish_simple_requirement (tree expr)
+finish_simple_requirement (location_t loc, tree expr)
 {
-  return build_nt (SIMPLE_REQ, expr);
+  tree r = build_nt (SIMPLE_REQ, expr);
+  SET_EXPR_LOCATION (r, loc);
+  return r;
 }
 
-/* Construct a requirement for the validity of TYPE. */
+/* Construct a requirement for the validity of TYPE.  */
+
 tree
-finish_type_requirement (tree type)
+finish_type_requirement (location_t loc, tree type)
 {
-  return build_nt (TYPE_REQ, type);
+  tree r = build_nt (TYPE_REQ, type);
+  SET_EXPR_LOCATION (r, loc);
+  return r;
 }
 
 /* Construct a requirement for the validity of EXPR, along with
    its properties. if TYPE is non-null, then it specifies either
    an implicit conversion or argument deduction constraint,
    depending on whether any placeholders occur in the type name.
-   NOEXCEPT_P is true iff the noexcept keyword was specified. */
+   NOEXCEPT_P is true iff the noexcept keyword was specified.  */
+
 tree
-finish_compound_requirement (tree expr, tree type, bool noexcept_p)
+finish_compound_requirement (location_t loc, tree expr, tree type, bool noexcept_p)
 {
   tree req = build_nt (COMPOUND_REQ, expr, type);
+  SET_EXPR_LOCATION (req, loc);
   COMPOUND_REQ_NOEXCEPT_P (req) = noexcept_p;
   return req;
 }
 
-/* Finish a nested requirement. */
+/* Finish a nested requirement.  */
+
 tree
-finish_nested_requirement (tree expr)
+finish_nested_requirement (location_t loc, tree expr)
 {
-  return build_nt (NESTED_REQ, expr);
+  tree r = build_nt (NESTED_REQ, expr);
+  SET_EXPR_LOCATION (r, loc);
+  return r;
 }
 
 /* Check that FN satisfies the structural requirements of a
@@ -2500,7 +2700,7 @@ check_function_concept (tree fn)
   if (TREE_CODE (body) == CLEANUP_POINT_EXPR)
     body = TREE_OPERAND (body, 0);
 
-  /* Check that the definition is written correctly. */
+  /* Check that the definition is written correctly.  */
   if (TREE_CODE (body) != RETURN_EXPR)
     {
       location_t loc = DECL_SOURCE_LOCATION (fn);
@@ -2580,108 +2780,93 @@ subsumes_constraints (tree a, tree b)
   return subsumes (a, b);
 }
 
-/* Returns true when the the constraints in A subsume those in B, but
-   the constraints in B do not subsume the constraints in A.  */
+/* Returns true when the the constraints in CI (with arguments
+   ARGS) strictly subsume the associated constraints of TMPL.  */
+
+bool
+strictly_subsumes (tree ci, tree args, tree tmpl)
+{
+  tree n1 = get_normalized_constraints_from_info (ci, args, NULL_TREE);
+  tree n2 = get_normalized_constraints_from_decl (tmpl);
+
+  return subsumes (n1, n2) && !subsumes (n2, n1);
+}
+
+/* REturns true when the constraints in CI (with arguments ARGS) subsume
+   the associated constraints of TMPL.  */
 
 bool
-strictly_subsumes (tree a, tree b)
+weakly_subsumes (tree ci, tree args, tree tmpl)
 {
-  return subsumes (a, b) && !subsumes (b, a);
+  tree n1 = get_normalized_constraints_from_info (ci, args, NULL_TREE);
+  tree n2 = get_normalized_constraints_from_decl (tmpl);
+
+  return subsumes (n1, n2);
 }
 
 /* Determines which of the declarations, A or B, is more constrained.
    That is, which declaration's constraints subsume but are not subsumed
    by the other's?
 
-   Returns 1 if A is more constrained than B, -1 if B is more constrained
-   than A, and 0 otherwise. */
+   Returns 1 if D1 is more constrained than D2, -1 if D2 is more constrained
+   than D1, and 0 otherwise. */
 
 int
 more_constrained (tree d1, tree d2)
 {
-  tree c1 = get_constraints (d1);
-  tree c2 = get_constraints (d2);
+  tree n1 = get_normalized_constraints_from_decl (d1);
+  tree n2 = get_normalized_constraints_from_decl (d2);
+
   int winner = 0;
-  if (subsumes_constraints (c1, c2))
+  if (subsumes (n1, n2))
     ++winner;
-  if (subsumes_constraints (c2, c1))
+  if (subsumes (n2, n1))
     --winner;
   return winner;
 }
 
-/* Returns true if D1 is at least as constrained as D2. That is, the
-   associated constraints of D1 subsume those of D2, or both declarations
-   are unconstrained. */
-
-bool
-at_least_as_constrained (tree d1, tree d2)
-{
-  tree c1 = get_constraints (d1);
-  tree c2 = get_constraints (d2);
-  return subsumes_constraints (c1, c2);
-}
-
-
 /*---------------------------------------------------------------------------
                         Constraint diagnostics
-
-FIXME: Normalize expressions into constraints before evaluating them.
-This should be the general pattern for all such diagnostics.
 ---------------------------------------------------------------------------*/
 
-/* The number of detailed constraint failures.  */
-
-int constraint_errors = 0;
-
-/* Do not generate errors after diagnosing this number of constraint
-   failures.
-
-   FIXME: This is a really arbitrary number. Provide better control of
-   constraint diagnostics with a command line option.  */
-
-int constraint_thresh = 20;
-
-
-/* Returns true if we should elide the diagnostic for a constraint failure.
-   This is the case when the number of errors has exceeded the pre-configured
-   threshold.  */
-
-inline bool
-elide_constraint_failure_p ()
-{
-  bool ret = constraint_thresh <= constraint_errors;
-  ++constraint_errors;
-  return ret;
-}
-
-/* Returns the number of undiagnosed errors. */
-
-inline int
-undiagnosed_constraint_failures ()
-{
-  return constraint_errors - constraint_thresh;
+/* Returns the best location to diagnose a constraint error.  */
+
+static location_t
+get_constraint_error_location (tree t)
+{
+  /* If we have a specific location give it.  */
+  tree expr = CONSTR_EXPR (t);
+  if (location_t loc = cp_expr_location (expr))
+    return loc;
+
+  /* If the constraint is normalized from a requires-clause, give
+     the location as that of the constrained declaration.  */
+  tree cxt = CONSTR_CONTEXT (t);
+  tree src = TREE_VALUE (cxt);
+  if (!src)
+    /* TODO: This only happens for constrained non-template declarations.  */
+    return input_location;
+  if (DECL_P (src))
+    return DECL_SOURCE_LOCATION (src);
+
+  /* Otherwise, give the location as the defining concept.  */
+  gcc_assert (concept_check_p (src));
+  tree id = unpack_concept_check (src);
+  tree tmpl = TREE_OPERAND (id, 0);
+  if (OVL_P (tmpl))
+    tmpl = OVL_FIRST (tmpl);
+  return DECL_SOURCE_LOCATION (tmpl);
 }
 
-/* The diagnosis of constraints performs a combination of normalization
-   and satisfaction testing. We recursively walk through the conjunction or
-   disjunction of associated constraints, testing each sub-constraint in
-   turn.  */
-
-namespace {
-
-void diagnose_constraint (location_t, tree, tree, tree);
-
-/* Emit a specific diagnostics for a failed trait.  */
+/* Emit a diagnostic for a failed trait.  */
 
 void
-diagnose_trait_expression (location_t loc, tree, tree cur, tree args)
+diagnose_trait_expr (tree expr, tree args)
 {
-  if (constraint_expression_satisfied_p (cur, args))
-    return;
-  if (elide_constraint_failure_p())
-    return;
+  location_t loc = cp_expr_location (expr);
 
-  tree expr = PRED_CONSTR_EXPR (cur);
+  /* Build a "fake" version of the instantiated trait, so we can
+     get the instantiated types from result.  */
   ++processing_template_decl;
   expr = tsubst_expr (expr, args, tf_none, NULL_TREE, false);
   --processing_template_decl;
@@ -2758,390 +2943,224 @@ diagnose_trait_expression (location_t loc, tree, tree cur, tree args)
     }
 }
 
-/* Diagnose the expression of a predicate constraint.  */
-
-void
-diagnose_other_expression (location_t loc, tree, tree cur, tree args)
+static tree
+diagnose_valid_expression (tree expr, tree args, tree in_decl)
 {
-  if (constraint_expression_satisfied_p (cur, args))
-    return;
-  if (elide_constraint_failure_p())
-    return;
-  inform (loc, "%qE evaluated to false", cur);
-}
+  tree result = tsubst_expr (expr, args, tf_none, in_decl, false);
+  if (result != error_mark_node)
+    return result;
 
-/* Do our best to infer meaning from predicates.  */
+  location_t loc = cp_expr_loc_or_input_loc (expr);
+  inform (loc, "the required expression %qE is invalid", expr);
 
-inline void
-diagnose_predicate_constraint (location_t loc, tree orig, tree cur, tree args)
-{
-  if (TREE_CODE (PRED_CONSTR_EXPR (cur)) == TRAIT_EXPR)
-    diagnose_trait_expression (loc, orig, cur, args);
-  else
-    diagnose_other_expression (loc, orig, cur, args);
-}
+  /* TODO: Replay the substitution to diagnose the error?  */
+  // tsubst_expr (expr, args, tf_error, in_decl, false);
 
-/* Diagnose a failed pack expansion, possibly containing constraints.  */
+  return error_mark_node;
+}
 
-void
-diagnose_pack_expansion (location_t loc, tree, tree cur, tree args)
+static tree
+diagnose_valid_type (tree type, tree args, tree in_decl)
 {
-  if (constraint_expression_satisfied_p (cur, args))
-    return;
-  if (elide_constraint_failure_p())
-    return;
-
-  /* Make sure that we don't have naked packs that we don't expect. */
-  if (!same_type_p (TREE_TYPE (cur), boolean_type_node))
-    {
-      inform (loc, "invalid pack expansion in constraint %qE", cur);
-      return;
-    }
+  tree result = tsubst (type, args, tf_none, in_decl);
+  if (result != error_mark_node)
+    return result;
 
-  inform (loc, "in the expansion of %qE", cur);
+  location_t loc = cp_expr_loc_or_input_loc (type);
+  inform (loc, "the required type %qT is invalid", type);
 
-  /* Get the vector of expanded arguments. Note that n must not
-     be 0 since this constraint is not satisfied.  */
-  ++processing_template_decl;
-  tree exprs = tsubst_pack_expansion (cur, args, tf_none, NULL_TREE);
-  --processing_template_decl;
-  if (exprs == error_mark_node)
-    {
-      /* TODO: This error message could be better. */
-      inform (loc, "    substitution failure occurred during expansion");
-      return;
-    }
+  /* TODO: Replay the substitution to diagnose the error?  */
+  // tsubst (type, args, tf_error, in_decl);
 
-  /* Check each expanded constraint separately. */
-  int n = TREE_VEC_LENGTH (exprs);
-  for (int i = 0; i < n; ++i)
-    {
-      tree expr = TREE_VEC_ELT (exprs, i);
-      if (!constraint_expression_satisfied_p (expr, args))
-        inform (loc, "    %qE was not satisfied", expr);
-    }
+  return error_mark_node;
 }
 
-/* Diagnose a potentially unsatisfied concept check constraint DECL<CARGS>.
-   Parameters are as for diagnose_constraint.  */
-
-void
-diagnose_check_constraint (location_t loc, tree orig, tree cur, tree args)
+static void
+diagnose_simple_requirement (tree req, tree args, tree in_decl)
 {
-  if (constraints_satisfied_p (cur, args))
-    return;
-
-  tree decl = CHECK_CONSTR_CONCEPT (cur);
-  tree cargs = CHECK_CONSTR_ARGS (cur);
-  tree tmpl = DECL_TI_TEMPLATE (decl);
-  tree check = build_nt (CHECK_CONSTR, decl, cargs);
-
-  /* Instantiate the concept check arguments.  */
-  tree targs = tsubst (cargs, args, tf_none, NULL_TREE);
-  if (targs == error_mark_node)
-    {
-      if (elide_constraint_failure_p ())
-        return;
-      inform (loc, "invalid use of the concept %qE", check);
-      tsubst (cargs, args, tf_warning_or_error, NULL_TREE);
-      return;
-    }
-
-  tree sub = build_tree_list (tmpl, targs);
-  /* Update to the expanded definitions. */
-  cur = expand_concept (decl, targs);
-  if (cur == error_mark_node)
-    {
-      if (elide_constraint_failure_p ())
-        return;
-      inform (loc, "in the expansion of concept %<%E %S%>", check, sub);
-      cur = get_concept_definition (decl);
-      tsubst_expr (cur, targs, tf_warning_or_error, NULL_TREE, false);
-      return;
-    }
-
-  orig = get_concept_definition (CHECK_CONSTR_CONCEPT (orig));
-  orig = normalize_expression (orig);
-
-  location_t dloc = DECL_SOURCE_LOCATION (decl);
-  inform (dloc, "within %qS", sub);
-  diagnose_constraint (dloc, orig, cur, targs);
+  diagnose_valid_expression (TREE_OPERAND (req, 0), args, in_decl);
 }
 
-/* Diagnose a potentially unsatisfied conjunction or disjunction.  Parameters
-   are as for diagnose_constraint.  */
-
-void
-diagnose_logical_constraint (location_t loc, tree orig, tree cur, tree args)
+static void
+diagnose_compound_requirement (tree req, tree args, tree in_decl)
 {
-  tree t0 = TREE_OPERAND (cur, 0);
-  tree t1 = TREE_OPERAND (cur, 1);
-  if (!constraints_satisfied_p (t0, args))
-    diagnose_constraint (loc, TREE_OPERAND (orig, 0), t0, args);
-  else if (TREE_CODE (orig) == TRUTH_ORIF_EXPR)
+  tree expr = TREE_OPERAND (req, 0);
+  expr = diagnose_valid_expression (expr, args, in_decl);
+  if (expr == error_mark_node)
     return;
-  if (!constraints_satisfied_p (t1, args))
-    diagnose_constraint (loc, TREE_OPERAND (orig, 1), t1, args);
-}
 
-/* Diagnose a potential expression constraint failure. */
+  location_t loc = cp_expr_loc_or_input_loc (expr);
 
-void
-diagnose_expression_constraint (location_t loc, tree orig, tree cur, tree args)
-{
-  if (constraints_satisfied_p (cur, args))
-    return;
-  if (elide_constraint_failure_p())
+  /* Check the noexcept condition.  */
+  if (COMPOUND_REQ_NOEXCEPT_P (req) && !expr_noexcept_p (expr, tf_none))
+    inform (loc, "%qE is not %<noexcept%>", expr);
+
+  tree type = TREE_OPERAND (req, 1);
+  type = diagnose_valid_type (type, args, in_decl);
+  if (type == error_mark_node)
     return;
 
-  tree expr = EXPR_CONSTR_EXPR (orig);
-  inform (loc, "the required expression %qE would be ill-formed", expr);
+  if (type)
+    {
+      subst_info quiet (tf_none, in_decl);
+      subst_info noisy (tf_error, in_decl);
 
-  // TODO: We should have a flag that controls this substitution.
-  // I'm finding it very useful for resolving concept check errors.
+      /* Check the expression against the result type.  */
+      if (tree placeholder = type_uses_auto (type))
+       {
+         if (!type_deducible_p (expr, type, placeholder, args, quiet))
+           {
+             tree orig_expr = TREE_OPERAND (req, 0);
+             inform (loc, "type deduction from %qE failed", orig_expr);
+
+             /* Further explain the reason for the error.  */
+             type_deducible_p (expr, type, placeholder, args, noisy);
+           }
+       }
+      else if (!expression_convertible_p (expr, type, quiet))
+       {
+         tree orig_expr = TREE_OPERAND (req, 0);
+         inform (loc, "cannot convert %qE to %qT", orig_expr, type);
 
-  // inform (input_location, "==== BEGIN DUMP ====");
-  // tsubst_expr (EXPR_CONSTR_EXPR (orig), args, tf_warning_or_error, NULL_TREE, false);
-  // inform (input_location, "==== END DUMP ====");
+         /* Further explain the reason for the error.  */
+         expression_convertible_p (expr, type, noisy);
+       }
+    }
 }
 
-/* Diagnose a potentially failed type constraint. */
-
-void
-diagnose_type_constraint (location_t loc, tree orig, tree cur, tree args)
+static void
+diagnose_type_requirement (tree req, tree args, tree in_decl)
 {
-  if (constraints_satisfied_p (cur, args))
-    return;
-  if (elide_constraint_failure_p())
-    return;
-
-  tree type = TYPE_CONSTR_TYPE (orig);
-  inform (loc, "the required type %qT would be ill-formed", type);
+  tree type = TREE_OPERAND (req, 0);
+  diagnose_valid_type (type, args, in_decl);
 }
 
-/* Diagnose a potentially unsatisfied conversion constraint. */
-
-void
-diagnose_implicit_conversion_constraint (location_t loc, tree orig, tree cur,
-                                        tree args)
+static void
+diagnose_nested_requirement (tree req, tree args)
 {
-  if (constraints_satisfied_p (cur, args))
-    return;
-
-  /* The expression and type will previously have been substituted into,
-     and therefore may already be an error. Also, we will have already
-     diagnosed substitution failures into an expression since this must be
-     part of a compound requirement.  */
-  tree expr = ICONV_CONSTR_EXPR (cur);
-  if (error_operand_p (expr))
+  tree expr = TREE_OPERAND (req, 0);
+  if (constraints_satisfied_p (expr, args))
     return;
+  location_t loc = cp_expr_location (expr);
+  inform (loc, "nested requirement %qE is not satisfied", expr);
 
-  /* Don't elide a previously diagnosed failure.  */
-  if (elide_constraint_failure_p())
-    return;
-
-  tree type = ICONV_CONSTR_TYPE (cur);
-  if (error_operand_p (type))
-    {
-      inform (loc, "substitution into type %qT failed",
-             ICONV_CONSTR_TYPE (orig));
-      return;
-    }
-
-  inform(loc, "%qE is not implicitly convertible to %qT", expr, type);
+  /* TODO: Replay the substitution to diagnose the error?  */
+  // subst_info noisy (tf_warning_or_error, NULL_TREE);
+  // constraints_satisfied_p (expr, args, noisy);
 }
 
-/* Diagnose an argument deduction constraint. */
-
-void
-diagnose_argument_deduction_constraint (location_t loc, tree orig, tree cur,
-                                       tree args)
+static void
+diagnose_requirement (tree req, tree args, tree in_decl)
 {
-  if (constraints_satisfied_p (cur, args))
-    return;
-
-  /* The expression and type will previously have been substituted into,
-     and therefore may already be an error. Also, we will have already
-     diagnosed substution failures into an expression since this must be
-     part of a compound requirement.  */
-  tree expr = DEDUCT_CONSTR_EXPR (cur);
-  if (error_operand_p (expr))
-    return;
-
-  /* Don't elide a previously diagnosed failure.  */
-  if (elide_constraint_failure_p ())
-    return;
-
-  tree pattern = DEDUCT_CONSTR_PATTERN (cur);
-  if (error_operand_p (pattern))
+  iloc_sentinel loc_s (cp_expr_location (req));
+  switch (TREE_CODE (req))
     {
-      inform (loc, "substitution into type %qT failed",
-             DEDUCT_CONSTR_PATTERN (orig));
-      return;
+    case SIMPLE_REQ:
+      return diagnose_simple_requirement (req, args, in_decl);
+    case COMPOUND_REQ:
+      return diagnose_compound_requirement (req, args, in_decl);
+    case TYPE_REQ:
+      return diagnose_type_requirement (req, args, in_decl);
+    case NESTED_REQ:
+      return diagnose_nested_requirement (req, args);
+    default:
+       gcc_unreachable ();
     }
-
-  inform (loc, "unable to deduce placeholder type %qT from %qE",
-         pattern, expr);
 }
 
-/* Diagnose an exception constraint. */
-
-void
-diagnose_exception_constraint (location_t loc, tree orig, tree cur, tree args)
+static void
+diagnose_requires_expr (tree expr, tree args, tree in_decl)
 {
-  if (constraints_satisfied_p (cur, args))
-    return;
-  if (elide_constraint_failure_p ())
-    return;
+  local_specialization_stack stack (lss_copy);
+  tree parms = TREE_OPERAND (expr, 0);
+  tree body = TREE_OPERAND (expr, 1);
 
-  /* Rebuild a noexcept expression. */
-  tree expr = EXCEPT_CONSTR_EXPR (cur);
-  if (error_operand_p (expr))
+  cp_unevaluated u;
+  subst_info info (tf_warning_or_error, NULL_TREE);
+  tree vars = tsubst_constraint_variables (parms, args, info);
+  if (vars == error_mark_node)
     return;
 
-  inform (loc, "%qE evaluated to false", EXCEPT_CONSTR_EXPR (orig));
+  tree p = body;
+  while (p)
+    {
+      tree req = TREE_VALUE (p);
+      diagnose_requirement (req, args, in_decl);
+      p = TREE_CHAIN (p);
+    }
 }
 
-/* Diagnose a potentially unsatisfied parameterized constraint.  */
+/* Diagnose a substitution failure in the atomic constraint T. Note that
+   ARGS have been previously instantiated through the parameter map.  */
 
-void
-diagnose_parameterized_constraint (location_t loc, tree orig, tree cur,
-                                  tree args)
+static void
+diagnose_atomic_constraint (tree t, tree args, subst_info info)
 {
-  if (constraints_satisfied_p (cur, args))
-    return;
-
-  local_specialization_stack stack;
-  tree parms = PARM_CONSTR_PARMS (cur);
-  tree vars = tsubst_constraint_variables (parms, args, tf_warning_or_error,
-                                          NULL_TREE);
-  if (vars == error_mark_node)
+  /* If the constraint is already ill-formed, we've previously diagnosed
+     the reason. We should still say why the constraints aren't satisfied.  */
+  if (t == error_mark_node)
     {
-      if (elide_constraint_failure_p ())
-        return;
-
-      /* TODO: Check which variable failed and use orig to diagnose
-         that substitution error.  */
-      inform (loc, "failed to instantiate constraint variables");
+      location_t loc;
+      if (info.in_decl)
+        loc = DECL_SOURCE_LOCATION (info.in_decl);
+      else
+        loc = input_location;
+      inform (loc, "invalid constraints");
       return;
     }
 
-  /* TODO: It would be better write these in a list. */
-  while (vars)
-    {
-      inform (loc, "    with %q#D", vars);
-      vars = TREE_CHAIN (vars);
-    }
-  orig = PARM_CONSTR_OPERAND (orig);
-  cur = PARM_CONSTR_OPERAND (cur);
-  return diagnose_constraint (loc, orig, cur, args);
-}
-
-/* Diagnose the constraint CUR for the given ARGS. This is only ever invoked
-   on the associated constraints, so we can only have conjunctions of
-   predicate constraints.  The ORIGinal (dependent) constructs follow
-   the current constraints to enable better diagnostics.  Note that ORIG
-   and CUR must be the same kinds of node, except when CUR is an error.  */
+  location_t loc = get_constraint_error_location (t);
+  iloc_sentinel loc_s (loc);
 
-void
-diagnose_constraint (location_t loc, tree orig, tree cur, tree args)
-{
-  switch (TREE_CODE (cur))
+  /* Generate better diagnostics for certain kinds of expressions.  */
+  tree expr = ATOMIC_CONSTR_EXPR (t);
+  STRIP_ANY_LOCATION_WRAPPER (expr);
+  switch (TREE_CODE (expr))
     {
-    case EXPR_CONSTR:
-      diagnose_expression_constraint (loc, orig, cur, args);
-      break;
-
-    case TYPE_CONSTR:
-      diagnose_type_constraint (loc, orig, cur, args);
-      break;
-
-    case ICONV_CONSTR:
-      diagnose_implicit_conversion_constraint (loc, orig, cur, args);
-      break;
-
-    case DEDUCT_CONSTR:
-      diagnose_argument_deduction_constraint (loc, orig, cur, args);
-      break;
-
-    case EXCEPT_CONSTR:
-      diagnose_exception_constraint (loc, orig, cur, args);
-      break;
-
-    case CONJ_CONSTR:
-    case DISJ_CONSTR:
-      diagnose_logical_constraint (loc, orig, cur, args);
-      break;
-
-    case PRED_CONSTR:
-      diagnose_predicate_constraint (loc, orig, cur, args);
-      break;
-
-    case PARM_CONSTR:
-      diagnose_parameterized_constraint (loc, orig, cur, args);
-      break;
-
-    case CHECK_CONSTR:
-      diagnose_check_constraint (loc, orig, cur, args);
+    case TRAIT_EXPR:
+      diagnose_trait_expr (expr, args);
       break;
-
-    case EXPR_PACK_EXPANSION:
-      diagnose_pack_expansion (loc, orig, cur, args);
+    case REQUIRES_EXPR:
+      diagnose_requires_expr (expr, args, info.in_decl);
       break;
-
-    case ERROR_MARK:
-      /* TODO: Can we improve the diagnostic with the original?  */
-      inform (input_location, "ill-formed constraint");
+    case INTEGER_CST:
+      /* This must be either 0 or false.  */
+      inform (loc, "%qE is never satisfied", expr);
       break;
-
     default:
-      gcc_unreachable ();
-      break;
+      inform (loc, "the expression %qE evaluated to %<false%>", expr);
     }
 }
 
-/* Diagnose the reason(s) why ARGS do not satisfy the constraints
-   of declaration DECL. */
-
-void
-diagnose_declaration_constraints (location_t loc, tree decl, tree args)
+diagnosing_failed_constraint::
+diagnosing_failed_constraint (tree t, tree args, bool diag)
+  : diagnosing_error (diag)
 {
-  inform (loc, "  constraints not satisfied");
-
-  /* Constraints are attached to the template.  */
-  if (tree ti = DECL_TEMPLATE_INFO (decl))
-    {
-      decl = TI_TEMPLATE (ti);
-      if (!args)
-       args = TI_ARGS (ti);
-    }
-
-  /* Recursively diagnose the associated constraints.  */
-  tree ci = get_constraints (decl);
-  tree t = CI_ASSOCIATED_CONSTRAINTS (ci);
-  diagnose_constraint (loc, t, t, args);
+  if (diagnosing_error)
+    current_failed_constraint = tree_cons (args, t, current_failed_constraint);
 }
 
-} // namespace
+diagnosing_failed_constraint::
+~diagnosing_failed_constraint ()
+{
+  if (diagnosing_error && current_failed_constraint)
+    current_failed_constraint = TREE_CHAIN (current_failed_constraint);
+}
 
-/* Emit diagnostics detailing the failure ARGS to satisfy the
-   constraints of T. Here, T can be either a constraint
-   or a declaration.  */
+/* Emit diagnostics detailing the failure ARGS to satisfy the constraints
+   of T. Here, T can be either a constraint or a declaration.  */
 
 void
 diagnose_constraints (location_t loc, tree t, tree args)
 {
-  constraint_errors = 0;
+  inform (loc, "constraints not satisfied");
 
-  if (constraint_p (t))
-    diagnose_constraint (loc, t, t, args);
-  else if (DECL_P (t))
-    diagnose_declaration_constraints (loc, t, args);
+  /* Replay satisfaction, but diagnose errors.  */
+  subst_info info (tf_warning_or_error, NULL_TREE);
+  if (!args)
+    constraints_satisfied_p (t, info);
   else
-    gcc_unreachable ();
-
-  /* Note the number of elided failures. */
-  int n = undiagnosed_constraint_failures ();
-  if (n > 0)
-    inform (loc, "... and %d more constraint errors not shown", n);
+    constraints_satisfied_p (t, args, info);
 }
+
+#include "gt-cp-constraint.h"
index 0a72231456bef76549c84272db0ce404e73c472b..f38b4e8cfcbce32b3d1d244e00731042a30de912 100644 (file)
@@ -381,6 +381,7 @@ cp_common_init_ts (void)
   /* New decls.  */
   MARK_TS_DECL_COMMON (TEMPLATE_DECL);
   MARK_TS_DECL_COMMON (WILDCARD_DECL);
+  MARK_TS_DECL_COMMON (CONCEPT_DECL);
 
   MARK_TS_DECL_NON_COMMON (USING_DECL);
 
@@ -458,17 +459,11 @@ cp_common_init_ts (void)
   MARK_TS_EXP (CHECK_CONSTR);
   MARK_TS_EXP (COMPOUND_REQ);
   MARK_TS_EXP (CONJ_CONSTR);
-  MARK_TS_EXP (DEDUCT_CONSTR);
   MARK_TS_EXP (DISJ_CONSTR);
-  MARK_TS_EXP (EXCEPT_CONSTR);
-  MARK_TS_EXP (EXPR_CONSTR);
-  MARK_TS_EXP (ICONV_CONSTR);
+  MARK_TS_EXP (ATOMIC_CONSTR);
   MARK_TS_EXP (NESTED_REQ);
-  MARK_TS_EXP (PARM_CONSTR);
-  MARK_TS_EXP (PRED_CONSTR);
   MARK_TS_EXP (REQUIRES_EXPR);
   MARK_TS_EXP (SIMPLE_REQ);
-  MARK_TS_EXP (TYPE_CONSTR);
   MARK_TS_EXP (TYPE_REQ);
 
   c_common_init_ts ();
index 4db1d3161245a6ec0e74b12c89a8a4dd34e3e829..845a7e251f376c53860931bb073534ca84019ce8 100644 (file)
@@ -507,6 +507,11 @@ DEFTREECODE (OMP_DEPOBJ, "omp_depobj", tcc_statement, 2)
 
 /* Extensions for Concepts. */
 
+/* Concept definition. This is not entirely different than a VAR_DECL
+   except that a) it must be a template, and b) doesn't have the wide
+   range of value and linkage options available to variables.  */
+DEFTREECODE (CONCEPT_DECL, "concept_decl", tcc_declaration, 0)
+
 /* Used to represent information associated with constrained declarations. */
 DEFTREECODE (CONSTRAINT_INFO, "constraint_info", tcc_exceptional, 0)
 
@@ -541,10 +546,24 @@ DEFTREECODE (NESTED_REQ, "nested_req", tcc_expression, 1)
    The operands of a constraint can be either types or expressions.
    Unlike expressions, constraints do not have a type. */
 
-/* A predicate constraint evaluates an expression E.
+/* An atomic constraint evaluates an expression E. The operand of the
+   constraint is its parameter mapping. The actual expression is stored
+   in the context.
+
+   ATOMIC_CONSTR_INFO provides source info to support diagnostics.
+   ATOMIC_CONSTR_EXPR has the expression to be evaluated.
+   ATOMIC_CONSTR_PARMS is the parameter mapping for the atomic constraint
+   and is stored in the type field.  */
+DEFTREECODE (ATOMIC_CONSTR, "atomic_constr", tcc_expression, 1)
+
+/* The conjunction and disjunction of two constraints, respectively.
+   Operands are accessed using TREE_OPERAND. The third operand provides
+   source info for diagnostics.
 
-   PRED_CONSTR_EXPR has the expression to be evaluated. */
-DEFTREECODE (PRED_CONSTR, "pred_constr", tcc_expression, 1)
+   CONJ_CONSTR_INFO and DISJ_CONSTR_INFO provide access to the source
+   information of constraints, which is stored in the TREE_TYPE.  */
+DEFTREECODE (CONJ_CONSTR, "conj_constr", tcc_expression, 2)
+DEFTREECODE (DISJ_CONSTR, "disj_constr", tcc_expression, 2)
 
 /* A check constraint represents the checking of a concept
    C. It has two operands: the template defining the concept
@@ -554,53 +573,6 @@ DEFTREECODE (PRED_CONSTR, "pred_constr", tcc_expression, 1)
    CHECK_CONSTR_ARGUMENTS are the template arguments */
 DEFTREECODE (CHECK_CONSTR, "check_constr", tcc_expression, 2)
 
-/* An expression constraint determines the validity of a expression E.
-
-   EXPR_CONST_EXPR has the expression being validated. */
-DEFTREECODE (EXPR_CONSTR, "expr_constr", tcc_expression, 1)
-
-/* A type constraint determines the validity of a type T. Note that
-
-   TYPE_CONST_TYPE has the type being validated */
-DEFTREECODE (TYPE_CONSTR, "type_constr", tcc_expression, 1)
-
-/* An implicit conversion constraint determines if an expression
-   E is implicitly convertible to a type T. Note that T may
-   be dependent but does not contain any placeholders.
-
-   ICONV_CONSTR_EXPR has the expression E.
-   ICONV_CONSTR_TYPE has the type T.
-   */
-DEFTREECODE (ICONV_CONSTR, "iconv_constr", tcc_expression, 2)
-
-/* An argument deduction constraint determines if the type of an
-   expression E can be deduced from a type pattern T. Note that
-   T must contain at least one place holder.
-
-   DEDUCT_CONSTR_EXPR has the expression E
-   DEDUCT_CONSTR_PATTERN has the type pattern T.
-   DEDUCT_CONSTR_PLACEHOLDERS has the list of placeholder nodes in T. */
-DEFTREECODE (DEDUCT_CONSTR, "deduct_constr", tcc_expression, 3)
-
-/* An exception constraint determines if, for an expression E,
-   noexcept(E) is true.
-
-   EXCEPT_CONSTR_EXPR has the expression E. */
-DEFTREECODE (EXCEPT_CONSTR, "except_constr", tcc_expression, 1)
-
-/* A parameterized constraint declares constraint variables, which
-   are used in expression, type, and exception constraints.
-
-   PARM_CONSTR_PARMS has a TREE_LIST of parameter declarations.
-   PARM_CONSTR_OPERAND has the nested constraint. */
-DEFTREECODE (PARM_CONSTR, "parm_constr", tcc_expression, 2)
-
-/* The conjunction and disjunction of two constraints, respectively.
-   Operands are accessed using TREE_OPERAND. */
-DEFTREECODE (CONJ_CONSTR, "conj_constr", tcc_expression, 2)
-DEFTREECODE (DISJ_CONSTR, "disj_constr", tcc_expression, 2)
-
-
 /*
 Local variables:
 mode:c
index b82b5808197f55a990ddc3a3a8197b0c3b9440eb..9ff617be2d43554042ebaaef38069cddf36986ec 100644 (file)
@@ -1522,9 +1522,37 @@ check_constraint_info (tree t)
 #define PLACEHOLDER_TYPE_CONSTRAINTS(NODE) \
   DECL_SIZE_UNIT (TYPE_NAME (NODE))
 
-/* The expression evaluated by the predicate constraint. */
-#define PRED_CONSTR_EXPR(NODE) \
-  TREE_OPERAND (TREE_CHECK (NODE, PRED_CONSTR), 0)
+/* Valid for any normalized constraint.  */
+#define CONSTR_CHECK(NODE) \
+  TREE_CHECK3 (NODE, ATOMIC_CONSTR, CONJ_CONSTR, DISJ_CONSTR)
+
+/* The CONSTR_INFO stores normalization data for a constraint. It refers to
+   the original expression and the expression or declaration
+   from which the constraint was normalized.
+
+   This is TREE_LIST whose TREE_PURPOSE is the original expression and whose
+   TREE_VALUE is a list of contexts.  */
+#define CONSTR_INFO(NODE) \
+  TREE_TYPE (CONSTR_CHECK (NODE))
+
+/* The expression evaluated by the constraint.  */
+#define CONSTR_EXPR(NODE) \
+  TREE_PURPOSE (CONSTR_INFO (NODE))
+
+/* The expression or declaration from which this constraint was normalized.
+   This is a TREE_LIST whose TREE_VALUE is either a template-id expression
+   denoting a concept check or the declaration introducing the constraint.
+   These are chained to other context objects.  */
+#define CONSTR_CONTEXT(NODE) \
+  TREE_VALUE (CONSTR_INFO (NODE))
+
+/* The parameter mapping for an atomic constraint. */
+#define ATOMIC_CONSTR_MAP(NODE) \
+  TREE_OPERAND (TREE_CHECK (NODE, ATOMIC_CONSTR), 0)
+
+/* The expression of an atomic constraint. */
+#define ATOMIC_CONSTR_EXPR(NODE) \
+  CONSTR_EXPR (ATOMIC_CONSTR_CHECK (NODE))
 
 /* The concept of a concept check. */
 #define CHECK_CONSTR_CONCEPT(NODE) \
@@ -1534,46 +1562,6 @@ check_constraint_info (tree t)
 #define CHECK_CONSTR_ARGS(NODE) \
   TREE_OPERAND (TREE_CHECK (NODE, CHECK_CONSTR), 1)
 
-/* The expression validated by the predicate constraint. */
-#define EXPR_CONSTR_EXPR(NODE) \
-  TREE_OPERAND (TREE_CHECK (NODE, EXPR_CONSTR), 0)
-
-/* The type validated by the predicate constraint. */
-#define TYPE_CONSTR_TYPE(NODE) \
-  TREE_OPERAND (TREE_CHECK (NODE, TYPE_CONSTR), 0)
-
-/* In an implicit conversion constraint, the source expression. */
-#define ICONV_CONSTR_EXPR(NODE) \
-  TREE_OPERAND (TREE_CHECK (NODE, ICONV_CONSTR), 0)
-
-/* In an implicit conversion constraint, the target type. */
-#define ICONV_CONSTR_TYPE(NODE) \
-  TREE_OPERAND (TREE_CHECK (NODE, ICONV_CONSTR), 1)
-
-/* In an argument deduction constraint, the source expression. */
-#define DEDUCT_CONSTR_EXPR(NODE) \
-  TREE_OPERAND (TREE_CHECK (NODE, DEDUCT_CONSTR), 0)
-
-/* In an argument deduction constraint, the target type pattern. */
-#define DEDUCT_CONSTR_PATTERN(NODE) \
-  TREE_OPERAND (TREE_CHECK (NODE, DEDUCT_CONSTR), 1)
-
-/* In an argument deduction constraint, the list of placeholder nodes. */
-#define DEDUCT_CONSTR_PLACEHOLDER(NODE) \
-  TREE_OPERAND (TREE_CHECK (NODE, DEDUCT_CONSTR), 2)
-
-/* The expression of an exception constraint. */
-#define EXCEPT_CONSTR_EXPR(NODE) \
-  TREE_OPERAND (TREE_CHECK (NODE, EXCEPT_CONSTR), 0)
-
-/* In a parameterized constraint, the local parameters. */
-#define PARM_CONSTR_PARMS(NODE) \
-  TREE_OPERAND (TREE_CHECK (NODE, PARM_CONSTR), 0)
-
-/* In a parameterized constraint, the operand. */
-#define PARM_CONSTR_OPERAND(NODE) \
-  TREE_OPERAND (TREE_CHECK (NODE, PARM_CONSTR), 1)
-
 /* Whether a PARM_DECL represents a local parameter in a
    requires-expression.  */
 #define CONSTRAINT_VAR_P(NODE) \
@@ -1663,6 +1651,7 @@ struct GTY(()) saved_scope {
 
   int x_processing_template_decl;
   int x_processing_specialization;
+  int x_processing_constraint;
   int suppress_location_wrappers;
   BOOL_BITFIELD x_processing_explicit_instantiation : 1;
   BOOL_BITFIELD need_pop_function_context : 1;
@@ -2627,7 +2616,8 @@ struct GTY(()) lang_decl_base {
    || TREE_CODE (NODE) == CONST_DECL           \
    || TREE_CODE (NODE) == TYPE_DECL            \
    || TREE_CODE (NODE) == TEMPLATE_DECL                \
-   || TREE_CODE (NODE) == USING_DECL)
+   || TREE_CODE (NODE) == USING_DECL            \
+   || TREE_CODE (NODE) == CONCEPT_DECL)
 
 /* DECL_LANG_SPECIFIC for the above codes.  */
 
@@ -3358,6 +3348,33 @@ struct GTY(()) lang_decl {
 #define TEMPLATE_DECL_COMPLEX_ALIAS_P(NODE) \
   DECL_LANG_FLAG_2 (TEMPLATE_DECL_CHECK (NODE))
 
+/* Returns t iff the node can have a TEMPLATE_INFO field.  */
+
+inline tree
+template_info_decl_check (const_tree t, const char* f, int l, const char* fn)
+{
+  switch (TREE_CODE (t))
+    {
+    case VAR_DECL:
+    case FUNCTION_DECL:
+    case FIELD_DECL:
+    case TYPE_DECL:
+    case CONCEPT_DECL:
+    case TEMPLATE_DECL:
+      return const_cast<tree>(t);
+    default:
+      break;
+    }
+  tree_check_failed (t, f, l, fn,
+                     VAR_DECL, FUNCTION_DECL, FIELD_DECL, TYPE_DECL,
+                     CONCEPT_DECL, TEMPLATE_DECL, 0);
+  gcc_unreachable ();
+}
+
+
+#define TEMPLATE_INFO_DECL_CHECK(NODE) \
+  template_info_decl_check ((NODE), __FILE__, __LINE__, __FUNCTION__)
+
 /* Nonzero for a type which is an alias for another type; i.e, a type
    which declaration was written 'using name-of-type =
    another-type'.  */
@@ -3367,8 +3384,8 @@ struct GTY(()) lang_decl {
    && TREE_CODE (TYPE_NAME (NODE)) == TYPE_DECL        \
    && TYPE_DECL_ALIAS_P (TYPE_NAME (NODE)))
 
-/* If non-NULL for a VAR_DECL, FUNCTION_DECL, TYPE_DECL or
-   TEMPLATE_DECL, the entity is either a template specialization (if
+/* If non-NULL for a VAR_DECL, FUNCTION_DECL, TYPE_DECL, TEMPLATE_DECL,
+   or CONCEPT_DECL, the entity is either a template specialization (if
    DECL_USE_TEMPLATE is nonzero) or the abstract instance of the
    template itself.
 
@@ -3387,7 +3404,7 @@ struct GTY(()) lang_decl {
    global function f.  In this case, DECL_TEMPLATE_INFO for S<int>::f
    will be non-NULL, but DECL_USE_TEMPLATE will be zero.  */
 #define DECL_TEMPLATE_INFO(NODE) \
-  (DECL_LANG_SPECIFIC (VAR_TEMPL_TYPE_FIELD_OR_FUNCTION_DECL_CHECK (NODE)) \
+  (DECL_LANG_SPECIFIC (TEMPLATE_INFO_DECL_CHECK (NODE)) \
    ->u.min.template_info)
 
 /* For a lambda capture proxy, its captured variable.  */
@@ -5222,6 +5239,8 @@ enum tsubst_flags {
                                declaration.  */
   tf_no_cleanup = 1 << 10,   /* Do not build a cleanup
                                (build_target_expr and friends) */
+  tf_norm = 1 << 11,            /* Build diagnostic information during
+                                   constraint normalization.  */
   /* Convenient substitution flags combinations.  */
   tf_warning_or_error = tf_warning | tf_error
 };
@@ -6121,43 +6140,6 @@ class_of_this_parm (const_tree fntype)
   return TREE_TYPE (type_of_this_parm (fntype));
 }
 
-/* True iff T is a variable template declaration. */
-inline bool
-variable_template_p (tree t)
-{
-  if (TREE_CODE (t) != TEMPLATE_DECL)
-    return false;
-  if (!PRIMARY_TEMPLATE_P (t))
-    return false;
-  if (tree r = DECL_TEMPLATE_RESULT (t))
-    return VAR_P (r);
-  return false;
-}
-
-/* True iff T is a variable concept definition. That is, T is
-   a variable template declared with the concept specifier. */
-inline bool
-variable_concept_p (tree t)
-{
-  if (TREE_CODE (t) != TEMPLATE_DECL)
-    return false;
-  if (tree r = DECL_TEMPLATE_RESULT (t))
-    return VAR_P (r) && DECL_DECLARED_CONCEPT_P (r);
-  return false;
-}
-
-/* True iff T is a concept definition. That is, T is a variable or function
-   template declared with the concept specifier. */
-inline bool
-concept_template_p (tree t)
-{
-  if (TREE_CODE (t) != TEMPLATE_DECL)
-    return false;
-  if (tree r = DECL_TEMPLATE_RESULT (t))
-    return VAR_OR_FUNCTION_DECL_P (r) && DECL_DECLARED_CONCEPT_P (r);
-  return false;
-}
-
 /* A parameter list indicating for a function with no parameters,
    e.g  "int f(void)".  */
 extern cp_parameter_declarator *no_parameters;
@@ -6614,6 +6596,9 @@ extern void finish_eh_spec_block          (tree, tree);
 extern tree build_eh_type_type                 (tree);
 extern tree cp_protect_cleanup_actions         (void);
 extern tree create_try_catch_expr               (tree, tree);
+extern tree template_parms_to_args             (tree);
+extern tree template_parms_level_to_args       (tree);
+extern tree generic_targs_for                  (tree);
 
 /* in expr.c */
 extern tree cplus_expand_constant              (tree);
@@ -6747,6 +6732,8 @@ extern void maybe_show_extern_c_location (void);
 extern bool literal_integer_zerop (const_tree);
 
 /* in pt.c */
+extern void push_access_scope                  (tree);
+extern void pop_access_scope                   (tree);
 extern bool check_template_shadow              (tree);
 extern bool check_auto_in_tmpl_args             (tree, tree);
 extern tree get_innermost_template_args                (tree, int);
@@ -6766,6 +6753,8 @@ extern int num_template_headers_for_class (tree);
 extern void check_template_variable            (tree);
 extern tree make_auto                          (void);
 extern tree make_decltype_auto                 (void);
+extern tree make_constrained_auto              (tree, tree);
+extern tree make_constrained_decltype_auto     (tree, tree);
 extern tree make_template_placeholder          (tree);
 extern bool template_placeholder_p             (tree);
 extern tree do_auto_deduction                   (tree, tree, tree,
@@ -6792,6 +6781,7 @@ extern bool check_default_tmpl_args             (tree, tree, bool, bool, int);
 extern tree push_template_decl                 (tree);
 extern tree push_template_decl_real            (tree, bool);
 extern tree add_inherited_template_parms       (tree, tree);
+extern void template_parm_level_and_index      (tree, int*, int*);
 extern bool redeclare_class_template           (tree, tree, tree);
 extern tree lookup_template_class              (tree, tree, tree, tree,
                                                 int, tsubst_flags_t);
@@ -6816,6 +6806,7 @@ extern bool always_instantiate_p          (tree);
 extern bool maybe_instantiate_noexcept         (tree, tsubst_flags_t = tf_warning_or_error);
 extern tree instantiate_decl                   (tree, bool, bool);
 extern int comp_template_parms                 (const_tree, const_tree);
+extern bool template_heads_equivalent_p                (const_tree, const_tree);
 extern bool builtin_pack_fn_p                  (tree);
 extern tree uses_parameter_packs                (tree);
 extern bool template_parameter_pack_p           (const_tree);
@@ -6844,7 +6835,11 @@ extern tree tsubst_copy_and_build                (tree, tree, tsubst_flags_t,
                                                 tree, bool, bool);
 extern tree tsubst_expr                         (tree, tree, tsubst_flags_t,
                                                  tree, bool);
-extern tree tsubst_pack_expansion               (tree, tree, tsubst_flags_t, tree);
+extern tree tsubst_pack_expansion              (tree, tree, tsubst_flags_t, tree);
+extern tree tsubst_argument_pack               (tree, tree, tsubst_flags_t, tree);
+extern tree tsubst_template_args               (tree, tree, tsubst_flags_t, tree);
+extern tree tsubst_template_arg                        (tree, tree, tsubst_flags_t, tree);
+extern tree tsubst_function_parms              (tree, tree, tsubst_flags_t, tree);
 extern tree most_general_template              (tree);
 extern tree get_mostly_instantiated_function_type (tree);
 extern bool problematic_instantiation_changed  (void);
@@ -6913,6 +6908,7 @@ extern bool deduction_guide_p                     (const_tree);
 extern bool copy_guide_p                       (const_tree);
 extern bool template_guide_p                   (const_tree);
 extern void store_explicit_specifier           (tree, tree);
+extern tree add_outermost_template_args                (tree, tree);
 
 /* in rtti.c */
 /* A vector of all tinfo decls that haven't been emitted yet.  */
@@ -7077,7 +7073,9 @@ extern tree finish_asm_stmt                       (location_t, int, tree, tree,
 extern tree finish_label_stmt                  (tree);
 extern void finish_label_decl                  (tree);
 extern cp_expr finish_parenthesized_expr       (cp_expr);
-extern tree force_paren_expr                   (tree);
+extern tree force_paren_expr                   (tree, bool = false);
+inline tree force_paren_expr_uneval            (tree t)
+{ return force_paren_expr (t, true); }
 extern tree maybe_undo_parenthesized_ref       (tree);
 extern tree maybe_strip_ref_conversion         (tree);
 extern tree finish_non_static_data_member       (tree, tree, tree);
@@ -7661,69 +7659,97 @@ typedef void cp_binding_oracle_function (enum cp_oracle_request, tree identifier
 
 extern cp_binding_oracle_function *cp_binding_oracle;
 
+/* Set during diagnostics to record the failed constraint. This is a
+   TREE_LIST whose VALUE is the constraint and whose PURPOSE are the
+   instantiation arguments Defined in pt.c.  */
+
+extern tree current_failed_constraint;
+
+/* An RAII class to manage the failed constraint.  */
+
+struct diagnosing_failed_constraint
+{
+  diagnosing_failed_constraint (tree, tree, bool);
+  ~diagnosing_failed_constraint ();
+
+  bool diagnosing_error;
+};
+
 /* in constraint.cc */
-extern void init_constraint_processing          ();
-extern bool constraint_p                        (tree);
-extern tree conjoin_constraints                 (tree, tree);
-extern tree conjoin_constraints                 (tree);
+
+extern void init_constraint_processing         ();
+extern cp_expr finish_constraint_or_expr       (location_t, cp_expr, cp_expr);
+extern cp_expr finish_constraint_and_expr      (location_t, cp_expr, cp_expr);
+extern cp_expr finish_constraint_primary_expr  (cp_expr);
+extern tree finish_concept_definition          (cp_expr, tree);
+extern tree combine_constraint_expressions      (tree, tree);
 extern tree get_constraints                     (tree);
 extern void set_constraints                     (tree, tree);
 extern void remove_constraints                  (tree);
 extern tree current_template_constraints       (void);
 extern tree associate_classtype_constraints     (tree);
 extern tree build_constraints                   (tree, tree);
+extern tree get_template_head_requirements     (tree);
+extern tree get_trailing_function_requirements (tree);
 extern tree get_shorthand_constraints           (tree);
-extern tree build_concept_check                 (tree, tree, tree = NULL_TREE);
+
+extern tree build_concept_id                   (tree);
+extern tree build_type_constraint              (tree, tree, tsubst_flags_t);
+extern tree build_concept_check                 (tree, tree, tsubst_flags_t);
+extern tree build_concept_check                 (tree, tree, tree, tsubst_flags_t);
+
+extern tree_pair finish_type_constraints       (tree, tree, tsubst_flags_t);
 extern tree build_constrained_parameter         (tree, tree, tree = NULL_TREE);
-extern tree make_constrained_auto               (tree, tree);
 extern void placeholder_extract_concept_and_args (tree, tree&, tree&);
 extern bool equivalent_placeholder_constraints  (tree, tree);
 extern hashval_t hash_placeholder_constraint   (tree);
 extern bool deduce_constrained_parameter        (tree, tree&, tree&);
 extern tree resolve_constraint_check            (tree);
 extern tree check_function_concept              (tree);
-extern tree finish_template_introduction        (tree, tree);
+extern tree finish_template_introduction        (tree, tree, location_t loc);
 extern bool valid_requirements_p                (tree);
 extern tree finish_concept_name                 (tree);
 extern tree finish_shorthand_constraint         (tree, tree);
-extern tree finish_requires_expr                (tree, tree);
-extern tree finish_simple_requirement           (tree);
-extern tree finish_type_requirement             (tree);
-extern tree finish_compound_requirement         (tree, tree, bool);
-extern tree finish_nested_requirement           (tree);
+extern tree finish_requires_expr                (location_t, tree, tree);
+extern tree finish_simple_requirement           (location_t, tree);
+extern tree finish_type_requirement             (location_t, tree);
+extern tree finish_compound_requirement         (location_t, tree, tree, bool);
+extern tree finish_nested_requirement           (location_t, tree);
 extern void check_constrained_friend            (tree, tree);
 extern tree tsubst_requires_expr                (tree, tree, tsubst_flags_t, tree);
 extern tree tsubst_constraint                   (tree, tree, tsubst_flags_t, tree);
 extern tree tsubst_constraint_info              (tree, tree, tsubst_flags_t, tree);
-extern bool function_concept_check_p            (tree);
-extern tree normalize_expression                (tree);
-extern tree expand_concept                      (tree, tree);
-extern bool expanding_concept                   ();
-extern tree evaluate_constraints                (tree, tree);
-extern tree evaluate_function_concept           (tree, tree);
-extern tree evaluate_variable_concept           (tree, tree);
-extern tree evaluate_constraint_expression      (tree, tree);
+extern tree tsubst_parameter_mapping           (tree, tree, tsubst_flags_t, tree);
+extern tree get_mapped_args                    (tree);
+
+struct processing_constraint_expression_sentinel
+{
+  processing_constraint_expression_sentinel ();
+  ~processing_constraint_expression_sentinel ();
+};
+
+extern bool processing_constraint_expression_p ();
+
+extern tree unpack_concept_check               (tree);
+extern tree evaluate_concept_check              (tree, tsubst_flags_t);
+extern tree satisfy_constraint_expression      (tree);
 extern bool constraints_satisfied_p             (tree);
 extern bool constraints_satisfied_p             (tree, tree);
-extern tree lookup_constraint_satisfaction      (tree, tree);
-extern tree memoize_constraint_satisfaction     (tree, tree, tree);
-extern tree lookup_concept_satisfaction         (tree, tree);
-extern tree memoize_concept_satisfaction        (tree, tree, tree);
-extern tree get_concept_expansion               (tree, tree);
-extern tree save_concept_expansion              (tree, tree, tree);
+extern void clear_satisfaction_cache           ();
 extern bool* lookup_subsumption_result          (tree, tree);
 extern bool save_subsumption_result             (tree, tree, bool);
-
+extern tree find_template_parameters           (tree, int);
 extern bool equivalent_constraints              (tree, tree);
 extern bool equivalently_constrained            (tree, tree);
 extern bool subsumes_constraints                (tree, tree);
-extern bool strictly_subsumes                  (tree, tree);
+extern bool strictly_subsumes                  (tree, tree, tree);
+extern bool weakly_subsumes                    (tree, tree, tree);
 extern int more_constrained                     (tree, tree);
-
+extern bool atomic_constraints_identical_p     (tree, tree);
+extern hashval_t hash_atomic_constraint                (tree);
 extern void diagnose_constraints                (location_t, tree, tree);
 
 /* in logic.cc */
-extern tree decompose_conclusions               (tree);
 extern bool subsumes                            (tree, tree);
 
 /* In class.c */
@@ -7773,7 +7799,7 @@ extern bool var_in_maybe_constexpr_fn           (tree);
 extern void explain_invalid_constexpr_fn        (tree);
 extern vec<tree> cx_error_context               (void);
 extern tree fold_sizeof_expr                   (tree);
-extern void clear_cv_and_fold_caches           (void);
+extern void clear_cv_and_fold_caches           (bool = true);
 extern tree unshare_constructor                        (tree CXX_MEM_STAT_INFO);
 
 /* In cp-ubsan.c */
@@ -7820,6 +7846,104 @@ null_node_p (const_tree expr)
   return expr == null_node;
 }
 
+/* True iff T is a variable template declaration. */
+inline bool
+variable_template_p (tree t)
+{
+  if (TREE_CODE (t) != TEMPLATE_DECL)
+    return false;
+  if (!PRIMARY_TEMPLATE_P (t))
+    return false;
+  if (tree r = DECL_TEMPLATE_RESULT (t))
+    return VAR_P (r);
+  return false;
+}
+
+/* True iff T is a standard concept definition. This will return
+   true for both the template and underlying declaration.  */
+
+inline bool
+standard_concept_p (tree t)
+{
+  if (TREE_CODE (t) == TEMPLATE_DECL)
+    t = DECL_TEMPLATE_RESULT (t);
+  return TREE_CODE (t) == CONCEPT_DECL;
+}
+
+/* True iff T is a variable concept definition. This will return
+   true for both the template and the underlying declaration.  */
+
+inline bool
+variable_concept_p (tree t)
+{
+  if (TREE_CODE (t) == TEMPLATE_DECL)
+    t = DECL_TEMPLATE_RESULT (t);
+  return VAR_P (t) && DECL_DECLARED_CONCEPT_P (t);
+}
+
+/* True iff T is a function concept definition or an overload set
+   containing multiple function concepts. This will return true for
+   both the template and the underlying declaration.  */
+
+inline bool
+function_concept_p (tree t)
+{
+  if (TREE_CODE (t) == OVERLOAD)
+    t = OVL_FIRST (t);
+  if (TREE_CODE (t) == TEMPLATE_DECL)
+    t = DECL_TEMPLATE_RESULT (t);
+  return TREE_CODE (t) == FUNCTION_DECL && DECL_DECLARED_CONCEPT_P (t);
+}
+
+/* True iff T is a standard, variable, or function concept.  */
+
+inline bool
+concept_definition_p (tree t)
+{
+  if (t == error_mark_node)
+    return false;
+
+  /* Adjust for function concept overloads.  */
+  if (TREE_CODE (t) == OVERLOAD)
+    t = OVL_FIRST (t);
+
+  /* See through templates. */
+  if (TREE_CODE (t) == TEMPLATE_DECL)
+    t = DECL_TEMPLATE_RESULT (t);
+
+  /* The obvious and easy case.  */
+  if (TREE_CODE (t) == CONCEPT_DECL)
+    return true;
+
+  /* Definitely not a concept.  */
+  if (!VAR_OR_FUNCTION_DECL_P (t))
+    return false;
+  if (!DECL_LANG_SPECIFIC (t))
+    return false;
+
+  return DECL_DECLARED_CONCEPT_P (t);
+}
+
+/* Same as above, but for const trees.  */
+
+inline bool
+concept_definition_p (const_tree t)
+{
+  return concept_definition_p (const_cast<tree> (t));
+}
+
+/* True if t is an expression that checks a concept.  */
+
+inline bool
+concept_check_p (const_tree t)
+{
+  if (TREE_CODE (t) == CALL_EXPR)
+    t = CALL_EXPR_FN (t);
+  if (t && TREE_CODE (t) == TEMPLATE_ID_EXPR)
+    return concept_definition_p (TREE_OPERAND (t, 0));
+  return false;
+}
+
 #if CHECKING_P
 namespace selftest {
   extern void run_cp_tests (void);
index 038f300903976e0ea7edba0cfdb140cf362f1b33..2a129a3bff7ccba034146b9cd4ac0122f30e9e94 100644 (file)
@@ -38,6 +38,7 @@ static void pp_cxx_typeid_expression (cxx_pretty_printer *, tree);
 static void pp_cxx_unary_left_fold_expression (cxx_pretty_printer *, tree);
 static void pp_cxx_unary_right_fold_expression (cxx_pretty_printer *, tree);
 static void pp_cxx_binary_fold_expression (cxx_pretty_printer *, tree);
+static void pp_cxx_concept_definition (cxx_pretty_printer *, tree);
 \f
 
 static inline void
@@ -237,7 +238,14 @@ pp_cxx_template_keyword_if_needed (cxx_pretty_printer *pp, tree scope, tree t)
 static void
 pp_cxx_nested_name_specifier (cxx_pretty_printer *pp, tree t)
 {
-  if (!SCOPE_FILE_SCOPE_P (t) && t != pp->enclosing_scope)
+  /* FIXME: When diagnosing references to concepts (especially as types?)
+     we end up adding too many '::' to the name. This is partially due
+     to the fact that pp->enclosing_namespace is null.  */
+  if (t == global_namespace)
+    {
+      pp_cxx_colon_colon (pp);
+    }
+  else if (!SCOPE_FILE_SCOPE_P (t) && t != pp->enclosing_scope)
     {
       tree scope = get_containing_scope (t);
       pp_cxx_nested_name_specifier (pp, scope);
@@ -1214,7 +1222,7 @@ cxx_pretty_printer::expression (tree t)
          }
       }
       break;
-      
+
     case LAMBDA_EXPR:
       pp_cxx_ws_string (this, "<lambda>");
       break;
@@ -1223,14 +1231,8 @@ cxx_pretty_printer::expression (tree t)
       pp_cxx_trait_expression (this, t);
       break;
 
-    case PRED_CONSTR:
+    case ATOMIC_CONSTR:
     case CHECK_CONSTR:
-    case EXPR_CONSTR:
-    case TYPE_CONSTR:
-    case ICONV_CONSTR:
-    case DEDUCT_CONSTR:
-    case EXCEPT_CONSTR:
-    case PARM_CONSTR:
     case CONJ_CONSTR:
     case DISJ_CONSTR:
       pp_cxx_constraint (this, t);
@@ -1349,6 +1351,8 @@ cxx_pretty_printer::simple_type_specifier (tree t)
     case TEMPLATE_PARM_INDEX:
     case BOUND_TEMPLATE_TEMPLATE_PARM:
       pp_cxx_unqualified_id (this, t);
+      if (tree c = PLACEHOLDER_TYPE_CONSTRAINTS (t))
+        pp_cxx_constrained_type_spec (this, c);
       break;
 
     case TYPENAME_TYPE:
@@ -1876,7 +1880,7 @@ pp_cxx_template_argument_list (cxx_pretty_printer *pp, tree t)
        {
          if (argpack)
            arg = TREE_VEC_ELT (argpack, idx);
-         
+
          if (need_comma)
            pp_cxx_separate_with (pp, ',');
          else
@@ -2302,24 +2306,28 @@ pp_cxx_canonical_template_parameter (cxx_pretty_printer *pp, tree parm)
 void
 pp_cxx_constrained_type_spec (cxx_pretty_printer *pp, tree c)
 {
-  tree t, a;
+  pp_cxx_whitespace (pp);
+  pp_cxx_left_bracket (pp);
+  pp->translate_string ("requires");
+  pp_cxx_whitespace (pp);
   if (c == error_mark_node)
     {
-      pp_cxx_ws_string(pp, "<unsatisfied-constrained-placeholder>");
+      pp_cxx_ws_string(pp, "<unsatisfied-type-constraint>");
       return;
     }
+  tree t, a;
   placeholder_extract_concept_and_args (c, t, a);
   pp->id_expression (t);
-  if (TREE_VEC_LENGTH (a) > 1)
-    {
-      pp_cxx_begin_template_argument_list (pp);
-      tree args = make_tree_vec (TREE_VEC_LENGTH (a) - 1);
-      for (int i = TREE_VEC_LENGTH (a) - 1; i > 0; --i)
-       TREE_VEC_ELT (args, i-1) = TREE_VEC_ELT (a, i);
-      pp_cxx_template_argument_list (pp, args);
-      ggc_free (args);
-      pp_cxx_end_template_argument_list (pp);
-    }
+  pp_cxx_begin_template_argument_list (pp);
+  pp_cxx_ws_string (pp, "<placeholder>");
+  pp_cxx_separate_with (pp, ',');
+  tree args = make_tree_vec (TREE_VEC_LENGTH (a) - 1);
+  for (int i = 0; i < TREE_VEC_LENGTH (a) - 1; ++i)
+    TREE_VEC_ELT (args, i) = TREE_VEC_ELT (a, i + 1);
+  pp_cxx_template_argument_list (pp, args);
+  ggc_free (args);
+  pp_cxx_end_template_argument_list (pp);
+  pp_cxx_right_bracket (pp);
 }
 
 /*
@@ -2358,6 +2366,8 @@ pp_cxx_template_declaration (cxx_pretty_printer *pp, tree t)
 
   if (TREE_CODE (t) == FUNCTION_DECL && DECL_SAVED_TREE (t))
     pp_cxx_function_definition (pp, t);
+  else if (TREE_CODE (t) == CONCEPT_DECL)
+    pp_cxx_concept_definition (pp, t);
   else
     pp_cxx_simple_declaration (pp, t);
 }
@@ -2374,6 +2384,17 @@ pp_cxx_explicit_instantiation (cxx_pretty_printer *pp, tree t)
   pp_unsupported_tree (pp, t);
 }
 
+static void
+pp_cxx_concept_definition (cxx_pretty_printer *pp, tree t)
+{
+  pp_cxx_unqualified_id (pp, DECL_NAME (t));
+  pp_cxx_whitespace (pp);
+  pp_cxx_ws_string (pp, "=");
+  pp_cxx_whitespace (pp);
+  pp->expression (DECL_INITIAL (t));
+  pp_cxx_semicolon (pp);
+}
+
 /*
     declaration:
        block-declaration
@@ -2841,6 +2862,7 @@ pp_cxx_compound_requirement (cxx_pretty_printer *pp, tree t)
 
   if (tree type = TREE_OPERAND (t, 1))
     {
+      pp_cxx_whitespace (pp);
       pp_cxx_ws_string (pp, "->");
       pp->type_id (type);
     }
@@ -2857,12 +2879,6 @@ pp_cxx_nested_requirement (cxx_pretty_printer *pp, tree t)
   pp_cxx_semicolon (pp);
 }
 
-void
-pp_cxx_predicate_constraint (cxx_pretty_printer *pp, tree t)
-{
-  pp->expression (TREE_OPERAND (t, 0));
-}
-
 void
 pp_cxx_check_constraint (cxx_pretty_printer *pp, tree t)
 {
@@ -2871,7 +2887,9 @@ pp_cxx_check_constraint (cxx_pretty_printer *pp, tree t)
   tree args = CHECK_CONSTR_ARGS (t);
   tree id = build_nt (TEMPLATE_ID_EXPR, tmpl, args);
 
-  if (VAR_P (decl))
+  if (TREE_CODE (decl) == CONCEPT_DECL)
+    pp->expression (id);
+  else if (VAR_P (decl))
     pp->expression (id);
   else if (TREE_CODE (decl) == FUNCTION_DECL)
     {
@@ -2884,77 +2902,60 @@ pp_cxx_check_constraint (cxx_pretty_printer *pp, tree t)
     gcc_unreachable ();
 }
 
-void
-pp_cxx_expression_constraint (cxx_pretty_printer *pp, tree t)
-{
-  pp_string (pp, "<valid-expression ");
-  pp_cxx_left_paren (pp);
-  pp->expression (TREE_OPERAND (t, 0));
-  pp_cxx_right_paren (pp);
-  pp_string (pp, ">");
-}
+/* Output the "[with ...]" clause for a parameter mapping of an atomic
+   constraint.   */
 
-void
-pp_cxx_type_constraint (cxx_pretty_printer *pp, tree t)
+static void
+pp_cxx_parameter_mapping (cxx_pretty_printer *pp, tree map)
 {
-  pp_string (pp, "<valid-type ");
-  pp->type_id (TREE_OPERAND (t, 0));
-  pp_string (pp, ">");
-}
+  for (tree p = map; p; p = TREE_CHAIN (p))
+    {
+      tree parm = TREE_VALUE (p);
+      tree arg = TREE_PURPOSE (p);
 
-void
-pp_cxx_implicit_conversion_constraint (cxx_pretty_printer *pp, tree t)
-{
-  pp_string (pp, "<implicitly-conversion ");
-  pp_cxx_left_paren (pp);
-  pp->expression (ICONV_CONSTR_EXPR (t));
-  pp_cxx_right_paren (pp);
-  pp_cxx_ws_string (pp, "to");
-  pp->type_id (ICONV_CONSTR_TYPE (t));
-  pp_string (pp, ">");
-}
+      if (TYPE_P (parm))
+       pp->type_id (parm);
+      else
+       pp_cxx_tree_identifier (pp, DECL_NAME (TEMPLATE_PARM_DECL (parm)));
 
-void
-pp_cxx_argument_deduction_constraint (cxx_pretty_printer *pp, tree t)
-{
-  pp_string (pp, "<argument-deduction ");
-  pp_cxx_left_paren (pp);
-  pp->expression (DEDUCT_CONSTR_EXPR (t));
-  pp_cxx_right_paren (pp);
-  pp_cxx_ws_string (pp, "as");
-  pp->expression (DEDUCT_CONSTR_PATTERN (t));
-  pp_string (pp, ">");
-}
+      pp_cxx_whitespace (pp);
+      pp_equal (pp);
+      pp_cxx_whitespace (pp);
 
-void
-pp_cxx_exception_constraint (cxx_pretty_printer *pp, tree t)
-{
-  pp_cxx_ws_string (pp, "noexcept");
-  pp_cxx_whitespace (pp);
-  pp_cxx_left_paren (pp);
-  pp->expression (TREE_OPERAND (t, 0));
-  pp_cxx_right_paren (pp);
+      if (TYPE_P (arg) || DECL_TEMPLATE_TEMPLATE_PARM_P (arg))
+       pp->type_id (arg);
+      else
+       pp->expression (arg);
+
+      if (TREE_CHAIN (p) != NULL_TREE)
+       pp_cxx_separate_with (pp, ';');
+    }
 }
 
 void
-pp_cxx_parameterized_constraint (cxx_pretty_printer *pp, tree t)
+pp_cxx_atomic_constraint (cxx_pretty_printer *pp, tree t)
 {
-  pp_left_paren (pp);
-  pp_string (pp, "<requires ");
-  if (tree parms = PARM_CONSTR_PARMS (t))
+  /* Emit the expression.  */
+  pp->expression (ATOMIC_CONSTR_EXPR (t));
+
+  /* Emit the parameter mapping.  */
+  tree map = ATOMIC_CONSTR_MAP (t);
+  if (map && map != error_mark_node)
     {
-       pp_cxx_parameter_declaration_clause (pp, parms);
       pp_cxx_whitespace (pp);
-    }
-  pp_cxx_constraint (pp, PARM_CONSTR_OPERAND (t));
-  pp_string (pp, ">");
+      pp_cxx_left_bracket (pp);
+      pp->translate_string ("with");
+      pp_cxx_whitespace (pp);
+      pp_cxx_parameter_mapping (pp, map);
+      pp_cxx_right_bracket (pp);
+   }
 }
 
 void
 pp_cxx_conjunction (cxx_pretty_printer *pp, tree t)
 {
   pp_cxx_constraint (pp, TREE_OPERAND (t, 0));
-  pp_string (pp, " and ");
+  pp_string (pp, " /\\ ");
   pp_cxx_constraint (pp, TREE_OPERAND (t, 1));
 }
 
@@ -2962,7 +2963,7 @@ void
 pp_cxx_disjunction (cxx_pretty_printer *pp, tree t)
 {
   pp_cxx_constraint (pp, TREE_OPERAND (t, 0));
-  pp_string (pp, " or ");
+  pp_string (pp, " \\/ ");
   pp_cxx_constraint (pp, TREE_OPERAND (t, 1));
 }
 
@@ -2974,38 +2975,14 @@ pp_cxx_constraint (cxx_pretty_printer *pp, tree t)
 
   switch (TREE_CODE (t))
     {
-    case PRED_CONSTR:
-      pp_cxx_predicate_constraint (pp, t);
+    case ATOMIC_CONSTR:
+      pp_cxx_atomic_constraint (pp, t);
       break;
 
     case CHECK_CONSTR:
       pp_cxx_check_constraint (pp, t);
       break;
 
-    case EXPR_CONSTR:
-      pp_cxx_expression_constraint (pp, t);
-      break;
-
-    case TYPE_CONSTR:
-      pp_cxx_type_constraint (pp, t);
-      break;
-
-    case ICONV_CONSTR:
-      pp_cxx_implicit_conversion_constraint (pp, t);
-      break;
-
-    case DEDUCT_CONSTR:
-      pp_cxx_argument_deduction_constraint (pp, t);
-      break;
-
-    case EXCEPT_CONSTR:
-      pp_cxx_exception_constraint (pp, t);
-      break;
-
-    case PARM_CONSTR:
-      pp_cxx_parameterized_constraint (pp, t);
-      break;
-
     case CONJ_CONSTR:
       pp_cxx_conjunction (pp, t);
       break;
index 72c02afa1058cf3dea9e2f5ebc4d1ab514bf068b..6f4c589b4b14d672a71075fc15136fd5010968c7 100644 (file)
@@ -920,6 +920,31 @@ determine_local_discriminator (tree decl)
 }
 
 \f
+
+/* Returns true if functions FN1 and FN2 have equivalent trailing
+   requires clauses.  */
+
+static bool
+function_requirements_equivalent_p (tree newfn, tree oldfn)
+{
+  /* In the concepts TS, the combined constraints are compared.  */
+  if (cxx_dialect < cxx2a)
+    {
+      tree ci1 = get_constraints (oldfn);
+      tree ci2 = get_constraints (newfn);
+      tree req1 = ci1 ? CI_ASSOCIATED_CONSTRAINTS (ci1) : NULL_TREE;
+      tree req2 = ci2 ? CI_ASSOCIATED_CONSTRAINTS (ci2) : NULL_TREE;
+      return cp_tree_equal (req1, req2);
+    }
+
+  /* Compare only trailing requirements.  */
+  tree reqs1 = get_trailing_function_requirements (newfn);
+  tree reqs2 = get_trailing_function_requirements (oldfn);
+  if ((reqs1 != NULL_TREE) != (reqs2 != NULL_TREE))
+    return false;
+  return cp_tree_equal (reqs1, reqs2);
+}
+
 /* Subroutine of duplicate_decls: return truthvalue of whether
    or not types of these decls match.
 
@@ -999,6 +1024,12 @@ decls_match (tree newdecl, tree olddecl, bool record_versions /* = true */)
       else
        types_match = 0;
 
+      /* Two function declarations match if either has a requires-clause
+         then both have a requires-clause and their constraints-expressions
+         are equivalent.  */
+      if (types_match && flag_concepts)
+       types_match = function_requirements_equivalent_p (newdecl, olddecl);
+
       /* The decls dont match if they correspond to two different versions
         of the same function.   Disallow extern "C" functions to be
         versions for now.  */
@@ -1013,23 +1044,21 @@ decls_match (tree newdecl, tree olddecl, bool record_versions /* = true */)
     }
   else if (TREE_CODE (newdecl) == TEMPLATE_DECL)
     {
+      if (!template_heads_equivalent_p (newdecl, olddecl))
+       return 0;
+
       tree oldres = DECL_TEMPLATE_RESULT (olddecl);
       tree newres = DECL_TEMPLATE_RESULT (newdecl);
 
       if (TREE_CODE (newres) != TREE_CODE (oldres))
        return 0;
 
-      if (!comp_template_parms (DECL_TEMPLATE_PARMS (newdecl),
-                               DECL_TEMPLATE_PARMS (olddecl)))
-       return 0;
-
-      if (TREE_CODE (DECL_TEMPLATE_RESULT (newdecl)) == TYPE_DECL)
-       types_match = (same_type_p (TREE_TYPE (oldres), TREE_TYPE (newres))
-                      && equivalently_constrained (olddecl, newdecl));
+      /* Two template types match if they are the same. Otherwise, compare
+         the underlying declarations.  */
+      if (TREE_CODE (newres) == TYPE_DECL)
+        types_match = same_type_p (TREE_TYPE (newres), TREE_TYPE (oldres));
       else
-       // We don't need to check equivalently_constrained for variable and
-       // function templates because we check it on the results.
-       types_match = decls_match (oldres, newres);
+       types_match = decls_match (newres, oldres);
     }
   else
     {
@@ -1057,11 +1086,6 @@ decls_match (tree newdecl, tree olddecl, bool record_versions /* = true */)
                                 COMPARE_REDECLARATION);
     }
 
-  // Normal functions can be constrained, as can variable partial
-  // specializations.
-  if (types_match && VAR_OR_FUNCTION_DECL_P (newdecl))
-    types_match = equivalently_constrained (newdecl, olddecl);
-
   return types_match;
 }
 
@@ -1336,6 +1360,46 @@ merge_attribute_bits (tree newdecl, tree olddecl)
                          && lookup_attribute ("gnu_inline",            \
                                               DECL_ATTRIBUTES (fn)))
 
+/* A subroutine of duplicate_decls. Emits a diagnostic when newdecl
+   ambiguates olddecl.  Returns true if an error occurs.  */
+
+static bool
+duplicate_function_template_decls (tree newdecl, tree olddecl)
+{
+
+  tree newres = DECL_TEMPLATE_RESULT (newdecl);
+  tree oldres = DECL_TEMPLATE_RESULT (olddecl);
+  /* Function template declarations can be differentiated by parameter
+     and return type.  */
+  if (compparms (TYPE_ARG_TYPES (TREE_TYPE (oldres)),
+                TYPE_ARG_TYPES (TREE_TYPE (newres)))
+       && same_type_p (TREE_TYPE (TREE_TYPE (newdecl)),
+                      TREE_TYPE (TREE_TYPE (olddecl))))
+    {
+      /* ... and also by their template-heads and requires-clauses.  */
+      if (template_heads_equivalent_p (newdecl, olddecl)
+         && function_requirements_equivalent_p (newres, oldres))
+       {
+         error ("ambiguating new declaration %q+#D", newdecl);
+         inform (DECL_SOURCE_LOCATION (olddecl),
+                 "old declaration %q#D", olddecl);
+         return true;
+       }
+
+      /* FIXME: The types are the same but the are differences
+        in either the template heads or function requirements.
+        We should be able to diagnose a set of common errors
+        stemming from these declarations. For example:
+
+          template<typename T> requires C void f(...);
+          template<typename T> void f(...) requires C;
+
+        These are functionally equivalent but not equivalent.  */
+    }
+
+  return false;
+}
+
 /* If NEWDECL is a redeclaration of OLDDECL, merge the declarations.
    If the redeclaration is invalid, a diagnostic is issued, and the
    error_mark_node is returned.  Otherwise, OLDDECL is returned.
@@ -1644,11 +1708,14 @@ duplicate_decls (tree newdecl, tree olddecl, bool newdecl_is_friend)
 
       if (TREE_CODE (newdecl) == TEMPLATE_DECL)
        {
+         tree oldres = DECL_TEMPLATE_RESULT (olddecl);
+         tree newres = DECL_TEMPLATE_RESULT (newdecl);
+
          /* The name of a class template may not be declared to refer to
             any other template, class, function, object, namespace, value,
             or type in the same scope.  */
-         if (TREE_CODE (DECL_TEMPLATE_RESULT (olddecl)) == TYPE_DECL
-             || TREE_CODE (DECL_TEMPLATE_RESULT (newdecl)) == TYPE_DECL)
+         if (TREE_CODE (oldres) == TYPE_DECL
+             || TREE_CODE (newres) == TYPE_DECL)
            {
              error_at (newdecl_loc,
                        "conflicting declaration of template %q#D", newdecl);
@@ -1656,24 +1723,13 @@ duplicate_decls (tree newdecl, tree olddecl, bool newdecl_is_friend)
                      "previous declaration %q#D", olddecl);
              return error_mark_node;
            }
-         else if (TREE_CODE (DECL_TEMPLATE_RESULT (olddecl)) == FUNCTION_DECL
-                  && TREE_CODE (DECL_TEMPLATE_RESULT (newdecl)) == FUNCTION_DECL
-                  && compparms (TYPE_ARG_TYPES (TREE_TYPE (DECL_TEMPLATE_RESULT (olddecl))),
-                                TYPE_ARG_TYPES (TREE_TYPE (DECL_TEMPLATE_RESULT (newdecl))))
-                  && comp_template_parms (DECL_TEMPLATE_PARMS (newdecl),
-                                          DECL_TEMPLATE_PARMS (olddecl))
-                  /* Template functions can be disambiguated by
-                     return type.  */
-                  && same_type_p (TREE_TYPE (TREE_TYPE (newdecl)),
-                                  TREE_TYPE (TREE_TYPE (olddecl)))
-                   /* Template functions can also be disambiguated by
-                     constraints.  */
-                   && equivalently_constrained (olddecl, newdecl))
+
+         else if (TREE_CODE (oldres) == FUNCTION_DECL
+                  && TREE_CODE (newres) == FUNCTION_DECL)
            {
-             error_at (newdecl_loc, "ambiguating new declaration %q#D",
-                       newdecl);
-             inform (olddecl_loc,
-                     "old declaration %q#D", olddecl);
+             if (duplicate_function_template_decls (newdecl, olddecl))
+               return error_mark_node;
+             return NULL_TREE;
            }
           else if (check_concept_refinement (olddecl, newdecl))
            return error_mark_node;
@@ -2916,6 +2972,9 @@ redeclaration_error_message (tree newdecl, tree olddecl)
          return NULL;
        }
 
+      if (TREE_CODE (DECL_TEMPLATE_RESULT (newdecl)) == CONCEPT_DECL)
+        return G_("redefinition of %q#D");
+
       if (TREE_CODE (DECL_TEMPLATE_RESULT (newdecl)) != FUNCTION_DECL
          || (DECL_TEMPLATE_RESULT (newdecl)
              == DECL_TEMPLATE_RESULT (olddecl)))
@@ -8985,12 +9044,12 @@ grokfndecl (tree ctype,
       tree tmpl_reqs = NULL_TREE;
       if (processing_template_decl > template_class_depth (ctype))
         tmpl_reqs = TEMPLATE_PARMS_CONSTRAINTS (current_template_parms);
-
-      /* Adjust the required expression into a constraint. */
-      if (decl_reqs)
-        decl_reqs = normalize_expression (decl_reqs);
-
       tree ci = build_constraints (tmpl_reqs, decl_reqs);
+      if (concept_p && ci)
+        {
+          error_at (location, "a function concept cannot be constrained");
+          ci = NULL_TREE;
+        }
       set_constraints (decl, ci);
     }
 
@@ -9630,12 +9689,18 @@ grokvardecl (tree type,
       if (!same_type_ignoring_top_level_qualifiers_p (type, boolean_type_node))
        error_at (declspecs->locations[ds_type_spec],
                  "concept must have type %<bool%>");
+      if (TEMPLATE_PARMS_CONSTRAINTS (current_template_parms))
+        {
+          error_at (location, "a variable concept cannot be constrained");
+          TEMPLATE_PARMS_CONSTRAINTS (current_template_parms) = NULL_TREE;
+        }
     }
   else if (flag_concepts
           && processing_template_decl > template_class_depth (scope))
     {
       tree reqs = TEMPLATE_PARMS_CONSTRAINTS (current_template_parms);
       tree ci = build_constraints (reqs, NULL_TREE);
+
       set_constraints (decl, ci);
     }
 
@@ -12543,12 +12608,18 @@ grokdeclarator (const cp_declarator *declarator,
       if (ctype || in_namespace)
        error ("cannot use %<::%> in parameter declaration");
 
-      if (type_uses_auto (type)
-         && !(cxx_dialect >= cxx17 && template_parm_flag))
+      tree auto_node = type_uses_auto (type);
+      if (auto_node && !(cxx_dialect >= cxx17 && template_parm_flag))
        {
          if (cxx_dialect >= cxx14)
-           error_at (typespec_loc,
-                     "%<auto%> parameter not permitted in this context");
+           {
+             if (decl_context == PARM && AUTO_IS_DECLTYPE (auto_node))
+               error_at (typespec_loc,
+                         "cannot declare a parameter with %<decltype(auto)%>");
+             else
+               error_at (typespec_loc,
+                         "%<auto%> parameter not permitted in this context");
+           }
          else
            error_at (typespec_loc, "parameter declared %<auto%>");
          type = error_mark_node;
@@ -16362,8 +16433,17 @@ finish_function (bool inline_p)
   if (!processing_template_decl && FNDECL_USED_AUTO (fndecl)
       && TREE_TYPE (fntype) == DECL_SAVED_AUTO_RETURN_TYPE (fndecl))
     {
-      if (is_auto (DECL_SAVED_AUTO_RETURN_TYPE (fndecl)))
+      if (is_auto (DECL_SAVED_AUTO_RETURN_TYPE (fndecl))
+          && !current_function_returns_value
+          && !current_function_returns_null)
        {
+         /* We haven't applied return type deduction because we haven't
+             seen any return statements. Do that now.  */
+         tree node = type_uses_auto (DECL_SAVED_AUTO_RETURN_TYPE (fndecl));
+         do_auto_deduction (DECL_SAVED_AUTO_RETURN_TYPE (fndecl),
+                            void_node, node, tf_warning_or_error,
+                             adc_return_type);
+
          apply_deduced_return_type (fndecl, void_type_node);
          fntype = TREE_TYPE (fndecl);
        }
@@ -16953,7 +17033,9 @@ require_deduced_type (tree decl, tsubst_flags_t complain)
 {
   if (undeduced_auto_decl (decl))
     {
-      if (complain & tf_error)
+      if (TREE_NO_WARNING (decl) && seen_error ())
+       /* We probably already complained about deduction failure.  */;
+      else if (complain & tf_error)
        error ("use of %qD before deduction of %<auto%>", decl);
       return false;
     }
index 56201345429c902cc272c4bf1891b1f188a9b37e..1fd87d2abea51ec7d243e8e27dfb3e0d0ee86689 100644 (file)
@@ -98,6 +98,7 @@ static void print_instantiation_full_context (diagnostic_context *);
 static void print_instantiation_partial_context (diagnostic_context *,
                                                 struct tinst_level *,
                                                 location_t);
+static void maybe_print_constraint_context (diagnostic_context *);
 static void cp_diagnostic_starter (diagnostic_context *, diagnostic_info *);
 static void cp_print_error_function (diagnostic_context *, diagnostic_info *);
 
@@ -545,9 +546,7 @@ dump_type (cxx_pretty_printer *pp, tree t, int flags)
 
     case TEMPLATE_TYPE_PARM:
       pp_cxx_cv_qualifier_seq (pp, t);
-      if (tree c = PLACEHOLDER_TYPE_CONSTRAINTS (t))
-       pp_cxx_constrained_type_spec (pp, c);
-      else if (template_placeholder_p (t))
+      if (template_placeholder_p (t))
        {
          t = TREE_TYPE (CLASS_PLACEHOLDER_TEMPLATE (t));
          pp_cxx_tree_identifier (pp, TYPE_IDENTIFIER (t));
@@ -558,6 +557,9 @@ dump_type (cxx_pretty_printer *pp, tree t, int flags)
       else
        pp_cxx_canonical_template_parameter
          (pp, TEMPLATE_TYPE_PARM_INDEX (t));
+      /* If this is a constrained placeholder, add the requirements.  */
+      if (tree c = PLACEHOLDER_TYPE_CONSTRAINTS (t))
+        pp_cxx_constrained_type_spec (pp, c);
       break;
 
       /* This is not always necessary for pointers and such, but doing this
@@ -1284,6 +1286,15 @@ dump_decl (cxx_pretty_printer *pp, tree t, int flags)
       dump_template_decl (pp, t, flags);
       break;
 
+    case CONCEPT_DECL:
+      pp_cxx_ws_string (pp, "concept");
+      dump_decl_name (pp, DECL_NAME (t), flags);
+      break;
+
+    case WILDCARD_DECL:
+      pp_string (pp, "<wildcard>");
+      break;
+
     case TEMPLATE_ID_EXPR:
       {
        tree name = TREE_OPERAND (t, 0);
@@ -1448,7 +1459,9 @@ dump_template_decl (cxx_pretty_printer *pp, tree t, int flags)
   else if (DECL_TEMPLATE_RESULT (t)
            && (VAR_P (DECL_TEMPLATE_RESULT (t))
               /* Alias template.  */
-              || DECL_TYPE_TEMPLATE_P (t)))
+              || DECL_TYPE_TEMPLATE_P (t)
+               /* Concept definition.  &*/
+               || TREE_CODE (DECL_TEMPLATE_RESULT (t)) == CONCEPT_DECL))
     dump_decl (pp, DECL_TEMPLATE_RESULT (t), flags | TFF_TEMPLATE_NAME);
   else
     {
@@ -2082,6 +2095,7 @@ dump_expr (cxx_pretty_printer *pp, tree t, int flags)
     case TEMPLATE_DECL:
     case NAMESPACE_DECL:
     case LABEL_DECL:
+    case WILDCARD_DECL:
     case OVERLOAD:
     case TYPE_DECL:
     case IDENTIFIER_NODE:
@@ -2848,18 +2862,14 @@ dump_expr (cxx_pretty_printer *pp, tree t, int flags)
       pp_cxx_nested_requirement (cxx_pp, t);
       break;
 
-    case PRED_CONSTR:
+    case ATOMIC_CONSTR:
     case CHECK_CONSTR:
-    case EXPR_CONSTR:
-    case TYPE_CONSTR:
-    case ICONV_CONSTR:
-    case DEDUCT_CONSTR:
-    case EXCEPT_CONSTR:
-    case PARM_CONSTR:
     case CONJ_CONSTR:
     case DISJ_CONSTR:
-      pp_cxx_constraint (cxx_pp, t);
-      break;
+      {
+        pp_cxx_constraint (cxx_pp, t);
+        break;
+      }
 
     case PLACEHOLDER_EXPR:
       pp_string (pp, M_("*this"));
@@ -3326,6 +3336,7 @@ cp_diagnostic_starter (diagnostic_context *context,
   cp_print_error_function (context, diagnostic);
   maybe_print_instantiation_context (context);
   maybe_print_constexpr_context (context);
+  maybe_print_constraint_context (context);
   pp_set_prefix (context->printer, diagnostic_build_prefix (context,
                                                                 diagnostic));
 }
@@ -3650,6 +3661,171 @@ maybe_print_constexpr_context (diagnostic_context *context)
 }
 \f
 
+static void
+print_location (diagnostic_context *context, location_t loc)
+{
+  expanded_location xloc = expand_location (loc);
+  if (context->show_column)
+    pp_verbatim (context->printer, _("%r%s:%d:%d:%R   "),
+                 "locus", xloc.file, xloc.line, xloc.column);
+  else
+    pp_verbatim (context->printer, _("%r%s:%d:%R   "),
+                 "locus", xloc.file, xloc.line);
+}
+
+/* Instantiate the concept check for the purpose of diagnosing an error.  */
+
+static tree
+rebuild_concept_check (tree expr, tree map, tree args)
+{
+  /* Instantiate the parameter mapping for the template-id.  */
+  map = tsubst_parameter_mapping (map, args, tf_none, NULL_TREE);
+  if (map == error_mark_node)
+    return error_mark_node;
+  args = get_mapped_args (map);
+
+  /* Rebuild the template id using substituted arguments. Substituting
+     directly through the expression will trigger recursive satisfaction,
+     so don't do that.  */
+  tree id = unpack_concept_check (expr);
+  args = tsubst_template_args (TREE_OPERAND (id, 1), args, tf_none, NULL_TREE);
+  if (args == error_mark_node)
+    return error_mark_node;
+  return build_nt (TEMPLATE_ID_EXPR, TREE_OPERAND (id, 0), args);
+}
+
+static void
+print_constrained_decl_info (diagnostic_context *context, tree decl)
+{
+  print_location (context, DECL_SOURCE_LOCATION (decl));
+  pp_verbatim (context->printer, "required by the constraints of %q#D\n", decl);
+}
+
+static void
+print_concept_check_info (diagnostic_context *context, tree expr, tree map, tree args)
+{
+  gcc_assert (concept_check_p (expr));
+
+  tree id = unpack_concept_check (expr);
+  tree tmpl = TREE_OPERAND (id, 0);
+  if (OVL_P (tmpl))
+    tmpl = OVL_FIRST (tmpl);
+  tree check = rebuild_concept_check (expr, map, args);
+  if (check == error_mark_node)
+    check = expr;
+
+  print_location (context, DECL_SOURCE_LOCATION (tmpl));
+  pp_verbatim (context->printer, "required for the satisfaction of %qE\n", check);
+}
+
+/* Diagnose the entry point into the satisfaction error. Returns the next
+   context, if any.  */
+
+static tree
+print_constraint_context_head (diagnostic_context *context, tree cxt, tree args)
+{
+  tree src = TREE_VALUE (cxt);
+  if (!src)
+    {
+      print_location (context, input_location);
+      pp_verbatim (context->printer, "required for constraint satisfaction\n");
+      return NULL_TREE;
+    }
+  if (DECL_P (src))
+    {
+      print_constrained_decl_info (context, src);
+      return NULL_TREE;
+    }
+  else
+    {
+      print_concept_check_info (context, src, TREE_PURPOSE (cxt), args);
+      return TREE_CHAIN (cxt);
+    }
+}
+
+static void
+print_requires_expression_info (diagnostic_context *context, tree constr, tree args)
+{
+
+  tree expr = ATOMIC_CONSTR_EXPR (constr);
+  tree map = ATOMIC_CONSTR_MAP (constr);
+  map = tsubst_parameter_mapping (map, args, tf_none, NULL_TREE);
+  if (map == error_mark_node)
+    return;
+  args = get_mapped_args (map);
+
+  print_location (context, cp_expr_loc_or_input_loc (expr));
+  pp_verbatim (context->printer, "in requirements ");
+
+  tree parms = TREE_OPERAND (expr, 0);
+  if (parms)
+    pp_verbatim (context->printer, "with ");
+  while (parms)
+    {
+      tree next = TREE_CHAIN (parms);
+
+      TREE_CHAIN (parms) = NULL_TREE;
+      cp_unevaluated u;
+      tree p = tsubst (parms, args, tf_none, NULL_TREE);
+      pp_verbatim (context->printer, "%q#D", p);
+      TREE_CHAIN (parms) = next;
+
+      if (next)
+        pp_separate_with_comma ((cxx_pretty_printer *)context->printer);
+
+      parms = next;
+    }
+
+  pp_verbatim (context->printer, "\n");
+}
+
+void
+maybe_print_single_constraint_context (diagnostic_context *context, tree failed)
+{
+  if (!failed)
+    return;
+
+  tree constr = TREE_VALUE (failed);
+  if (!constr || constr == error_mark_node)
+    return;
+  tree cxt = CONSTR_CONTEXT (constr);
+  if (!cxt)
+    return;
+  tree args = TREE_PURPOSE (failed);
+
+  /* Print the stack of requirements.  */
+  cxt = print_constraint_context_head (context, cxt, args);
+  while (cxt && !DECL_P (TREE_VALUE (cxt)))
+    {
+      tree expr = TREE_VALUE (cxt);
+      tree map = TREE_PURPOSE (cxt);
+      print_concept_check_info (context, expr, map, args);
+      cxt = TREE_CHAIN (cxt);
+    }
+
+  /* For certain constraints, we can provide additional context.  */
+  if (TREE_CODE (constr) == ATOMIC_CONSTR
+      && TREE_CODE (ATOMIC_CONSTR_EXPR (constr)) == REQUIRES_EXPR)
+    print_requires_expression_info (context, constr, args);
+}
+
+void
+maybe_print_constraint_context (diagnostic_context *context)
+{
+  if (!current_failed_constraint)
+    return;
+
+  tree cur = current_failed_constraint;
+
+  /* Recursively print nested contexts.  */
+  current_failed_constraint = TREE_CHAIN (current_failed_constraint);
+  if (current_failed_constraint)
+    maybe_print_constraint_context (context);
+
+  /* Print this context.  */
+  maybe_print_single_constraint_context (context, cur);
+}
+
 /* Return true iff TYPE_A and TYPE_B are template types that are
    meaningful to compare.  */
 
index c4fed1682692d878bfbcfb820e77e11ef9cd8305..b503e9743cf17b0606622eaa5a5346fae4d56677 100644 (file)
@@ -1040,9 +1040,9 @@ maybe_add_lambda_conv_op (tree type)
 
   bool const generic_lambda_p = generic_lambda_fn_p (callop);
 
-  if (!generic_lambda_p && DECL_INITIAL (callop) == NULL_TREE)
+  if (!generic_lambda_p && undeduced_auto_decl (callop))
     {
-      /* If the op() wasn't instantiated due to errors, give up.  */
+      /* If the op() wasn't deduced due to errors, give up.  */
       gcc_assert (errorcount || sorrycount);
       return;
     }
index 13cc321243677cdc8c07eeb8507c5e6261ecddbf..2d4abaf6eddce56cb4f5b12fe3c7c06ba10e0904 100644 (file)
@@ -47,729 +47,736 @@ along with GCC; see the file COPYING3.  If not see
 #include "toplev.h"
 #include "type-utils.h"
 
-namespace {
+/* Hash functions for atomic constrains.  */
 
-// Helper algorithms
-
-template<typename I>
-inline I
-next (I iter)
+struct constraint_hash : default_hash_traits<tree>
 {
-  return ++iter;
-}
+  static hashval_t hash (tree t)
+  {
+    return hash_atomic_constraint (t);
+  }
 
-template<typename I, typename P>
-inline bool
-any_p (I first, I last, P pred)
-{
-  while (first != last)
-    {
-      if (pred(*first))
-        return true;
-      ++first;
-    }
-  return false;
-}
+  static bool equal (tree t1, tree t2)
+  {
+    return atomic_constraints_identical_p (t1, t2);
+  }
+};
 
-bool prove_implication (tree, tree);
+/* A conjunctive or disjunctive clause.
 
-/*---------------------------------------------------------------------------
-                           Proof state
----------------------------------------------------------------------------*/
+   Each clause maintains an iterator that refers to the current
+   term, which is used in the linear decomposition of a formula
+   into CNF or DNF.  */
 
-struct term_entry
+struct clause
 {
-  tree t;
-};
+  typedef std::list<tree>::iterator iterator;
+  typedef std::list<tree>::const_iterator const_iterator;
 
-/* Hashing function and equality for constraint entries.  */
+  /* Initialize a clause with an initial term.  */
 
-struct term_hasher : ggc_ptr_hash<term_entry>
-{
-  static hashval_t hash (term_entry *e)
+  clause (tree t)
   {
-    return iterative_hash_template_arg (e->t, 0);
+    m_terms.push_back (t);
+    if (TREE_CODE (t) == ATOMIC_CONSTR)
+      m_set.add (t);
+
+    m_current = m_terms.begin ();
   }
 
-  static bool equal (term_entry *e1, term_entry *e2)
+  /* Create a copy of the current term. The current
+     iterator is set to point to the same position in the
+     copied list of terms.  */
+
+  clause (clause const& c)
+    : m_terms (c.m_terms), m_set (c.m_set), m_current (m_terms.begin ())
   {
-    return cp_tree_equal (e1->t, e2->t);
+    std::advance (m_current, std::distance (c.begin (), c.current ()));
   }
-};
 
-/* A term list is a list of atomic constraints. It is used
-   to maintain the lists of assumptions and conclusions in a
-   proof goal.
+  /* Returns true when all terms are atoms.  */
 
-   Each term list maintains an iterator that refers to the current
-   term. This can be used by various tactics to support iteration
-   and stateful manipulation of the list. */
-class term_list
-{
-public:
-  typedef std::list<tree>::iterator iterator;
+  bool done () const
+  {
+    return m_current == end ();
+  }
 
-  term_list ();
-  term_list (tree);
+  /* Advance to the next term.  */
 
-  bool includes (tree);
-  iterator insert (iterator, tree);
-  iterator push_back (tree);
-  iterator erase (iterator);
-  iterator replace (iterator, tree);
-  iterator replace (iterator, tree, tree);
+  void advance ()
+  {
+    gcc_assert (!done ());
+    ++m_current;
+  }
 
-  iterator begin() { return seq.begin(); }
-  iterator end() { return seq.end(); }
+  /* Replaces the current term at position ITER with T.  If
+     T is an atomic constraint that already appears in the
+     clause, remove but do not replace ITER. Returns a pair
+     containing an iterator to the replace object or past
+     the erased object and a boolean value which is true if
+     an object was erased.  */
 
-  std::list<tree>         seq;
-  hash_table<term_hasher> tab;
-};
+  std::pair<iterator, bool> replace (iterator iter, tree t)
+  {
+    gcc_assert (TREE_CODE (*iter) != ATOMIC_CONSTR);
+    if (TREE_CODE (t) == ATOMIC_CONSTR)
+      {
+       if (m_set.add (t))
+         return std::make_pair (m_terms.erase (iter), true);
+      }
+    *iter = t;
+    return std::make_pair (iter, false);
+  }
 
-inline
-term_list::term_list ()
-  : seq(), tab (11)
-{
-}
+  /* Inserts T before ITER in the list of terms.  If T has 
+     already is an atomic constraint that already appears in
+     the clause, no action is taken, and the current iterator
+     is returned. Returns a pair of an iterator to the inserted
+     object or ITER if no insertion occurred and a boolean
+     value which is true if an object was inserted.  */
 
-/* Initialize a term list with an initial term. */
+  std::pair<iterator, bool> insert (iterator iter, tree t)
+  {
+    if (TREE_CODE (t) == ATOMIC_CONSTR)
+    {
+      if (m_set.add (t))
+       return std::make_pair (iter, false);
+    }
+    return std::make_pair (m_terms.insert (iter, t), true);
+  }
 
-inline
-term_list::term_list (tree t)
-  : seq (), tab (11)
-{
-  push_back (t);
-}
+  /* Replaces the current term with T. In the case where the
+     current term is erased (because T is redundant), update
+     the position of the current term to the next term.  */
 
-/* Returns true if T is the in the tree. */
+  void replace (tree t)
+  {
+    m_current = replace (m_current, t).first;
+  }
 
-inline bool
-term_list::includes (tree t)
-{
-  term_entry ent = {t};
-  return tab.find (&ent);
-}
+  /* Replace the current term with T1 and T2, in that order.  */
 
-/* Append a term to the list. */
-inline term_list::iterator
-term_list::push_back (tree t)
-{
-  return insert (end(), t);
-}
+  void replace (tree t1, tree t2)
+  {
+    /* Replace the current term with t1. Ensure that iter points
+       to the term before which t2 will be inserted.  Update the
+       current term as needed.  */
+    std::pair<iterator, bool> rep = replace (m_current, t1);
+    if (rep.second)
+      m_current = rep.first;
+    else
+      ++rep.first;
 
-/* Insert a new (unseen) term T into the list before the proposition
-   indicated by ITER. Returns the iterator to the newly inserted
-   element.  */
+    /* Insert the t2. Make this the current term if we erased
+       the prior term.  */
+    std::pair<iterator, bool> ins = insert (rep.first, t2);
+    if (rep.second && ins.second)
+      m_current = ins.first;
+  }
 
-term_list::iterator
-term_list::insert (iterator iter, tree t)
-{
-  gcc_assert (!includes (t));
-  iter = seq.insert (iter, t);
-  term_entry ent = {t};
-  term_entry** slot = tab.find_slot (&ent, INSERT);
-  term_entry* ptr = ggc_alloc<term_entry> ();
-  *ptr = ent;
-  *slot = ptr;
-  return iter;
-}
+  /* Returns true if the clause contains the term T.  */
 
-/* Remove an existing term from the list. Returns an iterator referring
-   to the element after the removed term.  This may be end().  */
+  bool contains (tree t)
+  {
+    gcc_assert (TREE_CODE (t) == ATOMIC_CONSTR);
+    return m_set.contains (t);
+  }
 
-term_list::iterator
-term_list::erase (iterator iter)
-{
-  gcc_assert (includes (*iter));
-  term_entry ent = {*iter};
-  tab.remove_elt (&ent);
-  iter = seq.erase (iter);
-  return iter;
-}
 
-/* Replace the given term with that specified. If the term has
-   been previously seen, do not insert the term. Returns the
-   first iterator past the current term.  */
+  /* Returns an iterator to the first clause in the formula.  */
 
-term_list::iterator
-term_list::replace (iterator iter, tree t)
-{
-  iter = erase (iter);
-  if (!includes (t))
-    insert (iter, t);
-  return iter;
-}
+  iterator begin ()
+  {
+    return m_terms.begin ();
+  }
 
+  /* Returns an iterator to the first clause in the formula.  */
 
-/* Replace the term at the given position by the supplied T1
-   followed by t2. This is used in certain logical operators to
-   load a list of assumptions or conclusions.  */
+  const_iterator begin () const
+  {
+    return m_terms.begin ();
+  }
 
-term_list::iterator
-term_list::replace (iterator iter, tree t1, tree t2)
-{
-  iter = erase (iter);
-  if (!includes (t1))
-    insert (iter, t1);
-  if (!includes (t2))
-    insert (iter, t2);
-  return iter;
-}
+  /* Returns an iterator past the last clause in the formula.  */
+
+  iterator end ()
+  {
+    return m_terms.end ();
+  }
 
-/* A goal (or subgoal) models a sequent of the form
-   'A |- C' where A and C are lists of assumptions and
-   conclusions written as propositions in the constraint
-   language (i.e., lists of trees). */
+  /* Returns an iterator past the last clause in the formula.  */
 
-class proof_goal
-{
-public:
-  term_list assumptions;
-  term_list conclusions;
+  const_iterator end () const
+  {
+    return m_terms.end ();
+  }
+
+  /* Returns the current iterator.  */
+
+  const_iterator current () const
+  {
+    return m_current;
+  }
+
+  std::list<tree> m_terms; /* The list of terms.  */
+  hash_set<tree, false, constraint_hash> m_set; /* The set of atomic constraints.  */
+  iterator m_current; /* The current term.  */
 };
 
+
 /* A proof state owns a list of goals and tracks the
    current sub-goal. The class also provides facilities
    for managing subgoals and constructing term lists. */
 
-class proof_state : public std::list<proof_goal>
+struct formula
 {
-public:
-  proof_state ();
+  typedef std::list<clause>::iterator iterator;
+  typedef std::list<clause>::const_iterator const_iterator;
 
-  iterator branch (iterator i);
-  iterator discharge (iterator i);
-};
+  /* Construct a formula with an initial formula in a
+     single clause.  */
 
-/* Initialize the state with a single empty goal, and set that goal
-   as the current subgoal.  */
-
-inline
-proof_state::proof_state ()
-  : std::list<proof_goal> (1)
-{ }
+  formula (tree t)
+  {
+    /* This should call emplace_back(). There's a an extra copy being
+       invoked by using push_back().  */
+    m_clauses.push_back (t);
+    m_current = m_clauses.begin ();
+  }
 
+  /* Returns true when all clauses are atomic.  */
+  bool done () const
+  {
+    return m_current == end ();
+  }
 
-/* Branch the current goal by creating a new subgoal, returning a
-   reference to the new object. This does not update the current goal. */
+  /* Advance to the next term.  */
+  void advance ()
+  {
+    gcc_assert (!done ());
+    ++m_current;
+  }
 
-inline proof_state::iterator
-proof_state::branch (iterator i)
-{
-  gcc_assert (i != end());
-  proof_goal& g = *i;
-  return insert (++i, g);
-}
+  /* Insert a copy of clause into the formula. This corresponds
+     to a distribution of one logical operation over the other.  */
 
-/* Discharge the current goal, setting it equal to the
-   next non-satisfied goal. */
+  clause& branch ()
+  {
+    gcc_assert (!done ());
+    m_clauses.push_back (*m_current);
+    return m_clauses.back ();
+  }
 
-inline proof_state::iterator
-proof_state::discharge (iterator i)
-{
-  gcc_assert (i != end());
-  return erase (i);
-}
+  /* Returns the position of the current clause.  */
 
+  iterator current ()
+  {
+    return m_current;
+  }
 
-/*---------------------------------------------------------------------------
-                        Debugging
----------------------------------------------------------------------------*/
+  /* Returns an iterator to the first clause in the formula.  */
 
-// void
-// debug (term_list& ts)
-// {
-//   for (term_list::iterator i = ts.begin(); i != ts.end(); ++i)
-//     verbatim ("  # %E", *i);
-// }
-//
-// void
-// debug (proof_goal& g)
-// {
-//   debug (g.assumptions);
-//   verbatim ("       |-");
-//   debug (g.conclusions);
-// }
+  iterator begin ()
+  {
+    return m_clauses.begin ();
+  }
 
-/*---------------------------------------------------------------------------
-                        Atomicity of constraints
----------------------------------------------------------------------------*/
+  /* Returns an iterator to the first clause in the formula.  */
 
-/* Returns true if T is not an atomic constraint.  */
+  const_iterator begin () const
+  {
+    return m_clauses.begin ();
+  }
 
-bool
-non_atomic_constraint_p (tree t)
-{
-  switch (TREE_CODE (t))
-    {
-    case PRED_CONSTR:
-    case EXPR_CONSTR:
-    case TYPE_CONSTR:
-    case ICONV_CONSTR:
-    case DEDUCT_CONSTR:
-    case EXCEPT_CONSTR:
-      /* A pack expansion isn't atomic, but it can't decompose to prove an
-        atom, so it shouldn't cause analyze_atom to return undecided.  */
-    case EXPR_PACK_EXPANSION:
-      return false;
-    case CHECK_CONSTR:
-    case PARM_CONSTR:
-    case CONJ_CONSTR:
-    case DISJ_CONSTR:
-      return true;
-    default:
-      gcc_unreachable ();
-    }
-}
+  /* Returns an iterator past the last clause in the formula.  */
 
-/* Returns true if any constraints in T are not atomic.  */
+  iterator end ()
+  {
+    return m_clauses.end ();
+  }
 
-bool
-any_non_atomic_constraints_p (term_list& t)
-{
-  return any_p (t.begin(), t.end(), non_atomic_constraint_p);
-}
+  /* Returns an iterator past the last clause in the formula.  */
 
-/*---------------------------------------------------------------------------
-                           Proof validations
----------------------------------------------------------------------------*/
+  const_iterator end () const
+  {
+    return m_clauses.end ();
+  }
 
-enum proof_result
-{
-  invalid,
-  valid,
-  undecided
+  std::list<clause> m_clauses; /* The list of clauses.  */
+  iterator m_current; /* The current clause.  */
 };
 
-proof_result check_term (term_list&, tree);
-
-
-proof_result
-analyze_atom (term_list& ts, tree t)
+void
+debug (clause& c)
 {
-  /* FIXME: Hook into special cases, if any. */
-  /*
-  term_list::iterator iter = ts.begin();
-  term_list::iterator end = ts.end();
-  while (iter != end)
-    {
-      ++iter;
-    }
-  */
-
-  if (non_atomic_constraint_p (t))
-    return undecided;
-  if (any_non_atomic_constraints_p (ts))
-    return undecided;
-  return invalid;
+  for (clause::iterator i = c.begin(); i != c.end(); ++i)
+    verbatim ("  # %E", *i);
 }
 
-/* Search for a pack expansion in the list of assumptions that would
-   make this expansion valid.  */
-
-proof_result
-analyze_pack (term_list& ts, tree t)
+void
+debug (formula& f)
 {
-  tree c1 = normalize_expression (PACK_EXPANSION_PATTERN (t));
-  term_list::iterator iter = ts.begin();
-  term_list::iterator end = ts.end();
-  while (iter != end)
+  for (formula::iterator i = f.begin(); i != f.end(); ++i)
     {
-      if (TREE_CODE (*iter) == TREE_CODE (t))
-        {
-          tree c2 = normalize_expression (PACK_EXPANSION_PATTERN (*iter));
-          if (prove_implication (c2, c1))
-            return valid;
-          else
-            return invalid;
-        }
-      ++iter;
+      verbatim ("(((");
+      debug (*i);
+      verbatim (")))");
     }
-  return invalid;
 }
 
-/* Search for concept checks in TS that we know subsume T. */
+/* The logical rules used to analyze a logical formula. The
+   "left" and "right" refer to the position of formula in a
+   sequent (as in sequent calculus).  */
 
-proof_result
-search_known_subsumptions (term_list& ts, tree t)
+enum rules
 {
-  for (term_list::iterator i = ts.begin(); i != ts.end(); ++i)
-    if (TREE_CODE (*i) == CHECK_CONSTR)
-      {
-        if (bool* b = lookup_subsumption_result (*i, t))
-          return *b ? valid : invalid;
-      }
-  return undecided;
-}
+  left, right
+};
 
-/* Determine if the terms in TS provide sufficient support for proving
-   the proposition T. If any term in TS is a concept check that is known
-   to subsume T, then the proof is valid. Otherwise, we have to expand T
-   and continue searching for support.  */
+/* Distribution counting.  */
 
-proof_result
-analyze_check (term_list& ts, tree t)
+static inline bool
+disjunction_p (tree t)
 {
-  proof_result r = search_known_subsumptions (ts, t);
-  if (r != undecided)
-    return r;
-
-  tree tmpl = CHECK_CONSTR_CONCEPT (t);
-  tree args = CHECK_CONSTR_ARGS (t);
-  tree c = expand_concept (tmpl, args);
-  return check_term (ts, c);
+  return TREE_CODE (t) == DISJ_CONSTR;
 }
 
-/* Recursively check constraints of the parameterized constraint. */
-
-proof_result
-analyze_parameterized (term_list& ts, tree t)
+static inline bool
+conjunction_p (tree t)
 {
-  return check_term (ts, PARM_CONSTR_OPERAND (t));
+  return TREE_CODE (t) == CONJ_CONSTR;
 }
 
-proof_result
-analyze_conjunction (term_list& ts, tree t)
+static inline bool
+atomic_p (tree t)
 {
-  proof_result r = check_term (ts, TREE_OPERAND (t, 0));
-  if (r == invalid || r == undecided)
-    return r;
-  return check_term (ts, TREE_OPERAND (t, 1));
+  return TREE_CODE (t) == ATOMIC_CONSTR;
 }
 
-proof_result
-analyze_disjunction (term_list& ts, tree t)
-{
-  proof_result r = check_term (ts, TREE_OPERAND (t, 0));
-  if (r == valid)
-    return r;
-  return check_term (ts, TREE_OPERAND (t, 1));
-}
+/* Recursively count the number of clauses produced when converting T
+   to DNF. Returns a pair containing the number of clauses and a bool
+   value signifying that the the tree would be rewritten as a result of
+   distributing. In general, a conjunction for which this flag is set
+   is considered a disjunction for the purpose of counting.  */
 
-proof_result
-analyze_term (term_list& ts, tree t)
+static std::pair<int, bool>
+dnf_size_r (tree t)
 {
-  switch (TREE_CODE (t))
-    {
-    case CHECK_CONSTR:
-      return analyze_check (ts, t);
+  if (atomic_p (t))
+    /* Atomic constraints produce no clauses.  */
+    return std::make_pair (0, false);
 
-    case PARM_CONSTR:
-      return analyze_parameterized (ts, t);
+  /* For compound constraints, recursively count clauses and unpack
+     the results.  */
+  tree lhs = TREE_OPERAND (t, 0);
+  tree rhs = TREE_OPERAND (t, 1);
+  std::pair<int, bool> p1 = dnf_size_r (lhs);
+  std::pair<int, bool> p2 = dnf_size_r (rhs);
+  int n1 = p1.first, n2 = p2.first;
+  bool d1 = p1.second, d2 = p2.second;
 
-    case CONJ_CONSTR:
-      return analyze_conjunction (ts, t);
-    case DISJ_CONSTR:
-      return analyze_disjunction (ts, t);
-
-    case PRED_CONSTR:
-    case EXPR_CONSTR:
-    case TYPE_CONSTR:
-    case ICONV_CONSTR:
-    case DEDUCT_CONSTR:
-    case EXCEPT_CONSTR:
-      return analyze_atom (ts, t);
-
-    case EXPR_PACK_EXPANSION:
-      return analyze_pack (ts, t);
-
-    case ERROR_MARK:
-      /* Encountering an error anywhere in a constraint invalidates
-         the proof, since the constraint is ill-formed.  */
-      return invalid;
-    default:
-      gcc_unreachable ();
+  if (disjunction_p (t))
+    {
+      /* Matches constraints of the form P \/ Q. Disjunctions contribute
+        linearly to the number of constraints.  When both P and Q are
+        disjunctions, clauses are added. When only one of P and Q
+        is a disjunction, an additional clause is produced. When neither
+        P nor Q are disjunctions, two clauses are produced.  */
+      if (disjunction_p (lhs))
+       {
+         if (disjunction_p (rhs) || (conjunction_p (rhs) && d2))
+           /* Both P and Q are disjunctions.  */
+           return std::make_pair (n1 + n2, d1 | d2);
+         else
+           /* Only LHS is a disjunction.  */
+           return std::make_pair (1 + n1 + n2, d1 | d2);
+         gcc_unreachable ();
+       }
+      if (conjunction_p (lhs))
+       {
+         if ((disjunction_p (rhs) && d1) || (conjunction_p (rhs) && d1 && d2))
+           /* Both P and Q are disjunctions.  */
+           return std::make_pair (n1 + n2, d1 | d2);
+         if (disjunction_p (rhs)
+             || (conjunction_p (rhs) && d1 != d2)
+             || (atomic_p (rhs) && d1))
+           /* Either LHS or RHS is a disjunction.  */
+           return std::make_pair (1 + n1 + n2, d1 | d2);
+         else
+           /* Neither LHS nor RHS is a disjunction.  */
+           return std::make_pair (2, false);
+       }
+      if (atomic_p (lhs))
+       {
+         if (disjunction_p (rhs) || (conjunction_p (rhs) && d2))
+           /* Only RHS is a disjunction.  */
+           return std::make_pair (1 + n1 + n2, d1 | d2);
+         else
+           /* Neither LHS nor RHS is a disjunction.  */
+           return std::make_pair (2, false);
+       }
+    }
+  else /* conjunction_p (t)  */
+    {
+      /* Matches constraints of the form P /\ Q, possibly resulting
+         in the distribution of one side over the other. When both
+         P and Q are disjunctions, the number of clauses are multiplied.
+         When only one of P and Q is a disjunction, the the number of
+         clauses are added. Otherwise, neither side is a disjunction and
+         no clauses are created.  */
+      if (disjunction_p (lhs))
+       {
+         if (disjunction_p (rhs) || (conjunction_p (rhs) && d2))
+           /* Both P and Q are disjunctions.  */
+           return std::make_pair (n1 * n2, true);
+         else
+           /* Only LHS is a disjunction.  */
+           return std::make_pair (n1 + n2, true);
+         gcc_unreachable ();
+       }
+      if (conjunction_p (lhs))
+       {
+         if ((disjunction_p (rhs) && d1) || (conjunction_p (rhs) && d1 && d2))
+           /* Both P and Q are disjunctions.  */
+           return std::make_pair (n1 * n2, true);
+         if (disjunction_p (rhs)
+             || (conjunction_p (rhs) && d1 != d2)
+             || (atomic_p (rhs) && d1))
+           /* Either LHS or RHS is a disjunction.  */
+           return std::make_pair (n1 + n2, true);
+         else
+           /* Neither LHS nor RHS is a disjunction.  */
+           return std::make_pair (0, false);
+       }
+      if (atomic_p (lhs))
+       {
+         if (disjunction_p (rhs) || (conjunction_p (rhs) && d2))
+           /* Only RHS is a disjunction.  */
+           return std::make_pair (n1 + n2, true);
+         else
+           /* Neither LHS nor RHS is a disjunction.  */
+           return std::make_pair (0, false);
+       }
     }
+  gcc_unreachable ();
 }
 
-/* Check if a single term can be proven from a set of assumptions.
-   If the proof is not valid, then it is incomplete when either
-   the given term is non-atomic or any term in the list of assumptions
-   is not-atomic.  */
+/* Recursively count the number of clauses produced when converting T
+   to CNF. Returns a pair containing the number of clauses and a bool
+   value signifying that the the tree would be rewritten as a result of
+   distributing. In general, a disjunction for which this flag is set
+   is considered a conjunction for the purpose of counting.  */
 
-proof_result
-check_term (term_list& ts, tree t)
+static std::pair<int, bool>
+cnf_size_r (tree t)
 {
-  /* Try the easy way; search for an equivalent term.  */
-  if (ts.includes (t))
-    return valid;
+  if (atomic_p (t))
+    /* Atomic constraints produce no clauses.  */
+    return std::make_pair (0, false);
 
-  /* The hard way; actually consider what the term means.  */
-  return analyze_term (ts, t);
-}
+  /* For compound constraints, recursively count clauses and unpack
+     the results.  */
+  tree lhs = TREE_OPERAND (t, 0);
+  tree rhs = TREE_OPERAND (t, 1);
+  std::pair<int, bool> p1 = cnf_size_r (lhs);
+  std::pair<int, bool> p2 = cnf_size_r (rhs);
+  int n1 = p1.first, n2 = p2.first;
+  bool d1 = p1.second, d2 = p2.second;
 
-/* Check to see if any term is proven by the assumptions in the
-   proof goal. The proof is valid if the proof of any term is valid.
-   If validity cannot be determined, but any particular
-   check was undecided, then this goal is undecided.  */
-
-proof_result
-check_goal (proof_goal& g)
-{
-  term_list::iterator iter = g.conclusions.begin ();
-  term_list::iterator end = g.conclusions.end ();
-  bool incomplete = false;
-  while (iter != end)
+  if (disjunction_p (t))
     {
-      proof_result r = check_term (g.assumptions, *iter);
-      if (r == valid)
-        return r;
-      if (r == undecided)
-        incomplete = true;
-      ++iter;
+      /* Matches constraints of the form P \/ Q, possibly resulting
+         in the distribution of one side over the other. When both
+         P and Q are conjunctions, the number of clauses are multiplied.
+         When only one of P and Q is a conjunction, the the number of
+         clauses are added. Otherwise, neither side is a conjunction and
+         no clauses are created.  */
+      if (disjunction_p (lhs))
+       {
+         if ((disjunction_p (rhs) && d1 && d2) || (conjunction_p (rhs) && d1))
+           /* Both P and Q are conjunctions.  */
+           return std::make_pair (n1 * n2, true);
+         if ((disjunction_p (rhs) && d1 != d2)
+             || conjunction_p (rhs)
+             || (atomic_p (rhs) && d1))
+           /* Either LHS or RHS is a conjunction.  */
+           return std::make_pair (n1 + n2, true);
+         else
+           /* Neither LHS nor RHS is a conjunction.  */
+           return std::make_pair (0, false);
+         gcc_unreachable ();
+       }
+      if (conjunction_p (lhs))
+       {
+         if ((disjunction_p (rhs) && d2) || conjunction_p (rhs))
+           /* Both LHS and RHS are conjunctions.  */
+           return std::make_pair (n1 * n2, true);
+         else
+           /* Only LHS is a conjunction.  */
+           return std::make_pair (n1 + n2, true);
+       }
+      if (atomic_p (lhs))
+       {
+         if ((disjunction_p (rhs) && d2) || conjunction_p (rhs))
+           /* Only RHS is a disjunction.  */
+           return std::make_pair (n1 + n2, true);
+         else
+           /* Neither LHS nor RHS is a disjunction.  */
+           return std::make_pair (0, false);
+       }
     }
-
-    /* Was the proof complete? */
-    if (incomplete)
-      return undecided;
-    else
-      return invalid;
-}
-
-/* Check if the the proof is valid. This is the case when all
-   goals can be discharged. If any goal is invalid, then the
-   entire proof is invalid. Otherwise, the proof is undecided.  */
-
-proof_result
-check_proof (proof_state& p)
-{
-  proof_state::iterator iter = p.begin();
-  proof_state::iterator end = p.end();
-  while (iter != end)
+  else /* conjunction_p (t)  */
     {
-      proof_result r = check_goal (*iter);
-      if (r == invalid)
-        return r;
-      if (r == valid)
-        iter = p.discharge (iter);
-      else
-        ++iter;
+      /* Matches constraints of the form P /\ Q. Conjunctions contribute
+        linearly to the number of constraints.  When both P and Q are
+        conjunctions, clauses are added. When only one of P and Q
+        is a conjunction, an additional clause is produced. When neither
+        P nor Q are conjunctions, two clauses are produced.  */
+      if (disjunction_p (lhs))
+       {
+         if ((disjunction_p (rhs) && d1 && d2) || (conjunction_p (rhs) && d1))
+           /* Both P and Q are conjunctions.  */
+           return std::make_pair (n1 + n2, d1 | d2);
+         if ((disjunction_p (rhs) && d1 != d2)
+             || conjunction_p (rhs)
+             || (atomic_p (rhs) && d1))
+           /* Either LHS or RHS is a conjunction.  */
+           return std::make_pair (1 + n1 + n2, d1 | d2);
+         else
+           /* Neither LHS nor RHS is a conjunction.  */
+           return std::make_pair (2, false);
+         gcc_unreachable ();
+       }
+      if (conjunction_p (lhs))
+       {
+         if ((disjunction_p (rhs) && d2) || conjunction_p (rhs))
+           /* Both LHS and RHS are conjunctions.  */
+           return std::make_pair (n1 + n2, d1 | d2);
+         else
+           /* Only LHS is a conjunction.  */
+           return std::make_pair (1 + n1 + n2, d1 | d2);
+       }
+      if (atomic_p (lhs))
+       {
+         if ((disjunction_p (rhs) && d2) || conjunction_p (rhs))
+           /* Only RHS is a disjunction.  */
+           return std::make_pair (1 + n1 + n2, d1 | d2);
+         else
+           /* Neither LHS nor RHS is a disjunction.  */
+           return std::make_pair (2, false);
+       }
     }
-
-  /* If all goals are discharged, then the proof is valid.  */
-  if (p.empty())
-    return valid;
-  else
-    return undecided;
+  gcc_unreachable ();
 }
 
-/*---------------------------------------------------------------------------
-                           Left logical rules
----------------------------------------------------------------------------*/
+/* Count the number conjunctive clauses that would be created
+   when rewriting T to DNF. */
 
-term_list::iterator
-load_check_assumption (term_list& ts, term_list::iterator i)
+static int
+dnf_size (tree t)
 {
-  tree decl = CHECK_CONSTR_CONCEPT (*i);
-  tree tmpl = DECL_TI_TEMPLATE (decl);
-  tree args = CHECK_CONSTR_ARGS (*i);
-  return ts.replace(i, expand_concept (tmpl, args));
+  std::pair<int, bool> result = dnf_size_r (t);
+  return result.first == 0 ? 1 : result.first;
 }
 
-term_list::iterator
-load_parameterized_assumption (term_list& ts, term_list::iterator i)
-{
-  return ts.replace(i, PARM_CONSTR_OPERAND(*i));
-}
 
-term_list::iterator
-load_conjunction_assumption (term_list& ts, term_list::iterator i)
+/* Count the number disjunctive clauses that would be created
+   when rewriting T to CNF. */
+
+static int
+cnf_size (tree t)
 {
-  tree t1 = TREE_OPERAND (*i, 0);
-  tree t2 = TREE_OPERAND (*i, 1);
-  return ts.replace(i, t1, t2);
+  std::pair<int, bool> result = cnf_size_r (t);
+  return result.first == 0 ? 1 : result.first;
 }
 
-/* Examine the terms in the list, and apply left-logical rules to move
-   terms into the set of assumptions. */
+
+/* A left-conjunction is replaced by its operands.  */
 
 void
-load_assumptions (proof_goal& g)
+replace_term (clause& c, tree t)
 {
-  term_list::iterator iter = g.assumptions.begin();
-  term_list::iterator end = g.assumptions.end();
-  while (iter != end)
-    {
-      switch (TREE_CODE (*iter))
-        {
-        case CHECK_CONSTR:
-          iter = load_check_assumption (g.assumptions, iter);
-          break;
-        case PARM_CONSTR:
-          iter = load_parameterized_assumption (g.assumptions, iter);
-          break;
-        case CONJ_CONSTR:
-          iter = load_conjunction_assumption (g.assumptions, iter);
-          break;
-        default:
-          ++iter;
-          break;
-        }
-    }
+  tree t1 = TREE_OPERAND (t, 0);
+  tree t2 = TREE_OPERAND (t, 1);
+  return c.replace (t1, t2);
 }
 
-/* In each subgoal, load constraints into the assumption set.  */
+/* Create a new clause in the formula by copying the current
+   clause. In the current clause, the term at CI is replaced
+   by the first operand, and in the new clause, it is replaced
+   by the second.  */
 
 void
-load_assumptions(proof_state& p)
+branch_clause (formula& f, clause& c1, tree t)
 {
-  proof_state::iterator iter = p.begin();
-  while (iter != p.end())
-    {
-      load_assumptions (*iter);
-      ++iter;
-    }
+  tree t1 = TREE_OPERAND (t, 0);
+  tree t2 = TREE_OPERAND (t, 1);
+  clause& c2 = f.branch ();
+  c1.replace (t1);
+  c2.replace (t2);
 }
 
-void
-explode_disjunction (proof_state& p, proof_state::iterator gi, term_list::iterator ti1)
-{
-  tree t1 = TREE_OPERAND (*ti1, 0);
-  tree t2 = TREE_OPERAND (*ti1, 1);
+/* Decompose t1 /\ t2 according to the rules R.  */
 
-  /* Erase the current term from the goal. */
-  proof_goal& g1 = *gi;
-  proof_goal& g2 = *p.branch (gi);
+inline void
+decompose_conjuntion (formula& f, clause& c, tree t, rules r)
+{
+  if (r == left)
+    replace_term (c, t);
+  else
+    branch_clause (f, c, t);
+}
 
-  /* Get an iterator to the equivalent position in th enew goal. */
-  int n = std::distance (g1.assumptions.begin (), ti1);
-  term_list::iterator ti2 = g2.assumptions.begin ();
-  std::advance (ti2, n);
+/* Decompose t1 \/ t2 according to the rules R.  */
 
-  /* Replace the disjunction in both branches. */
-  g1.assumptions.replace (ti1, t1);
-  g2.assumptions.replace (ti2, t2);
+inline void
+decompose_disjunction (formula& f, clause& c, tree t, rules r)
+{
+  if (r == right)
+    replace_term (c, t);
+  else
+    branch_clause (f, c, t);
 }
 
+/* An atomic constraint is already decomposed.  */
+inline void
+decompose_atom (clause& c)
+{
+  c.advance ();
+}
 
-/* Search the assumptions of the goal for the first disjunction. */
+/* Decompose a term of clause C (in formula F) according to the
+   logical rules R. */
 
-bool
-explode_goal (proof_state& p, proof_state::iterator gi)
+void
+decompose_term (formula& f, clause& c, tree t, rules r)
 {
-  term_list& ts = gi->assumptions;
-  term_list::iterator ti = ts.begin();
-  term_list::iterator end = ts.end();
-  while (ti != end)
+  switch (TREE_CODE (t))
     {
-      if (TREE_CODE (*ti) == DISJ_CONSTR)
-        {
-          explode_disjunction (p, gi, ti);
-          return true;
-        }
-      else ++ti;
+      case CONJ_CONSTR:
+        return decompose_conjuntion (f, c, t, r);
+      case DISJ_CONSTR:
+       return decompose_disjunction (f, c, t, r);
+      default:
+       return decompose_atom (c);
     }
-  return false;
 }
 
-/* Search for the first goal with a disjunction, and then branch
-   creating a clone of that subgoal. */
+/* Decompose C (in F) using the logical rules R until it
+   is comprised of only atomic constraints.  */
 
 void
-explode_assumptions (proof_state& p)
+decompose_clause (formula& f, clause& c, rules r)
 {
-  proof_state::iterator iter = p.begin();
-  proof_state::iterator end = p.end();
-  while (iter != end)
-    {
-      if (explode_goal (p, iter))
-        return;
-      ++iter;
-    }
+  while (!c.done ())
+    decompose_term (f, c, *c.current (), r);
+  f.advance ();
 }
 
+/* Decompose the logical formula F according to the logical
+   rules determined by R.  The result is a formula containing
+   clauses that contain only atomic terms.  */
 
-/*---------------------------------------------------------------------------
-                           Right logical rules
----------------------------------------------------------------------------*/
-
-term_list::iterator
-load_disjunction_conclusion (term_list& g, term_list::iterator i)
+void
+decompose_formula (formula& f, rules r)
 {
-  tree t1 = TREE_OPERAND (*i, 0);
-  tree t2 = TREE_OPERAND (*i, 1);
-  return g.replace(i, t1, t2);
+  while (!f.done ())
+    decompose_clause (f, *f.current (), r);
 }
 
-/* Apply logical rules to the right hand side. This will load the
-   conclusion set with all tpp-level disjunctions.  */
+/* Fully decomposing T into a list of sequents, each comprised of
+   a list of atomic constraints, as if T were an antecedent.  */
 
-void
-load_conclusions (proof_goal& g)
+static formula
+decompose_antecedents (tree t)
 {
-  term_list::iterator iter = g.conclusions.begin();
-  term_list::iterator end = g.conclusions.end();
-  while (iter != end)
-    {
-      if (TREE_CODE (*iter) == DISJ_CONSTR)
-        iter = load_disjunction_conclusion (g.conclusions, iter);
-      else
-        ++iter;
-    }
+  formula f (t);
+  decompose_formula (f, left);
+  return f;
 }
 
-void
-load_conclusions (proof_state& p)
+/* Fully decomposing T into a list of sequents, each comprised of
+   a list of atomic constraints, as if T were a consequent.  */
+
+static formula
+decompose_consequents (tree t)
 {
-  proof_state::iterator iter = p.begin();
-  while (iter != p.end())
-    {
-      load_conclusions (*iter);
-      ++iter;
-    }
+  formula f (t);
+  decompose_formula (f, right);
+  return f;
 }
 
+static bool derive_proof (clause&, tree, rules);
 
-/*---------------------------------------------------------------------------
-                          High-level proof tactics
----------------------------------------------------------------------------*/
+/* Derive a proof of both operands of T.  */
 
-/* Given two constraints A and C, try to derive a proof that
-   A implies C.  */
+static bool
+derive_proof_for_both_operands (clause& c, tree t, rules r)
+{
+  if (!derive_proof (c, TREE_OPERAND (t, 0), r))
+    return false;
+  return derive_proof (c, TREE_OPERAND (t, 1), r);
+}
 
-bool
-prove_implication (tree a, tree c)
+/* Derive a proof of either operand of T.  */
+
+static bool
+derive_proof_for_either_operand (clause& c, tree t, rules r)
 {
-  /* Quick accept. */
-  if (cp_tree_equal (a, c))
+  if (derive_proof (c, TREE_OPERAND (t, 0), r))
     return true;
+  return derive_proof (c, TREE_OPERAND (t, 1), r);
+}
 
-  /* Build the initial proof state. */
-  proof_state proof;
-  proof_goal& goal = proof.front();
-  goal.assumptions.push_back(a);
-  goal.conclusions.push_back(c);
-
-  /* Perform an initial right-expansion in the off-chance that the right
-     hand side contains disjunctions. */
-  load_conclusions (proof);
+/* Derive a proof of the atomic constraint T in clause C.  */
 
-  int step_max = 1 << 10;
-  int step_count = 0;              /* FIXME: We shouldn't have this. */
-  std::size_t branch_limit = 1024; /* FIXME: This needs to be configurable. */
-  while (step_count < step_max && proof.size() < branch_limit)
-    {
-      /* Determine if we can prove that the assumptions entail the
-         conclusions. If so, we're done. */
-      load_assumptions (proof);
+static bool
+derive_atomic_proof (clause& c, tree t)
+{
+  return c.contains (t);
+}
 
-      /* Can we solve the proof based on this? */
-      proof_result r = check_proof (proof);
-      if (r != undecided)
-        return r == valid;
+/* Derive a proof of T from the terms in C.  */
 
-      /* If not, then we need to dig into disjunctions.  */
-      explode_assumptions (proof);
+static bool
+derive_proof (clause& c, tree t, rules r)
+{
+  switch (TREE_CODE (t))
+  {
+    case CONJ_CONSTR:
+      if (r == left)
+        return derive_proof_for_both_operands (c, t, r);
+      else
+       return derive_proof_for_either_operand (c, t, r);
+    case DISJ_CONSTR:
+      if (r == left)
+        return derive_proof_for_either_operand (c, t, r);
+      else
+       return derive_proof_for_both_operands (c, t, r);
+    default:
+      return derive_atomic_proof (c, t);
+  }
+}
 
-      ++step_count;
-    }
+/* Derive a proof of T from disjunctive clauses in F.  */
 
-  if (step_count == step_max)
-    error ("subsumption failed to resolve");
+static bool
+derive_proofs (formula& f, tree t, rules r)
+{
+  for (formula::iterator i = f.begin(); i != f.end(); ++i)
+    if (!derive_proof (*i, t, r))
+      return false;
+  return true;
+}
 
-  if (proof.size() == branch_limit)
-    error ("exceeded maximum number of branches");
+/* The largest number of clauses in CNF or DNF we accept as input
+   for subsumption. This an upper bound of 2^16 expressions.  */
+static int max_problem_size = 16;
 
+static inline bool
+diagnose_constraint_size (tree t)
+{
+  error_at (input_location, "%qE exceeds the maximum constraint complexity", t);
   return false;
 }
 
@@ -777,31 +784,51 @@ prove_implication (tree a, tree c)
    This is done by deriving a proof of the conclusions on the RIGHT
    from the assumptions on the LEFT assumptions.  */
 
-bool
-subsumes_constraints_nonnull (tree left, tree right)
+static bool
+subsumes_constraints_nonnull (tree lhs, tree rhs)
 {
-  gcc_assert (check_constraint_info (left));
-  gcc_assert (check_constraint_info (right));
-
   auto_timevar time (TV_CONSTRAINT_SUB);
-  tree a = CI_ASSOCIATED_CONSTRAINTS (left);
-  tree c = CI_ASSOCIATED_CONSTRAINTS (right);
-  return prove_implication (a, c);
-}
 
-} /* namespace */
+  int n1 = dnf_size (lhs);
+  int n2 = cnf_size (rhs);
+
+  /* Make sure we haven't exceeded the largest acceptable problem.  */
+  if (std::min (n1, n2) >= max_problem_size)
+    {
+      if (n1 < n2)
+        diagnose_constraint_size (lhs);
+      else
+       diagnose_constraint_size (rhs);
+      return false;
+    }
+
+  /* Decompose the smaller of the two formulas, and recursively
+     check the implication using the larger.  Note that for
+     constraints that are largely comprised of conjunctions the
+     it will usually be the case that n1 <= n2. */
+  if (n1 <= n2)
+    {
+      formula dnf = decompose_antecedents (lhs);
+      return derive_proofs (dnf, rhs, left);
+    }
+  else
+    {
+      formula cnf = decompose_consequents (rhs);
+      return derive_proofs (cnf, lhs, right);
+    }
+}
 
 /* Returns true if the LEFT constraints subsume the RIGHT
    constraints.  */
 
 bool
-subsumes (tree left, tree right)
+subsumes (tree lhs, tree rhs)
 {
-  if (left == right)
+  if (lhs == rhs)
     return true;
-  if (!left)
+  if (!lhs || lhs == error_mark_node)
     return false;
-  if (!right)
+  if (!rhs || rhs == error_mark_node)
     return true;
-  return subsumes_constraints_nonnull (left, right);
+  return subsumes_constraints_nonnull (lhs, rhs);
 }
index ff6d5ee698443122ee65684afb7bf618819ee810..57ab129c9ecf8ff5ca38535152d388e048b61085 100644 (file)
@@ -2268,7 +2268,13 @@ diagnose_name_conflict (tree decl, tree bval)
       && (TREE_CODE (decl) != TYPE_DECL
          || DECL_ARTIFICIAL (decl) == DECL_ARTIFICIAL (bval))
       && CP_DECL_CONTEXT (decl) == CP_DECL_CONTEXT (bval))
-    error ("redeclaration of %q#D", decl);
+    {
+      if (concept_definition_p (decl))
+        error ("redeclaration of %q#D with different template parameters",
+               decl);
+      else
+        error ("redeclaration of %q#D", decl);
+    }
   else
     error ("%q#D conflicts with a previous declaration", decl);
 
@@ -2334,6 +2340,9 @@ matching_fn_p (tree one, tree two)
        return false;
     }
 
+  if (!equivalently_constrained (one, two))
+    return false;
+
   return true;
 }
 
index c61e0b2cb9d0cbee372610f45041ba5f68beb284..b6e738c67d60d0fdd867cbe533b6c138c75991cc 100644 (file)
@@ -169,6 +169,7 @@ enum required_token {
   RT_TRY, /* try */
   RT_CATCH, /* catch */
   RT_THROW, /* throw */
+  RT_AUTO, /* auto */
   RT_LABEL, /* __label__ */
   RT_AT_TRY, /* @try */
   RT_AT_SYNCHRONIZED, /* @synchronized */
@@ -2196,6 +2197,8 @@ static tree cp_parser_type_specifier
    int *, bool *);
 static tree cp_parser_simple_type_specifier
   (cp_parser *, cp_decl_specifier_seq *, cp_parser_flags);
+static tree cp_parser_placeholder_type_specifier
+  (cp_parser *, location_t, tree, bool);
 static tree cp_parser_type_name
   (cp_parser *, bool);
 static tree cp_parser_nonclass_name
@@ -2369,6 +2372,8 @@ static tree cp_parser_type_parameter
   (cp_parser *, bool *);
 static tree cp_parser_template_id
   (cp_parser *, bool, bool, enum tag_types, bool);
+static tree cp_parser_template_id_expr
+  (cp_parser *, bool, bool, bool);
 static tree cp_parser_template_name
   (cp_parser *, bool, bool, bool, enum tag_types, bool *);
 static tree cp_parser_template_argument_list
@@ -2444,7 +2449,9 @@ static void cp_parser_label_declaration
 
 /* Concept Extensions */
 
-static tree cp_parser_requires_clause
+static tree cp_parser_concept_definition
+  (cp_parser *);
+static tree cp_parser_constraint_expression
   (cp_parser *);
 static tree cp_parser_requires_clause_opt
   (cp_parser *);
@@ -2454,7 +2461,7 @@ static tree cp_parser_requirement_parameter_list
   (cp_parser *);
 static tree cp_parser_requirement_body
   (cp_parser *);
-static tree cp_parser_requirement_list
+static tree cp_parser_requirement_seq
   (cp_parser *);
 static tree cp_parser_requirement
   (cp_parser *);
@@ -2687,11 +2694,6 @@ static bool cp_parser_init_statement_p
 static bool cp_parser_skip_to_closing_square_bracket
   (cp_parser *);
 
-/* Concept-related syntactic transformations */
-
-static tree cp_parser_maybe_concept_name       (cp_parser *, tree);
-static tree cp_parser_maybe_partial_concept_id (cp_parser *, tree, tree);
-
 // -------------------------------------------------------------------------- //
 // Unevaluated Operand Guard
 //
@@ -4874,6 +4876,8 @@ class token_pair
                              m_open_loc);
   }
 
+  location_t open_location () const { return m_open_loc; }
+
  private:
   location_t m_open_loc;
 };
@@ -4948,7 +4952,7 @@ cp_parser_statement_expr (cp_parser *parser)
 
    This returns the tree code corresponding to the matched operator
    as an int. When the current token matches a compound assignment
-   opertor, the resulting tree code is the negative value of the
+   operator, the resulting tree code is the negative value of the
    non-assignment operator. */
 
 static int
@@ -5904,11 +5908,10 @@ cp_parser_id_expression (cp_parser *parser,
 
       cp_parser_parse_tentatively (parser);
       /* Try a template-id.  */
-      id = cp_parser_template_id (parser,
-                                 /*template_keyword_p=*/false,
-                                 /*check_dependency_p=*/true,
-                                 none_type,
-                                 declarator_p);
+      id = cp_parser_template_id_expr (parser,
+                                      /*template_keyword_p=*/false,
+                                      /*check_dependency_p=*/true,
+                                      declarator_p);
       /* If that worked, we're done.  */
       if (cp_parser_parse_definitely (parser))
        return id;
@@ -5983,10 +5986,9 @@ cp_parser_unqualified_id (cp_parser* parser,
           template-id.  */
        cp_parser_parse_tentatively (parser);
        /* Try a template-id.  */
-       id = cp_parser_template_id (parser, template_keyword_p,
-                                   check_dependency_p,
-                                   none_type,
-                                   declarator_p);
+       id = cp_parser_template_id_expr (parser, template_keyword_p,
+                                        check_dependency_p,
+                                        declarator_p);
        /* If it worked, we're done.  */
        if (cp_parser_parse_definitely (parser))
          return id;
@@ -5995,10 +5997,9 @@ cp_parser_unqualified_id (cp_parser* parser,
       }
 
     case CPP_TEMPLATE_ID:
-      return cp_parser_template_id (parser, template_keyword_p,
-                                   check_dependency_p,
-                                   none_type,
-                                   declarator_p);
+      return cp_parser_template_id_expr (parser, template_keyword_p,
+                                        check_dependency_p,
+                                        declarator_p);
 
     case CPP_COMPL:
       {
@@ -6239,10 +6240,9 @@ cp_parser_unqualified_id (cp_parser* parser,
          /* This could be a template-id, so we try that first.  */
          cp_parser_parse_tentatively (parser);
          /* Try a template-id.  */
-         id = cp_parser_template_id (parser, template_keyword_p,
-                                     /*check_dependency_p=*/true,
-                                     none_type,
-                                     declarator_p);
+         id = cp_parser_template_id_expr (parser, template_keyword_p,
+                                          /*check_dependency_p=*/true,
+                                          declarator_p);
          /* If that worked, we're done.  */
          if (cp_parser_parse_definitely (parser))
            return id;
@@ -9638,6 +9638,8 @@ cp_parser_binary_expression (cp_parser* parser, bool cast_p,
            current.lhs = error_mark_node;
          else
            {
+             current.lhs.maybe_add_location_wrapper ();
+             rhs.maybe_add_location_wrapper ();
              current.lhs
                = build_min (current.tree_type,
                             TREE_CODE_CLASS (current.tree_type)
@@ -14059,10 +14061,26 @@ cp_parser_decl_specifier_seq (cp_parser* parser,
         case RID_CONCEPT:
           ds = ds_concept;
           cp_lexer_consume_token (parser->lexer);
+
+          /* Warn for concept as a decl-specifier. We'll rewrite these as
+             concept declarations later.  */
+          if (!flag_concepts_ts)
+            {
+             cp_token *next = cp_lexer_peek_token (parser->lexer);
+             if (next->keyword == RID_BOOL)
+               pedwarn (next->location, 0, "the %<bool%> keyword is not "
+                        "allowed in a C++20 concept definition");
+             else
+               pedwarn (token->location, 0, "C++20 concept definition syntax "
+                        "is %<concept <name> = <expr>%> ");
+            }
+
          /* In C++20 a concept definition is just 'concept name = expr;'
-            Support that syntax by pretending we've seen 'bool'.  */
+            Support that syntax as a TS extension by pretending we've seen
+            the 'bool' specifier.  */
          if (cp_lexer_next_token_is (parser->lexer, CPP_NAME)
-             && cp_lexer_nth_token_is (parser->lexer, 2, CPP_EQ))
+             && cp_lexer_nth_token_is (parser->lexer, 2, CPP_EQ)
+             && !decl_specs->any_type_specifiers_p)
            {
              cp_parser_set_decl_spec_type (decl_specs, boolean_type_node,
                                            token, /*type_definition*/false);
@@ -15800,15 +15818,15 @@ get_unqualified_id (cp_declarator *declarator)
     return NULL_TREE;
 }
 
-/* Returns true if DECL represents a constrained-parameter.  */
+/* Returns true if TYPE would declare a constrained constrained-parameter.  */
 
 static inline bool
-is_constrained_parameter (tree decl)
+is_constrained_parameter (tree type)
 {
-  return (decl
-          && TREE_CODE (decl) == TYPE_DECL
-          && CONSTRAINED_PARM_CONCEPT (decl)
-          && DECL_P (CONSTRAINED_PARM_CONCEPT (decl)));
+  return (type
+          && TREE_CODE (type) == TYPE_DECL
+          && CONSTRAINED_PARM_CONCEPT (type)
+          && DECL_P (CONSTRAINED_PARM_CONCEPT (type)));
 }
 
 /* Returns true if PARM declares a constrained-parameter. */
@@ -15894,8 +15912,8 @@ cp_parser_constrained_template_template_parm (cp_parser *parser,
    declarator.  */
 
 static tree
-constrained_non_type_template_parm (bool *is_non_type,
-                                    cp_parameter_declarator *parm)
+cp_parser_constrained_non_type_template_parm (bool *is_non_type,
+                                             cp_parameter_declarator *parm)
 {
   *is_non_type = true;
   cp_declarator *decl = parm->declarator;
@@ -15912,20 +15930,13 @@ constrained_non_type_template_parm (bool *is_non_type,
 static tree
 finish_constrained_parameter (cp_parser *parser,
                               cp_parameter_declarator *parmdecl,
-                              bool *is_non_type,
-                              bool *is_parameter_pack)
+                              bool *is_non_type)
 {
   tree decl = parmdecl->decl_specifiers.type;
   tree id = get_unqualified_id (parmdecl->declarator);
   tree def = parmdecl->default_argument;
   tree proto = DECL_INITIAL (decl);
 
-  /* A template parameter constrained by a variadic concept shall also
-     be declared as a template parameter pack.  */
-  bool is_variadic = template_parameter_pack_p (proto);
-  if (is_variadic && !*is_parameter_pack)
-    cp_parser_error (parser, "variadic constraint introduced without %<...%>");
-
   /* Build the parameter. Return an error if the declarator was invalid. */
   tree parm;
   if (TREE_CODE (proto) == TYPE_DECL)
@@ -15934,7 +15945,7 @@ finish_constrained_parameter (cp_parser *parser,
     parm = cp_parser_constrained_template_template_parm (parser, proto, id,
                                                         parmdecl);
   else
-    parm = constrained_non_type_template_parm (is_non_type, parmdecl);
+    parm = cp_parser_constrained_non_type_template_parm (is_non_type, parmdecl);
   if (parm == error_mark_node)
     return error_mark_node;
 
@@ -15949,14 +15960,13 @@ finish_constrained_parameter (cp_parser *parser,
 /* Returns true if the parsed type actually represents the declaration
    of a type template-parameter.  */
 
-static inline bool
+static bool
 declares_constrained_type_template_parameter (tree type)
 {
   return (is_constrained_parameter (type)
          && TREE_CODE (TREE_TYPE (type)) == TEMPLATE_TYPE_PARM);
 }
 
-
 /* Returns true if the parsed type actually represents the declaration of
    a template template-parameter.  */
 
@@ -16132,12 +16142,11 @@ cp_parser_template_parameter (cp_parser* parser, bool *is_non_type,
        cp_lexer_consume_token (parser->lexer);
     }
 
-  // The parameter may have been constrained.
+  /* The parameter may have been constrained type parameter.  */
   if (is_constrained_parameter (parameter_declarator))
     return finish_constrained_parameter (parser,
                                          parameter_declarator,
-                                         is_non_type,
-                                         is_parameter_pack);
+                                         is_non_type);
 
   // Now we're sure that the parameter is a non-type parameter.
   *is_non_type = true;
@@ -16263,8 +16272,8 @@ cp_parser_type_parameter (cp_parser* parser, bool *is_parameter_pack)
        if (flag_concepts)
           {
            tree reqs = get_shorthand_constraints (current_template_parms);
-           if (tree r = cp_parser_requires_clause_opt (parser))
-              reqs = conjoin_constraints (reqs, normalize_expression (r));
+           if (tree dreqs = cp_parser_requires_clause_opt (parser))
+              reqs = combine_constraint_expressions (reqs, dreqs);
            TEMPLATE_PARMS_CONSTRAINTS (current_template_parms) = reqs;
           }
 
@@ -16370,6 +16379,7 @@ cp_parser_template_id (cp_parser *parser,
   /* If the next token corresponds to a template-id, there is no need
      to reparse it.  */
   cp_token *token = cp_lexer_peek_token (parser->lexer);
+
   if (token->type == CPP_TEMPLATE_ID)
     {
       cp_lexer_consume_token (parser->lexer);
@@ -16512,7 +16522,7 @@ cp_parser_template_id (cp_parser *parser,
     = make_location (token->location, token->location, parser->lexer);
 
   /* Check for concepts autos where they don't belong.  We could
-     identify types in some cases of idnetifier TEMPL, looking ahead
+     identify types in some cases of identifier TEMPL, looking ahead
      for a CPP_SCOPE, but that would buy us nothing: we accept auto in
      types.  We reject them in functions, but if what we have is an
      identifier, even with none_type we can't conclude it's NOT a
@@ -16538,11 +16548,13 @@ cp_parser_template_id (cp_parser *parser,
       template_id
        = finish_template_type (templ, arguments, entering_scope);
     }
-  /* A template-like identifier may be a partial concept id. */
-  else if (flag_concepts
-           && (template_id = (cp_parser_maybe_partial_concept_id
-                             (parser, templ, arguments))))
-    return template_id;
+  else if (concept_definition_p (templ))
+    {
+      /* The caller will decide whether this is a concept check or type
+        constraint.  */
+      template_id = build2_loc (combined_loc, TEMPLATE_ID_EXPR,
+                               boolean_type_node, templ, arguments);
+    }
   else if (variable_template_p (templ))
     {
       template_id = lookup_template_variable (templ, arguments);
@@ -16599,6 +16611,23 @@ cp_parser_template_id (cp_parser *parser,
   return template_id;
 }
 
+/* Like cp_parser_template_id, called in non-type context.  */
+
+static tree
+cp_parser_template_id_expr (cp_parser *parser,
+                           bool template_keyword_p,
+                           bool check_dependency_p,
+                           bool is_declaration)
+{
+  tree x = cp_parser_template_id (parser, template_keyword_p, check_dependency_p,
+                                 none_type, is_declaration);
+  if (TREE_CODE (x) == TEMPLATE_ID_EXPR
+      && concept_check_p (x))
+    /* We didn't check the arguments in cp_parser_template_id; do that now.  */
+    return build_concept_id (x);
+  return x;
+}
+
 /* Parse a template-name.
 
    template-name:
@@ -17019,11 +17048,7 @@ cp_parser_template_argument (cp_parser* parser)
                                          /*check_dependency=*/true,
                                          /*ambiguous_decls=*/NULL,
                                          argument_start_token->location);
-      /* Handle a constrained-type-specifier for a non-type template
-        parameter.  */
-      if (tree decl = cp_parser_maybe_concept_name (parser, argument))
-       argument = decl;
-      else if (TREE_CODE (argument) != TEMPLATE_DECL
+      if (TREE_CODE (argument) != TEMPLATE_DECL
               && TREE_CODE (argument) != UNBOUND_CLASS_TEMPLATE)
        cp_parser_error (parser, "expected template-name");
     }
@@ -17772,7 +17797,7 @@ cp_parser_simple_type_specifier (cp_parser* parser,
          else if (!flag_concepts)
            pedwarn (token->location, 0,
                     "use of %<auto%> in parameter declaration "
-                    "only available with %<-fconcepts%>");
+                    "only available with %<-fconcepts-ts%>");
        }
       else
        type = make_auto ();
@@ -17888,6 +17913,10 @@ cp_parser_simple_type_specifier (cp_parser* parser,
       if (flags & CP_PARSER_FLAGS_OPTIONAL)
        cp_parser_parse_tentatively (parser);
 
+      /* Remember current tentative parsing state -- if we know we need
+        a type, we can give better diagnostics here.  */
+      bool tent = cp_parser_parsing_tentatively (parser);
+
       token = cp_lexer_peek_token (parser->lexer);
 
       /* Look for the optional `::' operator.  */
@@ -17942,13 +17971,44 @@ cp_parser_simple_type_specifier (cp_parser* parser,
            type = NULL_TREE;
        }
 
+      if (!type && flag_concepts && decl_specs)
+       {
+         /* Try for a type-constraint with template arguments.  We check
+            decl_specs here to avoid trying this for a functional cast.  */
+
+         cp_parser_parse_tentatively (parser);
+
+         type = cp_parser_template_id (parser,
+                                       /*template_keyword_p=*/false,
+                                       /*check_dependency_p=*/true,
+                                       none_type,
+                                       /*is_declaration=*/false);
+         if (type && concept_check_p (type))
+           {
+             location_t loc = EXPR_LOCATION (type);
+             type = cp_parser_placeholder_type_specifier (parser, loc,
+                                                          type, tent);
+             if (tent && type == error_mark_node)
+               /* Perhaps it's a concept-check expression.  */
+               cp_parser_simulate_error (parser);
+           }
+         else
+           cp_parser_simulate_error (parser);
+
+         if (!cp_parser_parse_definitely (parser))
+           type = NULL_TREE;
+       }
+
       if (!type && cxx_dialect >= cxx17)
        {
-         /* Try class template argument deduction.  */
+         /* Try class template argument deduction or type-constraint without
+            template arguments.  */
          tree name = cp_parser_identifier (parser);
          if (name && TREE_CODE (name) == IDENTIFIER_NODE
              && parser->scope != error_mark_node)
            {
+             location_t loc
+               = cp_lexer_previous_token (parser->lexer)->location;
              tree tmpl = cp_parser_lookup_name (parser, name,
                                                 none_type,
                                                 /*is_template=*/false,
@@ -17960,6 +18020,9 @@ cp_parser_simple_type_specifier (cp_parser* parser,
                  && (DECL_CLASS_TEMPLATE_P (tmpl)
                      || DECL_TEMPLATE_TEMPLATE_PARM_P (tmpl)))
                type = make_template_placeholder (tmpl);
+             else if (flag_concepts && tmpl && concept_definition_p (tmpl))
+               type = cp_parser_placeholder_type_specifier (parser, loc,
+                                                            tmpl, tent);
              else
                {
                  type = error_mark_node;
@@ -18031,6 +18094,140 @@ cp_parser_simple_type_specifier (cp_parser* parser,
   return type;
 }
 
+/* Parse the remainder of a placholder-type-specifier.
+
+   placeholder-type-specifier:
+     type-constraint_opt auto
+     type-constraint_opt decltype(auto)
+
+  The raw form of the constraint is parsed in cp_parser_simple_type_specifier
+  and passed as TMPL. This function converts TMPL to an actual type-constraint,
+  parses the placeholder type, and performs some contextual syntactic analysis.
+
+  LOC provides the location of the template name.
+
+  TENTATIVE is true if the type-specifier parsing is tentative; in that case,
+  don't give an error if TMPL isn't a valid type-constraint, as the template-id
+  might actually be a concept-check,
+
+  Note that the Concepts TS allows the auto or decltype(auto) to be
+  omitted in a constrained-type-specifier.  */
+
+tree
+cp_parser_placeholder_type_specifier (cp_parser *parser, location_t loc,
+                                     tree tmpl, bool tentative)
+{
+  if (tmpl == error_mark_node)
+    return error_mark_node;
+
+  tree orig_tmpl = tmpl;
+
+  /* Get the arguments as written for subsequent analysis.  */
+  tree args = NULL_TREE;
+  if (TREE_CODE (tmpl) == TEMPLATE_ID_EXPR)
+    {
+      args = TREE_OPERAND (tmpl, 1);
+      tmpl = TREE_OPERAND (tmpl, 0);
+    }
+  if (args == NULL_TREE)
+    /* A concept-name with no arguments can't be an expression.  */
+    tentative = false;
+
+  tsubst_flags_t complain = tentative ? tf_none : tf_warning_or_error;
+
+  /* Get the concept and prototype parameter for the constraint.  */
+  tree_pair info = finish_type_constraints (tmpl, args, complain);
+  tree con = info.first;
+  tree proto = info.second;
+  if (con == error_mark_node)
+    return error_mark_node;
+
+  /* As per the standard, require auto or decltype(auto), except in some
+     cases (template parameter lists, -fconcepts-ts enabled).  */
+  cp_token *placeholder = NULL, *open_paren = NULL, *close_paren = NULL;
+  if (cxx_dialect >= cxx2a)
+    {
+      if (cp_lexer_next_token_is_keyword (parser->lexer, RID_AUTO))
+       placeholder = cp_lexer_consume_token (parser->lexer);
+      else if (cp_lexer_next_token_is_keyword (parser->lexer, RID_DECLTYPE))
+       {
+         placeholder = cp_lexer_consume_token (parser->lexer);
+         open_paren = cp_parser_require (parser, CPP_OPEN_PAREN,
+                                         RT_OPEN_PAREN);
+         cp_parser_require_keyword (parser, RID_AUTO, RT_AUTO);
+          close_paren = cp_parser_require (parser, CPP_CLOSE_PAREN,
+                                          RT_CLOSE_PAREN,
+                                          open_paren->location);
+       }
+    }
+
+  /* A type constraint constrains a contextually determined type or type
+     parameter pack. However, the the Concepts TS does allow concepts
+     to introduce non-type and template template parameters.  */
+  if (TREE_CODE (proto) != TYPE_DECL)
+    {
+      if (!flag_concepts_ts
+         || !processing_template_parmlist)
+       {
+         error_at (loc, "%qE does not constrain a type", DECL_NAME (con));
+         inform (DECL_SOURCE_LOCATION (con), "concept defined here");
+         return error_mark_node;
+       }
+    }
+
+  /* In a template parameter list, a type-parameter can be introduced
+     by type-constraints alone.  */
+  if (processing_template_parmlist && !placeholder)
+    return build_constrained_parameter (con, proto, args);
+
+  /* Diagnose issues placeholder issues.  */
+  if (!flag_concepts_ts
+      && !parser->in_result_type_constraint_p
+      && !placeholder)
+    {
+      tree id = build_nt (TEMPLATE_ID_EXPR, tmpl, args);
+      tree expr = DECL_P (orig_tmpl) ? DECL_NAME (con) : id;
+      error_at (input_location,
+               "expected %<auto%> or %<decltype(auto)%> after %qE", expr);
+      /* Fall through. This is an error of omission.  */
+    }
+  else if (parser->in_result_type_constraint_p && placeholder)
+    {
+      /* A trailing return type only allows type-constraints.  */
+      error_at (input_location,
+               "unexpected placeholder in constrained result type");
+    }
+
+  /* In a parameter-declaration-clause, a placeholder-type-specifier
+     results in an invented template parameter.  */
+  if (parser->auto_is_implicit_function_template_parm_p)
+    {
+      if (placeholder && token_is_decltype (placeholder))
+       {
+         location_t loc = make_location (placeholder->location,
+                                         placeholder->location,
+                                         close_paren->location);
+         error_at (loc, "cannot declare a parameter with %<decltype(auto)%>");
+         return error_mark_node;
+       }
+      tree parm = build_constrained_parameter (con, proto, args);
+      return synthesize_implicit_template_parm (parser, parm);
+    }
+
+  /* Determine if the type should be deduced using template argument
+     deduction or decltype deduction. Note that the latter is always
+     used for type-constraints in trailing return types.  */
+  bool decltype_p = placeholder
+    ? placeholder->keyword == RID_DECLTYPE
+    : parser->in_result_type_constraint_p;
+
+  /* Otherwise, this is the type of a variable or return type.  */
+  if (decltype_p)
+    return make_constrained_decltype_auto (con, args);
+  else
+    return make_constrained_auto (con, args);
+}
+
 /* Parse a type-name.
 
    type-name:
@@ -18103,8 +18300,6 @@ cp_parser_type_name (cp_parser* parser, bool typename_keyword_p)
          && TREE_CODE (type_decl) == TYPE_DECL
          && TYPE_DECL_ALIAS_P (type_decl))
        gcc_assert (DECL_TEMPLATE_INSTANTIATION (type_decl));
-      else if (is_constrained_parameter (type_decl))
-        /* Don't do anything. */ ;
       else
        cp_parser_simulate_error (parser);
 
@@ -18116,105 +18311,6 @@ cp_parser_type_name (cp_parser* parser, bool typename_keyword_p)
   return type_decl;
 }
 
-/*  Check if DECL and ARGS can form a constrained-type-specifier.
-    If ARGS is non-null, we try to form a concept check of the
-    form DECL<?, ARGS> where ? is a wildcard that matches any
-    kind of template argument. If ARGS is NULL, then we try to
-    form a concept check of the form DECL<?>. */
-
-static tree
-cp_parser_maybe_constrained_type_specifier (cp_parser *parser,
-                                           tree decl, tree args)
-{
-  gcc_assert (args ? TREE_CODE (args) == TREE_VEC : true);
-
-  /* If we a constrained-type-specifier cannot be deduced. */
-  if (parser->prevent_constrained_type_specifiers)
-    return NULL_TREE;
-
-  /* A constrained type specifier can only be found in an
-     overload set or as a reference to a template declaration.
-
-     FIXME: This might be masking a bug.  It's possible that
-     that the deduction below is causing template specializations
-     to be formed with the wildcard as an argument.  */
-  if (TREE_CODE (decl) != OVERLOAD && TREE_CODE (decl) != TEMPLATE_DECL)
-    return NULL_TREE;
-
-  /* Try to build a call expression that evaluates the
-     concept. This can fail if the overload set refers
-     only to non-templates. */
-  tree placeholder = build_nt (WILDCARD_DECL);
-  tree check = build_concept_check (decl, placeholder, args);
-  if (check == error_mark_node)
-    return NULL_TREE;
-
-  /* Deduce the checked constraint and the prototype parameter.
-
-     FIXME: In certain cases, failure to deduce should be a
-     diagnosable error.  */
-  tree conc;
-  tree proto;
-  if (!deduce_constrained_parameter (check, conc, proto))
-    return NULL_TREE;
-
-  /* In template parameter scope, this results in a constrained
-     parameter. Return a descriptor of that parm. */
-  if (processing_template_parmlist)
-    return build_constrained_parameter (conc, proto, args);
-
-  /* In a parameter-declaration-clause, constrained-type
-     specifiers result in invented template parameters.  */
-  if (parser->auto_is_implicit_function_template_parm_p)
-    {
-      tree x = build_constrained_parameter (conc, proto, args);
-      return synthesize_implicit_template_parm (parser, x);
-    }
-  else
-    {
-     /* Otherwise, we're in a context where the constrained
-        type name is deduced and the constraint applies
-        after deduction. */
-      return make_constrained_auto (conc, args);
-    }
-
-  return NULL_TREE;
-}
-
-/* If DECL refers to a concept, return a TYPE_DECL representing
-   the result of using the constrained type specifier in the
-   current context.  DECL refers to a concept if
-
-  - it is an overload set containing a function concept taking a single
-    type argument, or
-
-  - it is a variable concept taking a single type argument.  */
-
-static tree
-cp_parser_maybe_concept_name (cp_parser* parser, tree decl)
-{
-  if (flag_concepts
-      && (TREE_CODE (decl) == OVERLOAD
-         || BASELINK_P (decl)
-         || variable_concept_p (decl)))
-    return cp_parser_maybe_constrained_type_specifier (parser, decl, NULL_TREE);
-  else
-    return NULL_TREE;
-}
-
-/* Check if DECL and ARGS form a partial-concept-id.  If so,
-   assign ID to the resulting constrained placeholder.
-
-   Returns true if the partial-concept-id designates a placeholder
-   and false otherwise. Note that *id is set to NULL_TREE in
-   this case. */
-
-static tree
-cp_parser_maybe_partial_concept_id (cp_parser *parser, tree decl, tree args)
-{
-  return cp_parser_maybe_constrained_type_specifier (parser, decl, args);
-}
-
 /* Parse a non-class type-name, that is, either an enum-name, a typedef-name,
    or a concept-name.
 
@@ -18245,10 +18341,6 @@ cp_parser_nonclass_name (cp_parser* parser)
 
   type_decl = strip_using_decl (type_decl);
 
-  /* If we found an overload set, then it may refer to a concept-name. */
-  if (tree decl = cp_parser_maybe_concept_name (parser, type_decl))
-    type_decl = decl;
-
   if (TREE_CODE (type_decl) != TYPE_DECL
       && (objc_is_id (identifier) || objc_is_class_name (identifier)))
     {
@@ -26885,31 +26977,280 @@ cp_parser_label_declaration (cp_parser* parser)
   cp_parser_require (parser, CPP_SEMICOLON, RT_SEMICOLON);
 }
 
+// -------------------------------------------------------------------------- //
+// Concept definitions
+
+static tree
+cp_parser_concept_definition (cp_parser *parser)
+{
+  gcc_assert (cp_lexer_next_token_is_keyword (parser->lexer, RID_CONCEPT));
+  cp_lexer_consume_token (parser->lexer);
+
+  cp_expr id = cp_parser_identifier (parser);
+  if (id == error_mark_node)
+    {
+      cp_parser_skip_to_end_of_statement (parser);
+      cp_parser_consume_semicolon_at_end_of_statement (parser);
+      return NULL_TREE;
+    }
+
+  if (!cp_parser_require (parser, CPP_EQ, RT_EQ))
+    {
+      cp_parser_skip_to_end_of_statement (parser);
+      cp_parser_consume_semicolon_at_end_of_statement (parser);
+      return error_mark_node;
+    }
+
+  processing_constraint_expression_sentinel parsing_constraint;
+  tree init = cp_parser_constraint_expression (parser);
+  if (init == error_mark_node)
+    cp_parser_skip_to_end_of_statement (parser);
+
+  /* Consume the trailing ';'. Diagnose the problem if it isn't there,
+     but continue as if it were.  */
+  cp_parser_consume_semicolon_at_end_of_statement (parser);
+
+  return finish_concept_definition (id, init);
+}
+
 // -------------------------------------------------------------------------- //
 // Requires Clause
 
-// Parse a requires clause.
-//
-//    requires-clause:
-//      'requires' logical-or-expression
-//
-// The required logical-or-expression must be a constant expression. Note
-// that we don't check that the expression is constepxr here. We defer until
-// we analyze constraints and then, we only check atomic constraints.
+/* Diagnose an expression that should appear in ()'s within a requires-clause
+   and suggest where to place those parentheses.  */
+
+static void
+cp_parser_diagnose_ungrouped_constraint_plain (location_t loc)
+{
+  error_at (loc, "expression after %<requires%> must be enclosed "
+                "in parentheses");
+}
+
+static void
+cp_parser_diagnose_ungrouped_constraint_rich (location_t loc)
+{
+  gcc_rich_location richloc (loc);
+  richloc.add_fixit_insert_before ("(");
+  richloc.add_fixit_insert_after (")");
+  error_at (&richloc, "expression after %<requires%> must be enclosed "
+                     "in parentheses");
+}
+
+/* Parse a primary expression within a constraint.  */
+
+static cp_expr
+cp_parser_constraint_primary_expression (cp_parser *parser)
+{
+  cp_parser_parse_tentatively (parser);
+  cp_id_kind idk;
+  location_t loc = input_location;
+  cp_expr expr = cp_parser_primary_expression (parser,
+                                              /*address_p=*/false,
+                                              /*cast_p=*/false,
+                                              /*template_arg_p=*/false,
+                                              &idk);
+  expr.maybe_add_location_wrapper ();
+  if (expr != error_mark_node)
+    expr = finish_constraint_primary_expr (expr);
+  if (cp_parser_parse_definitely (parser))
+    return expr;
+
+  /* Retry the parse at a lower precedence. If that succeeds, diagnose the
+     error, but return the expression as if it were valid.  */
+  cp_parser_parse_tentatively (parser);
+  expr = cp_parser_simple_cast_expression (parser);
+  if (cp_parser_parse_definitely (parser))
+    {
+      cp_parser_diagnose_ungrouped_constraint_rich (expr.get_location());
+      return expr;
+    }
+
+  /* Otherwise, something has gone wrong, but we can't generate a more
+     meaningful diagnostic or recover.  */
+  cp_parser_diagnose_ungrouped_constraint_plain (loc);
+  return error_mark_node;
+}
+
+/* Examine the token following EXPR. If it is an operator in a non-logical
+   binary expression, diagnose that as an error. Returns ERROR_MARK_NODE.  */
+
+static cp_expr
+cp_parser_check_non_logical_constraint (cp_parser *parser, cp_expr lhs)
+{
+  cp_token *token = cp_lexer_peek_token (parser->lexer);
+  switch (token->type)
+    {
+      default:
+        return lhs;
+
+      /* Arithmetic operators.  */
+      case CPP_PLUS:
+      case CPP_MINUS:
+      case CPP_MULT:
+      case CPP_DIV:
+      case CPP_MOD:
+      /* Bitwise operators.  */
+      case CPP_AND:
+      case CPP_OR:
+      case CPP_XOR:
+      case CPP_RSHIFT:
+      case CPP_LSHIFT:
+      /* Relational operators.  */
+      /* FIXME: Handle '<=>'.  */
+      case CPP_EQ_EQ:
+      case CPP_NOT_EQ:
+      case CPP_LESS:
+      case CPP_GREATER:
+      case CPP_LESS_EQ:
+      case CPP_GREATER_EQ:
+      /* Conditional operator */
+      case CPP_QUERY:
+      /* Pointer-to-member.  */
+      case CPP_DOT_STAR:
+      case CPP_DEREF_STAR:
+      /* Assignment operators.  */
+      case CPP_PLUS_EQ:
+      case CPP_MINUS_EQ:
+      case CPP_MULT_EQ:
+      case CPP_DIV_EQ:
+      case CPP_MOD_EQ:
+      case CPP_AND_EQ:
+      case CPP_OR_EQ:
+      case CPP_XOR_EQ:
+      case CPP_RSHIFT_EQ:
+      case CPP_LSHIFT_EQ:
+        break;
+
+      case CPP_EQ: {
+       /* An equal sign may be part of the the definition of a function,
+          and not an assignment operator, when parsing the expression
+          for a trailing requires-clause. For example:
+
+             template<typename T>
+             struct S {
+               S() requires C<T> = default;
+             }
+
+          This is not an error.  */
+       if (cp_lexer_nth_token_is_keyword (parser->lexer, 2, RID_DELETE)
+           || cp_lexer_nth_token_is_keyword (parser->lexer, 2, RID_DEFAULT))
+         return lhs;
+
+        break;
+      }
+   }
+
+   /* Try to parse the RHS as either the remainder of a conditional-expression
+      or a logical-or-expression so we can form a good diagnostic.  */
+  cp_parser_parse_tentatively (parser);
+  cp_expr rhs;
+  if (token->type == CPP_QUERY)
+    rhs = cp_parser_question_colon_clause (parser, lhs);
+  else
+    {
+      cp_lexer_consume_token (parser->lexer);
+      rhs = cp_parser_binary_expression (parser, false, false, false,
+                                        PREC_NOT_OPERATOR, NULL);
+    }
+
+  /* If we couldn't parse the RHS, then emit the best diagnostic we can.  */
+  if (!cp_parser_parse_definitely (parser))
+    {
+      cp_parser_diagnose_ungrouped_constraint_plain (token->location);
+      return error_mark_node;
+    }
+
+  /* Otherwise, emit a fixit for the complete binary expression.  */
+  location_t loc = make_location (token->location,
+                                 lhs.get_start(),
+                                 rhs.get_finish());
+  cp_parser_diagnose_ungrouped_constraint_rich (loc);
+  return error_mark_node;
+}
+
+/* Parse a constraint-logical-and-expression.
+
+     constraint-logical-and-expression:
+       primary-expression
+       constraint-logical-and-expression '&&' primary-expression  */
+
+static cp_expr
+cp_parser_constraint_logical_and_expression (cp_parser *parser)
+{
+  cp_expr lhs = cp_parser_constraint_primary_expression (parser);
+  while (cp_lexer_next_token_is (parser->lexer, CPP_AND_AND))
+    {
+      cp_token *op = cp_lexer_consume_token (parser->lexer);
+      tree rhs = cp_parser_constraint_primary_expression (parser);
+      lhs = finish_constraint_and_expr (op->location, lhs, rhs);
+    }
+  return cp_parser_check_non_logical_constraint (parser, lhs);
+}
+
+/* Parse a constraint-logical-or-expression.
+
+     constraint-logical-or-expression:
+       constraint-logical-and-expression
+       constraint-logical-or-expression '||' constraint-logical-and-expression  */
+
+static cp_expr
+cp_parser_constraint_logical_or_expression (cp_parser *parser)
+{
+  cp_expr lhs = cp_parser_constraint_logical_and_expression (parser);
+  while (cp_lexer_next_token_is (parser->lexer, CPP_OR_OR))
+    {
+      cp_token *op = cp_lexer_consume_token (parser->lexer);
+      cp_expr rhs = cp_parser_constraint_logical_and_expression (parser);
+      lhs = finish_constraint_or_expr (op->location, lhs, rhs);
+    }
+  return cp_parser_check_non_logical_constraint (parser, lhs);
+}
+
+/* Parse the expression after a requires-clause. This has a different grammar
+    than that in the concepts TS.  */
+
 static tree
-cp_parser_requires_clause (cp_parser *parser)
+cp_parser_requires_clause_expression (cp_parser *parser)
 {
-  // Parse the requires clause so that it is not automatically folded.
+  processing_constraint_expression_sentinel parsing_constraint;
   ++processing_template_decl;
-  tree expr = cp_parser_binary_expression (parser, false, false,
-                                          PREC_NOT_OPERATOR, NULL);
+  cp_expr expr = cp_parser_constraint_logical_or_expression (parser);
   if (check_for_bare_parameter_packs (expr))
     expr = error_mark_node;
   --processing_template_decl;
   return expr;
 }
 
-// Optionally parse a requires clause:
+/* Parse a expression after a requires clause.
+
+    constraint-expression:
+      logical-or-expression
+
+   The required logical-or-expression must be a constant expression. Note
+   that we don't check that the expression is constepxr here. We defer until
+   we analyze constraints and then, we only check atomic constraints.  */
+
+static tree
+cp_parser_constraint_expression (cp_parser *parser)
+{
+  processing_constraint_expression_sentinel parsing_constraint;
+  ++processing_template_decl;
+  cp_expr expr = cp_parser_binary_expression (parser, false, true,
+                                             PREC_NOT_OPERATOR, NULL);
+  if (check_for_bare_parameter_packs (expr))
+    expr = error_mark_node;
+  --processing_template_decl;
+  expr.maybe_add_location_wrapper ();
+  return expr;
+}
+
+/* Optionally parse a requires clause:
+
+      requires-clause:
+        `requires` constraint-logical-or-expression.
+   [ConceptsTS]
+        `requires constraint-expression.  */
+
 static tree
 cp_parser_requires_clause_opt (cp_parser *parser)
 {
@@ -26920,17 +27261,21 @@ cp_parser_requires_clause_opt (cp_parser *parser)
          && tok->u.value == ridpointers[RID_REQUIRES])
        {
          error_at (cp_lexer_peek_token (parser->lexer)->location,
-                   "%<requires%> only available with %<-fconcepts%>");
+                   "%<requires%> only available with "
+                    "%<-std=c++2a%> or %<-fconcepts%>");
          /* Parse and discard the requires-clause.  */
          cp_lexer_consume_token (parser->lexer);
-         cp_parser_requires_clause (parser);
+         cp_parser_constraint_expression (parser);
        }
       return NULL_TREE;
     }
   cp_lexer_consume_token (parser->lexer);
-  return cp_parser_requires_clause (parser);
-}
 
+  if (!flag_concepts_ts)
+    return cp_parser_requires_clause_expression (parser);
+  else
+    return cp_parser_constraint_expression (parser);
+}
 
 /*---------------------------------------------------------------------------
                            Requires expressions
@@ -26940,6 +27285,7 @@ cp_parser_requires_clause_opt (cp_parser *parser)
 
    requirement-expression:
        'requires' requirement-parameter-list [opt] requirement-body */
+
 static tree
 cp_parser_requires_expression (cp_parser *parser)
 {
@@ -26957,6 +27303,9 @@ cp_parser_requires_expression (cp_parser *parser)
       return error_mark_node;
     }
 
+  /* This is definitely a requires-expression.  */
+  cp_parser_commit_to_tentative_parse (parser);
+
   tree parms, reqs;
   {
     /* Local parameters are delared as variables within the scope
@@ -26997,13 +27346,15 @@ cp_parser_requires_expression (cp_parser *parser)
   /* This needs to happen after pop_bindings_and_leave_scope, as it reverses
      the parm chain.  */
   grokparms (parms, &parms);
-  return finish_requires_expr (parms, reqs);
+  loc = make_location (loc, loc, parser->lexer);
+  return finish_requires_expr (loc, parms, reqs);
 }
 
 /* Parse a parameterized requirement.
 
    requirement-parameter-list:
        '(' parameter-declaration-clause ')' */
+
 static tree
 cp_parser_requirement_parameter_list (cp_parser *parser)
 {
@@ -27031,7 +27382,7 @@ cp_parser_requirement_body (cp_parser *parser)
   if (!braces.require_open (parser))
     return error_mark_node;
 
-  tree reqs = cp_parser_requirement_list (parser);
+  tree reqs = cp_parser_requirement_seq (parser);
 
   if (!braces.require_close (parser))
     return error_mark_node;
@@ -27039,34 +27390,28 @@ cp_parser_requirement_body (cp_parser *parser)
   return reqs;
 }
 
-/* Parse a list of requirements.
+/* Parse a sequence of requirements.
 
-   requirement-list:
+   requirement-seq:
        requirement
-       requirement-list ';' requirement[opt] */
+       requirement-seq requirement */
+
 static tree
-cp_parser_requirement_list (cp_parser *parser)
+cp_parser_requirement_seq (cp_parser *parser)
 {
   tree result = NULL_TREE;
-  while (true)
+  do
     {
       tree req = cp_parser_requirement (parser);
-      if (req == error_mark_node)
-        return error_mark_node;
-
-      result = tree_cons (NULL_TREE, req, result);
-
-      /* If we see a semi-colon, consume it. */
-      if (cp_lexer_next_token_is (parser->lexer, CPP_SEMICOLON))
-       cp_lexer_consume_token (parser->lexer);
+      if (req != error_mark_node)
+       result = tree_cons (NULL_TREE, req, result);
+    } while (cp_lexer_next_token_is_not (parser->lexer, CPP_CLOSE_BRACE));
 
-      /* Stop processing at the end of the list. */
-      if (cp_lexer_next_token_is (parser->lexer, CPP_CLOSE_BRACE))
-        break;
-    }
+  /* If there are no valid requirements, this is not a valid expression. */
+  if (!result)
+    return error_mark_node;
 
-  /* Reverse the order of requirements so they are analyzed in
-     declaration order. */
+  /* Reverse the order of requirements so they are analyzed in order. */
   return nreverse (result);
 }
 
@@ -27077,6 +27422,7 @@ cp_parser_requirement_list (cp_parser *parser)
        compound-requirement
        type-requirement
        nested-requirement */
+
 static tree
 cp_parser_requirement (cp_parser *parser)
 {
@@ -27094,17 +27440,26 @@ cp_parser_requirement (cp_parser *parser)
 
      simple-requirement:
        expression ';' */
+
 static tree
 cp_parser_simple_requirement (cp_parser *parser)
 {
-  tree expr = cp_parser_expression (parser, NULL, false, false);
+  location_t start = cp_lexer_peek_token (parser->lexer)->location;
+  cp_expr expr = cp_parser_expression (parser, NULL, false, false);
+  if (expr == error_mark_node)
+    cp_parser_skip_to_end_of_statement (parser);
+
+  cp_parser_consume_semicolon_at_end_of_statement (parser);
+
   if (!expr || expr == error_mark_node)
     return error_mark_node;
 
-  if (!cp_parser_require (parser, CPP_SEMICOLON, RT_SEMICOLON))
-    return error_mark_node;
+  /* Sometimes we don't get locations, so use the cached token location
+     as a reasonable approximation.  */
+  if (expr.get_location() == UNKNOWN_LOCATION)
+    expr.set_location (start);
 
-  return finish_simple_requirement (expr);
+  return finish_simple_requirement (expr.get_location (), expr);
 }
 
 /* Parse a type requirement
@@ -27115,16 +27470,18 @@ cp_parser_simple_requirement (cp_parser *parser)
      required-type-name:
          type-name
          'template' [opt] simple-template-id  */
+
 static tree
 cp_parser_type_requirement (cp_parser *parser)
 {
-  cp_lexer_consume_token (parser->lexer);
+  cp_token *start_tok = cp_lexer_consume_token (parser->lexer);
+  location_t loc = cp_lexer_peek_token (parser->lexer)->location;
 
   // Save the scope before parsing name specifiers.
   tree saved_scope = parser->scope;
   tree saved_object_scope = parser->object_scope;
   tree saved_qualifying_scope = parser->qualifying_scope;
-  cp_parser_global_scope_opt (parser, /*current_scope_valid_p=*/true);
+  cp_parser_global_scope_opt (parser, /*current_scope_valid_p=*/false);
   cp_parser_nested_name_specifier_opt (parser,
                                        /*typename_keyword_p=*/true,
                                        /*check_dependency_p=*/false,
@@ -27156,18 +27513,20 @@ cp_parser_type_requirement (cp_parser *parser)
   if (type == error_mark_node)
     cp_parser_skip_to_end_of_statement (parser);
 
-  if (!cp_parser_require (parser, CPP_SEMICOLON, RT_SEMICOLON))
-    return error_mark_node;
+  cp_parser_consume_semicolon_at_end_of_statement (parser);
+
   if (type == error_mark_node)
     return error_mark_node;
 
-  return finish_type_requirement (type);
+  loc = make_location (loc, start_tok->location, parser->lexer);
+  return finish_type_requirement (loc, type);
 }
 
 /* Parse a compound requirement
 
      compound-requirement:
          '{' expression '}' 'noexcept' [opt] trailing-return-type [opt] ';' */
+
 static tree
 cp_parser_compound_requirement (cp_parser *parser)
 {
@@ -27176,12 +27535,26 @@ cp_parser_compound_requirement (cp_parser *parser)
   if (!braces.require_open (parser))
     return error_mark_node;
 
+  cp_token *expr_token = cp_lexer_peek_token (parser->lexer);
+
   tree expr = cp_parser_expression (parser, NULL, false, false);
-  if (!expr || expr == error_mark_node)
-    return error_mark_node;
+  if (expr == error_mark_node)
+    cp_parser_skip_to_closing_brace (parser);
 
   if (!braces.require_close (parser))
-    return error_mark_node;
+    {
+      cp_parser_skip_to_end_of_statement (parser);
+      cp_parser_consume_semicolon_at_end_of_statement (parser);
+      return error_mark_node;
+    }
+
+  /* If the expression was invalid, skip the remainder of the requirement.  */
+  if (!expr || expr == error_mark_node)
+    {
+      cp_parser_skip_to_end_of_statement (parser);
+      cp_parser_consume_semicolon_at_end_of_statement (parser);
+      return error_mark_node;
+    }
 
   /* Parse the optional noexcept. */
   bool noexcept_p = false;
@@ -27196,29 +27569,69 @@ cp_parser_compound_requirement (cp_parser *parser)
   if (cp_lexer_next_token_is (parser->lexer, CPP_DEREF))
     {
       cp_lexer_consume_token (parser->lexer);
+      cp_token *tok = cp_lexer_peek_token (parser->lexer);
+
       bool saved_result_type_constraint_p = parser->in_result_type_constraint_p;
       parser->in_result_type_constraint_p = true;
+      /* C++2a allows either a type-id or a type-constraint. Parsing
+         a type-id will subsume the parsing for a type-constraint but
+         allow for more syntactic forms (e.g., const C<T>*).  */
       type = cp_parser_trailing_type_id (parser);
       parser->in_result_type_constraint_p = saved_result_type_constraint_p;
       if (type == error_mark_node)
         return error_mark_node;
+
+      location_t type_loc = make_location (tok->location, tok->location,
+                                          parser->lexer);
+
+      /* Check that we haven't written something like 'const C<T>*'.  */
+      if (type_uses_auto (type))
+       {
+         if (!is_auto (type))
+           {
+             error_at (type_loc,
+                       "result type is not a plain type-constraint");
+             cp_parser_consume_semicolon_at_end_of_statement (parser);
+             return error_mark_node;
+           }
+       }
+      else if (!flag_concepts_ts)
+       /* P1452R2 removed the trailing-return-type option.  */
+       error_at (type_loc,
+                 "return-type-requirement is not a type-constraint");
     }
 
-  return finish_compound_requirement (expr, type, noexcept_p);
+  location_t loc = make_location (expr_token->location,
+                                 braces.open_location (),
+                                 parser->lexer);
+
+  cp_parser_consume_semicolon_at_end_of_statement (parser);
+
+  if (expr == error_mark_node || type == error_mark_node)
+    return error_mark_node;
+
+  return finish_compound_requirement (loc, expr, type, noexcept_p);
 }
 
 /* Parse a nested requirement. This is the same as a requires clause.
 
    nested-requirement:
      requires-clause */
+
 static tree
 cp_parser_nested_requirement (cp_parser *parser)
 {
-  cp_lexer_consume_token (parser->lexer);
-  tree req = cp_parser_requires_clause (parser);
+  gcc_assert (cp_lexer_next_token_is_keyword (parser->lexer, RID_REQUIRES));
+  cp_token *tok = cp_lexer_consume_token (parser->lexer);
+  location_t loc = cp_lexer_peek_token (parser->lexer)->location;
+  tree req = cp_parser_constraint_expression (parser);
+  if (req == error_mark_node)
+    cp_parser_skip_to_end_of_statement (parser);
+  loc = make_location (loc, tok->location, parser->lexer);
+  cp_parser_consume_semicolon_at_end_of_statement (parser);
   if (req == error_mark_node)
     return error_mark_node;
-  return finish_nested_requirement (req);
+  return finish_nested_requirement (loc, req);
 }
 
 /* Support Functions */
@@ -28145,6 +28558,11 @@ cp_parser_template_declaration_after_parameters (cp_parser* parser,
   else if (cxx_dialect >= cxx11
           && cp_lexer_next_token_is_keyword (parser->lexer, RID_USING))
     decl = cp_parser_alias_declaration (parser);
+  else if (cxx_dialect >= cxx2a /* Implies flag_concept.  */
+           && cp_lexer_next_token_is_keyword (parser->lexer, RID_CONCEPT)
+           && !cp_lexer_nth_token_is_keyword (parser->lexer, 2, RID_BOOL))
+    /* Allow 'concept bool' to be handled as per the TS.  */
+    decl = cp_parser_concept_definition (parser);
   else
     {
       /* There are no access checks when parsing a template, as we do not
@@ -28264,6 +28682,8 @@ cp_parser_template_introduction (cp_parser* parser, bool member_p)
   tree saved_object_scope = parser->object_scope;
   tree saved_qualifying_scope = parser->qualifying_scope;
 
+  cp_token *start_token = cp_lexer_peek_token (parser->lexer);
+
   /* Look for the optional `::' operator.  */
   cp_parser_global_scope_opt (parser,
                              /*current_scope_valid_p=*/false);
@@ -28285,12 +28705,14 @@ cp_parser_template_introduction (cp_parser* parser, bool member_p)
   parser->object_scope = saved_object_scope;
   parser->qualifying_scope = saved_qualifying_scope;
 
-  if (concept_name == error_mark_node)
+  if (concept_name == error_mark_node
+      || (seen_error () && !concept_definition_p (tmpl_decl)))
     cp_parser_simulate_error (parser);
 
   /* Look for opening brace for introduction.  */
   matching_braces braces;
   braces.require_open (parser);
+  location_t open_loc = input_location;
 
   if (!cp_parser_parse_definitely (parser))
     return false;
@@ -28321,15 +28743,26 @@ cp_parser_template_introduction (cp_parser* parser, bool member_p)
     }
 
   /* Build and associate the constraint.  */
-  tree parms = finish_template_introduction (tmpl_decl, introduction_list);
+  location_t introduction_loc = make_location (open_loc,
+                                              start_token->location,
+                                              parser->lexer);
+  tree parms = finish_template_introduction (tmpl_decl,
+                                            introduction_list,
+                                            introduction_loc);
   if (parms && parms != error_mark_node)
     {
+      if (!flag_concepts_ts)
+       pedwarn (introduction_loc, 0, "template-introductions"
+                " are not part of C++20 concepts [-fconcepts-ts]");
+
       cp_parser_template_declaration_after_parameters (parser, parms,
                                                       member_p);
       return true;
     }
 
-  error_at (token->location, "no matching concept for template-introduction");
+  if (parms == NULL_TREE)
+    error_at (token->location, "no matching concept for template-introduction");
+
   return true;
 }
 
@@ -28397,8 +28830,8 @@ cp_parser_explicit_template_declaration (cp_parser* parser, bool member_p)
   if (flag_concepts)
   {
     tree reqs = get_shorthand_constraints (current_template_parms);
-    if (tree r = cp_parser_requires_clause_opt (parser))
-      reqs = conjoin_constraints (reqs, normalize_expression (r));
+    if (tree treqs = cp_parser_requires_clause_opt (parser))
+      reqs = combine_constraint_expressions (reqs, treqs);
     TEMPLATE_PARMS_CONSTRAINTS (current_template_parms) = reqs;
   }
 
@@ -29635,6 +30068,9 @@ cp_parser_required_error (cp_parser *parser,
       case RT_THROW:
        gmsgid = G_("expected %<throw%>");
        break;
+      case RT_AUTO:
+        gmsgid = G_("expected %<auto%>");
+        break;
       case RT_LABEL:
        gmsgid = G_("expected %<__label__%>");
        break;
@@ -41910,36 +42346,15 @@ make_generic_type_name ()
 static tree
 synthesize_implicit_template_parm  (cp_parser *parser, tree constr)
 {
-  gcc_assert (current_binding_level->kind == sk_function_parms);
-
-   /* Before committing to modifying any scope, if we're in an
-      implicit template scope, and we're trying to synthesize a
-      constrained parameter, try to find a previous parameter with
-      the same name.  This is the same-type rule for abbreviated
-      function templates.
-
-      NOTE: We can generate implicit parameters when tentatively
-      parsing a nested name specifier, only to reject that parse
-      later. However, matching the same template-id as part of a
-      direct-declarator should generate an identical template
-      parameter, so this rule will merge them. */
-  if (parser->implicit_template_scope && constr)
-    {
-      tree t = parser->implicit_template_parms;
-      while (t)
-        {
-          if (equivalent_placeholder_constraints (TREE_TYPE (t), constr))
-           {
-             tree d = TREE_VALUE (t);
-             if (TREE_CODE (d) == PARM_DECL)
-               /* Return the TEMPLATE_PARM_INDEX.  */
-               d = DECL_INITIAL (d);
-             return d;
-           }
-          t = TREE_CHAIN (t);
-        }
+  /* A requires-clause is not a function and cannot have placeholders.  */
+  if (current_binding_level->kind == sk_block)
+    {
+      error ("placeholder type not allowed in this context");
+      return error_mark_node;
     }
 
+  gcc_assert (current_binding_level->kind == sk_function_parms);
+
   /* We are either continuing a function template that already contains implicit
      template parameters, creating a new fully-implicit function template, or
      extending an existing explicit function template with implicit template
@@ -42050,18 +42465,9 @@ synthesize_implicit_template_parm  (cp_parser *parser, tree constr)
   tree synth_tmpl_parm;
   bool non_type = false;
 
-  if (proto == NULL_TREE || TREE_CODE (proto) == TYPE_DECL)
-    synth_tmpl_parm
-      = finish_template_type_parm (class_type_node, synth_id);
-  else if (TREE_CODE (proto) == TEMPLATE_DECL)
-    synth_tmpl_parm
-      = finish_constrained_template_template_parm (proto, synth_id);
-  else
-    {
-      synth_tmpl_parm = copy_decl (proto);
-      DECL_NAME (synth_tmpl_parm) = synth_id;
-      non_type = true;
-    }
+  /* Synthesize the type template parameter.  */
+  gcc_assert(!proto || TREE_CODE (proto) == TYPE_DECL);
+  synth_tmpl_parm = finish_template_type_parm (class_type_node, synth_id);
 
   /* Attach the constraint to the parm before processing.  */
   tree node = build_tree_list (NULL_TREE, synth_tmpl_parm);
@@ -42073,6 +42479,13 @@ synthesize_implicit_template_parm  (cp_parser *parser, tree constr)
                             /*non_type=*/non_type,
                             /*param_pack=*/false);
 
+  /* Mark the synthetic declaration "virtual". This is used when
+     comparing template-heads to determine if whether an abbreviated
+     function template is equivalent to an explicit template.
+
+     Note that DECL_ARTIFICIAL is used elsewhere for template parameters.  */
+  DECL_VIRTUAL_P (TREE_VALUE (new_parm)) = true;
+
   // Chain the new parameter to the list of implicit parameters.
   if (parser->implicit_template_parms)
     parser->implicit_template_parms
@@ -42111,7 +42524,7 @@ synthesize_implicit_template_parm  (cp_parser *parser, tree constr)
   if (tree req = TEMPLATE_PARM_CONSTRAINTS (tree_last (new_parm)))
     {
       tree reqs = TEMPLATE_PARMS_CONSTRAINTS (current_template_parms);
-      reqs = conjoin_constraints (reqs, req);
+      reqs = combine_constraint_expressions (reqs, req);
       TEMPLATE_PARMS_CONSTRAINTS (current_template_parms) = reqs;
     }
 
index 6310e7b399beb5abda2daae854e6cd8cee6597dd..84464436991b2adb3dbcdd8ce0de0db81ee5cf75 100644 (file)
@@ -135,8 +135,6 @@ enum template_base_result {
   tbr_success
 };
 
-static void push_access_scope (tree);
-static void pop_access_scope (tree);
 static bool resolve_overloaded_unification (tree, tree, tree, tree,
                                            unification_kind_t, int,
                                            bool);
@@ -153,7 +151,6 @@ static tree coerce_innermost_template_parms (tree, tree, tree, tsubst_flags_t,
                                              bool, bool);
 static void tsubst_enum        (tree, tree, tree);
 static tree add_to_template_args (tree, tree);
-static tree add_outermost_template_args (tree, tree);
 static bool check_instantiated_args (tree, tree, tsubst_flags_t);
 static int check_non_deducible_conversion (tree, tree, int, int,
                                           struct conversion **, bool);
@@ -183,12 +180,9 @@ static int can_complete_type_without_circularity (tree);
 static tree get_bindings (tree, tree, tree, bool);
 static int template_decl_level (tree);
 static int check_cv_quals_for_unify (int, tree, tree);
-static void template_parm_level_and_index (tree, int*, int*);
 static int unify_pack_expansion (tree, tree, tree,
                                 tree, unification_kind_t, bool, bool);
 static tree copy_template_args (tree);
-static tree tsubst_template_arg (tree, tree, tsubst_flags_t, tree);
-static tree tsubst_template_args (tree, tree, tsubst_flags_t, tree);
 static tree tsubst_template_parms (tree, tree, tsubst_flags_t);
 static tree most_specialized_partial_spec (tree, tsubst_flags_t);
 static tree tsubst_aggr_type (tree, tree, tsubst_flags_t, tree, int);
@@ -232,7 +226,7 @@ static tree enclosing_instantiation_of (tree tctx);
    template, VAR_DECL for static member variable, or TYPE_DECL for
    alias template (needed by instantiate_decl).  */
 
-static void
+void
 push_access_scope (tree t)
 {
   gcc_assert (VAR_OR_FUNCTION_DECL_P (t)
@@ -255,7 +249,7 @@ push_access_scope (tree t)
 /* Restore the scope set up by push_access_scope.  T is the node we
    are processing.  */
 
-static void
+void
 pop_access_scope (tree t)
 {
   if (TREE_CODE (t) == FUNCTION_DECL)
@@ -582,7 +576,7 @@ add_to_template_args (tree args, tree extra_args)
    template arguments used to attain the full instantiation from the
    partial instantiation.  */
 
-static tree
+tree
 add_outermost_template_args (tree args, tree extra_args)
 {
   tree new_args;
@@ -3305,6 +3299,153 @@ comp_template_parms (const_tree parms1, const_tree parms2)
   return 1;
 }
 
+/* Returns true if two template parameters are declared with
+   equivalent constraints.  */
+
+static bool
+template_parameter_constraints_equivalent_p (const_tree parm1, const_tree parm2)
+{
+  tree req1 = TREE_TYPE (parm1);
+  tree req2 = TREE_TYPE (parm2);
+  if (!req1 != !req2)
+    return false;
+  if (req1)
+    return cp_tree_equal (req1, req2);
+  return true;
+}
+
+/* Returns true when two template parameters are equivalent.  */
+
+static bool
+template_parameters_equivalent_p (const_tree parm1, const_tree parm2)
+{
+  tree decl1 = TREE_VALUE (parm1);
+  tree decl2 = TREE_VALUE (parm2);
+
+  /* If either of the template parameters are invalid, assume
+     they match for the sake of error recovery. */
+  if (error_operand_p (decl1) || error_operand_p (decl2))
+    return true;
+
+  /* ... they declare parameters of the same kind.  */
+  if (TREE_CODE (decl1) != TREE_CODE (decl2))
+    return false;
+
+  /* ... one parameter was introduced by a parameter declaration, then
+     both are. This case arises as a result of eagerly rewriting declarations
+     during parsing.  */
+  if (DECL_VIRTUAL_P (decl1) != DECL_VIRTUAL_P (decl2))
+    return false;
+
+  /* ... if either declares a pack, they both do.  */
+  if (template_parameter_pack_p (decl1) != template_parameter_pack_p (decl2))
+    return false;
+
+  if (TREE_CODE (decl1) == PARM_DECL)
+    {
+      /* ... if they declare non-type parameters, the types are equivalent.  */
+      if (!same_type_p (TREE_TYPE (decl1), TREE_TYPE (decl2)))
+       return false;
+    }
+  else if (TREE_CODE (decl2) == TEMPLATE_DECL)
+    {
+      /* ... if they declare template template parameters, their template
+        parameter lists are equivalent.  */
+      if (!template_heads_equivalent_p (decl1, decl2))
+       return false;
+    }
+
+  /* ... if they are declared with a qualified-concept name, they both
+     are, and those names are equivalent.  */
+  return template_parameter_constraints_equivalent_p (parm1, parm2);
+}
+
+/* Returns true if two template parameters lists are equivalent.
+   Two template parameter lists are equivalent if they have the
+   same length and their corresponding parameters are equivalent.
+
+   PARMS1 and PARMS2 are TREE_LISTs containing TREE_VECs: the
+   data structure returned by DECL_TEMPLATE_PARMS.
+
+   This is generally the same implementation as comp_template_parms
+   except that it also the concept names and arguments used to
+   introduce parameters.  */
+
+static bool
+template_parameter_lists_equivalent_p (const_tree parms1, const_tree parms2)
+{
+  if (parms1 == parms2)
+    return true;
+
+  const_tree p1 = parms1;
+  const_tree p2 = parms2;
+  while (p1 != NULL_TREE && p2 != NULL_TREE)
+    {
+      tree list1 = TREE_VALUE (p1);
+      tree list2 = TREE_VALUE (p2);
+
+      if (TREE_VEC_LENGTH (list1) != TREE_VEC_LENGTH (list2))
+       return 0;
+
+      for (int i = 0; i < TREE_VEC_LENGTH (list2); ++i)
+       {
+         tree parm1 = TREE_VEC_ELT (list1, i);
+         tree parm2 = TREE_VEC_ELT (list2, i);
+         if (!template_parameters_equivalent_p (parm1, parm2))
+           return false;
+       }
+
+      p1 = TREE_CHAIN (p1);
+      p2 = TREE_CHAIN (p2);
+    }
+
+  if ((p1 != NULL_TREE) != (p2 != NULL_TREE))
+    return false;
+
+  return true;
+}
+
+/* Return true if the requires-clause of the template parameter lists are
+   equivalent and false otherwise.  */
+static bool
+template_requirements_equivalent_p (const_tree parms1, const_tree parms2)
+{
+  tree req1 = TEMPLATE_PARMS_CONSTRAINTS (parms1);
+  tree req2 = TEMPLATE_PARMS_CONSTRAINTS (parms2);
+  if ((req1 != NULL_TREE) != (req2 != NULL_TREE))
+    return false;
+  if (!cp_tree_equal (req1, req2))
+    return false;
+  return true;
+}
+
+/* Returns true if two template heads are equivalent. 17.6.6.1p6:
+   Two template heads are equivalent if their template parameter
+   lists are equivalent and their requires clauses are equivalent.
+
+   In pre-C++20, this is equivalent to calling comp_template_parms
+   for the template parameters of TMPL1 and TMPL2.  */
+
+bool
+template_heads_equivalent_p (const_tree tmpl1, const_tree tmpl2)
+{
+  tree parms1 = DECL_TEMPLATE_PARMS (tmpl1);
+  tree parms2 = DECL_TEMPLATE_PARMS (tmpl2);
+
+  /* Don't change the matching rules for pre-C++20.  */
+  if (cxx_dialect < cxx2a)
+    return comp_template_parms (parms1, parms2);
+
+  /* ... have the same number of template parameters, and their
+     corresponding parameters are equivalent.  */
+  if (!template_parameter_lists_equivalent_p (parms1, parms2))
+    return false;
+
+  /* ... if either has a requires-clause, they both do and their
+     corresponding constraint-expressions are equivalent.  */
+  return template_requirements_equivalent_p (parms1, parms2);
+}
+
 /* Determine whether PARM is a parameter pack.  */
 
 bool
@@ -4572,7 +4713,7 @@ template_parm_to_arg (tree t)
 /* Given a single level of template parameters (a TREE_VEC), return it
    as a set of template arguments.  */
 
-static tree
+tree
 template_parms_level_to_args (tree parms)
 {
   tree a = copy_node (parms);
@@ -4590,7 +4731,7 @@ template_parms_level_to_args (tree parms)
    arguments.  The template parameters are represented as a TREE_VEC, in
    the form documented in cp-tree.h for template arguments.  */
 
-static tree
+tree
 template_parms_to_args (tree parms)
 {
   tree header;
@@ -4626,6 +4767,26 @@ current_template_args (void)
   return template_parms_to_args (current_template_parms);
 }
 
+/* Return the fully generic arguments for of TMPL, i.e. what
+   current_template_args would be while parsing it.  */
+
+tree
+generic_targs_for (tree tmpl)
+{
+  if (tmpl == NULL_TREE)
+    return NULL_TREE;
+  if (DECL_TEMPLATE_TEMPLATE_PARM_P (tmpl)
+      || DECL_TEMPLATE_SPECIALIZATION (tmpl))
+    /* DECL_TEMPLATE_RESULT doesn't have the arguments we want.  For a template
+       template parameter, it has no TEMPLATE_INFO; for a partial
+       specialization, it has the arguments for the primary template, and we
+       want the arguments for the partial specialization.  */;
+  else if (tree result = DECL_TEMPLATE_RESULT (tmpl))
+    if (tree ti = get_template_info (result))
+      return TI_ARGS (ti);
+  return template_parms_to_args (DECL_TEMPLATE_PARMS (tmpl));
+}
+
 /* Update the declared TYPE by doing any lookups which were thought to be
    dependent, but are not now that we know the SCOPE of the declarator.  */
 
@@ -4855,7 +5016,7 @@ process_partial_specialization (tree decl)
   if (comp_template_args (inner_args, INNERMOST_TEMPLATE_ARGS (main_args))
       && (!flag_concepts
          || !strictly_subsumes (current_template_constraints (),
-                                get_constraints (maintmpl))))
+                                inner_args, maintmpl)))
     {
       if (!flag_concepts)
         error ("partial specialization %q+D does not specialize "
@@ -5564,6 +5725,8 @@ push_template_decl_real (tree decl, bool is_friend)
        gcc_assert (!DECL_ARTIFICIAL (decl));
       else if (VAR_P (decl))
        /* C++14 variable template. */;
+      else if (TREE_CODE (decl) == CONCEPT_DECL)
+       /* C++2a concept definitions.  */;
       else
        {
          error ("template declaration of %q#D", decl);
@@ -5966,6 +6129,19 @@ redeclare_class_template (tree type, tree parms, tree cons)
          return false;
        }
 
+      /* The parameters can be declared to introduce different
+        constraints.  */
+      tree p1 = TREE_VEC_ELT (tmpl_parms, i);
+      tree p2 = TREE_VEC_ELT (parms, i);
+      if (!template_parameter_constraints_equivalent_p (p1, p2))
+       {
+         error ("declaration of template parameter %q+#D with different "
+                "constraints", parm);
+         inform (DECL_SOURCE_LOCATION (tmpl_parm),
+                 "original declaration appeared here");
+         return false;
+       }
+
       if (tmpl_default != NULL_TREE && parm_default != NULL_TREE)
        {
          /* We have in [temp.param]:
@@ -5999,13 +6175,18 @@ redeclare_class_template (tree type, tree parms, tree cons)
        TEMPLATE_TYPE_PARM_FOR_CLASS (TREE_TYPE (parm)) = true;
     }
 
-  // Cannot redeclare a class template with a different set of constraints.
-  if (!equivalent_constraints (get_constraints (tmpl), cons))
+  tree ci = get_constraints (tmpl);
+  tree req1 = ci ? CI_TEMPLATE_REQS (ci) : NULL_TREE;
+  tree req2 = cons ? CI_TEMPLATE_REQS (cons) : NULL_TREE;
+
+  /* Two classes with different constraints declare different entities.  */
+  if (!cp_tree_equal (req1, req2))
     {
       error_at (input_location, "redeclaration %q#D with different "
                                 "constraints", tmpl);
       inform (DECL_SOURCE_LOCATION (tmpl),
               "original declaration appeared here");
+      return false;
     }
 
     return true;
@@ -6214,8 +6395,7 @@ get_underlying_template (tree tmpl)
              != num_innermost_template_parms (underlying)))
        break;
 
-      tree alias_args = INNERMOST_TEMPLATE_ARGS
-       (template_parms_to_args (DECL_TEMPLATE_PARMS (tmpl)));
+      tree alias_args = INNERMOST_TEMPLATE_ARGS (generic_targs_for (tmpl));
       if (!comp_template_args (TI_ARGS (tinfo), alias_args))
        break;
 
@@ -7322,14 +7502,7 @@ coerce_template_args_for_ttp (tree templ, tree arglist,
 
   tree outer = DECL_CONTEXT (templ);
   if (outer)
-    {
-      if (DECL_TEMPLATE_SPECIALIZATION (outer))
-       /* We want arguments for the partial specialization, not arguments for
-          the primary template.  */
-       outer = template_parms_to_args (DECL_TEMPLATE_PARMS (outer));
-      else
-       outer = TI_ARGS (get_template_info (DECL_TEMPLATE_RESULT (outer)));
-    }
+    outer = generic_targs_for (outer);
   else if (current_template_parms)
     {
       /* This is an argument of the current template, so we haven't set
@@ -7787,8 +7960,6 @@ is_compatible_template_arg (tree parm, tree arg)
   if (parm_cons == NULL_TREE)
     return true;
 
-  tree arg_cons = get_constraints (arg);
-
   /* If the template parameter is constrained, we need to rewrite its
      constraints in terms of the ARG's template parameters. This ensures
      that all of the template parameter types will have the same depth.
@@ -7796,17 +7967,18 @@ is_compatible_template_arg (tree parm, tree arg)
      Note that this is only valid when coerce_template_template_parm is
      true for the innermost template parameters of PARM and ARG. In other
      words, because coercion is successful, this conversion will be valid.  */
+  tree new_args = NULL_TREE;
   if (parm_cons)
     {
-      tree args = template_parms_to_args (DECL_TEMPLATE_PARMS (arg));
-      parm_cons = tsubst_constraint_info (parm_cons,
-                                         INNERMOST_TEMPLATE_ARGS (args),
+      tree aparms = DECL_INNERMOST_TEMPLATE_PARMS (arg);
+      new_args = template_parms_level_to_args (aparms);
+      parm_cons = tsubst_constraint_info (parm_cons, new_args,
                                          tf_none, NULL_TREE);
       if (parm_cons == error_mark_node)
         return false;
     }
 
-  return subsumes (parm_cons, arg_cons);
+  return weakly_subsumes (parm_cons, new_args, arg);
 }
 
 // Convert a placeholder argument into a binding to the original
@@ -8426,7 +8598,7 @@ coerce_template_parms (tree parms,
      template (DR 1430).  */
   else if (in_decl
           && (DECL_ALIAS_TEMPLATE_P (in_decl)
-              || concept_template_p (in_decl))
+              || concept_definition_p (in_decl))
           && variadic_args_p
           && nargs - variadic_args_p < nparms - variadic_p)
     {
@@ -8600,6 +8772,7 @@ coerce_template_parms (tree parms,
 
       if (arg == error_mark_node)
        lost++;
+
       TREE_VEC_ELT (new_inner_args, arg_idx - pack_adjust) = arg;
     }
 
@@ -8617,6 +8790,32 @@ coerce_template_parms (tree parms,
       /* We had some pack expansion arguments that will only work if the packs
         are empty, but wait until instantiation time to complain.
         See variadic-ttp3.C.  */
+
+      /* Except that we can't provide empty packs to alias templates or
+         concepts when there are no corresponding parameters. Basically,
+         we can get here with this:
+
+             template<typename T> concept C = true;
+
+             template<typename... Args>
+              requires C<Args...>
+             void f();
+
+         When parsing C<Args...>, we try to form a concept check of
+         C<?, Args...>. Without the extra check for substituting an empty
+         pack past the last parameter, we can accept the check as valid.
+
+         FIXME: This may be valid for alias templates (but I doubt it).
+
+         FIXME: The error could be better also.   */
+      if (in_decl && concept_definition_p (in_decl))
+       {
+         if (complain & tf_error)
+           error_at (location_of (TREE_VEC_ELT (args, arg_idx)),
+                     "too many arguments");
+         return error_mark_node;
+       }
+
       int len = nparms + (nargs - arg_idx);
       tree args = make_tree_vec (len);
       int i = 0;
@@ -9452,7 +9651,7 @@ lookup_template_class_1 (tree d1, tree arglist, tree in_decl, tree context,
           if (complain & tf_error)
             {
              auto_diagnostic_group d;
-              error ("template constraint failure");
+              error ("template constraint failure for %qD", gen_tmpl);
               diagnose_constraints (input_location, gen_tmpl, arglist);
             }
           return error_mark_node;
@@ -9785,13 +9984,12 @@ lookup_template_class (tree d1, tree arglist, tree in_decl, tree context,
 tree
 lookup_template_variable (tree templ, tree arglist)
 {
+  if (flag_concepts && variable_concept_p (templ))
+    return build_concept_check (templ, arglist, tf_none);
+
   /* The type of the expression is NULL_TREE since the template-id could refer
      to an explicit or partial specialization. */
-  tree type = NULL_TREE;
-  if (flag_concepts && variable_concept_p (templ))
-    /* Except that concepts are always bool.  */
-    type = boolean_type_node;
-  return build2 (TEMPLATE_ID_EXPR, type, templ, arglist);
+  return build2 (TEMPLATE_ID_EXPR, NULL_TREE, templ, arglist);
 }
 
 /* Instantiate a variable declaration from a TEMPLATE_ID_EXPR for use. */
@@ -9802,12 +10000,6 @@ finish_template_variable (tree var, tsubst_flags_t complain)
   tree templ = TREE_OPERAND (var, 0);
   tree arglist = TREE_OPERAND (var, 1);
 
-  /* We never want to return a VAR_DECL for a variable concept, since they
-     aren't instantiated.  In a template, leave the TEMPLATE_ID_EXPR alone.  */
-  bool concept_p = flag_concepts && variable_concept_p (templ);
-  if (concept_p && processing_template_decl)
-    return var;
-
   tree tmpl_args = DECL_TI_ARGS (DECL_TEMPLATE_RESULT (templ));
   arglist = add_outermost_template_args (tmpl_args, arglist);
 
@@ -9828,19 +10020,6 @@ finish_template_variable (tree var, tsubst_flags_t complain)
       return error_mark_node;
     }
 
-  /* If a template-id refers to a specialization of a variable
-     concept, then the expression is true if and only if the
-     concept's constraints are satisfied by the given template
-     arguments.
-
-     NOTE: This is an extension of Concepts Lite TS that
-     allows constraints to be used in expressions. */
-  if (concept_p)
-    {
-      tree decl = DECL_TEMPLATE_RESULT (templ);
-      return evaluate_variable_concept (decl, arglist);
-    }
-
   return instantiate_template (templ, arglist, complain);
 }
 
@@ -10041,6 +10220,26 @@ for_each_template_parm_r (tree *tp, int *walk_subtrees, void *d)
        return error_mark_node;
       break;
 
+    case SCOPE_REF:
+      if (pfd->include_nondeduced_p)
+       WALK_SUBTREE (TREE_OPERAND (t, 0));
+      break;
+
+    case REQUIRES_EXPR:
+      {
+       if (!fn)
+         return error_mark_node;
+
+       /* Recursively walk the type of each constraint variable.  */
+       tree p = TREE_OPERAND (t, 0);
+       while (p)
+         {
+           WALK_SUBTREE (TREE_TYPE (p));
+           p = TREE_CHAIN (p);
+         }
+      }
+      break;
+
     default:
       break;
     }
@@ -10104,6 +10303,119 @@ for_each_template_parm (tree t, tree_fn_t fn, void* data,
   return result;
 }
 
+struct find_template_parameter_info
+{
+  explicit find_template_parameter_info (int d)
+    : max_depth (d)
+  {}
+
+  hash_set<tree> visited;
+  hash_set<tree> parms;
+  int max_depth;
+};
+
+/* Appends the declaration of T to the list in DATA.  */
+
+static int
+keep_template_parm (tree t, void* data)
+{
+  find_template_parameter_info *ftpi = (find_template_parameter_info*)data;
+
+  /* Template parameters declared within the expression are not part of
+     the parameter mapping. For example, in this concept:
+
+       template<typename T>
+       concept C = requires { <expr> } -> same_as<int>;
+
+     the return specifier same_as<int> declares a new decltype parameter
+     that must not be part of the parameter mapping. The same is true
+     for generic lambda parameters, lambda template parameters, etc.  */
+  int level;
+  int index;
+  template_parm_level_and_index (t, &level, &index);
+  if (level > ftpi->max_depth)
+    return 0;
+
+  /* Arguments like const T yield parameters like const T. This means that
+     a template-id like X<T, const T> would yield two distinct parameters:
+     T and const T. Adjust types to their unqualified versions.  */
+  if (TYPE_P (t))
+    t = TYPE_MAIN_VARIANT (t);
+  ftpi->parms.add (t);
+
+  return 0;
+}
+
+/* Ensure that we recursively examine certain terms that are not normally
+   visited in for_each_template_parm_r.  */
+
+static int
+any_template_parm_r (tree t, void *data)
+{
+  find_template_parameter_info *ftpi = (find_template_parameter_info*)data;
+
+#define WALK_SUBTREE(NODE)                                             \
+  do                                                                   \
+    {                                                                  \
+      for_each_template_parm (NODE, keep_template_parm, data,          \
+                             &ftpi->visited, true,                     \
+                             any_template_parm_r);                     \
+    }                                                                  \
+  while (0)
+
+  switch (TREE_CODE (t))
+    {
+    case RECORD_TYPE:
+    case UNION_TYPE:
+    case ENUMERAL_TYPE:
+      /* Search for template parameters in type aliases.  */
+      if (alias_template_specialization_p (t))
+       {
+         tree tinfo = TYPE_ALIAS_TEMPLATE_INFO (t);
+         WALK_SUBTREE (TI_ARGS (tinfo));
+        }
+      break;
+
+    case TEMPLATE_TYPE_PARM:
+      /* Type constraints of a placeholder type may contain parameters.  */
+      if (is_auto (t))
+       if (tree constr = PLACEHOLDER_TYPE_CONSTRAINTS (t))
+         WALK_SUBTREE (constr);
+      break;
+
+    case TEMPLATE_ID_EXPR:
+      /* Search through references to variable templates.  */
+      WALK_SUBTREE (TREE_OPERAND (t, 0));
+      WALK_SUBTREE (TREE_OPERAND (t, 1));
+      break;
+
+    case CONSTRUCTOR:
+      if (TREE_TYPE (t))
+        WALK_SUBTREE (TREE_TYPE (t));
+      break;
+    default:
+      break;
+    }
+
+  /* Keep walking.  */
+  return 0;
+}
+
+/* Returns a list of unique template parameters found within T.  */
+
+tree
+find_template_parameters (tree t, int depth)
+{
+  find_template_parameter_info ftpi (depth);
+  for_each_template_parm (t, keep_template_parm, &ftpi, &ftpi.visited,
+                         /*include_nondeduced*/true, any_template_parm_r);
+  tree list = NULL_TREE;
+  for (hash_set<tree>::iterator iter = ftpi.parms.begin();
+       iter != ftpi.parms.end(); ++iter)
+    list = tree_cons (NULL_TREE, *iter, list);
+  return list;
+}
+
 /* Returns true if T depends on any template parameter.  */
 
 int
@@ -10431,6 +10743,30 @@ tsubst_friend_function (tree decl, tree args)
       DECL_USE_TEMPLATE (DECL_TEMPLATE_RESULT (new_friend)) = 0;
       DECL_SAVED_TREE (DECL_TEMPLATE_RESULT (new_friend))
        = DECL_SAVED_TREE (DECL_TEMPLATE_RESULT (decl));
+
+      /* Attach the template requirements to the new declaration
+         for declaration matching. We need to rebuild the requirements
+         so that parameter levels match.  */
+      if (tree ci = get_constraints (decl))
+       {
+         tree parms = DECL_TEMPLATE_PARMS (new_friend);
+         tree args = generic_targs_for (new_friend);
+         tree treqs = tsubst_constraint (CI_TEMPLATE_REQS (ci), args,
+                                         tf_warning_or_error, NULL_TREE);
+         tree freqs = tsubst_constraint (CI_DECLARATOR_REQS (ci), args,
+                                         tf_warning_or_error, NULL_TREE);
+
+         /* Update the constraints -- these won't really be valid for
+            checking, but that's not what we need them for. These ensure
+            that the declared function can find the friend during
+            declaration matching.  */
+         tree new_ci = get_constraints (new_friend);
+         CI_TEMPLATE_REQS (new_ci) = treqs;
+         CI_DECLARATOR_REQS (new_ci) = freqs;
+
+         /* Also update the template parameter list.  */
+         TEMPLATE_PARMS_CONSTRAINTS (parms) = treqs;
+       }
     }
 
   /* The mangled name for the NEW_FRIEND is incorrect.  The function
@@ -11529,7 +11865,7 @@ instantiate_class_template (tree type)
   return ret;
 }
 
-static tree
+tree
 tsubst_template_arg (tree t, tree args, tsubst_flags_t complain, tree in_decl)
 {
   tree r;
@@ -11547,6 +11883,7 @@ tsubst_template_arg (tree t, tree args, tsubst_flags_t complain, tree in_decl)
       if (!(complain & tf_warning))
        --c_inhibit_evaluation_warnings;
     }
+
   return r;
 }
 
@@ -11770,14 +12107,6 @@ gen_elem_of_pack_expansion_instantiation (tree pattern,
     t = tsubst_decl (pattern, args, complain);
   else if (pattern == error_mark_node)
     t = error_mark_node;
-  else if (constraint_p (pattern))
-    {
-      if (processing_template_decl)
-       t = tsubst_constraint (pattern, args, complain, in_decl);
-      else
-       t = (constraints_satisfied_p (pattern, args)
-            ? boolean_true_node : boolean_false_node);
-    }
   else if (!TYPE_P (pattern))
     t = tsubst_expr (pattern, args, complain, in_decl,
                     /*integral_constant_expression_p=*/false);
@@ -12257,7 +12586,6 @@ tsubst_pack_expansion (tree t, tree args, tsubst_flags_t complain,
         {
          int idx;
           template_parm_level_and_index (parm_pack, &level, &idx);
-
           if (level <= levels)
             arg_pack = TMPL_ARG (args, level, idx);
         }
@@ -12311,6 +12639,10 @@ tsubst_pack_expansion (tree t, tree args, tsubst_flags_t complain,
          /* We can't substitute for this parameter pack.  We use a flag as
             well as the missing_level counter because function parameter
             packs don't have a level.  */
+          if (!(processing_template_decl || is_auto (parm_pack)))
+           {
+             gcc_unreachable ();
+           }
          gcc_assert (processing_template_decl || is_auto (parm_pack));
          unsubstituted_packs = true;
        }
@@ -12519,9 +12851,33 @@ copy_template_args (tree t)
   return new_vec;
 }
 
+/* Substitute ARGS into the *_ARGUMENT_PACK orig_arg.  */
+
+tree
+tsubst_argument_pack (tree orig_arg, tree args, tsubst_flags_t complain,
+                     tree in_decl)
+{
+  /* Substitute into each of the arguments.  */
+  tree new_arg = TYPE_P (orig_arg)
+    ? cxx_make_type (TREE_CODE (orig_arg))
+    : make_node (TREE_CODE (orig_arg));
+
+  tree pack_args = tsubst_template_args (ARGUMENT_PACK_ARGS (orig_arg),
+                                        args, complain, in_decl);
+  if (pack_args == error_mark_node)
+    new_arg = error_mark_node;
+  else
+    SET_ARGUMENT_PACK_ARGS (new_arg, pack_args);
+
+  if (TREE_CODE (new_arg) == NONTYPE_ARGUMENT_PACK)
+    TREE_CONSTANT (new_arg) = TREE_CONSTANT (orig_arg);
+
+  return new_arg;
+}
+
 /* Substitute ARGS into the vector or list of template arguments T.  */
 
-static tree
+tree
 tsubst_template_args (tree t, tree args, tsubst_flags_t complain, tree in_decl)
 {
   tree orig_t = t;
@@ -12556,22 +12912,7 @@ tsubst_template_args (tree t, tree args, tsubst_flags_t complain, tree in_decl)
             expanded_len_adjust += TREE_VEC_LENGTH (new_arg) - 1;
         }
       else if (ARGUMENT_PACK_P (orig_arg))
-        {
-          /* Substitute into each of the arguments.  */
-          new_arg = TYPE_P (orig_arg)
-            ? cxx_make_type (TREE_CODE (orig_arg))
-            : make_node (TREE_CODE (orig_arg));
-
-         tree pack_args = tsubst_template_args (ARGUMENT_PACK_ARGS (orig_arg),
-                                                args, complain, in_decl);
-          if (pack_args == error_mark_node)
-            new_arg = error_mark_node;
-         else
-           SET_ARGUMENT_PACK_ARGS (new_arg, pack_args);
-
-          if (TREE_CODE (new_arg) == NONTYPE_ARGUMENT_PACK)
-           TREE_CONSTANT (new_arg) = TREE_CONSTANT (orig_arg);
-        }
+       new_arg = tsubst_argument_pack (orig_arg, args, complain, in_decl);
       else
        new_arg = tsubst_template_arg (orig_arg, args, complain, in_decl);
 
@@ -12704,6 +13045,7 @@ tsubst_template_parm (tree t, tree args, tsubst_flags_t complain)
 
   default_value = TREE_PURPOSE (t);
   parm_decl = TREE_VALUE (t);
+  tree constraint = TEMPLATE_PARM_CONSTRAINTS (t);
 
   parm_decl = tsubst (parm_decl, args, complain, NULL_TREE);
   if (TREE_CODE (parm_decl) == PARM_DECL
@@ -12711,8 +13053,11 @@ tsubst_template_parm (tree t, tree args, tsubst_flags_t complain)
     parm_decl = error_mark_node;
   default_value = tsubst_template_arg (default_value, args,
                                       complain, NULL_TREE);
+  constraint = tsubst_constraint (constraint, args, complain, NULL_TREE);
 
-  return build_tree_list (default_value, parm_decl);
+  tree r = build_tree_list (default_value, parm_decl);
+  TEMPLATE_PARM_CONSTRAINTS (r) = constraint;
+  return r;
 }
 
 /* Substitute the ARGS into the indicated aggregate (or enumeration)
@@ -13138,14 +13483,11 @@ tsubst_function_decl (tree t, tree args, tsubst_flags_t complain,
       && !grok_op_properties (r, /*complain=*/false))
     return error_mark_node;
 
-  /* When instantiating a constrained member, substitute
-     into the constraints to create a new constraint.  */
+  /* Associate the constraints directly with the instantiation. We
+     don't substitute through the constraints; that's only done when
+     they are checked.  */
   if (tree ci = get_constraints (t))
-    if (member)
-      {
-       ci = tsubst_constraint_info (ci, argvec, complain, NULL_TREE);
-       set_constraints (r, ci);
-      }
+    set_constraints (r, ci);
 
   if (DECL_FRIEND_P (t) && DECL_FRIEND_CONTEXT (t))
     SET_DECL_FRIEND_CONTEXT (r,
@@ -14001,6 +14343,17 @@ tsubst_decl (tree t, tree args, tsubst_flags_t complain)
   return r;
 }
 
+/* Substitute into the complete parameter type list PARMS.  */
+
+tree
+tsubst_function_parms (tree parms,
+                      tree args,
+                      tsubst_flags_t complain,
+                      tree in_decl)
+{
+  return tsubst_arg_types (parms, args, NULL_TREE, complain, in_decl);
+}
+
 /* Substitute into the ARG_TYPES of a function type.
    If END is a TREE_CHAIN, leave it and any following types
    un-substituted.  */
@@ -14565,6 +14918,14 @@ tsubst (tree t, tree args, tsubst_flags_t complain, tree in_decl)
            if (code == TEMPLATE_TYPE_PARM)
              {
                int quals;
+
+               /* When building concept checks for the purpose of
+                  deducing placeholders, we can end up with wildcards
+                  where types are expected. Adjust this to the deduced
+                  value.  */
+               if (TREE_CODE (arg) == WILDCARD_DECL)
+                 arg = TREE_TYPE (TREE_TYPE (arg));
+
                gcc_assert (TYPE_P (arg));
 
                quals = cp_type_quals (arg) | cp_type_quals (t);
@@ -14696,10 +15057,10 @@ tsubst (tree t, tree args, tsubst_flags_t complain, tree in_decl)
 
                 if (TREE_CODE (t) == TEMPLATE_TYPE_PARM)
                  {
-                   /* Propagate constraints on placeholders.  */
+                   /* Propagate constraints on placeholders since they are
+                      only instantiated during satisfaction.  */
                    if (tree constr = PLACEHOLDER_TYPE_CONSTRAINTS (t))
-                     PLACEHOLDER_TYPE_CONSTRAINTS (r)
-                       = tsubst_constraint (constr, args, complain, in_decl);
+                     PLACEHOLDER_TYPE_CONSTRAINTS (r) = constr;
                    else if (tree pl = CLASS_PLACEHOLDER_TEMPLATE (t))
                      {
                        pl = tsubst_copy (pl, args, complain, in_decl);
@@ -18347,6 +18708,28 @@ tsubst_copy_and_build (tree t,
            RETURN (templ);
          }
 
+       if (concept_definition_p (templ))
+         {
+           tree check = build_concept_check (templ, targs, complain);
+           if (check == error_mark_node)
+             RETURN (error_mark_node);
+
+           tree id = unpack_concept_check (check);
+
+           /* If we built a function concept check, return the underlying
+              template-id. So we can evaluate it as a function call.  */
+           if (function_concept_p (TREE_OPERAND (id, 0)))
+             RETURN (id);
+
+           /* Evaluate the concept, if needed.  */
+           tree args = TREE_OPERAND (id, 1);
+           if (!uses_template_parms (args)
+               && !processing_constraint_expression_p ())
+             RETURN (evaluate_concept_check (check, complain));
+
+           RETURN (check);
+         }
+
        if (variable_template_p (templ))
          RETURN (lookup_and_finish_template_variable (templ, targs, complain));
 
@@ -19111,6 +19494,25 @@ tsubst_copy_and_build (tree t,
                       /*fn_p=*/NULL,
                       complain));
          }
+       else if (concept_check_p (function))
+         {
+           /* FUNCTION is a template-id referring to a concept definition.  */
+           tree id = unpack_concept_check (function);
+           tree tmpl = TREE_OPERAND (id, 0);
+           tree args = TREE_OPERAND (id, 1);
+
+           /* Calls to standard and variable concepts should have been
+              previously diagnosed.  */
+           gcc_assert (function_concept_p (tmpl));
+
+           /* Ensure the result is wrapped as a call expression.  */
+           ret = build_concept_check (tmpl, args, tf_warning_or_error);
+
+           /* Possibly evaluate the check if it is non-dependent.   */
+           if (!uses_template_parms (args)
+               && !processing_constraint_expression_p ())
+             ret = evaluate_concept_check (ret, complain);
+         }
        else
          ret = finish_call_expr (function, &call_args,
                                  /*disallow_virtual=*/qualified_p,
@@ -23202,10 +23604,11 @@ more_specialized_fn (tree pat1, tree pat2, int len)
      constrained template.  */
   if (!lose1 && !lose2)
     {
-      tree c1 = get_constraints (DECL_TEMPLATE_RESULT (pat1));
-      tree c2 = get_constraints (DECL_TEMPLATE_RESULT (pat2));
-      lose1 = !subsumes_constraints (c1, c2);
-      lose2 = !subsumes_constraints (c2, c1);
+      int winner = more_constrained (decl1, decl2);
+      if (winner > 0)
+       lose2 = true;
+      else if (winner < 0)
+       lose1 = true;
     }
 
   /* All things being equal, if the next argument is a pack expansion
@@ -23277,7 +23680,7 @@ more_specialized_partial_spec (tree tmpl, tree pat1, tree pat2)
   /* If both deductions succeed, the partial ordering selects the more
      constrained template.  */
   if (!winner && any_deductions)
-    return more_constrained (tmpl1, tmpl2);
+    winner = more_constrained (tmpl1, tmpl2);
 
   /* In the case of a tie where at least one of the templates
      has a parameter pack at the end, the template with the most
@@ -25712,7 +26115,7 @@ value_dependent_expression_p (tree expression)
       }
 
     case TEMPLATE_ID_EXPR:
-      return variable_concept_p (TREE_OPERAND (expression, 0));
+      return concept_definition_p (TREE_OPERAND (expression, 0));
 
     case CONSTRUCTOR:
       {
@@ -26112,14 +26515,14 @@ instantiation_dependent_r (tree *tp, int *walk_subtrees,
       return *tp;
 
     case CALL_EXPR:
-      /* Treat calls to function concepts as dependent. */
-      if (function_concept_check_p (*tp))
+      /* Treat concept checks as dependent. */
+      if (concept_check_p (*tp))
         return *tp;
       break;
 
     case TEMPLATE_ID_EXPR:
-      /* And variable concepts.  */
-      if (variable_concept_p (TREE_OPERAND (*tp, 0)))
+      /* Treat concept checks as dependent.  */
+      if (concept_check_p (*tp))
        return *tp;
       break;
 
@@ -26657,10 +27060,10 @@ build_non_dependent_expr (tree expr)
       /* Don't do this during nsdmi parsing as it can lead to
         unexpected recursive instantiations.  */
       && !parsing_nsdmi ()
-      /* Don't do this during concept expansion either and for
+      /* Don't do this during concept processing either and for
          the same reason.  */
-      && !expanding_concept ())
-    fold_non_dependent_expr (expr, tf_none);
+      && !processing_constraint_expression_p ())
+    fold_non_dependent_expr (expr);
 
   STRIP_ANY_LOCATION_WRAPPER (expr);
 
@@ -26755,8 +27158,7 @@ static tree
 make_auto_1 (tree name, bool set_canonical)
 {
   tree au = cxx_make_type (TEMPLATE_TYPE_PARM);
-  TYPE_NAME (au) = build_decl (input_location,
-                              TYPE_DECL, name, au);
+  TYPE_NAME (au) = build_decl (input_location, TYPE_DECL, name, au);
   TYPE_STUB_DECL (au) = TYPE_NAME (au);
   TEMPLATE_TYPE_PARM_INDEX (au) = build_template_parm_index
     (0, processing_template_decl + 1, processing_template_decl + 1,
@@ -26801,31 +27203,87 @@ template_placeholder_p (tree t)
   return is_auto (t) && CLASS_PLACEHOLDER_TEMPLATE (t);
 }
 
-/* Make a "constrained auto" type-specifier. This is an
-   auto type with constraints that must be associated after
-   deduction.  The constraint is formed from the given
-   CONC and its optional sequence of arguments, which are
-   non-null if written as partial-concept-id.  */
+/* Make a "constrained auto" type-specifier. This is an auto or
+  decltype(auto) type with constraints that must be associated after
+  deduction.  The constraint is formed from the given concept CON
+  and its optional sequence of template arguments ARGS.
 
-tree
-make_constrained_auto (tree con, tree args)
-{
-  tree type = make_auto_1 (auto_identifier, false);
+  TYPE must be the result of make_auto_type or make_decltype_auto_type. */
 
+static tree
+make_constrained_placeholder_type (tree type, tree con, tree args)
+{
   /* Build the constraint. */
   tree tmpl = DECL_TI_TEMPLATE (con);
-  tree expr = VAR_P (con) ? tmpl : ovl_make (tmpl);
-  expr = build_concept_check (expr, type, args);
+  tree expr = tmpl;
+  if (TREE_CODE (con) == FUNCTION_DECL)
+    expr = ovl_make (tmpl);
+  expr = build_concept_check (expr, type, args, tf_warning_or_error);
 
-  tree constr = normalize_expression (expr);
-  PLACEHOLDER_TYPE_CONSTRAINTS (type) = constr;
+  PLACEHOLDER_TYPE_CONSTRAINTS (type) = expr;
 
   /* Our canonical type depends on the constraint.  */
   TYPE_CANONICAL (type) = canonical_type_parameter (type);
 
   /* Attach the constraint to the type declaration. */
-  tree decl = TYPE_NAME (type);
-  return decl;
+  return TYPE_NAME (type);
+}
+
+/* Make a "constrained auto" type-specifier.  */
+
+tree
+make_constrained_auto (tree con, tree args)
+{
+  tree type = make_auto_1 (auto_identifier, false);
+  return make_constrained_placeholder_type (type, con, args);
+}
+
+/* Make a "constrained decltype(auto)" type-specifier.  */
+
+tree
+make_constrained_decltype_auto (tree con, tree args)
+{
+  tree type = make_auto_1 (decltype_auto_identifier, false);
+  /* FIXME: I don't know why this isn't done in make_auto_1.  */
+  AUTO_IS_DECLTYPE (type) = true;
+  return make_constrained_placeholder_type (type, con, args);
+}
+
+/* Build and return a concept definition. Like other templates, the
+   CONCEPT_DECL node is wrapped by a TEMPLATE_DECL.  This returns the
+   the TEMPLATE_DECL. */
+
+tree
+finish_concept_definition (cp_expr id, tree init)
+{
+  gcc_assert (identifier_p (id));
+  gcc_assert (processing_template_decl);
+
+  location_t loc = id.get_location();
+
+  /* A concept-definition shall not have associated constraints.  */
+  if (TEMPLATE_PARMS_CONSTRAINTS (current_template_parms))
+    {
+      error_at (loc, "a concept cannot be constrained");
+      TEMPLATE_PARMS_CONSTRAINTS (current_template_parms) = NULL_TREE;
+    }
+
+  /* A concept-definition shall appear in namespace scope.  Templates
+     aren't allowed in block scope, so we only need to check for class
+     scope.  */
+  if (TYPE_P (current_scope()) || !DECL_NAMESPACE_SCOPE_P (current_scope ()))
+    {
+      error_at (loc, "concept %qE not in namespace scope", *id);
+      return error_mark_node;
+    }
+
+  /* Initially build the concept declaration; it's type is bool.  */
+  tree decl = build_lang_decl_loc (loc, CONCEPT_DECL, *id, boolean_type_node);
+  DECL_CONTEXT (decl) = current_scope ();
+  DECL_INITIAL (decl) = init;
+
+  /* Push the enclosing template.  */
+  return push_template_decl (decl);
 }
 
 /* Given type ARG, return std::initializer_list<ARG>.  */
@@ -27656,13 +28114,17 @@ do_auto_deduction (tree type, tree init, tree auto_node,
 
   /* Check any placeholder constraints against the deduced type. */
   if (flag_concepts && !processing_template_decl)
-    if (tree constr = NON_ERROR (PLACEHOLDER_TYPE_CONSTRAINTS (auto_node)))
+    if (tree check = NON_ERROR (PLACEHOLDER_TYPE_CONSTRAINTS (auto_node)))
       {
         /* Use the deduced type to check the associated constraints. If we
            have a partial-concept-id, rebuild the argument list so that
            we check using the extra arguments. */
-        gcc_assert (TREE_CODE (constr) == CHECK_CONSTR);
-        tree cargs = CHECK_CONSTR_ARGS (constr);
+       check = unpack_concept_check (check);
+       gcc_assert (TREE_CODE (check) == TEMPLATE_ID_EXPR);
+       tree cdecl = TREE_OPERAND (check, 0);
+       if (OVL_P (cdecl))
+         cdecl = OVL_FIRST (cdecl);
+        tree cargs = TREE_OPERAND (check, 1);
         if (TREE_VEC_LENGTH (cargs) > 1)
           {
             cargs = copy_node (cargs);
@@ -27670,7 +28132,11 @@ do_auto_deduction (tree type, tree init, tree auto_node,
           }
         else
           cargs = targs;
-        if (!constraints_satisfied_p (constr, cargs))
+
+       /* Rebuild the check using the deduced arguments.  */
+       check = build_concept_check (cdecl, cargs, tf_none);
+
+       if (!constraints_satisfied_p (check, cargs))
           {
             if (complain & tf_warning_or_error)
               {
@@ -27695,7 +28161,7 @@ do_auto_deduction (tree type, tree init, tree auto_node,
                            "placeholder constraints");
                     break;
                   }
-                diagnose_constraints (input_location, constr, targs);
+               diagnose_constraints (input_location, check, targs);
               }
             return error_mark_node;
           }
@@ -27936,6 +28402,23 @@ append_type_to_template_for_access_check (tree templ,
                                              scope, location);
 }
 
+/* Recursively walk over && expressions searching for EXPR. Return a reference
+   to that expression.  */
+
+static tree *find_template_requirement (tree *t, tree key)
+{
+  if (*t == key)
+    return t;
+  if (TREE_CODE (*t) == TRUTH_ANDIF_EXPR)
+    {
+      if (tree *p = find_template_requirement (&TREE_OPERAND (*t, 0), key))
+       return p;
+      if (tree *p = find_template_requirement (&TREE_OPERAND (*t, 1), key))
+       return p;
+    }
+  return 0;
+}
+
 /* Convert the generic type parameters in PARM that match the types given in the
    range [START_IDX, END_IDX) from the current_template_parms into generic type
    packs.  */
@@ -27957,9 +28440,8 @@ convert_generic_types_to_packs (tree parm, int start_idx, int end_idx)
       /* Create a distinct parameter pack type from the current parm and add it
         to the replacement args to tsubst below into the generic function
         parameter.  */
-
-      tree o = TREE_TYPE (TREE_VALUE
-                         (TREE_VEC_ELT (current, i)));
+      tree node = TREE_VEC_ELT (current, i);
+      tree o = TREE_TYPE (TREE_VALUE (node));
       tree t = copy_type (o);
       TEMPLATE_TYPE_PARM_INDEX (t)
        = reduce_template_parm_level (TEMPLATE_TYPE_PARM_INDEX (o),
@@ -27970,7 +28452,26 @@ convert_generic_types_to_packs (tree parm, int start_idx, int end_idx)
       TEMPLATE_TYPE_PARAMETER_PACK (t) = true;
       TYPE_CANONICAL (t) = canonical_type_parameter (t);
       TREE_VEC_ELT (replacement, i) = t;
-      TREE_VALUE (TREE_VEC_ELT (current, i)) = TREE_CHAIN (t);
+
+      /* Replace the current template parameter with new pack.  */
+      TREE_VALUE (node) = TREE_CHAIN (t);
+
+      /* Surgically adjust the associated constraint of adjusted parameter
+         and it's corresponding contribution to the current template
+         requirements.  */
+      if (tree constr = TEMPLATE_PARM_CONSTRAINTS (node))
+       {
+         tree id = unpack_concept_check (constr);
+         TREE_VEC_ELT (TREE_OPERAND (id, 1), 0) = template_parm_to_arg (t);
+         tree fold = finish_left_unary_fold_expr (constr, TRUTH_ANDIF_EXPR);
+         TEMPLATE_PARM_CONSTRAINTS (node) = fold;
+
+         /* If there was a constraint, we also need to replace that in
+            the template requirements, which we've already built.  */
+         tree *reqs = &TEMPLATE_PARMS_CONSTRAINTS (current_template_parms);
+         reqs = find_template_requirement (reqs, constr);
+         *reqs = fold;
+       }
     }
 
   for (int i = end_idx, e = TREE_VEC_LENGTH (current); i < e; ++i)
@@ -27987,27 +28488,6 @@ convert_generic_types_to_packs (tree parm, int start_idx, int end_idx)
   return tsubst (parm, replacement, tf_none, NULL_TREE);
 }
 
-/* Entries in the decl_constraint hash table. */
-struct GTY((for_user)) constr_entry
-{
-  tree decl;
-  tree ci;
-};
-
-/* Hashing function and equality for constraint entries. */
-struct constr_hasher : ggc_ptr_hash<constr_entry>
-{
-  static hashval_t hash (constr_entry *e)
-  {
-    return (hashval_t)DECL_UID (e->decl);
-  }
-
-  static bool equal (constr_entry *e1, constr_entry *e2)
-  {
-    return e1->decl == e2->decl;
-  }
-};
-
 /* A mapping from declarations to constraint information. Note that
    both templates and their underlying declarations are mapped to the
    same constraint information.
@@ -28015,7 +28495,7 @@ struct constr_hasher : ggc_ptr_hash<constr_entry>
    FIXME: This is defined in pt.c because garbage collection
    code is not being generated for constraint.cc. */
 
-static GTY (()) hash_table<constr_hasher> *decl_constraints;
+static GTY ((cache)) tree_cache_map *decl_constraints;
 
 /* Returns the template constraints of declaration T. If T is not
    constrained, return NULL_TREE. Note that T must be non-null. */
@@ -28025,14 +28505,15 @@ get_constraints (tree t)
 {
   if (!flag_concepts)
     return NULL_TREE;
+  if (!decl_constraints)
+    return NULL_TREE;
 
   gcc_assert (DECL_P (t));
   if (TREE_CODE (t) == TEMPLATE_DECL)
     t = DECL_TEMPLATE_RESULT (t);
-  constr_entry elt = { t, NULL_TREE };
-  constr_entry* found = decl_constraints->find (&elt);
+  tree* found = decl_constraints->get (t);
   if (found)
-    return found->ci;
+    return *found;
   else
     return NULL_TREE;
 }
@@ -28050,12 +28531,8 @@ set_constraints (tree t, tree ci)
   gcc_assert (t && flag_concepts);
   if (TREE_CODE (t) == TEMPLATE_DECL)
     t = DECL_TEMPLATE_RESULT (t);
-  gcc_assert (!get_constraints (t));
-  constr_entry elt = {t, ci};
-  constr_entry** slot = decl_constraints->find_slot (&elt, INSERT);
-  constr_entry* entry = ggc_alloc<constr_entry> ();
-  *entry = elt;
-  *slot = entry;
+  bool found = hash_map_safe_put<hm_ggc> (decl_constraints, t, ci);
+  gcc_assert (!found);
 }
 
 /* Remove the associated constraints of the declaration T.  */
@@ -28067,149 +28544,8 @@ remove_constraints (tree t)
   if (TREE_CODE (t) == TEMPLATE_DECL)
     t = DECL_TEMPLATE_RESULT (t);
 
-  constr_entry elt = {t, NULL_TREE};
-  constr_entry** slot = decl_constraints->find_slot (&elt, NO_INSERT);
-  if (slot)
-    decl_constraints->clear_slot (slot);
-}
-
-/* Memoized satisfaction results for declarations. This
-   maps the pair (constraint_info, arguments) to the result computed
-   by constraints_satisfied_p.  */
-
-struct GTY((for_user)) constraint_sat_entry
-{
-  tree ci;
-  tree args;
-  tree result;
-};
-
-/* Hashing function and equality for constraint entries. */
-
-struct constraint_sat_hasher : ggc_ptr_hash<constraint_sat_entry>
-{
-  static hashval_t hash (constraint_sat_entry *e)
-  {
-    hashval_t val = iterative_hash_object(e->ci, 0);
-    return iterative_hash_template_arg (e->args, val);
-  }
-
-  static bool equal (constraint_sat_entry *e1, constraint_sat_entry *e2)
-  {
-    return e1->ci == e2->ci && comp_template_args (e1->args, e2->args);
-  }
-};
-
-/* Memoized satisfaction results for concept checks. */
-
-struct GTY((for_user)) concept_spec_entry
-{
-  tree tmpl;
-  tree args;
-  tree result;
-};
-
-/* Hashing function and equality for constraint entries.  */
-
-struct concept_spec_hasher : ggc_ptr_hash<concept_spec_entry>
-{
-  static hashval_t hash (concept_spec_entry *e)
-  {
-    return hash_tmpl_and_args (e->tmpl, e->args);
-  }
-
-  static bool equal (concept_spec_entry *e1, concept_spec_entry *e2)
-  {
-    ++comparing_specializations;
-    bool eq = e1->tmpl == e2->tmpl && comp_template_args (e1->args, e2->args);
-    --comparing_specializations;
-    return eq;
-  }
-};
-
-static GTY (()) hash_table<constraint_sat_hasher> *constraint_memos;
-static GTY (()) hash_table<concept_spec_hasher> *concept_memos;
-
-/* Search for a memoized satisfaction result. Returns one of the
-   truth value nodes if previously memoized, or NULL_TREE otherwise.   */
-
-tree
-lookup_constraint_satisfaction (tree ci, tree args)
-{
-  constraint_sat_entry elt = { ci, args, NULL_TREE };
-  constraint_sat_entry* found = constraint_memos->find (&elt);
-  if (found)
-    return found->result;
-  else
-    return NULL_TREE;
-}
-
-/* Memoize the result of a satisfication test. Returns the saved result.  */
-
-tree
-memoize_constraint_satisfaction (tree ci, tree args, tree result)
-{
-  constraint_sat_entry elt = {ci, args, result};
-  constraint_sat_entry** slot = constraint_memos->find_slot (&elt, INSERT);
-  constraint_sat_entry* entry = ggc_alloc<constraint_sat_entry> ();
-  *entry = elt;
-  *slot = entry;
-  return result;
-}
-
-/* Search for a memoized satisfaction result for a concept. */
-
-tree
-lookup_concept_satisfaction (tree tmpl, tree args)
-{
-  concept_spec_entry elt = { tmpl, args, NULL_TREE };
-  concept_spec_entry* found = concept_memos->find (&elt);
-  if (found)
-    return found->result;
-  else
-    return NULL_TREE;
-}
-
-/* Memoize the result of a concept check. Returns the saved result.  */
-
-tree
-memoize_concept_satisfaction (tree tmpl, tree args, tree result)
-{
-  concept_spec_entry elt = {tmpl, args, result};
-  concept_spec_entry** slot = concept_memos->find_slot (&elt, INSERT);
-  concept_spec_entry* entry = ggc_alloc<concept_spec_entry> ();
-  *entry = elt;
-  *slot = entry;
-  return result;
-}
-
-static GTY (()) hash_table<concept_spec_hasher> *concept_expansions;
-
-/* Returns a prior concept specialization. This returns the substituted
-   and normalized constraints defined by the concept.  */
-
-tree
-get_concept_expansion (tree tmpl, tree args)
-{
-  concept_spec_entry elt = { tmpl, args, NULL_TREE };
-  concept_spec_entry* found = concept_expansions->find (&elt);
-  if (found)
-    return found->result;
-  else
-    return NULL_TREE;
-}
-
-/* Save a concept expansion for later.  */
-
-tree
-save_concept_expansion (tree tmpl, tree args, tree def)
-{
-  concept_spec_entry elt = {tmpl, args, def};
-  concept_spec_entry** slot = concept_expansions->find_slot (&elt, INSERT);
-  concept_spec_entry* entry = ggc_alloc<concept_spec_entry> ();
-  *entry = elt;
-  *slot = entry;
-  return def;
+  if (decl_constraints)
+    decl_constraints->remove (t);
 }
 
 static hashval_t
@@ -28306,13 +28642,11 @@ init_constraint_processing (void)
   if (!flag_concepts)
     return;
 
-  decl_constraints = hash_table<constr_hasher>::create_ggc(37);
-  constraint_memos = hash_table<constraint_sat_hasher>::create_ggc(37);
-  concept_memos = hash_table<concept_spec_hasher>::create_ggc(37);
-  concept_expansions = hash_table<concept_spec_hasher>::create_ggc(37);
   subsumption_table = hash_table<subsumption_hasher>::create_ggc(37);
 }
 
+GTY(()) tree current_failed_constraint;
+
 /* __integer_pack(N) in a pack expansion expands to a sequence of numbers from
    0..N-1.  */
 
index b441af8ff09cb1913106719dc027a739a5573175..5bc8679efd5f858a940b35eb0f1e5cfa61d20658 100644 (file)
@@ -835,7 +835,10 @@ accessible_p (tree type, tree decl, bool consider_local_p)
      in default arguments for template parameters), and access
      checking should be performed in the outermost parameter list.  */
   if (processing_template_decl
-      && !expanding_concept ()
+      /* FIXME CWG has been talking about doing access checking in the context
+        of the constraint-expression, rather than the constrained declaration,
+        in which case we would want to remove this test.  */
+      && !processing_constraint_expression_p ()
       && (!processing_template_parmlist || processing_template_decl > 1))
     return 1;
 
index 5a8c4d237e0d7905e65d29680ae56bea0b8caf02..1839013fbbcba74445dd68167f16b7bd31c23a9d 100644 (file)
@@ -1788,10 +1788,10 @@ finish_mem_initializers (tree mem_inits)
 
 /* Obfuscate EXPR if it looks like an id-expression or member access so
    that the call to finish_decltype in do_auto_deduction will give the
-   right result.  */
+   right result.  If EVEN_UNEVAL, do this even in unevaluated context.  */
 
 tree
-force_paren_expr (tree expr)
+force_paren_expr (tree expr, bool even_uneval)
 {
   /* This is only needed for decltype(auto) in C++14.  */
   if (cxx_dialect < cxx14)
@@ -1799,7 +1799,7 @@ force_paren_expr (tree expr)
 
   /* If we're in unevaluated context, we can't be deducing a
      return/initializer type, so we don't need to mess with this.  */
-  if (cp_unevaluated_operand)
+  if (cp_unevaluated_operand && !even_uneval)
     return expr;
 
   if (!DECL_P (tree_strip_any_location_wrapper (expr))
@@ -2589,6 +2589,27 @@ finish_call_expr (tree fn, vec<tree, va_gc> **args, bool disallow_virtual,
                                      /*fn_p=*/NULL,
                                      complain);
     }
+  else if (concept_check_p (fn))
+    {
+      /* FN is actually a template-id referring to a concept definition.  */
+      tree id = unpack_concept_check (fn);
+      tree tmpl = TREE_OPERAND (id, 0);
+      tree args = TREE_OPERAND (id, 1);
+
+      if (!function_concept_p (tmpl))
+       {
+         error_at (EXPR_LOC_OR_LOC (fn, input_location),
+                   "cannot call a concept as a function");
+         return error_mark_node;
+       }
+
+      /* Ensure the result is wrapped as a call expression.  */
+      result = build_concept_check (tmpl, args, tf_warning_or_error);
+
+      /* Evaluate the check if it is non-dependent.   */
+      if (!uses_template_parms (args))
+       result = evaluate_concept_check (result, complain);
+    }
   else if (is_overloaded_fn (fn))
     {
       /* If the function is an overloaded builtin, resolve it.  */
@@ -3833,9 +3854,10 @@ finish_id_expression_1 (tree id_expression,
       if (! error_operand_p (decl)
          && !dependent_p
          && integral_constant_expression_p
-         && ! decl_constant_var_p (decl)
+         && !decl_constant_var_p (decl)
          && TREE_CODE (decl) != CONST_DECL
-         && ! builtin_valid_in_constant_expr_p (decl))
+         && !builtin_valid_in_constant_expr_p (decl)
+         && !concept_check_p (decl))
        {
          if (!allow_non_integral_constant_expression_p)
            {
@@ -3851,12 +3873,23 @@ finish_id_expression_1 (tree id_expression,
        decl = wrap;
       else if (TREE_CODE (decl) == TEMPLATE_ID_EXPR
               && !dependent_p
-              && variable_template_p (TREE_OPERAND (decl, 0)))
+              && variable_template_p (TREE_OPERAND (decl, 0))
+              && !concept_check_p (decl))
        {
          decl = finish_template_variable (decl);
          mark_used (decl);
          decl = convert_from_reference (decl);
        }
+      else if (concept_check_p (decl))
+       {
+         /* If this is a standard or variable concept check, potentially
+            evaluate it. Function concepts need to be called as functions,
+            so don't try evaluating them here.  */
+         tree tmpl = TREE_OPERAND (decl, 0);
+         tree args = TREE_OPERAND (decl, 1);
+         if (!function_concept_p (tmpl) && !uses_template_parms (args))
+           decl = evaluate_concept_check (decl, tf_warning_or_error);
+       }
       else if (scope)
        {
          if (TREE_CODE (decl) == SCOPE_REF)
@@ -3929,6 +3962,16 @@ finish_id_expression_1 (tree id_expression,
 
          decl = baselink_for_fns (decl);
        }
+      else if (concept_check_p (decl))
+       {
+         /* If this is a standard or variable concept check, potentially
+            evaluate it. Function concepts need to be called as functions,
+            so don't try evaluating them here.  */
+         tree tmpl = TREE_OPERAND (decl, 0);
+         tree args = TREE_OPERAND (decl, 1);
+         if (!function_concept_p (tmpl) && !uses_template_parms (args))
+           decl = evaluate_concept_check (decl, tf_warning_or_error);
+       }
       else
        {
          if (DECL_P (decl) && DECL_NONLOCAL (decl)
index 1d9e9b8d05d48577269fb9720c394a7a106d3960..ba334e7d5cb12a9643b875d749ba2ca324261987 100644 (file)
@@ -9644,7 +9644,8 @@ check_return_expr (tree retval, bool *no_warning)
          if (!retval)
            retval = void_node;
          auto_node = type_uses_auto (pattern);
-         type = do_auto_deduction (pattern, retval, auto_node);
+         type = do_auto_deduction (pattern, retval, auto_node,
+                                   tf_warning_or_error, adc_return_type);
        }
 
       if (type == error_mark_node)
index 54d6b848157bbe3e6d6a1e03d2d1571a2224b35d..2402c38fdf3bdee85f1e7d340c915c9d353dd551 100644 (file)
@@ -923,7 +923,7 @@ store_init_value (tree decl, tree init, vec<tree, va_gc>** cleanups, int flags)
     return split_nonconstant_init (decl, value);
 
   /* DECL may change value; purge caches.  */
-  clear_cv_and_fold_caches ();
+  clear_cv_and_fold_caches (TREE_STATIC (decl));
 
   /* If the value is a constant, just put it in DECL_INITIAL.  If DECL
      is an automatic variable, the middle end will turn this into a
@@ -2242,8 +2242,7 @@ build_functional_cast (tree exp, tree parms, tsubst_flags_t complain)
       if (!CLASS_PLACEHOLDER_TEMPLATE (anode))
        {
          if (complain & tf_error)
-           error_at (DECL_SOURCE_LOCATION (TEMPLATE_TYPE_DECL (anode)),
-                     "invalid use of %qT", anode);
+           error ("invalid use of %qT", anode);
          return error_mark_node;
        }
       else if (!parms)
index 20e10c018048c8404a96dd01fc6bf6aafa755725..dc9a3e5e1a3f4f9b2f4798a85dd12f1b47502172 100644 (file)
@@ -2512,14 +2512,17 @@ exhaustion is signalled by throwing @code{std::bad_alloc}.  See also
 @samp{new (nothrow)}.
 
 @item -fconcepts
+@itemx -fconcepts-ts
 @opindex fconcepts
-Enable support for the C++ Extensions for Concepts Technical
-Specification, ISO 19217 (2015), which allows code like
-
-@smallexample
-template <class T> concept bool Addable = requires (T t) @{ t + t; @};
-template <Addable T> T add (T a, T b) @{ return a + b; @}
-@end smallexample
+@opindex fconcepts-ts
+Below @option{-std=c++2a}, @option{-fconcepts} enables support for the
+C++ Extensions for Concepts Technical Specification, ISO 19217 (2015).
+
+With @option{-std=c++2a} and above, Concepts are part of the language
+standard, so @option{-fconcepts} defaults to on.  But the standard
+specification of Concepts differs significantly from the TS, so some
+constructs that were allowed in the TS but didn't make it into the
+standard can still be enabled by @option{-fconcepts-ts}.
 
 @item -fconstexpr-depth=@var{n}
 @opindex fconstexpr-depth
index 26a3fcce24337beb5382b77d045e0da0d01f1715..931865552949bca8a28b0c7c9d317e7e77b4f890 100644 (file)
@@ -1,3 +1,8 @@
+2019-10-08  Andrew Sutton  <asutton@lock3software.com>
+
+       * lib/target-supports.exp (check_effective_target_concepts): Check
+       for std=c++2a.
+
 2019-10-09  Paolo Carlini  <paolo.carlini@oracle.com>
 
        * c-c++-common/Waddress-1.c: Test locations too.
diff --git a/gcc/testsuite/g++.dg/concepts/alias1.C b/gcc/testsuite/g++.dg/concepts/alias1.C
deleted file mode 100644 (file)
index 279a478..0000000
+++ /dev/null
@@ -1,16 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C() { return __is_class(T); }
-
-template<typename T>
-  requires C<T>()
-    using X = T*;
-
-struct S { };
-
-int main()
-{
-  X<S> x1;
-}
diff --git a/gcc/testsuite/g++.dg/concepts/alias2.C b/gcc/testsuite/g++.dg/concepts/alias2.C
deleted file mode 100644 (file)
index 06ffb1a..0000000
+++ /dev/null
@@ -1,14 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C() { return __is_class(T); }
-
-template<C T> using X = T*;
-
-struct S { };
-
-int main()
-{
-  X<S> x1;
-}
diff --git a/gcc/testsuite/g++.dg/concepts/alias3.C b/gcc/testsuite/g++.dg/concepts/alias3.C
deleted file mode 100644 (file)
index 2901c04..0000000
+++ /dev/null
@@ -1,14 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C() { return __is_class(T); }
-
-template<typename T>
-  requires C<T>()
-    using X = T*;
-
-int main()
-{
-  X<int> x1; // { dg-error "constraint|invalid" }
-}
diff --git a/gcc/testsuite/g++.dg/concepts/alias4.C b/gcc/testsuite/g++.dg/concepts/alias4.C
deleted file mode 100644 (file)
index 2c9f5de..0000000
+++ /dev/null
@@ -1,20 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C() { return __is_class(T); }
-
-template<typename T>
-  requires C<T>()
-    using X = T*;
-
-// BUG: Alias templates are expanded at the point of use, regardless
-// of whether or not they are dependent. This causes T* to be substituted
-// without acutally checking the constraints.
-template<typename T>
-  using Y = X<T>;
-
-int main()
-{
-  Y<int> y1; // { dg-error "" "" { xfail *-*-* } }
-}
index 2682940cedd36185c9dcdcd91a715c37ece4f147..e05330610fc66b823997a5846bddc3676691dd1b 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T1, class T2> class A { };
index abfb20191251a8e93aee61d86b728aa8aac9c147..27a6afa4ed97e64abf1aafd33eb67e5d4a8a3dfd 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class...> class tuple {};
index 4eb2ae8f7d53d7af1163b5505536cfc56bd3b6f2..8bf3fa9b1cea4223805f964e5e3928dce7f7f988 100644 (file)
@@ -1,6 +1,6 @@
 // PR c++/85006
-// { dg-do compile { target c++17 } }
-// { dg-additional-options "-fconcepts" }
+// { dg-do compile { target c++17_only } }
+// { dg-options "-fconcepts" }
 
 template<typename... Ts> struct A {};
 
index 936dd6826f0e4ede0bf1f25cc86334e9b6781991..3359700775292be787f402fc3d1e477d799cbb0d 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T>
diff --git a/gcc/testsuite/g++.dg/concepts/class.C b/gcc/testsuite/g++.dg/concepts/class.C
deleted file mode 100644 (file)
index dc5523e..0000000
+++ /dev/null
@@ -1,53 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool Class() { return __is_class(T); }
-
-template<typename T>
-  concept bool Union() { return __is_union(T); }
-
-
-// Check ordering of specializations
-template<typename T>
-  concept bool One() { return sizeof(T) >= 4; }
-
-template<typename T>
-  concept bool Two() { return One<T>() && sizeof(T) >= 8; }
-
-// Check non-overlapping specializations
-template<typename T>
-  struct S1 { static const int value = 0; };
-
-template<Class T>
-  struct S1<T> { static const int value = 1; };
-
-template<Union T>
-  struct S1<T> { static const int value = 2; };
-
-struct S { };
-union U { };
-
-static_assert(S1<int>::value == 0, "");
-static_assert(S1<S>::value == 1, "");
-static_assert(S1<U>::value == 2, "");
-
-
-// Check ordering of partial specializaitons
-template<typename T>
-  struct S2 { static const int value = 0;  };
-
-template<One T>
-  struct S2<T> { static const int value = 1; };
-
-template<Two T>
-  struct S2<T> { static const int value = 2; };
-
-struct one_type { char x[4]; };
-struct two_type { char x[8]; };
-
-static_assert(S2<char>::value == 0, "");
-static_assert(S2<one_type>::value == 1, "");
-static_assert(S2<two_type>::value == 2, "");
-
-int main() { }
diff --git a/gcc/testsuite/g++.dg/concepts/class1.C b/gcc/testsuite/g++.dg/concepts/class1.C
deleted file mode 100644 (file)
index a738e6e..0000000
+++ /dev/null
@@ -1,15 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C() { return __is_class(T); }
-
-template<typename T>
-  requires C<T>()
-    struct S { };
-
-struct X { };
-
-S<X> sx;
-
-int main() { }
diff --git a/gcc/testsuite/g++.dg/concepts/class2.C b/gcc/testsuite/g++.dg/concepts/class2.C
deleted file mode 100644 (file)
index ec87181..0000000
+++ /dev/null
@@ -1,15 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C() { return __is_class(T); }
-
-template<typename T>
-  requires C<T>()
-    struct S { };
-
-struct X { };
-
-S<int> sx; // { dg-error "constraint|invalid" }
-
-int main() { }
diff --git a/gcc/testsuite/g++.dg/concepts/class3.C b/gcc/testsuite/g++.dg/concepts/class3.C
deleted file mode 100644 (file)
index 256370d..0000000
+++ /dev/null
@@ -1,15 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C() { return __is_class(T); }
-
-// Check class redeclaration with alternative spellings.
-template<typename T> requires C<T>() struct S;
-template<C T> struct S { };
-
-struct X { };
-
-// S<X> sx;
-
-int main() { }
diff --git a/gcc/testsuite/g++.dg/concepts/class4.C b/gcc/testsuite/g++.dg/concepts/class4.C
deleted file mode 100644 (file)
index b583e55..0000000
+++ /dev/null
@@ -1,22 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool Class() { return __is_class(T); }
-
-template<typename T>
-  concept bool Union() { return __is_union(T); }
-
-// Check non-overlapping specializations
-template<typename T> struct S1 { static const int value = 0; };
-template<Class T> struct S1<T> { static const int value = 1; };
-template<Union T> struct S1<T> { static const int value = 2; };
-
-struct S { };
-union U { };
-
-static_assert(S1<int>::value == 0, "");
-static_assert(S1<S>::value == 1, "");
-static_assert(S1<U>::value == 2, "");
-
-int main() { }
index 7bf620edc5cafef3c436d72d9ddd5cca7af84f59..ac9d7e83e9de0376cf69c3a1170976dad65c63f3 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index bdd60918c8e096ea5477dcd0c583c757140ec8c4..f2345b19b0409bde2624dc27cb9d3c73aa76646a 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
@@ -15,5 +15,3 @@ struct one_type { char x[4]; };
 
 // Constraints are checked even when decls are not instantiatied.
 S4<one_type>* x4b; // { dg-error "constraint|invalid" }
-
-int main() { }
index 87f2ac90b20177a85399d9f2329d34d80efdfd3f..b9a544486ace3a784aebcca50bf5cf74e5bbe37c 100644 (file)
@@ -1,11 +1,11 @@
 // PR c++/84551
-// { dg-do compile { target c++17 } }
-// { dg-options "-g -O -fconcepts" }
+// { dg-do compile { target c++17_only } }
+// { dg-options "-fconcepts" }
 
 template<typename> concept bool C() { return true; }
 
-template<template<typename T> requires C<T>() class> struct A {};
+template<template<typename T> requires C<T>() class TT> struct A {};
 
-template<typename> requires true struct B {};
+template<typename U> requires C<U>() struct B {};
 
 A<B> a;
index 019a8ce1130ed579cd4c84cec515c71aea46bf21..6a461a50366928256cf8a90d2da8075f63a7ae73 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 typedef concept int CINT; // { dg-error "'concept' cannot appear in a typedef declaration" }
@@ -12,10 +12,10 @@ concept bool f3(); // { dg-error "14:concept .f3. has no definition" }
 struct X
 {
   template<typename T>
-  concept int f4() { return 0; } // { dg-error "return type|member function" }
-  concept bool f5() { return true; } // { dg-error "member function" }
+  concept int f4() { return 0; } // { dg-error "cannot be a member" }
+  concept f5 = true; // { dg-error "declared 'concept'" }
   template<typename T>
-  static concept bool f6() { return true; } // { dg-error "a concept cannot be a member function" }
+  static concept f6 = true; // { dg-error "declared 'concept'" }
   static concept bool x; // { dg-error "declared 'concept'" }
                         // { dg-error "uninitialized 'const" "" { target *-*-* } .-1 }
   concept int x2; // { dg-error "declared 'concept'" }
index bebbda1a1faea11a3c878a8bc4da9fefec8b35f6..eba57713089fe2bc5622d94a5f3bbd06c35ce0e7 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/67007
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class U>
index 9bb150605731aadc107482269ac664c7acbfdf01..ced56d400badc3a15ccbb50fd4abaff179e5b524 100644 (file)
@@ -1,19 +1,19 @@
 // PR c++/67159
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T, class U>
 concept bool SameAs = __is_same_as(T, U);
 
 template <class T>
-concept bool R1 = requires (T& t) {
-  { t.begin() } -> T
-  { t.end() } -> SameAs<T*>;
+concept bool R1 = requires (T& t) { // { dg-message "in requirements" }
+  { t.begin() } -> T;          // { dg-error "no match" }
+  { t.end() } -> SameAs<T*>;   // { dg-error "does not satisfy" }
 };
 
 template <class T>
-concept bool R2 = requires (T& t) {
-  { t.end() } -> SameAs<T*>;
+concept bool R2 = requires (T& t) { // { dg-message "in requirements" }
+  { t.end() } -> SameAs<T*>;   // { dg-error "does not satisfy" }
 };
 
 struct foo {
diff --git a/gcc/testsuite/g++.dg/concepts/disjunction1.C b/gcc/testsuite/g++.dg/concepts/disjunction1.C
deleted file mode 100644 (file)
index 930adf4..0000000
+++ /dev/null
@@ -1,60 +0,0 @@
-// PR c++/66962
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template <typename> struct remove_cv;
-template <typename> struct is_reference;
-template <typename> void declval();
-template <typename> struct is_constructible;
-template <typename> struct is_nothrow_constructible;
-template <typename _Tp> using remove_cv_t = typename remove_cv<_Tp>::type;
-template <typename> struct Trans_NS_extension_apply_list;
-template <typename T> using _t = typename T::type;
-template <class> void ImplicitlyConvertibleTo();
-template <class> void Assignable();
-template <class T, class... Args> int ConstructibleObject = requires { T{}; };
-template <class T, class... Args>
-concept bool BindableReference =
-    is_reference<T>::value &&is_constructible<T>::value;
-template <class T, class... Args> concept bool Constructible() {
-  return ConstructibleObject<T> || BindableReference<T, Args...>;
-}
-template <class T> concept bool DefaultConstructible() {
-  return Constructible<T>() && requires { new T[0]; };
-}
-template <class T> concept bool MoveConstructible() {
-  return Constructible<T>() && ImplicitlyConvertibleTo<T>;
-}
-template <class T> concept bool Movable() {
-  return MoveConstructible<T>() && Assignable<T &&>;
-}
-template <class, class> int Swappable_ = requires { 0; };
-template <class T, class U> int Swappable();
-template <class T> concept bool Dereferencable = requires{{0}};
-template <Dereferencable R> using RvalueReferenceType = decltype(0);
-template <class T> int IsValueType;
-template <class> struct value_type;
-template <class T>
-requires IsValueType<
-    _t<value_type<remove_cv_t<T>>>> using ValueType =
-    _t<value_type<remove_cv_t<T>>>;
-template <class I> concept bool Readable() {
-  return Movable<I>() && DefaultConstructible<I>() &&
-         Dereferencable<const I> && requires{{0}};
-}
-template <class Out, class T> concept bool MoveWritable() {
-  return Movable<Out>() && DefaultConstructible<Out>() &&
-         Dereferencable<Out>;
-}
-template <class In, class Out> concept bool IndirectlyMovable() {
-  return Readable<In>() && Movable<ValueType<In>>() &&
-         Constructible<ValueType<In>>() &&
-         MoveWritable<Out, RvalueReferenceType<In>>() &&
-         MoveWritable<Out, ValueType<In>>();
-}
-IndirectlyMovable { In, Out }
-int is_nothrow_indirectly_movable_v =
-    is_nothrow_constructible<ValueType<In>>::value;
-template <Readable R1, Readable R2>
-    requires IndirectlyMovable<R1, R2>() &&
-    IndirectlyMovable<R2, R1>() void iter_swap2();
index f865d5ec2c70f4fccc194c2f817684fc431be338..6f5bab1e106d3c9c4960967bbb03182cabf009b6 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/66092
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 #include <type_traits>
@@ -29,11 +29,14 @@ template <typename T, typename U, typename... Args>
   concept bool Similar = true;
 
 template <typename... Args>
-requires Same<Args...>() // { dg-error "invalid reference" }
+requires Same<Args...>() // { dg-error "" "" { xfail *-*-* } }
   void foo( Args... args ) {}
+// FIXME: The new method of building concept checks is suppressing the
+// diagnostic for the invalid substitution. This produces an invalid
+// requires-clause, which still prevents the function from being resolved.
 
 template <typename... Args>
-requires Similar<Args...> // { dg-error "invalid reference" }
+requires Similar<Args...> // { dg-error "pack expansion" }
   void bar( Args... args ) {}
 
 int main()
index faec3543461f7d8ccd8f469f4607969f914f36dc..640c2b5ec0ddfb0211f94c9d4ac3be357f0af448 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do link { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 // Check equivalence of short- and longhand declarations.
index 2094ca9f388b996f79b04ae2887b59c424c47e8c..dff719b86a5399112dc6fb2d87e889231e060451 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do run { target c++17 } }
+// { dg-do link { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 
@@ -21,9 +21,9 @@ int main() {
 
 void f1(C, C) { }
 
-template<C T>
-void f2(T, T) { }
+template<C T1, C T2>
+void f2(T1, T2) { }
 
-template<typename T>
-  requires C<T>
-void f3(T, T) { }
+template<typename T, typename U>
+  requires C<T> && C<U>
+void f3(T, U) { }
diff --git a/gcc/testsuite/g++.dg/concepts/explicit-inst4.C b/gcc/testsuite/g++.dg/concepts/explicit-inst4.C
deleted file mode 100644 (file)
index 20f4377..0000000
+++ /dev/null
@@ -1,18 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C() { return __is_class(T); }
-
-template<typename T>
-  concept bool D() { return C<T>() && __is_empty(T); }
-
-template<typename T>
-  struct S {
-    void g() requires C<T>() { } // #1
-    void g() requires D<T>() { } // #2
-  };
-
-template void S<int>::g(); // { dg-error "match" }
-
-int main() { }
diff --git a/gcc/testsuite/g++.dg/concepts/explicit-spec3.C b/gcc/testsuite/g++.dg/concepts/explicit-spec3.C
deleted file mode 100644 (file)
index fd48da1..0000000
+++ /dev/null
@@ -1,14 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C() { return __is_class(T); }
-
-template<C T> struct S;
-
-struct X { };
-
-// Not a valid explicit specialization, int does not satisfy C.
-template<> struct S<int> { }; // { dg-error "constraint" }
-
-int main() { }
index 33dad0a47a6892aa77c9e88981b0433df8733cc6..ba4c48d7dcc649c3d0c3fa94b6eb93e8843b71e6 100644 (file)
@@ -1,6 +1,8 @@
-// { dg-do run { target c++17 } }
+// { dg-do run { target c++17_only } }
 // { dg-options "-fconcepts" }
 
+// TODO: ICE on gimplify 16?
+
 #include <cassert>
 #include <iostream>
 
@@ -8,13 +10,9 @@ template<typename T>
   concept bool C1 = __is_class(T);
 
 template<typename T>
-  concept bool C2() { return __is_class(T); }
-
-template<typename T>
-  concept bool C3() { return requires (T a) { ++a; }; }
+  concept bool C3 = requires (T a) { ++a; };
 
 int main() {
   if (C1<int>) assert(false);
-  if (C2<int>()) assert(false);
-  if (!C3<int>()) assert(false);
+  if (!C3<int>) assert(false);
 }
index c5447df1d87a439e0a78f957d70f5fe51d421eb7..1cff60542f5a52e25f19ce18c4e913d90bb56781 100644 (file)
@@ -1,16 +1,16 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
 concept bool C1()
 {
-  return requires (T t) { t.f(); };
+  return requires (T t) { t.f(); }; // { dg-message "in requirements" }
 }
 
 template<typename T>
 concept bool C2()
 {
-  return requires { typename T::type; };
+  return requires { typename T::type; }; // { dg-message "in requirements" }
 }
 
 template<typename T>
@@ -22,7 +22,7 @@ template<typename T>
 void f2(T x) { }
 
 // Note that these declarations are private and therefore
-// cannot satisify the constraints.
+// cannot satisfy the constraints.
 class S
 {
   using type = int;
@@ -31,12 +31,12 @@ class S
 
 int main()
 {
-  f1(s); // { dg-error "cannot call" }
+  f1(s); // { dg-error "cannot call|private" }
   f2(s); // { dg-error "" }
 
   // When used in non-SFINAE contexts, make sure that we fail
   // the constraint check before emitting the access check
-  // failures. The context is being presented constistently
+  // failures. The context is being presented consistently
   // in both cases.
   static_assert(C1<S>(), ""); // { dg-error "failed" }
   static_assert(C2<S>(), ""); // { dg-error "" }
index 26b829d338dca05e5d433c1324fd5380b2d9c7d2..676468112843ced0d703fbd3bd6d3778c7fff2cf 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
diff --git a/gcc/testsuite/g++.dg/concepts/feature-macro.C b/gcc/testsuite/g++.dg/concepts/feature-macro.C
deleted file mode 100644 (file)
index d3d9b54..0000000
+++ /dev/null
@@ -1,6 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-#ifndef __cpp_concepts
-#error __cpp_concepts not defined
-#endif
index a4ade7c628d1da5e1a88906efbf92e0bf12cbf1c..d1b4c0c59f3147717f576dfb1bada2a488d6bcf2 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index 0d70728abe87412e048dd61a1ce525a434c4d2b3..899988c37a368e0c0606bc08c50df016143a61f7 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index b664ccf3da1f8154c78e2903ff66f0f995f8fa27..245380388fa9e7926cb8b9748210f08680e94e53 100644 (file)
@@ -1,4 +1,3 @@
-// Out-of-line generic member function definitions.
 // { dg-do compile { target c++14 } }
 // { dg-additional-options "-fconcepts" }
 
index 17f14b9a46bfec4fb104af7b082c0d39cfde759b..f23c057ab6b494fe6d3778072f1b788afc4fda62 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index 6993f34a89fc11b422c3943e3060678d36ee17ed..8d0a2e1d2028964d2bccb491c45c9f9ee4017567 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 // Test that constraint satisfaction checks work even when
@@ -40,7 +40,7 @@ template <typename T>
 template <typename T>
   concept bool Concept()
   {
-    return requires( T t ) {
+    return requires( T t ) { // { dg-message "in requirements" }
       requires Float<decltype( project(t) )>();
     };
   }
index 250e0a8713aee9a48ac7cfd30596f92fa0d454a7..debb3238a6ee18cd182e36cd0346e75c7f6bd9c5 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index bc0e126c96c88ae6e6caf1bf28daae9dc5424c27..07b8e3a89ba2e878af18c2e6fda169fea4ccbcd6 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do run { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 #include <cassert>
index 830a1747865ecff5413f33fed2bf6fb817960e9c..bbaac46c9ff91f57bbddaeace4a3f08c27d972bc 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index 018b12f86e111371e3d8b22d1ac5aa3145d4bbae..7714788c3c0512b5bc23cf2c99e76509e3faab21 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 // Check shorthand notation.
index 97155f8eb8651c87c9c197e52f67eb50374d7105..031e87fdf233d243efd00f0f92e2b8626b3d2628 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 // Redefinition errors.
index 0052f1aee73943060dd5914c92716d3fb9e85dbb..869cb9c93910c14630322869031dc2ed72f81be4 100644 (file)
@@ -1,8 +1,6 @@
-// { dg-do link { target c++17 } }
+// { dg-do link { target c++14 } }
 // { dg-options "-fconcepts" }
 
-// FIXME: What is this actually testing?
-
 void f() requires true { }
 
 int main() { }
index a3daf4e1bade0d31b5a2ad60534defba9ffae8dc..ffcce4f02208adecee6c52fd33092e2bf3eb2253 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index 84ed77ce1c1de47336389bfda0d4fbe88669a3a4..2f5e88b945c6962bb4fb537dc5f239cf1b09e91a 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do run { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 #include <cassert>
index 5a9556531e75067a864949ca4b08fb6085eabf74..816072d8d3ad6e1e7dd62e9149f37652c0c8c7f3 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
@@ -10,10 +10,10 @@ template<int N>
 template<template<typename> class X>
   concept bool Template() { return true; }
 
-struct S { };
+void f1(Int) { }      // { dg-error "does not constrain a type" }
+void f2(Template) { } // { dg-error "does not constrain a type" }
 
-void f1(Int) { }      // { dg-error "" }
-void f2(Template) { } // { dg-error "" }
+struct S { };
 
 struct S1 {
   void f1(auto x) { }
index 3b10327a493e03c4f23dbfa04b76888493c53022..257608a57a7fb1db44c09fc9a0f1fe628a0d28e6 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do run { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 #include <cassert>
diff --git a/gcc/testsuite/g++.dg/concepts/iconv1.C b/gcc/testsuite/g++.dg/concepts/iconv1.C
deleted file mode 100644 (file)
index e99254f..0000000
+++ /dev/null
@@ -1,21 +0,0 @@
-// PR c++/67240
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-int foo(int x)
-{
-    return x;
-}
-template <typename T>
-concept bool C1 = requires (T x) {
-    {foo(x)} -> int&;
-};
-
-template <typename T>
-concept bool C2 = requires (T x) {
-    {foo(x)} -> void;
-};
-static_assert( C1<int> );      // { dg-error "assert" }
-static_assert( C2<int> );      // { dg-error "assert" }
index 4b3f56193317b3d3988a5cd3fa0545f8a309cffc..b137791bbb6a68843f82fee88aa569dbbb34ca84 100644 (file)
@@ -1,23 +1,33 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept bool C = __is_class(T);
+
+struct X { };
 
 template<typename T>
-  struct S1 { S1(double) requires C<T>() { } };
+  struct Base {
+    Base(double) requires C<T> { } 
+  };
 
-struct S2 : S1<int> {
-  using S1<int>::S1;
+struct Ok1 : Base<X> {
+  using Base<X>::Base;
+};
+
+struct Err1 : Base<int> {
+  using Base<int>::Base;
 };
 
 template<typename T>
-  struct S3 : S1<T> {
-    using S1<T>::S1;
+  struct Generic : Base<T> {
+    using Base<T>::Base;
   };
 
-struct X { };
 
 int main() {
-  S3<X> s(0.0);
+  Ok1 x1(0.0);
+  Err1 x2(0.0); // { dg-error "no matching" }
+  Generic<X> x3(0.0);
+  Generic<int> x4(0.0); // { dg-error "no matching" }
 }
index 6f046323346a3dcfe6cf232db619b7a8220dddae..abfe96e824019ef898f718d5b7f4746196e8cbfa 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
@@ -11,12 +11,13 @@ template<typename T>
   };
 
 template<typename T>
-  struct S2 : S1<T> {
-    using S1<T>::S1;
+  struct S2 : S1<T> { // { dg-error "no matching function" }
+    using S1<T>::S1; // { dg-error "no matching function" }
   };
 
 struct X { } x;
 
 int main() {
-  S2<X> s = x;
+  S2<X> s1(0); // { dg-error "use of deleted function" }
+  S2<X> s2; // { dg-error "use of deleted function" }
 }
diff --git a/gcc/testsuite/g++.dg/concepts/inherit-ctor4.C b/gcc/testsuite/g++.dg/concepts/inherit-ctor4.C
deleted file mode 100644 (file)
index 43df6e6..0000000
+++ /dev/null
@@ -1,19 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C() { return __is_class(T); }
-
-template<typename T>
-  struct S1 {
-    template<C U> S1(U x) { }
-  };
-
-template<typename T>
-  struct S2 : S1<T> {
-    using S1<T>::S1;
-  };
-
-int main() {
-  S2<int> s(0); // { dg-error "no matching function" }
-}
index 84fa6dbbed4ec35e035a0f127b24b8cdff43bb75..5f9bb7e08f0e34bb865843d5c12aea3d938c1c7c 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index 9c7c1733c6df1b3958d83ae6da0f00971253514d..206777d1b94533d71afb71894b323f9d73cb07d4 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do run { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 #include <cassert>
index 5e93f31327058bf2eb1cd94c9a50a7371f39b3af..f02f1bea2473bd872d68176c63803919edb6a602 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename ... T>
index a7e513535aaffc1721c1b674cab31e790a07617b..0b275e14bf2cfc04674bc12c4e008be34c10b8ef 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename ... T>
@@ -18,11 +18,11 @@ template<int N>
 template<typename T, typename U = int>
   concept bool C5() { return __is_class(U); }
 
-C1{...A, B} void f1() {}; // { dg-error "no matching|wrong number" }
-C1{A} void f2() {} // { dg-error "cannot match pack|no matching concept" }
-C2{A, B} void f3() {}; // { dg-error "cannot match pack|no matching concept" }
-C3{...A} void f4() {}; // { dg-error "cannot match pack|no matching concept" }
-C4{A} void f5() {}; // { dg-error "no matching concept" }
+C1{...A, B} void f1() {}; // { dg-error "cannot deduce template parameters" }
+C1{A} void f2() {}
+C2{A, B} void f3() {};
+C3{...A} void f4() {}; // { dg-error "cannot be introduced" }
+C4{A} void f5() {}; // { dg-error "cannot deduce template parameters" }
 C5{A, B} void f6() {};
 
 int main()
index e7cd7a48066417b7841071139d2d2479aa9f9f9e..bbfef7bed9ee4ddd9d69b861cbd43cebaf560d5c 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T, typename U = int>
@@ -7,10 +7,5 @@ template<typename T, typename U = int>
      return sizeof(U) == sizeof(int);
   }
 
-C{A} void f1() {}
+C{A} void f1() {} // { dg-error "all template parameters" }
 
-int main()
-{
-  f1<char>();
-  return 0;
-}
index 57b325a0e38fb4e3b15873402879cc61ccde7169..233c5bcec7be3d76f4b416f2453a6f1de8c1e01d 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/67003
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 namespace X {
index d5bdc7e2789e09c2f446aa50fb50dd2f8ebaf440..343fe7a9824a1a5c127de5b32769a6a9ebec17b4 100644 (file)
@@ -1,14 +1,14 @@
 // PR c++/66985
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <template <class> class T>
-concept bool _Valid = requires { typename T<int>; };
+concept bool Valid = requires { typename T<int>; };
 
 template <template <class> class T>
 struct __defer { };
 
-_Valid{T}
+Valid{T}
 struct __defer<T> {
   using type = T<int>;
 };
index 33c3b6227c739f3ae41abadd3ddedf4d9148b987..fbad42f69524ceed1b49a1b4716cb494e470e8d3 100644 (file)
@@ -1,5 +1,5 @@
-// { dg-do compile { target c++17 } }
-// { dg-additional-options "-fconcepts" }
+// { dg-do compile { target c++17_only } }
+// { dg-options "-fconcepts" }
 
 struct S
 {
diff --git a/gcc/testsuite/g++.dg/concepts/memtmpl1.C b/gcc/testsuite/g++.dg/concepts/memtmpl1.C
deleted file mode 100644 (file)
index dc00a07..0000000
+++ /dev/null
@@ -1,16 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template <class T>
-struct A {
-  template <class U>
-  requires sizeof(T) == 1
-    static void f(U);
-  template <class U>
-  requires sizeof(T) == 2
-    static void f(U);
-  void g()
-  {
-    f(42);
-  }
-};
index 5f0f3468ea120fa271c55090f90fe1ccd752c85c..6b66b78fb8b743023201bf7c70da9b131c1db9fe 100644 (file)
@@ -1,27 +1,37 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
-  concept bool Type() { return true; }
+  concept bool Type = true;
 
 template<typename T, typename U>
-  concept bool Same() { return __is_same_as(T, U); }
+  concept bool Same = __is_same_as(T, U);
 
 template<typename T, typename U>
-  concept bool C1() { return true; }
+  concept bool C1 = true;
 
 template<typename T, typename... Args>
-  concept bool C2() { return true; }
+  concept bool C2 = true;
+
+template<typename T, typename U>
+  concept bool C3 = __is_same_as(T, int) && __is_same_as(U, double);
 
 template<Same<int> T> struct S1 { };
 template<typename T, Same<T> U> struct S2 { };
 
-void f(Same<int> q) { }
-void g(Type a, Same<decltype(a)> b) { }
+template<Same<int> Q>
+void f(Q q) { }
+template<Type A, Same<decltype(A{})> B>
+void g(A a, B b) { }
 
-void h0(Same<int>* a) { }
-void h1(C1<int>* a) { }
-void h2(C2<char, short, int, long>* a) { }
+template<Same<int> A>
+void h0(A* a) { }
+template<C1<int> A>
+void h1(A* a) { }
+template<C2<char, short, int, long> A>
+void h2(A* a) { }
+template<C3<double> A>
+void h3(A* a) { }
 
 int main() {
   S1<int> s1;
@@ -30,5 +40,6 @@ int main() {
   g(0, 1);
   h0((int*)0);
   h1((int*)0);
-  h2((int*)0);
+  // h2((int*)0);
+  h3((int*)0);
 }
index e51894bb1c02f71de16046d8bbb6cdf005a52153..2c14576f374cd9ae8cd76603bddacc86ce44891a 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 // Make sure that we check partial concept ids
index a5b853cfb3f3325ba39273277a95b130892602bd..bec6715ea65456e635cecbc51d3c03d49273fcbf 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/67138
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T>
diff --git a/gcc/testsuite/g++.dg/concepts/placeholder1.C b/gcc/testsuite/g++.dg/concepts/placeholder1.C
deleted file mode 100644 (file)
index edd0003..0000000
+++ /dev/null
@@ -1,35 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T, typename U>
-struct is_same
-{
-  static constexpr bool value = false;
-};
-
-template<typename T>
-struct is_same<T, T>
-{
-  static constexpr bool value = true;
-};
-
-template<class T, class U>
-concept bool Same = is_same<T, U>::value;
-
-template<typename T>
-concept bool C1 = true;
-
-template<typename T, typename U>
-concept bool C2 = true;
-
-template<typename T>
-concept bool C3() { return true; }
-
-template<typename T, typename U>
-concept bool C4() { return true; }
-
-C1      c1 = 0;
-C2<int> c2 = 0;
-C3      c3 = 0;
-C4<int> c4 = 0;
-Same<int> s1 = 'a'; // { dg-error "does not satisfy|is_same" }
index 3d6fc813a6acc860075939784e96e440924bcf81..0c6f91abcafcf232073e18715a0b6681be0c80c3 100644 (file)
@@ -1,9 +1,6 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
-// Check argument deduction constraints.
-// TODO: We shoul have more of these...
-
 template<typename T>
 concept bool C1 = sizeof(T) == 0;
 
@@ -12,22 +9,16 @@ concept bool C2 = __is_same_as(T, U);
 
 
 template<typename T>
-concept bool D1()
-{
-  return requires (T t) { { t } -> C1; };
-}
+concept bool D1 = requires (T t) { { t } -> C1; };
 
 template<typename T>
-concept bool D2()
-{
-  return requires (T t) { { t } -> C2<void>; };
-}
+concept bool D2 = requires (T t) { { t } -> C2<void>; };
 
-void f1(D1) { }
-void f2(D2) { }
+void f1(auto D1) { } // OK: D1 is declared as a parameter
+void f2(auto D2) { } // OK: D2 is declared as a parameter
 
 int main()
 {
-  f1(0); // { dg-error "cannot call" }
-  f2(0); // { dg-error "cannot call" }
+  f1(0);
+  f2(0);
 }
index 93f0c0d161fa3eeb06ef3ceea497a7622205f5e3..4f8600bd07fc0bc978998392555648fd76a81b79 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/66218
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T, class U>
@@ -7,8 +7,8 @@ concept bool Same = __is_same_as(T, U);
 
 template <class T>
 concept bool C =
-  requires {
-    { 0 } -> Same<T>;
+  requires { // { dg-message "in requirements" }
+    { 0 } -> Same<T>;          // { dg-error "does not satisfy" }
   };
 
 template <C c>
index d1308137c0481077ba8c03a21fc68d1bcc8e715c..2b5afbbc6f2c7afb2f31c48db7f85be1bdfb1259 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/66218
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T, class U>
@@ -7,8 +7,8 @@ concept bool Same = __is_same_as(T, U);
 
 template <class T>
 concept bool C =
-  requires {
-    { 0 } -> Same<T>;
+  requires { // { dg-message "in requirements" }
+    { 0 } -> Same<T>;          // { dg-error "does not satisfy" }
   };
 
 template <class T>
index 245e27a75ed3d6d9ca809249ec57f64666164aa7..7881a40a22a55986ededddd8257a05f425abb599 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T, class U>
@@ -8,11 +8,11 @@ const int i = 0;
 template <class T>
 concept bool C =
   requires {
-    { &i } -> const Same<T>*;
+    { &i } -> const Same<T>*; // { dg-error "not a plain type-constraint" }
   };
 
 template <C c>
 constexpr bool f() { return true; }
 
-static_assert(f<double>(), "");        // { dg-error "" }
-static_assert(f<int>(), "");
+static_assert(f<double>(), "");        // { dg-error "cannot call|as type" }
+static_assert(f<int>(), ""); // { dg-error "cannot call|as type" }
index 51282d93a5d302ec1ab3a10226267a549258f694..20b9c9316489092e91ec05ef38d46a1c49a66cf1 100644 (file)
@@ -1,11 +1,8 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <int I> struct B { static const int i = I; };
 template <int I> concept bool Few = I < 10;
 
-constexpr int g(B<Few> b) { return b.i; }
+constexpr int g(B<Few> b); // { dg-error "does not constrain a type|invalid" }
 
-#define SA(X) static_assert((X),#X)
-SA(g(B<2>{}) == 2);
-SA(g(B<10>{}) == 10);          // { dg-error "" }
index e383653b6a3b3e65afbb643ced05c5363651bcb0..5fcb38a5bf47717ca8b95bae5ece4f4eebb48e28 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index 2aec0e545d2e6aac68088048e49b76767515972b..f927c9abefca7d876e447d3e924085c16f19af56 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 using TD = int;
index 0d8a69d0bd1acb8f64244768af19abc49ad012c6..67153d63dfbcaf53f925ef39efc7f36f36b07ec6 100644 (file)
@@ -1,66 +1,45 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
-concept bool C()
-{
-  return requires (T t) { t.mf(); };
-}
+concept bool C = requires (T t) { t.mf(); };
 
 template<typename T>
-concept bool CA1()
-{
-  return C<typename T::ca1_type>();
-}
+concept bool CA1 = C<typename T::ca1_type>;
 
 template<typename T>
-concept bool CA2()
-{
-  return CA1<T>() && requires () { typename T::ca2_type; };
-}
+concept bool CA2 = CA1<T> && requires () { typename T::ca2_type; };
 
 template<typename T>
-concept bool CA3()
-{
-  return CA2<T>() && requires () { typename T::ca3_type; };
-}
+concept bool CA3 = CA2<T> && requires () { typename T::ca3_type; };
 
 template<typename T>
-concept bool CB1()
-{
-  return requires () { typename T::cb1_type; };
-}
+concept bool CB1 = requires () { typename T::cb1_type; };
 
 template<typename T>
-concept bool CB2()
-{
-  return CB1<T>() && requires () { typename T::cb2_type; };
-}
+concept bool CB2 = CB1<T> && requires () { typename T::cb2_type; };
 
 template<typename T>
-concept bool CB3()
-{
-  return CB2<T>() && requires () { typename T::cb3_type; };
-}
+concept bool CB3 = CB2<T> && requires () { typename T::cb3_type; };
 
 
 struct MC { void mf(); };
-static_assert(C<MC>(), "");
+static_assert(C<MC>, "");
 
 
 struct MA1 { using ca1_type = MC; };
 struct MA2 : MA1 { using ca2_type = int; };
 struct MA3 : MA2 { using ca3_type = int; };
-static_assert(CA1<MA1>(), "");
-static_assert(CA2<MA2>(), "");
-static_assert(CA3<MA3>(), "");
+static_assert(CA1<MA1>, "");
+static_assert(CA2<MA2>, "");
+static_assert(CA3<MA3>, "");
 
 struct MB1 { using cb1_type = int; };
 struct MB2 : MB1 { using cb2_type = int; };
 struct MB3 : MB2 { using cb3_type = int; };
-static_assert(CB1<MB1>(), "");
-static_assert(CB2<MB2>(), "");
-static_assert(CB3<MB3>(), "");
+static_assert(CB1<MB1>, "");
+static_assert(CB2<MB2>, "");
+static_assert(CB3<MB3>, "");
 
 
 template<typename T1, typename T2>
@@ -73,29 +52,19 @@ struct S<T1, T2> // Specialization #1
 };
 
 template<CA1 T1, CB2 T2>
-  requires !CA2<T1>()
+  requires (!CA2<T1>)
 struct S<T1, T2> // Specialization #2
 {
   static constexpr int value = 2;
 };
 
 template<CA2 T1, CB3 T2>
-  requires !CA3<T1>()
+  requires (!CA3<T1>)
 struct S<T1, T2> // Specialization #3
 {
   static constexpr int value = 3;
 };
 
-S<MA1,MB1> s11;
-S<MA1,MB2> s12;
-S<MA1,MB3> s13;
-S<MA2,MB1> s21;
-S<MA2,MB2> s22;
-S<MA2,MB3> s23;
-S<MA3,MB1> s31;
-S<MA3,MB2> s32;
-S<MA3,MB3> s33;
-
 static_assert(S<MA1,MB1>::value == 1, "");
 static_assert(S<MA1,MB2>::value == 2, "");
 static_assert(S<MA1,MB3>::value == 2, "");
index 067801844dccdca91cf9933fc779473592d343b0..ea3077d84fba52d69a2648e985f94a8d27d7b478 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 // Performance test... This should be fast.
index a0eca9ba8c0e72955ff0a50884994d35253289b9..382eba110a6d94b7a867161728302ad88df14e00 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<class T> concept bool C1 = true;
index af4f677644aeb2f248817fa7a895ece67dae619a..600176027b8976982f2276dab01ea899f7a0b987 100644 (file)
@@ -1,4 +1,5 @@
-// { dg-additional-options "-fconcepts" }
+// { dg-do compile { target c++11 } }
+// { dg-options "-fconcepts" }
 
 template <typename T> struct A
 {
index 7b5d712512eeb5d249b9f5a2129de3444f8c6fd5..7199e0517d42bc1717a8ae618f3219f3d1633e97 100644 (file)
@@ -1,10 +1,10 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
-template <class X> concept bool allocatable = requires{{new X}->X * };
+template <class X> concept bool allocatable = requires{{new X}->X *; };
 template <class X> concept bool semiregular = allocatable<X>;
-template <class X> concept bool readable = requires{requires semiregular<X>};
-template <class> int weak_input_iterator = requires{{0}->readable};
+template <class X> concept bool readable = requires{requires semiregular<X>;};
+template <class> int weak_input_iterator = requires{{0}->readable;};
 template <class X> bool input_iterator{weak_input_iterator<X>}; // { dg-warning "narrowing conversion" }
 template <class X> bool forward_iterator{input_iterator<X>};
 template <class X> bool bidirectional_iterator{forward_iterator<X>};
index b1554218aad2b18ca68c31481670a82b5989adc9..ddd069c798771f552d98950d354f251780de98d3 100644 (file)
@@ -1,7 +1,7 @@
 // { dg-do compile { target c++11 } }
 // { dg-additional-options "-fconcepts" }
 
-template<class... Xs>                 
+template<class... Xs>
 void consume(Xs&&...) {}
 
 template<class... Xs>
index fc458989c3430b762ea6f585b7d3d3ba5fb79468..16868ba042c058845286adb459c760502b063344 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class>
index 3ec6514b4671fb4d2904905673d4061ea5a404fa..224eaa3d2f7a03616ed15c002d87183f102ddd66 100644 (file)
@@ -1,6 +1,7 @@
-// { dg-do compile { target c++14 } }
-// { dg-additional-options "-fconcepts" }
+// { dg-do compile { target c++17_only } }
+// { dg-options "-fconcepts" }
 
 template<template<typename> class T>
 concept bool C = T<int>::value;
-C c = 1;  // { dg-error "invalid reference to concept" }
+
+C c = 1;  // { dg-error "does not constrain a type" }
index 8b4eb41f7adecb4150254ebd2d3b975c594985a3..a150e37d78d8e92091bee6e1351e1de2910effa0 100644 (file)
@@ -1,5 +1,5 @@
-// { dg-do compile { target c++14 } }
-// { dg-additional-options "-fconcepts" }
+// { dg-do compile { target c++17_only } }
+// { dg-options "-fconcepts" }
 
 template<typename T>
 concept bool C() { return true; }
index 02f1999672d9ddea01e66e1931c128c49f4a3d75..675d66d9d7567d8fc6f442f033a46d13cbfecc3a 100644 (file)
@@ -1,6 +1,7 @@
-// { dg-do compile { target c++14 } }
-// { dg-additional-options "-fconcepts" }
+// { dg-do compile { target c++17_only } }
+// { dg-options "-fconcepts" }
 
 template<template<typename> class T>
 concept bool C = true;
-C c = 1;  // { dg-error "invalid reference to concept" }
+
+C c = 1;  // { dg-error "does not constrain a type" }
index 42d21f57161578a8e33e51a7f2edbd4eb1627355..fb754253612eddf13d2cd5f9b9a830de27cda3d0 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<class T>
index 975ee8941db8e56f26a3c7726dbb5d43ce14ea25..0c2f45602d194e2963ed270ac784f535d09bc1da 100644 (file)
@@ -1,12 +1,15 @@
 // PR c++/84330
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 struct A
 {
-  template<typename T> requires sizeof(T) >> 0 void foo(T);  // { dg-error "predicate constraint" }
+  template<typename T>
+    requires (sizeof(T) >> 0)
+  void foo(T);
 
   void bar()
   {
-    foo(0);  // { dg-error "no matching" }
+    foo(0);  // { dg-error "no matching function" }
   }
 };
index 861a838dd29b81fcf4422e86c97552157138137c..52a42647948ac1f4dba4f293d711280a0ae4ce86 100644 (file)
@@ -1,6 +1,6 @@
-// { dg-do compile { target c++14 } }
-// { dg-additional-options "-fconcepts" }
+// { dg-do compile { target c++17_only } }
+// { dg-options "-fconcepts" }
 
 template<int> concept bool C = true;
 
-C c = 0;  // { dg-error "invalid reference to concept" }
+C c = 0;  // { dg-error "does not constrain a type" }
diff --git a/gcc/testsuite/g++.dg/concepts/req-neg1.C b/gcc/testsuite/g++.dg/concepts/req-neg1.C
deleted file mode 100644 (file)
index 637f993..0000000
+++ /dev/null
@@ -1,11 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-void f1(int a) requires true;         // OK
-auto f2(int a) -> bool requires true; // OK
-auto f3(int a) requires true -> bool; // { dg-error "" } requires-clause precedes trailing-return-type
-typedef void fn_t() requires true;    // { dg-error "typedef" }
-void (*pf)() requires true;           // { dg-error "non-function" }
-void (*fn(int))() requires false;     // { dg-error "return type" }
-void g(int (*)() requires true);      // { dg-error "parameter|non-function" }
-auto* p = new (void(*)(char) requires true); // { dg-error "type-id" }
diff --git a/gcc/testsuite/g++.dg/concepts/req1.C b/gcc/testsuite/g++.dg/concepts/req1.C
deleted file mode 100644 (file)
index fedea73..0000000
+++ /dev/null
@@ -1,29 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool Class () { return __is_class(T); }
-
-// Allow a requires-expression with no parms.
-template<typename T>
-  concept bool C = requires { typename T::type; };
-
-void f1(auto a) requires Class<decltype(a)>() { }
-void f2(auto a) requires requires (decltype(a) x) { -x; } { }
-
-struct S { } s;
-
-// Allow non-type template parms as constraints.
-template<bool B> requires B struct S0; // OK
-
-template<int N> requires N struct S1 { };      // { dg-error "does not have type" }
-template<int N> requires N == 0 struct S2 { }; // OK
-
-template<typename T, T X> requires X struct S3 { }; // OK
-S3<int, 0> s3a;      // { dg-error "constraint failure|does not have type" }
-S3<bool, false> s3b; // { dg-error "constraint failure" }
-
-int main() {
-  f1(s);
-  f2(0);
-}
diff --git a/gcc/testsuite/g++.dg/concepts/req10.C b/gcc/testsuite/g++.dg/concepts/req10.C
deleted file mode 100644 (file)
index 949859c..0000000
+++ /dev/null
@@ -1,19 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-// Test that standard conversions are checked with
-// implicit conversion constraints.
-
-template<typename T, typename U>
-concept bool C()
-{
-  return requires(T& t) { {t} -> U&; };
-}
-
-struct B { };
-class D : B { };
-
-int main()
-{
-  static_assert(C<D, B>(), ""); // { dg-error "failed" }
-}
diff --git a/gcc/testsuite/g++.dg/concepts/req11.C b/gcc/testsuite/g++.dg/concepts/req11.C
deleted file mode 100644 (file)
index 8891cce..0000000
+++ /dev/null
@@ -1,29 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-// Check that we can evaluate constant requires-expressions
-// as constant expressions, for the curious case when they
-// appear within predicate constraints.
-
-template<typename... Ts> struct variant { };
-
-template<typename T>
-concept bool Streamable()
-{
-  return requires (T t) { t; };
-}
-
-template<typename T>
-concept bool Range()
-{
-  return requires (T t) { t; };
-}
-
-template<class T>
-  requires Streamable<T>() and not Range<T>()
-void print(const T& x) { }
-
-int main()
-{
-  print("hello"); // { dg-error "cannot call" }
-}
diff --git a/gcc/testsuite/g++.dg/concepts/req12.C b/gcc/testsuite/g++.dg/concepts/req12.C
deleted file mode 100644 (file)
index c6b345a..0000000
+++ /dev/null
@@ -1,26 +0,0 @@
-// PR c++/66218
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-#include <type_traits>
-
-template <class T, class U>
-concept bool Same =
-  std::is_same<T, U>::value;
-
-template <class T>
-concept bool C =
-  requires(T t) {
-    { t } -> Same<T>;
-  };
-
-template <class>
-constexpr bool f() { return false; }
-template <C>
-constexpr bool f() { return true; }
-
-static_assert(f<char>(), "");
-static_assert(f<int>(), "");
-static_assert(f<double>(), "");
-
-int main() {}
diff --git a/gcc/testsuite/g++.dg/concepts/req13.C b/gcc/testsuite/g++.dg/concepts/req13.C
deleted file mode 100644 (file)
index 4fd2312..0000000
+++ /dev/null
@@ -1,14 +0,0 @@
-// PR c++/66758
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template <class T, class...Args>
-concept bool Constructible =
-  requires(Args&&...args) {
-    T{ ((Args&&)(args))... };
-    new T{((Args&&)(args))...};
-  };
-
-template <Constructible T> struct A { };
-A<int> a;
-
diff --git a/gcc/testsuite/g++.dg/concepts/req16.C b/gcc/testsuite/g++.dg/concepts/req16.C
deleted file mode 100644 (file)
index ca04d60..0000000
+++ /dev/null
@@ -1,20 +0,0 @@
-// PR c++/66988
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-#include <type_traits>
-
-template <template <class> class T, class U>
-concept bool _Valid = requires { typename T<U>; };
-
-template <class T>
-using __t = typename T::type;
-
-template <class T>
-struct __has_type : std::false_type { };
-
-template <class T>
-  requires _Valid<__t, T>
-struct __has_type<T> : std::true_type { };
-
-static_assert(!__has_type<int>(), "");
diff --git a/gcc/testsuite/g++.dg/concepts/req18.C b/gcc/testsuite/g++.dg/concepts/req18.C
deleted file mode 100644 (file)
index cccfaed..0000000
+++ /dev/null
@@ -1,18 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template <class> struct all_same {
-  static constexpr bool value = 1;
-};
-template <class T> concept bool Assignable
-= requires(T t)
-{
-  requires all_same<decltype(t = 0)>::value;
-};
-
-template <class I> requires !Assignable<I>
-int dispatch();
-template <Assignable>
-void dispatch();
-
-int main() { dispatch<int *>(); }
diff --git a/gcc/testsuite/g++.dg/concepts/req19.C b/gcc/testsuite/g++.dg/concepts/req19.C
deleted file mode 100644 (file)
index 97cd9e5..0000000
+++ /dev/null
@@ -1,14 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-struct B
-{
-  template <class T> void f(T t)
-    requires requires (T tt) { tt; }
-  { }
-};
-
-int main()
-{
-  B().f(42);
-}
diff --git a/gcc/testsuite/g++.dg/concepts/req2.C b/gcc/testsuite/g++.dg/concepts/req2.C
deleted file mode 100644 (file)
index b32845a..0000000
+++ /dev/null
@@ -1,20 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool Class () { return __is_class(T); }
-
-void f1(auto a) requires Class<decltype(a)>() { }
-
-// FIXME: This is generating excess errors related to pretty
-// printing the trailing requires expression.
-void f2(auto a)
-  requires requires (decltype(a) x) { -x; }
-{ }
-
-struct S { } s;
-
-int main() {
-  f1(0); // { dg-error "cannot call" }
-  f2((void*)0); // { dg-error "cannot call" }
-}
diff --git a/gcc/testsuite/g++.dg/concepts/req20.C b/gcc/testsuite/g++.dg/concepts/req20.C
deleted file mode 100644 (file)
index bd6b0f9..0000000
+++ /dev/null
@@ -1,21 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template <class T> concept bool C = true;
-
-template <class T>
-requires C<typename T::foo>
-void f(T t) { }
-
-void f(...);
-
-template <class T>
-requires C<T>
-void g(T t) { }
-
-int main()
-{
-  f(42);
-  g(42);
-}
-
diff --git a/gcc/testsuite/g++.dg/concepts/req3.C b/gcc/testsuite/g++.dg/concepts/req3.C
deleted file mode 100644 (file)
index 8ce58e5..0000000
+++ /dev/null
@@ -1,17 +0,0 @@
-// { dg-do run { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool Class () { return __is_class(T); }
-
-struct Test {
-  void f(auto a) requires Class<decltype(a)>();
-} test;
-
-struct S { }s;
-
-int main() {
-  test.f(s);
-}
-
-void Test::f(auto a) requires Class<decltype(a)>() { }
diff --git a/gcc/testsuite/g++.dg/concepts/req4.C b/gcc/testsuite/g++.dg/concepts/req4.C
deleted file mode 100644 (file)
index fcc13c6..0000000
+++ /dev/null
@@ -1,19 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-struct fool {
-  constexpr fool operator&&(fool) const { return {}; }
-  constexpr fool operator||(fool) const { return {}; }
-};
-
-template<typename T> constexpr fool p1() { return {}; }
-template<typename T> constexpr fool p2() { return {}; }
-
-template<typename T>
-  concept bool C() { return p1<T>() && p2<T>(); }
-
-template<C T> void f(T x) { }
-
-int main() {
-  f(0); // { dg-error "cannot call|uses overloaded operator" }
-}
diff --git a/gcc/testsuite/g++.dg/concepts/req5.C b/gcc/testsuite/g++.dg/concepts/req5.C
deleted file mode 100644 (file)
index 7ad1cab..0000000
+++ /dev/null
@@ -1,19 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-struct fool { };
-
-constexpr fool operator&&(fool, fool) { return {}; }
-constexpr fool operator||(fool, fool) { return {}; }
-
-template<typename T> constexpr fool p1() { return {}; }
-template<typename T> constexpr fool p2() { return {}; }
-
-template<typename T>
-  concept bool C() { return p1<T>() && p2<T>(); }
-
-template<C T> void f(T x) { }
-
-int main() {
-  f(0); // { dg-error "cannot call|uses overloaded operator" }
-}
diff --git a/gcc/testsuite/g++.dg/concepts/req6.C b/gcc/testsuite/g++.dg/concepts/req6.C
deleted file mode 100644 (file)
index dd7dbdd..0000000
+++ /dev/null
@@ -1,19 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-struct X { };
-int operator==(X, X) { return 0; }
-
-template<typename T>
-  concept bool C1() { return X(); } // { dg-error "bool" }
-
-template<C1 T>
-  void h(T) { } // OK until used.
-
-void f()
-{
-  h(0); // { dg-error "does not have|cannot call" }
-}
-
-template<typename T>
-  concept bool C2() { return X() == X(); } // OK
diff --git a/gcc/testsuite/g++.dg/concepts/req7.C b/gcc/testsuite/g++.dg/concepts/req7.C
deleted file mode 100644 (file)
index a6cfb4b..0000000
+++ /dev/null
@@ -1,25 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-#include <vector>
-
-using namespace std;
-
-template<typename T>
-  struct Sequence : std::false_type { };
-
-template<typename T>
-  struct Predicate : std::false_type { };
-
-template<typename Seq, typename Fn>
-  requires Sequence<Seq>{} and Predicate<Fn>{}
-    bool all(const Seq& seq, Fn fn) {
-      for(const auto& x : seq)
-        if (not fn(x))
-          return false;
-      return true;
-    }
-
-int main() {
-  all(vector<int>{0, 2}, true); // { dg-error "not|bool" }
-}
diff --git a/gcc/testsuite/g++.dg/concepts/req8.C b/gcc/testsuite/g++.dg/concepts/req8.C
deleted file mode 100644 (file)
index 201be37..0000000
+++ /dev/null
@@ -1,17 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-// Check that type requirements are normalized correctly.
-
-template<typename T>
-  concept bool Has_member_type() {
-    return requires() { typename T::type; };
-  }
-
-template<typename T>
-  concept bool Concept() {
-    return true && Has_member_type<T>();
-  }
-
-template<Concept T>
-  void foo( T t  ) { }
diff --git a/gcc/testsuite/g++.dg/concepts/req9.C b/gcc/testsuite/g++.dg/concepts/req9.C
deleted file mode 100644 (file)
index 497154c..0000000
+++ /dev/null
@@ -1,25 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-struct S1 {};
-
-template<typename T>
-concept bool C() { return requires(T x) { { x.fn() } -> S1<T>; }; }
-
-template<C U>
-void fn(U x)
-{
-  x.fn();
-}
-
-struct S2
-{
-  auto fn() const { return S1<S2>(); }
-};
-
-int main()
-{
-  fn(S2{});
-  return 0;
-}
diff --git a/gcc/testsuite/g++.dg/concepts/template-parm1.C b/gcc/testsuite/g++.dg/concepts/template-parm1.C
deleted file mode 100644 (file)
index 192226f..0000000
+++ /dev/null
@@ -1,35 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C1 = __is_same_as(T, int);
-
-template<int N>
-  concept bool C2 = N == 0;
-
-template<template<typename> class X>
-  concept bool C3 = true;
-
-template<typename> struct Foo;
-
-// Type template parameters
-template<C1 T = int> struct S1 { };
-template<C1 = int> struct S2;
-template<C1 T> struct S2 { };
-
-// Non-type template parameters
-template<C2 N = 0> struct S3 { };
-template<C2 = 0> struct S4;
-template<C2 N> struct S4 { };
-
-// Template template parameters
-template<C3 X = Foo> struct S5 { };
-template<C3 = Foo> struct S6;
-template<C3 X> struct S6 { };
-
-S1<> s1;
-S2<> s2;
-S3<> s3;
-S4<> s4;
-S5<> s5;
-S6<> s6;
diff --git a/gcc/testsuite/g++.dg/concepts/template-parm10.C b/gcc/testsuite/g++.dg/concepts/template-parm10.C
deleted file mode 100644 (file)
index 33bf372..0000000
+++ /dev/null
@@ -1,18 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<int N, class T>
-  concept bool P() { return true; }
-
-template<template<typename> class X, class T>
-  concept bool Q() { return true; }
-
-template<P<int> N> void f() { }
-template<Q<int> X> void g() { }
-
-template<typename> struct S { };
-
-int main() {
-  f<0>();
-  g<S>();
-}
index 04e11e2e69788f1c5d2235816ebb13914dcdd6fe..07ad6e284e9b429486fa99e88998b4259a00e993 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index 8745bb1d9216d8e454f43c592d10a02aae98abdc..cb3e2c6b55af66024a0fbf36dba1fbdda5ee4697 100644 (file)
@@ -1,6 +1,6 @@
-// Conceptized version of template/ttp23.C
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
+// Conceptized version of template/ttp23.C
 
 template <class T> concept bool Foo = true;
 
index adecc12f0f9ec17e8e8ea19fd1db8214e4214e7e..d708fd06f018fd1b947164776cf79f827d87de07 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index 3d37e9c4c250e3da0138c6bc7536e706014671c0..028149c13dce3bd3b6ca05d36f0559aac06ebe97 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index f546a758c06d696b92c6bbef8ec5e8a2097bc0d9..d93dbc7f2dc534a45b501292bfa0c42c1c92decc 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
diff --git a/gcc/testsuite/g++.dg/concepts/template-parm5.C b/gcc/testsuite/g++.dg/concepts/template-parm5.C
deleted file mode 100644 (file)
index cd93c60..0000000
+++ /dev/null
@@ -1,20 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C1 = __is_same_as(T, int);
-
-template<int N>
-  concept bool C2 = N == 0;
-
-template<template<typename> class X>
-  concept bool C3 = true;
-
-template<typename> struct Foo;
-
-template<C1... Ts = int> struct S1; // { dg-error "default argument" }
-template<C1... = int> struct S2; // { dg-error "default argument" }
-template<C2... Ns = 0> struct S3; // { dg-error "default argument" }
-template<C2... = 0> struct S4; // { dg-error "default argument" }
-template<C3... Ts = Foo> struct S5; // { dg-error "default argument" }
-template<C3... = Foo> struct S6; // { dg-error "default argument" }
diff --git a/gcc/testsuite/g++.dg/concepts/template-parm6.C b/gcc/testsuite/g++.dg/concepts/template-parm6.C
deleted file mode 100644 (file)
index 9efe409..0000000
+++ /dev/null
@@ -1,42 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename... Ts> struct are_same;
-
-template<>
-  struct are_same<> {
-    static constexpr bool value = true;
-  };
-
-template<typename T>
-  struct are_same<T> {
-    static constexpr bool value = true;
-  };
-
-template<typename T, typename U, typename... Ts>
-  struct are_same<T, U, Ts...> {
-    static constexpr bool value =
-      __is_same_as(T, U) && are_same<U, Ts...>::value;
-  };
-
-constexpr bool all_of() { return true; }
-constexpr bool all_of(bool b) { return b; }
-
-template<typename... Ts>
-  constexpr bool all_of(bool a, bool b, Ts... args) {
-    return (a && b) && all_of(b, args...);
-  }
-
-template<typename... Ts>
-  concept bool C1 = are_same<Ts...>::value;
-
-template<bool... Bs>
-  concept bool C2 = all_of(Bs...);
-
-template<C1... Ts> struct S1 { };
-template<C1...> struct S2 { };
-template<C2... Bs> struct S4 { };
-template<C2...> struct S5 { };
-
-S1<int, int, int> s1;
-S4<true, true, true> s4;
diff --git a/gcc/testsuite/g++.dg/concepts/template-parm7.C b/gcc/testsuite/g++.dg/concepts/template-parm7.C
deleted file mode 100644 (file)
index 1dfa0d1..0000000
+++ /dev/null
@@ -1,45 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename... Ts> struct are_same;
-
-template<>
-  struct are_same<> {
-    static constexpr bool value = true;
-  };
-
-template<typename T>
-  struct are_same<T> {
-    static constexpr bool value = true;
-  };
-
-template<typename T, typename U, typename... Ts>
-  struct are_same<T, U, Ts...> {
-    static constexpr bool value =
-      __is_same_as(T, U) && are_same<U, Ts...>::value;
-  };
-
-constexpr bool all_of() { return true; }
-constexpr bool all_of(bool b) { return b; }
-
-template<typename... Ts>
-  constexpr bool all_of(bool a, bool b, Ts... args) {
-    return (a && b) && all_of(b, args...);
-  }
-
-
-template<typename... Ts>
-  concept bool C1 = are_same<Ts...>::value;
-
-template<bool... Bs>
-  concept bool C2 = all_of(Bs...);
-
-template<C1... Ts> struct S1 { }; // OK
-S1<int, int, char> s1; // { dg-error "constraint failure|invalid type" }
-template<C1 Ts> struct S2 { }; // { dg-error "variadic constraint"  }
-
-template<C2... Bs> struct S3 { }; // OK
-S3<true, true, false> s3; // { dg-error "constraint failure|invalid type" }
-template<C2 Bs> struct S4 { }; // { dg-error "variadic constraint" }
-
-int main() { }
diff --git a/gcc/testsuite/g++.dg/concepts/template-parm9.C b/gcc/testsuite/g++.dg/concepts/template-parm9.C
deleted file mode 100644 (file)
index 64308cd..0000000
+++ /dev/null
@@ -1,19 +0,0 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
-template<typename T>
-  concept bool C() { return __is_class(T); }
-
-template<typename T>
-  concept bool D() { return C<T>() and __is_empty(T); }
-
-template<template<typename Q> requires C<Q>() class X>
-  struct S { };
-
-template<typename A> requires true struct T0 { };
-template<typename A> requires D<A>() struct T1 { };
-
-S<T0> x3; // { dg-error "constraint mismatch|invalid type" }
-S<T1> x4; // { dg-error "constraint mismatch|invalid type" }
-
-int main() { }
index 6c4dc2cec3149449b47c02908a2458664a04ac12..d701859d1273bc53713389a8c37ff32ad42d2b9a 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/66937
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 #include <tuple>
index 90a88d83cc9947c9b865000256917bbdd0d95d57..21a4915551b8b3d060527fff408f37250f874874 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index c16d3e4a57e0e2c00073fc1f6816b71bb79f6ab7..4ff00a008b9aebd6a9c3a25ac41c3a4043de776b 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
index f3d642b1a5352cbe15d90e9167817dcfbeac098f..144c0ea1a24c07a96221b25b84e2d5676fb5c0bb 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T>
@@ -12,12 +12,19 @@ template<typename T>
 
 
 template<typename U>
-  requires C1<U>() // { dg-error "" }
+  requires C1<U>() // { dg-error "cannot be used as a function" }
   void f1(U) { }
 
 template<typename U>
-  requires C2<U> // { dg-error "invalid reference" }
+  requires C2<U> // { dg-error "must be called" }
   void f2(U) { }
 
 template<C3 T>  // { dg-error "not a type" }
-  void f(T) { } // { dg-error "" }
+  void f(T) { } // { dg-error "declared void|not declared" }
+
+void foo()
+{
+  struct S { } s;
+  f2(s);
+  // f2(0);
+}
index 3864c9db99c84442745868c57f0d463e5c1940aa..a7839ee5f85cc2435cab1df743fce0289ff5056e 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T, typename U>
index b1e9cb5379c864d4a67405f305a102fbfea97d39..d8fa2984f15c2043eaec958566d065bdaf2f2ffc 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename T1, typename T2>
@@ -8,8 +8,8 @@ template<typename T1, typename T2, typename T3>
 concept bool C2 = true;
 
 
-template<C1 T> // { dg-error "not a type" }
+template<C1 T> // { dg-error "wrong number of template arguments" }
 constexpr bool f1( )  { return true; }
 
-template<C2<int> T> // { dg-error "expected|not a type" }
+template<C2<int> T> // { dg-error "wrong number of template arguments" }
 constexpr bool f2( )  { return true; }
index 8f5ac62953821484667e315a50872842e68fee4a..80984a7ab1ec988e2019ddb5de3dd92d937c5e8e 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T>
index 8371b373f6472352a1b07c9c0d0f3e58f0291c05..2cfe2666e16b400fde2c4329febbd07ac0626931 100644 (file)
@@ -1,6 +1,6 @@
 // PR c++/85133
-// { dg-do compile { target c++17 } }
-// { dg-additional-options "-fconcepts" }
+// { dg-do compile { target c++17_only } }
+// { dg-options "-fconcepts" }
 
 template<typename> concept bool C; // { dg-error "no initializer" }
 
index b69d7d8d6a2aa61861713d42b8e2cd195930feb7..4ac578f88bc2499a7adb7041e543681a8692877e 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/67117
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T>
index ffe5f1ffe05670ec5d5fb8422532c8ba4fd4d68a..1b8890a789b331399d53de3cef8a4c35c7b77ad1 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/67139
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T>
index 22f07eef827aa8d885451bcb899811fc96ccfd00..cc5ee5fa8e65040848584f5eb9e833080780ed5b 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/68666
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 struct A {
index 4c0f437c32c9f94b98fa8a11fa243039f987b81a..c3bc7f61513066ea6e93687bbe5703977213853b 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/66712
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T, class...Args>
index 4675d97ca1885c0164f433ec804b323b49e88d36..7b220097f9806fff52fdb1b389cf9f6c2161de5d 100644 (file)
@@ -1,17 +1,18 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T> concept bool Copyable = requires (T t) { T(t); };
 template <class T> concept bool Constructable = requires { T(); };
 template <class T> concept bool Both = Copyable<T> && Constructable<T>;
 
-template <Copyable... Ts>
+template <Copyable... Ts> // requires (Copyable<Ts> && ...)
 constexpr int f(Ts...) { return 0; } // #1
 
-template <Both... Ts>
+template <Both... Ts> // requires (Both<Ts> && ...)
 constexpr int f(Ts...) { return 1; }     // #2
 
 int main()
 {
-  static_assert(f(42) == 1);
+  static_assert(f(42) == 1); // { dg-error "ambiguous" }
+  // The associated constraints of the two functions are incomparable.
 }
index f980e991775b0eecc53478a2283b2d4ec72b93d1..bd2f381a1a88ad18b86666092ab8332702c4c2c2 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/70036
-// { dg-do compile { target c++14 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template <class T> concept bool C = true;
index e0f7903cac557e4f63f3ca6ef98f4ff8408b0e3d..d6eea49b95894e93a75c3b2b2d37e323502f8680 100644 (file)
@@ -1,5 +1,5 @@
 // PR c++/73456
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 template<typename...> struct list {};
@@ -7,13 +7,14 @@ template<typename...> struct list {};
 template<typename Seq>
 concept bool Sequence = true;
 
-template<Sequence... Seqs>
+template<Sequence... Seqs> // requires (Sequence<Seqs> && ...)
 struct zip;
 
 template<Sequence... Seqs>
-    requires requires { typename list<Seqs...>; }
-// main.cpp:12:8: internal compiler error: in non_atomic_constraint_p, at cp/logic.cc:315
-struct zip<Seqs...> {};
+    requires requires { typename list<Seqs...>; } // && (Sequence<Seqs> && ...)
+struct zip<Seqs...> {}; // { dg-error "does not specialize" }
+// The constraints of the specialization and the sequence are not
+// comparable; the specializations are unordered.
 
 int main()
 {
index d33f927a63ecf3e4f0c2df997e1cc33bf5107952..c277d33d00225cf4f565778f977af666f8a1a736 100644 (file)
@@ -3,4 +3,4 @@
 
 using T = auto() -> int;
 using U = void() -> int; // { dg-error "11:.type name. function with trailing return type not declared with .auto." }
-using W = auto(); // { dg-error "11:invalid use of .auto." }
+using W = auto(); // { dg-error "11:.*auto." }
index aaa80f47000602ae89a4ea23f2ce1e46f1e02287..3399ce7c866510ebaef7e745239e7d386c7abb02 100644 (file)
@@ -6,7 +6,7 @@ int main()
   auto a = []() { return true; };
   auto b = []() { return a(); };  // { dg-error "'a' is not captured" }
   int c, d;
-  while (b() && c < d) // { dg-error "could not convert" }
+  while (b() && c < d)
     {
     }
 }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-access1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-access1.C
new file mode 100644 (file)
index 0000000..bbb1be0
--- /dev/null
@@ -0,0 +1,15 @@
+// { dg-do compile { target concepts } }
+
+class A
+{
+  static void f(int);
+public:
+  template <class T> void g(T t)
+    requires requires { f(t); }
+  {}
+};
+
+int main()
+{
+  A().g(42);
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-alias.C b/gcc/testsuite/g++.dg/cpp2a/concepts-alias.C
new file mode 100644 (file)
index 0000000..6b2ab0d
--- /dev/null
@@ -0,0 +1,25 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+concept Class = __is_class(T);
+
+template<typename T>
+  requires Class<T>
+using X = T*;
+
+// BUG: Alias templates are expanded at the point of use, regardless
+// of whether or not they are dependent. This causes T* to be substituted
+// without acutally checking the constraints. See the declaration of y1
+// below.
+template<typename T>
+using Y = X<T>;
+
+template<Class T> using Z = T*;
+
+struct S { };
+
+X<S> x1; // OK
+X<int> x2; // { dg-error "template constraint failure" }
+Y<int> y1; // { dg-error "" "" { xfail *-*-* } }
+Z<S> z1; // ok
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-alias2.C b/gcc/testsuite/g++.dg/cpp2a/concepts-alias2.C
new file mode 100644 (file)
index 0000000..b784030
--- /dev/null
@@ -0,0 +1,12 @@
+// { dg-do compile { target concepts } }
+
+using Bool = bool;
+template <class T>
+const Bool b = true;
+
+template <class T>
+concept BoolC = b<T>;
+
+template <BoolC T> struct A { };
+
+A<int> a;
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-class.C b/gcc/testsuite/g++.dg/cpp2a/concepts-class.C
new file mode 100644 (file)
index 0000000..aca5c44
--- /dev/null
@@ -0,0 +1,115 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+concept Class = __is_class(T);
+
+template<typename T>
+concept Union = __is_union(T);
+
+template<typename T>
+concept One = sizeof(T) >= 4;
+
+template<typename T>
+concept Two = One<T> && sizeof(T) >= 8;
+
+// Basic checks
+template<typename T> requires true struct ok { };
+template<typename T> requires false struct err { };
+
+ok<int> ok1;
+err<int> err1; // { dg-error "template constraint failure" }
+err<int>* err2; // { dg-error "template constraint failure" }
+
+// Redeclarations
+template<typename T>
+  requires Class<T>
+struct S1;
+
+template<Class T> // { dg-error "template parameter | different constraints" }
+struct S1 { };
+
+template<typename T>
+  requires Class<T>
+struct S2;
+
+template<typename T>
+  requires Union<T>
+struct S2; // { dg-error "redeclaration | different constraints" }
+
+
+// Check non-overlapping specializations
+template<typename T>
+struct S3 { static const int value = 0; };
+
+template<typename T>
+  requires Class<T>
+struct S3<T> { static const int value = 1; };
+
+template<typename T>
+  requires Union<T>
+struct S3<T> { static const int value = 2; };
+
+struct S { };
+union U { };
+
+static_assert(S3<int>::value == 0, "");
+static_assert(S3<S>::value == 1, "");
+static_assert(S3<U>::value == 2, "");
+
+// Check ordering of partial specializations
+template<typename T>
+struct S4 { static const int value = 0;  };
+
+template<typename T>
+  requires One<T>
+struct S4<T> { static const int value = 1; };
+
+template<typename T>
+  requires Two<T>
+struct S4<T> { static const int value = 2; };
+
+struct one_type { char x[4]; };
+struct two_type { char x[8]; };
+
+static_assert(S4<char>::value == 0, "");
+static_assert(S4<one_type>::value == 1, "");
+static_assert(S4<two_type>::value == 2, "");
+
+// Specializations are more specialized.
+template<typename T> requires Two<T> struct S5 { };
+template<typename T> requires One<T> struct S5<T> { }; // { dg-error "does not specialize" }
+
+// Constraints are checked even when decls are not instantiatied.
+S5<one_type>* x4b; // { dg-error "constraint|invalid" }
+
+// Deduction guides
+template <class T>
+concept IsInt = __is_same_as(T,int);
+
+template<typename T>
+struct A
+{
+  int i;
+  A(...);
+};
+
+template<typename I>
+  requires IsInt<I>
+A(I) -> A<I>;
+
+A a(1);
+A a2(1.0);      // { dg-error "class template argument deduction | no matching function for call" }
+
+
+template<typename T>
+struct S6
+{
+  template<typename U>
+    requires true
+  struct Inner;
+};
+
+template<typename T>
+template<typename U>
+struct S6<T>::Inner { }; // { dg-error "does not match" }
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-cmath.C b/gcc/testsuite/g++.dg/cpp2a/concepts-cmath.C
new file mode 100644 (file)
index 0000000..b090c3a
--- /dev/null
@@ -0,0 +1,4 @@
+// { dg-do compile { target c++2a } }
+
+#include <cmath>
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-complete1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-complete1.C
new file mode 100644 (file)
index 0000000..25c4ca0
--- /dev/null
@@ -0,0 +1,11 @@
+// { dg-do compile { target c++2a } }
+
+template <class T> concept has_mem_type = requires { typename T::type; };
+
+template <has_mem_type T> int f () { return 0; }
+template <class T> char f() { return 0; }
+
+struct A;
+static_assert (sizeof (f<A>()) == 1);
+struct A { typedef int type; };
+static_assert (sizeof (f<A>()) > 1);
similarity index 82%
rename from gcc/testsuite/g++.dg/concepts/constrained-parm.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-constrained-parm.C
index c2b6614aeea3955997afef377ed2e8f9e91c9bde..bb7e31d8b6c51248e69725ddb885e9242efe7713 100644 (file)
@@ -1,8 +1,7 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept C = __is_class(T);
 
 template<const C T> struct S1 { };    // { dg-error "cv-qualified" }
 template<volatile C T> struct S2 { }; // { dg-error "cv-qualified" }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-decltype.C b/gcc/testsuite/g++.dg/cpp2a/concepts-decltype.C
new file mode 100644 (file)
index 0000000..db3cfdf
--- /dev/null
@@ -0,0 +1,67 @@
+// { dg-do compile { target c++2a } }
+
+// Tests constrained decltype(auto).
+
+template<typename T>
+concept Type = true;
+
+template<typename T>
+concept Int = __is_same_as(T, int);
+
+template<typename T, typename U>
+concept SameAs = __is_same_as(T, U);
+
+template<typename T, typename U>
+  requires SameAs<T, U>
+constexpr bool check = true;
+
+int z = 0;
+const int cz = 0;
+
+Type decltype(auto) x1 = 0;
+static_assert(check<decltype(x1), int>);
+Type decltype(auto) x2 = z;
+static_assert(check<decltype(x2), int>);
+Type decltype(auto) x3 = (z);
+static_assert(check<decltype(x3), int&>);
+Type decltype(auto) x4 = cz;
+static_assert(check<decltype(x4), const int>);
+Type decltype(auto) x5 = (cz);
+static_assert(check<decltype(x5), const int&>);
+
+Type decltype(auto) f1() { return 0; }
+static_assert(check<decltype(f1()), int>);
+Type decltype(auto) f2() { return z; }
+static_assert(check<decltype(f2()), int>);
+Type decltype(auto) f3() { return (z); }
+static_assert(check<decltype(f3()), int&>);
+Type decltype(auto) f4() { return cz; }
+static_assert(check<decltype(f4()), int>); // Top-level const is removed?
+Type decltype(auto) f5() { return (cz); }
+static_assert(check<decltype(f5()), const int&>);
+
+bool b = true;
+const bool cb = true;
+
+Int decltype(auto) b1 = true; // { dg-error "deduced initializer" }
+Int decltype(auto) b2 = (b);  // { dg-error "deduced initializer" }
+Int decltype(auto) b3 = (cb); // { dg-error "deduced initializer" }
+
+Int decltype(auto) g1() { } // { dg-error "deduced return type" }
+Int decltype(auto) g2() { return; } // { dg-error "deduced return type" }
+Int decltype(auto) g3() { return true; } // { dg-error "deduced return type" }
+int g4(Type decltype(auto) x) { return 0; } // { dg-error "cannot declare" }
+int g5(decltype(auto) x) { return 0; } // { dg-error "cannot declare" }
+
+template<Type decltype(auto) X, typename T>
+  requires SameAs<decltype(X), T>
+constexpr bool deduced_as = true;
+
+constexpr int Z = 10;
+
+static_assert(deduced_as<0, int>);
+static_assert(deduced_as<0, int&>); // { dg-error "invalid variable template" }
+static_assert(deduced_as<Z, const int>);
+static_assert(deduced_as<(Z), const int>); // { dg-error "invalid variable template" }
+static_assert(deduced_as<(Z), const int&>);
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-defarg1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-defarg1.C
new file mode 100644 (file)
index 0000000..c3ed309
--- /dev/null
@@ -0,0 +1,7 @@
+// { dg-do compile { target concepts } }
+
+template<typename T, typename U = T> concept C3 = true;
+template<class T> struct s1
+{
+  template <C3<T> U> void f() { }
+};
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-dep1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-dep1.C
new file mode 100644 (file)
index 0000000..b4f6113
--- /dev/null
@@ -0,0 +1,5 @@
+// { dg-do compile { target concepts } }
+
+template <class T> concept True = true;
+template <class T, int I = static_cast<int>(True<T>)> struct A { };
+template <class T> struct B: A<T> { };
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-dr1430.C b/gcc/testsuite/g++.dg/cpp2a/concepts-dr1430.C
new file mode 100644 (file)
index 0000000..f39921f
--- /dev/null
@@ -0,0 +1,14 @@
+// PR c++/66092
+// { dg-do compile { target c++2a } }
+
+template <typename T, typename U, typename... Args>
+  concept Similar = true;
+
+template <typename... Args>
+  requires Similar<Args...> // { dg-error "pack expansion" }
+void foo( Args... args ) {}
+
+int main()
+{
+  foo(1, 2, 3); // { dg-error "cannot call" }
+}
similarity index 56%
rename from gcc/testsuite/g++.dg/concepts/explicit-inst1.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-explicit-inst1.C
index 99bd72e069c88f58fd32d66a485b6c67c1564eb3..14d994c9b041aa3e0fa94993d7c90c3919d735bb 100644 (file)
@@ -1,11 +1,13 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
+// { dg-final { scan-assembler "_Z1gI1XEvT_" } }
+// { dg-final { scan-assembler "_Z1gI1YEvT_" } }
+// { dg-final { scan-assembler "_Z1gIiEvT_" } }
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept C = __is_class(T);
 
 template<typename T>
-  concept bool D() { return C<T>() && __is_empty(T); }
+  concept D = C<T> && __is_empty(T);
 
 struct X { };
 struct Y { int n; };
@@ -14,7 +16,6 @@ template<typename T> void g(T) { } // #1
 template<C T> void g(T) { } // #2
 template<D T> void g(T) { } // #3
 
-// FIXME: How do I test that these generate the right symbols?
 template void g(int); // Instantiate #1
 template void g(X); // Instantitae #3
 template void g(Y); // Instantiate #2
similarity index 72%
rename from gcc/testsuite/g++.dg/concepts/explicit-inst2.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-explicit-inst2.C
index ea313876f017945c7f86efe283418a18294c2a7f..6074bc78682632c0465d4bb0285d9086b360b70b 100644 (file)
@@ -1,11 +1,10 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept C = __is_class(T);
 
 template<typename T>
-  concept bool D() { return C<T>() && __is_empty(T); }
+  concept D = C<T> && __is_empty(T);
 
 struct X { };
 struct Y { int n; };
similarity index 51%
rename from gcc/testsuite/g++.dg/concepts/explicit-inst3.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-explicit-inst3.C
index 18d3c496f38919a3e06e6542b431a41010c8ef13..03ec9e9a698d6d0dcede1d91afb15796f809e69d 100644 (file)
@@ -1,11 +1,10 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept C = __is_class(T);
 
 template<typename T>
-  concept bool D() { return C<T>() && __is_empty(T); }
+  concept D = C<T> && __is_empty(T);
 
 struct X { };
 struct Y { int n; };
@@ -13,10 +12,10 @@ struct Y { int n; };
 template<typename T>
   struct S {
     void f() { }                 // #1
-    void f() requires C<T>() { } // #2
+    void f() requires C<T> { } // #2
 
-    void g() requires C<T>() { } // #1
-    void g() requires D<T>() { } // #2
+    void g() requires C<T> { } // #1
+    void g() requires D<T> { } // #2
   };
 
 template void S<int>::f(); // #1
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-explicit-inst4.C b/gcc/testsuite/g++.dg/cpp2a/concepts-explicit-inst4.C
new file mode 100644 (file)
index 0000000..81bc081
--- /dev/null
@@ -0,0 +1,17 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+  concept C = __is_class(T);
+
+template<typename T>
+  concept D = C<T> && __is_empty(T);
+
+template<typename T>
+  struct S {
+    void g() requires C<T> { } // #1
+    void g() requires D<T> { } // #2
+  };
+
+template void S<int>::g(); // { dg-error "match" }
+
+int main() { }
similarity index 76%
rename from gcc/testsuite/g++.dg/concepts/explicit-spec1.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec1.C
index bff06f21b6fe6433e17c84493fa154e855c0b762..d54bcdb32119b753872d710e9b49bfec7a4f6359 100644 (file)
@@ -1,13 +1,12 @@
-// { dg-do run { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do run { target c++2a } }
 
 #include <cassert>
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept C = __is_class(T);
 
 template<typename T>
-  concept bool D() { return C<T>() && __is_empty(T); }
+  concept D = C<T> && __is_empty(T);
 
 struct X { } x;
 struct Y { int n; } y;
similarity index 54%
rename from gcc/testsuite/g++.dg/concepts/explicit-spec2.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec2.C
index ca8b8a037bcbc9f3d289d13bf406c6f4921997d1..4103714b0043a13e28a53e72672e7409424c6e01 100644 (file)
@@ -1,8 +1,7 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept C = __is_class(T);
 
 struct X { };
 
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec3.C b/gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec3.C
new file mode 100644 (file)
index 0000000..76c6fb9
--- /dev/null
@@ -0,0 +1,13 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+  concept C = __is_class(T);
+
+template<C T> struct S;
+
+struct X { };
+
+// Not a valid explicit specialization, int does not satisfy C.
+template<> struct S<int> { }; // { dg-error "constraint failure" }
+
+int main() { }
similarity index 70%
rename from gcc/testsuite/g++.dg/concepts/explicit-spec4.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec4.C
index 75a2dec6f49b3e5dffd90d77486bfde3b30be7c0..0634eafb24875067c61b7ab41f2fccdd0f2b43b6 100644 (file)
@@ -1,13 +1,12 @@
-// { dg-do run { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do run { target c++2a } }
 
 #include <cassert>
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept C = __is_class(T);
 
 template<typename T>
-  concept bool D() { return C<T>() && __is_empty(T); }
+  concept D = C<T> && __is_empty(T);
 
 struct X { } x;
 struct Y { int n; } y;
@@ -17,10 +16,10 @@ int called = 0;
 template<typename T>
   struct S {
     void f() { called = 0; }                 // #1
-    void f() requires C<T>() { called = 0; } // #2
+    void f() requires C<T> { called = 0; } // #2
 
-    void g() requires C<T>() { } // #1
-    void g() requires D<T>() { } // #2
+    void g() requires C<T> { } // #1
+    void g() requires D<T> { } // #2
   };
 
 template<> void S<int>::f() { called = 1; } // Spec of #1
similarity index 53%
rename from gcc/testsuite/g++.dg/concepts/explicit-spec5.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec5.C
index d83eec11bc288592c1b2c2626735e5d846129ff2..b682b0d884688750456de87fe4229851109d3578 100644 (file)
@@ -1,13 +1,12 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 #include <cassert>
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept C = __is_class(T);
 
 template<typename T>
-  concept bool D() { return C<T>() && __is_empty(T); }
+  concept D = C<T> && __is_empty(T);
 
 struct X { } x;
 struct Y { int n; } y;
@@ -16,7 +15,7 @@ int called = 0;
 
 template<typename T>
   struct S {
-    void f() requires C<T>();
+    void f() requires C<T>;
   };
 
 template<> void S<int>::f() { called = 1; } // { dg-error "match" }
similarity index 84%
rename from gcc/testsuite/g++.dg/concepts/explicit-spec6.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-explicit-spec6.C
index b5487072e22e6300a530ae93b2f68c9e56648a09..13f04d7759418b40bd5989219030232d005ca5b6 100644 (file)
@@ -1,5 +1,4 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template<typename T>
 struct A {
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-feature-macro.C b/gcc/testsuite/g++.dg/cpp2a/concepts-feature-macro.C
new file mode 100644 (file)
index 0000000..56fbb68
--- /dev/null
@@ -0,0 +1,5 @@
+// { dg-do compile { target c++2a } }
+
+#ifndef __cpp_concepts
+#error __cpp_concepts not defined
+#endif
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-fn1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-fn1.C
new file mode 100644 (file)
index 0000000..33f3a74
--- /dev/null
@@ -0,0 +1,248 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+concept Type = true;
+
+template<typename T>
+concept False = false;
+
+template<typename T>
+concept Class = __is_class(T);
+
+template<typename T>
+concept EmptyClass = Class<T> && __is_empty(T);
+
+template<typename T, typename U>
+concept Classes = __is_class(T) && __is_class (U);
+
+struct empty { };
+
+struct nonempty { int n; };
+
+static_assert(Type<int>);
+static_assert(False<int>); // { dg-error "static assertion failed" }
+
+// Basic checks
+
+template<typename T>
+  requires Type<T>
+int fn1(T t) { return 0; }
+
+template<typename T>
+  requires False<T>
+int fn2(T t) { return 0; }
+
+void driver()
+{
+  fn1(0); // OK
+  fn2(0); // { dg-error "cannot call function" }
+}
+
+// Ordering
+
+template<typename T>
+concept Cf = requires (T t) { t.f(); };
+
+template<typename T>
+concept Cfg = Cf<T> && requires (T t) { t.g(); };
+
+template<typename T>
+concept Cf2 = requires (T t) { t.f(); };
+
+template<typename T>
+constexpr int algo(T t) { return 0; }
+
+template<typename T> requires Cf<T>
+constexpr int algo(T t) { return 1; }
+
+template<typename T> requires Cfg<T>
+constexpr int algo(T t) { return 2; }
+
+template<typename T> requires Cf<T>
+constexpr int ambig(T t) { return 1; }
+
+template<typename T> requires Cf2<T>
+constexpr int ambig(T t) { return 1; }
+
+struct T1 {
+  void f() { }
+};
+
+struct T2 : T1 {
+  void g() { }
+};
+
+void driver_0()
+{
+  T1 x;
+  T2 y;
+  static_assert(algo(0) == 0);
+  static_assert(algo(x) == 1);
+  static_assert(algo(y) == 2);
+  ambig(x); // { dg-error "call of overload | is ambiguous" }
+}
+
+template<typename T>
+struct S
+{
+  void f() requires Class<T> { }
+
+  template<typename U>
+  struct Inner
+  {
+    void g() requires Classes<T, U> { }
+  };
+
+  template<typename U> requires Classes<T, U> void h(U) { }
+};
+
+struct X { };
+
+void driver_1()
+{
+  S<X> s1;
+  s1.f(); // OK
+  s1.h(s1); // OK
+  s1.h(0); // { dg-error "no matching function" }
+
+  S<int> s2;
+  s2.f(); // { dg-error "no matching function" }
+
+  S<X>::Inner<X> si1;
+  si1.g();
+
+  S<X>::Inner<int> si2;
+  si2.g(); // { dg-error "no matching function" }
+}
+
+// Check constraints on non-dependent arguments, even when in a
+// dependent context.
+
+template<typename T> requires Class<T> void f1(T x) { }
+
+// fn1-2.C -- Dependent checks
+template<typename T>
+void caller_1(T x) 
+{ 
+  f1(x); // Unchecked dependent arg.
+  f1(empty{}); // Checked non-dependent arg, but OK
+  f1(0); // { dg-error "cannot call function" }
+}
+
+// fn3.c -- Ordering
+template<typename T> 
+  requires Class<T> 
+constexpr int f2(T x) { return 1; }
+
+template<typename T> 
+  requires EmptyClass<T> 
+constexpr int f2(T x) { return 2; }
+
+template<typename T> 
+constexpr int f3(T x) requires Class<T> { return 1; }
+
+template<typename T> 
+constexpr int f3(T x) requires EmptyClass<T> { return 2; }
+
+void driver_2() 
+{
+  f2(0); // { dg-error "no matching function" }
+  static_assert (f2(empty{}) == 2);
+  static_assert (f2(nonempty{}) == 1);
+  f3(0); // { dg-error "no matching function" }
+  static_assert (f3(empty{}) == 2);
+  static_assert (f3(nonempty{}) == 1);
+}
+
+// fn8.C -- Address of overloaded functions
+template<typename T> requires Type<T> void ok(T) { }
+template<typename T> requires Class<T> void err(T) { }
+
+auto p1 = &ok<int>;
+auto p2 = &err<int>; // { dg-error "no matches" }
+void (*p3)(int) = &ok<int>;
+void (*p4)(int) = &err<int>; // { dg-error "no matches" }
+void (*p5)(int) = &ok;
+void (*p6)(int) = &err; // { dg-error "no matches" }
+
+template<typename T> void g(T x) { }
+
+void driver_3 () 
+{
+  g(&ok<int>);
+  g(&err<int>); // { dg-error "no matches" }
+}
+
+
+struct S2 {
+  template<typename T> requires Type<T> int ok(T) { return 0; }
+  template<typename T> requires Class<T> int err(T) { return 0; }
+};
+
+auto p7 = &S2::ok<int>;
+auto p8 = &S2::err<int>; // { dg-error "no matches" }
+int (S2::*p9)(int) = &S2::ok<int>;
+int (S2::*p10)(int) = &S2::err<int>; // { dg-error "no matches" }
+int (S2::*p11)(int) = &S2::ok;
+int (S2::*p12)(int) = &S2::err; // { dg-error "no matches" }
+
+// fn9.C -- Ordering with function address
+template<typename T> 
+  requires Class<T> 
+constexpr int fn(T) { return 1; }
+
+template<typename T> 
+  requires EmptyClass<T>
+constexpr int fn(T) { return 2; }
+
+struct S3 
+{
+  template<typename T> 
+    requires Class<T> 
+  constexpr int fn(T) const { return 1; }
+  
+  template<typename T> 
+    requires EmptyClass<T> 
+  constexpr int fn(T) const { return 2; }
+};
+
+void driver_5 () {
+  struct X { };
+  struct Y { X x; };
+
+  constexpr X x;
+  constexpr Y y;
+  constexpr S3 s;
+
+  constexpr auto p1 = &fn<X>; // Empty f
+  static_assert (p1(x) == 2);
+
+  constexpr auto p2 = &fn<Y>; // Class f
+  static_assert(p2(y) == 1);
+
+  constexpr auto p3 = &S3::fn<X>; // Empty f
+  static_assert((s.*p3)(x) == 2);
+
+  constexpr auto p4 = &S3::fn<Y>; // Empty f
+  static_assert((s.*p4)(y) == 1);
+}
+
+
+// Redeclarations
+
+// FIXME: This should be a diagnosable error. The programmer has moved
+// the requirements from the template-head to the declarator.
+template<typename T> requires Type<T> void f3();
+template<typename T> void f3() requires Type<T>;
+
+void driver_4()
+{
+  f3<int>(); // { dg-error "call of overload | ambiguous" }
+}
+
+template<template<typename T> requires true class X> void f4();
+template<template<typename T> class X> void f4(); // OK: different declarations
+
+template<typename T> requires Type<T> void def() { }
+template<typename T> requires Type<T> void def() { } // { dg-error "redefinition" }
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-fn2.C b/gcc/testsuite/g++.dg/cpp2a/concepts-fn2.C
new file mode 100644 (file)
index 0000000..ddf99aa
--- /dev/null
@@ -0,0 +1,111 @@
+// { dg-do run { target c++2a } }
+
+#define assert(E) if (!(E)) __builtin_abort();
+
+template<typename T>
+  concept C = __is_class(T);
+
+template<typename T>
+  concept D = __is_empty(T);
+
+struct X { } x;
+struct Y { int n; } y;
+
+int called = 0;
+
+// Test constrained member definitions
+template<typename T>
+  struct S1 {
+    void f1() requires C<T> { }
+
+    void f2() requires C<T> { called = 1; }
+    void f2() requires (!C<T>) { called = 2; }
+
+    void f3() { called = 1; }
+    void f3() requires C<T> { called = 2; }
+    void f3() requires C<T> && D<T> { called = 3; }
+
+    void g1() requires C<T> && true;
+
+    void g2() requires C<T>;
+    void g2() requires (!C<T>);
+
+    void g3();
+    void g3() requires C<T>;
+    void g3() requires C<T> and D<T>;
+
+    template<C U> void h1(U u) { called = 1; }
+    template<C U> void h2(U u);
+    template<C U> void h3(U u) requires D<U>;
+  };
+
+template<C T>
+  struct S2 {
+    void f(T) requires D<T>;
+  };
+
+
+int main() {
+  S1<X> sx;
+  S1<Y> sy;
+  S1<int> si;
+
+  // Defined in-class
+  sx.f1();
+  sx.f2(); assert(called == 1);  
+  sx.f3(); assert(called == 3);
+
+  sy.f1();
+  sy.f2(); assert(called == 1);
+  sy.f3(); assert(called == 2);
+
+  si.f2(); assert(called == 2);
+  si.f3(); assert(called == 1);
+
+  // Member function template tests
+  S1<int> s1i;
+  s1i.h1(x); assert(called == 1);
+  s1i.h2(x); assert(called == 2);
+  s1i.h3(x); assert(called == 3);
+
+  // Defined out of class.
+  sx.g1();
+  sx.g2(); assert(called == 1);
+  sx.g3(); assert(called == 3);
+
+  sy.g1();
+  sy.g2(); assert(called == 1);
+  sy.g3(); assert(called == 2);
+
+  si.g2(); assert(called == 2);
+  si.g3(); assert(called == 1);
+}
+
+template<typename T>
+  void S1<T>::g1() requires C<T> and true { }
+
+template<typename T>
+  void S1<T>::g2() requires C<T> { called = 1; }
+
+template<typename T>
+  void S1<T>::g2() requires (!C<T>) { called = 2; }
+
+template<typename T>
+  void S1<T>::g3() { called = 1; }
+
+template<typename T>
+  void S1<T>::g3() requires C<T> { called = 2; }
+
+template<typename T>
+  void S1<T>::g3() requires C<T> and D<T> { called = 3; }
+
+template<typename T>
+  template<C U>
+    void S1<T>::h2(U u) { called = 2; }
+
+template<typename T>
+  template<C U>
+      void S1<T>::h3(U u) requires D<U> { called = 3; }
+
+template<C T>
+  void S2<T>::f(T t) requires D<T> { called = 4; }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-fn3.C b/gcc/testsuite/g++.dg/cpp2a/concepts-fn3.C
new file mode 100644 (file)
index 0000000..e73ae23
--- /dev/null
@@ -0,0 +1,49 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+concept type = true;
+
+template<typename T, typename U>
+concept same_as = __is_same_as(T, U);
+
+template<typename T>
+concept integral = __is_same_as(T, int);
+
+template<typename... Ts>
+concept all_integral = (integral<Ts> && ...);
+
+void f1(integral auto... args) { }
+void f2(all_integral auto... args) { }
+
+template<type T> requires true
+void f3(T, integral auto... args) { }
+
+template<type T>
+struct S
+{
+  void f1(integral auto... args) { }
+  void f2(all_integral auto... args) { }
+
+  template<type U> requires true
+  void f3(U, integral auto... args) { }
+};
+
+int main()
+{
+  f1(1, 2, 3);
+  f1(1, 2, 3u); // { dg-error "cannot call" }
+  f2(1, 2, 3);
+  f2(1, 2, 3u); // { dg-error "cannot call" }
+  f3(1, 2, 3);
+  f3(1, 2, 3u); // { dg-error "cannot call" }
+  f3(1u, 2, 3);
+
+  S<void> s;
+  s.f1(1, 2, 3);
+  s.f1(1, 2, 3u); // { dg-error "no matching function" }
+  s.f2(1, 2, 3);
+  s.f2(1, 2, 3u); // { dg-error "no matching function" }
+  s.f3(1, 2, 3);
+  s.f3(1, 2, 3u); // { dg-error "no matching function" }
+  s.f3(1u, 2, 3);
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-fnparm1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-fnparm1.C
new file mode 100644 (file)
index 0000000..586b510
--- /dev/null
@@ -0,0 +1,10 @@
+// { dg-do compile { target concepts } }
+
+template <class T> void f(T t)
+  requires requires { static_cast<T&&>(t); }
+{}
+
+int main()
+{
+  f(42);
+}
similarity index 71%
rename from gcc/testsuite/g++.dg/concepts/friend1.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-friend1.C
index 9050b557b3bc9a02d951cde7afc3a161823bf326..da97575687fd7fafaa1160f7ff19b8447fdf3b0a 100644 (file)
@@ -1,8 +1,7 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template<typename T>
-  concept bool Eq() { return requires(T t) { t == t; }; }
+  concept Eq = requires(T t) { t == t; }; // { dg-message "in requirements" }
 
 struct Nt {
   template<Eq T> friend void f(T) { }
@@ -15,7 +14,7 @@ template<Eq T>
 
 template<typename T>
   struct S {
-    friend bool operator==(S, S) requires Eq<T>() { return true; }
+    friend bool operator==(S, S) requires Eq<T> { return true; }
 
     friend void proc<>(S*); // { dg-error "does not match any template declaration" }
   };
similarity index 66%
rename from gcc/testsuite/g++.dg/concepts/friend2.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-friend2.C
index 8ef600222e5cc0dd2a71eaabf5921de2e757d701..6aa9d9617791b0ff54c84521cffa055365ed5ebf 100644 (file)
@@ -1,8 +1,7 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template<typename T>
-  concept bool Eq() { return requires(T t) { t == t; }; }
+  concept Eq = requires(T t) { t == t; }; // { dg-message "in requirements" }
 
 template<Eq T> struct Foo { };
 
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-friend3.C b/gcc/testsuite/g++.dg/cpp2a/concepts-friend3.C
new file mode 100644 (file)
index 0000000..4f49358
--- /dev/null
@@ -0,0 +1,20 @@
+// { dg-do compile { target concepts } }
+
+template <class T> concept True = true;
+
+template <True U> struct B { int i = ++U::x; };
+template <True U> void f() { ++U::x; }
+
+template <class V> class C
+{
+  static int x;
+
+  template <True U> friend struct B;
+  template <True U> friend void f();
+};
+
+int main()
+{
+  f<C<int>>();
+  B<C<int>>();
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-iconv1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-iconv1.C
new file mode 100644 (file)
index 0000000..cc2ce7e
--- /dev/null
@@ -0,0 +1,22 @@
+// PR c++/67240
+// { dg-do compile { target c++2a } }
+
+template <class T, class U> concept Same = __is_same_as(T,U);
+
+int foo(int x)
+{
+    return x;
+}
+template <typename T>
+concept C1 = requires (T x) {
+    {foo(x)} -> Same<int&>;
+};
+
+template <typename T>
+concept C2 = requires (T x) {
+    {foo(x)} -> Same<void>;
+};
+static_assert( C1<int> );      // { dg-error "assert" }
+static_assert( C2<int> );      // { dg-error "assert" }
similarity index 60%
rename from gcc/testsuite/g++.dg/concepts/inherit-ctor2.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-inherit-ctor2.C
index cb81d13e6d909a7a7a5dba2980ee6f974ae711d4..aa244bc04c1222d1aee680910d178563addb2b1f 100644 (file)
@@ -1,12 +1,11 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept C = __is_class(T);
 
 template<typename T>
   struct S1 {
-    S1(double) requires C<T>() { }
+    S1(double) requires C<T> { }
   };
 
 template<typename T>
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-inherit-ctor4.C b/gcc/testsuite/g++.dg/cpp2a/concepts-inherit-ctor4.C
new file mode 100644 (file)
index 0000000..75190eb
--- /dev/null
@@ -0,0 +1,18 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+  concept C = __is_class(T);
+
+template<typename T>
+  struct S1 {
+    template<C U> S1(U x) { }
+  };
+
+template<typename T>
+  struct S2 : S1<T> {
+    using S1<T>::S1; // { dg-error "no matching function" }
+  };
+
+int main() {
+  S2<int> s(0); // { dg-error "use of deleted function" }
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-inherit-ctor5.C b/gcc/testsuite/g++.dg/cpp2a/concepts-inherit-ctor5.C
new file mode 100644 (file)
index 0000000..2044ab5
--- /dev/null
@@ -0,0 +1,18 @@
+// { dg-do compile { target concepts } }
+
+template <class T> struct A
+{
+  constexpr A(T) requires (sizeof(T) > 1) {}
+
+  A(T);
+};
+
+template <class T> struct B: A<T>
+{
+  using A<T>::A;
+};
+
+int main()
+{
+  constexpr B<int> b = 42;
+}
similarity index 56%
rename from gcc/testsuite/g++.dg/concepts/lambda1.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-lambda1.C
index a77e65459b7c5e04679b5be09ad27b168799e02e..ef1968899f6ab322858ab366ce4a85ee6c9c53fb 100644 (file)
@@ -1,6 +1,5 @@
 // PR c++/82565
-// { dg-do compile { target c++14 } }
-// { dg-additional-options -fconcepts }
+// { dg-do compile { target c++2a } }
 
 struct string
 {
@@ -9,18 +8,20 @@ struct string
   bool empty() const;
 };
 
+template<class From, class To>
+concept convertible_to = requires(From (&f)(), void (&g)(To)) { g(f()); };
+
 template<typename T, typename ReturnType>
-concept bool Concept() {
-  return requires(T t, const string& s) {
-    { t(s) } -> ReturnType;
+concept Concept =
+  requires(T t, const string& s) {
+    { t(s) } -> convertible_to<ReturnType>;
   };
-}
 
 struct test {
   string _str;
 
   template<typename Visitor>
-    requires Concept<Visitor, bool>()
+    requires Concept<Visitor, bool>
   decltype(auto) visit(Visitor&& visitor) const {
     return visitor(_str);
   }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-locations1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-locations1.C
new file mode 100644 (file)
index 0000000..6c81c14
--- /dev/null
@@ -0,0 +1,22 @@
+// { dg-do compile { target c++2a } }
+
+struct S
+{
+  concept S();  // { dg-error "3:a constructor cannot be .concept." }
+  // { dg-error "concept definition syntax" "" { target *-*-* } .-1 }
+  concept int s = 1;  // { dg-error "3:non-static data member .s. declared .concept." }
+  // { dg-error "concept definition syntax" "" { target *-*-* } .-1 }
+  concept void foo();  // { dg-error "3:a concept cannot be a member function" }
+  // { dg-error "concept definition syntax" "" { target *-*-* } .-1 }
+  concept ~S();  // { dg-error "3:a destructor cannot be .concept." }
+  // { dg-error "concept definition syntax" "" { target *-*-* } .-1 }
+};
+
+typedef concept int my_int;  // { dg-error "9:.concept. cannot appear in a typedef declaration" }
+// { dg-error "concept definition syntax" "" { target *-*-* } .-1 }
+
+void bar(concept int);  // { dg-error "10:a parameter cannot be declared .concept." }
+// { dg-error "concept definition syntax" "" { target *-*-* } .-1 }
+
+concept int i = 0;  // { dg-error "1:a non-template variable cannot be .concept." }
+// { dg-error "concept definition syntax" "" { target *-*-* } .-1 }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-member-concept.C b/gcc/testsuite/g++.dg/cpp2a/concepts-member-concept.C
new file mode 100644 (file)
index 0000000..f3a2d06
--- /dev/null
@@ -0,0 +1,21 @@
+// { dg-do compile { target c++2a } }
+
+// FIXME: Diagnostics should be better.
+
+struct Base {
+  template<typename T>
+  static concept D = __is_same_as(T, int); // { dg-error "static data member" }
+
+  template<typename T, typename U>
+  static concept E = __is_same_as(T, U); // { dg-error "static data member" }
+
+  template<typename T>
+  concept C1 = __is_same_as(T, int); // { dg-error "not in namespace scope" }
+};
+
+union U {
+  template<typename T>
+  concept C = true; // // { dg-error "not in namespace scope" }
+};
+
+// { dg-excess-errors "deprecated" }
similarity index 54%
rename from gcc/testsuite/g++.dg/concepts/memfun-err.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-memfun-err.C
index 6e1ad00c5cfa6ca8dac00795dd3f0c1110c29671..acfa188af21974fc0024f279ffc069896c634ac0 100644 (file)
@@ -1,12 +1,10 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
-
+// { dg-do compile { target c++2a } }
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept C = __is_class(T);
 
 template<typename T>
-  concept bool D() { return __is_empty(T); }
+  concept D = __is_empty(T);
 
 struct X { } x;
 struct Y { int n; } y;
@@ -16,15 +14,15 @@ int called = 0;
 // Test constrained member definitions
 template<typename T>
   struct S1 { // { dg-message "defined here" }
-    void f1() requires C<T>() { }
-    void g1() requires C<T>() and true;
+    void f1() requires C<T> { }
+    void g1() requires C<T> and true;
     template<C U> void h1(U u) { called = 1; }
 
-    void g2() requires C<T>(); // { dg-message "candidate" }
+    void g2() requires C<T>; // { dg-message "candidate" }
   };
 
 template<typename T>
-  void S1<T>::g2() requires D<T>() { } // { dg-error "no declaration matches" }
+  void S1<T>::g2() requires D<T> { } // { dg-error "no declaration matches" }
 
 int main() {
   S1<X> sx;
similarity index 56%
rename from gcc/testsuite/g++.dg/concepts/memfun.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-memfun.C
index 818c7e61c8a862343b559cf5df18d2bc5374b3c0..f6ad519635244bf6fb7c1d88a70af7a6bdd3d592 100644 (file)
@@ -1,13 +1,12 @@
-// { dg-do run { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do run { target c++2a } }
 
 #include <cassert>
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept C = __is_class(T);
 
 template<typename T>
-  concept bool D() { return __is_empty(T); }
+  concept D = __is_empty(T);
 
 struct X { } x;
 struct Y { int n; } y;
@@ -17,32 +16,32 @@ int called = 0;
 // Test constrained member definitions
 template<typename T>
   struct S1 {
-    void f1() requires C<T>() { }
+    void f1() requires C<T> { }
 
-    void f2() requires C<T>() { called = 1; }
-    void f2() requires not C<T>() { called = 2; }
+    void f2() requires C<T> { called = 1; }
+    void f2() requires (not C<T>) { called = 2; }
 
     void f3() { called = 1; }
-    void f3() requires C<T>() { called = 2; }
-    void f3() requires C<T>() and D<T>() { called = 3; }
+    void f3() requires C<T> { called = 2; }
+    void f3() requires C<T> and D<T> { called = 3; }
 
-    void g1() requires C<T>() and true;
+    void g1() requires C<T> and true;
 
-    void g2() requires C<T>();
-    void g2() requires not C<T>();
+    void g2() requires C<T>;
+    void g2() requires (not C<T>);
 
     void g3();
-    void g3() requires C<T>();
-    void g3() requires C<T>() and D<T>();
+    void g3() requires C<T>;
+    void g3() requires C<T> and D<T>;
 
     template<C U> void h1(U u) { called = 1; }
     template<C U> void h2(U u);
-    template<C U> void h3(U u) requires D<U>();
+    template<C U> void h3(U u) requires D<U>;
   };
 
 template<C T>
   struct S2 {
-    void f(T) requires D<T>();
+    void f(T) requires D<T>;
   };
 
 
@@ -83,22 +82,22 @@ int main() {
 }
 
 template<typename T>
-  void S1<T>::g1() requires C<T>() and true { }
+  void S1<T>::g1() requires C<T> and true { }
 
 template<typename T>
-  void S1<T>::g2() requires C<T>() { called = 1; }
+  void S1<T>::g2() requires C<T> { called = 1; }
 
 template<typename T>
-  void S1<T>::g2() requires not C<T>() { called = 2; }
+  void S1<T>::g2() requires (not C<T>) { called = 2; }
 
 template<typename T>
   void S1<T>::g3() { called = 1; }
 
 template<typename T>
-  void S1<T>::g3() requires C<T>() { called = 2; }
+  void S1<T>::g3() requires C<T> { called = 2; }
 
 template<typename T>
-  void S1<T>::g3() requires C<T>() and D<T>() { called = 3; }
+  void S1<T>::g3() requires C<T> and D<T> { called = 3; }
 
 template<typename T>
   template<C U>
@@ -106,7 +105,7 @@ template<typename T>
 
 template<typename T>
   template<C U>
-      void S1<T>::h3(U u) requires D<U>() { called = 3; }
+      void S1<T>::h3(U u) requires D<U> { called = 3; }
 
 template<C T>
-  void S2<T>::f(T t) requires D<T>() { called = 4; }
+  void S2<T>::f(T t) requires D<T> { called = 4; }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-memtmpl1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-memtmpl1.C
new file mode 100644 (file)
index 0000000..ee28d5b
--- /dev/null
@@ -0,0 +1,17 @@
+// { dg-do compile { target c++2a } }
+
+template <class T>
+struct A {
+  template <class U>
+    requires (sizeof(T) == 1)
+      static void f(U);
+  
+  template <class U>
+    requires (sizeof(T) == 2)
+      static void f(U);
+  
+  void g()
+  {
+    f(42);
+  }
+};
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-memtmpl2.C b/gcc/testsuite/g++.dg/cpp2a/concepts-memtmpl2.C
new file mode 100644 (file)
index 0000000..f60b220
--- /dev/null
@@ -0,0 +1,15 @@
+// { dg-do compile { target concepts } }
+
+template <class T, class U> concept NotSame = !__is_same_as (T, U);
+
+template <class T, class X>
+struct A
+{
+  template <NotSame<A> U> void f(U) { }
+  template <class U> void f(U);
+};
+
+int main()
+{
+  A<int,int>().f<char>(0);
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-nested1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-nested1.C
new file mode 100644 (file)
index 0000000..8fc965f
--- /dev/null
@@ -0,0 +1,13 @@
+// { dg-do compile { target concepts } }
+
+namespace N { template <class T> concept True = true; }
+template <class T> struct A { };
+
+template <class T>
+requires N::True<T> && requires { typename A<T>; }
+void f();
+
+int main()
+{
+  f<int>();
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-noexcept1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-noexcept1.C
new file mode 100644 (file)
index 0000000..e5a9b72
--- /dev/null
@@ -0,0 +1,25 @@
+// { dg-do compile { target c++2a } }
+
+void f1(int);
+void f2(int) noexcept;
+
+template<typename T>
+concept C1 = requires (T t) { // { dg-message "in requirements" }
+  { f1(t) } noexcept;
+};
+
+template<typename T>
+concept C2 = requires (T t) {
+  { f2(t) } noexcept;
+};
+
+template<C1 T>
+void g1(T t);
+
+template<C2 T>
+void g2(T t);
+
+void test() {
+  g1(0); // { dg-error "cannot call" }
+  g2(0);
+}
\ No newline at end of file
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-p1141.C b/gcc/testsuite/g++.dg/cpp2a/concepts-p1141.C
new file mode 100644 (file)
index 0000000..deb409a
--- /dev/null
@@ -0,0 +1,98 @@
+// { dg-do compile }
+// { dg-options "-std=c++2a" }
+
+template<typename T>
+concept Type = true;
+
+template<typename T>
+concept Bottom = false;
+
+template<typename T>
+concept Class = __is_class(T);
+
+template<auto N>
+concept Number = true;
+
+template<template<typename> class T>
+concept Template = true;
+
+struct empty { };
+
+Type x1 = 0; // { dg-error "expected 'auto'" }
+Type auto x2 = 0;
+
+Number x3 = 0; // { dg-error "does not constrain a type" }
+Template x4 = 0; // { dg-error "does not constrain a type" }
+
+Type const& x5 = 0; // { dg-error "expected 'auto'" }
+const Type& x6 = 0; // { dg-error "expected 'auto'" }
+Type auto const& x7 = 0;
+const Type auto& x8 = 0;
+Type const auto& x9 = 0; // { dg-error "expected 'auto'|two or more data types" }
+
+template<Type T> // OK: T is a type parameter.
+void f1(T);
+
+template<Number N> // { dg-error "does not constrain a type" }
+void f2();
+
+template<Template X> // { dg-error "does not constrain a type" }
+void f3();
+
+template<Type auto N> // OK: N is a non-type parameter.
+void f4() { }
+
+template<Bottom auto N> // OK: but never usable.
+void f5();
+
+void driver() {
+  f4<0>();
+  f5<0>(); // { dg-error "no matching function for call | constraints not satisfied" }
+}
+
+Type f6() { return 0; } // { dg-error "expected 'auto'" }
+static_assert(__is_same_as(decltype(f6), int()));
+
+Type auto f7() { return 0; }
+static_assert(__is_same_as(decltype(f7), int()));
+
+Type f8() { return 0; } // { dg-error "expected 'auto'" }
+auto f9() -> Type { return 0; } // { dg-error "expected 'auto'" }
+
+Type f10() { } // { dg-error "expected 'auto'" }
+auto f11() -> Type { } // { dg-error "expected" }
+
+Number f12(); // { dg-error "does not constrain a type" }
+auto f13() -> Number; // { dg-error "does not constrain a type" }
+
+Template f14(); // { dg-error "does not constrain a type" }
+auto f15() -> Template; // { dg-error "does not constrain a type" }
+
+Type f16() { return 0; } // { dg-error "expected 'auto'" }
+auto f17() -> Type { return 0; } // { dg-error "expected 'auto'" }
+
+// Abbreviated function templates
+
+void f18(Class x) { } // { dg-error "expected 'auto'" }
+void f19(Class auto x) { }
+void f20(Class auto x, Class auto y) { }
+
+void driver_1()
+{
+  f19(0); // { dg-error "cannot call function" }
+  f19(empty{});
+  f20(0, empty{}); // { dg-error "cannot call function" }
+  f20(empty{}, empty{});
+}
+
+// Abbreviated function redeclarations
+
+// Functionally equivalent but not equivalent.
+void f21(Class auto x);
+template<Class T> void f21(T x);
+
+void driver_2()
+{
+  f21(empty{}); // { dg-error "call of overload | ambiguous" }
+}
+
similarity index 84%
rename from gcc/testsuite/g++.dg/concepts/partial-spec.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec.C
index 0ff8ec2fa1a4d15fa4b6a5a929a1d6ba6f9c63d6..25b4e2a40001e2cdfe74d94d6a550b8eb2e3cc8e 100644 (file)
@@ -1,4 +1,4 @@
-// { dg-do compile { target c++17 } }
+// { dg-do compile { target c++17_only } }
 // { dg-options "-fconcepts" }
 
 // Check that constraints don't break unconstrained partial
similarity index 67%
rename from gcc/testsuite/g++.dg/concepts/partial-spec2.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec2.C
index 2449c37fc9930183aa960d97178d96adc49d36c0..566d8dd55b653edf7ddd6066747f7bf417f5b043 100644 (file)
@@ -1,6 +1,11 @@
 // PR c++/67084
 // { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-additional-options "-fconcepts" }
+
+template <class T>
+concept True = true;
+template <class T>
+concept False = false;
 
 template <class T>
 constexpr bool p = false;
@@ -9,11 +14,11 @@ template <class T>
 constexpr bool p<T*> = false;
 
 template <class T>
-  requires true
+  requires True<T>
 constexpr bool p<T*> = false;
 
 template <class T>
-  requires true && T() == 0
+  requires True<T> && (T() == 0)
 constexpr bool p<T*> = true;
 
 template <class T>
@@ -23,11 +28,11 @@ template <class T>
 constexpr bool q<T*> = true;
 
 template <class T>
-  requires false
+  requires False<T>
 constexpr bool q<T*> = false;
 
 template <class T>
-  requires false && T() != 0
+  requires False<T> && (T() != 0)
 constexpr bool q<T*> = false;
 
 static_assert (p<int*>,"");
similarity index 82%
rename from gcc/testsuite/g++.dg/concepts/partial-spec3.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec3.C
index d4071814909ea49ee0dd57b9f29e3af8df03fe15..2970af100f98dac7df0e3b5078cedd0bdcf35296 100644 (file)
@@ -1,5 +1,5 @@
 // { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-additional-options "-fconcepts" }
 
 template <class T> struct A { };
 template <class T> requires false struct A<T*> { };
similarity index 87%
rename from gcc/testsuite/g++.dg/concepts/partial-spec4.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec4.C
index 70461b4370f6f16da3c525e9db8f7d43c8473d74..5ba3ab19c23fc4b688c76e165e157973d2e4631e 100644 (file)
@@ -1,7 +1,6 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
-template <class T> concept bool is_int = __is_same_as(T,int);
+template <class T> concept is_int = __is_same_as(T,int);
 
 template <class T> struct A { };
 template <is_int T> struct A<T*> {
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec5.C b/gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec5.C
new file mode 100644 (file)
index 0000000..f33f749
--- /dev/null
@@ -0,0 +1,12 @@
+// PR c++/67138
+// { dg-do compile { target c++2a } }
+
+template <class T>
+concept Auto = true;
+
+template <Auto T>
+struct test {};
+
+template <Auto T>
+  requires requires (T t) { t + t; }
+struct test<T> {};
similarity index 72%
rename from gcc/testsuite/g++.dg/concepts/partial-spec6.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-partial-spec6.C
index 1d80ee36e5c38945455da269a9fda8763b83e7a8..2bb4bacb2ef37fd1e5fc32e5b17dc84a24247835 100644 (file)
@@ -1,9 +1,8 @@
 // PR c++/67152
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template <class T>
-concept bool HasType = requires { typename T::type; };
+concept HasType = requires { typename T::type; };
 
 template<class T>
 struct trait {
@@ -19,7 +18,7 @@ trait<has_type>::type foo() {}
 // constraint so this partial specialization would not have been
 // selected.
 template<class T>
-  requires !HasType<T>
+  requires (!HasType<T>)
 struct trait<T> {
   using type = void;
 };
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-placeholder1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-placeholder1.C
new file mode 100644 (file)
index 0000000..cbea81d
--- /dev/null
@@ -0,0 +1,14 @@
+// { dg-do compile { target c++2a } }
+
+template<class T, class U>
+concept Same = __is_same_as(T, U);
+
+template<typename T>
+concept C1 = true;
+
+template<typename T, typename U>
+concept C2 = true;
+
+C1 auto        c1 = 0;
+C2<int> auto   c2 = 0;
+Same<int> auto s1 = 'a'; // { dg-error "does not satisfy|is_same" }
similarity index 75%
rename from gcc/testsuite/g++.dg/concepts/regress/alias-decl-42.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-pr59200.C
index 6d5a4d5fdea8587f53f27ac2c4709efdc5116854..995672c0694bc2dfa823129e18102c3f94df73a5 100644 (file)
@@ -1,5 +1,6 @@
 // PR c++/59200
-// { dg-options "-std=c++17 -fconcepts" }
+// { dg-do compile { target c++17 } }
+// { dg-additional-options "-fconcepts" }
 
 struct A
 {
similarity index 63%
rename from gcc/testsuite/g++.dg/concepts/pr65552.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-pr65552.C
index 318fdccfbdc9439e854f086d861f9dbe7b71d884..5af2a40e9a3df67806d039143df01e4d098888c7 100644 (file)
@@ -1,13 +1,11 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template<typename T>
-concept bool Concept() {
-  return requires () {
+concept Concept =
+  requires () {
     typename T::member_type1;
     typename T::member_type2;
   };
-}
 
 struct model {
   using member_type1 = int;
similarity index 63%
rename from gcc/testsuite/g++.dg/concepts/pr65575.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-pr65575.C
index efaf958121bfeee2be6fad7146d698183406d6b7..3bfde5e2c75f0aaf48fd674b7dfce2e930617614 100644 (file)
@@ -1,5 +1,6 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// PR c++/65575
+// { dg-do compile { target c++17_only } }
+// { dg-additional-options "-fconcepts" }
 
 template<typename T>
 concept bool C = false;
@@ -11,12 +12,10 @@ auto f4() -> int& requires false;
 auto f5() -> int* requires false;
 auto f6() -> int requires false;
 
-int (*p)() requires true; // { dg-error "" }
-int (&p)() requires true; // { dg-error "" }
+int (*p1)() requires true; // { dg-error "" }
+int (&p2)() requires true; // { dg-error "" }
 int g(int (*)() requires true); // { dg-error "" }
 
-int f() { return 0; }
-
 int
 main()
 {
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr65634.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr65634.C
new file mode 100644 (file)
index 0000000..52f24ec
--- /dev/null
@@ -0,0 +1,18 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+concept C1 =
+    requires () {
+        { T::smf() } noexcept;
+    };
+
+struct M1 {
+    static void smf() noexcept;
+};
+template<typename T>
+concept C2 = C1<typename T::type>;
+
+struct M2 {
+    using type = M1;
+};
+static_assert(C2<M2>, "");
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr65636.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr65636.C
new file mode 100644 (file)
index 0000000..b99a343
--- /dev/null
@@ -0,0 +1,8 @@
+// { dg-do compile { target c++2a } }
+
+using TD = int;
+
+template<typename T>
+concept C = requires () { typename TD; };
+
+static_assert(C<int>, "");
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr65848.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr65848.C
new file mode 100644 (file)
index 0000000..b246d25
--- /dev/null
@@ -0,0 +1,59 @@
+// { dg-do compile { target c++2a } }
+
+// Performance test... This should be fast.
+
+#include <type_traits>
+
+template<typename T>
+concept Destructible = std::is_destructible<T>::value;
+template<typename T, typename... Args>
+concept Constructible = Destructible<T> && std::is_constructible<T, Args...>::value;
+template<typename T>
+concept Move_constructible = Constructible<T, T&&>;
+template<typename T>
+concept Copy_constructible = Move_constructible<T> && Constructible<T, const T&>;
+template<typename T, typename U>
+concept Assignable = std::is_assignable<T, U>::value;
+template<typename T>
+concept Move_assignable = Assignable<T&, T&&>;
+template<typename T>
+concept Copy_assignable = Move_assignable<T> && Assignable<T&, const T&>;
+template<typename T>
+concept Copyable = Copy_constructible<T> && Copy_assignable<T>;
+
+template<typename T>
+concept C1 = Copyable<T>;
+template<typename T>
+concept C2 = C1<T>;
+template<typename T>
+concept C3 = C2<T>;
+template<typename T>
+concept C4 = C3<T>;
+template<typename T>
+concept C5 = C4<T>;
+template<typename T>
+concept C6 = C5<T>;
+template<typename T>
+concept C7 = C6<T>;
+template<typename T>
+concept C8 = C7<T>;
+template<typename T>
+concept C9 = C8<T>;
+template<typename T>
+concept C10 = C9<T>;
+template<typename T>
+concept C11 = C10<T>;
+
+struct S1 {};
+struct S2 {};
+struct S3 {};
+struct S4 {};
+struct S5 {};
+struct S6 {};
+
+static_assert(C11<S1>, "");
+static_assert(C11<S2>, "");
+static_assert(C11<S3>, "");
+static_assert(C11<S4>, "");
+static_assert(C11<S5>, "");
+static_assert(C11<S6>, "");
similarity index 70%
rename from gcc/testsuite/g++.dg/concepts/pr65854.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-pr65854.C
index 28eac885a4628e9c63dabbf6c43c2e5b8d044f52..a21073c4d67bc2668ab0390877fc7dd5bbdd8c29 100644 (file)
@@ -1,5 +1,5 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// PR c++/65854
+// { dg-do compile { target c++2a } }
 
 // Handle alias templates in type requirements.
 
@@ -13,13 +13,10 @@ template<typename T1, typename T2>
 using Alias1 = typename BTT<T1, T2>::type;
 
 template<typename T1, typename T2>
-concept bool C()
-{
-  return requires() { typename Alias1<T1, T2>; };
-}
+concept C = requires() { typename Alias1<T1, T2>; }; // { dg-message "in requirements" }
 
 template<typename T1, typename T2>
-  requires C<T1, T2>()
+  requires C<T1, T2>
 int f();
 
 auto i = f<char, int>(); // { dg-error "cannot call function" }
similarity index 75%
rename from gcc/testsuite/g++.dg/concepts/pr66091.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-pr66091.C
index a71cd7b8bee77ad29901008e61bfaa5f52dabe1a..ea51e31817e2afee2bc83d1391bd18051da5fef6 100644 (file)
@@ -1,5 +1,6 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// PR c++/66091
+// { dg-do compile { target c++17_only } }
+// { dg-additional-options "-fconcepts" }
 
 template<typename T>
 concept bool C1()
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr66844.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr66844.C
new file mode 100644 (file)
index 0000000..64baab1
--- /dev/null
@@ -0,0 +1,16 @@
+// PR c++/66844
+// { dg-do compile { target c++2a } }
+
+template <class T, class U>
+concept Same = __is_same_as(T, U);
+
+template <class T>
+concept C = requires (T t) {   // { dg-error "invalid parameter|in requirements" }
+    requires Same<decltype(t),void>;
+  };
+
+template <typename T>
+  requires C<T>
+constexpr bool is_c() { return true; }
+
+static_assert(is_c<void>(), ""); // { dg-error "cannot call" }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr66962.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr66962.C
new file mode 100644 (file)
index 0000000..ca57a09
--- /dev/null
@@ -0,0 +1,80 @@
+// PR c++/66962
+// { dg-do compile { target c++2a } }
+
+template <typename> struct remove_cv;
+template <typename> struct is_reference;
+template <typename> void declval();
+template <typename> struct is_constructible;
+template <typename> struct is_nothrow_constructible;
+template <typename _Tp> using remove_cv_t = typename remove_cv<_Tp>::type;
+template <typename> struct Trans_NS_extension_apply_list;
+template <typename T> using _t = typename T::type;
+template <class> void ImplicitlyConvertibleTo();
+template <class> void Assignable();
+template <class T, class... Args> int ConstructibleObject = requires { T{}; };
+
+template <class T, class... Args>
+concept BindableReference = 
+  is_reference<T>::value && is_constructible<T>::value;
+
+template <class T, class... Args> 
+concept Constructible = 
+  ConstructibleObject<T> || BindableReference<T, Args...>;
+
+template <class T> 
+concept DefaultConstructible = 
+  Constructible<T> && requires { new T[0]; };
+
+template <class T> 
+concept MoveConstructible =
+  Constructible<T> && ImplicitlyConvertibleTo<T>;
+
+template <class T>
+concept Movable =
+  MoveConstructible<T> && Assignable<T &&>;
+
+template <class, class>
+int Swappable_ = requires { 0; };
+
+template <class T, class U> 
+int Swappable();
+
+template <class T> 
+concept Dereferencable = requires{{0};};
+
+template <Dereferencable R> 
+using RvalueReferenceType = decltype(0);
+
+template <class T> 
+int IsValueType;
+
+template <class>
+struct value_type;
+
+template <class T>
+  requires IsValueType<_t<value_type<remove_cv_t<T>>>>
+using ValueType = _t<value_type<remove_cv_t<T>>>;
+
+template <class I> 
+concept Readable =
+  Movable<I> && DefaultConstructible<I> && Dereferencable<const I> && requires{{0};};
+
+template <class Out, class T> 
+concept MoveWritable =
+  Movable<Out> && DefaultConstructible<Out> && Dereferencable<Out>;
+
+template <class In, class Out> 
+concept IndirectlyMovable =
+  Readable<In> && 
+  Movable<ValueType<In>> && 
+  Constructible<ValueType<In>> &&
+  MoveWritable<Out, RvalueReferenceType<In>> &&
+  MoveWritable<Out, ValueType<In>>;
+
+template<typename In, typename Out>
+  requires IndirectlyMovable<In, Out>
+int is_nothrow_indirectly_movable_v = is_nothrow_constructible<ValueType<In>>::value;
+
+template <Readable R1, Readable R2>
+  requires IndirectlyMovable<R1, R2> && IndirectlyMovable<R2, R1>
+void iter_swap2();
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67070.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67070.C
new file mode 100644 (file)
index 0000000..548cb40
--- /dev/null
@@ -0,0 +1,51 @@
+// { dg-do compile { target c++2a } }
+
+template <class T>
+concept C1 =
+  requires { typename T::type; } && T::type::value;
+
+template <class T>
+concept C2 =
+  requires {
+    typename T::Type;
+    requires T::Type::value;
+  };
+
+template <class T>
+  requires (!C1<T>)
+void f1() { }
+
+template <class T>
+  requires (!C2<T>)
+void f2() { }
+
+struct S { };
+
+void test()
+{
+  f1<S>();
+  f2<S>();
+}
+
+// ------------------
+
+
+template<typename T>
+concept C = requires (T t) { t.f(); };
+
+template<typename A, typename B>
+  requires (!(C<A> && C<B>))
+void g1() { }
+
+template<typename A, typename B>
+  requires (!C<A> || !C<B>)
+void g2() { }
+
+struct X {
+  void f();
+};
+
+void test2() {
+  g1<X, X>(); // { dg-error "cannot call" }
+  g2<X, X>(); // { dg-error "cannot call" }
+}
\ No newline at end of file
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67147.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67147.C
new file mode 100644 (file)
index 0000000..db8c37e
--- /dev/null
@@ -0,0 +1,25 @@
+// PR c++/67147
+// { dg-do compile { target c++2a } }
+
+template <class F, class I1, class I2 = I1>
+concept IndirectCallableRelation = true;
+
+template <class, class, class = void>
+constexpr bool indirect_relation() { return false; }
+
+// FIXME: The original bug was found using the introducer syntax.
+
+template<typename F, typename I1>
+  requires IndirectCallableRelation<F, I1>
+constexpr bool indirect_relation() { return true; }
+
+template<typename F, typename I1, typename I2>
+  requires IndirectCallableRelation<F, I1, I2>
+constexpr bool indirect_relation() { return true; }
+
+// This was added to the discussion thread as a minimum repro.
+
+template<typename T, int = sizeof(T)> 
+concept C1 = true;
+
+template <C1 T> int test();
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67148.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67148.C
new file mode 100644 (file)
index 0000000..c593996
--- /dev/null
@@ -0,0 +1,121 @@
+// PR c++/67148
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+namespace std
+{
+  template<typename T>
+  T declval();
+
+  typedef unsigned int size_t;
+  typedef int ptrdiff_t;
+  typedef decltype(nullptr) nullptr_t;
+  template<typename _Tp, _Tp... _Idx>
+    struct integer_sequence
+    {
+      typedef _Tp value_type;
+      static constexpr size_t size() { return sizeof...(_Idx); }
+    };
+
+  template <class T, T Value>
+  struct integral_constant {
+    using type = integral_constant;
+    using value_type = T;
+    constexpr operator T() const { return Value; }
+    constexpr T operator()() const { return Value; }
+    static constexpr T value {Value};
+  };
+  template <class T, T Value>
+  constexpr T integral_constant<T, Value>::value;
+  using true_type = integral_constant<bool, true>;
+  using false_type = integral_constant<bool, false>;
+
+  template <class T, class U>
+  struct is_same : false_type {};
+  template <class T>
+  struct is_same<T,T> : true_type {};
+}
+       
+namespace meta
+{
+    inline namespace v1
+    {
+        template <typename T>
+        using _t = typename T::type;
+        template <bool... Bools>
+        using and_c = std::is_same<std::integer_sequence<bool, Bools...>,
+                                   std::integer_sequence<bool, (Bools || true)...>>;
+    }
+}
+
+namespace stl2 { inline namespace v1 {
+using std::declval;
+namespace detail {
+template <class...>
+struct all_same : std::true_type {};
+template <class T, class...Rest>
+struct all_same<T, Rest...> :
+  meta::and_c<__is_same_as(T, Rest)...> {};
+}
+template <class...Ts>
+concept bool Same() {
+  return detail::all_same<Ts...>::value;
+}
+template <class F, class...Args>
+using ResultType = decltype(declval<F>()(declval<Args>()...));
+template <class>
+struct value_type {};
+template <class T>
+struct value_type<T*> {
+  using type = T;
+};
+template <class T>
+using ValueType =
+  typename value_type<T>::type;
+
+template <class F, class...Args>
+concept bool Function() {
+  return requires (F& f, Args&&...args) {
+    f((Args&&)args...);
+    requires Same<decltype(f((Args&&)args...)), ResultType<F, Args...> >();
+  };
+}
+
+template <class, class...> struct __function : std::false_type {};
+Function{F, ...Args} struct __function<F, Args...> : std::true_type {};
+
+template <class F, class I1, class I2>
+concept bool IndirectCallable() {
+  return Function<F, ValueType<I1>, ValueType<I2>>();
+}
+
+template <class F, class I1, class I2>
+concept bool IndirectCallable2() {
+  return __function<F, ValueType<I1>, ValueType<I2>>::value;
+}
+
+namespace ext { namespace models {
+template <class, class, class>
+constexpr bool indirect_callable() { return false; }
+IndirectCallable{F, I1, I2}
+constexpr bool indirect_callable() { return true; }
+
+template <class, class, class>
+constexpr bool indirect_callable2() { return false; }
+IndirectCallable2{F, I1, I2}
+constexpr bool indirect_callable2() { return true; }
+}}
+}}
+
+namespace models = stl2::ext::models;
+
+template <class T = void>
+struct plus {
+  T operator()(T, T) const;
+};
+
+static_assert((models::indirect_callable<::plus<int>, int*, int*>()));
+static_assert((models::indirect_callable2<::plus<int>, int*, int*>()));
+
+static_assert((models::indirect_callable<::plus<int>, int**, int*>())); // { dg-error "static assertion failed" }
+static_assert((models::indirect_callable2<::plus<int>, int**, int*>())); // { dg-error "static assertion failed" }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67178.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67178.C
new file mode 100644 (file)
index 0000000..f76f2e3
--- /dev/null
@@ -0,0 +1,26 @@
+// PR c++/67178
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+concept c = true;
+
+template<typename T>
+concept C0 = requires (auto x) { // { dg-error "placeholder type" }
+  x;
+};
+
+template<typename T>
+concept C1 = requires (C1 auto x) { // { dg-error "not been declared|placeholder|two or more|in requirements" }
+  x; // { dg-error "not declared" }
+  { x } -> c; // { dg-error "not declared|does not satisfy" }
+};
+
+template<typename T>
+  requires C1<T>
+void f(T) {}
+
+int main() {
+  f(1); // { dg-error "cannot call" }
+}
+
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67210.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67210.C
new file mode 100644 (file)
index 0000000..044e677
--- /dev/null
@@ -0,0 +1,10 @@
+// PR c++/67210
+// { dg-do compile { target c++2a } }
+
+template <class T, class U>
+concept C = true;
+
+template <class T>
+struct A {};
+
+void f(A<C<int> auto >) {} 
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67217.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67217.C
new file mode 100644 (file)
index 0000000..35618ae
--- /dev/null
@@ -0,0 +1,8 @@
+// PR c++/67217
+// { dg-do compile { target c++2a } }
+
+template <class T>
+  requires __is_same_as(T, double)
+union A {};
+
+int main() { A<int>{}; } // { dg-error "template constraint failure" }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-1.C
new file mode 100644 (file)
index 0000000..1c5d73a
--- /dev/null
@@ -0,0 +1,32 @@
+// PR c++/67225
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template <class T, class U> 
+concept bool Same() 
+{
+    return true;
+}
+
+template <class T> struct WrapT {T t;};
+
+template <class T>
+concept bool Destructible()
+{
+    return requires(T t, const T ct, WrapT<T>& wt) // { dg-message "in requirements" }
+    {
+        {wt.~WrapT()} noexcept;
+        // {&t} -> Same<T*>; // #1
+        //{&t} -> T*; // #2
+    };
+}
+
+template <Destructible T>
+void f() {}
+
+struct Y {private: ~Y();};
+
+int main()
+{
+    f<Y>(); // { dg-error "cannot call" }
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-2.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-2.C
new file mode 100644 (file)
index 0000000..6218176
--- /dev/null
@@ -0,0 +1,36 @@
+// PR c++/67225
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<typename Target>
+// template<typename Target, typename... Ts>
+concept bool has_resize ()
+{
+  return requires (Target tgt)
+  {
+    { tgt.resize () };
+  };
+}
+
+template<typename Target>
+void resize (Target tgt)
+{
+  if constexpr (has_resize<Target> ())
+  {
+    tgt.resize ();
+  }
+}
+
+class MyClass
+{
+  private:
+    int foo (int i)
+    {
+      return i * 2;
+    }
+};
+
+int main ()
+{
+  return MyClass {}.foo (7); // { dg-error "private within this context" }
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-3.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-3.C
new file mode 100644 (file)
index 0000000..d08efb6
--- /dev/null
@@ -0,0 +1,21 @@
+// PR c++/67225
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template <class>
+concept bool Dummy = true;
+
+template <typename>
+class example {
+    template <Dummy U>
+    friend auto func();
+};
+
+class test {
+    test() = default;
+};
+
+int main()
+{
+    test t; // { dg-error "private within this context" }
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-4.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-4.C
new file mode 100644 (file)
index 0000000..0ef5ed8
--- /dev/null
@@ -0,0 +1,14 @@
+// PR c++/67225
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template <class, class>
+concept bool C1 = true;
+
+template <class>
+concept bool C2 = requires { { 42 } -> C1<int>; };
+
+int main() {
+    class A { int x; } a;
+    a.x = 42; // { dg-error "private within this context" }
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-5.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67225-5.C
new file mode 100644 (file)
index 0000000..9a25831
--- /dev/null
@@ -0,0 +1,17 @@
+// PR c++/67225
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<typename A, typename T>
+concept bool SomeConcept = true;
+
+template <typename T>
+void breaker(SomeConcept<int>);
+
+class SomeClass {
+    int privateMember;
+};
+
+int main() {
+    return SomeClass().privateMember; // { dg-error "private within this context" }
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67319.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67319.C
new file mode 100644 (file)
index 0000000..6eb1c44
--- /dev/null
@@ -0,0 +1,24 @@
+// PR c++/67319
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template <typename T>
+concept bool Any()
+{
+  return requires (T t) { +t; };
+}
+
+struct my_struct
+{
+  template <Any... Args>
+  auto sample(Args... args) -> void;
+};
+
+int main()
+{
+  my_struct{}.sample();
+  my_struct{}.sample(0);
+  my_struct{}.sample(0, 'a');
+  my_struct{}.sample(nullptr); // { dg-error "" }
+}
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67427.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67427.C
new file mode 100644 (file)
index 0000000..fcad301
--- /dev/null
@@ -0,0 +1,22 @@
+// PR c++/67427
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template <class S, class I>
+concept bool Sentinel =
+  requires (I i) { i; };
+
+template <class I, class S>
+concept bool SizedIteratorRange =
+  Sentinel<S, I> && true;
+
+Sentinel{S, I}
+void distance(I first, S last) {}
+
+template <class I, class S>
+  requires SizedIteratorRange<I, S>
+void distance(I first, S last) {}
+
+int main() {
+  distance(42, 43); // { dg-error "ambiguous" }
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67654.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67654.C
new file mode 100644 (file)
index 0000000..27ee205
--- /dev/null
@@ -0,0 +1,30 @@
+// PR c++/67427
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template <bool... Values> struct and_c_impl {
+  static constexpr bool value = true;
+};
+
+template <bool ValueFirst, bool... ValuesRest>
+struct and_c_impl<ValueFirst, ValuesRest...> {
+  static constexpr bool value = ValueFirst && and_c_impl<ValuesRest...>::value;
+};
+
+template <bool... Values> constexpr bool and_c() {
+  return and_c_impl<Values...>::value;
+}
+
+template<class T> concept bool C() {
+  return true;
+}
+
+template<class... Tx>
+struct A {
+  A() requires and_c<C<Tx>()...>() = default;
+};
+
+int main() {
+  A<int, double> a;
+  return 0;
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67658.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67658.C
new file mode 100644 (file)
index 0000000..087f4fc
--- /dev/null
@@ -0,0 +1,14 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<class T> concept bool C1() { return false; }
+template<C1 T> concept bool C2() { return true; } // { dg-error "cannot be constrained" }
+
+void f(C2 x) {
+}
+
+struct A {} a;
+
+int main() {
+  f(a);
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67684.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67684.C
new file mode 100644 (file)
index 0000000..35e9295
--- /dev/null
@@ -0,0 +1,63 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<class T>
+class A {
+ public:
+  template<int I, class S>
+    requires I > 0
+  friend int f1(const A<S>&);
+
+  template<int I, class S>
+  friend int f2(const A<S>&) requires I > 0;
+
+ private:
+  int x = 2;
+};
+
+template<int I, class S>
+  requires I > 0
+int f1(const A<S>& a)  { 
+  return a.x;
+} 
+
+template<int I, class S>
+int f2(const A<S>& a) requires I > 0 { 
+  return a.x;
+} 
+
+class B {
+ public:
+  template<int I>
+    requires I > 0
+  friend int f3(const B&);
+
+  template<int I>
+  friend int f4(const B&) requires I > 0;
+
+ private:
+  int x = 2;
+};
+
+template<int I>
+  requires I > 0
+int f3(const B& a) {
+  return a.x;
+}
+
+template<int I>
+int f4(const B& a) requires I > 0 {
+  return a.x;
+}
+
+int main() { 
+  A<double> a;
+  f1<2>(a);
+  f2<2>(a);
+
+  B b;
+  f3<2>(b);
+  f4<2>(b);
+
+  return 0;
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67685.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67685.C
new file mode 100644 (file)
index 0000000..82f068e
--- /dev/null
@@ -0,0 +1,7 @@
+// PR c++/67685
+// { dg-do compile { target c++14 } }
+// { dg-additional-options "-fconcepts" }
+
+void f(auto i) {requires {i;};}
+
+int main() {f(0);}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67692.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67692.C
new file mode 100644 (file)
index 0000000..139ecd6
--- /dev/null
@@ -0,0 +1,13 @@
+// { dg-do compile { target c++2a } }
+
+template<class T>
+bool f(T x) {
+  return requires(T x) {
+    ++x;
+  };
+}
+
+int main() {
+  f(3);
+  return 0;
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67697.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67697.C
new file mode 100644 (file)
index 0000000..44c077b
--- /dev/null
@@ -0,0 +1,15 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<class X>
+concept bool C() {
+  return requires(X x, bool b) {
+    requires b; // { dg-error "not a constant expression" }
+    x++;
+  };
+}
+
+int main() {
+  C<int>();
+  return 0;
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67719.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67719.C
new file mode 100644 (file)
index 0000000..d28b445
--- /dev/null
@@ -0,0 +1,15 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<class X> concept bool C() {
+  return __is_same_as(X, int) || __is_same_as(X, long);
+}
+
+template<C... Tx>
+struct Ax {};
+
+int main() {
+  Ax<int, long> a;
+  Ax<int, long, void> b; // { dg-error "template constraint failure" }
+  return 0;
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67774.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67774.C
new file mode 100644 (file)
index 0000000..d363c59
--- /dev/null
@@ -0,0 +1,26 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+#include <type_traits>
+#include <utility>
+#include <iostream>
+
+template <class X> concept bool cpt_RealScalar() {
+  return std::is_floating_point<X>::value;
+}
+
+namespace detail {
+template <class, class> constexpr bool k_evaluator_impl = false;
+
+template <std::size_t... Indexes, class E>
+constexpr bool k_evaluator_impl<std::index_sequence<Indexes...>, E> = true;
+}
+
+template <class X, std::size_t K> concept bool cpt_KEvaluator =
+  detail::k_evaluator_impl<std::make_index_sequence<K>, X>;
+
+int main() {
+  auto f = [](int, int, int) -> double { return 3; };
+  std::cout << cpt_KEvaluator<decltype(f)> << '\n'; // { dg-error "wrong number of template arguments" }
+  return 0;
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67825.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67825.C
new file mode 100644 (file)
index 0000000..95698e9
--- /dev/null
@@ -0,0 +1,20 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+struct A {
+  template <class T>
+  double operator()(T x) const {
+    return 0;
+  }
+};
+
+template <class X> concept bool C() {
+  return requires {
+    &X::operator();
+  };
+}
+
+int main() {
+  static_assert(C<A>());
+  return 0;
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67860.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67860.C
new file mode 100644 (file)
index 0000000..8bad031
--- /dev/null
@@ -0,0 +1,61 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+#include <type_traits>
+
+inline constexpr bool and_impl() { return true; }
+
+template <class OperandFirst, class... OperandsRest>
+constexpr bool and_impl(OperandFirst operand_first,
+                        OperandsRest... operands_rest) {
+  return operand_first && and_impl(operands_rest...);
+}
+
+template <class... Operands> constexpr bool and_(Operands... operands) {
+  return and_impl(operands...);
+}
+
+template <class X> concept bool C() { return true; }
+
+// v1
+template<int, class... Xs>
+  requires and_(C<Xs>()...)
+constexpr int f(const Xs&... xs) {
+  return 0;
+}
+
+// v2
+template<int, class... Xs>
+constexpr int f(const Xs&... xs) {
+  return 1;
+}
+
+int main() {
+  static_assert(f<10>(3.0, 2.0f) == 0);
+  return 0;
+}
+
+// 2nd example
+
+template <typename T, typename... Us>
+concept bool AreType() {
+  return (std::is_same<T,Us>::value && ...);
+  // return true; gives the same overloaded error
+}
+
+// Function with constraint
+template<typename T, AreType<T>... Us>
+constexpr bool isValid(Us... us) {
+  return true;
+}
+
+// Function with no constraint
+template<typename T, typename... U>
+constexpr bool isValid(U... u) {
+  return false;
+}
+
+int main2() {
+  static_assert(isValid<int>(1)); // also isValid<int>(1, 2, 3); etc
+  return 0;
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67862.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67862.C
new file mode 100644 (file)
index 0000000..222e528
--- /dev/null
@@ -0,0 +1,162 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+typedef int size_t;
+template <typename _Tp> struct A { static constexpr _Tp value = 1; };
+template <typename _Tp> _Tp declval();
+template <typename _From, typename _To> struct __is_convertible_helper {
+  template <typename, typename> static A<bool> __test(int);
+  typedef decltype(__test<_From, _To>(0)) type;
+};
+template <typename, typename>
+struct is_convertible : __is_convertible_helper<int, int>::type {};
+template <typename> struct remove_reference;
+template <typename _Tp> struct remove_reference<_Tp &> { typedef _Tp type; };
+struct base;
+struct general;
+template <typename _Tp, _Tp...> struct B;
+template <typename _Tp, _Tp> using make_integer_sequence = B<int>;
+template <size_t... _Idx> using index_sequence = B<size_t, _Idx...>;
+template <size_t _Num>
+using make_index_sequence = make_integer_sequence<size_t, _Num>;
+template <bool...> struct and_c_impl { static constexpr bool value = true; };
+template <bool...> constexpr bool and_c() { return and_c_impl<>::value; }
+
+template <class X, class Y> concept bool cpt_Convertible() {
+  return is_convertible<X, Y>::value;
+}
+
+template <class T> using uncvref_t = typename remove_reference<T>::type;
+struct Plus;
+using index_t = int;
+template <class> bool cpt_Index;
+template <class... Extents>
+requires and_c<cpt_Index<Extents>()...>() class Dimensionality;
+namespace detail_concept {
+template <class> bool match_dimensionality;
+template <class... Extents>
+constexpr bool match_dimensionality<Dimensionality<Extents...>> = true;
+}
+template <class X> concept bool cpt_Dimensionality() {
+  return detail_concept::match_dimensionality<X>;
+}
+
+template <class X> concept bool cpt_Shaped() { return requires(X x){{x};}; }
+
+template <class X> concept bool cpt_Dimensioned() { return cpt_Shaped<X>(); }
+
+template <class... Extents>
+requires and_c<cpt_Index<Extents>()...>() class Dimensionality {
+public:
+  static constexpr size_t num_dimensions = sizeof...(Extents);
+};
+template <index_t...> using DimensionalityC = Dimensionality<>;
+template <class> struct dimensionality_type_impl;
+template <cpt_Dimensioned X> struct dimensionality_type_impl<X> {
+  using type = uncvref_t<decltype(declval<X>().dimensionality())>;
+};
+template <cpt_Dimensioned X>
+using dimensionality_type = typename dimensionality_type_impl<X>::type;
+template <class Functor, class... Expressibles>
+requires requires(Functor functor, Expressibles... expressibles) {
+  map_expressions_impl(functor, expressibles...);
+}
+
+decltype(auto) map_impl(Functor, Expressibles...);
+void cpt_ContinualScalar();
+template <class> concept bool cpt_Scalar() { return cpt_ContinualScalar; }
+
+template <class X> concept bool cpt_FlatEvaluator() {
+  return requires(X x){{x}->cpt_Scalar;};
+}
+
+template <class, class> bool k_evaluator_impl;
+template <size_t... Indexes, class Evaluator>
+constexpr bool k_evaluator_impl<index_sequence<Indexes...>, Evaluator> = true;
+template <class X, size_t K> concept bool cpt_KEvaluator() {
+  return k_evaluator_impl<make_index_sequence<K>, X>;
+}
+
+template <class X, size_t K> concept bool cpt_KCompatibleEvaluator() {
+  return cpt_KEvaluator<X, K>();
+}
+
+template <class X> concept bool cpt_Structure() {
+  return cpt_Convertible<X, base>();
+}
+
+template <cpt_Dimensionality Dimensionality, cpt_Structure,
+          cpt_KCompatibleEvaluator<Dimensionality::num_dimensions> Evaluator>
+class NumericArrayExpression;
+namespace detail_concept {
+
+template <class> bool match_numeric_array_expression;
+
+template <cpt_Dimensionality Dimensionality,
+          cpt_Structure Structure,
+          cpt_KCompatibleEvaluator<Dimensionality::num_dimensions> Evaluator>
+constexpr bool match_numeric_array_expression<
+    NumericArrayExpression<Dimensionality, Structure, Evaluator>> = true;
+
+}
+template <class X> concept bool cpt_NumericArrayExpression() {
+  return detail_concept::match_numeric_array_expression<X>;
+}
+
+namespace expression_traits {
+namespace detail_expression_traits {
+template <class...> struct first_numeric_array_expression_impl;
+template <cpt_NumericArrayExpression ExpressionFirst, class... ExpressionsRest>
+struct first_numeric_array_expression_impl<ExpressionFirst,
+                                           ExpressionsRest...> {
+  using type = ExpressionFirst;
+};
+}
+template <class... Expressions>
+using first_numeric_array_expression =
+    typename detail_expression_traits::first_numeric_array_expression_impl<
+        Expressions...>::type;
+template <class... Expressions>
+using first_expression_dimensionality =
+    dimensionality_type<first_numeric_array_expression<Expressions...>>;
+}
+template <cpt_Dimensionality Dimensionality, cpt_Structure,
+          cpt_KCompatibleEvaluator<Dimensionality::num_dimensions> Evaluator>
+class NumericArrayExpression {
+public:
+  NumericArrayExpression(Dimensionality, Evaluator) {}
+  Dimensionality &dimensionality();
+};
+
+template <cpt_Structure Structure, cpt_Dimensionality Dimensionality,
+          cpt_KCompatibleEvaluator<Dimensionality::num_dimensions> Evaluator>
+auto make_numeric_array_expression(Dimensionality dimensionality,
+                                   Evaluator evaluator) {
+  return NumericArrayExpression<Dimensionality, Structure, Evaluator>(
+      dimensionality, evaluator);
+}
+
+template <size_t, class Functor, class... Evaluators>
+auto make_map_evaluator_impl(Functor) requires
+    and_(cpt_FlatEvaluator<Evaluators>()...);
+template <class Functor, class... Expressions>
+requires
+requires(Expressions... expressions,
+         expression_traits::first_expression_dimensionality<Expressions...>
+             dimensionality) {
+  make_map_evaluator_impl<decltype(dimensionality)::num_dimensions>(
+      expressions...);
+}
+
+decltype(auto) map_expressions_impl(Functor, Expressions...);
+template <class Functor, class... Expressibles> concept bool cpt_Mappable() {
+  return requires(Functor functor, Expressibles... expressibles) {
+    map_impl(functor, expressibles...);
+  };
+}
+
+void ____C_A_T_C_H____T_E_S_T____8() {
+  auto e1 = make_numeric_array_expression<general>(DimensionalityC<>(), [] {});
+  using E1 = decltype(e1);
+  cpt_Mappable<Plus, E1>();
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr67969.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr67969.C
new file mode 100644 (file)
index 0000000..4f2ab51
--- /dev/null
@@ -0,0 +1,32 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+template <class, class>
+class NumericArray {};
+
+template <class>
+constexpr bool match_numeric_array = false;
+template <class Scalar, class Shape>
+constexpr bool
+    match_numeric_array<NumericArray<Scalar, Shape>> =
+        true;
+template <class T>
+concept bool cpt_NumericArrayContainer() {
+  return match_numeric_array<T>;
+}
+
+template <class X>
+concept bool cpt_NumericArray() {
+  return requires{requires cpt_NumericArrayContainer<X>();};
+}
+
+
+template <class X>
+requires !cpt_NumericArray<X>() auto func(int, X) {}
+
+template <class X>
+requires cpt_NumericArray<X>() auto func(int, X) {}
+
+int main() {
+  NumericArray<double, int> v5;
+  func(0, v5);
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr68093-1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr68093-1.C
new file mode 100644 (file)
index 0000000..bc4ddee
--- /dev/null
@@ -0,0 +1,13 @@
+// { dg-do compile { target c++2a } }
+template <typename t>
+struct S
+{
+  template <typename t2>
+    requires false
+  friend void foobar(S, t2) {}
+};
+
+int main()
+{
+  foobar(S<double>{}, int{}); // { dg-error "cannot call" }
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr68093-2.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr68093-2.C
new file mode 100644 (file)
index 0000000..91e3c80
--- /dev/null
@@ -0,0 +1,14 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+template <class T>
+concept bool True = true;
+
+template <class T>
+struct S {
+  friend bool operator==(S, int) requires True<T> { return true; }
+  friend bool operator==(S, int) requires !True<T> { return true; }
+};
+
+int main() {
+  S<int> s;
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr68372.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr68372.C
new file mode 100644 (file)
index 0000000..d1416eb
--- /dev/null
@@ -0,0 +1,48 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<typename F>
+concept bool FCallable()
+{
+  return requires(F)
+  {
+      F::f();
+  };
+}
+
+class Test1
+{
+public:
+  template<FCallable P, FCallable... Pp>
+  static void g()
+  {
+    (Pp::f(), ...);
+  }
+};
+
+class A
+{
+public:
+  static void f() {}
+};
+
+template<typename X> concept bool C = true;
+
+template<C... X>
+void bar(X...)
+{}
+
+struct foo
+{
+  template<C... X>
+  void bar(X...)
+  {}
+};
+
+int main()
+{
+  Test1::template g<A>();
+  bar();
+  foo {}.bar();
+}
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr68434.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr68434.C
new file mode 100644 (file)
index 0000000..23ecf4d
--- /dev/null
@@ -0,0 +1,18 @@
+// { dg-do compile { target c++2a } }
+
+template <class>
+concept C1 = true;
+
+template <class>
+concept C2 = true;
+
+template <class Expr>
+concept C3 =
+  requires (Expr expr) {
+      {expr}->C1;
+      {expr}->C2;
+  };
+
+template<C3 T>
+auto f (T);
+
similarity index 66%
rename from gcc/testsuite/g++.dg/concepts/pr68683.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-pr68683.C
index ff7709e40aaac911468daddd308450ede5626331..f11d69f75fd203ef56b10f8559ae08b9273c0d1a 100644 (file)
@@ -1,5 +1,5 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// PR c++/68683
+// { dg-do compile { target c++2a } }
 
 template <typename, typename>
 struct is_same {
@@ -7,10 +7,10 @@ struct is_same {
 };
 
 template <typename T, typename U>
-concept bool Same = is_same<T, U>::value;
+concept Same = is_same<T, U>::value;
 
 template <typename T>
-concept bool Integral = requires {
+concept Integral = requires {
   { T () } -> Same<typename T::value_type>;
 };
 
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr68812.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr68812.C
new file mode 100644 (file)
index 0000000..2d809e8
--- /dev/null
@@ -0,0 +1,34 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+namespace zero
+{
+  template<int... s>
+  struct S
+  {
+    template<int... f>
+      requires(... and (s == f))
+    static void F()
+    {
+    }
+  };
+
+  void foo(S<>) {}
+}
+
+namespace one
+{
+  template<typename X, typename Y> concept bool Foo = true;
+
+  template<typename... T>
+  struct foo
+  {
+    template<typename... U>
+    foo(U...)
+      requires (Foo<T, U> && ...)
+    {}
+  };
+
+  void bar(foo<int, long, double>) {}
+}
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr69235.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr69235.C
new file mode 100644 (file)
index 0000000..5e96d0e
--- /dev/null
@@ -0,0 +1,48 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<typename T>
+concept bool Boolean()
+{
+  return requires(T t)
+  {
+    { t } -> bool;
+  };
+}
+
+template<typename T>
+concept bool C()
+{
+  return requires (T t)
+  {
+    { t } -> Boolean;
+  };
+}
+
+template<typename T>
+struct X;
+
+template<typename T>
+  requires ! C<typename T::type>()
+struct X<T>
+{
+  using type = int;
+};
+
+template<typename T>
+  requires C<typename T::type>()
+struct X<T>
+{
+  using type = int;
+};
+
+struct S
+{
+  using type = char;
+};
+
+void f()
+{
+  X<S>::type x;
+}
+
similarity index 61%
rename from gcc/testsuite/g++.dg/concepts/pr71368.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-pr71368.C
index 5cd2b54cd7cd5ce7bf3f1fa7d4cec6877fb69155..c67e632d4438ab5ae11dc32b041eb672fe326990 100644 (file)
@@ -1,14 +1,16 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// PR c++/71368
+// { dg-do compile { target c++2a } }
+
+template <class T, class U> concept Same = __is_same_as(T,U);
 
 struct inner;
 
-template<typename X> concept bool CompoundReq = requires {
+template<typename X> concept CompoundReq = requires {
     // fine with concrete type in trailing type, i.e. inner& instead of X&
-    { X::inner_member() } -> X&;
+    { X::inner_member() } -> Same<X&>;
 };
 
-template<typename X> concept bool Concept = requires {
+template<typename X> concept Concept = requires {
     { X::outer_member() } -> CompoundReq;
 };
 
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr71385.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr71385.C
new file mode 100644 (file)
index 0000000..f31997c
--- /dev/null
@@ -0,0 +1,14 @@
+// { dg-do compile { target c++2a } }
+
+template<class From, class To>
+concept convertible_to = requires(From (&f)(), void (&g)(To)) { g(f()); };
+
+template<class T>
+concept Addable =
+ requires(T x){
+  {x + x} -> convertible_to<T>;
+ };
+
+int main(){
+ Addable auto t = 0;
+}
similarity index 77%
rename from gcc/testsuite/g++.dg/concepts/pr71965.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-pr71965.C
index 6bfaef19211fa0d76cc2ab3b930836e70ecc70e3..4619cf993afd5e3bd619daa27b295ea3204d3db3 100644 (file)
@@ -1,13 +1,10 @@
-// { dg-do compile { target c++14 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template <class T>
-concept bool Destructible() {
-    return false;
-}
+concept Destructible = false;
 
 template <class T, class...Args>
-concept bool ConstructibleObject =
+concept ConstructibleObject =
     // Concept evaluation should short-circuit even the template
     // substitution, so we shouldn't even substitute into the requires
     // constraint and the unimplemented multi-dimensional new T{...}
@@ -15,7 +12,7 @@ concept bool ConstructibleObject =
     // sorry() message we used to for such constructs when asked not
     // to issue errors, this shouldn't be a problem for this and
     // similar cases.
-    Destructible<T>() && requires (Args&&...args) {
+    Destructible<T> && requires (Args&&...args) {
         new T{ (Args&&)args... };
     };
 
similarity index 94%
rename from gcc/testsuite/g++.dg/concepts/memfun2.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-pr72415.C
index 78a2cf7badaefb63e643c19067e80a84868d1a63..04bdc514fb9fbc1e26600dafd7833589ee10cc73 100644 (file)
@@ -2,7 +2,7 @@
 // { dg-do compile { target c++17 } }
 // { dg-options "-fconcepts" }
 
-template<int... Indices>
+template<int... Xs>
 struct indices {};
 
 template<typename Dummy>
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr78752.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr78752.C
new file mode 100644 (file)
index 0000000..40eeaa7
--- /dev/null
@@ -0,0 +1,18 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+#include <type_traits>
+
+template <class T, class U>
+concept bool Same = std::is_same<T, U>::value;
+
+struct test {
+  template <Same<int>... Ints>
+  void func(Ints... ints) {}
+};
+
+int main()
+{
+  test t;
+  t.func(1, 2, 3);
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr79759.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr79759.C
new file mode 100644 (file)
index 0000000..a99103c
--- /dev/null
@@ -0,0 +1,7 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<typename T, T N>
+concept bool C0() { return true; }
+
+void f(C0<0>);
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr80746.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr80746.C
new file mode 100644 (file)
index 0000000..69e2fbe
--- /dev/null
@@ -0,0 +1,14 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<typename T, typename T::type>
+concept bool C = true;
+
+template<C<0> T> class ct {};
+
+struct S
+{
+  using type = int;
+};
+
+template class ct<S>;
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr80773.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr80773.C
new file mode 100644 (file)
index 0000000..34b96c5
--- /dev/null
@@ -0,0 +1,33 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<typename F>
+concept bool FCallable()
+{
+  return requires(F)
+  {
+    F::f();
+  };
+}
+
+class Test1
+{
+public:
+  template<FCallable P, FCallable... Pp>
+  static void g()
+  {
+    (Pp::f(), ...);
+  }
+};
+
+class A
+{
+public:
+  static void f() {}
+};
+
+int main()
+{
+  Test1::template g<A>();
+}
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr82507.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr82507.C
new file mode 100644 (file)
index 0000000..f743855
--- /dev/null
@@ -0,0 +1,17 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<class I>
+concept bool True = true;
+
+template<class T>
+concept bool HasType = requires { typename T::type; };
+
+template<class T>
+struct S
+{
+  void foo() requires HasType<T> && True<typename T::type>;
+};
+
+S<int> s;
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr82740.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr82740.C
new file mode 100644 (file)
index 0000000..6e0f062
--- /dev/null
@@ -0,0 +1,32 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<class T>
+concept bool C = requires(const T& t) { t.foo(); };
+
+template<class T>
+struct Base
+{
+  constexpr T const& derived() const
+  {
+    return static_cast<T const&>(*this);
+  }
+  constexpr bool bar() const
+    requires requires(const T& t) { t.foo(); }
+  {
+    derived().foo();
+    return true;
+  }
+};
+
+template<class T>
+struct Derived : Base<Derived<T>>
+{
+  constexpr void foo() const {}
+};
+
+int main()
+{
+  static_assert(Derived<int>{}.bar());
+}
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr84140.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr84140.C
new file mode 100644 (file)
index 0000000..d901ab2
--- /dev/null
@@ -0,0 +1,38 @@
+// { dg-do run { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<class, class> constexpr bool is_same_v = false;
+template<class T> constexpr bool is_same_v<T, T> = true;
+
+template<class T, class U>
+concept bool Same = is_same_v<T, U>;
+
+template<class T, class U>
+concept bool Diff = requires(T& t, U& u) { u - t; };
+
+template<class I, class S>
+int distance(I, S) { return 0; }
+
+template<class I, Diff<I> S>
+int distance(I first, S last)
+{
+  return last - first;
+}
+
+template<class T>
+struct I
+{
+  template<class U>
+    requires Same<T, U>
+  friend int operator-(I const&, I<U> const&)
+  {
+    static_assert(Same<T, U>);
+    return 42;
+  }
+};
+
+int main()
+{
+  return distance(I<int>{}, I<void>{});
+}
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr84551.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr84551.C
new file mode 100644 (file)
index 0000000..e40796f
--- /dev/null
@@ -0,0 +1,11 @@
+// PR c++/84551
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-g -O" }
+
+template<typename> concept C = true;
+
+template<template<typename T> requires C<T> class TT> struct A {};
+
+template<typename U> requires true struct B {};
+
+A<B> a;                                // { dg-error "" }
similarity index 65%
rename from gcc/testsuite/g++.dg/concepts/pr84980.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-pr84980.C
index 619c0bd140bdaa5e9c41f3f8faa7802c7d82b858..1703de05f07a9aa18cbade41f671127664b47cd8 100644 (file)
@@ -1,6 +1,4 @@
-// { dg-do compile { target c++14 } }
+// { dg-do compile { target c++17_only } }
 // { dg-additional-options "-fconcepts" }
 
 template<T> concept bool C = true;  // { dg-error "has not been declared" }
-
-template<C...> struct A;
similarity index 70%
rename from gcc/testsuite/g++.dg/concepts/pr85265.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-pr85265.C
index 86124ceb712ac21d54880b660b7e835fa6585eaf..96aac69ed6a0f9de9f043dbc8aa1ec52036b2ed2 100644 (file)
@@ -1,4 +1,5 @@
-// { dg-do compile { target c++14 } }
+// PR c++/85265
+// { dg-do compile { target c++17_only } }
 // { dg-additional-options "-fconcepts" }
 
 template<typename> concept bool C = true;
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr85808.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr85808.C
new file mode 100644 (file)
index 0000000..bcba830
--- /dev/null
@@ -0,0 +1,19 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+namespace X
+{
+  template<class> constexpr bool x = true;
+}
+
+template<int> using helper = void;
+
+template<typename T>
+concept bool C =
+  requires
+  {
+    requires X::x<T>;
+    typename helper<T{}>;
+  };
+
+static_assert(C<int>);
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr86269.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr86269.C
new file mode 100644 (file)
index 0000000..4428ee1
--- /dev/null
@@ -0,0 +1,19 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+#include <type_traits>
+
+template<typename t2, typename t = std::remove_reference_t<t2>>
+concept bool IntegralOrIntegralRef = std::is_integral_v<t>;
+
+template<IntegralOrIntegralRef t>
+auto foo(t && v)
+{
+  return v;
+}
+
+int main()
+{
+  int i = 7;
+  foo(8);
+  return foo(i);
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-pr87441.C b/gcc/testsuite/g++.dg/cpp2a/concepts-pr87441.C
new file mode 100644 (file)
index 0000000..bcd4ba5
--- /dev/null
@@ -0,0 +1,10 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<typename X, typename Y = X>
+concept bool HasBinaryAdd = requires(X x, Y y)
+{
+  {x + y} -> decltype(x + y);
+};
+
+void proc(HasBinaryAdd x, HasBinaryAdd y);
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires1.C
new file mode 100644 (file)
index 0000000..f1603af
--- /dev/null
@@ -0,0 +1,71 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+concept Class = __is_class(T);
+
+// Allow a requires-expression with no parms.
+template<typename T>
+concept C = requires { typename T::type; };
+
+void f1(int a) requires true;         // OK
+auto f2(int a) -> bool requires true; // OK
+auto f3(int a) requires true -> bool; // { dg-error "" } requires-clause precedes trailing-return-type
+typedef void fn_t() requires true;    // { dg-error "typedef" }
+void (*pf)() requires true;           // { dg-error "non-function" }
+void (*fn(int))() requires false;     // { dg-error "return type" }
+void g(int (*)() requires true);      // { dg-error "parameter|non-function" }
+auto* p = new (void(*)(char) requires true); // { dg-error "type-id" }
+void f4(auto a) requires Class<decltype(a)> { }
+void f5(auto a) requires requires (decltype(a) x) { -x; } { } // { dg-message "in requirements" }
+
+struct Test {
+  void f(auto a) requires Class<decltype(a)>;
+} test;
+
+void driver_1() {
+  struct S { } s;
+  f4(s);
+  f5(0);
+  f5((void*)0); // { dg-error "cannot call" }
+  test.f(s);
+}
+
+void Test::f(auto a) requires Class<decltype(a)> { }
+
+template<bool B> requires B struct S0; // OK
+
+template<int N> requires N struct S1 { }; // { dg-error "does not have type" }
+S1<1> x0; // { dg-error "template constraint failure|does not have type" }
+
+template<int N> requires N == 0 struct S2 { }; // { dg-error "does not have type|must be enclosed" }
+
+template<int N> requires (N == 0) struct S3 { }; // OK
+
+template<typename T, T X> requires X struct S4 { }; // OK
+S4<int, 0> x1;      // { dg-error "template constraint failure|does not have type" }
+S4<bool, true> x2; // OK
+S4<bool, false> x3; // { dg-error "template constraint failure" }
+
+
+// req11.C
+template<typename T>
+concept Streamable = requires (T t) { t; };
+
+template<typename T>
+concept Range = requires (T t) { t; };
+
+// FIXME: There are two syntax errors here when there should be
+// just one.Note that !Range<T> is not a primary-expression and needs to
+// be wrapped in parens to be syntactically valid.
+template<class T>
+  requires Streamable<T> && !Range<T> // { dg-error "must be enclosed" }
+void print1(const T& x) { }
+
+template<class T>
+  requires Streamable<T> && (!Range<T>)
+void print2(const T& x) { }
+
+void driver_3()
+{
+  print2("hello"); // { dg-error "cannot call" }
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires10.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires10.C
new file mode 100644 (file)
index 0000000..bb0e3b0
--- /dev/null
@@ -0,0 +1,32 @@
+// PR c++/66988
+// { dg-do compile { target c++2a } }
+
+template<bool B>
+struct bool_constant {
+  static constexpr bool value = B;
+  constexpr operator bool() const { return value; }
+};
+
+using true_type = bool_constant<true>;
+using false_type = bool_constant<false>;
+
+template <template <class> class T, class U>
+concept _Valid = requires { typename T<U>; };
+
+template <class T>
+using nested_type = typename T::type;
+
+template <class T>
+struct has_nested_type : false_type { };
+
+template <class T>
+  requires _Valid<nested_type, T>
+struct has_nested_type<T> : true_type { };
+
+struct Nested
+{
+  using type = int;
+};
+
+static_assert(!has_nested_type<int>());
+static_assert(has_nested_type<Nested>());
similarity index 56%
rename from gcc/testsuite/g++.dg/concepts/req17.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-requires11.C
index 472cfef34a0a8882a8984233d472ff1e2c13d8a8..be6409ebad817ac59812c41f52c33a22c3c42034 100644 (file)
@@ -1,20 +1,20 @@
 // PR c++/67018
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template <typename T>
 constexpr bool Val = true;
 
 template <class I>
-concept bool InputIterator = requires (I i) {
-  requires Val <decltype(i++)>;
+concept InputIterator = requires (I i) {
+  requires Val<decltype(i++)>;
 };
 
 template <class I>
-concept bool ForwardIterator = InputIterator<I> && true;
+concept ForwardIterator = InputIterator<I> && true;
 
 template<InputIterator>
 constexpr bool f() { return false; }
+
 template<ForwardIterator>
 constexpr bool f() { return true; }
 
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires12.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires12.C
new file mode 100644 (file)
index 0000000..c8e3cfd
--- /dev/null
@@ -0,0 +1,21 @@
+// { dg-do compile { target c++2a } }
+
+template <class> struct all_same {
+  static constexpr bool value = true;
+};
+
+template <class T> 
+concept Assignable = requires(T t)
+{
+  requires all_same<decltype(t = 0)>::value;
+};
+
+template <class I> 
+  requires (!Assignable<I>)
+int dispatch();
+
+template <class I>
+  requires Assignable<I>
+void dispatch();
+
+int main() { dispatch<int *>(); }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires13.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires13.C
new file mode 100644 (file)
index 0000000..8ba3862
--- /dev/null
@@ -0,0 +1,20 @@
+// { dg-do compile { target c++2a } }
+
+template <class T> concept C = true;
+
+template <class T>
+  requires C<typename T::foo>
+void f(T t) { }
+
+void f(...);
+
+template <class T>
+  requires C<T>
+void g(T t) { }
+
+int main()
+{
+  f(42);
+  g(42);
+}
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires14.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires14.C
new file mode 100644 (file)
index 0000000..e2893c0
--- /dev/null
@@ -0,0 +1,24 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+concept A = sizeof(T) >= 4;
+
+template<typename T>
+concept B = __is_class(T);
+
+template<A T>
+void ok1(T a) {
+  return;
+}
+
+template<typename T>
+  requires B<T>
+void ok2(T a) {
+  return;
+}
+
+template<A T>
+  requires B<T>
+void fun(T a) {
+  return;
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires15.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires15.C
new file mode 100644 (file)
index 0000000..81d9196
--- /dev/null
@@ -0,0 +1,15 @@
+// { dg-do compile { target c++2a } }
+
+struct string;
+
+template<typename T>
+concept C = string; // { dg-error "expected primary-expression" }
+
+template<C T>
+void fun(T s) { }
+
+int main(int, char **) {
+  fun((int *)0); // { dg-error "cannot call function" }
+  return 0;
+}
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires16.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires16.C
new file mode 100644 (file)
index 0000000..209c916
--- /dev/null
@@ -0,0 +1,47 @@
+// { dg-do compile { target c++2a } }
+
+// A poor mans Integral concept.
+template<typename T>
+concept Integral = __is_same_as(T, int);
+
+template<int N>
+concept Nonnegative = N >= 0;
+
+template<typename... Args>
+concept UnaryPack = (sizeof...(Args) == 1);
+
+template<typename... Args>
+  requires Integral<Args...> // { dg-error "non-pack parameter" }
+void f1();
+
+template<typename... Args>
+  requires Integral<Args>... // { dg-error "parameter packs not expanded|expected unqualified-id" }
+void f2();
+
+template<typename... Args>
+  requires (Integral<Args> && ...)
+void f3() { }
+
+template<Integral... Args>
+void f4() { }
+
+// FIXME: This syntax is likely to be made invalid.
+template<Nonnegative... Args> // { dg-error "does not constrain a type" }
+void f5() { }
+
+template<UnaryPack Arg> // requires UnaryPack<Arg>
+void f6() { }
+
+template<UnaryPack... Args> // requires (... && UnaryPack<Args>)
+void f7() { }
+
+void driver()
+{
+  f1<int, int>(); // { dg-error "cannot call function" }
+  f3<int, int>();
+  f3<int, void>(); // { dg-error "cannot call function" }
+  f4<int, int>();
+  f4<int, void>(); // { dg-error "cannot call function" }
+  f7<int>();
+  f7<int, int>();
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires17.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires17.C
new file mode 100644 (file)
index 0000000..1ec1d59
--- /dev/null
@@ -0,0 +1,11 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+concept Type = true;
+
+template<typename T>
+concept C =
+  requires (T a) {
+    { a.f() } -> Type; // OK
+    { a.g() } -> const Type*; // { dg-error "not a plain type-constraint" }
+  };
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires2.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires2.C
new file mode 100644 (file)
index 0000000..45bb423
--- /dev/null
@@ -0,0 +1,74 @@
+// { dg-do compile { target c++2a } }
+
+// Test the types of atomic constraints
+
+// req5.C
+struct fool {
+  constexpr fool operator&&(fool) const { return {}; }
+  constexpr fool operator||(fool) const { return {}; }
+};
+
+template<typename T> constexpr fool p1() { return {}; }
+template<typename T> constexpr fool p2() { return {}; }
+
+template<typename T>
+concept Bad = p1<T>() && p2<T>();
+
+template<typename T> requires Bad<T> void bad(T x) { }
+
+void driver_2()
+{
+  bad(0); // { dg-error "cannot call" }
+}
+
+// req6.C
+struct X { };
+int operator==(X, X) { return 0; }
+
+template<typename T>
+concept C1 = (X());
+
+template<typename T>
+concept C2 = (X() == X());
+
+template<typename T>
+  requires C1<T>
+void h1(T) { } 
+
+template<typename T>
+  requires C2<T>
+void h2(T);
+
+void driver_3()
+{
+  h1(0); // { dg-error "cannot call" }
+  h2(0); // { dg-error "cannot call" } 
+}
+
+// req7.C
+template<bool B>
+struct boolean_constant 
+{
+  constexpr operator bool() const { return B; }
+};
+
+using true_type = boolean_constant<true>;
+using false_type = boolean_constant<false>;
+
+template<typename T>
+struct dependent_true : true_type { };
+
+template<typename T>
+struct dependent_false : false_type { };
+
+template<typename T>
+  requires (dependent_true<T>{}) // { dg-message "bool" }
+struct S5 { };
+
+template<typename T>
+  requires (dependent_false<T>{}) // { dg-message "bool" }
+struct S6 { };
+
+S5<int> x5; // { dg-error "template constraint failure" }
+S6<int> x6; // { dg-error "template constraint failure" }
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires3.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires3.C
new file mode 100644 (file)
index 0000000..bbcba0d
--- /dev/null
@@ -0,0 +1,32 @@
+// { dg-do compile { target c++2a } }
+
+// Test basic expression requirements
+
+// req13.C
+
+template<class T, class...Args>
+concept Constructible =
+  requires(Args&&...args) {
+    T {((Args&&)(args))...};
+    new T{((Args&&)(args))...};
+  };
+
+template<typename T>
+  requires Constructible<T> 
+struct A { };
+
+A<int> a;
+
+// req19.C
+
+struct B
+{
+  template <class T> 
+  void f(T t) requires requires (T tt) { tt; }
+  { }
+};
+
+int main()
+{
+  B().f(42);
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires4.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires4.C
new file mode 100644 (file)
index 0000000..61aa72d
--- /dev/null
@@ -0,0 +1,15 @@
+// { dg-do compile { target c++2a } }
+
+// Test associated type requirements
+
+// req8.C
+
+template<typename T>
+concept Has_member_type = requires { typename T::type; };
+
+template<typename T>
+concept Concept = true && Has_member_type<T>;
+
+template<typename T>
+  requires Concept<T>
+void foo(T t) { }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires5.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires5.C
new file mode 100644 (file)
index 0000000..fe37ed4
--- /dev/null
@@ -0,0 +1,45 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options -fconcepts-ts }
+
+// Test conversion requirements (not in C++20)
+
+// req9.C
+
+template<typename T>
+struct S1 { };
+
+template<typename T>
+concept C = requires(T x) { { x.fn() } -> S1<T>; };
+
+template<typename U>
+  requires C<U>
+void fn(U x)
+{
+  x.fn();
+}
+
+struct S2
+{
+  auto fn() const { return S1<S2>(); }
+};
+
+int driver_1()
+{
+  fn(S2{});
+  return 0;
+}
+
+// req10.C
+// Test implicit conversion requirements
+
+template<typename T, typename U>
+concept ConvertibleTo = requires(T& t) { {t} -> U&; };
+
+struct B { };
+class D : /*private*/ B { };
+
+void driver_2()
+{
+  static_assert(ConvertibleTo<D, B>()); // { dg-error "cannot be used as a function" }
+  static_assert(ConvertibleTo<D, B>); // { dg-error "static assertion failed" }
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires6.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires6.C
new file mode 100644 (file)
index 0000000..20df78b
--- /dev/null
@@ -0,0 +1,34 @@
+// { dg-do compile { target c++2a } }
+
+// Test deduction requirements.
+
+// req12.C
+
+template <typename T, typename U>
+concept SameAs = __is_same_as(T, U);
+
+template <typename T>
+concept C1 = requires(T t) { // { dg-message "in requirements" }
+  { t } -> SameAs<T>; // NOTE: t deduced as decltype((t))
+  // { dg-error "does not satisfy placeholder constraints" "" { target *-*-* } .-1 }
+};
+
+template <typename T>
+  requires C1<T>
+constexpr bool f1() { return true; }
+
+static_assert(f1<char>()); // { dg-error "cannot call" }
+static_assert(f1<int>()); // { dg-error "cannot call" }
+static_assert(f1<double>()); // { dg-error "cannot call" }
+
+
+template <typename T>
+concept C2 = requires(T t) {
+  { t } -> SameAs<T&>; // NOTE: t deduced as decltype((t))
+};
+
+template <typename T>
+  requires C2<T>
+constexpr bool f2() { return true; }
+
+static_assert(f2<int>()); // OK
similarity index 52%
rename from gcc/testsuite/g++.dg/concepts/req14.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-requires7.C
index 6e60b6f194f0939b301e9332738e9dca3da357e1..89057c4244121ed7c907555c30d83fb92176ea6a 100644 (file)
@@ -1,14 +1,15 @@
 // PR c++/66758
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template <class T, class U>
-concept bool C = requires (T t, U u) { t + u; };
+concept C = requires (T t, U u) { t + u; };
 
 template <class T, class U>
-requires C<T,U>
+  requires C<T,U>
 void f(T t, U u) { t + u; }
 
+struct non_addable { };
+
 int main()
 {
   using T = decltype(f(42, 24));
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-requires8.C b/gcc/testsuite/g++.dg/cpp2a/concepts-requires8.C
new file mode 100644 (file)
index 0000000..f10e4bc
--- /dev/null
@@ -0,0 +1,17 @@
+// { dg-do compile { target c++2a } }
+
+template <class T, class U>
+concept C = requires (T t, U u) { t + u; }; // { dg-message "in requirements" }
+
+template <class T, class U>
+  requires C<T,U>
+void f(T t, U u) { t + u; }
+
+struct non_addable { };
+
+int main()
+{
+  // FIXME: This diagnostic is being emitted twice, when it should
+  // be emitted just once.
+  using U = decltype(f(42, non_addable{})); // { dg-error "cannot call function" }
+}
similarity index 81%
rename from gcc/testsuite/g++.dg/concepts/req15.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-requires9.C
index 385d51da8cd6b90d5d0c78b25651ca69016f9b1a..3594176627fe833e1bcbafa6cbf66ffb29b5c5fd 100644 (file)
@@ -1,6 +1,5 @@
 // PR c++/66832
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template <class T, class U, unsigned N>
   requires requires (T& t, U &u) { t.foo(); u.foo(); }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-sfinae1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-sfinae1.C
new file mode 100644 (file)
index 0000000..87b5d75
--- /dev/null
@@ -0,0 +1,10 @@
+// { dg-do compile { target concepts } }
+
+template <class T> struct A
+{
+  static const int x = 42;
+};
+
+template <class T> concept R42 = A<T&>::x == 42;
+
+static_assert (!R42<void>);
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-template-parm1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-template-parm1.C
new file mode 100644 (file)
index 0000000..4ff401a
--- /dev/null
@@ -0,0 +1,19 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+concept Int = __is_same_as(T, int);
+
+// Type template parameters
+template<Int T = int> struct S1 { };
+template<Int = int> struct S2;
+template<Int T> struct S2 { };
+
+// Non-type template parameters
+template<Int auto N = 0> struct S3 { };
+template<Int auto = 0> struct S4;
+template<Int auto N> struct S4 { };
+
+S1<> s1;
+S2<> s2;
+S3<> s3;
+S4<> s4;
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-template-parm10.C b/gcc/testsuite/g++.dg/cpp2a/concepts-template-parm10.C
new file mode 100644 (file)
index 0000000..239b485
--- /dev/null
@@ -0,0 +1,10 @@
+// { dg-do compile { target c++2a } }
+
+template <class>
+concept Dummy = true;
+
+template <typename>
+class example {
+    template <Dummy<> U>
+    friend auto func();
+};
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-template-parm2.C b/gcc/testsuite/g++.dg/cpp2a/concepts-template-parm2.C
new file mode 100644 (file)
index 0000000..7c1d69b
--- /dev/null
@@ -0,0 +1,15 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+  concept Int = __is_same_as(T, int);
+
+template<typename> struct Foo;
+
+// Instantiation of default arguments happens at the point of
+// instantiation for the class.
+
+template<Int T = char> struct S1 { };
+template<Int auto X = false> struct S2 { };
+
+S1<> s1; // { dg-error "constraint failure" }
+S2<> s2; // { dg-error "placeholder constraints not satisfied" }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-template-parm5.C b/gcc/testsuite/g++.dg/cpp2a/concepts-template-parm5.C
new file mode 100644 (file)
index 0000000..78b6f1c
--- /dev/null
@@ -0,0 +1,9 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+  concept Int = __is_same_as(T, int);
+
+template<Int... Ts = int> struct S1; // { dg-error "default argument" }
+template<Int... = int> struct S2; // { dg-error "default argument" }
+template<Int auto... Ns = 0> struct S3; // { dg-error "default argument" }
+template<Int auto... = 0> struct S4; // { dg-error "default argument" }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-template-parm6.C b/gcc/testsuite/g++.dg/cpp2a/concepts-template-parm6.C
new file mode 100644 (file)
index 0000000..47c8ef7
--- /dev/null
@@ -0,0 +1,44 @@
+// { dg-do compile { target c++2a } }
+
+template<typename... Ts> struct are_same;
+
+template<>
+  struct are_same<> {
+    static constexpr bool value = true;
+  };
+
+template<typename T>
+  struct are_same<T> {
+    static constexpr bool value = true;
+  };
+
+template<typename T, typename U, typename... Ts>
+  struct are_same<T, U, Ts...> {
+    static constexpr bool value =
+      __is_same_as(T, U) && are_same<U, Ts...>::value;
+  };
+
+template<typename... Ts>
+concept Same = are_same<Ts...>::value;
+
+template<typename T>
+concept Int = __is_same_as(T, int);
+
+// NOTE: The use of Same is misleading; it constraints each T, not
+// the pack of Ts.
+template<Same... Ts> struct S1 { }; // requires (C<Ts> && ...)
+S1<int, int, int> s1;
+S1<int, int, bool> s2;
+
+// // NOTE: The use of Same is misleading; it constraints each deduced
+// // type, not the pack of Ts.
+template<Same auto... Xs> struct S2 { }; // requires (C<X> && ...) with each X deduced
+S2<true, true, true> s3;
+S2<true, true, 'a'> s4;  // OK
+
+template<Int... Ts> struct S3 { }; // requires (C<Ts> && ...)
+S3<int, int, char> x0; // { dg-error "template constraint failure" }
+
+template<Int auto... Xs> struct S4 { }; // requires (C<X> && ...) with each X deduced
+S4<0, 1, 2, 'a'> x1; // { dg-error "placeholder constraints not satisfied" }
+
similarity index 50%
rename from gcc/testsuite/g++.dg/concepts/template-parm8.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-template-parm8.C
index 63c451242226de810dabf9cda8659c5874785c25..adcffed060c40d957219fdccd3b5eeccc96b3ad7 100644 (file)
@@ -1,13 +1,12 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template<typename T>
-  concept bool C() { return __is_class(T); }
+  concept C = __is_class(T);
 
 template<typename T>
-  concept bool D() { return C<T>() and __is_empty(T); }
+  concept D = C<T> and __is_empty(T);
 
-template<template<typename Q> requires C<Q>() class X>
+template<template<typename Q> requires C<Q> class X>
   struct S { };
 
 // An unconstrained template can be used as an argument for any
@@ -16,7 +15,7 @@ template<typename A> struct T0 { };
 S<T0> x1;
 
 // Matching constraints are valid.
-template<typename A> requires C<A>() struct T1 { };
+template<typename A> requires C<A> struct T1 { };
 S<T1> x2;
 
 int main() { }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-template-parm9.C b/gcc/testsuite/g++.dg/cpp2a/concepts-template-parm9.C
new file mode 100644 (file)
index 0000000..d578b35
--- /dev/null
@@ -0,0 +1,18 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+  concept C = __is_class(T);
+
+template<typename T>
+  concept D = C<T> and __is_empty(T);
+
+template<template<typename Q> requires C<Q> class X>
+  struct S { };
+
+template<typename A> requires true struct T0 { };
+template<typename A> requires D<A> struct T1 { };
+
+S<T0> x3; // { dg-error "constraint mismatch|invalid type" }
+S<T1> x4; // { dg-error "constraint mismatch|invalid type" }
+
+int main() { }
similarity index 60%
rename from gcc/testsuite/g++.dg/concepts/traits1.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-traits1.C
index 27610e25a5403ab43696c7e9811fcd2ca0c43987..0ea529ef4a67adbbe3fe56a20dedafa3025448f2 100644 (file)
@@ -1,62 +1,61 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-options "-std=c++2a" }
 
 template<typename T>
-  concept bool Nothrow_assignable() { return __has_nothrow_assign(T); }
+  concept Nothrow_assignable = __has_nothrow_assign(T);
 
 template<typename T>
-  concept bool Nothrow_constructible() { return __has_nothrow_constructor(T); }
+  concept Nothrow_constructible = __has_nothrow_constructor(T);
 
 template<typename T>
-  concept bool Nothrow_copyable() { return __has_nothrow_copy(T); }
+  concept Nothrow_copyable = __has_nothrow_copy(T);
 
 template<typename T>
-  concept bool Trivially_assignable() { return __has_trivial_assign(T); }
+  concept Trivially_assignable = __has_trivial_assign(T);
 
 template<typename T>
-  concept bool Trivially_constructible() { return __has_trivial_constructor(T); }
+  concept Trivially_constructible = __has_trivial_constructor(T);
 
 template<typename T>
-  concept bool Trivially_copyable() { return __has_trivial_copy(T); }
+  concept Trivially_copyable = __has_trivial_copy(T);
 
 template<typename T>
-  concept bool Trivially_destructible() { return __has_trivial_destructor(T); }
+  concept Trivially_destructible = __has_trivial_destructor(T);
 
 template<typename T>
-  concept bool Dynamically_destructible() { return __has_virtual_destructor(T); }
+  concept Dynamically_destructible = __has_virtual_destructor(T);
 
 template<typename T>
-  concept bool Abstract() { return __is_abstract(T); }
+  concept Abstract = __is_abstract(T);
 
 template<typename T>
-  concept bool Polymorphic() { return __is_polymorphic(T); }
+  concept Polymorphic = __is_polymorphic(T);
 
 template<typename T>
-  concept bool Class() { return __is_class(T); }
+  concept Class = __is_class(T);
 
 template<typename T>
-  concept bool Empty() { return __is_empty(T); }
+  concept Empty = __is_empty(T);
 
 template<typename T>
-  concept bool Enum() { return __is_enum(T); }
+  concept Enum = __is_enum(T);
 
 template<typename T>
-  concept bool Final() { return __is_final(T); }
+  concept Final = __is_final(T);
 
 template<typename T>
-  concept bool Literal_type() { return __is_literal_type(T); }
+  concept Literal_type = __is_literal_type(T);
 
 template<typename T>
-  concept bool Pod() { return __is_pod(T); }
+  concept Pod = __is_pod(T);
 
 template<typename T>
-  concept bool Standard_layout() { return __is_standard_layout(T); }
+  concept Standard_layout = __is_standard_layout(T);
 
 template<typename T>
-  concept bool Trivial() { return __is_trivial(T); }
+  concept Trivial = __is_trivial(T);
 
 template<typename T>
-  concept bool Union() { return __is_union(T); }
+  concept Union = __is_union(T);
 
 template<Nothrow_assignable T> void f1() { }
 template<Nothrow_copyable T> void f2() { }
similarity index 59%
rename from gcc/testsuite/g++.dg/concepts/traits2.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-traits2.C
index 71dcfd37e6a5c2b98ab8ed2b781229f17158c328..1dc5ffe9c00f944ee378364ee97403dd9a5191f1 100644 (file)
@@ -1,62 +1,61 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
 
 template<typename T>
-  concept bool Nothrow_assignable() { return __has_nothrow_assign(T); }
+  concept Nothrow_assignable = __has_nothrow_assign(T);
 
 template<typename T>
-  concept bool Nothrow_constructible() { return __has_nothrow_constructor(T); }
+  concept Nothrow_constructible = __has_nothrow_constructor(T);
 
 template<typename T>
-  concept bool Nothrow_copyable() { return __has_nothrow_copy(T); }
+  concept Nothrow_copyable = __has_nothrow_copy(T);
 
 template<typename T>
-  concept bool Trivially_assignable() { return __has_trivial_assign(T); }
+  concept Trivially_assignable = __has_trivial_assign(T);
 
 template<typename T>
-  concept bool Trivially_constructible() { return __has_trivial_constructor(T); }
+  concept Trivially_constructible = __has_trivial_constructor(T);
 
 template<typename T>
-  concept bool Trivially_copyable() { return __has_trivial_copy(T); }
+  concept Trivially_copyable = __has_trivial_copy(T);
 
 template<typename T>
-  concept bool Trivially_destructible() { return __has_trivial_destructor(T); }
+  concept Trivially_destructible = __has_trivial_destructor(T);
 
 template<typename T>
-  concept bool Dynamically_destructible() { return __has_virtual_destructor(T); }
+  concept Dynamically_destructible = __has_virtual_destructor(T);
 
 template<typename T>
-  concept bool Abstract() { return __is_abstract(T); }
+  concept Abstract = __is_abstract(T);
 
 template<typename T>
-  concept bool Polymorphic() { return __is_polymorphic(T); }
+  concept Polymorphic = __is_polymorphic(T);
 
 template<typename T>
-  concept bool Class() { return __is_class(T); }
+  concept Class = __is_class(T);
 
 template<typename T>
-  concept bool Empty() { return __is_empty(T); }
+  concept Empty = __is_empty(T);
 
 template<typename T>
-  concept bool Enum() { return __is_enum(T); }
+  concept Enum = __is_enum(T);
 
 template<typename T>
-  concept bool Final() { return __is_final(T); }
+  concept Final = __is_final(T);
 
 template<typename T>
-  concept bool Literal_type() { return __is_literal_type(T); }
+  concept Literal_type = __is_literal_type(T);
 
 template<typename T>
-  concept bool Pod() { return __is_pod(T); }
+  concept Pod = __is_pod(T);
 
 template<typename T>
-  concept bool Standard_layout() { return __is_standard_layout(T); }
+  concept Standard_layout = __is_standard_layout(T);
 
 template<typename T>
-  concept bool Trivial() { return __is_trivial(T); }
+  concept Trivial = __is_trivial(T);
 
 template<typename T>
-  concept bool Union() { return __is_union(T); }
+  concept Union = __is_union(T);
 
 template<Nothrow_assignable T> void f1();
 template<Nothrow_copyable T> void f2();
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-ts1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-ts1.C
new file mode 100644 (file)
index 0000000..b819ad4
--- /dev/null
@@ -0,0 +1,49 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+// This tests the terse notation.
+
+template<typename T>
+concept True = true;
+
+template<typename T>
+concept False = false;
+
+template<typename T, typename U>
+concept SameAs = __is_same_as(T, U);
+
+True x1 = 0;
+False x2 = 0; // { dg-error "deduced initializer does not satisfy" }
+
+void f1(True t) { }
+void f2(False t) { }
+void f3(SameAs<int> q) { }
+void f4(True a, SameAs<decltype(a)> b) { }
+
+True f5() { return 0; }
+False f6() { return 0; } // { dg-error "deduced return type" }
+SameAs<int> f7() { return 0; }
+SameAs<int> f8() { return 'a'; } // { dg-error "deduced return type" }
+auto f9() -> True { return 0; }
+auto f10() -> False { return 0; } // { dg-error "deduced return type" }
+auto f11() -> SameAs<int> { return 0; }
+auto f12() -> SameAs<char> { return 0; } // { dg-error "deduced return type" }
+auto f13(int n) -> SameAs<decltype(n)> { return n; }
+auto f14(int n) -> SameAs<decltype(n)> { return 'a'; } // { dg-error "deduced return type" }
+auto f15(auto x) -> SameAs<decltype(x)> { return 0; } // { dg-error "deduced return type" }
+
+void driver()
+{
+  f1(0);
+  f2(0); // { dg-error "cannot call" }
+  f3(0);
+  f3('a'); // { dg-error "cannot call" }
+  f4(0, 0);
+  f4(0, 'a'); // { dg-error "cannot call" }
+  f15(0);
+  f15('a'); // { dg-message "" }
+}
+
+template<class T> concept bool C1() { return false; }
+template<C1 T> concept bool C2() { return true; } // { dg-error "cannot be constrained" }
+template<C1 T> concept bool C3 = true; // { dg-error "cannot be constrained" }
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-ts2.C b/gcc/testsuite/g++.dg/cpp2a/concepts-ts2.C
new file mode 100644 (file)
index 0000000..9d2dbee
--- /dev/null
@@ -0,0 +1,260 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+// Basic tests using function concepts.
+
+template<typename T>
+concept bool Type() { return true; }
+
+template<typename T>
+concept bool False() { return false; }
+
+template<typename T>
+concept bool Class() { return __is_class(T); }
+
+template<typename T>
+concept bool EmptyClass() { return Class<T>() && __is_empty(T); }
+
+template<typename T, typename U>
+concept bool Classes() { return __is_class(T) && __is_class (U); }
+
+struct empty { };
+
+struct nonempty { int n; };
+
+static_assert(Type<int>());
+static_assert(False<int>()); // { dg-error "static assertion failed" }
+
+// Basic checks
+
+template<typename T>
+  requires (Type<T>())
+int fn1(T t) { return 0; }
+
+template<typename T>
+  requires (False<T>())
+int fn2(T t) { return 0; }
+
+void driver()
+{
+  fn1(0); // OK
+  fn2(0); // { dg-error "cannot call function" }
+}
+
+// Ordering
+
+template<typename T>
+concept bool Cf() { return requires (T t) { t.f(); }; }
+
+template<typename T>
+concept bool Cfg() { return Cf<T>() && requires (T t) { t.g(); }; }
+
+template<typename T>
+concept bool Cf2() { return requires (T t) { t.f(); }; }
+
+template<typename T>
+constexpr int algo(T t) { return 0; }
+
+template<typename T> requires (Cf<T>())
+constexpr int algo(T t) { return 1; }
+
+template<typename T> requires (Cfg<T>())
+constexpr int algo(T t) { return 2; }
+
+template<typename T> requires (Cf<T>())
+constexpr int ambig(T t) { return 1; }
+
+template<typename T> requires (Cf2<T>())
+constexpr int ambig(T t) { return 1; }
+
+struct T1 {
+  void f() { }
+};
+
+struct T2 : T1 {
+  void g() { }
+};
+
+void driver_0()
+{
+  T1 x;
+  T2 y;
+  static_assert(algo(0) == 0);
+  static_assert(algo(x) == 1);
+  static_assert(algo(y) == 2);
+  ambig(x); // { dg-error "call of overload | is ambiguous" }
+}
+
+template<typename T>
+struct S
+{
+  void f() requires (Class<T>()) { }
+  
+  template<typename U>
+  struct Inner
+  {
+    void g() requires (Classes<T, U>()) { }
+  };
+
+  template<typename U> requires (Classes<T, U>()) void h(U) { }
+};
+
+struct X { };
+
+void driver_1()
+{
+  S<X> s1;
+  s1.f(); // OK
+  s1.h(s1); // OK
+  s1.h(0); // { dg-error "no matching function" }
+
+  S<int> s2;
+  s2.f(); // { dg-error "no matching function" }
+
+  S<X>::Inner<X> si1;
+  si1.g();
+  
+  S<X>::Inner<int> si2;
+  si2.g(); // { dg-error "no matching function" }
+}
+
+// Check constraints on non-dependent arguments, even when in a
+// dependent context.
+
+template<typename T> requires (Class<T>()) void f1(T x) { }
+
+// fn1-2.C -- Dependent checks
+template<typename T>
+void caller_1(T x) 
+{ 
+  f1(x); // Unchecked dependent arg.
+  f1(empty{}); // Checked non-dependent arg, but OK
+  f1(0); // { dg-error "cannot call function" }
+}
+
+// fn3.c -- Ordering
+template<typename T> 
+  requires (Class<T>()) 
+constexpr int f2(T x) { return 1; }
+
+template<typename T> 
+  requires (EmptyClass<T>())
+constexpr int f2(T x) { return 2; }
+
+template<typename T> 
+constexpr int f3(T x) requires (Class<T>()) { return 1; }
+
+template<typename T> 
+constexpr int f3(T x) requires (EmptyClass<T>()) { return 2; }
+
+void driver_2() 
+{
+  f2(0); // { dg-error "no matching function" }
+  static_assert (f2(empty{}) == 2);
+  static_assert (f2(nonempty{}) == 1);
+  f3(0); // { dg-error "no matching function" }
+  static_assert (f3(empty{}) == 2);
+  static_assert (f3(nonempty{}) == 1);
+}
+
+// fn8.C -- Address of overloaded functions
+template<typename T> requires (Type<T>()) void ok(T) { }
+template<typename T> requires (Class<T>()) void err(T) { }
+
+auto p1 = &ok<int>;
+auto p2 = &err<int>; // { dg-error "no matches" }
+void (*p3)(int) = &ok<int>;
+void (*p4)(int) = &err<int>; // { dg-error "no matches" }
+void (*p5)(int) = &ok;
+void (*p6)(int) = &err; // { dg-error "no matches" }
+
+template<typename T> void g(T x) { }
+
+void driver_3 () 
+{
+  g(&ok<int>);
+  g(&err<int>); // { dg-error "no matches" }
+}
+
+
+struct S2 {
+  template<typename T> requires (Type<T>()) int ok(T) { return 0; }
+  template<typename T> requires (Class<T>()) int err(T) { return 0; }
+};
+
+auto p7 = &S2::ok<int>;
+auto p8 = &S2::err<int>; // { dg-error "no matches" }
+int (S2::*p9)(int) = &S2::ok<int>;
+int (S2::*p10)(int) = &S2::err<int>; // { dg-error "no matches" }
+int (S2::*p11)(int) = &S2::ok;
+int (S2::*p12)(int) = &S2::err; // { dg-error "no matches" }
+
+// fn9.C -- Ordering with function address
+template<typename T> 
+  requires (Class<T>())
+constexpr int fn(T) { return 1; }
+
+template<typename T> 
+  requires (EmptyClass<T>())
+constexpr int fn(T) { return 2; }
+
+struct S3 
+{
+  template<typename T> 
+    requires (Class<T>())
+  constexpr int fn(T) const { return 1; }
+  
+  template<typename T> 
+    requires (EmptyClass<T>())
+  constexpr int fn(T) const { return 2; }
+};
+
+void driver_5 () {
+  struct X { };
+  struct Y { X x; };
+
+  constexpr X x;
+  constexpr Y y;
+  constexpr S3 s;
+
+  constexpr auto p1 = &fn<X>; // Empty f
+  static_assert (p1(x) == 2);
+
+  constexpr auto p2 = &fn<Y>; // Class f
+  static_assert(p2(y) == 1);
+
+  constexpr auto p3 = &S3::fn<X>; // Empty f
+  static_assert((s.*p3)(x) == 2);
+
+  constexpr auto p4 = &S3::fn<Y>; // Empty f
+  static_assert((s.*p4)(y) == 1);
+}
+
+
+// Redeclarations
+
+// FIXME: This should be a diagnosable error. The programmer has moved
+// the requirements from the template-head to the declarator.
+template<typename T> requires (Type<T>()) void f3();
+template<typename T> void f3() requires (Type<T>());
+
+void driver_4()
+{
+  f3<int>(); // { dg-error "call of overload | ambiguous" }
+}
+
+template<template<typename T> requires true class X> void f4();
+template<template<typename T> class X> void f4(); // OK: different declarations
+
+template<typename T> requires (Type<T>()) void def() { }
+template<typename T> requires (Type<T>()) void def() { } // { dg-error "redefinition" }
+
+// fn-concept1.C
+
+template<typename T>
+concept bool Tuple() { // { dg-error "multiple statements" }
+  static_assert(T::value, "");
+  return true;
+}
+
+void f(Tuple&);
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-ts3.C b/gcc/testsuite/g++.dg/cpp2a/concepts-ts3.C
new file mode 100644 (file)
index 0000000..c1f4ba9
--- /dev/null
@@ -0,0 +1,251 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+// Basic tests using variable concepts.
+
+template<typename T>
+concept bool Type = true;
+
+template<typename T>
+concept bool False = false;
+
+template<typename T>
+concept bool Class = __is_class(T);
+
+template<typename T>
+concept bool EmptyClass = Class<T> && __is_empty(T);
+
+template<typename T, typename U>
+concept bool Classes = __is_class(T) && __is_class (U);
+
+struct empty { };
+
+struct nonempty { int n; };
+
+static_assert(Type<int>);
+static_assert(False<int>); // { dg-error "static assertion failed" }
+
+// Basic checks
+
+template<typename T>
+  requires Type<T>
+int fn1(T t) { return 0; }
+
+template<typename T>
+  requires False<T>
+int fn2(T t) { return 0; }
+
+void driver()
+{
+  fn1(0); // OK
+  fn2(0); // { dg-error "cannot call function" }
+}
+
+// Ordering
+
+template<typename T>
+concept bool Cf = requires (T t) { t.f(); };
+
+template<typename T>
+concept bool Cfg = Cf<T> && requires (T t) { t.g(); };
+
+template<typename T>
+concept bool Cf2 = requires (T t) { t.f(); };
+
+template<typename T>
+constexpr int algo(T t) { return 0; }
+
+template<typename T> requires Cf<T>
+constexpr int algo(T t) { return 1; }
+
+template<typename T> requires Cfg<T>
+constexpr int algo(T t) { return 2; }
+
+template<typename T> requires Cf<T>
+constexpr int ambig(T t) { return 1; }
+
+template<typename T> requires Cf2<T>
+constexpr int ambig(T t) { return 1; }
+
+struct T1 {
+  void f() { }
+};
+
+struct T2 : T1 {
+  void g() { }
+};
+
+void driver_0()
+{
+  T1 x;
+  T2 y;
+  static_assert(algo(0) == 0);
+  static_assert(algo(x) == 1);
+  static_assert(algo(y) == 2);
+  ambig(x); // { dg-error "call of overload | is ambiguous" }
+}
+
+template<typename T>
+struct S
+{
+  void f() requires Class<T> { }
+  
+  template<typename U>
+  struct Inner
+  {
+    void g() requires Classes<T, U> { }
+  };
+
+  template<typename U> requires Classes<T, U> void h(U) { }
+};
+
+struct X { };
+
+void driver_1()
+{
+  S<X> s1;
+  s1.f(); // OK
+  s1.h(s1); // OK
+  s1.h(0); // { dg-error "no matching function" }
+
+  S<int> s2;
+  s2.f(); // { dg-error "no matching function" }
+
+  S<X>::Inner<X> si1;
+  si1.g();
+  
+  S<X>::Inner<int> si2;
+  si2.g(); // { dg-error "no matching function" }
+}
+
+// Check constraints on non-dependent arguments, even when in a
+// dependent context.
+
+template<typename T> requires Class<T> void f1(T x) { }
+
+// fn1-2.C -- Dependent checks
+template<typename T>
+void caller_1(T x) 
+{ 
+  f1(x); // Unchecked dependent arg.
+  f1(empty{}); // Checked non-dependent arg, but OK
+  f1(0); // { dg-error "cannot call function" }
+}
+
+// fn3.c -- Ordering
+template<typename T> 
+  requires Class<T> 
+constexpr int f2(T x) { return 1; }
+
+template<typename T> 
+  requires EmptyClass<T>
+constexpr int f2(T x) { return 2; }
+
+template<typename T> 
+constexpr int f3(T x) requires Class<T> { return 1; }
+
+template<typename T> 
+constexpr int f3(T x) requires EmptyClass<T> { return 2; }
+
+void driver_2() 
+{
+  f2(0); // { dg-error "no matching function" }
+  static_assert (f2(empty{}) == 2);
+  static_assert (f2(nonempty{}) == 1);
+  f3(0); // { dg-error "no matching function" }
+  static_assert (f3(empty{}) == 2);
+  static_assert (f3(nonempty{}) == 1);
+}
+
+// fn8.C -- Address of overloaded functions
+template<typename T> requires Type<T> void ok(T) { }
+template<typename T> requires Class<T> void err(T) { }
+
+auto p1 = &ok<int>;
+auto p2 = &err<int>; // { dg-error "no matches" }
+void (*p3)(int) = &ok<int>;
+void (*p4)(int) = &err<int>; // { dg-error "no matches" }
+void (*p5)(int) = &ok;
+void (*p6)(int) = &err; // { dg-error "no matches" }
+
+template<typename T> void g(T x) { }
+
+void driver_3 () 
+{
+  g(&ok<int>);
+  g(&err<int>); // { dg-error "no matches" }
+}
+
+
+struct S2 {
+  template<typename T> requires Type<T> int ok(T) { return 0; }
+  template<typename T> requires Class<T> int err(T) { return 0; }
+};
+
+auto p7 = &S2::ok<int>;
+auto p8 = &S2::err<int>; // { dg-error "no matches" }
+int (S2::*p9)(int) = &S2::ok<int>;
+int (S2::*p10)(int) = &S2::err<int>; // { dg-error "no matches" }
+int (S2::*p11)(int) = &S2::ok;
+int (S2::*p12)(int) = &S2::err; // { dg-error "no matches" }
+
+// fn9.C -- Ordering with function address
+template<typename T> 
+  requires Class<T>
+constexpr int fn(T) { return 1; }
+
+template<typename T> 
+  requires EmptyClass<T>
+constexpr int fn(T) { return 2; }
+
+struct S3 
+{
+  template<typename T> 
+    requires Class<T>
+  constexpr int fn(T) const { return 1; }
+  
+  template<typename T> 
+    requires EmptyClass<T>
+  constexpr int fn(T) const { return 2; }
+};
+
+void driver_5 () {
+  struct X { };
+  struct Y { X x; };
+
+  constexpr X x;
+  constexpr Y y;
+  constexpr S3 s;
+
+  constexpr auto p1 = &fn<X>; // Empty f
+  static_assert (p1(x) == 2);
+
+  constexpr auto p2 = &fn<Y>; // Class f
+  static_assert(p2(y) == 1);
+
+  constexpr auto p3 = &S3::fn<X>; // Empty f
+  static_assert((s.*p3)(x) == 2);
+
+  constexpr auto p4 = &S3::fn<Y>; // Empty f
+  static_assert((s.*p4)(y) == 1);
+}
+
+
+// Redeclarations
+
+// FIXME: This should be a diagnosable error. The programmer has moved
+// the requirements from the template-head to the declarator.
+template<typename T> requires Type<T> void f3();
+template<typename T> void f3() requires Type<T>;
+
+void driver_4()
+{
+  f3<int>(); // { dg-error "call of overload | ambiguous" }
+}
+
+template<template<typename T> requires true class X> void f4();
+template<template<typename T> class X> void f4(); // OK: different declarations
+
+template<typename T> requires Type<T> void def() { }
+template<typename T> requires Type<T> void def() { } // { dg-error "redefinition" }
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-ts4.C b/gcc/testsuite/g++.dg/cpp2a/concepts-ts4.C
new file mode 100644 (file)
index 0000000..3d720e8
--- /dev/null
@@ -0,0 +1,34 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+// Basic tests for introduction syntax.
+
+template<typename T>
+concept True = true;
+
+template<typename T>
+concept False = false;
+
+template<typename T, typename U>
+concept Same = __is_same_as(T, U);
+
+template<int N>
+concept Pos = N >= 0;
+
+True{T} void f1(T) { }
+False{T} void f2(T) { }
+Same{X, Y} void same();
+Pos{N} void fn();
+
+void driver()
+{
+  f1(0);
+  f2(0); // { dg-error "cannot call function" }
+
+  same<int, int>();
+  same<int, char>(); // { dg-error "cannot call function" }
+
+  fn<0>(); // OK
+  fn<-1>(); // { dg-error "cannot call function" }
+  fn<int>(); // { dg-error "no matching function" }
+}
similarity index 78%
rename from gcc/testsuite/g++.dg/concepts/member-concept.C
rename to gcc/testsuite/g++.dg/cpp2a/concepts-ts5.C
index ef577a19d0770dd1327e0a9e73383902680c1937..cdc40df4387a5d4bd07791137dfc2043a0d1f996 100644 (file)
@@ -1,5 +1,5 @@
-// { dg-do compile { target c++17 } }
-// { dg-options "-fconcepts" }
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
 
 struct Base {
   template<typename T>
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-ts6.C b/gcc/testsuite/g++.dg/cpp2a/concepts-ts6.C
new file mode 100644 (file)
index 0000000..bf665aa
--- /dev/null
@@ -0,0 +1,72 @@
+// { dg-do compile { target c++2a } }
+// { dg-additional-options "-fconcepts-ts" }
+
+template<typename T, int N, typename... Xs> concept bool C1 = true;
+
+template<template<typename> class X> concept bool C2 = true;
+
+template<typename... Ts> concept bool C3 = true;
+
+C1{A, B, ...C} struct S1 { };
+
+C2{T} void f();
+
+C2{...Ts} void g(); // { dg-error "cannot be introduced" }
+
+C3{...Ts} struct S2 { };
+C3{T, U, V} struct S3 { };
+C3{...Ts, U} struct S4 { }; // { dg-error "cannot deduce template parameters" }
+
+template<typename> struct X { };
+
+void driver1() {
+  S1<int, 0, char, bool, float> s1a;
+  S1<0, 0, char, bool, float> s1b; // { dg-error "type/value mismatch" }
+
+  f<X>();
+  f<int>(); // { dg-error "no matching function for call" }
+
+  S2<int> s2a;
+  S2<char, signed char, unsigned char> s2b;
+  S2<0> s2c; // { dg-error "type/value mismatch" }
+
+  S3<int, int, int> s3a;
+  S3<int, int> s3b; // { dg-error "wrong number of template arguments" }
+}
+
+template<typename... Args>
+struct all_same;
+
+template<typename T, typename U, typename... Args>
+struct all_same<T, U, Args...>
+{
+  static constexpr bool value = __is_same_as(T, U) && all_same<U, Args...>::value;
+};
+
+template<typename T>
+struct all_same<T>
+{
+  static constexpr bool value = true;
+};
+
+template<>
+struct all_same<>
+{
+  static constexpr bool value = true;
+};
+
+template<typename... Ts>
+concept AllSame = all_same<Ts...>::value;
+
+AllSame{...Ts} struct S5 { };
+AllSame{T, U} struct S6 { };
+
+void driver2()
+{
+  S5<int, int> s5a;
+  S5<int, int, int, int> s5b;
+  S5<int, int, int, char> s5c; // { dg-error "template constraint failure" }
+  S6<void, void> s6a;
+  S6<void, int> s6c; // { dg-error "template constraint failure" }
+  S6<void, void, void> s6b; // { dg-error "wrong number of template arguments" }
+}
\ No newline at end of file
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts-using1.C b/gcc/testsuite/g++.dg/cpp2a/concepts-using1.C
new file mode 100644 (file)
index 0000000..733382d
--- /dev/null
@@ -0,0 +1,17 @@
+// PR c++/79591
+// { dg-do compile { target c++2a } }
+
+template <class> concept True = true;
+
+// Fine.
+namespace X {
+  void f(auto) {}
+  void f(True auto) {}
+}
+
+void f(auto) {}
+namespace Y {
+  void f(True auto) {}
+  using ::f;
+  // error: 'template<class auto:3> void f(auto:3)' conflicts with a previous declaration
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts.C b/gcc/testsuite/g++.dg/cpp2a/concepts.C
new file mode 100644 (file)
index 0000000..5839207
--- /dev/null
@@ -0,0 +1,57 @@
+// { dg-do compile { target c++2a } }
+
+// Change in grammar for the expression trailing `requires`.
+template<typename T>
+  requires true != false // { dg-error "must be enclosed" }
+void f1(T)
+{ }
+
+template<typename T>
+void f3(T) requires true != false // { dg-error "must be enclosed" }
+{ }
+
+template<typename T>
+  requires requires {
+    requires true == false; // OK: logical-or-expressions allowed here.
+  }
+void f3(T)
+{ }
+
+template<typename T>
+concept bool C1 = true; // { dg-error "bool" }
+template<typename T>
+bool concept C2 = true; // { dg-error "concept definition syntax" }
+
+template<typename T>
+concept C3 = true; // OK
+template<typename T>
+concept C3 = true; // { dg-error "redefinition" }
+template<typename T, typename U>
+concept C3 = true; // { dg-error "different template parameters" }
+template<int>
+concept C3 = true; // { dg-error "different template parameters" }
+int C3 = 0; // { dg-error "different kind of entity" }
+
+int C4 = 0;
+template<typename T>
+concept C4 = true; // { dg-error "different kind of entity" }
+
+// Concepts as expressions
+
+template<typename T>
+concept True = true;
+
+template<typename T>
+concept False = false;
+
+static_assert(True<int>);
+static_assert(False<int>); // { dg-error "static assertion failed" }
+
+void f4(True auto);
+
+template<True T> concept C5 = true; // { dg-error "cannot be constrained" }
+
+
+template<typename T>
+concept Recursive = Recursive<T>; // { dg-error "not declared|expected" }
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts1.C b/gcc/testsuite/g++.dg/cpp2a/concepts1.C
new file mode 100644 (file)
index 0000000..f22464a
--- /dev/null
@@ -0,0 +1,27 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+concept Class = __is_class(T);
+
+template<Class T>
+void f1(T) { }
+
+struct empty { };
+
+// Redeclarations involving brief template parameters.
+
+template<Class T>
+void decl1(T);
+
+template<typename T>
+  requires Class<T>
+void decl1(T);
+
+void driver_1()
+{
+  f1(0); // { dg-error "cannot call function" }
+  f1(empty{});
+
+  decl1(empty{}); // { dg-error "call of overload | ambiguous" }
+}
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts2.C b/gcc/testsuite/g++.dg/cpp2a/concepts2.C
new file mode 100644 (file)
index 0000000..d3b45f7
--- /dev/null
@@ -0,0 +1,69 @@
+// { dg-do compile { target c++2a } }
+
+template<typename T>
+concept True = true;
+
+template<typename T>
+concept False = false;
+
+template<typename T>
+concept Int = __is_same_as(T, int);
+
+static_assert(True<int>);
+static_assert(!False<int>);
+static_assert(False<int>); // { dg-error "static assertion failed" }
+
+constexpr bool will_be_true() {
+  if (True<int>)
+    return true;
+  return false;
+}
+
+constexpr bool will_be_false() {
+  if (!False<int>)
+    return true;
+  return false;
+}
+
+static_assert(will_be_true());
+static_assert(will_be_false());
+
+template<typename T>
+constexpr bool is_int() {
+  if (Int<T>)
+    return true;
+  return false;
+}
+
+static_assert(is_int<int>());
+static_assert(is_int<void>()); // { dg-error "static assertion failed" }
+
+template<typename T>
+constexpr bool f1() {
+  if (Int<int>) //  Note: always true.
+    return true;
+  return false;
+}
+static_assert(f1<int>());
+static_assert(f1<void>());
+
+
+template<typename T>
+constexpr bool f2() {
+  if constexpr (Int<int>) // Note: always true.
+    return true;
+  return false;
+}
+static_assert(f2<int>());
+static_assert(f2<void>());
+
+template<typename T>
+concept C = true;
+
+int driver() {
+  bool c_int = (C<int>); 
+  if((C<int>))
+    ;
+  return c_int;
+}
+
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts3.C b/gcc/testsuite/g++.dg/cpp2a/concepts3.C
new file mode 100644 (file)
index 0000000..d962426
--- /dev/null
@@ -0,0 +1,48 @@
+// { dg-do compile { target c++2a } }
+
+template <class T, class U> concept same_as = __is_same_as(T,U);
+
+template<typename T>
+concept C1 = requires (T& t) { // { dg-message "in requirements" }
+  t.~T(); 
+};
+
+template<typename T>
+concept C2 = requires (T& t) { // { dg-message "in requirements" }
+  { t.~T() } -> same_as<void>;
+};
+
+template<typename T>
+concept C3 = requires { // { dg-message "in requirements" }
+  typename T::type;
+};
+
+class S1
+{
+  ~S1() { }
+  using type = int;
+};
+
+void driver() {
+  static_assert(C1<S1>, ""); // { dg-error "static assertion failed" }
+  static_assert(C2<S1>, ""); // { dg-error "static assertion failed" }
+  static_assert(C3<S1>, ""); // { dg-error "static assertion failed" }
+}
+
+template<typename T>
+  requires C1<T>
+void f1() { }
+
+template<typename T>
+  requires C2<T>
+void f2() { }
+
+template<typename T>
+  requires C3<T>
+void f3() { }
+
+void driver2() {
+  f1<S1>(); // { dg-error "cannot call|is private" }
+  f2<S1>(); // { dg-error "cannot call|is private" }
+  f3<S1>(); // { dg-error "cannot call|is private" }
+}
diff --git a/gcc/testsuite/g++.dg/cpp2a/concepts4.C b/gcc/testsuite/g++.dg/cpp2a/concepts4.C
new file mode 100644 (file)
index 0000000..388fad7
--- /dev/null
@@ -0,0 +1,10 @@
+// { dg-do compile { target c++2a } }
+
+template <class T> struct A { static const int x = 42; };
+
+template <class Ta> concept A42 = A<Ta>::x == 42;
+template <class Tv> concept Void = __is_same_as(Tv, void);
+template <class Tb, class Ub> concept A42b = Void<Tb> || A42<Ub>;
+template <class Tc> concept R42c = A42b<Tc, Tc&>;
+
+static_assert (R42c<void>);
diff --git a/gcc/testsuite/g++.dg/cpp2a/cond-triv2.C b/gcc/testsuite/g++.dg/cpp2a/cond-triv2.C
new file mode 100644 (file)
index 0000000..554f8a3
--- /dev/null
@@ -0,0 +1,30 @@
+// PR c++/67348
+// { dg-do compile { target c++2a } }
+
+#include <type_traits>
+#include <vector>
+using namespace std;
+
+template <class...Ts>
+  requires (is_destructible<Ts>::value && ...)
+struct variant {
+  ~variant() { /* ... */ }
+  ~variant()
+    requires (is_trivially_destructible<Ts>::value && ...) = default;
+
+  variant(variant&&) { /* ... */ }
+  variant(variant&&)
+    requires (is_trivially_move_constructible<Ts>::value && ...) = default;
+
+  variant& operator=(variant&&) { /* ... */ }
+  variant& operator=(variant&&)
+    requires (is_trivially_move_assignable<Ts>::value && ...) = default;
+
+  // ...similar treatment for copy construction / assignment...
+};
+
+static_assert(is_trivially_destructible<variant<int, float>>());
+static_assert(!is_trivially_destructible<variant<int, vector<int>>>());
+
+static_assert(is_trivially_move_constructible<variant<int, float>>());
+static_assert(!is_trivially_move_constructible<variant<int, vector<int>>>());
index ddeb4a2adc7b8a47bbd7ec1136b772ab57913a17..77f29db166036859bdb37f1ec3069039689933b9 100644 (file)
@@ -52,7 +52,7 @@ proc g++-dg-runtest { testcases flags default-extra-flags } {
            if { [llength $gpp_std_list] > 0 } {
                set std_list $gpp_std_list
            } else {
-               set std_list { 98 14 17 }
+               set std_list { 98 14 17 2a }
            }
            set option_list { }
            foreach x $std_list {
index 179202f0f7d4a41f3ad2698ce670551f13341b0e..f8a235d5c872da6a937912bb09b94ce9df6ee729 100644 (file)
@@ -8584,7 +8584,7 @@ proc check_effective_target_c++2a { } {
 
 # Check for C++ Concepts TS support, i.e. -fconcepts flag.
 proc check_effective_target_concepts { } {
-    return [check-flags { "" { } { -fconcepts } }]
+    return [check-flags { "" { } { -fconcepts -std=*2a } }]
 }
 
 # Return 1 if expensive testcases should be run.
index 3551a433b1f279fead2e5fdc7d7fa025498b2a73..357fcfd65c5931fc3d78d98ab49f9614903f3b83 100644 (file)
@@ -141,8 +141,9 @@ DEFTIMEVAR (TV_PARSE_INLINE          , "parser inl. func. body")
 DEFTIMEVAR (TV_PARSE_INMETH          , "parser inl. meth. body")
 DEFTIMEVAR (TV_TEMPLATE_INST         , "template instantiation")
 DEFTIMEVAR (TV_CONSTEXPR            , "constant expression evaluation")
-DEFTIMEVAR (TV_CONSTRAINT_SAT        , "constraint satisfaction")
-DEFTIMEVAR (TV_CONSTRAINT_SUB        , "constraint subsumption")
+DEFTIMEVAR (TV_CONSTRAINT_NORM      , "constraint normalization")
+DEFTIMEVAR (TV_CONSTRAINT_SAT       , "constraint satisfaction")
+DEFTIMEVAR (TV_CONSTRAINT_SUB       , "constraint subsumption")
 DEFTIMEVAR (TV_FLATTEN_INLINING      , "flatten inlining")
 DEFTIMEVAR (TV_EARLY_INLINING        , "early inlining heuristics")
 DEFTIMEVAR (TV_INLINE_PARAMETERS     , "inline parameters")