aboutsummaryrefslogtreecommitdiff
path: root/gcc/cp/lambda.c
diff options
context:
space:
mode:
Diffstat (limited to 'gcc/cp/lambda.c')
-rw-r--r--gcc/cp/lambda.c1583
1 files changed, 0 insertions, 1583 deletions
diff --git a/gcc/cp/lambda.c b/gcc/cp/lambda.c
deleted file mode 100644
index 4cc3a47..0000000
--- a/gcc/cp/lambda.c
+++ /dev/null
@@ -1,1583 +0,0 @@
-/* Perform the semantic phase of lambda parsing, i.e., the process of
- building tree structure, checking semantic consistency, and
- building RTL. These routines are used both during actual parsing
- and during the instantiation of template functions.
-
- Copyright (C) 1998-2022 Free Software Foundation, Inc.
-
- This file is part of GCC.
-
- GCC is free software; you can redistribute it and/or modify it
- under the terms of the GNU General Public License as published by
- the Free Software Foundation; either version 3, or (at your option)
- any later version.
-
- GCC is distributed in the hope that it will be useful, but
- WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- General Public License for more details.
-
-You should have received a copy of the GNU General Public License
-along with GCC; see the file COPYING3. If not see
-<http://www.gnu.org/licenses/>. */
-
-#include "config.h"
-#include "system.h"
-#include "coretypes.h"
-#include "cp-tree.h"
-#include "stringpool.h"
-#include "cgraph.h"
-#include "tree-iterator.h"
-#include "toplev.h"
-#include "gimplify.h"
-#include "target.h"
-
-/* Constructor for a lambda expression. */
-
-tree
-build_lambda_expr (void)
-{
- tree lambda = make_node (LAMBDA_EXPR);
- LAMBDA_EXPR_DEFAULT_CAPTURE_MODE (lambda) = CPLD_NONE;
- LAMBDA_EXPR_CAPTURE_LIST (lambda) = NULL_TREE;
- LAMBDA_EXPR_THIS_CAPTURE (lambda) = NULL_TREE;
- LAMBDA_EXPR_REGEN_INFO (lambda) = NULL_TREE;
- LAMBDA_EXPR_PENDING_PROXIES (lambda) = NULL;
- LAMBDA_EXPR_MUTABLE_P (lambda) = false;
- return lambda;
-}
-
-/* Create the closure object for a LAMBDA_EXPR. */
-
-tree
-build_lambda_object (tree lambda_expr)
-{
- /* Build aggregate constructor call.
- - cp_parser_braced_list
- - cp_parser_functional_cast */
- vec<constructor_elt, va_gc> *elts = NULL;
- tree node, expr, type;
-
- if (processing_template_decl || lambda_expr == error_mark_node)
- return lambda_expr;
-
- /* Make sure any error messages refer to the lambda-introducer. */
- location_t loc = LAMBDA_EXPR_LOCATION (lambda_expr);
- iloc_sentinel il (loc);
-
- for (node = LAMBDA_EXPR_CAPTURE_LIST (lambda_expr);
- node;
- node = TREE_CHAIN (node))
- {
- tree field = TREE_PURPOSE (node);
- tree val = TREE_VALUE (node);
-
- if (field == error_mark_node)
- {
- expr = error_mark_node;
- goto out;
- }
-
- if (TREE_CODE (val) == TREE_LIST)
- val = build_x_compound_expr_from_list (val, ELK_INIT,
- tf_warning_or_error);
-
- if (DECL_P (val))
- mark_used (val);
-
- /* Mere mortals can't copy arrays with aggregate initialization, so
- do some magic to make it work here. */
- if (TREE_CODE (TREE_TYPE (field)) == ARRAY_TYPE)
- val = build_array_copy (val);
- else if (DECL_NORMAL_CAPTURE_P (field)
- && !DECL_VLA_CAPTURE_P (field)
- && !TYPE_REF_P (TREE_TYPE (field)))
- {
- /* "the entities that are captured by copy are used to
- direct-initialize each corresponding non-static data
- member of the resulting closure object."
-
- There's normally no way to express direct-initialization
- from an element of a CONSTRUCTOR, so we build up a special
- TARGET_EXPR to bypass the usual copy-initialization. */
- val = force_rvalue (val, tf_warning_or_error);
- if (TREE_CODE (val) == TARGET_EXPR)
- TARGET_EXPR_DIRECT_INIT_P (val) = true;
- }
-
- CONSTRUCTOR_APPEND_ELT (elts, DECL_NAME (field), val);
- }
-
- expr = build_constructor (init_list_type_node, elts);
- CONSTRUCTOR_IS_DIRECT_INIT (expr) = 1;
-
- /* N2927: "[The closure] class type is not an aggregate."
- But we briefly treat it as an aggregate to make this simpler. */
- type = LAMBDA_EXPR_CLOSURE (lambda_expr);
- CLASSTYPE_NON_AGGREGATE (type) = 0;
- expr = finish_compound_literal (type, expr, tf_warning_or_error);
- protected_set_expr_location (expr, loc);
- CLASSTYPE_NON_AGGREGATE (type) = 1;
-
- out:
- return expr;
-}
-
-/* Return an initialized RECORD_TYPE for LAMBDA.
- LAMBDA must have its explicit captures already. */
-
-tree
-begin_lambda_type (tree lambda)
-{
- /* Lambda names are nearly but not quite anonymous. */
- tree name = make_anon_name ();
- IDENTIFIER_LAMBDA_P (name) = true;
-
- /* Create the new RECORD_TYPE for this lambda. */
- tree type = xref_tag (/*tag_code=*/record_type, name);
- if (type == error_mark_node)
- return error_mark_node;
-
- /* Designate it as a struct so that we can use aggregate initialization. */
- CLASSTYPE_DECLARED_CLASS (type) = false;
-
- /* Cross-reference the expression and the type. */
- LAMBDA_EXPR_CLOSURE (lambda) = type;
- CLASSTYPE_LAMBDA_EXPR (type) = lambda;
-
- /* In C++17, assume the closure is literal; we'll clear the flag later if
- necessary. */
- if (cxx_dialect >= cxx17)
- CLASSTYPE_LITERAL_P (type) = true;
-
- /* Clear base types. */
- xref_basetypes (type, /*bases=*/NULL_TREE);
-
- /* Start the class. */
- type = begin_class_definition (type);
-
- return type;
-}
-
-/* Given a LAMBDA_EXPR or closure type LAMBDA, return the op() of the
- closure type. */
-
-tree
-lambda_function (tree lambda)
-{
- tree type;
- if (TREE_CODE (lambda) == LAMBDA_EXPR)
- type = LAMBDA_EXPR_CLOSURE (lambda);
- else
- type = lambda;
- gcc_assert (LAMBDA_TYPE_P (type));
- /* Don't let debug_tree cause instantiation. */
- if (CLASSTYPE_TEMPLATE_INSTANTIATION (type)
- && !COMPLETE_OR_OPEN_TYPE_P (type))
- return NULL_TREE;
- lambda = lookup_member (type, call_op_identifier,
- /*protect=*/0, /*want_type=*/false,
- tf_warning_or_error);
- if (lambda)
- lambda = STRIP_TEMPLATE (get_first_fn (lambda));
- return lambda;
-}
-
-/* Returns the type to use for the FIELD_DECL corresponding to the
- capture of EXPR. EXPLICIT_INIT_P indicates whether this is a
- C++14 init capture, and BY_REFERENCE_P indicates whether we're
- capturing by reference. */
-
-tree
-lambda_capture_field_type (tree expr, bool explicit_init_p,
- bool by_reference_p)
-{
- tree type;
- bool is_this = is_this_parameter (tree_strip_nop_conversions (expr));
-
- if (is_this)
- type = TREE_TYPE (expr);
- else if (explicit_init_p)
- {
- tree auto_node = make_auto ();
-
- type = auto_node;
- if (by_reference_p)
- /* Add the reference now, so deduction doesn't lose
- outermost CV qualifiers of EXPR. */
- type = build_reference_type (type);
- if (uses_parameter_packs (expr))
- /* Stick with 'auto' even if the type could be deduced. */;
- else
- type = do_auto_deduction (type, expr, auto_node);
- }
- else if (type_dependent_expression_p (expr))
- {
- type = cxx_make_type (DECLTYPE_TYPE);
- DECLTYPE_TYPE_EXPR (type) = expr;
- DECLTYPE_FOR_LAMBDA_CAPTURE (type) = true;
- DECLTYPE_FOR_REF_CAPTURE (type) = by_reference_p;
- SET_TYPE_STRUCTURAL_EQUALITY (type);
- }
- else
- {
- STRIP_ANY_LOCATION_WRAPPER (expr);
-
- if (!by_reference_p && is_capture_proxy (expr))
- {
- /* When capturing by-value another capture proxy from an enclosing
- lambda, consider the type of the corresponding field instead,
- as the proxy may be additionally const-qualifed if the enclosing
- lambda is non-mutable (PR94376). */
- gcc_assert (TREE_CODE (DECL_VALUE_EXPR (expr)) == COMPONENT_REF);
- expr = TREE_OPERAND (DECL_VALUE_EXPR (expr), 1);
- }
-
- type = non_reference (unlowered_expr_type (expr));
-
- if (by_reference_p || TREE_CODE (type) == FUNCTION_TYPE)
- type = build_reference_type (type);
- }
-
- return type;
-}
-
-/* Returns true iff DECL is a lambda capture proxy variable created by
- build_capture_proxy. */
-
-bool
-is_capture_proxy (tree decl)
-{
- /* Location wrappers should be stripped or otherwise handled by the
- caller before using this predicate. */
- gcc_checking_assert (!location_wrapper_p (decl));
-
- return (VAR_P (decl)
- && DECL_HAS_VALUE_EXPR_P (decl)
- && !DECL_ANON_UNION_VAR_P (decl)
- && !DECL_DECOMPOSITION_P (decl)
- && !DECL_FNAME_P (decl)
- && !(DECL_ARTIFICIAL (decl)
- && DECL_LANG_SPECIFIC (decl)
- && DECL_OMP_PRIVATIZED_MEMBER (decl))
- && LAMBDA_FUNCTION_P (DECL_CONTEXT (decl)));
-}
-
-/* Returns true iff DECL is a capture proxy for a normal capture
- (i.e. without explicit initializer). */
-
-bool
-is_normal_capture_proxy (tree decl)
-{
- if (!is_capture_proxy (decl))
- /* It's not a capture proxy. */
- return false;
-
- return (DECL_LANG_SPECIFIC (decl)
- && DECL_CAPTURED_VARIABLE (decl));
-}
-
-/* Returns true iff DECL is a capture proxy for a normal capture
- of a constant variable. */
-
-bool
-is_constant_capture_proxy (tree decl)
-{
- if (is_normal_capture_proxy (decl))
- return decl_constant_var_p (DECL_CAPTURED_VARIABLE (decl));
- return false;
-}
-
-/* VAR is a capture proxy created by build_capture_proxy; add it to the
- current function, which is the operator() for the appropriate lambda. */
-
-void
-insert_capture_proxy (tree var)
-{
- if (is_normal_capture_proxy (var))
- {
- tree cap = DECL_CAPTURED_VARIABLE (var);
- if (CHECKING_P)
- {
- gcc_assert (!is_normal_capture_proxy (cap));
- tree old = retrieve_local_specialization (cap);
- if (old)
- gcc_assert (DECL_CONTEXT (old) != DECL_CONTEXT (var));
- }
- register_local_specialization (var, cap);
- }
-
- /* Put the capture proxy in the extra body block so that it won't clash
- with a later local variable. */
- pushdecl_outermost_localscope (var);
-
- /* And put a DECL_EXPR in the STATEMENT_LIST for the same block. */
- var = build_stmt (DECL_SOURCE_LOCATION (var), DECL_EXPR, var);
- tree stmt_list = (*stmt_list_stack)[1];
- gcc_assert (stmt_list);
- append_to_statement_list_force (var, &stmt_list);
-}
-
-/* We've just finished processing a lambda; if the containing scope is also
- a lambda, insert any capture proxies that were created while processing
- the nested lambda. */
-
-void
-insert_pending_capture_proxies (void)
-{
- tree lam;
- vec<tree, va_gc> *proxies;
- unsigned i;
-
- if (!current_function_decl || !LAMBDA_FUNCTION_P (current_function_decl))
- return;
-
- lam = CLASSTYPE_LAMBDA_EXPR (DECL_CONTEXT (current_function_decl));
- proxies = LAMBDA_EXPR_PENDING_PROXIES (lam);
- for (i = 0; i < vec_safe_length (proxies); ++i)
- {
- tree var = (*proxies)[i];
- insert_capture_proxy (var);
- }
- release_tree_vector (LAMBDA_EXPR_PENDING_PROXIES (lam));
- LAMBDA_EXPR_PENDING_PROXIES (lam) = NULL;
-}
-
-/* Given REF, a COMPONENT_REF designating a field in the lambda closure,
- return the type we want the proxy to have: the type of the field itself,
- with added const-qualification if the lambda isn't mutable and the
- capture is by value. */
-
-tree
-lambda_proxy_type (tree ref)
-{
- tree type;
- if (ref == error_mark_node)
- return error_mark_node;
- if (REFERENCE_REF_P (ref))
- ref = TREE_OPERAND (ref, 0);
- gcc_assert (TREE_CODE (ref) == COMPONENT_REF);
- type = TREE_TYPE (ref);
- if (!type || WILDCARD_TYPE_P (non_reference (type)))
- {
- type = cxx_make_type (DECLTYPE_TYPE);
- DECLTYPE_TYPE_EXPR (type) = ref;
- DECLTYPE_FOR_LAMBDA_PROXY (type) = true;
- SET_TYPE_STRUCTURAL_EQUALITY (type);
- }
- if (DECL_PACK_P (TREE_OPERAND (ref, 1)))
- type = make_pack_expansion (type);
- return type;
-}
-
-/* MEMBER is a capture field in a lambda closure class. Now that we're
- inside the operator(), build a placeholder var for future lookups and
- debugging. */
-
-static tree
-build_capture_proxy (tree member, tree init)
-{
- tree var, object, fn, closure, name, lam, type;
-
- if (PACK_EXPANSION_P (member))
- member = PACK_EXPANSION_PATTERN (member);
-
- closure = DECL_CONTEXT (member);
- fn = lambda_function (closure);
- lam = CLASSTYPE_LAMBDA_EXPR (closure);
-
- /* The proxy variable forwards to the capture field. */
- object = build_fold_indirect_ref (DECL_ARGUMENTS (fn));
- object = finish_non_static_data_member (member, object, NULL_TREE);
- if (REFERENCE_REF_P (object))
- object = TREE_OPERAND (object, 0);
-
- /* Remove the __ inserted by add_capture. */
- name = get_identifier (IDENTIFIER_POINTER (DECL_NAME (member)) + 2);
-
- type = lambda_proxy_type (object);
-
- if (name == this_identifier && !INDIRECT_TYPE_P (type))
- {
- type = build_pointer_type (type);
- type = cp_build_qualified_type (type, TYPE_QUAL_CONST);
- object = build_fold_addr_expr_with_type (object, type);
- }
-
- if (DECL_VLA_CAPTURE_P (member))
- {
- /* Rebuild the VLA type from the pointer and maxindex. */
- tree field = next_initializable_field (TYPE_FIELDS (type));
- tree ptr = build_simple_component_ref (object, field);
- field = next_initializable_field (DECL_CHAIN (field));
- tree max = build_simple_component_ref (object, field);
- type = build_cplus_array_type (TREE_TYPE (TREE_TYPE (ptr)),
- build_index_type (max));
- type = build_reference_type (type);
- object = convert (type, ptr);
- }
-
- complete_type (type);
-
- var = build_decl (input_location, VAR_DECL, name, type);
- SET_DECL_VALUE_EXPR (var, object);
- DECL_HAS_VALUE_EXPR_P (var) = 1;
- DECL_ARTIFICIAL (var) = 1;
- TREE_USED (var) = 1;
- DECL_CONTEXT (var) = fn;
-
- if (DECL_NORMAL_CAPTURE_P (member))
- {
- if (DECL_VLA_CAPTURE_P (member))
- {
- init = CONSTRUCTOR_ELT (init, 0)->value;
- init = TREE_OPERAND (init, 0); // Strip ADDR_EXPR.
- init = TREE_OPERAND (init, 0); // Strip ARRAY_REF.
- }
- else
- {
- if (PACK_EXPANSION_P (init))
- init = PACK_EXPANSION_PATTERN (init);
- }
-
- if (INDIRECT_REF_P (init))
- init = TREE_OPERAND (init, 0);
- STRIP_NOPS (init);
-
- gcc_assert (VAR_P (init) || TREE_CODE (init) == PARM_DECL);
- while (is_normal_capture_proxy (init))
- init = DECL_CAPTURED_VARIABLE (init);
- retrofit_lang_decl (var);
- DECL_CAPTURED_VARIABLE (var) = init;
- }
-
- if (name == this_identifier)
- {
- gcc_assert (LAMBDA_EXPR_THIS_CAPTURE (lam) == member);
- LAMBDA_EXPR_THIS_CAPTURE (lam) = var;
- }
-
- if (fn == current_function_decl)
- insert_capture_proxy (var);
- else
- vec_safe_push (LAMBDA_EXPR_PENDING_PROXIES (lam), var);
-
- return var;
-}
-
-static GTY(()) tree ptr_id;
-static GTY(()) tree max_id;
-
-/* Return a struct containing a pointer and a length for lambda capture of
- an array of runtime length. */
-
-static tree
-vla_capture_type (tree array_type)
-{
- tree type = xref_tag (record_type, make_anon_name ());
- xref_basetypes (type, NULL_TREE);
- type = begin_class_definition (type);
- if (!ptr_id)
- {
- ptr_id = get_identifier ("ptr");
- max_id = get_identifier ("max");
- }
- tree ptrtype = build_pointer_type (TREE_TYPE (array_type));
- tree field = build_decl (input_location, FIELD_DECL, ptr_id, ptrtype);
- finish_member_declaration (field);
- field = build_decl (input_location, FIELD_DECL, max_id, sizetype);
- finish_member_declaration (field);
- return finish_struct (type, NULL_TREE);
-}
-
-/* From an ID and INITIALIZER, create a capture (by reference if
- BY_REFERENCE_P is true), add it to the capture-list for LAMBDA,
- and return it. If ID is `this', BY_REFERENCE_P says whether
- `*this' is captured by reference. */
-
-tree
-add_capture (tree lambda, tree id, tree orig_init, bool by_reference_p,
- bool explicit_init_p)
-{
- char *buf;
- tree type, member, name;
- bool vla = false;
- bool variadic = false;
- tree initializer = orig_init;
-
- if (PACK_EXPANSION_P (initializer))
- {
- initializer = PACK_EXPANSION_PATTERN (initializer);
- variadic = true;
- }
-
- if (TREE_CODE (initializer) == TREE_LIST
- /* A pack expansion might end up with multiple elements. */
- && !PACK_EXPANSION_P (TREE_VALUE (initializer)))
- initializer = build_x_compound_expr_from_list (initializer, ELK_INIT,
- tf_warning_or_error);
- type = TREE_TYPE (initializer);
- if (type == error_mark_node)
- return error_mark_node;
-
- if (!dependent_type_p (type) && array_of_runtime_bound_p (type))
- {
- vla = true;
- if (!by_reference_p)
- error ("array of runtime bound cannot be captured by copy, "
- "only by reference");
-
- /* For a VLA, we capture the address of the first element and the
- maximum index, and then reconstruct the VLA for the proxy. */
- tree elt = cp_build_array_ref (input_location, initializer,
- integer_zero_node, tf_warning_or_error);
- initializer = build_constructor_va (init_list_type_node, 2,
- NULL_TREE, build_address (elt),
- NULL_TREE, array_type_nelts (type));
- type = vla_capture_type (type);
- }
- else if (!dependent_type_p (type)
- && variably_modified_type_p (type, NULL_TREE))
- {
- sorry ("capture of variably-modified type %qT that is not an N3639 array "
- "of runtime bound", type);
- if (TREE_CODE (type) == ARRAY_TYPE
- && variably_modified_type_p (TREE_TYPE (type), NULL_TREE))
- inform (input_location, "because the array element type %qT has "
- "variable size", TREE_TYPE (type));
- return error_mark_node;
- }
- else
- {
- type = lambda_capture_field_type (initializer, explicit_init_p,
- by_reference_p);
- if (type == error_mark_node)
- return error_mark_node;
-
- if (id == this_identifier && !by_reference_p)
- {
- gcc_assert (INDIRECT_TYPE_P (type));
- type = TREE_TYPE (type);
- initializer = cp_build_fold_indirect_ref (initializer);
- }
-
- if (dependent_type_p (type))
- ;
- else if (id != this_identifier && by_reference_p)
- {
- if (!lvalue_p (initializer))
- {
- error ("cannot capture %qE by reference", initializer);
- return error_mark_node;
- }
- }
- else
- {
- /* Capture by copy requires a complete type. */
- type = complete_type (type);
- if (!COMPLETE_TYPE_P (type))
- {
- error ("capture by copy of incomplete type %qT", type);
- cxx_incomplete_type_inform (type);
- return error_mark_node;
- }
- else if (!verify_type_context (input_location,
- TCTX_CAPTURE_BY_COPY, type))
- return error_mark_node;
- }
- }
-
- /* Add __ to the beginning of the field name so that user code
- won't find the field with name lookup. We can't just leave the name
- unset because template instantiation uses the name to find
- instantiated fields. */
- buf = (char *) alloca (IDENTIFIER_LENGTH (id) + 3);
- buf[1] = buf[0] = '_';
- memcpy (buf + 2, IDENTIFIER_POINTER (id),
- IDENTIFIER_LENGTH (id) + 1);
- name = get_identifier (buf);
-
- if (variadic)
- {
- type = make_pack_expansion (type);
- if (explicit_init_p)
- /* With an explicit initializer 'type' is auto, which isn't really a
- parameter pack in this context. We will want as many fields as we
- have elements in the expansion of the initializer, so use its packs
- instead. */
- {
- PACK_EXPANSION_PARAMETER_PACKS (type)
- = uses_parameter_packs (initializer);
- PACK_EXPANSION_AUTO_P (type) = true;
- }
- }
-
- /* Make member variable. */
- member = build_decl (input_location, FIELD_DECL, name, type);
- DECL_VLA_CAPTURE_P (member) = vla;
-
- if (!explicit_init_p)
- /* Normal captures are invisible to name lookup but uses are replaced
- with references to the capture field; we implement this by only
- really making them invisible in unevaluated context; see
- qualify_lookup. For now, let's make explicitly initialized captures
- always visible. */
- DECL_NORMAL_CAPTURE_P (member) = true;
-
- if (id == this_identifier)
- LAMBDA_EXPR_THIS_CAPTURE (lambda) = member;
-
- /* Add it to the appropriate closure class if we've started it. */
- if (current_class_type
- && current_class_type == LAMBDA_EXPR_CLOSURE (lambda))
- {
- if (COMPLETE_TYPE_P (current_class_type))
- internal_error ("trying to capture %qD in instantiation of "
- "generic lambda", id);
- finish_member_declaration (member);
- }
-
- tree listmem = member;
- if (variadic)
- {
- listmem = make_pack_expansion (member);
- initializer = orig_init;
- }
- LAMBDA_EXPR_CAPTURE_LIST (lambda)
- = tree_cons (listmem, initializer, LAMBDA_EXPR_CAPTURE_LIST (lambda));
-
- if (LAMBDA_EXPR_CLOSURE (lambda))
- return build_capture_proxy (member, initializer);
- /* For explicit captures we haven't started the function yet, so we wait
- and build the proxy from cp_parser_lambda_body. */
- LAMBDA_CAPTURE_EXPLICIT_P (LAMBDA_EXPR_CAPTURE_LIST (lambda)) = true;
- return NULL_TREE;
-}
-
-/* Register all the capture members on the list CAPTURES, which is the
- LAMBDA_EXPR_CAPTURE_LIST for the lambda after the introducer. */
-
-void
-register_capture_members (tree captures)
-{
- if (captures == NULL_TREE)
- return;
-
- register_capture_members (TREE_CHAIN (captures));
-
- tree field = TREE_PURPOSE (captures);
- if (PACK_EXPANSION_P (field))
- field = PACK_EXPANSION_PATTERN (field);
-
- finish_member_declaration (field);
-}
-
-/* Similar to add_capture, except this works on a stack of nested lambdas.
- BY_REFERENCE_P in this case is derived from the default capture mode.
- Returns the capture for the lambda at the bottom of the stack. */
-
-tree
-add_default_capture (tree lambda_stack, tree id, tree initializer)
-{
- bool this_capture_p = (id == this_identifier);
- tree var = NULL_TREE;
- tree saved_class_type = current_class_type;
-
- for (tree node = lambda_stack;
- node;
- node = TREE_CHAIN (node))
- {
- tree lambda = TREE_VALUE (node);
-
- current_class_type = LAMBDA_EXPR_CLOSURE (lambda);
- if (DECL_PACK_P (initializer))
- initializer = make_pack_expansion (initializer);
- var = add_capture (lambda,
- id,
- initializer,
- /*by_reference_p=*/
- (this_capture_p
- || (LAMBDA_EXPR_DEFAULT_CAPTURE_MODE (lambda)
- == CPLD_REFERENCE)),
- /*explicit_init_p=*/false);
- initializer = convert_from_reference (var);
-
- /* Warn about deprecated implicit capture of this via [=]. */
- if (cxx_dialect >= cxx20
- && this_capture_p
- && LAMBDA_EXPR_DEFAULT_CAPTURE_MODE (lambda) == CPLD_COPY)
- {
- if (warning_at (LAMBDA_EXPR_LOCATION (lambda), OPT_Wdeprecated,
- "implicit capture of %qE via %<[=]%> is deprecated "
- "in C++20", this_identifier))
- inform (LAMBDA_EXPR_LOCATION (lambda), "add explicit %<this%> or "
- "%<*this%> capture");
- }
- }
-
- current_class_type = saved_class_type;
-
- return var;
-}
-
-/* Return the capture pertaining to a use of 'this' in LAMBDA, in the
- form of an INDIRECT_REF, possibly adding it through default
- capturing, if ADD_CAPTURE_P is nonzero. If ADD_CAPTURE_P is negative,
- try to capture but don't complain if we can't. */
-
-tree
-lambda_expr_this_capture (tree lambda, int add_capture_p)
-{
- tree result;
-
- tree this_capture = LAMBDA_EXPR_THIS_CAPTURE (lambda);
-
- /* In unevaluated context this isn't an odr-use, so don't capture. */
- if (cp_unevaluated_operand)
- add_capture_p = false;
-
- /* Try to default capture 'this' if we can. */
- if (!this_capture)
- {
- tree lambda_stack = NULL_TREE;
- tree init = NULL_TREE;
-
- /* If we are in a lambda function, we can move out until we hit:
- 1. a non-lambda function or NSDMI,
- 2. a lambda function capturing 'this', or
- 3. a non-default capturing lambda function. */
- for (tree tlambda = lambda; ;)
- {
- if (add_capture_p
- && LAMBDA_EXPR_DEFAULT_CAPTURE_MODE (tlambda) == CPLD_NONE)
- /* tlambda won't let us capture 'this'. */
- break;
-
- if (add_capture_p)
- lambda_stack = tree_cons (NULL_TREE,
- tlambda,
- lambda_stack);
-
- tree closure = LAMBDA_EXPR_CLOSURE (tlambda);
- tree containing_function
- = decl_function_context (TYPE_NAME (closure));
-
- tree ex = LAMBDA_EXPR_EXTRA_SCOPE (tlambda);
- if (ex && TREE_CODE (ex) == FIELD_DECL)
- {
- /* Lambda in an NSDMI. We don't have a function to look up
- 'this' in, but we can find (or rebuild) the fake one from
- inject_this_parameter. */
- if (!containing_function && !COMPLETE_TYPE_P (closure))
- /* If we're parsing a lambda in a non-local class,
- we can find the fake 'this' in scope_chain. */
- init = scope_chain->x_current_class_ptr;
- else
- /* Otherwise it's either gone or buried in
- function_context_stack, so make another. */
- init = build_this_parm (NULL_TREE, DECL_CONTEXT (ex),
- TYPE_UNQUALIFIED);
- gcc_checking_assert
- (init && (TREE_TYPE (TREE_TYPE (init))
- == current_nonlambda_class_type ()));
- break;
- }
-
- if (containing_function == NULL_TREE)
- /* We ran out of scopes; there's no 'this' to capture. */
- break;
-
- if (!LAMBDA_FUNCTION_P (containing_function))
- {
- /* We found a non-lambda function. */
- if (DECL_NONSTATIC_MEMBER_FUNCTION_P (containing_function))
- /* First parameter is 'this'. */
- init = DECL_ARGUMENTS (containing_function);
- break;
- }
-
- tlambda
- = CLASSTYPE_LAMBDA_EXPR (DECL_CONTEXT (containing_function));
-
- if (LAMBDA_EXPR_THIS_CAPTURE (tlambda))
- {
- /* An outer lambda has already captured 'this'. */
- init = LAMBDA_EXPR_THIS_CAPTURE (tlambda);
- break;
- }
- }
-
- if (init)
- {
- if (add_capture_p)
- this_capture = add_default_capture (lambda_stack,
- /*id=*/this_identifier,
- init);
- else
- this_capture = init;
- }
- }
-
- if (cp_unevaluated_operand)
- result = this_capture;
- else if (!this_capture)
- {
- if (add_capture_p == 1)
- {
- error ("%<this%> was not captured for this lambda function");
- result = error_mark_node;
- }
- else
- result = NULL_TREE;
- }
- else
- {
- /* To make sure that current_class_ref is for the lambda. */
- gcc_assert (TYPE_MAIN_VARIANT (TREE_TYPE (current_class_ref))
- == LAMBDA_EXPR_CLOSURE (lambda));
-
- result = this_capture;
-
- /* If 'this' is captured, each use of 'this' is transformed into an
- access to the corresponding unnamed data member of the closure
- type cast (_expr.cast_ 5.4) to the type of 'this'. [ The cast
- ensures that the transformed expression is an rvalue. ] */
- result = rvalue (result);
- }
-
- return result;
-}
-
-/* Return the innermost LAMBDA_EXPR we're currently in, if any. */
-
-tree
-current_lambda_expr (void)
-{
- tree type = current_class_type;
- while (type && !LAMBDA_TYPE_P (type))
- type = decl_type_context (TYPE_NAME (type));
- if (type)
- return CLASSTYPE_LAMBDA_EXPR (type);
- else
- return NULL_TREE;
-}
-
-/* Return the current LAMBDA_EXPR, if this is a resolvable dummy
- object. NULL otherwise.. */
-
-static tree
-resolvable_dummy_lambda (tree object)
-{
- if (!is_dummy_object (object))
- return NULL_TREE;
-
- tree type = TYPE_MAIN_VARIANT (TREE_TYPE (object));
- gcc_assert (!TYPE_PTR_P (type));
-
- if (type != current_class_type
- && current_class_type
- && LAMBDA_TYPE_P (current_class_type)
- && lambda_function (current_class_type)
- && DERIVED_FROM_P (type, nonlambda_method_basetype()))
- return CLASSTYPE_LAMBDA_EXPR (current_class_type);
-
- return NULL_TREE;
-}
-
-/* We don't want to capture 'this' until we know we need it, i.e. after
- overload resolution has chosen a non-static member function. At that
- point we call this function to turn a dummy object into a use of the
- 'this' capture. */
-
-tree
-maybe_resolve_dummy (tree object, bool add_capture_p)
-{
- if (tree lam = resolvable_dummy_lambda (object))
- if (tree cap = lambda_expr_this_capture (lam, add_capture_p))
- if (cap != error_mark_node)
- object = build_fold_indirect_ref (cap);
-
- return object;
-}
-
-/* When parsing a generic lambda containing an argument-dependent
- member function call we defer overload resolution to instantiation
- time. But we have to know now whether to capture this or not.
- Do that if FNS contains any non-static fns.
- The std doesn't anticipate this case, but I expect this to be the
- outcome of discussion. */
-
-void
-maybe_generic_this_capture (tree object, tree fns)
-{
- if (tree lam = resolvable_dummy_lambda (object))
- if (!LAMBDA_EXPR_THIS_CAPTURE (lam))
- {
- /* We've not yet captured, so look at the function set of
- interest. */
- if (BASELINK_P (fns))
- fns = BASELINK_FUNCTIONS (fns);
- bool id_expr = TREE_CODE (fns) == TEMPLATE_ID_EXPR;
- if (id_expr)
- fns = TREE_OPERAND (fns, 0);
-
- for (lkp_iterator iter (fns); iter; ++iter)
- if (((!id_expr && TREE_CODE (*iter) != USING_DECL)
- || TREE_CODE (*iter) == TEMPLATE_DECL)
- && DECL_NONSTATIC_MEMBER_FUNCTION_P (*iter))
- {
- /* Found a non-static member. Capture this. */
- lambda_expr_this_capture (lam, /*maybe*/-1);
- break;
- }
- }
-}
-
-/* Returns the innermost non-lambda function. */
-
-tree
-current_nonlambda_function (void)
-{
- tree fn = current_function_decl;
- while (fn && LAMBDA_FUNCTION_P (fn))
- fn = decl_function_context (fn);
- return fn;
-}
-
-/* Returns the method basetype of the innermost non-lambda function, including
- a hypothetical constructor if inside an NSDMI, or NULL_TREE if none. */
-
-tree
-nonlambda_method_basetype (void)
-{
- if (!current_class_ref)
- return NULL_TREE;
-
- tree type = current_class_type;
- if (!type || !LAMBDA_TYPE_P (type))
- return type;
-
- while (true)
- {
- tree lam = CLASSTYPE_LAMBDA_EXPR (type);
- tree ex = LAMBDA_EXPR_EXTRA_SCOPE (lam);
- if (ex && TREE_CODE (ex) == FIELD_DECL)
- /* Lambda in an NSDMI. */
- return DECL_CONTEXT (ex);
-
- tree fn = TYPE_CONTEXT (type);
- if (!fn || TREE_CODE (fn) != FUNCTION_DECL
- || !DECL_NONSTATIC_MEMBER_FUNCTION_P (fn))
- /* No enclosing non-lambda method. */
- return NULL_TREE;
- if (!LAMBDA_FUNCTION_P (fn))
- /* Found an enclosing non-lambda method. */
- return TYPE_METHOD_BASETYPE (TREE_TYPE (fn));
- type = DECL_CONTEXT (fn);
- }
-}
-
-/* Like current_scope, but looking through lambdas. */
-
-tree
-current_nonlambda_scope (void)
-{
- tree scope = current_scope ();
- for (;;)
- {
- if (TREE_CODE (scope) == FUNCTION_DECL
- && LAMBDA_FUNCTION_P (scope))
- {
- scope = CP_TYPE_CONTEXT (DECL_CONTEXT (scope));
- continue;
- }
- else if (LAMBDA_TYPE_P (scope))
- {
- scope = CP_TYPE_CONTEXT (scope);
- continue;
- }
- break;
- }
- return scope;
-}
-
-/* Helper function for maybe_add_lambda_conv_op; build a CALL_EXPR with
- indicated FN and NARGS, but do not initialize the return type or any of the
- argument slots. */
-
-static tree
-prepare_op_call (tree fn, int nargs)
-{
- tree t;
-
- t = build_vl_exp (CALL_EXPR, nargs + 3);
- CALL_EXPR_FN (t) = fn;
- CALL_EXPR_STATIC_CHAIN (t) = NULL;
-
- return t;
-}
-
-/* Return true iff CALLOP is the op() for a generic lambda. */
-
-bool
-generic_lambda_fn_p (tree callop)
-{
- return (LAMBDA_FUNCTION_P (callop)
- && DECL_TEMPLATE_INFO (callop)
- && PRIMARY_TEMPLATE_P (DECL_TI_TEMPLATE (callop)));
-}
-
-/* If the closure TYPE has a static op(), also add a conversion to function
- pointer. */
-
-void
-maybe_add_lambda_conv_op (tree type)
-{
- bool nested = (cfun != NULL);
- bool nested_def = decl_function_context (TYPE_MAIN_DECL (type));
- tree callop = lambda_function (type);
- tree lam = CLASSTYPE_LAMBDA_EXPR (type);
-
- if (LAMBDA_EXPR_CAPTURE_LIST (lam) != NULL_TREE
- || LAMBDA_EXPR_DEFAULT_CAPTURE_MODE (lam) != CPLD_NONE)
- return;
-
- if (processing_template_decl)
- return;
-
- bool const generic_lambda_p = generic_lambda_fn_p (callop);
-
- if (!generic_lambda_p && undeduced_auto_decl (callop))
- {
- /* If the op() wasn't deduced due to errors, give up. */
- gcc_assert (errorcount || sorrycount);
- return;
- }
-
- /* Non-generic non-capturing lambdas only have a conversion function to
- pointer to function when the trailing requires-clause's constraints are
- satisfied. */
- if (!generic_lambda_p && !constraints_satisfied_p (callop))
- return;
-
- /* Non-template conversion operators are defined directly with build_call_a
- and using DIRECT_ARGVEC for arguments (including 'this'). Templates are
- deferred and the CALL is built in-place. In the case of a deduced return
- call op, the decltype expression, DECLTYPE_CALL, used as a substitute for
- the return type is also built in-place. The arguments of DECLTYPE_CALL in
- the return expression may differ in flags from those in the body CALL. In
- particular, parameter pack expansions are marked PACK_EXPANSION_LOCAL_P in
- the body CALL, but not in DECLTYPE_CALL. */
-
- vec<tree, va_gc> *direct_argvec = 0;
- tree decltype_call = 0, call = 0;
- tree optype = TREE_TYPE (callop);
- tree fn_result = TREE_TYPE (optype);
-
- tree thisarg = build_int_cst (TREE_TYPE (DECL_ARGUMENTS (callop)), 0);
- if (generic_lambda_p)
- {
- ++processing_template_decl;
-
- /* Prepare the dependent member call for the static member function
- '_FUN' and, potentially, prepare another call to be used in a decltype
- return expression for a deduced return call op to allow for simple
- implementation of the conversion operator. */
-
- tree instance = cp_build_fold_indirect_ref (thisarg);
- tree objfn = lookup_template_function (DECL_NAME (callop),
- DECL_TI_ARGS (callop));
- objfn = build_min (COMPONENT_REF, NULL_TREE,
- instance, objfn, NULL_TREE);
- int nargs = list_length (DECL_ARGUMENTS (callop)) - 1;
-
- call = prepare_op_call (objfn, nargs);
- if (type_uses_auto (fn_result))
- decltype_call = prepare_op_call (objfn, nargs);
- }
- else
- {
- direct_argvec = make_tree_vector ();
- direct_argvec->quick_push (thisarg);
- }
-
- /* Copy CALLOP's argument list (as per 'copy_list') as FN_ARGS in order to
- declare the static member function "_FUN" below. For each arg append to
- DIRECT_ARGVEC (for the non-template case) or populate the pre-allocated
- call args (for the template case). If a parameter pack is found, expand
- it, flagging it as PACK_EXPANSION_LOCAL_P for the body call. */
-
- tree fn_args = NULL_TREE;
- {
- int ix = 0;
- tree src = DECL_CHAIN (DECL_ARGUMENTS (callop));
- tree tgt = NULL;
-
- while (src)
- {
- tree new_node = copy_node (src);
- /* We set DECL_CONTEXT of NEW_NODE to the statfn below.
- Notice this is creating a recursive type! */
-
- /* Clear TREE_ADDRESSABLE on thunk arguments. */
- TREE_ADDRESSABLE (new_node) = 0;
-
- if (!fn_args)
- fn_args = tgt = new_node;
- else
- {
- TREE_CHAIN (tgt) = new_node;
- tgt = new_node;
- }
-
- mark_exp_read (tgt);
-
- if (generic_lambda_p)
- {
- tree a = tgt;
- if (DECL_PACK_P (tgt))
- {
- a = make_pack_expansion (a);
- PACK_EXPANSION_LOCAL_P (a) = true;
- }
- CALL_EXPR_ARG (call, ix) = a;
-
- if (decltype_call)
- {
- /* Avoid capturing variables in this context. */
- ++cp_unevaluated_operand;
- CALL_EXPR_ARG (decltype_call, ix) = forward_parm (tgt);
- --cp_unevaluated_operand;
- }
-
- ++ix;
- }
- else
- vec_safe_push (direct_argvec, tgt);
-
- src = TREE_CHAIN (src);
- }
- }
-
- if (generic_lambda_p)
- {
- if (decltype_call)
- {
- fn_result = finish_decltype_type
- (decltype_call, /*id_expression_or_member_access_p=*/false,
- tf_warning_or_error);
- }
- }
- else
- call = build_call_a (callop,
- direct_argvec->length (),
- direct_argvec->address ());
-
- CALL_FROM_THUNK_P (call) = 1;
- SET_EXPR_LOCATION (call, UNKNOWN_LOCATION);
-
- tree stattype = build_function_type (fn_result, FUNCTION_ARG_CHAIN (callop));
- stattype = (cp_build_type_attribute_variant
- (stattype, TYPE_ATTRIBUTES (optype)));
- if (flag_noexcept_type
- && TYPE_NOTHROW_P (TREE_TYPE (callop)))
- stattype = build_exception_variant (stattype, noexcept_true_spec);
-
- if (generic_lambda_p)
- --processing_template_decl;
-
- /* First build up the conversion op. */
-
- tree rettype = build_pointer_type (stattype);
- tree name = make_conv_op_name (rettype);
- tree thistype = cp_build_qualified_type (type, TYPE_QUAL_CONST);
- tree fntype = build_method_type_directly (thistype, rettype, void_list_node);
- /* DR 1722: The conversion function should be noexcept. */
- fntype = build_exception_variant (fntype, noexcept_true_spec);
- tree convfn = build_lang_decl (FUNCTION_DECL, name, fntype);
- SET_DECL_LANGUAGE (convfn, lang_cplusplus);
- tree fn = convfn;
- DECL_SOURCE_LOCATION (fn) = DECL_SOURCE_LOCATION (callop);
- SET_DECL_ALIGN (fn, MINIMUM_METHOD_BOUNDARY);
- grokclassfn (type, fn, NO_SPECIAL);
- set_linkage_according_to_type (type, fn);
- rest_of_decl_compilation (fn, namespace_bindings_p (), at_eof);
- DECL_IN_AGGR_P (fn) = 1;
- DECL_ARTIFICIAL (fn) = 1;
- DECL_NOT_REALLY_EXTERN (fn) = 1;
- DECL_DECLARED_INLINE_P (fn) = 1;
- DECL_DECLARED_CONSTEXPR_P (fn) = DECL_DECLARED_CONSTEXPR_P (callop);
- if (DECL_IMMEDIATE_FUNCTION_P (callop))
- SET_DECL_IMMEDIATE_FUNCTION_P (fn);
- DECL_ARGUMENTS (fn) = build_this_parm (fn, fntype, TYPE_QUAL_CONST);
-
- if (nested_def)
- DECL_INTERFACE_KNOWN (fn) = 1;
-
- if (generic_lambda_p)
- fn = add_inherited_template_parms (fn, DECL_TI_TEMPLATE (callop));
-
- add_method (type, fn, false);
-
- /* Generic thunk code fails for varargs; we'll complain in mark_used if
- the conversion op is used. */
- if (varargs_function_p (callop))
- {
- DECL_DELETED_FN (fn) = 1;
- return;
- }
-
- /* Now build up the thunk to be returned. */
-
- tree statfn = build_lang_decl (FUNCTION_DECL, fun_identifier, stattype);
- SET_DECL_LANGUAGE (statfn, lang_cplusplus);
- fn = statfn;
- DECL_SOURCE_LOCATION (fn) = DECL_SOURCE_LOCATION (callop);
- grokclassfn (type, fn, NO_SPECIAL);
- set_linkage_according_to_type (type, fn);
- rest_of_decl_compilation (fn, namespace_bindings_p (), at_eof);
- DECL_IN_AGGR_P (fn) = 1;
- DECL_ARTIFICIAL (fn) = 1;
- DECL_NOT_REALLY_EXTERN (fn) = 1;
- DECL_DECLARED_INLINE_P (fn) = 1;
- DECL_STATIC_FUNCTION_P (fn) = 1;
- DECL_DECLARED_CONSTEXPR_P (fn) = DECL_DECLARED_CONSTEXPR_P (callop);
- if (DECL_IMMEDIATE_FUNCTION_P (callop))
- SET_DECL_IMMEDIATE_FUNCTION_P (fn);
- DECL_ARGUMENTS (fn) = fn_args;
- for (tree arg = fn_args; arg; arg = DECL_CHAIN (arg))
- {
- /* Avoid duplicate -Wshadow warnings. */
- DECL_NAME (arg) = NULL_TREE;
- DECL_CONTEXT (arg) = fn;
- }
- if (nested_def)
- DECL_INTERFACE_KNOWN (fn) = 1;
-
- if (generic_lambda_p)
- fn = add_inherited_template_parms (fn, DECL_TI_TEMPLATE (callop));
-
- if (flag_sanitize & SANITIZE_NULL)
- /* Don't UBsan this function; we're deliberately calling op() with a null
- object argument. */
- add_no_sanitize_value (fn, SANITIZE_UNDEFINED);
-
- add_method (type, fn, false);
-
- if (nested)
- push_function_context ();
- else
- /* Still increment function_depth so that we don't GC in the
- middle of an expression. */
- ++function_depth;
-
- /* Generate the body of the thunk. */
-
- start_preparsed_function (statfn, NULL_TREE,
- SF_PRE_PARSED | SF_INCLASS_INLINE);
- tree body = begin_function_body ();
- tree compound_stmt = begin_compound_stmt (0);
- if (!generic_lambda_p)
- {
- set_flags_from_callee (call);
- if (MAYBE_CLASS_TYPE_P (TREE_TYPE (call)))
- call = build_cplus_new (TREE_TYPE (call), call, tf_warning_or_error);
- }
- call = convert_from_reference (call);
- finish_return_stmt (call);
-
- finish_compound_stmt (compound_stmt);
- finish_function_body (body);
-
- fn = finish_function (/*inline_p=*/true);
- if (!generic_lambda_p)
- expand_or_defer_fn (fn);
-
- /* Generate the body of the conversion op. */
-
- start_preparsed_function (convfn, NULL_TREE,
- SF_PRE_PARSED | SF_INCLASS_INLINE);
- body = begin_function_body ();
- compound_stmt = begin_compound_stmt (0);
-
- /* decl_needed_p needs to see that it's used. */
- TREE_USED (statfn) = 1;
- finish_return_stmt (decay_conversion (statfn, tf_warning_or_error));
-
- finish_compound_stmt (compound_stmt);
- finish_function_body (body);
-
- fn = finish_function (/*inline_p=*/true);
- if (!generic_lambda_p)
- expand_or_defer_fn (fn);
-
- if (nested)
- pop_function_context ();
- else
- --function_depth;
-}
-
-/* True if FN is the static function "_FUN" that gets returned from the lambda
- conversion operator. */
-
-bool
-lambda_static_thunk_p (tree fn)
-{
- return (fn && TREE_CODE (fn) == FUNCTION_DECL
- && DECL_ARTIFICIAL (fn)
- && DECL_STATIC_FUNCTION_P (fn)
- && LAMBDA_TYPE_P (CP_DECL_CONTEXT (fn)));
-}
-
-bool
-call_from_lambda_thunk_p (tree call)
-{
- return (CALL_FROM_THUNK_P (call)
- && lambda_static_thunk_p (current_function_decl));
-}
-
-/* Returns true iff VAL is a lambda-related declaration which should
- be ignored by unqualified lookup. */
-
-bool
-is_lambda_ignored_entity (tree val)
-{
- /* Look past normal, non-VLA capture proxies. */
- if (is_normal_capture_proxy (val)
- && !variably_modified_type_p (TREE_TYPE (val), NULL_TREE))
- return true;
-
- /* Always ignore lambda fields, their names are only for debugging. */
- if (TREE_CODE (val) == FIELD_DECL
- && CLASSTYPE_LAMBDA_EXPR (DECL_CONTEXT (val)))
- return true;
-
- /* None of the lookups that use qualify_lookup want the op() from the
- lambda; they want the one from the enclosing class. */
- if (tree fns = maybe_get_fns (val))
- if (LAMBDA_FUNCTION_P (OVL_FIRST (fns)))
- return true;
-
- return false;
-}
-
-/* Lambdas that appear in variable initializer or default argument scope
- get that in their mangling, so we need to record it. We might as well
- use the count for function and namespace scopes as well. */
-static GTY(()) tree lambda_scope;
-static GTY(()) int lambda_count;
-struct GTY(()) tree_int
-{
- tree t;
- int i;
-};
-static GTY(()) vec<tree_int, va_gc> *lambda_scope_stack;
-
-void
-start_lambda_scope (tree decl)
-{
- tree_int ti;
- gcc_assert (decl);
- /* Once we're inside a function, we ignore variable scope and just push
- the function again so that popping works properly. */
- if (current_function_decl && TREE_CODE (decl) == VAR_DECL)
- decl = current_function_decl;
- ti.t = lambda_scope;
- ti.i = lambda_count;
- vec_safe_push (lambda_scope_stack, ti);
- if (lambda_scope != decl)
- {
- /* Don't reset the count if we're still in the same function. */
- lambda_scope = decl;
- lambda_count = 0;
- }
-}
-
-void
-record_lambda_scope (tree lambda)
-{
- LAMBDA_EXPR_EXTRA_SCOPE (lambda) = lambda_scope;
- LAMBDA_EXPR_DISCRIMINATOR (lambda) = lambda_count++;
- if (lambda_scope)
- {
- tree closure = LAMBDA_EXPR_CLOSURE (lambda);
- gcc_checking_assert (closure);
- maybe_attach_decl (lambda_scope, TYPE_NAME (closure));
- }
-}
-
-/* This lambda is an instantiation of a lambda in a template default argument
- that got no LAMBDA_EXPR_EXTRA_SCOPE, so this shouldn't either. But we do
- need to use and increment the global count to avoid collisions. */
-
-void
-record_null_lambda_scope (tree lambda)
-{
- if (vec_safe_is_empty (lambda_scope_stack))
- record_lambda_scope (lambda);
- else
- {
- tree_int *p = lambda_scope_stack->begin();
- LAMBDA_EXPR_EXTRA_SCOPE (lambda) = p->t;
- LAMBDA_EXPR_DISCRIMINATOR (lambda) = p->i++;
- }
- gcc_assert (LAMBDA_EXPR_EXTRA_SCOPE (lambda) == NULL_TREE);
-}
-
-void
-finish_lambda_scope (void)
-{
- tree_int *p = &lambda_scope_stack->last ();
- if (lambda_scope != p->t)
- {
- lambda_scope = p->t;
- lambda_count = p->i;
- }
- lambda_scope_stack->pop ();
-}
-
-tree
-start_lambda_function (tree fco, tree lambda_expr)
-{
- /* Let the front end know that we are going to be defining this
- function. */
- start_preparsed_function (fco,
- NULL_TREE,
- SF_PRE_PARSED | SF_INCLASS_INLINE);
-
- tree body = begin_function_body ();
-
- /* Push the proxies for any explicit captures. */
- for (tree cap = LAMBDA_EXPR_CAPTURE_LIST (lambda_expr); cap;
- cap = TREE_CHAIN (cap))
- build_capture_proxy (TREE_PURPOSE (cap), TREE_VALUE (cap));
-
- return body;
-}
-
-/* Subroutine of prune_lambda_captures: CAP is a node in
- LAMBDA_EXPR_CAPTURE_LIST. Return the variable it captures for which we
- might optimize away the capture, or NULL_TREE if there is no such
- variable. */
-
-static tree
-var_to_maybe_prune (tree cap)
-{
- if (LAMBDA_CAPTURE_EXPLICIT_P (cap))
- /* Don't prune explicit captures. */
- return NULL_TREE;
-
- tree mem = TREE_PURPOSE (cap);
- if (!DECL_P (mem) || !DECL_NORMAL_CAPTURE_P (mem))
- /* Packs and init-captures aren't captures of constant vars. */
- return NULL_TREE;
-
- tree init = TREE_VALUE (cap);
- if (is_normal_capture_proxy (init))
- init = DECL_CAPTURED_VARIABLE (init);
- if (decl_constant_var_p (init))
- return init;
-
- return NULL_TREE;
-}
-
-/* walk_tree helper for prune_lambda_captures: Remember which capture proxies
- for constant variables are actually used in the lambda body.
-
- There will always be a DECL_EXPR for the capture proxy; remember it when we
- see it, but replace it with any other use. */
-
-static tree
-mark_const_cap_r (tree *t, int *walk_subtrees, void *data)
-{
- hash_map<tree,tree*> &const_vars = *(hash_map<tree,tree*>*)data;
-
- tree var = NULL_TREE;
- if (TREE_CODE (*t) == DECL_EXPR)
- {
- tree decl = DECL_EXPR_DECL (*t);
- if (is_constant_capture_proxy (decl))
- {
- var = DECL_CAPTURED_VARIABLE (decl);
- *walk_subtrees = 0;
- }
- }
- else if (!location_wrapper_p (*t) /* is_capture_proxy dislikes them. */
- && is_constant_capture_proxy (*t))
- var = DECL_CAPTURED_VARIABLE (*t);
-
- if (var)
- {
- tree *&slot = const_vars.get_or_insert (var);
- if (!slot || VAR_P (*t))
- slot = t;
- }
-
- return NULL_TREE;
-}
-
-/* We're at the end of processing a lambda; go back and remove any captures of
- constant variables for which we've folded away all uses. */
-
-static void
-prune_lambda_captures (tree body)
-{
- tree lam = current_lambda_expr ();
- if (!LAMBDA_EXPR_CAPTURE_OPTIMIZED (lam))
- /* No uses were optimized away. */
- return;
- if (LAMBDA_EXPR_DEFAULT_CAPTURE_MODE (lam) == CPLD_NONE)
- /* No default captures, and we don't prune explicit captures. */
- return;
-
- hash_map<tree,tree*> const_vars;
-
- cp_walk_tree_without_duplicates (&body, mark_const_cap_r, &const_vars);
-
- tree *fieldp = &TYPE_FIELDS (LAMBDA_EXPR_CLOSURE (lam));
- for (tree *capp = &LAMBDA_EXPR_CAPTURE_LIST (lam); *capp; )
- {
- tree cap = *capp;
- if (tree var = var_to_maybe_prune (cap))
- {
- tree **use = const_vars.get (var);
- if (use && TREE_CODE (**use) == DECL_EXPR)
- {
- /* All uses of this capture were folded away, leaving only the
- proxy declaration. */
-
- /* Splice the capture out of LAMBDA_EXPR_CAPTURE_LIST. */
- *capp = TREE_CHAIN (cap);
-
- /* And out of TYPE_FIELDS. */
- tree field = TREE_PURPOSE (cap);
- while (*fieldp != field)
- fieldp = &DECL_CHAIN (*fieldp);
- *fieldp = DECL_CHAIN (*fieldp);
-
- /* And remove the capture proxy declaration. */
- **use = void_node;
- continue;
- }
- }
-
- capp = &TREE_CHAIN (cap);
- }
-}
-
-void
-finish_lambda_function (tree body)
-{
- finish_function_body (body);
-
- prune_lambda_captures (body);
-
- /* Finish the function and generate code for it if necessary. */
- tree fn = finish_function (/*inline_p=*/true);
-
- /* Only expand if the call op is not a template. */
- if (!DECL_TEMPLATE_INFO (fn))
- expand_or_defer_fn (fn);
-}
-
-#include "gt-cp-lambda.h"