mirror of
https://git.FreeBSD.org/src.git
synced 2024-12-26 11:47:31 +00:00
c6d2f3514a
branch as of May 26th, 2000. [these are changes March 31 - May 24th]
9877 lines
286 KiB
C
9877 lines
286 KiB
C
/* Handle parameterized types (templates) for GNU C++.
|
||
Copyright (C) 1992, 93-97, 1998, 1999 Free Software Foundation, Inc.
|
||
Written by Ken Raeburn (raeburn@cygnus.com) while at Watchmaker Computing.
|
||
Rewritten by Jason Merrill (jason@cygnus.com).
|
||
|
||
This file is part of GNU CC.
|
||
|
||
GNU CC 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 2, or (at your option)
|
||
any later version.
|
||
|
||
GNU CC 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 GNU CC; see the file COPYING. If not, write to
|
||
the Free Software Foundation, 59 Temple Place - Suite 330,
|
||
Boston, MA 02111-1307, USA. */
|
||
|
||
/* Known bugs or deficiencies include:
|
||
|
||
all methods must be provided in header files; can't use a source
|
||
file that contains only the method templates and "just win". */
|
||
|
||
#include "config.h"
|
||
#include "system.h"
|
||
#include "obstack.h"
|
||
|
||
#include "tree.h"
|
||
#include "flags.h"
|
||
#include "cp-tree.h"
|
||
#include "decl.h"
|
||
#include "parse.h"
|
||
#include "lex.h"
|
||
#include "output.h"
|
||
#include "defaults.h"
|
||
#include "except.h"
|
||
#include "toplev.h"
|
||
#include "rtl.h"
|
||
#include "varray.h"
|
||
|
||
/* The type of functions taking a tree, and some additional data, and
|
||
returning an int. */
|
||
typedef int (*tree_fn_t) PROTO((tree, void*));
|
||
|
||
extern struct obstack permanent_obstack;
|
||
|
||
extern int lineno;
|
||
extern char *input_filename;
|
||
|
||
tree current_template_parms;
|
||
HOST_WIDE_INT processing_template_decl;
|
||
|
||
/* The PENDING_TEMPLATES is a TREE_LIST of templates whose
|
||
instantiations have been deferred, either because their definitions
|
||
were not yet available, or because we were putting off doing the
|
||
work. The TREE_PURPOSE of each entry is a SRCLOC indicating where
|
||
the instantiate request occurred; the TREE_VALUE is a either a DECL
|
||
(for a function or static data member), or a TYPE (for a class)
|
||
indicating what we are hoping to instantiate. */
|
||
static tree pending_templates;
|
||
static tree *template_tail = &pending_templates;
|
||
|
||
static tree maybe_templates;
|
||
static tree *maybe_template_tail = &maybe_templates;
|
||
|
||
int minimal_parse_mode;
|
||
|
||
int processing_specialization;
|
||
int processing_explicit_instantiation;
|
||
int processing_template_parmlist;
|
||
static int template_header_count;
|
||
|
||
static tree saved_trees;
|
||
static varray_type inline_parm_levels;
|
||
static size_t inline_parm_levels_used;
|
||
|
||
#define obstack_chunk_alloc xmalloc
|
||
#define obstack_chunk_free free
|
||
|
||
#define UNIFY_ALLOW_NONE 0
|
||
#define UNIFY_ALLOW_MORE_CV_QUAL 1
|
||
#define UNIFY_ALLOW_LESS_CV_QUAL 2
|
||
#define UNIFY_ALLOW_DERIVED 4
|
||
#define UNIFY_ALLOW_INTEGER 8
|
||
|
||
#define GTB_VIA_VIRTUAL 1 /* The base class we are examining is
|
||
virtual, or a base class of a virtual
|
||
base. */
|
||
#define GTB_IGNORE_TYPE 2 /* We don't need to try to unify the current
|
||
type with the desired type. */
|
||
|
||
static int resolve_overloaded_unification PROTO((tree, tree, tree, tree,
|
||
unification_kind_t, int));
|
||
static int try_one_overload PROTO((tree, tree, tree, tree, tree,
|
||
unification_kind_t, int));
|
||
static int unify PROTO((tree, tree, tree, tree, int));
|
||
static void add_pending_template PROTO((tree));
|
||
static int push_tinst_level PROTO((tree));
|
||
static tree classtype_mangled_name PROTO((tree));
|
||
static char *mangle_class_name_for_template PROTO((char *, tree, tree));
|
||
static tree tsubst_expr_values PROTO((tree, tree));
|
||
static int list_eq PROTO((tree, tree));
|
||
static tree get_class_bindings PROTO((tree, tree, tree));
|
||
static tree coerce_template_parms PROTO((tree, tree, tree, int, int));
|
||
static void tsubst_enum PROTO((tree, tree, tree));
|
||
static tree add_to_template_args PROTO((tree, tree));
|
||
static tree add_outermost_template_args PROTO((tree, tree));
|
||
static void maybe_adjust_types_for_deduction PROTO((unification_kind_t, tree*,
|
||
tree*));
|
||
static int type_unification_real PROTO((tree, tree, tree, tree,
|
||
int, unification_kind_t, int));
|
||
static void note_template_header PROTO((int));
|
||
static tree maybe_fold_nontype_arg PROTO((tree));
|
||
static tree convert_nontype_argument PROTO((tree, tree));
|
||
static tree convert_template_argument PROTO ((tree, tree, tree, int,
|
||
int , tree));
|
||
static tree get_bindings_overload PROTO((tree, tree, tree));
|
||
static int for_each_template_parm PROTO((tree, tree_fn_t, void*));
|
||
static tree build_template_parm_index PROTO((int, int, int, tree, tree));
|
||
static int inline_needs_template_parms PROTO((tree));
|
||
static void push_inline_template_parms_recursive PROTO((tree, int));
|
||
static tree retrieve_specialization PROTO((tree, tree));
|
||
static tree register_specialization PROTO((tree, tree, tree));
|
||
static int unregister_specialization PROTO((tree, tree));
|
||
static tree reduce_template_parm_level PROTO((tree, tree, int));
|
||
static tree build_template_decl PROTO((tree, tree));
|
||
static int mark_template_parm PROTO((tree, void *));
|
||
static tree tsubst_friend_function PROTO((tree, tree));
|
||
static tree tsubst_friend_class PROTO((tree, tree));
|
||
static tree get_bindings_real PROTO((tree, tree, tree, int));
|
||
static int template_decl_level PROTO((tree));
|
||
static tree maybe_get_template_decl_from_type_decl PROTO((tree));
|
||
static int check_cv_quals_for_unify PROTO((int, tree, tree));
|
||
static tree tsubst_template_arg_vector PROTO((tree, tree, int));
|
||
static tree tsubst_template_parms PROTO((tree, tree, int));
|
||
static void regenerate_decl_from_template PROTO((tree, tree));
|
||
static tree most_specialized PROTO((tree, tree, tree));
|
||
static tree most_specialized_class PROTO((tree, tree));
|
||
static tree most_general_template PROTO((tree));
|
||
static void set_mangled_name_for_template_decl PROTO((tree));
|
||
static int template_class_depth_real PROTO((tree, int));
|
||
static tree tsubst_aggr_type PROTO((tree, tree, int, tree, int));
|
||
static tree tsubst_decl PROTO((tree, tree, tree, tree));
|
||
static tree tsubst_arg_types PROTO((tree, tree, int, tree));
|
||
static tree tsubst_function_type PROTO((tree, tree, int, tree));
|
||
static void check_specialization_scope PROTO((void));
|
||
static tree process_partial_specialization PROTO((tree));
|
||
static void set_current_access_from_decl PROTO((tree));
|
||
static void check_default_tmpl_args PROTO((tree, tree, int, int));
|
||
static tree tsubst_call_declarator_parms PROTO((tree, tree, int, tree));
|
||
static tree get_template_base_recursive PROTO((tree, tree,
|
||
tree, tree, tree, int));
|
||
static tree get_template_base PROTO((tree, tree, tree, tree));
|
||
static tree try_class_unification PROTO((tree, tree, tree, tree));
|
||
static int coerce_template_template_parms PROTO((tree, tree, int,
|
||
tree, tree));
|
||
static tree determine_specialization PROTO((tree, tree, tree *, int));
|
||
static int template_args_equal PROTO((tree, tree));
|
||
static void print_template_context PROTO((int));
|
||
static int has_pvbases_p PROTO((tree, tree));
|
||
|
||
/* We use TREE_VECs to hold template arguments. If there is only one
|
||
level of template arguments, then the TREE_VEC contains the
|
||
arguments directly. If there is more than one level of template
|
||
arguments, then each entry in the TREE_VEC is itself a TREE_VEC,
|
||
containing the template arguments for a single level. The first
|
||
entry in the outer TREE_VEC is the outermost level of template
|
||
parameters; the last is the innermost.
|
||
|
||
It is incorrect to ever form a template argument vector containing
|
||
only one level of arguments, but which is a TREE_VEC containing as
|
||
its only entry the TREE_VEC for that level. */
|
||
|
||
/* Non-zero if the template arguments is actually a vector of vectors,
|
||
rather than just a vector. */
|
||
#define TMPL_ARGS_HAVE_MULTIPLE_LEVELS(NODE) \
|
||
(NODE != NULL_TREE \
|
||
&& TREE_CODE (NODE) == TREE_VEC \
|
||
&& TREE_VEC_LENGTH (NODE) > 0 \
|
||
&& TREE_VEC_ELT (NODE, 0) != NULL_TREE \
|
||
&& TREE_CODE (TREE_VEC_ELT (NODE, 0)) == TREE_VEC)
|
||
|
||
/* The depth of a template argument vector. When called directly by
|
||
the parser, we use a TREE_LIST rather than a TREE_VEC to represent
|
||
template arguments. In fact, we may even see NULL_TREE if there
|
||
are no template arguments. In both of those cases, there is only
|
||
one level of template arguments. */
|
||
#define TMPL_ARGS_DEPTH(NODE) \
|
||
(TMPL_ARGS_HAVE_MULTIPLE_LEVELS (NODE) ? TREE_VEC_LENGTH (NODE) : 1)
|
||
|
||
/* The LEVELth level of the template ARGS. Note that template
|
||
parameter levels are indexed from 1, not from 0. */
|
||
#define TMPL_ARGS_LEVEL(ARGS, LEVEL) \
|
||
(TMPL_ARGS_HAVE_MULTIPLE_LEVELS (ARGS) \
|
||
? TREE_VEC_ELT ((ARGS), (LEVEL) - 1) : ARGS)
|
||
|
||
/* Set the LEVELth level of the template ARGS to VAL. This macro does
|
||
not work with single-level argument vectors. */
|
||
#define SET_TMPL_ARGS_LEVEL(ARGS, LEVEL, VAL) \
|
||
(TREE_VEC_ELT ((ARGS), (LEVEL) - 1) = (VAL))
|
||
|
||
/* Accesses the IDXth parameter in the LEVELth level of the ARGS. */
|
||
#define TMPL_ARG(ARGS, LEVEL, IDX) \
|
||
(TREE_VEC_ELT (TMPL_ARGS_LEVEL (ARGS, LEVEL), IDX))
|
||
|
||
/* Set the IDXth element in the LEVELth level of ARGS to VAL. This
|
||
macro does not work with single-level argument vectors. */
|
||
#define SET_TMPL_ARG(ARGS, LEVEL, IDX, VAL) \
|
||
(TREE_VEC_ELT (TREE_VEC_ELT ((ARGS), (LEVEL) - 1), (IDX)) = (VAL))
|
||
|
||
/* Given a single level of template arguments in NODE, return the
|
||
number of arguments. */
|
||
#define NUM_TMPL_ARGS(NODE) \
|
||
((NODE) == NULL_TREE ? 0 \
|
||
: (TREE_CODE (NODE) == TREE_VEC \
|
||
? TREE_VEC_LENGTH (NODE) : list_length (NODE)))
|
||
|
||
/* The number of levels of template parameters given by NODE. */
|
||
#define TMPL_PARMS_DEPTH(NODE) \
|
||
(TREE_INT_CST_HIGH (TREE_PURPOSE (NODE)))
|
||
|
||
/* Do any processing required when DECL (a member template declaration
|
||
using TEMPLATE_PARAMETERS as its innermost parameter list) is
|
||
finished. Returns the TEMPLATE_DECL corresponding to DECL, unless
|
||
it is a specialization, in which case the DECL itself is returned. */
|
||
|
||
tree
|
||
finish_member_template_decl (decl)
|
||
tree decl;
|
||
{
|
||
if (decl == NULL_TREE || decl == void_type_node)
|
||
return NULL_TREE;
|
||
else if (decl == error_mark_node)
|
||
/* By returning NULL_TREE, the parser will just ignore this
|
||
declaration. We have already issued the error. */
|
||
return NULL_TREE;
|
||
else if (TREE_CODE (decl) == TREE_LIST)
|
||
{
|
||
/* Assume that the class is the only declspec. */
|
||
decl = TREE_VALUE (decl);
|
||
if (IS_AGGR_TYPE (decl) && CLASSTYPE_TEMPLATE_INFO (decl)
|
||
&& ! CLASSTYPE_TEMPLATE_SPECIALIZATION (decl))
|
||
{
|
||
tree tmpl = CLASSTYPE_TI_TEMPLATE (decl);
|
||
check_member_template (tmpl);
|
||
return tmpl;
|
||
}
|
||
return NULL_TREE;
|
||
}
|
||
else if (DECL_TEMPLATE_INFO (decl))
|
||
{
|
||
if (!DECL_TEMPLATE_SPECIALIZATION (decl))
|
||
{
|
||
check_member_template (DECL_TI_TEMPLATE (decl));
|
||
return DECL_TI_TEMPLATE (decl);
|
||
}
|
||
else
|
||
return decl;
|
||
}
|
||
else
|
||
cp_error ("invalid member template declaration `%D'", decl);
|
||
|
||
return error_mark_node;
|
||
}
|
||
|
||
/* Returns the template nesting level of the indicated class TYPE.
|
||
|
||
For example, in:
|
||
template <class T>
|
||
struct A
|
||
{
|
||
template <class U>
|
||
struct B {};
|
||
};
|
||
|
||
A<T>::B<U> has depth two, while A<T> has depth one.
|
||
Both A<T>::B<int> and A<int>::B<U> have depth one, if
|
||
COUNT_SPECIALIZATIONS is 0 or if they are instantiations, not
|
||
specializations.
|
||
|
||
This function is guaranteed to return 0 if passed NULL_TREE so
|
||
that, for example, `template_class_depth (current_class_type)' is
|
||
always safe. */
|
||
|
||
static int
|
||
template_class_depth_real (type, count_specializations)
|
||
tree type;
|
||
int count_specializations;
|
||
{
|
||
int depth;
|
||
|
||
for (depth = 0;
|
||
type && TREE_CODE (type) != NAMESPACE_DECL;
|
||
type = (TREE_CODE (type) == FUNCTION_DECL)
|
||
? DECL_REAL_CONTEXT (type) : TYPE_CONTEXT (type))
|
||
{
|
||
if (TREE_CODE (type) != FUNCTION_DECL)
|
||
{
|
||
if (CLASSTYPE_TEMPLATE_INFO (type)
|
||
&& PRIMARY_TEMPLATE_P (CLASSTYPE_TI_TEMPLATE (type))
|
||
&& ((count_specializations
|
||
&& CLASSTYPE_TEMPLATE_SPECIALIZATION (type))
|
||
|| uses_template_parms (CLASSTYPE_TI_ARGS (type))))
|
||
++depth;
|
||
}
|
||
else
|
||
{
|
||
if (DECL_TEMPLATE_INFO (type)
|
||
&& PRIMARY_TEMPLATE_P (DECL_TI_TEMPLATE (type))
|
||
&& ((count_specializations
|
||
&& DECL_TEMPLATE_SPECIALIZATION (type))
|
||
|| uses_template_parms (DECL_TI_ARGS (type))))
|
||
++depth;
|
||
}
|
||
}
|
||
|
||
return depth;
|
||
}
|
||
|
||
/* Returns the template nesting level of the indicated class TYPE.
|
||
Like template_class_depth_real, but instantiations do not count in
|
||
the depth. */
|
||
|
||
int
|
||
template_class_depth (type)
|
||
tree type;
|
||
{
|
||
return template_class_depth_real (type, /*count_specializations=*/0);
|
||
}
|
||
|
||
/* Returns 1 if processing DECL as part of do_pending_inlines
|
||
needs us to push template parms. */
|
||
|
||
static int
|
||
inline_needs_template_parms (decl)
|
||
tree decl;
|
||
{
|
||
if (! DECL_TEMPLATE_INFO (decl))
|
||
return 0;
|
||
|
||
return (TMPL_PARMS_DEPTH (DECL_TEMPLATE_PARMS (most_general_template (decl)))
|
||
> (processing_template_decl + DECL_TEMPLATE_SPECIALIZATION (decl)));
|
||
}
|
||
|
||
/* Subroutine of maybe_begin_member_template_processing.
|
||
Push the template parms in PARMS, starting from LEVELS steps into the
|
||
chain, and ending at the beginning, since template parms are listed
|
||
innermost first. */
|
||
|
||
static void
|
||
push_inline_template_parms_recursive (parmlist, levels)
|
||
tree parmlist;
|
||
int levels;
|
||
{
|
||
tree parms = TREE_VALUE (parmlist);
|
||
int i;
|
||
|
||
if (levels > 1)
|
||
push_inline_template_parms_recursive (TREE_CHAIN (parmlist), levels - 1);
|
||
|
||
++processing_template_decl;
|
||
current_template_parms
|
||
= tree_cons (build_int_2 (0, processing_template_decl),
|
||
parms, current_template_parms);
|
||
TEMPLATE_PARMS_FOR_INLINE (current_template_parms) = 1;
|
||
|
||
pushlevel (0);
|
||
for (i = 0; i < TREE_VEC_LENGTH (parms); ++i)
|
||
{
|
||
tree parm = TREE_VALUE (TREE_VEC_ELT (parms, i));
|
||
my_friendly_assert (TREE_CODE_CLASS (TREE_CODE (parm)) == 'd', 0);
|
||
|
||
switch (TREE_CODE (parm))
|
||
{
|
||
case TYPE_DECL:
|
||
case TEMPLATE_DECL:
|
||
pushdecl (parm);
|
||
break;
|
||
|
||
case PARM_DECL:
|
||
{
|
||
/* Make a CONST_DECL as is done in process_template_parm.
|
||
It is ugly that we recreate this here; the original
|
||
version built in process_template_parm is no longer
|
||
available. */
|
||
tree decl = build_decl (CONST_DECL, DECL_NAME (parm),
|
||
TREE_TYPE (parm));
|
||
SET_DECL_ARTIFICIAL (decl);
|
||
DECL_INITIAL (decl) = DECL_INITIAL (parm);
|
||
DECL_TEMPLATE_PARM_P (decl) = 1;
|
||
pushdecl (decl);
|
||
}
|
||
break;
|
||
|
||
default:
|
||
my_friendly_abort (0);
|
||
}
|
||
}
|
||
}
|
||
|
||
/* Restore the template parameter context for a member template or
|
||
a friend template defined in a class definition. */
|
||
|
||
void
|
||
maybe_begin_member_template_processing (decl)
|
||
tree decl;
|
||
{
|
||
tree parms;
|
||
int levels = 0;
|
||
|
||
if (inline_needs_template_parms (decl))
|
||
{
|
||
parms = DECL_TEMPLATE_PARMS (most_general_template (decl));
|
||
levels = TMPL_PARMS_DEPTH (parms) - processing_template_decl;
|
||
|
||
if (DECL_TEMPLATE_SPECIALIZATION (decl))
|
||
{
|
||
--levels;
|
||
parms = TREE_CHAIN (parms);
|
||
}
|
||
|
||
push_inline_template_parms_recursive (parms, levels);
|
||
}
|
||
|
||
/* Remember how many levels of template parameters we pushed so that
|
||
we can pop them later. */
|
||
if (!inline_parm_levels)
|
||
VARRAY_INT_INIT (inline_parm_levels, 4, "inline_parm_levels");
|
||
if (inline_parm_levels_used == inline_parm_levels->num_elements)
|
||
VARRAY_GROW (inline_parm_levels, 2 * inline_parm_levels_used);
|
||
VARRAY_INT (inline_parm_levels, inline_parm_levels_used) = levels;
|
||
++inline_parm_levels_used;
|
||
}
|
||
|
||
/* Undo the effects of begin_member_template_processing. */
|
||
|
||
void
|
||
maybe_end_member_template_processing ()
|
||
{
|
||
int i;
|
||
|
||
if (!inline_parm_levels_used)
|
||
return;
|
||
|
||
--inline_parm_levels_used;
|
||
for (i = 0;
|
||
i < VARRAY_INT (inline_parm_levels, inline_parm_levels_used);
|
||
++i)
|
||
{
|
||
--processing_template_decl;
|
||
current_template_parms = TREE_CHAIN (current_template_parms);
|
||
poplevel (0, 0, 0);
|
||
}
|
||
}
|
||
|
||
/* Returns non-zero iff T is a member template function. We must be
|
||
careful as in
|
||
|
||
template <class T> class C { void f(); }
|
||
|
||
Here, f is a template function, and a member, but not a member
|
||
template. This function does not concern itself with the origin of
|
||
T, only its present state. So if we have
|
||
|
||
template <class T> class C { template <class U> void f(U); }
|
||
|
||
then neither C<int>::f<char> nor C<T>::f<double> is considered
|
||
to be a member template. But, `template <class U> void
|
||
C<int>::f(U)' is considered a member template. */
|
||
|
||
int
|
||
is_member_template (t)
|
||
tree t;
|
||
{
|
||
if (!DECL_FUNCTION_TEMPLATE_P (t))
|
||
/* Anything that isn't a function or a template function is
|
||
certainly not a member template. */
|
||
return 0;
|
||
|
||
/* A local class can't have member templates. */
|
||
if (hack_decl_function_context (t))
|
||
return 0;
|
||
|
||
return (DECL_FUNCTION_MEMBER_P (DECL_TEMPLATE_RESULT (t))
|
||
/* If there are more levels of template parameters than
|
||
there are template classes surrounding the declaration,
|
||
then we have a member template. */
|
||
&& (TMPL_PARMS_DEPTH (DECL_TEMPLATE_PARMS (t)) >
|
||
template_class_depth (DECL_CLASS_CONTEXT (t))));
|
||
}
|
||
|
||
#if 0 /* UNUSED */
|
||
/* Returns non-zero iff T is a member template class. See
|
||
is_member_template for a description of what precisely constitutes
|
||
a member template. */
|
||
|
||
int
|
||
is_member_template_class (t)
|
||
tree t;
|
||
{
|
||
if (!DECL_CLASS_TEMPLATE_P (t))
|
||
/* Anything that isn't a class template, is certainly not a member
|
||
template. */
|
||
return 0;
|
||
|
||
if (!DECL_CLASS_SCOPE_P (t))
|
||
/* Anything whose context isn't a class type is surely not a
|
||
member template. */
|
||
return 0;
|
||
|
||
/* If there are more levels of template parameters than there are
|
||
template classes surrounding the declaration, then we have a
|
||
member template. */
|
||
return (TMPL_PARMS_DEPTH (DECL_TEMPLATE_PARMS (t)) >
|
||
template_class_depth (DECL_CONTEXT (t)));
|
||
}
|
||
#endif
|
||
|
||
/* Return a new template argument vector which contains all of ARGS,
|
||
but has as its innermost set of arguments the EXTRA_ARGS. The
|
||
resulting vector will be built on a temporary obstack, and so must
|
||
be explicitly copied to the permanent obstack, if required. */
|
||
|
||
static tree
|
||
add_to_template_args (args, extra_args)
|
||
tree args;
|
||
tree extra_args;
|
||
{
|
||
tree new_args;
|
||
int extra_depth;
|
||
int i;
|
||
int j;
|
||
|
||
extra_depth = TMPL_ARGS_DEPTH (extra_args);
|
||
new_args = make_temp_vec (TMPL_ARGS_DEPTH (args) + extra_depth);
|
||
|
||
for (i = 1; i <= TMPL_ARGS_DEPTH (args); ++i)
|
||
SET_TMPL_ARGS_LEVEL (new_args, i, TMPL_ARGS_LEVEL (args, i));
|
||
|
||
for (j = 1; j <= extra_depth; ++j, ++i)
|
||
SET_TMPL_ARGS_LEVEL (new_args, i, TMPL_ARGS_LEVEL (extra_args, j));
|
||
|
||
return new_args;
|
||
}
|
||
|
||
/* Like add_to_template_args, but only the outermost ARGS are added to
|
||
the EXTRA_ARGS. In particular, all but TMPL_ARGS_DEPTH
|
||
(EXTRA_ARGS) levels are added. This function is used to combine
|
||
the template arguments from a partial instantiation with the
|
||
template arguments used to attain the full instantiation from the
|
||
partial instantiation. */
|
||
|
||
static tree
|
||
add_outermost_template_args (args, extra_args)
|
||
tree args;
|
||
tree extra_args;
|
||
{
|
||
tree new_args;
|
||
|
||
/* If there are more levels of EXTRA_ARGS than there are ARGS,
|
||
something very fishy is going on. */
|
||
my_friendly_assert (TMPL_ARGS_DEPTH (args) >= TMPL_ARGS_DEPTH (extra_args),
|
||
0);
|
||
|
||
/* If *all* the new arguments will be the EXTRA_ARGS, just return
|
||
them. */
|
||
if (TMPL_ARGS_DEPTH (args) == TMPL_ARGS_DEPTH (extra_args))
|
||
return extra_args;
|
||
|
||
/* For the moment, we make ARGS look like it contains fewer levels. */
|
||
TREE_VEC_LENGTH (args) -= TMPL_ARGS_DEPTH (extra_args);
|
||
|
||
new_args = add_to_template_args (args, extra_args);
|
||
|
||
/* Now, we restore ARGS to its full dimensions. */
|
||
TREE_VEC_LENGTH (args) += TMPL_ARGS_DEPTH (extra_args);
|
||
|
||
return new_args;
|
||
}
|
||
|
||
/* We've got a template header coming up; push to a new level for storing
|
||
the parms. */
|
||
|
||
void
|
||
begin_template_parm_list ()
|
||
{
|
||
/* We use a non-tag-transparent scope here, which causes pushtag to
|
||
put tags in this scope, rather than in the enclosing class or
|
||
namespace scope. This is the right thing, since we want
|
||
TEMPLATE_DECLS, and not TYPE_DECLS for template classes. For a
|
||
global template class, push_template_decl handles putting the
|
||
TEMPLATE_DECL into top-level scope. For a nested template class,
|
||
e.g.:
|
||
|
||
template <class T> struct S1 {
|
||
template <class T> struct S2 {};
|
||
};
|
||
|
||
pushtag contains special code to call pushdecl_with_scope on the
|
||
TEMPLATE_DECL for S2. */
|
||
pushlevel (0);
|
||
declare_pseudo_global_level ();
|
||
++processing_template_decl;
|
||
++processing_template_parmlist;
|
||
note_template_header (0);
|
||
}
|
||
|
||
/* This routine is called when a specialization is declared. If it is
|
||
illegal to declare a specialization here, an error is reported. */
|
||
|
||
static void
|
||
check_specialization_scope ()
|
||
{
|
||
tree scope = current_scope ();
|
||
|
||
/* [temp.expl.spec]
|
||
|
||
An explicit specialization shall be declared in the namespace of
|
||
which the template is a member, or, for member templates, in the
|
||
namespace of which the enclosing class or enclosing class
|
||
template is a member. An explicit specialization of a member
|
||
function, member class or static data member of a class template
|
||
shall be declared in the namespace of which the class template
|
||
is a member. */
|
||
if (scope && TREE_CODE (scope) != NAMESPACE_DECL)
|
||
cp_error ("explicit specialization in non-namespace scope `%D'",
|
||
scope);
|
||
|
||
/* [temp.expl.spec]
|
||
|
||
In an explicit specialization declaration for a member of a class
|
||
template or a member template that appears in namespace scope,
|
||
the member template and some of its enclosing class templates may
|
||
remain unspecialized, except that the declaration shall not
|
||
explicitly specialize a class member template if its enclosing
|
||
class templates are not explicitly specialized as well. */
|
||
if (current_template_parms)
|
||
cp_error ("enclosing class templates are not explicitly specialized");
|
||
}
|
||
|
||
/* We've just seen template <>. */
|
||
|
||
void
|
||
begin_specialization ()
|
||
{
|
||
note_template_header (1);
|
||
check_specialization_scope ();
|
||
}
|
||
|
||
/* Called at then end of processing a declaration preceeded by
|
||
template<>. */
|
||
|
||
void
|
||
end_specialization ()
|
||
{
|
||
reset_specialization ();
|
||
}
|
||
|
||
/* Any template <>'s that we have seen thus far are not referring to a
|
||
function specialization. */
|
||
|
||
void
|
||
reset_specialization ()
|
||
{
|
||
processing_specialization = 0;
|
||
template_header_count = 0;
|
||
}
|
||
|
||
/* We've just seen a template header. If SPECIALIZATION is non-zero,
|
||
it was of the form template <>. */
|
||
|
||
static void
|
||
note_template_header (specialization)
|
||
int specialization;
|
||
{
|
||
processing_specialization = specialization;
|
||
template_header_count++;
|
||
}
|
||
|
||
/* We're beginning an explicit instantiation. */
|
||
|
||
void
|
||
begin_explicit_instantiation ()
|
||
{
|
||
++processing_explicit_instantiation;
|
||
}
|
||
|
||
|
||
void
|
||
end_explicit_instantiation ()
|
||
{
|
||
my_friendly_assert(processing_explicit_instantiation > 0, 0);
|
||
--processing_explicit_instantiation;
|
||
}
|
||
|
||
/* The TYPE is being declared. If it is a template type, that means it
|
||
is a partial specialization. Do appropriate error-checking. */
|
||
|
||
void
|
||
maybe_process_partial_specialization (type)
|
||
tree type;
|
||
{
|
||
if (IS_AGGR_TYPE (type) && CLASSTYPE_USE_TEMPLATE (type))
|
||
{
|
||
if (CLASSTYPE_IMPLICIT_INSTANTIATION (type)
|
||
&& TYPE_SIZE (type) == NULL_TREE)
|
||
{
|
||
if (current_namespace
|
||
!= decl_namespace_context (CLASSTYPE_TI_TEMPLATE (type)))
|
||
{
|
||
cp_pedwarn ("specializing `%#T' in different namespace", type);
|
||
cp_pedwarn_at (" from definition of `%#D'",
|
||
CLASSTYPE_TI_TEMPLATE (type));
|
||
}
|
||
SET_CLASSTYPE_TEMPLATE_SPECIALIZATION (type);
|
||
if (processing_template_decl)
|
||
push_template_decl (TYPE_MAIN_DECL (type));
|
||
}
|
||
else if (CLASSTYPE_TEMPLATE_INSTANTIATION (type))
|
||
cp_error ("specialization of `%T' after instantiation", type);
|
||
}
|
||
else if (processing_specialization)
|
||
cp_error ("explicit specialization of non-template `%T'", type);
|
||
}
|
||
|
||
/* Retrieve the specialization (in the sense of [temp.spec] - a
|
||
specialization is either an instantiation or an explicit
|
||
specialization) of TMPL for the given template ARGS. If there is
|
||
no such specialization, return NULL_TREE. The ARGS are a vector of
|
||
arguments, or a vector of vectors of arguments, in the case of
|
||
templates with more than one level of parameters. */
|
||
|
||
static tree
|
||
retrieve_specialization (tmpl, args)
|
||
tree tmpl;
|
||
tree args;
|
||
{
|
||
tree s;
|
||
|
||
my_friendly_assert (TREE_CODE (tmpl) == TEMPLATE_DECL, 0);
|
||
|
||
/* There should be as many levels of arguments as there are
|
||
levels of parameters. */
|
||
my_friendly_assert (TMPL_ARGS_DEPTH (args)
|
||
== TMPL_PARMS_DEPTH (DECL_TEMPLATE_PARMS (tmpl)),
|
||
0);
|
||
|
||
for (s = DECL_TEMPLATE_SPECIALIZATIONS (tmpl);
|
||
s != NULL_TREE;
|
||
s = TREE_CHAIN (s))
|
||
if (comp_template_args (TREE_PURPOSE (s), args))
|
||
return TREE_VALUE (s);
|
||
|
||
return NULL_TREE;
|
||
}
|
||
|
||
/* Returns non-zero iff DECL is a specialization of TMPL. */
|
||
|
||
int
|
||
is_specialization_of (decl, tmpl)
|
||
tree decl;
|
||
tree tmpl;
|
||
{
|
||
tree t;
|
||
|
||
if (TREE_CODE (decl) == FUNCTION_DECL)
|
||
{
|
||
for (t = decl;
|
||
t != NULL_TREE;
|
||
t = DECL_TEMPLATE_INFO (t) ? DECL_TI_TEMPLATE (t) : NULL_TREE)
|
||
if (t == tmpl)
|
||
return 1;
|
||
}
|
||
else
|
||
{
|
||
my_friendly_assert (TREE_CODE (decl) == TYPE_DECL, 0);
|
||
|
||
for (t = TREE_TYPE (decl);
|
||
t != NULL_TREE;
|
||
t = CLASSTYPE_USE_TEMPLATE (t)
|
||
? TREE_TYPE (CLASSTYPE_TI_TEMPLATE (t)) : NULL_TREE)
|
||
if (same_type_p (TYPE_MAIN_VARIANT (t),
|
||
TYPE_MAIN_VARIANT (TREE_TYPE (tmpl))))
|
||
return 1;
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
/* Register the specialization SPEC as a specialization of TMPL with
|
||
the indicated ARGS. Returns SPEC, or an equivalent prior
|
||
declaration, if available. */
|
||
|
||
static tree
|
||
register_specialization (spec, tmpl, args)
|
||
tree spec;
|
||
tree tmpl;
|
||
tree args;
|
||
{
|
||
tree s;
|
||
|
||
my_friendly_assert (TREE_CODE (tmpl) == TEMPLATE_DECL, 0);
|
||
|
||
if (TREE_CODE (spec) == FUNCTION_DECL
|
||
&& uses_template_parms (DECL_TI_ARGS (spec)))
|
||
/* This is the FUNCTION_DECL for a partial instantiation. Don't
|
||
register it; we want the corresponding TEMPLATE_DECL instead.
|
||
We use `uses_template_parms (DECL_TI_ARGS (spec))' rather than
|
||
the more obvious `uses_template_parms (spec)' to avoid problems
|
||
with default function arguments. In particular, given
|
||
something like this:
|
||
|
||
template <class T> void f(T t1, T t = T())
|
||
|
||
the default argument expression is not substituted for in an
|
||
instantiation unless and until it is actually needed. */
|
||
return spec;
|
||
|
||
/* There should be as many levels of arguments as there are
|
||
levels of parameters. */
|
||
my_friendly_assert (TMPL_ARGS_DEPTH (args)
|
||
== TMPL_PARMS_DEPTH (DECL_TEMPLATE_PARMS (tmpl)),
|
||
0);
|
||
|
||
for (s = DECL_TEMPLATE_SPECIALIZATIONS (tmpl);
|
||
s != NULL_TREE;
|
||
s = TREE_CHAIN (s))
|
||
if (comp_template_args (TREE_PURPOSE (s), args))
|
||
{
|
||
tree fn = TREE_VALUE (s);
|
||
|
||
if (DECL_TEMPLATE_SPECIALIZATION (spec))
|
||
{
|
||
if (DECL_TEMPLATE_INSTANTIATION (fn))
|
||
{
|
||
if (TREE_USED (fn)
|
||
|| DECL_EXPLICIT_INSTANTIATION (fn))
|
||
{
|
||
cp_error ("specialization of %D after instantiation",
|
||
fn);
|
||
return spec;
|
||
}
|
||
else
|
||
{
|
||
/* This situation should occur only if the first
|
||
specialization is an implicit instantiation,
|
||
the second is an explicit specialization, and
|
||
the implicit instantiation has not yet been
|
||
used. That situation can occur if we have
|
||
implicitly instantiated a member function and
|
||
then specialized it later.
|
||
|
||
We can also wind up here if a friend
|
||
declaration that looked like an instantiation
|
||
turns out to be a specialization:
|
||
|
||
template <class T> void foo(T);
|
||
class S { friend void foo<>(int) };
|
||
template <> void foo(int);
|
||
|
||
We transform the existing DECL in place so that
|
||
any pointers to it become pointers to the
|
||
updated declaration.
|
||
|
||
If there was a definition for the template, but
|
||
not for the specialization, we want this to
|
||
look as if there is no definition, and vice
|
||
versa. */
|
||
DECL_INITIAL (fn) = NULL_TREE;
|
||
duplicate_decls (spec, fn);
|
||
|
||
return fn;
|
||
}
|
||
}
|
||
else if (DECL_TEMPLATE_SPECIALIZATION (fn))
|
||
{
|
||
duplicate_decls (spec, fn);
|
||
return fn;
|
||
}
|
||
}
|
||
}
|
||
|
||
DECL_TEMPLATE_SPECIALIZATIONS (tmpl)
|
||
= perm_tree_cons (args, spec, DECL_TEMPLATE_SPECIALIZATIONS (tmpl));
|
||
|
||
return spec;
|
||
}
|
||
|
||
/* Unregister the specialization SPEC as a specialization of TMPL.
|
||
Returns nonzero if the SPEC was listed as a specialization of
|
||
TMPL. */
|
||
|
||
static int
|
||
unregister_specialization (spec, tmpl)
|
||
tree spec;
|
||
tree tmpl;
|
||
{
|
||
tree* s;
|
||
|
||
for (s = &DECL_TEMPLATE_SPECIALIZATIONS (tmpl);
|
||
*s != NULL_TREE;
|
||
s = &TREE_CHAIN (*s))
|
||
if (TREE_VALUE (*s) == spec)
|
||
{
|
||
*s = TREE_CHAIN (*s);
|
||
return 1;
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
/* Print the list of candidate FNS in an error message. */
|
||
|
||
void
|
||
print_candidates (fns)
|
||
tree fns;
|
||
{
|
||
tree fn;
|
||
|
||
const char *str = "candidates are:";
|
||
|
||
for (fn = fns; fn != NULL_TREE; fn = TREE_CHAIN (fn))
|
||
{
|
||
tree f;
|
||
|
||
for (f = TREE_VALUE (fn); f; f = OVL_NEXT (f))
|
||
cp_error_at ("%s %+#D", str, OVL_CURRENT (f));
|
||
str = " ";
|
||
}
|
||
}
|
||
|
||
/* Returns the template (one of the functions given by TEMPLATE_ID)
|
||
which can be specialized to match the indicated DECL with the
|
||
explicit template args given in TEMPLATE_ID. The DECL may be
|
||
NULL_TREE if none is available. In that case, the functions in
|
||
TEMPLATE_ID are non-members.
|
||
|
||
If NEED_MEMBER_TEMPLATE is non-zero the function is known to be a
|
||
specialization of a member template.
|
||
|
||
The template args (those explicitly specified and those deduced)
|
||
are output in a newly created vector *TARGS_OUT.
|
||
|
||
If it is impossible to determine the result, an error message is
|
||
issued. The error_mark_node is returned to indicate failure. */
|
||
|
||
static tree
|
||
determine_specialization (template_id, decl, targs_out,
|
||
need_member_template)
|
||
tree template_id;
|
||
tree decl;
|
||
tree* targs_out;
|
||
int need_member_template;
|
||
{
|
||
tree fn;
|
||
tree fns;
|
||
tree targs;
|
||
tree explicit_targs;
|
||
tree candidates = NULL_TREE;
|
||
tree templates = NULL_TREE;
|
||
|
||
*targs_out = NULL_TREE;
|
||
|
||
if (template_id == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
fns = TREE_OPERAND (template_id, 0);
|
||
explicit_targs = TREE_OPERAND (template_id, 1);
|
||
|
||
if (fns == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
/* Check for baselinks. */
|
||
if (TREE_CODE (fns) == TREE_LIST)
|
||
fns = TREE_VALUE (fns);
|
||
|
||
for (; fns; fns = OVL_NEXT (fns))
|
||
{
|
||
tree tmpl;
|
||
|
||
fn = OVL_CURRENT (fns);
|
||
|
||
if (TREE_CODE (fn) == TEMPLATE_DECL)
|
||
/* DECL might be a specialization of FN. */
|
||
tmpl = fn;
|
||
else if (need_member_template)
|
||
/* FN is an ordinary member function, and we need a
|
||
specialization of a member template. */
|
||
continue;
|
||
else if (TREE_CODE (fn) != FUNCTION_DECL)
|
||
/* We can get IDENTIFIER_NODEs here in certain erroneous
|
||
cases. */
|
||
continue;
|
||
else if (!DECL_FUNCTION_MEMBER_P (fn))
|
||
/* This is just an ordinary non-member function. Nothing can
|
||
be a specialization of that. */
|
||
continue;
|
||
else
|
||
{
|
||
tree decl_arg_types;
|
||
|
||
/* This is an ordinary member function. However, since
|
||
we're here, we can assume it's enclosing class is a
|
||
template class. For example,
|
||
|
||
template <typename T> struct S { void f(); };
|
||
template <> void S<int>::f() {}
|
||
|
||
Here, S<int>::f is a non-template, but S<int> is a
|
||
template class. If FN has the same type as DECL, we
|
||
might be in business. */
|
||
if (!same_type_p (TREE_TYPE (TREE_TYPE (decl)),
|
||
TREE_TYPE (TREE_TYPE (fn))))
|
||
/* The return types differ. */
|
||
continue;
|
||
|
||
/* Adjust the type of DECL in case FN is a static member. */
|
||
decl_arg_types = TYPE_ARG_TYPES (TREE_TYPE (decl));
|
||
if (DECL_STATIC_FUNCTION_P (fn)
|
||
&& DECL_NONSTATIC_MEMBER_FUNCTION_P (decl))
|
||
decl_arg_types = TREE_CHAIN (decl_arg_types);
|
||
|
||
if (compparms (TYPE_ARG_TYPES (TREE_TYPE (fn)),
|
||
decl_arg_types))
|
||
/* They match! */
|
||
candidates = tree_cons (NULL_TREE, fn, candidates);
|
||
|
||
continue;
|
||
}
|
||
|
||
/* See whether this function might be a specialization of this
|
||
template. */
|
||
targs = get_bindings (tmpl, decl, explicit_targs);
|
||
|
||
if (!targs)
|
||
/* We cannot deduce template arguments that when used to
|
||
specialize TMPL will produce DECL. */
|
||
continue;
|
||
|
||
/* Save this template, and the arguments deduced. */
|
||
templates = scratch_tree_cons (targs, tmpl, templates);
|
||
}
|
||
|
||
if (templates && TREE_CHAIN (templates))
|
||
{
|
||
/* We have:
|
||
|
||
[temp.expl.spec]
|
||
|
||
It is possible for a specialization with a given function
|
||
signature to be instantiated from more than one function
|
||
template. In such cases, explicit specification of the
|
||
template arguments must be used to uniquely identify the
|
||
function template specialization being specialized.
|
||
|
||
Note that here, there's no suggestion that we're supposed to
|
||
determine which of the candidate templates is most
|
||
specialized. However, we, also have:
|
||
|
||
[temp.func.order]
|
||
|
||
Partial ordering of overloaded function template
|
||
declarations is used in the following contexts to select
|
||
the function template to which a function template
|
||
specialization refers:
|
||
|
||
-- when an explicit specialization refers to a function
|
||
template.
|
||
|
||
So, we do use the partial ordering rules, at least for now.
|
||
This extension can only serve to make illegal programs legal,
|
||
so it's safe. And, there is strong anecdotal evidence that
|
||
the committee intended the partial ordering rules to apply;
|
||
the EDG front-end has that behavior, and John Spicer claims
|
||
that the committee simply forgot to delete the wording in
|
||
[temp.expl.spec]. */
|
||
tree tmpl = most_specialized (templates, decl, explicit_targs);
|
||
if (tmpl && tmpl != error_mark_node)
|
||
{
|
||
targs = get_bindings (tmpl, decl, explicit_targs);
|
||
templates = scratch_tree_cons (targs, tmpl, NULL_TREE);
|
||
}
|
||
}
|
||
|
||
if (templates == NULL_TREE && candidates == NULL_TREE)
|
||
{
|
||
cp_error_at ("template-id `%D' for `%+D' does not match any template declaration",
|
||
template_id, decl);
|
||
return error_mark_node;
|
||
}
|
||
else if ((templates && TREE_CHAIN (templates))
|
||
|| (candidates && TREE_CHAIN (candidates))
|
||
|| (templates && candidates))
|
||
{
|
||
cp_error_at ("ambiguous template specialization `%D' for `%+D'",
|
||
template_id, decl);
|
||
chainon (candidates, templates);
|
||
print_candidates (candidates);
|
||
return error_mark_node;
|
||
}
|
||
|
||
/* We have one, and exactly one, match. */
|
||
if (candidates)
|
||
{
|
||
/* It was a specialization of an ordinary member function in a
|
||
template class. */
|
||
*targs_out = copy_node (DECL_TI_ARGS (TREE_VALUE (candidates)));
|
||
return DECL_TI_TEMPLATE (TREE_VALUE (candidates));
|
||
}
|
||
|
||
/* It was a specialization of a template. */
|
||
targs = DECL_TI_ARGS (DECL_RESULT (TREE_VALUE (templates)));
|
||
if (TMPL_ARGS_HAVE_MULTIPLE_LEVELS (targs))
|
||
{
|
||
*targs_out = copy_node (targs);
|
||
SET_TMPL_ARGS_LEVEL (*targs_out,
|
||
TMPL_ARGS_DEPTH (*targs_out),
|
||
TREE_PURPOSE (templates));
|
||
}
|
||
else
|
||
*targs_out = TREE_PURPOSE (templates);
|
||
return TREE_VALUE (templates);
|
||
}
|
||
|
||
/* Check to see if the function just declared, as indicated in
|
||
DECLARATOR, and in DECL, is a specialization of a function
|
||
template. We may also discover that the declaration is an explicit
|
||
instantiation at this point.
|
||
|
||
Returns DECL, or an equivalent declaration that should be used
|
||
instead if all goes well. Issues an error message if something is
|
||
amiss. Returns error_mark_node if the error is not easily
|
||
recoverable.
|
||
|
||
FLAGS is a bitmask consisting of the following flags:
|
||
|
||
2: The function has a definition.
|
||
4: The function is a friend.
|
||
|
||
The TEMPLATE_COUNT is the number of references to qualifying
|
||
template classes that appeared in the name of the function. For
|
||
example, in
|
||
|
||
template <class T> struct S { void f(); };
|
||
void S<int>::f();
|
||
|
||
the TEMPLATE_COUNT would be 1. However, explicitly specialized
|
||
classes are not counted in the TEMPLATE_COUNT, so that in
|
||
|
||
template <class T> struct S {};
|
||
template <> struct S<int> { void f(); }
|
||
template <> void S<int>::f();
|
||
|
||
the TEMPLATE_COUNT would be 0. (Note that this declaration is
|
||
illegal; there should be no template <>.)
|
||
|
||
If the function is a specialization, it is marked as such via
|
||
DECL_TEMPLATE_SPECIALIZATION. Furthermore, its DECL_TEMPLATE_INFO
|
||
is set up correctly, and it is added to the list of specializations
|
||
for that template. */
|
||
|
||
tree
|
||
check_explicit_specialization (declarator, decl, template_count, flags)
|
||
tree declarator;
|
||
tree decl;
|
||
int template_count;
|
||
int flags;
|
||
{
|
||
int have_def = flags & 2;
|
||
int is_friend = flags & 4;
|
||
int specialization = 0;
|
||
int explicit_instantiation = 0;
|
||
int member_specialization = 0;
|
||
|
||
tree ctype = DECL_CLASS_CONTEXT (decl);
|
||
tree dname = DECL_NAME (decl);
|
||
|
||
if (processing_specialization)
|
||
{
|
||
/* The last template header was of the form template <>. */
|
||
|
||
if (template_header_count > template_count)
|
||
{
|
||
/* There were more template headers than qualifying template
|
||
classes. */
|
||
if (template_header_count - template_count > 1)
|
||
/* There shouldn't be that many template parameter lists.
|
||
There can be at most one parameter list for every
|
||
qualifying class, plus one for the function itself. */
|
||
cp_error ("too many template parameter lists in declaration of `%D'", decl);
|
||
|
||
SET_DECL_TEMPLATE_SPECIALIZATION (decl);
|
||
if (ctype)
|
||
member_specialization = 1;
|
||
else
|
||
specialization = 1;
|
||
}
|
||
else if (template_header_count == template_count)
|
||
{
|
||
/* The counts are equal. So, this might be a
|
||
specialization, but it is not a specialization of a
|
||
member template. It might be something like
|
||
|
||
template <class T> struct S {
|
||
void f(int i);
|
||
};
|
||
template <>
|
||
void S<int>::f(int i) {} */
|
||
specialization = 1;
|
||
SET_DECL_TEMPLATE_SPECIALIZATION (decl);
|
||
}
|
||
else
|
||
{
|
||
/* This cannot be an explicit specialization. There are not
|
||
enough headers for all of the qualifying classes. For
|
||
example, we might have:
|
||
|
||
template <>
|
||
void S<int>::T<char>::f();
|
||
|
||
But, we're missing another template <>. */
|
||
cp_error("too few template parameter lists in declaration of `%D'", decl);
|
||
return decl;
|
||
}
|
||
}
|
||
else if (processing_explicit_instantiation)
|
||
{
|
||
if (template_header_count)
|
||
cp_error ("template parameter list used in explicit instantiation");
|
||
|
||
if (have_def)
|
||
cp_error ("definition provided for explicit instantiation");
|
||
|
||
explicit_instantiation = 1;
|
||
}
|
||
else if (ctype != NULL_TREE
|
||
&& !TYPE_BEING_DEFINED (ctype)
|
||
&& CLASSTYPE_TEMPLATE_INSTANTIATION (ctype)
|
||
&& !is_friend)
|
||
{
|
||
/* This case catches outdated code that looks like this:
|
||
|
||
template <class T> struct S { void f(); };
|
||
void S<int>::f() {} // Missing template <>
|
||
|
||
We disable this check when the type is being defined to
|
||
avoid complaining about default compiler-generated
|
||
constructors, destructors, and assignment operators.
|
||
Since the type is an instantiation, not a specialization,
|
||
these are the only functions that can be defined before
|
||
the class is complete. */
|
||
|
||
/* If they said
|
||
template <class T> void S<int>::f() {}
|
||
that's bogus. */
|
||
if (template_header_count)
|
||
{
|
||
cp_error ("template parameters specified in specialization");
|
||
return decl;
|
||
}
|
||
|
||
if (pedantic)
|
||
cp_pedwarn
|
||
("explicit specialization not preceded by `template <>'");
|
||
specialization = 1;
|
||
SET_DECL_TEMPLATE_SPECIALIZATION (decl);
|
||
}
|
||
else if (TREE_CODE (declarator) == TEMPLATE_ID_EXPR)
|
||
{
|
||
if (is_friend)
|
||
/* This could be something like:
|
||
|
||
template <class T> void f(T);
|
||
class S { friend void f<>(int); } */
|
||
specialization = 1;
|
||
else
|
||
{
|
||
/* This case handles bogus declarations like template <>
|
||
template <class T> void f<int>(); */
|
||
|
||
cp_error ("template-id `%D' in declaration of primary template",
|
||
declarator);
|
||
return decl;
|
||
}
|
||
}
|
||
|
||
if (specialization || member_specialization)
|
||
{
|
||
tree t = TYPE_ARG_TYPES (TREE_TYPE (decl));
|
||
for (; t; t = TREE_CHAIN (t))
|
||
if (TREE_PURPOSE (t))
|
||
{
|
||
cp_pedwarn
|
||
("default argument specified in explicit specialization");
|
||
break;
|
||
}
|
||
if (current_lang_name == lang_name_c)
|
||
cp_error ("template specialization with C linkage");
|
||
}
|
||
|
||
if (specialization || member_specialization || explicit_instantiation)
|
||
{
|
||
tree tmpl = NULL_TREE;
|
||
tree targs = NULL_TREE;
|
||
|
||
/* Make sure that the declarator is a TEMPLATE_ID_EXPR. */
|
||
if (TREE_CODE (declarator) != TEMPLATE_ID_EXPR)
|
||
{
|
||
tree fns;
|
||
|
||
my_friendly_assert (TREE_CODE (declarator) == IDENTIFIER_NODE,
|
||
0);
|
||
if (!ctype)
|
||
fns = IDENTIFIER_NAMESPACE_VALUE (dname);
|
||
else
|
||
fns = dname;
|
||
|
||
declarator =
|
||
lookup_template_function (fns, NULL_TREE);
|
||
}
|
||
|
||
if (declarator == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
if (ctype != NULL_TREE && TYPE_BEING_DEFINED (ctype))
|
||
{
|
||
if (!explicit_instantiation)
|
||
/* A specialization in class scope. This is illegal,
|
||
but the error will already have been flagged by
|
||
check_specialization_scope. */
|
||
return error_mark_node;
|
||
else
|
||
{
|
||
/* It's not legal to write an explicit instantiation in
|
||
class scope, e.g.:
|
||
|
||
class C { template void f(); }
|
||
|
||
This case is caught by the parser. However, on
|
||
something like:
|
||
|
||
template class C { void f(); };
|
||
|
||
(which is illegal) we can get here. The error will be
|
||
issued later. */
|
||
;
|
||
}
|
||
|
||
return decl;
|
||
}
|
||
else if (TREE_CODE (TREE_OPERAND (declarator, 0)) == LOOKUP_EXPR)
|
||
{
|
||
/* A friend declaration. We can't do much, because we don't
|
||
know what this resolves to, yet. */
|
||
my_friendly_assert (is_friend != 0, 0);
|
||
my_friendly_assert (!explicit_instantiation, 0);
|
||
SET_DECL_IMPLICIT_INSTANTIATION (decl);
|
||
return decl;
|
||
}
|
||
else if (ctype != NULL_TREE
|
||
&& (TREE_CODE (TREE_OPERAND (declarator, 0)) ==
|
||
IDENTIFIER_NODE))
|
||
{
|
||
/* Find the list of functions in ctype that have the same
|
||
name as the declared function. */
|
||
tree name = TREE_OPERAND (declarator, 0);
|
||
tree fns = NULL_TREE;
|
||
int idx;
|
||
|
||
if (name == constructor_name (ctype)
|
||
|| name == constructor_name_full (ctype))
|
||
{
|
||
int is_constructor = DECL_CONSTRUCTOR_P (decl);
|
||
|
||
if (is_constructor ? !TYPE_HAS_CONSTRUCTOR (ctype)
|
||
: !TYPE_HAS_DESTRUCTOR (ctype))
|
||
{
|
||
/* From [temp.expl.spec]:
|
||
|
||
If such an explicit specialization for the member
|
||
of a class template names an implicitly-declared
|
||
special member function (clause _special_), the
|
||
program is ill-formed.
|
||
|
||
Similar language is found in [temp.explicit]. */
|
||
cp_error ("specialization of implicitly-declared special member function");
|
||
return error_mark_node;
|
||
}
|
||
|
||
name = is_constructor ? ctor_identifier : dtor_identifier;
|
||
}
|
||
|
||
if (!IDENTIFIER_TYPENAME_P (name))
|
||
{
|
||
idx = lookup_fnfields_1 (ctype, name);
|
||
if (idx >= 0)
|
||
fns = TREE_VEC_ELT (CLASSTYPE_METHOD_VEC (ctype), idx);
|
||
}
|
||
else
|
||
{
|
||
tree methods;
|
||
|
||
/* For a type-conversion operator, we cannot do a
|
||
name-based lookup. We might be looking for `operator
|
||
int' which will be a specialization of `operator T'.
|
||
So, we find *all* the conversion operators, and then
|
||
select from them. */
|
||
fns = NULL_TREE;
|
||
|
||
methods = CLASSTYPE_METHOD_VEC (ctype);
|
||
if (methods)
|
||
for (idx = 2; idx < TREE_VEC_LENGTH (methods); ++idx)
|
||
{
|
||
tree ovl = TREE_VEC_ELT (methods, idx);
|
||
|
||
if (!ovl || !DECL_CONV_FN_P (OVL_CURRENT (ovl)))
|
||
/* There are no more conversion functions. */
|
||
break;
|
||
|
||
/* Glue all these conversion functions together
|
||
with those we already have. */
|
||
for (; ovl; ovl = OVL_NEXT (ovl))
|
||
fns = ovl_cons (OVL_CURRENT (ovl), fns);
|
||
}
|
||
}
|
||
|
||
if (fns == NULL_TREE)
|
||
{
|
||
cp_error ("no member function `%D' declared in `%T'",
|
||
name, ctype);
|
||
return error_mark_node;
|
||
}
|
||
else
|
||
TREE_OPERAND (declarator, 0) = fns;
|
||
}
|
||
|
||
/* Figure out what exactly is being specialized at this point.
|
||
Note that for an explicit instantiation, even one for a
|
||
member function, we cannot tell apriori whether the
|
||
instantiation is for a member template, or just a member
|
||
function of a template class. Even if a member template is
|
||
being instantiated, the member template arguments may be
|
||
elided if they can be deduced from the rest of the
|
||
declaration. */
|
||
tmpl = determine_specialization (declarator, decl,
|
||
&targs,
|
||
member_specialization);
|
||
|
||
if (!tmpl || tmpl == error_mark_node)
|
||
/* We couldn't figure out what this declaration was
|
||
specializing. */
|
||
return error_mark_node;
|
||
else
|
||
{
|
||
tree gen_tmpl = most_general_template (tmpl);
|
||
|
||
if (explicit_instantiation)
|
||
{
|
||
/* We don't set DECL_EXPLICIT_INSTANTIATION here; that
|
||
is done by do_decl_instantiation later. */
|
||
|
||
int arg_depth = TMPL_ARGS_DEPTH (targs);
|
||
int parm_depth = TMPL_PARMS_DEPTH (DECL_TEMPLATE_PARMS (tmpl));
|
||
|
||
if (arg_depth > parm_depth)
|
||
{
|
||
/* If TMPL is not the most general template (for
|
||
example, if TMPL is a friend template that is
|
||
injected into namespace scope), then there will
|
||
be too many levels fo TARGS. Remove some of them
|
||
here. */
|
||
int i;
|
||
tree new_targs;
|
||
|
||
new_targs = make_temp_vec (parm_depth);
|
||
for (i = arg_depth - parm_depth; i < arg_depth; ++i)
|
||
TREE_VEC_ELT (new_targs, i - (arg_depth - parm_depth))
|
||
= TREE_VEC_ELT (targs, i);
|
||
targs = new_targs;
|
||
}
|
||
|
||
decl = instantiate_template (tmpl, targs);
|
||
return decl;
|
||
}
|
||
|
||
/* If we though that the DECL was a member function, but it
|
||
turns out to be specializing a static member function,
|
||
make DECL a static member function as well. */
|
||
if (DECL_STATIC_FUNCTION_P (tmpl)
|
||
&& DECL_NONSTATIC_MEMBER_FUNCTION_P (decl))
|
||
{
|
||
revert_static_member_fn (&decl, 0, 0);
|
||
last_function_parms = TREE_CHAIN (last_function_parms);
|
||
}
|
||
|
||
/* Set up the DECL_TEMPLATE_INFO for DECL. */
|
||
DECL_TEMPLATE_INFO (decl)
|
||
= perm_tree_cons (tmpl, targs, NULL_TREE);
|
||
|
||
/* Mangle the function name appropriately. Note that we do
|
||
not mangle specializations of non-template member
|
||
functions of template classes, e.g. with
|
||
|
||
template <class T> struct S { void f(); }
|
||
|
||
and given the specialization
|
||
|
||
template <> void S<int>::f() {}
|
||
|
||
we do not mangle S<int>::f() here. That's because it's
|
||
just an ordinary member function and doesn't need special
|
||
treatment. We do this here so that the ordinary,
|
||
non-template, name-mangling algorith will not be used
|
||
later. */
|
||
if ((is_member_template (tmpl) || ctype == NULL_TREE)
|
||
&& name_mangling_version >= 1)
|
||
set_mangled_name_for_template_decl (decl);
|
||
|
||
if (is_friend && !have_def)
|
||
/* This is not really a declaration of a specialization.
|
||
It's just the name of an instantiation. But, it's not
|
||
a request for an instantiation, either. */
|
||
SET_DECL_IMPLICIT_INSTANTIATION (decl);
|
||
|
||
/* Register this specialization so that we can find it
|
||
again. */
|
||
decl = register_specialization (decl, gen_tmpl, targs);
|
||
}
|
||
}
|
||
|
||
return decl;
|
||
}
|
||
|
||
/* TYPE is being declared. Verify that the use of template headers
|
||
and such is reasonable. Issue error messages if not. */
|
||
|
||
void
|
||
maybe_check_template_type (type)
|
||
tree type;
|
||
{
|
||
if (template_header_count)
|
||
{
|
||
/* We are in the scope of some `template <...>' header. */
|
||
|
||
int context_depth
|
||
= template_class_depth_real (TYPE_CONTEXT (type),
|
||
/*count_specializations=*/1);
|
||
|
||
if (template_header_count <= context_depth)
|
||
/* This is OK; the template headers are for the context. We
|
||
are actually too lenient here; like
|
||
check_explicit_specialization we should consider the number
|
||
of template types included in the actual declaration. For
|
||
example,
|
||
|
||
template <class T> struct S {
|
||
template <class U> template <class V>
|
||
struct I {};
|
||
};
|
||
|
||
is illegal, but:
|
||
|
||
template <class T> struct S {
|
||
template <class U> struct I;
|
||
};
|
||
|
||
template <class T> template <class U.
|
||
struct S<T>::I {};
|
||
|
||
is not. */
|
||
;
|
||
else if (template_header_count > context_depth + 1)
|
||
/* There are two many template parameter lists. */
|
||
cp_error ("too many template parameter lists in declaration of `%T'", type);
|
||
}
|
||
}
|
||
|
||
/* Returns 1 iff PARMS1 and PARMS2 are identical sets of template
|
||
parameters. These are represented in the same format used for
|
||
DECL_TEMPLATE_PARMS. */
|
||
|
||
int comp_template_parms (parms1, parms2)
|
||
tree parms1;
|
||
tree parms2;
|
||
{
|
||
tree p1;
|
||
tree p2;
|
||
|
||
if (parms1 == parms2)
|
||
return 1;
|
||
|
||
for (p1 = parms1, p2 = parms2;
|
||
p1 != NULL_TREE && p2 != NULL_TREE;
|
||
p1 = TREE_CHAIN (p1), p2 = TREE_CHAIN (p2))
|
||
{
|
||
tree t1 = TREE_VALUE (p1);
|
||
tree t2 = TREE_VALUE (p2);
|
||
int i;
|
||
|
||
my_friendly_assert (TREE_CODE (t1) == TREE_VEC, 0);
|
||
my_friendly_assert (TREE_CODE (t2) == TREE_VEC, 0);
|
||
|
||
if (TREE_VEC_LENGTH (t1) != TREE_VEC_LENGTH (t2))
|
||
return 0;
|
||
|
||
for (i = 0; i < TREE_VEC_LENGTH (t2); ++i)
|
||
{
|
||
tree parm1 = TREE_VALUE (TREE_VEC_ELT (t1, i));
|
||
tree parm2 = TREE_VALUE (TREE_VEC_ELT (t2, i));
|
||
|
||
if (TREE_CODE (parm1) != TREE_CODE (parm2))
|
||
return 0;
|
||
|
||
if (TREE_CODE (parm1) == TEMPLATE_TYPE_PARM)
|
||
continue;
|
||
else if (!same_type_p (TREE_TYPE (parm1), TREE_TYPE (parm2)))
|
||
return 0;
|
||
}
|
||
}
|
||
|
||
if ((p1 != NULL_TREE) != (p2 != NULL_TREE))
|
||
/* One set of parameters has more parameters lists than the
|
||
other. */
|
||
return 0;
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* Complain if DECL shadows a template parameter.
|
||
|
||
[temp.local]: A template-parameter shall not be redeclared within its
|
||
scope (including nested scopes). */
|
||
|
||
void
|
||
check_template_shadow (decl)
|
||
tree decl;
|
||
{
|
||
tree olddecl;
|
||
|
||
/* If we're not in a template, we can't possibly shadow a template
|
||
parameter. */
|
||
if (!current_template_parms)
|
||
return;
|
||
|
||
/* Figure out what we're shadowing. */
|
||
if (TREE_CODE (decl) == OVERLOAD)
|
||
decl = OVL_CURRENT (decl);
|
||
olddecl = IDENTIFIER_VALUE (DECL_NAME (decl));
|
||
|
||
/* If there's no previous binding for this name, we're not shadowing
|
||
anything, let alone a template parameter. */
|
||
if (!olddecl)
|
||
return;
|
||
|
||
/* If we're not shadowing a template parameter, we're done. Note
|
||
that OLDDECL might be an OVERLOAD (or perhaps even an
|
||
ERROR_MARK), so we can't just blithely assume it to be a _DECL
|
||
node. */
|
||
if (TREE_CODE_CLASS (TREE_CODE (olddecl)) != 'd'
|
||
|| !DECL_TEMPLATE_PARM_P (olddecl))
|
||
return;
|
||
|
||
/* We check for decl != olddecl to avoid bogus errors for using a
|
||
name inside a class. We check TPFI to avoid duplicate errors for
|
||
inline member templates. */
|
||
if (decl == olddecl
|
||
|| TEMPLATE_PARMS_FOR_INLINE (current_template_parms))
|
||
return;
|
||
|
||
cp_error_at ("declaration of `%#D'", decl);
|
||
cp_error_at (" shadows template parm `%#D'", olddecl);
|
||
}
|
||
|
||
/* Return a new TEMPLATE_PARM_INDEX with the indicated INDEX, LEVEL,
|
||
ORIG_LEVEL, DECL, and TYPE. */
|
||
|
||
static tree
|
||
build_template_parm_index (index, level, orig_level, decl, type)
|
||
int index;
|
||
int level;
|
||
int orig_level;
|
||
tree decl;
|
||
tree type;
|
||
{
|
||
tree t = make_node (TEMPLATE_PARM_INDEX);
|
||
TEMPLATE_PARM_IDX (t) = index;
|
||
TEMPLATE_PARM_LEVEL (t) = level;
|
||
TEMPLATE_PARM_ORIG_LEVEL (t) = orig_level;
|
||
TEMPLATE_PARM_DECL (t) = decl;
|
||
TREE_TYPE (t) = type;
|
||
|
||
return t;
|
||
}
|
||
|
||
/* Return a TEMPLATE_PARM_INDEX, similar to INDEX, but whose
|
||
TEMPLATE_PARM_LEVEL has been decreased by LEVELS. If such a
|
||
TEMPLATE_PARM_INDEX already exists, it is returned; otherwise, a
|
||
new one is created. */
|
||
|
||
static tree
|
||
reduce_template_parm_level (index, type, levels)
|
||
tree index;
|
||
tree type;
|
||
int levels;
|
||
{
|
||
if (TEMPLATE_PARM_DESCENDANTS (index) == NULL_TREE
|
||
|| (TEMPLATE_PARM_LEVEL (TEMPLATE_PARM_DESCENDANTS (index))
|
||
!= TEMPLATE_PARM_LEVEL (index) - levels))
|
||
{
|
||
tree decl
|
||
= build_decl (TREE_CODE (TEMPLATE_PARM_DECL (index)),
|
||
DECL_NAME (TEMPLATE_PARM_DECL (index)),
|
||
type);
|
||
tree t
|
||
= build_template_parm_index (TEMPLATE_PARM_IDX (index),
|
||
TEMPLATE_PARM_LEVEL (index) - levels,
|
||
TEMPLATE_PARM_ORIG_LEVEL (index),
|
||
decl, type);
|
||
TEMPLATE_PARM_DESCENDANTS (index) = t;
|
||
|
||
/* Template template parameters need this. */
|
||
DECL_TEMPLATE_PARMS (decl)
|
||
= DECL_TEMPLATE_PARMS (TEMPLATE_PARM_DECL (index));
|
||
}
|
||
|
||
return TEMPLATE_PARM_DESCENDANTS (index);
|
||
}
|
||
|
||
/* Process information from new template parameter NEXT and append it to the
|
||
LIST being built. */
|
||
|
||
tree
|
||
process_template_parm (list, next)
|
||
tree list, next;
|
||
{
|
||
tree parm;
|
||
tree decl = 0;
|
||
tree defval;
|
||
int is_type, idx;
|
||
|
||
parm = next;
|
||
my_friendly_assert (TREE_CODE (parm) == TREE_LIST, 259);
|
||
defval = TREE_PURPOSE (parm);
|
||
parm = TREE_VALUE (parm);
|
||
is_type = TREE_PURPOSE (parm) == class_type_node;
|
||
|
||
if (list)
|
||
{
|
||
tree p = TREE_VALUE (tree_last (list));
|
||
|
||
if (TREE_CODE (p) == TYPE_DECL)
|
||
idx = TEMPLATE_TYPE_IDX (TREE_TYPE (p));
|
||
else if (TREE_CODE (p) == TEMPLATE_DECL)
|
||
idx = TEMPLATE_TYPE_IDX (TREE_TYPE (DECL_TEMPLATE_RESULT (p)));
|
||
else
|
||
idx = TEMPLATE_PARM_IDX (DECL_INITIAL (p));
|
||
++idx;
|
||
}
|
||
else
|
||
idx = 0;
|
||
|
||
if (!is_type)
|
||
{
|
||
my_friendly_assert (TREE_CODE (TREE_PURPOSE (parm)) == TREE_LIST, 260);
|
||
/* is a const-param */
|
||
parm = grokdeclarator (TREE_VALUE (parm), TREE_PURPOSE (parm),
|
||
PARM, 0, NULL_TREE);
|
||
|
||
/* [temp.param]
|
||
|
||
The top-level cv-qualifiers on the template-parameter are
|
||
ignored when determining its type. */
|
||
TREE_TYPE (parm) = TYPE_MAIN_VARIANT (TREE_TYPE (parm));
|
||
|
||
/* A template parameter is not modifiable. */
|
||
TREE_READONLY (parm) = 1;
|
||
if (IS_AGGR_TYPE (TREE_TYPE (parm))
|
||
&& TREE_CODE (TREE_TYPE (parm)) != TEMPLATE_TYPE_PARM
|
||
&& TREE_CODE (TREE_TYPE (parm)) != TYPENAME_TYPE)
|
||
{
|
||
cp_error ("`%#T' is not a valid type for a template constant parameter",
|
||
TREE_TYPE (parm));
|
||
if (DECL_NAME (parm) == NULL_TREE)
|
||
error (" a template type parameter must begin with `class' or `typename'");
|
||
TREE_TYPE (parm) = void_type_node;
|
||
}
|
||
else if (pedantic
|
||
&& (TREE_CODE (TREE_TYPE (parm)) == REAL_TYPE
|
||
|| TREE_CODE (TREE_TYPE (parm)) == COMPLEX_TYPE))
|
||
cp_pedwarn ("`%T' is not a valid type for a template constant parameter",
|
||
TREE_TYPE (parm));
|
||
if (TREE_PERMANENT (parm) == 0)
|
||
{
|
||
parm = copy_node (parm);
|
||
TREE_PERMANENT (parm) = 1;
|
||
}
|
||
decl = build_decl (CONST_DECL, DECL_NAME (parm), TREE_TYPE (parm));
|
||
DECL_INITIAL (parm) = DECL_INITIAL (decl)
|
||
= build_template_parm_index (idx, processing_template_decl,
|
||
processing_template_decl,
|
||
decl, TREE_TYPE (parm));
|
||
}
|
||
else
|
||
{
|
||
tree t;
|
||
parm = TREE_VALUE (parm);
|
||
|
||
if (parm && TREE_CODE (parm) == TEMPLATE_DECL)
|
||
{
|
||
t = make_lang_type (TEMPLATE_TEMPLATE_PARM);
|
||
/* This is for distinguishing between real templates and template
|
||
template parameters */
|
||
TREE_TYPE (parm) = t;
|
||
TREE_TYPE (DECL_TEMPLATE_RESULT (parm)) = t;
|
||
decl = parm;
|
||
}
|
||
else
|
||
{
|
||
t = make_lang_type (TEMPLATE_TYPE_PARM);
|
||
/* parm is either IDENTIFIER_NODE or NULL_TREE */
|
||
decl = build_decl (TYPE_DECL, parm, t);
|
||
}
|
||
|
||
TYPE_NAME (t) = decl;
|
||
TYPE_STUB_DECL (t) = decl;
|
||
parm = decl;
|
||
TEMPLATE_TYPE_PARM_INDEX (t)
|
||
= build_template_parm_index (idx, processing_template_decl,
|
||
processing_template_decl,
|
||
decl, TREE_TYPE (parm));
|
||
}
|
||
SET_DECL_ARTIFICIAL (decl);
|
||
DECL_TEMPLATE_PARM_P (decl) = 1;
|
||
pushdecl (decl);
|
||
parm = build_tree_list (defval, parm);
|
||
return chainon (list, parm);
|
||
}
|
||
|
||
/* The end of a template parameter list has been reached. Process the
|
||
tree list into a parameter vector, converting each parameter into a more
|
||
useful form. Type parameters are saved as IDENTIFIER_NODEs, and others
|
||
as PARM_DECLs. */
|
||
|
||
tree
|
||
end_template_parm_list (parms)
|
||
tree parms;
|
||
{
|
||
int nparms;
|
||
tree parm;
|
||
tree saved_parmlist = make_tree_vec (list_length (parms));
|
||
|
||
current_template_parms
|
||
= tree_cons (build_int_2 (0, processing_template_decl),
|
||
saved_parmlist, current_template_parms);
|
||
|
||
for (parm = parms, nparms = 0; parm; parm = TREE_CHAIN (parm), nparms++)
|
||
TREE_VEC_ELT (saved_parmlist, nparms) = parm;
|
||
|
||
--processing_template_parmlist;
|
||
|
||
return saved_parmlist;
|
||
}
|
||
|
||
/* end_template_decl is called after a template declaration is seen. */
|
||
|
||
void
|
||
end_template_decl ()
|
||
{
|
||
reset_specialization ();
|
||
|
||
if (! processing_template_decl)
|
||
return;
|
||
|
||
/* This matches the pushlevel in begin_template_parm_list. */
|
||
poplevel (0, 0, 0);
|
||
|
||
--processing_template_decl;
|
||
current_template_parms = TREE_CHAIN (current_template_parms);
|
||
(void) get_pending_sizes (); /* Why? */
|
||
}
|
||
|
||
/* Given a template argument vector containing the template PARMS.
|
||
The innermost PARMS are given first. */
|
||
|
||
tree
|
||
current_template_args ()
|
||
{
|
||
tree header;
|
||
tree args = NULL_TREE;
|
||
int length = TMPL_PARMS_DEPTH (current_template_parms);
|
||
int l = length;
|
||
|
||
/* If there is only one level of template parameters, we do not
|
||
create a TREE_VEC of TREE_VECs. Instead, we return a single
|
||
TREE_VEC containing the arguments. */
|
||
if (length > 1)
|
||
args = make_tree_vec (length);
|
||
|
||
for (header = current_template_parms; header; header = TREE_CHAIN (header))
|
||
{
|
||
tree a = copy_node (TREE_VALUE (header));
|
||
int i;
|
||
|
||
TREE_TYPE (a) = NULL_TREE;
|
||
for (i = TREE_VEC_LENGTH (a) - 1; i >= 0; --i)
|
||
{
|
||
tree t = TREE_VEC_ELT (a, i);
|
||
|
||
/* T will be a list if we are called from within a
|
||
begin/end_template_parm_list pair, but a vector directly
|
||
if within a begin/end_member_template_processing pair. */
|
||
if (TREE_CODE (t) == TREE_LIST)
|
||
{
|
||
t = TREE_VALUE (t);
|
||
|
||
if (TREE_CODE (t) == TYPE_DECL
|
||
|| TREE_CODE (t) == TEMPLATE_DECL)
|
||
t = TREE_TYPE (t);
|
||
else
|
||
t = DECL_INITIAL (t);
|
||
TREE_VEC_ELT (a, i) = t;
|
||
}
|
||
}
|
||
|
||
if (length > 1)
|
||
TREE_VEC_ELT (args, --l) = a;
|
||
else
|
||
args = a;
|
||
}
|
||
|
||
return args;
|
||
}
|
||
|
||
/* Return a TEMPLATE_DECL corresponding to DECL, using the indicated
|
||
template PARMS. Used by push_template_decl below. */
|
||
|
||
static tree
|
||
build_template_decl (decl, parms)
|
||
tree decl;
|
||
tree parms;
|
||
{
|
||
tree tmpl = build_lang_decl (TEMPLATE_DECL, DECL_NAME (decl), NULL_TREE);
|
||
DECL_TEMPLATE_PARMS (tmpl) = parms;
|
||
DECL_CONTEXT (tmpl) = DECL_CONTEXT (decl);
|
||
if (DECL_LANG_SPECIFIC (decl))
|
||
{
|
||
DECL_CLASS_CONTEXT (tmpl) = DECL_CLASS_CONTEXT (decl);
|
||
DECL_STATIC_FUNCTION_P (tmpl) = DECL_STATIC_FUNCTION_P (decl);
|
||
DECL_CONSTRUCTOR_P (tmpl) = DECL_CONSTRUCTOR_P (decl);
|
||
}
|
||
|
||
return tmpl;
|
||
}
|
||
|
||
struct template_parm_data
|
||
{
|
||
/* The level of the template parameters we are currently
|
||
processing. */
|
||
int level;
|
||
|
||
/* The index of the specialization argument we are currently
|
||
processing. */
|
||
int current_arg;
|
||
|
||
/* An array whose size is the number of template parameters. The
|
||
elements are non-zero if the parameter has been used in any one
|
||
of the arguments processed so far. */
|
||
int* parms;
|
||
|
||
/* An array whose size is the number of template arguments. The
|
||
elements are non-zero if the argument makes use of template
|
||
parameters of this level. */
|
||
int* arg_uses_template_parms;
|
||
};
|
||
|
||
/* Subroutine of push_template_decl used to see if each template
|
||
parameter in a partial specialization is used in the explicit
|
||
argument list. If T is of the LEVEL given in DATA (which is
|
||
treated as a template_parm_data*), then DATA->PARMS is marked
|
||
appropriately. */
|
||
|
||
static int
|
||
mark_template_parm (t, data)
|
||
tree t;
|
||
void* data;
|
||
{
|
||
int level;
|
||
int idx;
|
||
struct template_parm_data* tpd = (struct template_parm_data*) data;
|
||
|
||
if (TREE_CODE (t) == TEMPLATE_PARM_INDEX)
|
||
{
|
||
level = TEMPLATE_PARM_LEVEL (t);
|
||
idx = TEMPLATE_PARM_IDX (t);
|
||
}
|
||
else
|
||
{
|
||
level = TEMPLATE_TYPE_LEVEL (t);
|
||
idx = TEMPLATE_TYPE_IDX (t);
|
||
}
|
||
|
||
if (level == tpd->level)
|
||
{
|
||
tpd->parms[idx] = 1;
|
||
tpd->arg_uses_template_parms[tpd->current_arg] = 1;
|
||
}
|
||
|
||
/* Return zero so that for_each_template_parm will continue the
|
||
traversal of the tree; we want to mark *every* template parm. */
|
||
return 0;
|
||
}
|
||
|
||
/* Process the partial specialization DECL. */
|
||
|
||
static tree
|
||
process_partial_specialization (decl)
|
||
tree decl;
|
||
{
|
||
tree type = TREE_TYPE (decl);
|
||
tree maintmpl = CLASSTYPE_TI_TEMPLATE (type);
|
||
tree specargs = CLASSTYPE_TI_ARGS (type);
|
||
tree inner_args = innermost_args (specargs);
|
||
tree inner_parms = INNERMOST_TEMPLATE_PARMS (current_template_parms);
|
||
tree main_inner_parms = DECL_INNERMOST_TEMPLATE_PARMS (maintmpl);
|
||
int nargs = TREE_VEC_LENGTH (inner_args);
|
||
int ntparms = TREE_VEC_LENGTH (inner_parms);
|
||
int i;
|
||
int did_error_intro = 0;
|
||
struct template_parm_data tpd;
|
||
struct template_parm_data tpd2;
|
||
|
||
/* We check that each of the template parameters given in the
|
||
partial specialization is used in the argument list to the
|
||
specialization. For example:
|
||
|
||
template <class T> struct S;
|
||
template <class T> struct S<T*>;
|
||
|
||
The second declaration is OK because `T*' uses the template
|
||
parameter T, whereas
|
||
|
||
template <class T> struct S<int>;
|
||
|
||
is no good. Even trickier is:
|
||
|
||
template <class T>
|
||
struct S1
|
||
{
|
||
template <class U>
|
||
struct S2;
|
||
template <class U>
|
||
struct S2<T>;
|
||
};
|
||
|
||
The S2<T> declaration is actually illegal; it is a
|
||
full-specialization. Of course,
|
||
|
||
template <class U>
|
||
struct S2<T (*)(U)>;
|
||
|
||
or some such would have been OK. */
|
||
tpd.level = TMPL_PARMS_DEPTH (current_template_parms);
|
||
tpd.parms = alloca (sizeof (int) * ntparms);
|
||
bzero ((PTR) tpd.parms, sizeof (int) * ntparms);
|
||
|
||
tpd.arg_uses_template_parms = alloca (sizeof (int) * nargs);
|
||
bzero ((PTR) tpd.arg_uses_template_parms, sizeof (int) * nargs);
|
||
for (i = 0; i < nargs; ++i)
|
||
{
|
||
tpd.current_arg = i;
|
||
for_each_template_parm (TREE_VEC_ELT (inner_args, i),
|
||
&mark_template_parm,
|
||
&tpd);
|
||
}
|
||
for (i = 0; i < ntparms; ++i)
|
||
if (tpd.parms[i] == 0)
|
||
{
|
||
/* One of the template parms was not used in the
|
||
specialization. */
|
||
if (!did_error_intro)
|
||
{
|
||
cp_error ("template parameters not used in partial specialization:");
|
||
did_error_intro = 1;
|
||
}
|
||
|
||
cp_error (" `%D'",
|
||
TREE_VALUE (TREE_VEC_ELT (inner_parms, i)));
|
||
}
|
||
|
||
/* [temp.class.spec]
|
||
|
||
The argument list of the specialization shall not be identical to
|
||
the implicit argument list of the primary template. */
|
||
if (comp_template_args (inner_args,
|
||
innermost_args (CLASSTYPE_TI_ARGS (TREE_TYPE
|
||
(maintmpl)))))
|
||
cp_error ("partial specialization `%T' does not specialize any template arguments", type);
|
||
|
||
/* [temp.class.spec]
|
||
|
||
A partially specialized non-type argument expression shall not
|
||
involve template parameters of the partial specialization except
|
||
when the argument expression is a simple identifier.
|
||
|
||
The type of a template parameter corresponding to a specialized
|
||
non-type argument shall not be dependent on a parameter of the
|
||
specialization. */
|
||
my_friendly_assert (nargs == DECL_NTPARMS (maintmpl), 0);
|
||
tpd2.parms = 0;
|
||
for (i = 0; i < nargs; ++i)
|
||
{
|
||
tree arg = TREE_VEC_ELT (inner_args, i);
|
||
if (/* These first two lines are the `non-type' bit. */
|
||
TREE_CODE_CLASS (TREE_CODE (arg)) != 't'
|
||
&& TREE_CODE (arg) != TEMPLATE_DECL
|
||
/* This next line is the `argument expression is not just a
|
||
simple identifier' condition and also the `specialized
|
||
non-type argument' bit. */
|
||
&& TREE_CODE (arg) != TEMPLATE_PARM_INDEX)
|
||
{
|
||
if (tpd.arg_uses_template_parms[i])
|
||
cp_error ("template argument `%E' involves template parameter(s)", arg);
|
||
else
|
||
{
|
||
/* Look at the corresponding template parameter,
|
||
marking which template parameters its type depends
|
||
upon. */
|
||
tree type =
|
||
TREE_TYPE (TREE_VALUE (TREE_VEC_ELT (main_inner_parms,
|
||
i)));
|
||
|
||
if (!tpd2.parms)
|
||
{
|
||
/* We haven't yet initialized TPD2. Do so now. */
|
||
tpd2.arg_uses_template_parms
|
||
= (int*) alloca (sizeof (int) * nargs);
|
||
/* The number of parameters here is the number in the
|
||
main template, which, as checked in the assertion
|
||
above, is NARGS. */
|
||
tpd2.parms = (int*) alloca (sizeof (int) * nargs);
|
||
tpd2.level =
|
||
TMPL_PARMS_DEPTH (DECL_TEMPLATE_PARMS (maintmpl));
|
||
}
|
||
|
||
/* Mark the template parameters. But this time, we're
|
||
looking for the template parameters of the main
|
||
template, not in the specialization. */
|
||
tpd2.current_arg = i;
|
||
tpd2.arg_uses_template_parms[i] = 0;
|
||
bzero ((PTR) tpd2.parms, sizeof (int) * nargs);
|
||
for_each_template_parm (type,
|
||
&mark_template_parm,
|
||
&tpd2);
|
||
|
||
if (tpd2.arg_uses_template_parms [i])
|
||
{
|
||
/* The type depended on some template parameters.
|
||
If they are fully specialized in the
|
||
specialization, that's OK. */
|
||
int j;
|
||
for (j = 0; j < nargs; ++j)
|
||
if (tpd2.parms[j] != 0
|
||
&& tpd.arg_uses_template_parms [j])
|
||
{
|
||
cp_error ("type `%T' of template argument `%E' depends on template parameter(s)",
|
||
type,
|
||
arg);
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
if (retrieve_specialization (maintmpl, specargs))
|
||
/* We've already got this specialization. */
|
||
return decl;
|
||
|
||
DECL_TEMPLATE_SPECIALIZATIONS (maintmpl) = CLASSTYPE_TI_SPEC_INFO (type)
|
||
= perm_tree_cons (inner_args, inner_parms,
|
||
DECL_TEMPLATE_SPECIALIZATIONS (maintmpl));
|
||
TREE_TYPE (DECL_TEMPLATE_SPECIALIZATIONS (maintmpl)) = type;
|
||
return decl;
|
||
}
|
||
|
||
/* Check that a template declaration's use of default arguments is not
|
||
invalid. Here, PARMS are the template parameters. IS_PRIMARY is
|
||
non-zero if DECL is the thing declared by a primary template.
|
||
IS_PARTIAL is non-zero if DECL is a partial specialization. */
|
||
|
||
static void
|
||
check_default_tmpl_args (decl, parms, is_primary, is_partial)
|
||
tree decl;
|
||
tree parms;
|
||
int is_primary;
|
||
int is_partial;
|
||
{
|
||
const char *msg;
|
||
int last_level_to_check;
|
||
|
||
/* [temp.param]
|
||
|
||
A default template-argument shall not be specified in a
|
||
function template declaration or a function template definition, nor
|
||
in the template-parameter-list of the definition of a member of a
|
||
class template. */
|
||
|
||
if (current_class_type
|
||
&& !TYPE_BEING_DEFINED (current_class_type)
|
||
&& DECL_LANG_SPECIFIC (decl)
|
||
/* If this is either a friend defined in the scope of the class
|
||
or a member function. */
|
||
&& DECL_CLASS_CONTEXT (decl) == current_class_type
|
||
/* And, if it was a member function, it really was defined in
|
||
the scope of the class. */
|
||
&& (!DECL_FUNCTION_MEMBER_P (decl) || DECL_DEFINED_IN_CLASS_P (decl)))
|
||
/* We already checked these parameters when the template was
|
||
declared, so there's no need to do it again now. This function
|
||
was defined in class scope, but we're processing it's body now
|
||
that the class is complete. */
|
||
return;
|
||
|
||
if (TREE_CODE (decl) != TYPE_DECL || is_partial || !is_primary)
|
||
/* For an ordinary class template, default template arguments are
|
||
allowed at the innermost level, e.g.:
|
||
template <class T = int>
|
||
struct S {};
|
||
but, in a partial specialization, they're not allowed even
|
||
there, as we have in [temp.class.spec]:
|
||
|
||
The template parameter list of a specialization shall not
|
||
contain default template argument values.
|
||
|
||
So, for a partial specialization, or for a function template,
|
||
we look at all of them. */
|
||
;
|
||
else
|
||
/* But, for a primary class template that is not a partial
|
||
specialization we look at all template parameters except the
|
||
innermost ones. */
|
||
parms = TREE_CHAIN (parms);
|
||
|
||
/* Figure out what error message to issue. */
|
||
if (TREE_CODE (decl) == FUNCTION_DECL)
|
||
msg = "default argument for template parameter in function template `%D'";
|
||
else if (is_partial)
|
||
msg = "default argument in partial specialization `%D'";
|
||
else
|
||
msg = "default argument for template parameter for class enclosing `%D'";
|
||
|
||
if (current_class_type && TYPE_BEING_DEFINED (current_class_type))
|
||
/* If we're inside a class definition, there's no need to
|
||
examine the parameters to the class itself. On the one
|
||
hand, they will be checked when the class is defined, and,
|
||
on the other, default arguments are legal in things like:
|
||
template <class T = double>
|
||
struct S { template <class U> void f(U); };
|
||
Here the default argument for `S' has no bearing on the
|
||
declaration of `f'. */
|
||
last_level_to_check = template_class_depth (current_class_type) + 1;
|
||
else
|
||
/* Check everything. */
|
||
last_level_to_check = 0;
|
||
|
||
for (; parms && TMPL_PARMS_DEPTH (parms) >= last_level_to_check;
|
||
parms = TREE_CHAIN (parms))
|
||
{
|
||
tree inner_parms = TREE_VALUE (parms);
|
||
int i, ntparms;
|
||
|
||
ntparms = TREE_VEC_LENGTH (inner_parms);
|
||
for (i = 0; i < ntparms; ++i)
|
||
if (TREE_PURPOSE (TREE_VEC_ELT (inner_parms, i)))
|
||
{
|
||
if (msg)
|
||
{
|
||
cp_error (msg, decl);
|
||
msg = 0;
|
||
}
|
||
|
||
/* Clear out the default argument so that we are not
|
||
confused later. */
|
||
TREE_PURPOSE (TREE_VEC_ELT (inner_parms, i)) = NULL_TREE;
|
||
}
|
||
|
||
/* At this point, if we're still interested in issuing messages,
|
||
they must apply to classes surrounding the object declared. */
|
||
if (msg)
|
||
msg = "default argument for template parameter for class enclosing `%D'";
|
||
}
|
||
}
|
||
|
||
/* Creates a TEMPLATE_DECL for the indicated DECL using the template
|
||
parameters given by current_template_args, or reuses a
|
||
previously existing one, if appropriate. Returns the DECL, or an
|
||
equivalent one, if it is replaced via a call to duplicate_decls.
|
||
|
||
If IS_FRIEND is non-zero, DECL is a friend declaration. */
|
||
|
||
tree
|
||
push_template_decl_real (decl, is_friend)
|
||
tree decl;
|
||
int is_friend;
|
||
{
|
||
tree tmpl;
|
||
tree args;
|
||
tree info;
|
||
tree ctx;
|
||
int primary;
|
||
int is_partial;
|
||
|
||
/* See if this is a partial specialization. */
|
||
is_partial = (TREE_CODE (decl) == TYPE_DECL && DECL_ARTIFICIAL (decl)
|
||
&& TREE_CODE (TREE_TYPE (decl)) != ENUMERAL_TYPE
|
||
&& CLASSTYPE_TEMPLATE_SPECIALIZATION (TREE_TYPE (decl)));
|
||
|
||
is_friend |= (TREE_CODE (decl) == FUNCTION_DECL && DECL_FRIEND_P (decl));
|
||
|
||
if (is_friend)
|
||
/* For a friend, we want the context of the friend function, not
|
||
the type of which it is a friend. */
|
||
ctx = DECL_CONTEXT (decl);
|
||
else if (DECL_REAL_CONTEXT (decl)
|
||
&& TREE_CODE (DECL_REAL_CONTEXT (decl)) != NAMESPACE_DECL)
|
||
/* In the case of a virtual function, we want the class in which
|
||
it is defined. */
|
||
ctx = DECL_REAL_CONTEXT (decl);
|
||
else
|
||
/* Otherwise, if we're currently definining some class, the DECL
|
||
is assumed to be a member of the class. */
|
||
ctx = current_class_type;
|
||
|
||
if (ctx && TREE_CODE (ctx) == NAMESPACE_DECL)
|
||
ctx = NULL_TREE;
|
||
|
||
if (!DECL_CONTEXT (decl))
|
||
DECL_CONTEXT (decl) = FROB_CONTEXT (current_namespace);
|
||
|
||
/* For determining whether this is a primary template or not, we're really
|
||
interested in the lexical context, not the true context. */
|
||
if (is_friend)
|
||
info = current_class_type;
|
||
else
|
||
info = ctx;
|
||
|
||
/* See if this is a primary template. */
|
||
if (info && TREE_CODE (info) == FUNCTION_DECL)
|
||
primary = 0;
|
||
/* Note that template_class_depth returns 0 if given NULL_TREE, so
|
||
this next line works even when we are at global scope. */
|
||
else if (processing_template_decl > template_class_depth (info))
|
||
primary = 1;
|
||
else
|
||
primary = 0;
|
||
|
||
if (primary)
|
||
{
|
||
if (current_lang_name == lang_name_c)
|
||
cp_error ("template with C linkage");
|
||
if (TREE_CODE (decl) == TYPE_DECL && ANON_AGGRNAME_P (DECL_NAME (decl)))
|
||
cp_error ("template class without a name");
|
||
if (TREE_CODE (decl) == TYPE_DECL
|
||
&& TREE_CODE (TREE_TYPE (decl)) == ENUMERAL_TYPE)
|
||
cp_error ("template declaration of `%#T'", TREE_TYPE (decl));
|
||
}
|
||
|
||
/* Check to see that the rules regarding the use of default
|
||
arguments are not being violated. */
|
||
check_default_tmpl_args (decl, current_template_parms,
|
||
primary, is_partial);
|
||
|
||
if (is_partial)
|
||
return process_partial_specialization (decl);
|
||
|
||
args = current_template_args ();
|
||
|
||
if (!ctx
|
||
|| TREE_CODE (ctx) == FUNCTION_DECL
|
||
|| TYPE_BEING_DEFINED (ctx)
|
||
|| (is_friend && !DECL_TEMPLATE_INFO (decl)))
|
||
{
|
||
if (DECL_LANG_SPECIFIC (decl)
|
||
&& DECL_TEMPLATE_INFO (decl)
|
||
&& DECL_TI_TEMPLATE (decl))
|
||
tmpl = DECL_TI_TEMPLATE (decl);
|
||
else
|
||
{
|
||
tmpl = build_template_decl (decl, current_template_parms);
|
||
|
||
if (DECL_LANG_SPECIFIC (decl)
|
||
&& DECL_TEMPLATE_SPECIALIZATION (decl))
|
||
{
|
||
/* A specialization of a member template of a template
|
||
class. */
|
||
SET_DECL_TEMPLATE_SPECIALIZATION (tmpl);
|
||
DECL_TEMPLATE_INFO (tmpl) = DECL_TEMPLATE_INFO (decl);
|
||
DECL_TEMPLATE_INFO (decl) = NULL_TREE;
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
tree a, t, current, parms;
|
||
int i;
|
||
|
||
if (CLASSTYPE_TEMPLATE_INSTANTIATION (ctx))
|
||
cp_error ("must specialize `%#T' before defining member `%#D'",
|
||
ctx, decl);
|
||
if (TREE_CODE (decl) == TYPE_DECL)
|
||
{
|
||
if ((IS_AGGR_TYPE_CODE (TREE_CODE (TREE_TYPE (decl)))
|
||
|| TREE_CODE (TREE_TYPE (decl)) == ENUMERAL_TYPE)
|
||
&& TYPE_TEMPLATE_INFO (TREE_TYPE (decl))
|
||
&& TYPE_TI_TEMPLATE (TREE_TYPE (decl)))
|
||
tmpl = TYPE_TI_TEMPLATE (TREE_TYPE (decl));
|
||
else
|
||
{
|
||
cp_error ("`%D' does not declare a template type", decl);
|
||
return decl;
|
||
}
|
||
}
|
||
else if (! DECL_TEMPLATE_INFO (decl))
|
||
{
|
||
cp_error ("template definition of non-template `%#D'", decl);
|
||
return decl;
|
||
}
|
||
else
|
||
tmpl = DECL_TI_TEMPLATE (decl);
|
||
|
||
if (is_member_template (tmpl)
|
||
&& DECL_FUNCTION_TEMPLATE_P (tmpl)
|
||
&& DECL_TEMPLATE_INFO (decl) && DECL_TI_ARGS (decl)
|
||
&& DECL_TEMPLATE_SPECIALIZATION (decl))
|
||
{
|
||
tree new_tmpl;
|
||
|
||
/* The declaration is a specialization of a member
|
||
template, declared outside the class. Therefore, the
|
||
innermost template arguments will be NULL, so we
|
||
replace them with the arguments determined by the
|
||
earlier call to check_explicit_specialization. */
|
||
args = DECL_TI_ARGS (decl);
|
||
|
||
new_tmpl
|
||
= build_template_decl (decl, current_template_parms);
|
||
DECL_TEMPLATE_RESULT (new_tmpl) = decl;
|
||
TREE_TYPE (new_tmpl) = TREE_TYPE (decl);
|
||
DECL_TI_TEMPLATE (decl) = new_tmpl;
|
||
SET_DECL_TEMPLATE_SPECIALIZATION (new_tmpl);
|
||
DECL_TEMPLATE_INFO (new_tmpl) =
|
||
perm_tree_cons (tmpl, args, NULL_TREE);
|
||
|
||
register_specialization (new_tmpl, tmpl, args);
|
||
return decl;
|
||
}
|
||
|
||
/* Make sure the template headers we got make sense. */
|
||
|
||
parms = DECL_TEMPLATE_PARMS (tmpl);
|
||
i = TMPL_PARMS_DEPTH (parms);
|
||
if (TMPL_ARGS_DEPTH (args) != i)
|
||
{
|
||
cp_error ("expected %d levels of template parms for `%#D', got %d",
|
||
i, decl, TMPL_ARGS_DEPTH (args));
|
||
}
|
||
else
|
||
for (current = decl; i > 0; --i, parms = TREE_CHAIN (parms))
|
||
{
|
||
a = TMPL_ARGS_LEVEL (args, i);
|
||
t = INNERMOST_TEMPLATE_PARMS (parms);
|
||
|
||
if (TREE_VEC_LENGTH (t) != TREE_VEC_LENGTH (a))
|
||
{
|
||
if (current == decl)
|
||
cp_error ("got %d template parameters for `%#D'",
|
||
TREE_VEC_LENGTH (a), decl);
|
||
else
|
||
cp_error ("got %d template parameters for `%#T'",
|
||
TREE_VEC_LENGTH (a), current);
|
||
cp_error (" but %d required", TREE_VEC_LENGTH (t));
|
||
}
|
||
|
||
/* Perhaps we should also check that the parms are used in the
|
||
appropriate qualifying scopes in the declarator? */
|
||
|
||
if (current == decl)
|
||
current = ctx;
|
||
else
|
||
current = TYPE_CONTEXT (current);
|
||
}
|
||
}
|
||
|
||
DECL_TEMPLATE_RESULT (tmpl) = decl;
|
||
TREE_TYPE (tmpl) = TREE_TYPE (decl);
|
||
|
||
/* Push template declarations for global functions and types. Note
|
||
that we do not try to push a global template friend declared in a
|
||
template class; such a thing may well depend on the template
|
||
parameters of the class. */
|
||
if (! ctx
|
||
&& !(is_friend && template_class_depth (current_class_type) > 0))
|
||
tmpl = pushdecl_namespace_level (tmpl);
|
||
|
||
if (primary)
|
||
DECL_PRIMARY_TEMPLATE (tmpl) = tmpl;
|
||
|
||
info = perm_tree_cons (tmpl, args, NULL_TREE);
|
||
|
||
if (TREE_CODE (decl) == TYPE_DECL && DECL_ARTIFICIAL (decl))
|
||
{
|
||
SET_TYPE_TEMPLATE_INFO (TREE_TYPE (tmpl), info);
|
||
if ((!ctx || TREE_CODE (ctx) != FUNCTION_DECL)
|
||
&& TREE_CODE (TREE_TYPE (decl)) != ENUMERAL_TYPE)
|
||
DECL_NAME (decl) = classtype_mangled_name (TREE_TYPE (decl));
|
||
}
|
||
else if (! DECL_LANG_SPECIFIC (decl))
|
||
cp_error ("template declaration of `%#D'", decl);
|
||
else
|
||
DECL_TEMPLATE_INFO (decl) = info;
|
||
|
||
return DECL_TEMPLATE_RESULT (tmpl);
|
||
}
|
||
|
||
tree
|
||
push_template_decl (decl)
|
||
tree decl;
|
||
{
|
||
return push_template_decl_real (decl, 0);
|
||
}
|
||
|
||
/* Called when a class template TYPE is redeclared with the indicated
|
||
template PARMS, e.g.:
|
||
|
||
template <class T> struct S;
|
||
template <class T> struct S {}; */
|
||
|
||
void
|
||
redeclare_class_template (type, parms)
|
||
tree type;
|
||
tree parms;
|
||
{
|
||
tree tmpl;
|
||
tree tmpl_parms;
|
||
int i;
|
||
|
||
if (!TYPE_TEMPLATE_INFO (type))
|
||
{
|
||
cp_error ("`%T' is not a template type", type);
|
||
return;
|
||
}
|
||
|
||
tmpl = TYPE_TI_TEMPLATE (type);
|
||
if (!PRIMARY_TEMPLATE_P (tmpl))
|
||
/* The type is nested in some template class. Nothing to worry
|
||
about here; there are no new template parameters for the nested
|
||
type. */
|
||
return;
|
||
|
||
parms = INNERMOST_TEMPLATE_PARMS (parms);
|
||
tmpl_parms = DECL_INNERMOST_TEMPLATE_PARMS (tmpl);
|
||
|
||
if (TREE_VEC_LENGTH (parms) != TREE_VEC_LENGTH (tmpl_parms))
|
||
{
|
||
cp_error_at ("previous declaration `%D'", tmpl);
|
||
cp_error ("used %d template parameter%s instead of %d",
|
||
TREE_VEC_LENGTH (tmpl_parms),
|
||
TREE_VEC_LENGTH (tmpl_parms) == 1 ? "" : "s",
|
||
TREE_VEC_LENGTH (parms));
|
||
return;
|
||
}
|
||
|
||
for (i = 0; i < TREE_VEC_LENGTH (tmpl_parms); ++i)
|
||
{
|
||
tree tmpl_parm = TREE_VALUE (TREE_VEC_ELT (tmpl_parms, i));
|
||
tree parm = TREE_VALUE (TREE_VEC_ELT (parms, i));
|
||
tree tmpl_default = TREE_PURPOSE (TREE_VEC_ELT (tmpl_parms, i));
|
||
tree parm_default = TREE_PURPOSE (TREE_VEC_ELT (parms, i));
|
||
|
||
if (TREE_CODE (tmpl_parm) != TREE_CODE (parm))
|
||
{
|
||
cp_error_at ("template parameter `%#D'", tmpl_parm);
|
||
cp_error ("redeclared here as `%#D'", parm);
|
||
return;
|
||
}
|
||
|
||
if (tmpl_default != NULL_TREE && parm_default != NULL_TREE)
|
||
{
|
||
/* We have in [temp.param]:
|
||
|
||
A template-parameter may not be given default arguments
|
||
by two different declarations in the same scope. */
|
||
cp_error ("redefinition of default argument for `%#D'", parm);
|
||
cp_error_at (" original definition appeared here", tmpl_parm);
|
||
return;
|
||
}
|
||
|
||
if (parm_default != NULL_TREE)
|
||
/* Update the previous template parameters (which are the ones
|
||
that will really count) with the new default value. */
|
||
TREE_PURPOSE (TREE_VEC_ELT (tmpl_parms, i)) = parm_default;
|
||
}
|
||
}
|
||
|
||
/* Attempt to convert the non-type template parameter EXPR to the
|
||
indicated TYPE. If the conversion is successful, return the
|
||
converted value. If the conversion is unsuccesful, return
|
||
NULL_TREE if we issued an error message, or error_mark_node if we
|
||
did not. We issue error messages for out-and-out bad template
|
||
parameters, but not simply because the conversion failed, since we
|
||
might be just trying to do argument deduction. By the time this
|
||
function is called, neither TYPE nor EXPR may make use of template
|
||
parameters. */
|
||
|
||
static tree
|
||
convert_nontype_argument (type, expr)
|
||
tree type;
|
||
tree expr;
|
||
{
|
||
tree expr_type = TREE_TYPE (expr);
|
||
|
||
/* A template-argument for a non-type, non-template
|
||
template-parameter shall be one of:
|
||
|
||
--an integral constant-expression of integral or enumeration
|
||
type; or
|
||
|
||
--the name of a non-type template-parameter; or
|
||
|
||
--the name of an object or function with external linkage,
|
||
including function templates and function template-ids but
|
||
excluding non-static class members, expressed as id-expression;
|
||
or
|
||
|
||
--the address of an object or function with external linkage,
|
||
including function templates and function template-ids but
|
||
excluding non-static class members, expressed as & id-expression
|
||
where the & is optional if the name refers to a function or
|
||
array; or
|
||
|
||
--a pointer to member expressed as described in _expr.unary.op_. */
|
||
|
||
/* An integral constant-expression can include const variables
|
||
or enumerators. */
|
||
if (INTEGRAL_TYPE_P (expr_type) && TREE_READONLY_DECL_P (expr))
|
||
expr = decl_constant_value (expr);
|
||
|
||
if (is_overloaded_fn (expr))
|
||
/* OK for now. We'll check that it has external linkage later.
|
||
Check this first since if expr_type is the unknown_type_node
|
||
we would otherwise complain below. */
|
||
;
|
||
else if (TYPE_PTRMEM_P (expr_type)
|
||
|| TYPE_PTRMEMFUNC_P (expr_type))
|
||
{
|
||
if (TREE_CODE (expr) != PTRMEM_CST)
|
||
goto bad_argument;
|
||
}
|
||
else if (TYPE_PTR_P (expr_type)
|
||
|| TYPE_PTRMEM_P (expr_type)
|
||
|| TREE_CODE (expr_type) == ARRAY_TYPE
|
||
|| TREE_CODE (type) == REFERENCE_TYPE
|
||
/* If expr is the address of an overloaded function, we
|
||
will get the unknown_type_node at this point. */
|
||
|| expr_type == unknown_type_node)
|
||
{
|
||
tree referent;
|
||
tree e = expr;
|
||
STRIP_NOPS (e);
|
||
|
||
if (TREE_CODE (type) == REFERENCE_TYPE
|
||
|| TREE_CODE (expr_type) == ARRAY_TYPE)
|
||
referent = e;
|
||
else
|
||
{
|
||
if (TREE_CODE (e) != ADDR_EXPR)
|
||
{
|
||
bad_argument:
|
||
cp_error ("`%E' is not a valid template argument", expr);
|
||
if (TYPE_PTR_P (expr_type))
|
||
{
|
||
if (TREE_CODE (TREE_TYPE (expr_type)) == FUNCTION_TYPE)
|
||
cp_error ("it must be the address of a function with external linkage");
|
||
else
|
||
cp_error ("it must be the address of an object with external linkage");
|
||
}
|
||
else if (TYPE_PTRMEM_P (expr_type)
|
||
|| TYPE_PTRMEMFUNC_P (expr_type))
|
||
cp_error ("it must be a pointer-to-member of the form `&X::Y'");
|
||
|
||
return NULL_TREE;
|
||
}
|
||
|
||
referent = TREE_OPERAND (e, 0);
|
||
STRIP_NOPS (referent);
|
||
}
|
||
|
||
if (TREE_CODE (referent) == STRING_CST)
|
||
{
|
||
cp_error ("string literal %E is not a valid template argument",
|
||
referent);
|
||
error ("because it is the address of an object with static linkage");
|
||
return NULL_TREE;
|
||
}
|
||
|
||
if (is_overloaded_fn (referent))
|
||
/* We'll check that it has external linkage later. */
|
||
;
|
||
else if (TREE_CODE (referent) != VAR_DECL)
|
||
goto bad_argument;
|
||
else if (!TREE_PUBLIC (referent))
|
||
{
|
||
cp_error ("address of non-extern `%E' cannot be used as template argument", referent);
|
||
return error_mark_node;
|
||
}
|
||
}
|
||
else if (INTEGRAL_TYPE_P (expr_type)
|
||
|| TYPE_PTRMEM_P (expr_type)
|
||
|| TYPE_PTRMEMFUNC_P (expr_type)
|
||
/* The next two are g++ extensions. */
|
||
|| TREE_CODE (expr_type) == REAL_TYPE
|
||
|| TREE_CODE (expr_type) == COMPLEX_TYPE)
|
||
{
|
||
if (! TREE_CONSTANT (expr))
|
||
{
|
||
non_constant:
|
||
cp_error ("non-constant `%E' cannot be used as template argument",
|
||
expr);
|
||
return NULL_TREE;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
cp_error ("object `%E' cannot be used as template argument", expr);
|
||
return NULL_TREE;
|
||
}
|
||
|
||
switch (TREE_CODE (type))
|
||
{
|
||
case INTEGER_TYPE:
|
||
case BOOLEAN_TYPE:
|
||
case ENUMERAL_TYPE:
|
||
/* For a non-type template-parameter of integral or enumeration
|
||
type, integral promotions (_conv.prom_) and integral
|
||
conversions (_conv.integral_) are applied. */
|
||
if (!INTEGRAL_TYPE_P (expr_type))
|
||
return error_mark_node;
|
||
|
||
/* It's safe to call digest_init in this case; we know we're
|
||
just converting one integral constant expression to another. */
|
||
expr = digest_init (type, expr, (tree*) 0);
|
||
|
||
if (TREE_CODE (expr) != INTEGER_CST)
|
||
/* Curiously, some TREE_CONSTANT integral expressions do not
|
||
simplify to integer constants. For example, `3 % 0',
|
||
remains a TRUNC_MOD_EXPR. */
|
||
goto non_constant;
|
||
|
||
return expr;
|
||
|
||
case REAL_TYPE:
|
||
case COMPLEX_TYPE:
|
||
/* These are g++ extensions. */
|
||
if (TREE_CODE (expr_type) != TREE_CODE (type))
|
||
return error_mark_node;
|
||
|
||
expr = digest_init (type, expr, (tree*) 0);
|
||
|
||
if (TREE_CODE (expr) != REAL_CST)
|
||
goto non_constant;
|
||
|
||
return expr;
|
||
|
||
case POINTER_TYPE:
|
||
{
|
||
tree type_pointed_to = TREE_TYPE (type);
|
||
|
||
if (TYPE_PTRMEM_P (type))
|
||
{
|
||
tree e;
|
||
|
||
/* For a non-type template-parameter of type pointer to data
|
||
member, qualification conversions (_conv.qual_) are
|
||
applied. */
|
||
e = perform_qualification_conversions (type, expr);
|
||
if (TREE_CODE (e) == NOP_EXPR)
|
||
/* The call to perform_qualification_conversions will
|
||
insert a NOP_EXPR over EXPR to do express conversion,
|
||
if necessary. But, that will confuse us if we use
|
||
this (converted) template parameter to instantiate
|
||
another template; then the thing will not look like a
|
||
valid template argument. So, just make a new
|
||
constant, of the appropriate type. */
|
||
e = make_ptrmem_cst (type, PTRMEM_CST_MEMBER (expr));
|
||
return e;
|
||
}
|
||
else if (TREE_CODE (type_pointed_to) == FUNCTION_TYPE)
|
||
{
|
||
/* For a non-type template-parameter of type pointer to
|
||
function, only the function-to-pointer conversion
|
||
(_conv.func_) is applied. If the template-argument
|
||
represents a set of overloaded functions (or a pointer to
|
||
such), the matching function is selected from the set
|
||
(_over.over_). */
|
||
tree fns;
|
||
tree fn;
|
||
|
||
if (TREE_CODE (expr) == ADDR_EXPR)
|
||
fns = TREE_OPERAND (expr, 0);
|
||
else
|
||
fns = expr;
|
||
|
||
fn = instantiate_type (type_pointed_to, fns, 0);
|
||
|
||
if (fn == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
if (!TREE_PUBLIC (fn))
|
||
{
|
||
if (really_overloaded_fn (fns))
|
||
return error_mark_node;
|
||
else
|
||
goto bad_argument;
|
||
}
|
||
|
||
expr = build_unary_op (ADDR_EXPR, fn, 0);
|
||
|
||
my_friendly_assert (same_type_p (type, TREE_TYPE (expr)),
|
||
0);
|
||
return expr;
|
||
}
|
||
else
|
||
{
|
||
/* For a non-type template-parameter of type pointer to
|
||
object, qualification conversions (_conv.qual_) and the
|
||
array-to-pointer conversion (_conv.array_) are applied.
|
||
[Note: In particular, neither the null pointer conversion
|
||
(_conv.ptr_) nor the derived-to-base conversion
|
||
(_conv.ptr_) are applied. Although 0 is a valid
|
||
template-argument for a non-type template-parameter of
|
||
integral type, it is not a valid template-argument for a
|
||
non-type template-parameter of pointer type.]
|
||
|
||
The call to decay_conversion performs the
|
||
array-to-pointer conversion, if appropriate. */
|
||
expr = decay_conversion (expr);
|
||
|
||
if (expr == error_mark_node)
|
||
return error_mark_node;
|
||
else
|
||
return perform_qualification_conversions (type, expr);
|
||
}
|
||
}
|
||
break;
|
||
|
||
case REFERENCE_TYPE:
|
||
{
|
||
tree type_referred_to = TREE_TYPE (type);
|
||
|
||
if (TREE_CODE (type_referred_to) == FUNCTION_TYPE)
|
||
{
|
||
/* For a non-type template-parameter of type reference to
|
||
function, no conversions apply. If the
|
||
template-argument represents a set of overloaded
|
||
functions, the matching function is selected from the
|
||
set (_over.over_). */
|
||
tree fns = expr;
|
||
tree fn;
|
||
|
||
fn = instantiate_type (type_referred_to, fns, 0);
|
||
|
||
if (fn == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
if (!TREE_PUBLIC (fn))
|
||
{
|
||
if (really_overloaded_fn (fns))
|
||
/* Don't issue an error here; we might get a different
|
||
function if the overloading had worked out
|
||
differently. */
|
||
return error_mark_node;
|
||
else
|
||
goto bad_argument;
|
||
}
|
||
|
||
my_friendly_assert (same_type_p (type_referred_to,
|
||
TREE_TYPE (fn)),
|
||
0);
|
||
|
||
return fn;
|
||
}
|
||
else
|
||
{
|
||
/* For a non-type template-parameter of type reference to
|
||
object, no conversions apply. The type referred to by the
|
||
reference may be more cv-qualified than the (otherwise
|
||
identical) type of the template-argument. The
|
||
template-parameter is bound directly to the
|
||
template-argument, which must be an lvalue. */
|
||
if ((TYPE_MAIN_VARIANT (expr_type)
|
||
!= TYPE_MAIN_VARIANT (type_referred_to))
|
||
|| !at_least_as_qualified_p (type_referred_to,
|
||
expr_type)
|
||
|| !real_lvalue_p (expr))
|
||
return error_mark_node;
|
||
else
|
||
return expr;
|
||
}
|
||
}
|
||
break;
|
||
|
||
case RECORD_TYPE:
|
||
{
|
||
if (!TYPE_PTRMEMFUNC_P (type))
|
||
/* This handles templates like
|
||
template<class T, T t> void f();
|
||
when T is substituted with any class. The second template
|
||
parameter becomes invalid and the template candidate is
|
||
rejected. */
|
||
return error_mark_node;
|
||
|
||
/* For a non-type template-parameter of type pointer to member
|
||
function, no conversions apply. If the template-argument
|
||
represents a set of overloaded member functions, the
|
||
matching member function is selected from the set
|
||
(_over.over_). */
|
||
|
||
if (!TYPE_PTRMEMFUNC_P (expr_type) &&
|
||
expr_type != unknown_type_node)
|
||
return error_mark_node;
|
||
|
||
if (TREE_CODE (expr) == PTRMEM_CST)
|
||
{
|
||
/* A ptr-to-member constant. */
|
||
if (!same_type_p (type, expr_type))
|
||
return error_mark_node;
|
||
else
|
||
return expr;
|
||
}
|
||
|
||
if (TREE_CODE (expr) != ADDR_EXPR)
|
||
return error_mark_node;
|
||
|
||
expr = instantiate_type (type, expr, 0);
|
||
|
||
if (expr == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
my_friendly_assert (same_type_p (type, TREE_TYPE (expr)),
|
||
0);
|
||
return expr;
|
||
}
|
||
break;
|
||
|
||
default:
|
||
/* All non-type parameters must have one of these types. */
|
||
my_friendly_abort (0);
|
||
break;
|
||
}
|
||
|
||
return error_mark_node;
|
||
}
|
||
|
||
/* Return 1 if PARM_PARMS and ARG_PARMS matches using rule for
|
||
template template parameters. Both PARM_PARMS and ARG_PARMS are
|
||
vectors of TREE_LIST nodes containing TYPE_DECL, TEMPLATE_DECL
|
||
or PARM_DECL.
|
||
|
||
ARG_PARMS may contain more parameters than PARM_PARMS. If this is
|
||
the case, then extra parameters must have default arguments.
|
||
|
||
Consider the example:
|
||
template <class T, class Allocator = allocator> class vector;
|
||
template<template <class U> class TT> class C;
|
||
|
||
C<vector> is a valid instantiation. PARM_PARMS for the above code
|
||
contains a TYPE_DECL (for U), ARG_PARMS contains two TYPE_DECLs (for
|
||
T and Allocator) and OUTER_ARGS contains the argument that is used to
|
||
substitute the TT parameter. */
|
||
|
||
static int
|
||
coerce_template_template_parms (parm_parms, arg_parms, complain,
|
||
in_decl, outer_args)
|
||
tree parm_parms, arg_parms;
|
||
int complain;
|
||
tree in_decl, outer_args;
|
||
{
|
||
int nparms, nargs, i;
|
||
tree parm, arg;
|
||
|
||
my_friendly_assert (TREE_CODE (parm_parms) == TREE_VEC, 0);
|
||
my_friendly_assert (TREE_CODE (arg_parms) == TREE_VEC, 0);
|
||
|
||
nparms = TREE_VEC_LENGTH (parm_parms);
|
||
nargs = TREE_VEC_LENGTH (arg_parms);
|
||
|
||
/* The rule here is opposite of coerce_template_parms. */
|
||
if (nargs < nparms
|
||
|| (nargs > nparms
|
||
&& TREE_PURPOSE (TREE_VEC_ELT (arg_parms, nparms)) == NULL_TREE))
|
||
return 0;
|
||
|
||
for (i = 0; i < nparms; ++i)
|
||
{
|
||
parm = TREE_VALUE (TREE_VEC_ELT (parm_parms, i));
|
||
arg = TREE_VALUE (TREE_VEC_ELT (arg_parms, i));
|
||
|
||
if (arg == NULL_TREE || arg == error_mark_node
|
||
|| parm == NULL_TREE || parm == error_mark_node)
|
||
return 0;
|
||
|
||
if (TREE_CODE (arg) != TREE_CODE (parm))
|
||
return 0;
|
||
|
||
switch (TREE_CODE (parm))
|
||
{
|
||
case TYPE_DECL:
|
||
break;
|
||
|
||
case TEMPLATE_DECL:
|
||
/* We encounter instantiations of templates like
|
||
template <template <template <class> class> class TT>
|
||
class C; */
|
||
{
|
||
tree parmparm = DECL_INNERMOST_TEMPLATE_PARMS (parm);
|
||
tree argparm = DECL_INNERMOST_TEMPLATE_PARMS (arg);
|
||
|
||
if (!coerce_template_template_parms (parmparm, argparm,
|
||
complain, in_decl,
|
||
outer_args))
|
||
return 0;
|
||
}
|
||
break;
|
||
|
||
case PARM_DECL:
|
||
/* The tsubst call is used to handle cases such as
|
||
template <class T, template <T> class TT> class D;
|
||
i.e. the parameter list of TT depends on earlier parameters. */
|
||
if (!same_type_p (tsubst (TREE_TYPE (parm), outer_args,
|
||
complain, in_decl),
|
||
TREE_TYPE (arg)))
|
||
return 0;
|
||
break;
|
||
|
||
default:
|
||
my_friendly_abort (0);
|
||
}
|
||
}
|
||
return 1;
|
||
}
|
||
|
||
/* Convert the indicated template ARG as necessary to match the
|
||
indicated template PARM. Returns the converted ARG, or
|
||
error_mark_node if the conversion was unsuccessful. Error messages
|
||
are issued if COMPLAIN is non-zero. This conversion is for the Ith
|
||
parameter in the parameter list. ARGS is the full set of template
|
||
arguments deduced so far. */
|
||
|
||
static tree
|
||
convert_template_argument (parm, arg, args, complain, i, in_decl)
|
||
tree parm;
|
||
tree arg;
|
||
tree args;
|
||
int complain;
|
||
int i;
|
||
tree in_decl;
|
||
{
|
||
tree val;
|
||
tree inner_args;
|
||
int is_type, requires_type, is_tmpl_type, requires_tmpl_type;
|
||
|
||
inner_args = innermost_args (args);
|
||
|
||
if (TREE_CODE (arg) == TREE_LIST
|
||
&& TREE_TYPE (arg) != NULL_TREE
|
||
&& TREE_CODE (TREE_TYPE (arg)) == OFFSET_TYPE)
|
||
{
|
||
/* The template argument was the name of some
|
||
member function. That's usually
|
||
illegal, but static members are OK. In any
|
||
case, grab the underlying fields/functions
|
||
and issue an error later if required. */
|
||
arg = TREE_VALUE (arg);
|
||
TREE_TYPE (arg) = unknown_type_node;
|
||
}
|
||
|
||
requires_tmpl_type = TREE_CODE (parm) == TEMPLATE_DECL;
|
||
requires_type = (TREE_CODE (parm) == TYPE_DECL
|
||
|| requires_tmpl_type);
|
||
|
||
/* Check if it is a class template. If REQUIRES_TMPL_TYPE is true,
|
||
we also accept implicitly created TYPE_DECL as a valid argument.
|
||
This is necessary to handle the case where we pass a template name
|
||
to a template template parameter in a scope where we've derived from
|
||
in instantiation of that template, so the template name refers to that
|
||
instantiation. We really ought to handle this better. */
|
||
is_tmpl_type
|
||
= ((TREE_CODE (arg) == TEMPLATE_DECL
|
||
&& TREE_CODE (DECL_TEMPLATE_RESULT (arg)) == TYPE_DECL)
|
||
|| (TREE_CODE (arg) == TEMPLATE_TEMPLATE_PARM
|
||
&& !TEMPLATE_TEMPLATE_PARM_TEMPLATE_INFO (arg))
|
||
|| (TREE_CODE (arg) == RECORD_TYPE
|
||
&& CLASSTYPE_TEMPLATE_INFO (arg)
|
||
&& TREE_CODE (TYPE_NAME (arg)) == TYPE_DECL
|
||
&& DECL_ARTIFICIAL (TYPE_NAME (arg))
|
||
&& requires_tmpl_type
|
||
&& is_base_of_enclosing_class (arg, current_class_type)));
|
||
if (is_tmpl_type && TREE_CODE (arg) == TEMPLATE_TEMPLATE_PARM)
|
||
arg = TYPE_STUB_DECL (arg);
|
||
else if (is_tmpl_type && TREE_CODE (arg) == RECORD_TYPE)
|
||
arg = CLASSTYPE_TI_TEMPLATE (arg);
|
||
|
||
is_type = TREE_CODE_CLASS (TREE_CODE (arg)) == 't' || is_tmpl_type;
|
||
|
||
if (requires_type && ! is_type && TREE_CODE (arg) == SCOPE_REF
|
||
&& TREE_CODE (TREE_OPERAND (arg, 0)) == TEMPLATE_TYPE_PARM)
|
||
{
|
||
cp_pedwarn ("to refer to a type member of a template parameter,");
|
||
cp_pedwarn (" use `typename %E'", arg);
|
||
|
||
arg = make_typename_type (TREE_OPERAND (arg, 0),
|
||
TREE_OPERAND (arg, 1));
|
||
is_type = 1;
|
||
}
|
||
if (is_type != requires_type)
|
||
{
|
||
if (in_decl)
|
||
{
|
||
if (complain)
|
||
{
|
||
cp_error ("type/value mismatch at argument %d in template parameter list for `%D'",
|
||
i + 1, in_decl);
|
||
if (is_type)
|
||
cp_error (" expected a constant of type `%T', got `%T'",
|
||
TREE_TYPE (parm),
|
||
(is_tmpl_type ? DECL_NAME (arg) : arg));
|
||
else
|
||
cp_error (" expected a type, got `%E'", arg);
|
||
}
|
||
}
|
||
return error_mark_node;
|
||
}
|
||
if (is_tmpl_type ^ requires_tmpl_type)
|
||
{
|
||
if (in_decl && complain)
|
||
{
|
||
cp_error ("type/value mismatch at argument %d in template parameter list for `%D'",
|
||
i + 1, in_decl);
|
||
if (is_tmpl_type)
|
||
cp_error (" expected a type, got `%T'", DECL_NAME (arg));
|
||
else
|
||
cp_error (" expected a class template, got `%T'", arg);
|
||
}
|
||
return error_mark_node;
|
||
}
|
||
|
||
if (is_type)
|
||
{
|
||
if (requires_tmpl_type)
|
||
{
|
||
tree parmparm = DECL_INNERMOST_TEMPLATE_PARMS (parm);
|
||
tree argparm = DECL_INNERMOST_TEMPLATE_PARMS (arg);
|
||
|
||
if (coerce_template_template_parms (parmparm, argparm, complain,
|
||
in_decl, inner_args))
|
||
{
|
||
val = arg;
|
||
|
||
/* TEMPLATE_TEMPLATE_PARM node is preferred over
|
||
TEMPLATE_DECL. */
|
||
if (val != error_mark_node
|
||
&& DECL_TEMPLATE_TEMPLATE_PARM_P (val))
|
||
val = TREE_TYPE (val);
|
||
}
|
||
else
|
||
{
|
||
if (in_decl && complain)
|
||
{
|
||
cp_error ("type/value mismatch at argument %d in template parameter list for `%D'",
|
||
i + 1, in_decl);
|
||
cp_error (" expected a template of type `%D', got `%D'", parm, arg);
|
||
}
|
||
|
||
val = error_mark_node;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
val = groktypename (arg);
|
||
if (! processing_template_decl)
|
||
{
|
||
/* [basic.link]: A name with no linkage (notably, the
|
||
name of a class or enumeration declared in a local
|
||
scope) shall not be used to declare an entity with
|
||
linkage. This implies that names with no linkage
|
||
cannot be used as template arguments. */
|
||
tree t = no_linkage_check (val);
|
||
if (t)
|
||
{
|
||
if (ANON_AGGRNAME_P (TYPE_IDENTIFIER (t)))
|
||
cp_pedwarn
|
||
("template-argument `%T' uses anonymous type", val);
|
||
else
|
||
cp_error
|
||
("template-argument `%T' uses local type `%T'",
|
||
val, t);
|
||
return error_mark_node;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
tree t = tsubst (TREE_TYPE (parm), args, complain, in_decl);
|
||
|
||
if (processing_template_decl)
|
||
arg = maybe_fold_nontype_arg (arg);
|
||
|
||
if (!uses_template_parms (arg) && !uses_template_parms (t))
|
||
/* We used to call digest_init here. However, digest_init
|
||
will report errors, which we don't want when complain
|
||
is zero. More importantly, digest_init will try too
|
||
hard to convert things: for example, `0' should not be
|
||
converted to pointer type at this point according to
|
||
the standard. Accepting this is not merely an
|
||
extension, since deciding whether or not these
|
||
conversions can occur is part of determining which
|
||
function template to call, or whether a given epxlicit
|
||
argument specification is legal. */
|
||
val = convert_nontype_argument (t, arg);
|
||
else
|
||
val = arg;
|
||
|
||
if (val == NULL_TREE)
|
||
val = error_mark_node;
|
||
else if (val == error_mark_node && complain)
|
||
cp_error ("could not convert template argument `%E' to `%T'",
|
||
arg, t);
|
||
}
|
||
|
||
return val;
|
||
}
|
||
|
||
/* Convert all template arguments to their appropriate types, and
|
||
return a vector containing the innermost resulting template
|
||
arguments. If any error occurs, return error_mark_node, and, if
|
||
COMPLAIN is non-zero, issue an error message. Some error messages
|
||
are issued even if COMPLAIN is zero; for instance, if a template
|
||
argument is composed from a local class.
|
||
|
||
If REQUIRE_ALL_ARGUMENTS is non-zero, all arguments must be
|
||
provided in ARGLIST, or else trailing parameters must have default
|
||
values. If REQUIRE_ALL_ARGUMENTS is zero, we will attempt argument
|
||
deduction for any unspecified trailing arguments.
|
||
|
||
The resulting TREE_VEC is allocated on a temporary obstack, and
|
||
must be explicitly copied if it will be permanent. */
|
||
|
||
static tree
|
||
coerce_template_parms (parms, args, in_decl,
|
||
complain,
|
||
require_all_arguments)
|
||
tree parms, args;
|
||
tree in_decl;
|
||
int complain;
|
||
int require_all_arguments;
|
||
{
|
||
int nparms, nargs, i, lost = 0;
|
||
tree inner_args;
|
||
tree new_args;
|
||
tree new_inner_args;
|
||
|
||
inner_args = innermost_args (args);
|
||
nargs = NUM_TMPL_ARGS (inner_args);
|
||
nparms = TREE_VEC_LENGTH (parms);
|
||
|
||
if (nargs > nparms
|
||
|| (nargs < nparms
|
||
&& require_all_arguments
|
||
&& TREE_PURPOSE (TREE_VEC_ELT (parms, nargs)) == NULL_TREE))
|
||
{
|
||
if (complain)
|
||
{
|
||
cp_error ("wrong number of template arguments (%d, should be %d)",
|
||
nargs, nparms);
|
||
|
||
if (in_decl)
|
||
cp_error_at ("provided for `%D'", in_decl);
|
||
}
|
||
|
||
return error_mark_node;
|
||
}
|
||
|
||
new_inner_args = make_temp_vec (nparms);
|
||
new_args = add_outermost_template_args (args, new_inner_args);
|
||
for (i = 0; i < nparms; i++)
|
||
{
|
||
tree arg;
|
||
tree parm;
|
||
|
||
/* Get the Ith template parameter. */
|
||
parm = TREE_VEC_ELT (parms, i);
|
||
|
||
/* Calculate the Ith argument. */
|
||
if (inner_args && TREE_CODE (inner_args) == TREE_LIST)
|
||
{
|
||
arg = TREE_VALUE (inner_args);
|
||
inner_args = TREE_CHAIN (inner_args);
|
||
}
|
||
else if (i < nargs)
|
||
arg = TREE_VEC_ELT (inner_args, i);
|
||
/* If no template argument was supplied, look for a default
|
||
value. */
|
||
else if (TREE_PURPOSE (parm) == NULL_TREE)
|
||
{
|
||
/* There was no default value. */
|
||
my_friendly_assert (!require_all_arguments, 0);
|
||
break;
|
||
}
|
||
else if (TREE_CODE (TREE_VALUE (parm)) == TYPE_DECL)
|
||
arg = tsubst (TREE_PURPOSE (parm), new_args, complain, in_decl);
|
||
else
|
||
arg = tsubst_expr (TREE_PURPOSE (parm), new_args, complain,
|
||
in_decl);
|
||
|
||
/* Now, convert the Ith argument, as necessary. */
|
||
if (arg == NULL_TREE)
|
||
/* We're out of arguments. */
|
||
{
|
||
my_friendly_assert (!require_all_arguments, 0);
|
||
break;
|
||
}
|
||
else if (arg == error_mark_node)
|
||
{
|
||
cp_error ("template argument %d is invalid", i + 1);
|
||
arg = error_mark_node;
|
||
}
|
||
else
|
||
arg = convert_template_argument (TREE_VALUE (parm),
|
||
arg, new_args, complain, i,
|
||
in_decl);
|
||
|
||
if (arg == error_mark_node)
|
||
lost++;
|
||
TREE_VEC_ELT (new_inner_args, i) = arg;
|
||
}
|
||
|
||
if (lost)
|
||
return error_mark_node;
|
||
|
||
return new_inner_args;
|
||
}
|
||
|
||
/* Returns 1 if template args OT and NT are equivalent. */
|
||
|
||
static int
|
||
template_args_equal (ot, nt)
|
||
tree ot, nt;
|
||
{
|
||
if (nt == ot)
|
||
return 1;
|
||
if (TREE_CODE (nt) != TREE_CODE (ot))
|
||
return 0;
|
||
if (TREE_CODE (nt) == TREE_VEC)
|
||
/* For member templates */
|
||
return comp_template_args (ot, nt);
|
||
else if (TREE_CODE_CLASS (TREE_CODE (ot)) == 't')
|
||
return same_type_p (ot, nt);
|
||
else
|
||
return (cp_tree_equal (ot, nt) > 0);
|
||
}
|
||
|
||
/* Returns 1 iff the OLDARGS and NEWARGS are in fact identical sets
|
||
of template arguments. Returns 0 otherwise. */
|
||
|
||
int
|
||
comp_template_args (oldargs, newargs)
|
||
tree oldargs, newargs;
|
||
{
|
||
int i;
|
||
|
||
if (TREE_VEC_LENGTH (oldargs) != TREE_VEC_LENGTH (newargs))
|
||
return 0;
|
||
|
||
for (i = 0; i < TREE_VEC_LENGTH (oldargs); ++i)
|
||
{
|
||
tree nt = TREE_VEC_ELT (newargs, i);
|
||
tree ot = TREE_VEC_ELT (oldargs, i);
|
||
|
||
if (! template_args_equal (ot, nt))
|
||
return 0;
|
||
}
|
||
return 1;
|
||
}
|
||
|
||
/* Given class template name and parameter list, produce a user-friendly name
|
||
for the instantiation. */
|
||
|
||
static char *
|
||
mangle_class_name_for_template (name, parms, arglist)
|
||
char *name;
|
||
tree parms, arglist;
|
||
{
|
||
static struct obstack scratch_obstack;
|
||
static char *scratch_firstobj;
|
||
int i, nparms;
|
||
|
||
if (!scratch_firstobj)
|
||
gcc_obstack_init (&scratch_obstack);
|
||
else
|
||
obstack_free (&scratch_obstack, scratch_firstobj);
|
||
scratch_firstobj = obstack_alloc (&scratch_obstack, 1);
|
||
|
||
#define ccat(c) obstack_1grow (&scratch_obstack, (c));
|
||
#define cat(s) obstack_grow (&scratch_obstack, (s), strlen (s))
|
||
|
||
cat (name);
|
||
ccat ('<');
|
||
nparms = TREE_VEC_LENGTH (parms);
|
||
arglist = innermost_args (arglist);
|
||
my_friendly_assert (nparms == TREE_VEC_LENGTH (arglist), 268);
|
||
for (i = 0; i < nparms; i++)
|
||
{
|
||
tree parm = TREE_VALUE (TREE_VEC_ELT (parms, i));
|
||
tree arg = TREE_VEC_ELT (arglist, i);
|
||
|
||
if (i)
|
||
ccat (',');
|
||
|
||
if (TREE_CODE (parm) == TYPE_DECL)
|
||
{
|
||
cat (type_as_string_real (arg, 0, 1));
|
||
continue;
|
||
}
|
||
else if (TREE_CODE (parm) == TEMPLATE_DECL)
|
||
{
|
||
if (TREE_CODE (arg) == TEMPLATE_DECL)
|
||
{
|
||
/* Already substituted with real template. Just output
|
||
the template name here */
|
||
tree context = DECL_CONTEXT (arg);
|
||
if (context)
|
||
{
|
||
my_friendly_assert (TREE_CODE (context) == NAMESPACE_DECL, 980422);
|
||
cat(decl_as_string (DECL_CONTEXT (arg), 0));
|
||
cat("::");
|
||
}
|
||
cat (IDENTIFIER_POINTER (DECL_NAME (arg)));
|
||
}
|
||
else
|
||
/* Output the parameter declaration */
|
||
cat (type_as_string_real (arg, 0, 1));
|
||
continue;
|
||
}
|
||
else
|
||
my_friendly_assert (TREE_CODE (parm) == PARM_DECL, 269);
|
||
|
||
if (TREE_CODE (arg) == TREE_LIST)
|
||
{
|
||
/* New list cell was built because old chain link was in
|
||
use. */
|
||
my_friendly_assert (TREE_PURPOSE (arg) == NULL_TREE, 270);
|
||
arg = TREE_VALUE (arg);
|
||
}
|
||
/* No need to check arglist against parmlist here; we did that
|
||
in coerce_template_parms, called from lookup_template_class. */
|
||
cat (expr_as_string (arg, 0));
|
||
}
|
||
{
|
||
char *bufp = obstack_next_free (&scratch_obstack);
|
||
int offset = 0;
|
||
while (bufp[offset - 1] == ' ')
|
||
offset--;
|
||
obstack_blank_fast (&scratch_obstack, offset);
|
||
|
||
/* B<C<char> >, not B<C<char>> */
|
||
if (bufp[offset - 1] == '>')
|
||
ccat (' ');
|
||
}
|
||
ccat ('>');
|
||
ccat ('\0');
|
||
return (char *) obstack_base (&scratch_obstack);
|
||
}
|
||
|
||
static tree
|
||
classtype_mangled_name (t)
|
||
tree t;
|
||
{
|
||
if (CLASSTYPE_TEMPLATE_INFO (t)
|
||
/* Specializations have already had their names set up in
|
||
lookup_template_class. */
|
||
&& !CLASSTYPE_TEMPLATE_SPECIALIZATION (t))
|
||
{
|
||
tree tmpl = most_general_template (CLASSTYPE_TI_TEMPLATE (t));
|
||
|
||
/* For non-primary templates, the template parameters are
|
||
implicit from their surrounding context. */
|
||
if (PRIMARY_TEMPLATE_P (tmpl))
|
||
{
|
||
tree name = DECL_NAME (tmpl);
|
||
char *mangled_name = mangle_class_name_for_template
|
||
(IDENTIFIER_POINTER (name),
|
||
DECL_INNERMOST_TEMPLATE_PARMS (tmpl),
|
||
CLASSTYPE_TI_ARGS (t));
|
||
tree id = get_identifier (mangled_name);
|
||
IDENTIFIER_TEMPLATE (id) = name;
|
||
return id;
|
||
}
|
||
}
|
||
|
||
return TYPE_IDENTIFIER (t);
|
||
}
|
||
|
||
static void
|
||
add_pending_template (d)
|
||
tree d;
|
||
{
|
||
tree ti;
|
||
|
||
if (TREE_CODE_CLASS (TREE_CODE (d)) == 't')
|
||
ti = CLASSTYPE_TEMPLATE_INFO (d);
|
||
else
|
||
ti = DECL_TEMPLATE_INFO (d);
|
||
|
||
if (TI_PENDING_TEMPLATE_FLAG (ti))
|
||
return;
|
||
|
||
*template_tail = perm_tree_cons
|
||
(build_srcloc_here (), d, NULL_TREE);
|
||
template_tail = &TREE_CHAIN (*template_tail);
|
||
TI_PENDING_TEMPLATE_FLAG (ti) = 1;
|
||
}
|
||
|
||
|
||
/* Return a TEMPLATE_ID_EXPR corresponding to the indicated FNS (which
|
||
may be either a _DECL or an overloaded function or an
|
||
IDENTIFIER_NODE), and ARGLIST. */
|
||
|
||
tree
|
||
lookup_template_function (fns, arglist)
|
||
tree fns, arglist;
|
||
{
|
||
tree type;
|
||
|
||
if (fns == NULL_TREE)
|
||
{
|
||
cp_error ("non-template used as template");
|
||
return error_mark_node;
|
||
}
|
||
|
||
type = TREE_TYPE (fns);
|
||
if (TREE_CODE (fns) == OVERLOAD || !type)
|
||
type = unknown_type_node;
|
||
|
||
if (processing_template_decl)
|
||
return build_min (TEMPLATE_ID_EXPR, type, fns, arglist);
|
||
else
|
||
return build (TEMPLATE_ID_EXPR, type, fns, arglist);
|
||
}
|
||
|
||
/* Within the scope of a template class S<T>, the name S gets bound
|
||
(in build_self_reference) to a TYPE_DECL for the class, not a
|
||
TEMPLATE_DECL. If DECL is a TYPE_DECL for current_class_type,
|
||
or one of its enclosing classes, and that type is a template,
|
||
return the associated TEMPLATE_DECL. Otherwise, the original
|
||
DECL is returned. */
|
||
|
||
static tree
|
||
maybe_get_template_decl_from_type_decl (decl)
|
||
tree decl;
|
||
{
|
||
return (decl != NULL_TREE
|
||
&& TREE_CODE (decl) == TYPE_DECL
|
||
&& DECL_ARTIFICIAL (decl)
|
||
&& CLASS_TYPE_P (TREE_TYPE (decl))
|
||
&& CLASSTYPE_TEMPLATE_INFO (TREE_TYPE (decl)))
|
||
? CLASSTYPE_TI_TEMPLATE (TREE_TYPE (decl)) : decl;
|
||
}
|
||
|
||
/* Given an IDENTIFIER_NODE (type TEMPLATE_DECL) and a chain of
|
||
parameters, find the desired type.
|
||
|
||
D1 is the PTYPENAME terminal, and ARGLIST is the list of arguments.
|
||
(Actually ARGLIST may be either a TREE_LIST or a TREE_VEC. It will
|
||
be a TREE_LIST if called directly from the parser, and a TREE_VEC
|
||
otherwise.) Since ARGLIST is build on the decl_obstack, we must
|
||
copy it here to keep it from being reclaimed when the decl storage
|
||
is reclaimed.
|
||
|
||
IN_DECL, if non-NULL, is the template declaration we are trying to
|
||
instantiate.
|
||
|
||
If ENTERING_SCOPE is non-zero, we are about to enter the scope of
|
||
the class we are looking up.
|
||
|
||
If the template class is really a local class in a template
|
||
function, then the FUNCTION_CONTEXT is the function in which it is
|
||
being instantiated. */
|
||
|
||
tree
|
||
lookup_template_class (d1, arglist, in_decl, context, entering_scope)
|
||
tree d1, arglist;
|
||
tree in_decl;
|
||
tree context;
|
||
int entering_scope;
|
||
{
|
||
tree template = NULL_TREE, parmlist;
|
||
tree t;
|
||
|
||
if (TREE_CODE (d1) == IDENTIFIER_NODE)
|
||
{
|
||
if (IDENTIFIER_VALUE (d1)
|
||
&& DECL_TEMPLATE_TEMPLATE_PARM_P (IDENTIFIER_VALUE (d1)))
|
||
template = IDENTIFIER_VALUE (d1);
|
||
else
|
||
{
|
||
if (context)
|
||
push_decl_namespace (context);
|
||
if (current_class_type != NULL_TREE)
|
||
template =
|
||
maybe_get_template_decl_from_type_decl
|
||
(IDENTIFIER_CLASS_VALUE (d1));
|
||
if (template == NULL_TREE)
|
||
template = lookup_name_nonclass (d1);
|
||
if (context)
|
||
pop_decl_namespace ();
|
||
}
|
||
if (template)
|
||
context = DECL_CONTEXT (template);
|
||
}
|
||
else if (TREE_CODE (d1) == TYPE_DECL && IS_AGGR_TYPE (TREE_TYPE (d1)))
|
||
{
|
||
tree type = TREE_TYPE (d1);
|
||
|
||
/* If we are declaring a constructor, say A<T>::A<T>, we will get
|
||
an implicit typename for the second A. Deal with it. */
|
||
if (TREE_CODE (type) == TYPENAME_TYPE && TREE_TYPE (type))
|
||
type = TREE_TYPE (type);
|
||
|
||
if (CLASSTYPE_TEMPLATE_INFO (type))
|
||
{
|
||
template = CLASSTYPE_TI_TEMPLATE (type);
|
||
d1 = DECL_NAME (template);
|
||
}
|
||
}
|
||
else if (TREE_CODE (d1) == ENUMERAL_TYPE
|
||
|| (TREE_CODE_CLASS (TREE_CODE (d1)) == 't'
|
||
&& IS_AGGR_TYPE (d1)))
|
||
{
|
||
template = TYPE_TI_TEMPLATE (d1);
|
||
d1 = DECL_NAME (template);
|
||
}
|
||
else if (TREE_CODE (d1) == TEMPLATE_DECL
|
||
&& TREE_CODE (DECL_RESULT (d1)) == TYPE_DECL)
|
||
{
|
||
template = d1;
|
||
d1 = DECL_NAME (template);
|
||
context = DECL_CONTEXT (template);
|
||
}
|
||
else
|
||
my_friendly_abort (272);
|
||
|
||
/* With something like `template <class T> class X class X { ... };'
|
||
we could end up with D1 having nothing but an IDENTIFIER_VALUE.
|
||
We don't want to do that, but we have to deal with the situation,
|
||
so let's give them some syntax errors to chew on instead of a
|
||
crash. */
|
||
if (! template)
|
||
{
|
||
cp_error ("`%T' is not a template", d1);
|
||
return error_mark_node;
|
||
}
|
||
|
||
if (context == NULL_TREE)
|
||
context = global_namespace;
|
||
|
||
if (TREE_CODE (template) != TEMPLATE_DECL)
|
||
{
|
||
cp_error ("non-template type `%T' used as a template", d1);
|
||
if (in_decl)
|
||
cp_error_at ("for template declaration `%D'", in_decl);
|
||
return error_mark_node;
|
||
}
|
||
|
||
if (DECL_TEMPLATE_TEMPLATE_PARM_P (template))
|
||
{
|
||
/* Create a new TEMPLATE_DECL and TEMPLATE_TEMPLATE_PARM node to store
|
||
template arguments */
|
||
|
||
tree parm = copy_template_template_parm (TREE_TYPE (template));
|
||
tree template2 = TYPE_STUB_DECL (parm);
|
||
tree arglist2;
|
||
|
||
parmlist = DECL_INNERMOST_TEMPLATE_PARMS (template);
|
||
|
||
arglist2 = coerce_template_parms (parmlist, arglist, template, 1, 1);
|
||
if (arglist2 == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
arglist2 = copy_to_permanent (arglist2);
|
||
TEMPLATE_TEMPLATE_PARM_TEMPLATE_INFO (parm)
|
||
= perm_tree_cons (template2, arglist2, NULL_TREE);
|
||
TYPE_SIZE (parm) = 0;
|
||
return parm;
|
||
}
|
||
else
|
||
{
|
||
tree template_type = TREE_TYPE (template);
|
||
tree gen_tmpl;
|
||
tree type_decl;
|
||
tree found = NULL_TREE;
|
||
int arg_depth;
|
||
int parm_depth;
|
||
int is_partial_instantiation;
|
||
|
||
gen_tmpl = most_general_template (template);
|
||
parmlist = DECL_TEMPLATE_PARMS (gen_tmpl);
|
||
parm_depth = TMPL_PARMS_DEPTH (parmlist);
|
||
arg_depth = TMPL_ARGS_DEPTH (arglist);
|
||
|
||
/* We build up the coerced arguments and such on the
|
||
momentary_obstack. */
|
||
push_momentary ();
|
||
|
||
if (arg_depth == 1 && parm_depth > 1)
|
||
{
|
||
/* We've been given an incomplete set of template arguments.
|
||
For example, given:
|
||
|
||
template <class T> struct S1 {
|
||
template <class U> struct S2 {};
|
||
template <class U> struct S2<U*> {};
|
||
};
|
||
|
||
we will be called with an ARGLIST of `U*', but the
|
||
TEMPLATE will be `template <class T> template
|
||
<class U> struct S1<T>::S2'. We must fill in the missing
|
||
arguments. */
|
||
arglist
|
||
= add_outermost_template_args (TYPE_TI_ARGS (TREE_TYPE (template)),
|
||
arglist);
|
||
arg_depth = TMPL_ARGS_DEPTH (arglist);
|
||
}
|
||
|
||
/* Now we should enough arguments. */
|
||
my_friendly_assert (parm_depth == arg_depth, 0);
|
||
|
||
/* From here on, we're only interested in the most general
|
||
template. */
|
||
template = gen_tmpl;
|
||
|
||
/* Calculate the BOUND_ARGS. These will be the args that are
|
||
actually tsubst'd into the definition to create the
|
||
instantiation. */
|
||
if (parm_depth > 1)
|
||
{
|
||
/* We have multiple levels of arguments to coerce, at once. */
|
||
int i;
|
||
int saved_depth = TMPL_ARGS_DEPTH (arglist);
|
||
|
||
tree bound_args = make_temp_vec (parm_depth);
|
||
|
||
for (i = saved_depth,
|
||
t = DECL_TEMPLATE_PARMS (template);
|
||
i > 0 && t != NULL_TREE;
|
||
--i, t = TREE_CHAIN (t))
|
||
{
|
||
tree a = coerce_template_parms (TREE_VALUE (t),
|
||
arglist, template, 1, 1);
|
||
SET_TMPL_ARGS_LEVEL (bound_args, i, a);
|
||
|
||
/* We temporarily reduce the length of the ARGLIST so
|
||
that coerce_template_parms will see only the arguments
|
||
corresponding to the template parameters it is
|
||
examining. */
|
||
TREE_VEC_LENGTH (arglist)--;
|
||
}
|
||
|
||
/* Restore the ARGLIST to its full size. */
|
||
TREE_VEC_LENGTH (arglist) = saved_depth;
|
||
|
||
arglist = bound_args;
|
||
}
|
||
else
|
||
arglist
|
||
= coerce_template_parms (INNERMOST_TEMPLATE_PARMS (parmlist),
|
||
innermost_args (arglist),
|
||
template, 1, 1);
|
||
|
||
if (arglist == error_mark_node)
|
||
/* We were unable to bind the arguments. */
|
||
return error_mark_node;
|
||
|
||
/* In the scope of a template class, explicit references to the
|
||
template class refer to the type of the template, not any
|
||
instantiation of it. For example, in:
|
||
|
||
template <class T> class C { void f(C<T>); }
|
||
|
||
the `C<T>' is just the same as `C'. Outside of the
|
||
class, however, such a reference is an instantiation. */
|
||
if (comp_template_args (TYPE_TI_ARGS (template_type),
|
||
arglist))
|
||
{
|
||
found = template_type;
|
||
|
||
if (!entering_scope && PRIMARY_TEMPLATE_P (template))
|
||
{
|
||
tree ctx;
|
||
|
||
/* Note that we use DECL_CONTEXT, rather than
|
||
CP_DECL_CONTEXT, so that the termination test is
|
||
always just `ctx'. We're not interested in namepace
|
||
scopes. */
|
||
for (ctx = current_class_type;
|
||
ctx;
|
||
ctx = (TREE_CODE_CLASS (TREE_CODE (ctx)) == 't')
|
||
? TYPE_CONTEXT (ctx) : DECL_CONTEXT (ctx))
|
||
if (same_type_p (ctx, template_type))
|
||
break;
|
||
|
||
if (!ctx)
|
||
/* We're not in the scope of the class, so the
|
||
TEMPLATE_TYPE is not the type we want after
|
||
all. */
|
||
found = NULL_TREE;
|
||
}
|
||
}
|
||
|
||
if (!found)
|
||
{
|
||
for (found = DECL_TEMPLATE_INSTANTIATIONS (template);
|
||
found; found = TREE_CHAIN (found))
|
||
if (comp_template_args (TREE_PURPOSE (found), arglist))
|
||
break;
|
||
|
||
if (found)
|
||
found = TREE_VALUE (found);
|
||
}
|
||
|
||
if (found)
|
||
{
|
||
pop_momentary ();
|
||
return found;
|
||
}
|
||
|
||
/* Since we didn't find the type, we'll have to create it.
|
||
Since we'll be saving this type on the
|
||
DECL_TEMPLATE_INSTANTIATIONS list, it must be permanent. */
|
||
push_obstacks (&permanent_obstack, &permanent_obstack);
|
||
|
||
/* This type is a "partial instantiation" if any of the template
|
||
arguments still inolve template parameters. Note that we set
|
||
IS_PARTIAL_INSTANTIATION for partial specializations as
|
||
well. */
|
||
is_partial_instantiation = uses_template_parms (arglist);
|
||
|
||
/* Create the type. */
|
||
if (TREE_CODE (template_type) == ENUMERAL_TYPE)
|
||
{
|
||
if (!is_partial_instantiation)
|
||
t = start_enum (TYPE_IDENTIFIER (template_type));
|
||
else
|
||
/* We don't want to call start_enum for this type, since
|
||
the values for the enumeration constants may involve
|
||
template parameters. And, no one should be interested
|
||
in the enumeration constants for such a type. */
|
||
t = make_node (ENUMERAL_TYPE);
|
||
}
|
||
else
|
||
{
|
||
t = make_lang_type (TREE_CODE (template_type));
|
||
CLASSTYPE_DECLARED_CLASS (t)
|
||
= CLASSTYPE_DECLARED_CLASS (template_type);
|
||
CLASSTYPE_GOT_SEMICOLON (t) = 1;
|
||
SET_CLASSTYPE_IMPLICIT_INSTANTIATION (t);
|
||
TYPE_FOR_JAVA (t) = TYPE_FOR_JAVA (template_type);
|
||
}
|
||
|
||
/* If we called start_enum above, this information will already
|
||
be set up. */
|
||
if (!TYPE_NAME (t))
|
||
{
|
||
TYPE_CONTEXT (t) = FROB_CONTEXT (context);
|
||
|
||
/* Create a stub TYPE_DECL for it. */
|
||
type_decl = build_decl (TYPE_DECL, DECL_NAME (template), t);
|
||
SET_DECL_ARTIFICIAL (type_decl);
|
||
DECL_CONTEXT (type_decl) = TYPE_CONTEXT (t);
|
||
DECL_SOURCE_FILE (type_decl)
|
||
= DECL_SOURCE_FILE (TYPE_STUB_DECL (template_type));
|
||
DECL_SOURCE_LINE (type_decl)
|
||
= DECL_SOURCE_LINE (TYPE_STUB_DECL (template_type));
|
||
TYPE_STUB_DECL (t) = TYPE_NAME (t) = type_decl;
|
||
}
|
||
else
|
||
type_decl = TYPE_NAME (t);
|
||
|
||
/* Set up the template information. We have to figure out which
|
||
template is the immediate parent if this is a full
|
||
instantiation. */
|
||
if (parm_depth == 1 || is_partial_instantiation
|
||
|| !PRIMARY_TEMPLATE_P (template))
|
||
/* This case is easy; there are no member templates involved. */
|
||
found = template;
|
||
else
|
||
{
|
||
/* This is a full instantiation of a member template. There
|
||
should be some partial instantiation of which this is an
|
||
instance. */
|
||
|
||
for (found = DECL_TEMPLATE_INSTANTIATIONS (template);
|
||
found; found = TREE_CHAIN (found))
|
||
{
|
||
int success;
|
||
tree tmpl = CLASSTYPE_TI_TEMPLATE (TREE_VALUE (found));
|
||
|
||
/* We only want partial instantiations, here, not
|
||
specializations or full instantiations. */
|
||
if (CLASSTYPE_TEMPLATE_SPECIALIZATION (TREE_VALUE (found))
|
||
|| !uses_template_parms (TREE_VALUE (found)))
|
||
continue;
|
||
|
||
/* Temporarily reduce by one the number of levels in the
|
||
ARGLIST and in FOUND so as to avoid comparing the
|
||
last set of arguments. */
|
||
TREE_VEC_LENGTH (arglist)--;
|
||
TREE_VEC_LENGTH (TREE_PURPOSE (found)) --;
|
||
|
||
/* See if the arguments match. If they do, then TMPL is
|
||
the partial instantiation we want. */
|
||
success = comp_template_args (TREE_PURPOSE (found), arglist);
|
||
|
||
/* Restore the argument vectors to their full size. */
|
||
TREE_VEC_LENGTH (arglist)++;
|
||
TREE_VEC_LENGTH (TREE_PURPOSE (found))++;
|
||
|
||
if (success)
|
||
{
|
||
found = tmpl;
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (!found)
|
||
my_friendly_abort (0);
|
||
}
|
||
|
||
arglist = copy_to_permanent (arglist);
|
||
SET_TYPE_TEMPLATE_INFO (t,
|
||
tree_cons (found, arglist, NULL_TREE));
|
||
DECL_TEMPLATE_INSTANTIATIONS (template)
|
||
= tree_cons (arglist, t,
|
||
DECL_TEMPLATE_INSTANTIATIONS (template));
|
||
|
||
if (TREE_CODE (t) == ENUMERAL_TYPE
|
||
&& !is_partial_instantiation)
|
||
/* Now that the type has been registered on the instantiations
|
||
list, we set up the enumerators. Because the enumeration
|
||
constants may involve the enumeration type itself, we make
|
||
sure to register the type first, and then create the
|
||
constants. That way, doing tsubst_expr for the enumeration
|
||
constants won't result in recursive calls here; we'll find
|
||
the instantiation and exit above. */
|
||
tsubst_enum (template_type, t, arglist);
|
||
|
||
/* We're done with the permanent obstack, now. */
|
||
pop_obstacks ();
|
||
/* We're also done with the momentary allocation we started
|
||
above. */
|
||
pop_momentary ();
|
||
|
||
/* Reset the name of the type, now that CLASSTYPE_TEMPLATE_INFO
|
||
is set up. */
|
||
if (TREE_CODE (t) != ENUMERAL_TYPE)
|
||
DECL_NAME (type_decl) = classtype_mangled_name (t);
|
||
DECL_ASSEMBLER_NAME (type_decl) = DECL_NAME (type_decl);
|
||
if (!is_partial_instantiation)
|
||
{
|
||
DECL_ASSEMBLER_NAME (type_decl)
|
||
= get_identifier (build_overload_name (t, 1, 1));
|
||
|
||
/* For backwards compatibility; code that uses
|
||
-fexternal-templates expects looking up a template to
|
||
instantiate it. I think DDD still relies on this.
|
||
(jason 8/20/1998) */
|
||
if (TREE_CODE (t) != ENUMERAL_TYPE
|
||
&& flag_external_templates
|
||
&& CLASSTYPE_INTERFACE_KNOWN (TREE_TYPE (template))
|
||
&& ! CLASSTYPE_INTERFACE_ONLY (TREE_TYPE (template)))
|
||
add_pending_template (t);
|
||
}
|
||
else
|
||
/* If the type makes use of template parameters, the
|
||
code that generates debugging information will crash. */
|
||
DECL_IGNORED_P (TYPE_STUB_DECL (t)) = 1;
|
||
|
||
return t;
|
||
}
|
||
}
|
||
|
||
/* For each TEMPLATE_TYPE_PARM, TEMPLATE_TEMPLATE_PARM, or
|
||
TEMPLATE_PARM_INDEX in T, call FN with the parameter and the DATA.
|
||
If FN returns non-zero, the iteration is terminated, and
|
||
for_each_template_parm returns 1. Otherwise, the iteration
|
||
continues. If FN never returns a non-zero value, the value
|
||
returned by for_each_template_parm is 0. If FN is NULL, it is
|
||
considered to be the function which always returns 1. */
|
||
|
||
static int
|
||
for_each_template_parm (t, fn, data)
|
||
tree t;
|
||
tree_fn_t fn;
|
||
void* data;
|
||
{
|
||
if (!t)
|
||
return 0;
|
||
|
||
if (TREE_CODE_CLASS (TREE_CODE (t)) == 't'
|
||
&& for_each_template_parm (TYPE_CONTEXT (t), fn, data))
|
||
return 1;
|
||
|
||
switch (TREE_CODE (t))
|
||
{
|
||
case INDIRECT_REF:
|
||
case COMPONENT_REF:
|
||
/* We assume that the object must be instantiated in order to build
|
||
the COMPONENT_REF, so we test only whether the type of the
|
||
COMPONENT_REF uses template parms. */
|
||
return for_each_template_parm (TREE_TYPE (t), fn, data);
|
||
|
||
case ARRAY_REF:
|
||
case OFFSET_REF:
|
||
return (for_each_template_parm (TREE_OPERAND (t, 0), fn, data)
|
||
|| for_each_template_parm (TREE_OPERAND (t, 1), fn, data));
|
||
|
||
case IDENTIFIER_NODE:
|
||
if (!IDENTIFIER_TEMPLATE (t))
|
||
return 0;
|
||
my_friendly_abort (42);
|
||
|
||
/* aggregates of tree nodes */
|
||
case TREE_VEC:
|
||
{
|
||
int i = TREE_VEC_LENGTH (t);
|
||
while (i--)
|
||
if (for_each_template_parm (TREE_VEC_ELT (t, i), fn, data))
|
||
return 1;
|
||
return 0;
|
||
}
|
||
case TREE_LIST:
|
||
if (for_each_template_parm (TREE_PURPOSE (t), fn, data)
|
||
|| for_each_template_parm (TREE_VALUE (t), fn, data))
|
||
return 1;
|
||
return for_each_template_parm (TREE_CHAIN (t), fn, data);
|
||
|
||
case OVERLOAD:
|
||
if (for_each_template_parm (OVL_FUNCTION (t), fn, data))
|
||
return 1;
|
||
return for_each_template_parm (OVL_CHAIN (t), fn, data);
|
||
|
||
/* constructed type nodes */
|
||
case POINTER_TYPE:
|
||
case REFERENCE_TYPE:
|
||
return for_each_template_parm (TREE_TYPE (t), fn, data);
|
||
|
||
case RECORD_TYPE:
|
||
if (TYPE_PTRMEMFUNC_FLAG (t))
|
||
return for_each_template_parm (TYPE_PTRMEMFUNC_FN_TYPE (t),
|
||
fn, data);
|
||
/* Fall through. */
|
||
|
||
case UNION_TYPE:
|
||
case ENUMERAL_TYPE:
|
||
if (! TYPE_TEMPLATE_INFO (t))
|
||
return 0;
|
||
return for_each_template_parm (TREE_VALUE
|
||
(TYPE_TEMPLATE_INFO (t)),
|
||
fn, data);
|
||
case METHOD_TYPE:
|
||
if (for_each_template_parm (TYPE_METHOD_BASETYPE (t), fn, data))
|
||
return 1;
|
||
/* Fall through. */
|
||
|
||
case FUNCTION_TYPE:
|
||
/* Check the parameter types. Since default arguments are not
|
||
instantiated until they are needed, the TYPE_ARG_TYPES may
|
||
contain expressions that involve template parameters. But,
|
||
no-one should be looking at them yet. And, once they're
|
||
instantiated, they don't contain template parameters, so
|
||
there's no point in looking at them then, either. */
|
||
{
|
||
tree parm;
|
||
|
||
for (parm = TYPE_ARG_TYPES (t); parm; parm = TREE_CHAIN (parm))
|
||
if (for_each_template_parm (TREE_VALUE (parm), fn, data))
|
||
return 1;
|
||
}
|
||
|
||
/* Check the return type, too. */
|
||
return for_each_template_parm (TREE_TYPE (t), fn, data);
|
||
|
||
case ARRAY_TYPE:
|
||
if (for_each_template_parm (TYPE_DOMAIN (t), fn, data))
|
||
return 1;
|
||
return for_each_template_parm (TREE_TYPE (t), fn, data);
|
||
case OFFSET_TYPE:
|
||
if (for_each_template_parm (TYPE_OFFSET_BASETYPE (t), fn, data))
|
||
return 1;
|
||
return for_each_template_parm (TREE_TYPE (t), fn, data);
|
||
|
||
/* decl nodes */
|
||
case TYPE_DECL:
|
||
return for_each_template_parm (TREE_TYPE (t), fn, data);
|
||
|
||
case TEMPLATE_DECL:
|
||
/* A template template parameter is encountered */
|
||
if (DECL_TEMPLATE_TEMPLATE_PARM_P (t))
|
||
return for_each_template_parm (TREE_TYPE (t), fn, data);
|
||
/* Already substituted template template parameter */
|
||
return 0;
|
||
|
||
case CONST_DECL:
|
||
if (for_each_template_parm (DECL_INITIAL (t), fn, data))
|
||
return 1;
|
||
goto check_type_and_context;
|
||
|
||
case FUNCTION_DECL:
|
||
case VAR_DECL:
|
||
/* ??? What about FIELD_DECLs? */
|
||
if (DECL_LANG_SPECIFIC (t) && DECL_TEMPLATE_INFO (t)
|
||
&& for_each_template_parm (DECL_TI_ARGS (t), fn, data))
|
||
return 1;
|
||
/* fall through */
|
||
case PARM_DECL:
|
||
check_type_and_context:
|
||
if (for_each_template_parm (TREE_TYPE (t), fn, data))
|
||
return 1;
|
||
if (DECL_CONTEXT (t)
|
||
&& for_each_template_parm (DECL_CONTEXT (t), fn, data))
|
||
return 1;
|
||
return 0;
|
||
|
||
case CALL_EXPR:
|
||
return (for_each_template_parm (TREE_OPERAND (t, 0), fn, data)
|
||
|| for_each_template_parm (TREE_OPERAND (t, 1), fn, data));
|
||
|
||
case ADDR_EXPR:
|
||
return for_each_template_parm (TREE_OPERAND (t, 0), fn, data);
|
||
|
||
/* template parm nodes */
|
||
case TEMPLATE_TEMPLATE_PARM:
|
||
/* Record template parameters such as `T' inside `TT<T>'. */
|
||
if (TEMPLATE_TEMPLATE_PARM_TEMPLATE_INFO (t)
|
||
&& for_each_template_parm (TYPE_TI_ARGS (t), fn, data))
|
||
return 1;
|
||
case TEMPLATE_TYPE_PARM:
|
||
case TEMPLATE_PARM_INDEX:
|
||
if (fn)
|
||
return (*fn)(t, data);
|
||
else
|
||
return 1;
|
||
|
||
/* simple type nodes */
|
||
case INTEGER_TYPE:
|
||
if (for_each_template_parm (TYPE_MIN_VALUE (t), fn, data))
|
||
return 1;
|
||
return for_each_template_parm (TYPE_MAX_VALUE (t), fn, data);
|
||
|
||
case REAL_TYPE:
|
||
case COMPLEX_TYPE:
|
||
case VOID_TYPE:
|
||
case BOOLEAN_TYPE:
|
||
case NAMESPACE_DECL:
|
||
return 0;
|
||
|
||
/* constants */
|
||
case INTEGER_CST:
|
||
case REAL_CST:
|
||
case STRING_CST:
|
||
return 0;
|
||
|
||
case ERROR_MARK:
|
||
/* Non-error_mark_node ERROR_MARKs are bad things. */
|
||
my_friendly_assert (t == error_mark_node, 274);
|
||
/* NOTREACHED */
|
||
return 0;
|
||
|
||
case LOOKUP_EXPR:
|
||
case TYPENAME_TYPE:
|
||
return 1;
|
||
|
||
case PTRMEM_CST:
|
||
return for_each_template_parm (TREE_TYPE (t), fn, data);
|
||
|
||
case SCOPE_REF:
|
||
return for_each_template_parm (TREE_OPERAND (t, 0), fn, data);
|
||
|
||
case CONSTRUCTOR:
|
||
if (TREE_TYPE (t) && TYPE_PTRMEMFUNC_P (TREE_TYPE (t)))
|
||
return for_each_template_parm (TYPE_PTRMEMFUNC_FN_TYPE
|
||
(TREE_TYPE (t)), fn, data);
|
||
return for_each_template_parm (TREE_OPERAND (t, 1), fn, data);
|
||
|
||
case MODOP_EXPR:
|
||
case CAST_EXPR:
|
||
case REINTERPRET_CAST_EXPR:
|
||
case CONST_CAST_EXPR:
|
||
case STATIC_CAST_EXPR:
|
||
case DYNAMIC_CAST_EXPR:
|
||
case ARROW_EXPR:
|
||
case DOTSTAR_EXPR:
|
||
case TYPEID_EXPR:
|
||
return 1;
|
||
|
||
case SIZEOF_EXPR:
|
||
case ALIGNOF_EXPR:
|
||
return for_each_template_parm (TREE_OPERAND (t, 0), fn, data);
|
||
|
||
default:
|
||
switch (TREE_CODE_CLASS (TREE_CODE (t)))
|
||
{
|
||
case '1':
|
||
case '2':
|
||
case 'e':
|
||
case '<':
|
||
{
|
||
int i;
|
||
for (i = first_rtl_op (TREE_CODE (t)); --i >= 0;)
|
||
if (for_each_template_parm (TREE_OPERAND (t, i), fn, data))
|
||
return 1;
|
||
return 0;
|
||
}
|
||
default:
|
||
break;
|
||
}
|
||
sorry ("testing %s for template parms",
|
||
tree_code_name [(int) TREE_CODE (t)]);
|
||
my_friendly_abort (82);
|
||
/* NOTREACHED */
|
||
return 0;
|
||
}
|
||
}
|
||
|
||
int
|
||
uses_template_parms (t)
|
||
tree t;
|
||
{
|
||
return for_each_template_parm (t, 0, 0);
|
||
}
|
||
|
||
static struct tinst_level *current_tinst_level;
|
||
static struct tinst_level *free_tinst_level;
|
||
static int tinst_depth;
|
||
extern int max_tinst_depth;
|
||
#ifdef GATHER_STATISTICS
|
||
int depth_reached;
|
||
#endif
|
||
int tinst_level_tick;
|
||
int last_template_error_tick;
|
||
|
||
/* Print out all the template instantiations that we are currently
|
||
working on. If ERR, we are being called from cp_thing, so do
|
||
the right thing for an error message. */
|
||
|
||
static void
|
||
print_template_context (err)
|
||
int err;
|
||
{
|
||
struct tinst_level *p = current_tinst_level;
|
||
int line = lineno;
|
||
char *file = input_filename;
|
||
|
||
if (err && p)
|
||
{
|
||
if (current_function_decl != p->decl
|
||
&& current_function_decl != NULL_TREE)
|
||
/* We can get here during the processing of some synthesized
|
||
method. Then, p->decl will be the function that's causing
|
||
the synthesis. */
|
||
;
|
||
else
|
||
{
|
||
if (current_function_decl == p->decl)
|
||
/* Avoid redundancy with the the "In function" line. */;
|
||
else
|
||
fprintf (stderr, "%s: In instantiation of `%s':\n",
|
||
file, decl_as_string (p->decl, 0));
|
||
|
||
line = p->line;
|
||
file = p->file;
|
||
p = p->next;
|
||
}
|
||
}
|
||
|
||
for (; p; p = p->next)
|
||
{
|
||
fprintf (stderr, "%s:%d: instantiated from `%s'\n", file, line,
|
||
decl_as_string (p->decl, 0));
|
||
line = p->line;
|
||
file = p->file;
|
||
}
|
||
fprintf (stderr, "%s:%d: instantiated from here\n", file, line);
|
||
}
|
||
|
||
/* Called from cp_thing to print the template context for an error. */
|
||
|
||
void
|
||
maybe_print_template_context ()
|
||
{
|
||
if (last_template_error_tick == tinst_level_tick
|
||
|| current_tinst_level == 0)
|
||
return;
|
||
|
||
last_template_error_tick = tinst_level_tick;
|
||
print_template_context (1);
|
||
}
|
||
|
||
static int
|
||
push_tinst_level (d)
|
||
tree d;
|
||
{
|
||
struct tinst_level *new;
|
||
|
||
if (tinst_depth >= max_tinst_depth)
|
||
{
|
||
/* If the instantiation in question still has unbound template parms,
|
||
we don't really care if we can't instantiate it, so just return.
|
||
This happens with base instantiation for implicit `typename'. */
|
||
if (uses_template_parms (d))
|
||
return 0;
|
||
|
||
last_template_error_tick = tinst_level_tick;
|
||
error ("template instantiation depth exceeds maximum of %d",
|
||
max_tinst_depth);
|
||
error (" (use -ftemplate-depth-NN to increase the maximum)");
|
||
cp_error (" instantiating `%D'", d);
|
||
|
||
print_template_context (0);
|
||
|
||
return 0;
|
||
}
|
||
|
||
if (free_tinst_level)
|
||
{
|
||
new = free_tinst_level;
|
||
free_tinst_level = new->next;
|
||
}
|
||
else
|
||
new = (struct tinst_level *) xmalloc (sizeof (struct tinst_level));
|
||
|
||
new->decl = d;
|
||
new->line = lineno;
|
||
new->file = input_filename;
|
||
new->next = current_tinst_level;
|
||
current_tinst_level = new;
|
||
|
||
++tinst_depth;
|
||
#ifdef GATHER_STATISTICS
|
||
if (tinst_depth > depth_reached)
|
||
depth_reached = tinst_depth;
|
||
#endif
|
||
|
||
++tinst_level_tick;
|
||
return 1;
|
||
}
|
||
|
||
void
|
||
pop_tinst_level ()
|
||
{
|
||
struct tinst_level *old = current_tinst_level;
|
||
|
||
/* Restore the filename and line number stashed away when we started
|
||
this instantiation. */
|
||
lineno = old->line;
|
||
input_filename = old->file;
|
||
extract_interface_info ();
|
||
|
||
current_tinst_level = old->next;
|
||
old->next = free_tinst_level;
|
||
free_tinst_level = old;
|
||
--tinst_depth;
|
||
++tinst_level_tick;
|
||
}
|
||
|
||
struct tinst_level *
|
||
tinst_for_decl ()
|
||
{
|
||
struct tinst_level *p = current_tinst_level;
|
||
|
||
if (p)
|
||
for (; p->next ; p = p->next )
|
||
;
|
||
return p;
|
||
}
|
||
|
||
/* DECL is a friend FUNCTION_DECL or TEMPLATE_DECL. ARGS is the
|
||
vector of template arguments, as for tsubst.
|
||
|
||
Returns an appropriate tsbust'd friend declaration. */
|
||
|
||
static tree
|
||
tsubst_friend_function (decl, args)
|
||
tree decl;
|
||
tree args;
|
||
{
|
||
tree new_friend;
|
||
int line = lineno;
|
||
char *file = input_filename;
|
||
|
||
lineno = DECL_SOURCE_LINE (decl);
|
||
input_filename = DECL_SOURCE_FILE (decl);
|
||
|
||
if (TREE_CODE (decl) == FUNCTION_DECL
|
||
&& DECL_TEMPLATE_INSTANTIATION (decl)
|
||
&& TREE_CODE (DECL_TI_TEMPLATE (decl)) != TEMPLATE_DECL)
|
||
/* This was a friend declared with an explicit template
|
||
argument list, e.g.:
|
||
|
||
friend void f<>(T);
|
||
|
||
to indicate that f was a template instantiation, not a new
|
||
function declaration. Now, we have to figure out what
|
||
instantiation of what template. */
|
||
{
|
||
tree template_id;
|
||
tree new_args;
|
||
tree tmpl;
|
||
|
||
template_id
|
||
= lookup_template_function (tsubst_expr (DECL_TI_TEMPLATE (decl),
|
||
args, /*complain=*/1,
|
||
NULL_TREE),
|
||
tsubst (DECL_TI_ARGS (decl),
|
||
args, /*complain=*/1,
|
||
NULL_TREE));
|
||
/* FIXME: The decl we create via the next tsubst could be
|
||
created on a temporary obstack. */
|
||
new_friend = tsubst (decl, args, /*complain=*/1, NULL_TREE);
|
||
tmpl = determine_specialization (template_id, new_friend,
|
||
&new_args,
|
||
/*need_member_template=*/0);
|
||
new_friend = instantiate_template (tmpl, new_args);
|
||
goto done;
|
||
}
|
||
|
||
new_friend = tsubst (decl, args, /*complain=*/1, NULL_TREE);
|
||
|
||
/* The NEW_FRIEND will look like an instantiation, to the
|
||
compiler, but is not an instantiation from the point of view of
|
||
the language. For example, we might have had:
|
||
|
||
template <class T> struct S {
|
||
template <class U> friend void f(T, U);
|
||
};
|
||
|
||
Then, in S<int>, template <class U> void f(int, U) is not an
|
||
instantiation of anything. */
|
||
DECL_USE_TEMPLATE (new_friend) = 0;
|
||
if (TREE_CODE (decl) == TEMPLATE_DECL)
|
||
DECL_USE_TEMPLATE (DECL_TEMPLATE_RESULT (new_friend)) = 0;
|
||
|
||
/* The mangled name for the NEW_FRIEND is incorrect. The call to
|
||
tsubst will have resulted in a call to
|
||
set_mangled_name_for_template_decl. But, the function is not a
|
||
template instantiation and should not be mangled like one.
|
||
Therefore, we remangle the function name. We don't have to do
|
||
this if the NEW_FRIEND is a template since
|
||
set_mangled_name_for_template_decl doesn't do anything if the
|
||
function declaration still uses template arguments. */
|
||
if (TREE_CODE (new_friend) != TEMPLATE_DECL)
|
||
{
|
||
set_mangled_name_for_decl (new_friend);
|
||
DECL_RTL (new_friend) = 0;
|
||
make_decl_rtl (new_friend, NULL_PTR, 1);
|
||
}
|
||
|
||
if (DECL_NAMESPACE_SCOPE_P (new_friend))
|
||
{
|
||
tree old_decl;
|
||
tree new_friend_template_info;
|
||
tree new_friend_result_template_info;
|
||
int new_friend_is_defn;
|
||
|
||
/* We must save some information from NEW_FRIEND before calling
|
||
duplicate decls since that function will free NEW_FRIEND if
|
||
possible. */
|
||
new_friend_template_info = DECL_TEMPLATE_INFO (new_friend);
|
||
if (TREE_CODE (new_friend) == TEMPLATE_DECL)
|
||
{
|
||
/* This declaration is a `primary' template. */
|
||
DECL_PRIMARY_TEMPLATE (new_friend) = new_friend;
|
||
|
||
new_friend_is_defn
|
||
= DECL_INITIAL (DECL_RESULT (new_friend)) != NULL_TREE;
|
||
new_friend_result_template_info
|
||
= DECL_TEMPLATE_INFO (DECL_RESULT (new_friend));
|
||
}
|
||
else
|
||
{
|
||
new_friend_is_defn = DECL_INITIAL (new_friend) != NULL_TREE;
|
||
new_friend_result_template_info = NULL_TREE;
|
||
}
|
||
|
||
old_decl = pushdecl_namespace_level (new_friend);
|
||
|
||
if (old_decl != new_friend)
|
||
{
|
||
/* This new friend declaration matched an existing
|
||
declaration. For example, given:
|
||
|
||
template <class T> void f(T);
|
||
template <class U> class C {
|
||
template <class T> friend void f(T) {}
|
||
};
|
||
|
||
the friend declaration actually provides the definition
|
||
of `f', once C has been instantiated for some type. So,
|
||
old_decl will be the out-of-class template declaration,
|
||
while new_friend is the in-class definition.
|
||
|
||
But, if `f' was called before this point, the
|
||
instantiation of `f' will have DECL_TI_ARGS corresponding
|
||
to `T' but not to `U', references to which might appear
|
||
in the definition of `f'. Previously, the most general
|
||
template for an instantiation of `f' was the out-of-class
|
||
version; now it is the in-class version. Therefore, we
|
||
run through all specialization of `f', adding to their
|
||
DECL_TI_ARGS appropriately. In particular, they need a
|
||
new set of outer arguments, corresponding to the
|
||
arguments for this class instantiation.
|
||
|
||
The same situation can arise with something like this:
|
||
|
||
friend void f(int);
|
||
template <class T> class C {
|
||
friend void f(T) {}
|
||
};
|
||
|
||
when `C<int>' is instantiated. Now, `f(int)' is defined
|
||
in the class. */
|
||
|
||
if (!new_friend_is_defn)
|
||
/* On the other hand, if the in-class declaration does
|
||
*not* provide a definition, then we don't want to alter
|
||
existing definitions. We can just leave everything
|
||
alone. */
|
||
;
|
||
else
|
||
{
|
||
/* Overwrite whatever template info was there before, if
|
||
any, with the new template information pertaining to
|
||
the declaration. */
|
||
DECL_TEMPLATE_INFO (old_decl) = new_friend_template_info;
|
||
|
||
if (TREE_CODE (old_decl) != TEMPLATE_DECL)
|
||
/* duplicate_decls will take care of this case. */
|
||
;
|
||
else
|
||
{
|
||
tree t;
|
||
tree new_friend_args;
|
||
|
||
DECL_TEMPLATE_INFO (DECL_RESULT (old_decl))
|
||
= new_friend_result_template_info;
|
||
|
||
new_friend_args = TI_ARGS (new_friend_template_info);
|
||
for (t = DECL_TEMPLATE_SPECIALIZATIONS (old_decl);
|
||
t != NULL_TREE;
|
||
t = TREE_CHAIN (t))
|
||
{
|
||
tree spec = TREE_VALUE (t);
|
||
|
||
DECL_TI_ARGS (spec)
|
||
= add_outermost_template_args (new_friend_args,
|
||
DECL_TI_ARGS (spec));
|
||
DECL_TI_ARGS (spec)
|
||
= copy_to_permanent (DECL_TI_ARGS (spec));
|
||
}
|
||
|
||
/* Now, since specializations are always supposed to
|
||
hang off of the most general template, we must move
|
||
them. */
|
||
t = most_general_template (old_decl);
|
||
if (t != old_decl)
|
||
{
|
||
DECL_TEMPLATE_SPECIALIZATIONS (t)
|
||
= chainon (DECL_TEMPLATE_SPECIALIZATIONS (t),
|
||
DECL_TEMPLATE_SPECIALIZATIONS (old_decl));
|
||
DECL_TEMPLATE_SPECIALIZATIONS (old_decl) = NULL_TREE;
|
||
}
|
||
}
|
||
}
|
||
|
||
/* The information from NEW_FRIEND has been merged into OLD_DECL
|
||
by duplicate_decls. */
|
||
new_friend = old_decl;
|
||
}
|
||
}
|
||
else if (TYPE_SIZE (DECL_CONTEXT (new_friend)))
|
||
{
|
||
/* Check to see that the declaration is really present, and,
|
||
possibly obtain an improved declaration. */
|
||
tree fn = check_classfn (DECL_CONTEXT (new_friend),
|
||
new_friend);
|
||
|
||
if (fn)
|
||
new_friend = fn;
|
||
}
|
||
|
||
done:
|
||
lineno = line;
|
||
input_filename = file;
|
||
return new_friend;
|
||
}
|
||
|
||
/* FRIEND_TMPL is a friend TEMPLATE_DECL. ARGS is the vector of
|
||
template arguments, as for tsubst.
|
||
|
||
Returns an appropriate tsbust'd friend type. */
|
||
|
||
static tree
|
||
tsubst_friend_class (friend_tmpl, args)
|
||
tree friend_tmpl;
|
||
tree args;
|
||
{
|
||
tree friend_type;
|
||
tree tmpl;
|
||
|
||
/* First, we look for a class template. */
|
||
tmpl = lookup_name (DECL_NAME (friend_tmpl), /*prefer_type=*/0);
|
||
|
||
/* But, if we don't find one, it might be because we're in a
|
||
situation like this:
|
||
|
||
template <class T>
|
||
struct S {
|
||
template <class U>
|
||
friend struct S;
|
||
};
|
||
|
||
Here, in the scope of (say) S<int>, `S' is bound to a TYPE_DECL
|
||
for `S<int>', not the TEMPLATE_DECL. */
|
||
if (!tmpl || !DECL_CLASS_TEMPLATE_P (tmpl))
|
||
{
|
||
tmpl = lookup_name (DECL_NAME (friend_tmpl), /*prefer_type=*/1);
|
||
tmpl = maybe_get_template_decl_from_type_decl (tmpl);
|
||
}
|
||
|
||
if (tmpl && DECL_CLASS_TEMPLATE_P (tmpl))
|
||
{
|
||
/* The friend template has already been declared. Just
|
||
check to see that the declarations match, and install any new
|
||
default parameters. We must tsubst the default parameters,
|
||
of course. We only need the innermost template parameters
|
||
because that is all that redeclare_class_template will look
|
||
at. */
|
||
tree parms
|
||
= tsubst_template_parms (DECL_TEMPLATE_PARMS (friend_tmpl),
|
||
args, /*complain=*/1);
|
||
redeclare_class_template (TREE_TYPE (tmpl), parms);
|
||
friend_type = TREE_TYPE (tmpl);
|
||
}
|
||
else
|
||
{
|
||
/* The friend template has not already been declared. In this
|
||
case, the instantiation of the template class will cause the
|
||
injection of this template into the global scope. */
|
||
tmpl = tsubst (friend_tmpl, args, /*complain=*/1, NULL_TREE);
|
||
|
||
/* The new TMPL is not an instantiation of anything, so we
|
||
forget its origins. We don't reset CLASSTYPE_TI_TEMPLATE for
|
||
the new type because that is supposed to be the corresponding
|
||
template decl, i.e., TMPL. */
|
||
DECL_USE_TEMPLATE (tmpl) = 0;
|
||
DECL_TEMPLATE_INFO (tmpl) = NULL_TREE;
|
||
CLASSTYPE_USE_TEMPLATE (TREE_TYPE (tmpl)) = 0;
|
||
|
||
/* Inject this template into the global scope. */
|
||
friend_type = TREE_TYPE (pushdecl_top_level (tmpl));
|
||
}
|
||
|
||
return friend_type;
|
||
}
|
||
|
||
static int
|
||
has_pvbases_p (t, pattern)
|
||
tree t, pattern;
|
||
{
|
||
if (!TYPE_USES_VIRTUAL_BASECLASSES (t))
|
||
return 0;
|
||
|
||
if (TYPE_USES_PVBASES (pattern))
|
||
return 1;
|
||
|
||
for (t = CLASSTYPE_VBASECLASSES (t); t; t = TREE_CHAIN (t))
|
||
if (TYPE_VIRTUAL_P (BINFO_TYPE (t)))
|
||
return 1;
|
||
|
||
return 0;
|
||
}
|
||
|
||
tree
|
||
instantiate_class_template (type)
|
||
tree type;
|
||
{
|
||
tree template, args, pattern, t;
|
||
tree typedecl;
|
||
|
||
if (type == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
if (TYPE_BEING_DEFINED (type) || TYPE_SIZE (type))
|
||
return type;
|
||
|
||
/* We want to allocate temporary vectors of template arguments and
|
||
template argument expressions on the momentary obstack, not on
|
||
the expression obstack. Otherwise, all the space allocated in
|
||
argument coercion and such is simply lost. */
|
||
push_momentary ();
|
||
|
||
/* Figure out which template is being instantiated. */
|
||
template = most_general_template (CLASSTYPE_TI_TEMPLATE (type));
|
||
my_friendly_assert (TREE_CODE (template) == TEMPLATE_DECL, 279);
|
||
|
||
/* Figure out which arguments are being used to do the
|
||
instantiation. */
|
||
args = CLASSTYPE_TI_ARGS (type);
|
||
PARTIAL_INSTANTIATION_P (type) = uses_template_parms (args);
|
||
|
||
if (pedantic && PARTIAL_INSTANTIATION_P (type))
|
||
/* If this is a partial instantiation, then we can't instantiate
|
||
the type; there's no telling whether or not one of the
|
||
template parameters might eventually be instantiated to some
|
||
value that results in a specialization being used. For
|
||
example, consider:
|
||
|
||
template <class T>
|
||
struct S {};
|
||
|
||
template <class U>
|
||
void f(S<U>);
|
||
|
||
template <>
|
||
struct S<int> {};
|
||
|
||
Now, the `S<U>' in `f<int>' is the specialization, not an
|
||
instantiation of the original template. */
|
||
goto end;
|
||
|
||
/* Determine what specialization of the original template to
|
||
instantiate. */
|
||
if (PARTIAL_INSTANTIATION_P (type))
|
||
/* There's no telling which specialization is appropriate at this
|
||
point. Since all peeking at the innards of this partial
|
||
instantiation are extensions (like the "implicit typename"
|
||
extension, which allows users to omit the keyword `typename' on
|
||
names that are declared as types in template base classes), we
|
||
are free to do what we please.
|
||
|
||
Trying to figure out which partial instantiation to use can
|
||
cause a crash. (Some of the template arguments don't even have
|
||
types.) So, we just use the most general version. */
|
||
t = NULL_TREE;
|
||
else
|
||
{
|
||
t = most_specialized_class (template, args);
|
||
|
||
if (t == error_mark_node)
|
||
{
|
||
const char *str = "candidates are:";
|
||
cp_error ("ambiguous class template instantiation for `%#T'", type);
|
||
for (t = DECL_TEMPLATE_SPECIALIZATIONS (template); t;
|
||
t = TREE_CHAIN (t))
|
||
{
|
||
if (get_class_bindings (TREE_VALUE (t), TREE_PURPOSE (t),
|
||
args))
|
||
{
|
||
cp_error_at ("%s %+#T", str, TREE_TYPE (t));
|
||
str = " ";
|
||
}
|
||
}
|
||
TYPE_BEING_DEFINED (type) = 1;
|
||
type = error_mark_node;
|
||
goto end;
|
||
}
|
||
}
|
||
|
||
if (t)
|
||
pattern = TREE_TYPE (t);
|
||
else
|
||
pattern = TREE_TYPE (template);
|
||
|
||
/* If the template we're instantiating is incomplete, then clearly
|
||
there's nothing we can do. */
|
||
if (TYPE_SIZE (pattern) == NULL_TREE)
|
||
goto end;
|
||
|
||
/* If this is a partial instantiation, don't tsubst anything. We will
|
||
only use this type for implicit typename, so the actual contents don't
|
||
matter. All that matters is whether a particular name is a type. */
|
||
if (PARTIAL_INSTANTIATION_P (type))
|
||
{
|
||
/* The fields set here must be kept in sync with those cleared
|
||
in begin_class_definition. */
|
||
TYPE_BINFO_BASETYPES (type) = TYPE_BINFO_BASETYPES (pattern);
|
||
TYPE_FIELDS (type) = TYPE_FIELDS (pattern);
|
||
TYPE_METHODS (type) = TYPE_METHODS (pattern);
|
||
CLASSTYPE_TAGS (type) = CLASSTYPE_TAGS (pattern);
|
||
/* Pretend that the type is complete, so that we will look
|
||
inside it during name lookup and such. */
|
||
TYPE_SIZE (type) = integer_zero_node;
|
||
goto end;
|
||
}
|
||
|
||
/* If we've recursively instantiated too many templates, stop. */
|
||
if (! push_tinst_level (type))
|
||
goto end;
|
||
|
||
/* Now we're really doing the instantiation. Mark the type as in
|
||
the process of being defined. */
|
||
TYPE_BEING_DEFINED (type) = 1;
|
||
|
||
maybe_push_to_top_level (uses_template_parms (type));
|
||
|
||
if (t)
|
||
{
|
||
/* This TYPE is actually a instantiation of of a partial
|
||
specialization. We replace the innermost set of ARGS with
|
||
the arguments appropriate for substitution. For example,
|
||
given:
|
||
|
||
template <class T> struct S {};
|
||
template <class T> struct S<T*> {};
|
||
|
||
and supposing that we are instantiating S<int*>, ARGS will
|
||
present be {int*} but we need {int}. */
|
||
tree inner_args
|
||
= get_class_bindings (TREE_VALUE (t), TREE_PURPOSE (t),
|
||
args);
|
||
|
||
/* If there were multiple levels in ARGS, replacing the
|
||
innermost level would alter CLASSTYPE_TI_ARGS, which we don't
|
||
want, so we make a copy first. */
|
||
if (TMPL_ARGS_HAVE_MULTIPLE_LEVELS (args))
|
||
{
|
||
args = copy_node (args);
|
||
SET_TMPL_ARGS_LEVEL (args, TMPL_ARGS_DEPTH (args), inner_args);
|
||
}
|
||
else
|
||
args = inner_args;
|
||
}
|
||
|
||
if (flag_external_templates)
|
||
{
|
||
if (flag_alt_external_templates)
|
||
{
|
||
CLASSTYPE_INTERFACE_ONLY (type) = interface_only;
|
||
SET_CLASSTYPE_INTERFACE_UNKNOWN_X (type, interface_unknown);
|
||
CLASSTYPE_VTABLE_NEEDS_WRITING (type)
|
||
= (! CLASSTYPE_INTERFACE_ONLY (type)
|
||
&& CLASSTYPE_INTERFACE_KNOWN (type));
|
||
}
|
||
else
|
||
{
|
||
CLASSTYPE_INTERFACE_ONLY (type) = CLASSTYPE_INTERFACE_ONLY (pattern);
|
||
SET_CLASSTYPE_INTERFACE_UNKNOWN_X
|
||
(type, CLASSTYPE_INTERFACE_UNKNOWN (pattern));
|
||
CLASSTYPE_VTABLE_NEEDS_WRITING (type)
|
||
= (! CLASSTYPE_INTERFACE_ONLY (type)
|
||
&& CLASSTYPE_INTERFACE_KNOWN (type));
|
||
}
|
||
}
|
||
else
|
||
{
|
||
SET_CLASSTYPE_INTERFACE_UNKNOWN (type);
|
||
CLASSTYPE_VTABLE_NEEDS_WRITING (type) = 1;
|
||
}
|
||
|
||
TYPE_HAS_CONSTRUCTOR (type) = TYPE_HAS_CONSTRUCTOR (pattern);
|
||
TYPE_HAS_DESTRUCTOR (type) = TYPE_HAS_DESTRUCTOR (pattern);
|
||
TYPE_OVERLOADS_CALL_EXPR (type) = TYPE_OVERLOADS_CALL_EXPR (pattern);
|
||
TYPE_OVERLOADS_ARRAY_REF (type) = TYPE_OVERLOADS_ARRAY_REF (pattern);
|
||
TYPE_OVERLOADS_ARROW (type) = TYPE_OVERLOADS_ARROW (pattern);
|
||
TYPE_GETS_NEW (type) = TYPE_GETS_NEW (pattern);
|
||
TYPE_GETS_DELETE (type) = TYPE_GETS_DELETE (pattern);
|
||
TYPE_VEC_DELETE_TAKES_SIZE (type) = TYPE_VEC_DELETE_TAKES_SIZE (pattern);
|
||
TYPE_HAS_ASSIGN_REF (type) = TYPE_HAS_ASSIGN_REF (pattern);
|
||
TYPE_HAS_CONST_ASSIGN_REF (type) = TYPE_HAS_CONST_ASSIGN_REF (pattern);
|
||
TYPE_HAS_ABSTRACT_ASSIGN_REF (type) = TYPE_HAS_ABSTRACT_ASSIGN_REF (pattern);
|
||
TYPE_HAS_INIT_REF (type) = TYPE_HAS_INIT_REF (pattern);
|
||
TYPE_HAS_CONST_INIT_REF (type) = TYPE_HAS_CONST_INIT_REF (pattern);
|
||
TYPE_HAS_DEFAULT_CONSTRUCTOR (type) = TYPE_HAS_DEFAULT_CONSTRUCTOR (pattern);
|
||
TYPE_HAS_CONVERSION (type) = TYPE_HAS_CONVERSION (pattern);
|
||
TYPE_USES_COMPLEX_INHERITANCE (type)
|
||
= TYPE_USES_COMPLEX_INHERITANCE (pattern);
|
||
TYPE_USES_MULTIPLE_INHERITANCE (type)
|
||
= TYPE_USES_MULTIPLE_INHERITANCE (pattern);
|
||
TYPE_USES_VIRTUAL_BASECLASSES (type)
|
||
= TYPE_USES_VIRTUAL_BASECLASSES (pattern);
|
||
TYPE_PACKED (type) = TYPE_PACKED (pattern);
|
||
TYPE_ALIGN (type) = TYPE_ALIGN (pattern);
|
||
TYPE_FOR_JAVA (type) = TYPE_FOR_JAVA (pattern); /* For libjava's JArray<T> */
|
||
if (ANON_UNION_TYPE_P (pattern))
|
||
SET_ANON_UNION_TYPE_P (type);
|
||
|
||
/* We must copy the arguments to the permanent obstack since
|
||
during the tsubst'ing below they may wind up in the
|
||
DECL_TI_ARGS of some instantiated member template. */
|
||
args = copy_to_permanent (args);
|
||
|
||
if (TYPE_BINFO_BASETYPES (pattern))
|
||
{
|
||
tree base_list = NULL_TREE;
|
||
tree pbases = TYPE_BINFO_BASETYPES (pattern);
|
||
int i;
|
||
|
||
/* Substitute into each of the bases to determine the actual
|
||
basetypes. */
|
||
for (i = 0; i < TREE_VEC_LENGTH (pbases); ++i)
|
||
{
|
||
tree base;
|
||
tree access;
|
||
tree pbase;
|
||
|
||
pbase = TREE_VEC_ELT (pbases, i);
|
||
|
||
/* Substitue to figure out the base class. */
|
||
base = tsubst (BINFO_TYPE (pbase), args,
|
||
/*complain=*/1, NULL_TREE);
|
||
if (base == error_mark_node)
|
||
continue;
|
||
|
||
/* Calculate the correct access node. */
|
||
if (TREE_VIA_VIRTUAL (pbase))
|
||
{
|
||
if (TREE_VIA_PUBLIC (pbase))
|
||
access = access_public_virtual_node;
|
||
else if (TREE_VIA_PROTECTED (pbase))
|
||
access = access_protected_virtual_node;
|
||
else
|
||
access = access_private_virtual_node;
|
||
}
|
||
else
|
||
{
|
||
if (TREE_VIA_PUBLIC (pbase))
|
||
access = access_public_node;
|
||
else if (TREE_VIA_PROTECTED (pbase))
|
||
access = access_protected_node;
|
||
else
|
||
access = access_private_node;
|
||
}
|
||
|
||
base_list = tree_cons (access, base, base_list);
|
||
}
|
||
|
||
/* The list is now in reverse order; correct that. */
|
||
base_list = nreverse (base_list);
|
||
|
||
/* Now call xref_basetypes to set up all the base-class
|
||
information. */
|
||
xref_basetypes (TREE_CODE (pattern) == RECORD_TYPE
|
||
? (CLASSTYPE_DECLARED_CLASS (pattern)
|
||
? class_type_node : record_type_node)
|
||
: union_type_node,
|
||
DECL_NAME (TYPE_NAME (pattern)),
|
||
type,
|
||
base_list);
|
||
}
|
||
|
||
/* Now that our base classes are set up, enter the scope of the
|
||
class, so that name lookups into base classes, etc. will work
|
||
corectly. This is precisely analagous to what we do in
|
||
begin_class_definition when defining an ordinary non-template
|
||
class. */
|
||
pushclass (type, 1);
|
||
|
||
for (t = CLASSTYPE_TAGS (pattern); t; t = TREE_CHAIN (t))
|
||
{
|
||
tree tag = TREE_VALUE (t);
|
||
tree name = TYPE_IDENTIFIER (tag);
|
||
tree newtag;
|
||
|
||
newtag = tsubst (tag, args, /*complain=*/1, NULL_TREE);
|
||
if (TREE_CODE (newtag) != ENUMERAL_TYPE)
|
||
{
|
||
if (TYPE_LANG_SPECIFIC (tag) && CLASSTYPE_IS_TEMPLATE (tag))
|
||
/* Unfortunately, lookup_template_class sets
|
||
CLASSTYPE_IMPLICIT_INSTANTIATION for a partial
|
||
instantiation (i.e., for the type of a member template
|
||
class nested within a template class.) This behavior is
|
||
required for maybe_process_partial_specialization to work
|
||
correctly, but is not accurate in this case; the TAG is not
|
||
an instantiation of anything. (The corresponding
|
||
TEMPLATE_DECL is an instantiation, but the TYPE is not.) */
|
||
CLASSTYPE_USE_TEMPLATE (newtag) = 0;
|
||
|
||
/* Now, we call pushtag to put this NEWTAG into the scope of
|
||
TYPE. We first set up the IDENTIFIER_TYPE_VALUE to avoid
|
||
pushtag calling push_template_decl. We don't have to do
|
||
this for enums because it will already have been done in
|
||
tsubst_enum. */
|
||
if (name)
|
||
SET_IDENTIFIER_TYPE_VALUE (name, newtag);
|
||
pushtag (name, newtag, /*globalize=*/0);
|
||
}
|
||
}
|
||
|
||
/* Don't replace enum constants here. */
|
||
for (t = TYPE_FIELDS (pattern); t; t = TREE_CHAIN (t))
|
||
if (TREE_CODE (t) != CONST_DECL)
|
||
{
|
||
tree r;
|
||
|
||
/* The the file and line for this declaration, to assist in
|
||
error message reporting. Since we called push_tinst_level
|
||
above, we don't need to restore these. */
|
||
lineno = DECL_SOURCE_LINE (t);
|
||
input_filename = DECL_SOURCE_FILE (t);
|
||
|
||
r = tsubst (t, args, /*complain=*/1, NULL_TREE);
|
||
if (TREE_CODE (r) == VAR_DECL)
|
||
{
|
||
tree init;
|
||
|
||
if (DECL_DEFINED_IN_CLASS_P (r))
|
||
init = tsubst_expr (DECL_INITIAL (t), args,
|
||
/*complain=*/1, NULL_TREE);
|
||
else
|
||
init = NULL_TREE;
|
||
|
||
finish_static_data_member_decl (r, init,
|
||
/*asmspec_tree=*/NULL_TREE,
|
||
/*need_pop=*/0,
|
||
/*flags=*/0);
|
||
|
||
if (DECL_DEFINED_IN_CLASS_P (r))
|
||
check_static_variable_definition (r, TREE_TYPE (r));
|
||
}
|
||
|
||
/* R will have a TREE_CHAIN if and only if it has already been
|
||
processed by finish_member_declaration. This can happen
|
||
if, for example, it is a TYPE_DECL for a class-scoped
|
||
ENUMERAL_TYPE; such a thing will already have been added to
|
||
the field list by tsubst_enum above. */
|
||
if (!TREE_CHAIN (r))
|
||
{
|
||
set_current_access_from_decl (r);
|
||
finish_member_declaration (r);
|
||
}
|
||
}
|
||
|
||
/* After we have calculated the bases, we can now compute whether we
|
||
have polymorphic vbases. This needs to happen before we
|
||
instantiate the methods, because the constructors may take
|
||
additional arguments. */
|
||
if (flag_vtable_thunks >= 2)
|
||
TYPE_USES_PVBASES (type) = has_pvbases_p (type, pattern);
|
||
|
||
/* Set up the list (TYPE_METHODS) and vector (CLASSTYPE_METHOD_VEC)
|
||
for this instantiation. */
|
||
for (t = TYPE_METHODS (pattern); t; t = TREE_CHAIN (t))
|
||
{
|
||
tree r = tsubst (t, args, /*complain=*/1, NULL_TREE);
|
||
set_current_access_from_decl (r);
|
||
finish_member_declaration (r);
|
||
}
|
||
|
||
/* Construct the DECL_FRIENDLIST for the new class type. */
|
||
typedecl = TYPE_MAIN_DECL (type);
|
||
for (t = DECL_FRIENDLIST (TYPE_MAIN_DECL (pattern));
|
||
t != NULL_TREE;
|
||
t = TREE_CHAIN (t))
|
||
{
|
||
tree friends;
|
||
|
||
for (friends = TREE_VALUE (t);
|
||
friends != NULL_TREE;
|
||
friends = TREE_CHAIN (friends))
|
||
if (TREE_PURPOSE (friends) == error_mark_node)
|
||
add_friend (type,
|
||
tsubst_friend_function (TREE_VALUE (friends),
|
||
args));
|
||
else
|
||
add_friends (type,
|
||
tsubst_copy (TREE_PURPOSE (t), args,
|
||
/*complain=*/1, NULL_TREE),
|
||
tsubst (TREE_PURPOSE (friends), args,
|
||
/*complain=*/1, NULL_TREE));
|
||
}
|
||
|
||
for (t = CLASSTYPE_FRIEND_CLASSES (pattern);
|
||
t != NULL_TREE;
|
||
t = TREE_CHAIN (t))
|
||
{
|
||
tree friend_type = TREE_VALUE (t);
|
||
tree new_friend_type;
|
||
|
||
if (TREE_CODE (friend_type) == TEMPLATE_DECL)
|
||
new_friend_type = tsubst_friend_class (friend_type, args);
|
||
else if (uses_template_parms (friend_type))
|
||
new_friend_type = tsubst (friend_type, args, /*complain=*/1,
|
||
NULL_TREE);
|
||
else
|
||
/* The call to xref_tag_from_type does injection for friend
|
||
classes. */
|
||
new_friend_type =
|
||
xref_tag_from_type (friend_type, NULL_TREE, 1);
|
||
|
||
|
||
if (TREE_CODE (friend_type) == TEMPLATE_DECL)
|
||
/* Trick make_friend_class into realizing that the friend
|
||
we're adding is a template, not an ordinary class. It's
|
||
important that we use make_friend_class since it will
|
||
perform some error-checking and output cross-reference
|
||
information. */
|
||
++processing_template_decl;
|
||
|
||
make_friend_class (type, new_friend_type);
|
||
|
||
if (TREE_CODE (friend_type) == TEMPLATE_DECL)
|
||
--processing_template_decl;
|
||
}
|
||
|
||
/* This does injection for friend functions. */
|
||
if (!processing_template_decl)
|
||
{
|
||
t = tsubst (DECL_TEMPLATE_INJECT (template), args,
|
||
/*complain=*/1, NULL_TREE);
|
||
|
||
for (; t; t = TREE_CHAIN (t))
|
||
{
|
||
tree d = TREE_VALUE (t);
|
||
|
||
if (TREE_CODE (d) == TYPE_DECL)
|
||
/* Already injected. */;
|
||
else
|
||
pushdecl (d);
|
||
}
|
||
}
|
||
|
||
for (t = TYPE_FIELDS (type); t; t = TREE_CHAIN (t))
|
||
if (TREE_CODE (t) == FIELD_DECL)
|
||
{
|
||
TREE_TYPE (t) = complete_type (TREE_TYPE (t));
|
||
require_complete_type (t);
|
||
}
|
||
|
||
/* Set the file and line number information to whatever is given for
|
||
the class itself. This puts error messages involving generated
|
||
implicit functions at a predictable point, and the same point
|
||
that would be used for non-template classes. */
|
||
lineno = DECL_SOURCE_LINE (typedecl);
|
||
input_filename = DECL_SOURCE_FILE (typedecl);
|
||
|
||
unreverse_member_declarations (type);
|
||
finish_struct_1 (type, 0);
|
||
CLASSTYPE_GOT_SEMICOLON (type) = 1;
|
||
|
||
/* Clear this now so repo_template_used is happy. */
|
||
TYPE_BEING_DEFINED (type) = 0;
|
||
repo_template_used (type);
|
||
|
||
popclass ();
|
||
pop_from_top_level ();
|
||
pop_tinst_level ();
|
||
|
||
end:
|
||
pop_momentary ();
|
||
|
||
return type;
|
||
}
|
||
|
||
static int
|
||
list_eq (t1, t2)
|
||
tree t1, t2;
|
||
{
|
||
if (t1 == NULL_TREE)
|
||
return t2 == NULL_TREE;
|
||
if (t2 == NULL_TREE)
|
||
return 0;
|
||
/* Don't care if one declares its arg const and the other doesn't -- the
|
||
main variant of the arg type is all that matters. */
|
||
if (TYPE_MAIN_VARIANT (TREE_VALUE (t1))
|
||
!= TYPE_MAIN_VARIANT (TREE_VALUE (t2)))
|
||
return 0;
|
||
return list_eq (TREE_CHAIN (t1), TREE_CHAIN (t2));
|
||
}
|
||
|
||
/* If arg is a non-type template parameter that does not depend on template
|
||
arguments, fold it like we weren't in the body of a template. */
|
||
|
||
static tree
|
||
maybe_fold_nontype_arg (arg)
|
||
tree arg;
|
||
{
|
||
if (TREE_CODE_CLASS (TREE_CODE (arg)) != 't'
|
||
&& !uses_template_parms (arg))
|
||
{
|
||
/* Sometimes, one of the args was an expression involving a
|
||
template constant parameter, like N - 1. Now that we've
|
||
tsubst'd, we might have something like 2 - 1. This will
|
||
confuse lookup_template_class, so we do constant folding
|
||
here. We have to unset processing_template_decl, to
|
||
fool build_expr_from_tree() into building an actual
|
||
tree. */
|
||
|
||
int saved_processing_template_decl = processing_template_decl;
|
||
processing_template_decl = 0;
|
||
arg = fold (build_expr_from_tree (arg));
|
||
processing_template_decl = saved_processing_template_decl;
|
||
}
|
||
return arg;
|
||
}
|
||
|
||
/* Return the TREE_VEC with the arguments for the innermost template header,
|
||
where ARGS is either that or the VEC of VECs for all the
|
||
arguments. */
|
||
|
||
tree
|
||
innermost_args (args)
|
||
tree args;
|
||
{
|
||
return TMPL_ARGS_LEVEL (args, TMPL_ARGS_DEPTH (args));
|
||
}
|
||
|
||
/* Substitute ARGS into the vector of template arguments T. */
|
||
|
||
static tree
|
||
tsubst_template_arg_vector (t, args, complain)
|
||
tree t;
|
||
tree args;
|
||
int complain;
|
||
{
|
||
int len = TREE_VEC_LENGTH (t), need_new = 0, i;
|
||
tree *elts = (tree *) alloca (len * sizeof (tree));
|
||
|
||
bzero ((char *) elts, len * sizeof (tree));
|
||
|
||
for (i = 0; i < len; i++)
|
||
{
|
||
if (TREE_VEC_ELT (t, i) != NULL_TREE
|
||
&& TREE_CODE (TREE_VEC_ELT (t, i)) == TREE_VEC)
|
||
elts[i] = tsubst_template_arg_vector (TREE_VEC_ELT (t, i),
|
||
args, complain);
|
||
else
|
||
elts[i] = maybe_fold_nontype_arg
|
||
(tsubst_expr (TREE_VEC_ELT (t, i), args, complain,
|
||
NULL_TREE));
|
||
|
||
if (elts[i] != TREE_VEC_ELT (t, i))
|
||
need_new = 1;
|
||
}
|
||
|
||
if (!need_new)
|
||
return t;
|
||
|
||
t = make_temp_vec (len);
|
||
for (i = 0; i < len; i++)
|
||
TREE_VEC_ELT (t, i) = elts[i];
|
||
|
||
return t;
|
||
}
|
||
|
||
/* Return the result of substituting ARGS into the template parameters
|
||
given by PARMS. If there are m levels of ARGS and m + n levels of
|
||
PARMS, then the result will contain n levels of PARMS. For
|
||
example, if PARMS is `template <class T> template <class U>
|
||
template <T*, U, class V>' and ARGS is {{int}, {double}} then the
|
||
result will be `template <int*, double, class V>'. */
|
||
|
||
static tree
|
||
tsubst_template_parms (parms, args, complain)
|
||
tree parms;
|
||
tree args;
|
||
int complain;
|
||
{
|
||
tree r;
|
||
tree* new_parms = &r;
|
||
|
||
for (new_parms = &r;
|
||
TMPL_PARMS_DEPTH (parms) > TMPL_ARGS_DEPTH (args);
|
||
new_parms = &(TREE_CHAIN (*new_parms)),
|
||
parms = TREE_CHAIN (parms))
|
||
{
|
||
tree new_vec =
|
||
make_tree_vec (TREE_VEC_LENGTH (TREE_VALUE (parms)));
|
||
int i;
|
||
|
||
for (i = 0; i < TREE_VEC_LENGTH (new_vec); ++i)
|
||
{
|
||
tree default_value =
|
||
TREE_PURPOSE (TREE_VEC_ELT (TREE_VALUE (parms), i));
|
||
tree parm_decl =
|
||
TREE_VALUE (TREE_VEC_ELT (TREE_VALUE (parms), i));
|
||
|
||
TREE_VEC_ELT (new_vec, i)
|
||
= build_tree_list (tsubst (default_value, args, complain,
|
||
NULL_TREE),
|
||
tsubst (parm_decl, args, complain,
|
||
NULL_TREE));
|
||
}
|
||
|
||
*new_parms =
|
||
tree_cons (build_int_2 (0, (TMPL_PARMS_DEPTH (parms)
|
||
- TMPL_ARGS_DEPTH (args))),
|
||
new_vec, NULL_TREE);
|
||
}
|
||
|
||
return r;
|
||
}
|
||
|
||
/* Substitute the ARGS into the indicated aggregate (or enumeration)
|
||
type T. If T is not an aggregate or enumeration type, it is
|
||
handled as if by tsubst. IN_DECL is as for tsubst. If
|
||
ENTERING_SCOPE is non-zero, T is the context for a template which
|
||
we are presently tsubst'ing. Return the subsituted value. */
|
||
|
||
static tree
|
||
tsubst_aggr_type (t, args, complain, in_decl, entering_scope)
|
||
tree t;
|
||
tree args;
|
||
int complain;
|
||
tree in_decl;
|
||
int entering_scope;
|
||
{
|
||
if (t == NULL_TREE)
|
||
return NULL_TREE;
|
||
|
||
switch (TREE_CODE (t))
|
||
{
|
||
case RECORD_TYPE:
|
||
if (TYPE_PTRMEMFUNC_P (t))
|
||
{
|
||
tree r = build_ptrmemfunc_type
|
||
(tsubst (TYPE_PTRMEMFUNC_FN_TYPE (t), args, complain, in_decl));
|
||
return cp_build_qualified_type (r, TYPE_QUALS (t));
|
||
}
|
||
|
||
/* else fall through */
|
||
case ENUMERAL_TYPE:
|
||
case UNION_TYPE:
|
||
if (TYPE_TEMPLATE_INFO (t))
|
||
{
|
||
tree argvec;
|
||
tree context;
|
||
tree r;
|
||
|
||
/* First, determine the context for the type we are looking
|
||
up. */
|
||
if (TYPE_CONTEXT (t) != NULL_TREE)
|
||
context = tsubst_aggr_type (TYPE_CONTEXT (t), args,
|
||
complain,
|
||
in_decl, /*entering_scope=*/1);
|
||
else
|
||
context = NULL_TREE;
|
||
|
||
/* Then, figure out what arguments are appropriate for the
|
||
type we are trying to find. For example, given:
|
||
|
||
template <class T> struct S;
|
||
template <class T, class U> void f(T, U) { S<U> su; }
|
||
|
||
and supposing that we are instantiating f<int, double>,
|
||
then our ARGS will be {int, double}, but, when looking up
|
||
S we only want {double}. */
|
||
push_momentary ();
|
||
argvec = tsubst_template_arg_vector (TYPE_TI_ARGS (t), args,
|
||
complain);
|
||
|
||
r = lookup_template_class (t, argvec, in_decl, context,
|
||
entering_scope);
|
||
pop_momentary ();
|
||
|
||
return cp_build_qualified_type (r, TYPE_QUALS (t));
|
||
}
|
||
else
|
||
/* This is not a template type, so there's nothing to do. */
|
||
return t;
|
||
|
||
default:
|
||
return tsubst (t, args, complain, in_decl);
|
||
}
|
||
}
|
||
|
||
/* Substitute the ARGS into the T, which is a _DECL. TYPE is the
|
||
(already computed) substitution of ARGS into TREE_TYPE (T), if
|
||
appropriate. Return the result of the substitution. IN_DECL is as
|
||
for tsubst. */
|
||
|
||
static tree
|
||
tsubst_decl (t, args, type, in_decl)
|
||
tree t;
|
||
tree args;
|
||
tree type;
|
||
tree in_decl;
|
||
{
|
||
int saved_lineno;
|
||
char* saved_filename;
|
||
tree r = NULL_TREE;
|
||
|
||
/* Set the filename and linenumber to improve error-reporting. */
|
||
saved_lineno = lineno;
|
||
saved_filename = input_filename;
|
||
lineno = DECL_SOURCE_LINE (t);
|
||
input_filename = DECL_SOURCE_FILE (t);
|
||
|
||
switch (TREE_CODE (t))
|
||
{
|
||
case TEMPLATE_DECL:
|
||
{
|
||
/* We can get here when processing a member template function
|
||
of a template class. */
|
||
tree decl = DECL_TEMPLATE_RESULT (t);
|
||
tree spec;
|
||
int is_template_template_parm = DECL_TEMPLATE_TEMPLATE_PARM_P (t);
|
||
|
||
if (!is_template_template_parm)
|
||
{
|
||
/* We might already have an instance of this template.
|
||
The ARGS are for the surrounding class type, so the
|
||
full args contain the tsubst'd args for the context,
|
||
plus the innermost args from the template decl. */
|
||
tree tmpl_args = DECL_CLASS_TEMPLATE_P (t)
|
||
? CLASSTYPE_TI_ARGS (TREE_TYPE (t))
|
||
: DECL_TI_ARGS (DECL_RESULT (t));
|
||
tree full_args;
|
||
|
||
push_momentary ();
|
||
full_args = tsubst_template_arg_vector (tmpl_args, args,
|
||
/*complain=*/1);
|
||
|
||
/* tsubst_template_arg_vector doesn't copy the vector if
|
||
nothing changed. But, *something* should have
|
||
changed. */
|
||
my_friendly_assert (full_args != tmpl_args, 0);
|
||
|
||
spec = retrieve_specialization (t, full_args);
|
||
pop_momentary ();
|
||
if (spec != NULL_TREE)
|
||
{
|
||
r = spec;
|
||
break;
|
||
}
|
||
}
|
||
|
||
/* Make a new template decl. It will be similar to the
|
||
original, but will record the current template arguments.
|
||
We also create a new function declaration, which is just
|
||
like the old one, but points to this new template, rather
|
||
than the old one. */
|
||
r = copy_node (t);
|
||
copy_lang_decl (r);
|
||
my_friendly_assert (DECL_LANG_SPECIFIC (r) != 0, 0);
|
||
TREE_CHAIN (r) = NULL_TREE;
|
||
|
||
if (is_template_template_parm)
|
||
{
|
||
tree new_decl = tsubst (decl, args, /*complain=*/1, in_decl);
|
||
DECL_RESULT (r) = new_decl;
|
||
TREE_TYPE (r) = TREE_TYPE (new_decl);
|
||
break;
|
||
}
|
||
|
||
DECL_CONTEXT (r)
|
||
= tsubst_aggr_type (DECL_CONTEXT (t), args, /*complain=*/1,
|
||
in_decl, /*entering_scope=*/1);
|
||
DECL_CLASS_CONTEXT (r)
|
||
= tsubst_aggr_type (DECL_CLASS_CONTEXT (t), args,
|
||
/*complain=*/1, in_decl,
|
||
/*entering_scope=*/1);
|
||
DECL_TEMPLATE_INFO (r) = build_tree_list (t, args);
|
||
|
||
if (TREE_CODE (decl) == TYPE_DECL)
|
||
{
|
||
tree new_type = tsubst (TREE_TYPE (t), args,
|
||
/*complain=*/1, in_decl);
|
||
TREE_TYPE (r) = new_type;
|
||
CLASSTYPE_TI_TEMPLATE (new_type) = r;
|
||
DECL_RESULT (r) = TYPE_MAIN_DECL (new_type);
|
||
DECL_TI_ARGS (r) = CLASSTYPE_TI_ARGS (new_type);
|
||
}
|
||
else
|
||
{
|
||
tree new_decl = tsubst (decl, args, /*complain=*/1, in_decl);
|
||
DECL_RESULT (r) = new_decl;
|
||
DECL_TI_TEMPLATE (new_decl) = r;
|
||
TREE_TYPE (r) = TREE_TYPE (new_decl);
|
||
DECL_TI_ARGS (r) = DECL_TI_ARGS (new_decl);
|
||
}
|
||
|
||
SET_DECL_IMPLICIT_INSTANTIATION (r);
|
||
DECL_TEMPLATE_INSTANTIATIONS (r) = NULL_TREE;
|
||
DECL_TEMPLATE_SPECIALIZATIONS (r) = NULL_TREE;
|
||
|
||
/* The template parameters for this new template are all the
|
||
template parameters for the old template, except the
|
||
outermost level of parameters. */
|
||
DECL_TEMPLATE_PARMS (r)
|
||
= tsubst_template_parms (DECL_TEMPLATE_PARMS (t), args,
|
||
/*complain=*/1);
|
||
|
||
if (PRIMARY_TEMPLATE_P (t))
|
||
DECL_PRIMARY_TEMPLATE (r) = r;
|
||
|
||
/* We don't partially instantiate partial specializations. */
|
||
if (TREE_CODE (decl) == TYPE_DECL)
|
||
break;
|
||
|
||
for (spec = DECL_TEMPLATE_SPECIALIZATIONS (t);
|
||
spec != NULL_TREE;
|
||
spec = TREE_CHAIN (spec))
|
||
{
|
||
/* It helps to consider example here. Consider:
|
||
|
||
template <class T>
|
||
struct S {
|
||
template <class U>
|
||
void f(U u);
|
||
|
||
template <>
|
||
void f(T* t) {}
|
||
};
|
||
|
||
Now, for example, we are instantiating S<int>::f(U u).
|
||
We want to make a template:
|
||
|
||
template <class U>
|
||
void S<int>::f(U);
|
||
|
||
It will have a specialization, for the case U = int*, of
|
||
the form:
|
||
|
||
template <>
|
||
void S<int>::f<int*>(int*);
|
||
|
||
This specialization will be an instantiation of
|
||
the specialization given in the declaration of S, with
|
||
argument list int*. */
|
||
|
||
tree fn = TREE_VALUE (spec);
|
||
tree spec_args;
|
||
tree new_fn;
|
||
|
||
if (!DECL_TEMPLATE_SPECIALIZATION (fn))
|
||
/* Instantiations are on the same list, but they're of
|
||
no concern to us. */
|
||
continue;
|
||
|
||
if (TREE_CODE (fn) != TEMPLATE_DECL)
|
||
/* A full specialization. There's no need to record
|
||
that here. */
|
||
continue;
|
||
|
||
spec_args = tsubst (DECL_TI_ARGS (fn), args,
|
||
/*complain=*/1, in_decl);
|
||
new_fn = tsubst (DECL_RESULT (most_general_template (fn)),
|
||
spec_args, /*complain=*/1, in_decl);
|
||
DECL_TI_TEMPLATE (new_fn) = fn;
|
||
register_specialization (new_fn, r,
|
||
innermost_args (spec_args));
|
||
}
|
||
|
||
/* Record this partial instantiation. */
|
||
register_specialization (r, t,
|
||
DECL_TI_ARGS (DECL_RESULT (r)));
|
||
|
||
}
|
||
break;
|
||
|
||
case FUNCTION_DECL:
|
||
{
|
||
tree ctx;
|
||
tree argvec = NULL_TREE;
|
||
tree *friends;
|
||
tree gen_tmpl;
|
||
int member;
|
||
int args_depth;
|
||
int parms_depth;
|
||
|
||
/* Nobody should be tsubst'ing into non-template functions. */
|
||
my_friendly_assert (DECL_TEMPLATE_INFO (t) != NULL_TREE, 0);
|
||
|
||
if (TREE_CODE (DECL_TI_TEMPLATE (t)) == TEMPLATE_DECL)
|
||
{
|
||
tree spec;
|
||
|
||
/* Allocate template arguments on the momentary obstack,
|
||
in case we don't need to keep them. */
|
||
push_momentary ();
|
||
|
||
/* Calculate the most general template of which R is a
|
||
specialization, and the complete set of arguments used to
|
||
specialize R. */
|
||
gen_tmpl = most_general_template (DECL_TI_TEMPLATE (t));
|
||
argvec
|
||
= tsubst_template_arg_vector (DECL_TI_ARGS
|
||
(DECL_TEMPLATE_RESULT (gen_tmpl)),
|
||
args, /*complain=*/1);
|
||
|
||
/* Check to see if we already have this specialization. */
|
||
spec = retrieve_specialization (gen_tmpl, argvec);
|
||
|
||
if (spec)
|
||
{
|
||
r = spec;
|
||
pop_momentary ();
|
||
break;
|
||
}
|
||
|
||
/* We're going to need to keep the ARGVEC, so we copy it
|
||
here. */
|
||
argvec = copy_to_permanent (argvec);
|
||
pop_momentary ();
|
||
|
||
/* Here, we deal with the peculiar case:
|
||
|
||
template <class T> struct S {
|
||
template <class U> friend void f();
|
||
};
|
||
template <class U> friend void f() {}
|
||
template S<int>;
|
||
template void f<double>();
|
||
|
||
Here, the ARGS for the instantiation of will be {int,
|
||
double}. But, we only need as many ARGS as there are
|
||
levels of template parameters in CODE_PATTERN. We are
|
||
careful not to get fooled into reducing the ARGS in
|
||
situations like:
|
||
|
||
template <class T> struct S { template <class U> void f(U); }
|
||
template <class T> template <> void S<T>::f(int) {}
|
||
|
||
which we can spot because the pattern will be a
|
||
specialization in this case. */
|
||
args_depth = TMPL_ARGS_DEPTH (args);
|
||
parms_depth =
|
||
TMPL_PARMS_DEPTH (DECL_TEMPLATE_PARMS (DECL_TI_TEMPLATE (t)));
|
||
if (args_depth > parms_depth
|
||
&& !DECL_TEMPLATE_SPECIALIZATION (t))
|
||
{
|
||
my_friendly_assert (DECL_FRIEND_P (t), 0);
|
||
|
||
if (parms_depth > 1)
|
||
{
|
||
int i;
|
||
|
||
args = make_temp_vec (parms_depth);
|
||
for (i = 0; i < parms_depth; ++i)
|
||
TREE_VEC_ELT (args, i) =
|
||
TREE_VEC_ELT (args, i + (args_depth - parms_depth));
|
||
}
|
||
else
|
||
args = TREE_VEC_ELT (args, args_depth - parms_depth);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
/* This special case arises when we have something like this:
|
||
|
||
template <class T> struct S {
|
||
friend void f<int>(int, double);
|
||
};
|
||
|
||
Here, the DECL_TI_TEMPLATE for the friend declaration
|
||
will be a LOOKUP_EXPR or an IDENTIFIER_NODE. We are
|
||
being called from tsubst_friend_function, and we want
|
||
only to create a new decl (R) with appropriate types so
|
||
that we can call determine_specialization. */
|
||
my_friendly_assert ((TREE_CODE (DECL_TI_TEMPLATE (t))
|
||
== LOOKUP_EXPR)
|
||
|| (TREE_CODE (DECL_TI_TEMPLATE (t))
|
||
== IDENTIFIER_NODE), 0);
|
||
gen_tmpl = NULL_TREE;
|
||
}
|
||
|
||
if (DECL_CLASS_SCOPE_P (t))
|
||
{
|
||
if (DECL_NAME (t) == constructor_name (DECL_CONTEXT (t)))
|
||
member = 2;
|
||
else
|
||
member = 1;
|
||
ctx = tsubst_aggr_type (DECL_CLASS_CONTEXT (t), args,
|
||
/*complain=*/1, t,
|
||
/*entering_scope=*/1);
|
||
}
|
||
else
|
||
{
|
||
member = 0;
|
||
ctx = NULL_TREE;
|
||
}
|
||
type = tsubst (type, args, /*complain=*/1, in_decl);
|
||
|
||
/* We do NOT check for matching decls pushed separately at this
|
||
point, as they may not represent instantiations of this
|
||
template, and in any case are considered separate under the
|
||
discrete model. Instead, see add_maybe_template. */
|
||
|
||
r = copy_node (t);
|
||
copy_lang_decl (r);
|
||
DECL_USE_TEMPLATE (r) = 0;
|
||
TREE_TYPE (r) = type;
|
||
|
||
DECL_CONTEXT (r)
|
||
= tsubst_aggr_type (DECL_CONTEXT (t), args, /*complain=*/1, t,
|
||
/*entering_scope=*/1);
|
||
DECL_CLASS_CONTEXT (r) = ctx;
|
||
|
||
if (member && IDENTIFIER_TYPENAME_P (DECL_NAME (r)))
|
||
/* Type-conversion operator. Reconstruct the name, in
|
||
case it's the name of one of the template's parameters. */
|
||
DECL_NAME (r) = build_typename_overload (TREE_TYPE (type));
|
||
|
||
DECL_ARGUMENTS (r) = tsubst (DECL_ARGUMENTS (t), args,
|
||
/*complain=*/1, t);
|
||
DECL_MAIN_VARIANT (r) = r;
|
||
DECL_RESULT (r) = NULL_TREE;
|
||
|
||
TREE_STATIC (r) = 0;
|
||
TREE_PUBLIC (r) = TREE_PUBLIC (t);
|
||
DECL_EXTERNAL (r) = 1;
|
||
DECL_INTERFACE_KNOWN (r) = 0;
|
||
DECL_DEFER_OUTPUT (r) = 0;
|
||
TREE_CHAIN (r) = NULL_TREE;
|
||
DECL_PENDING_INLINE_INFO (r) = 0;
|
||
TREE_USED (r) = 0;
|
||
|
||
/* Set up the DECL_TEMPLATE_INFO for R and compute its mangled
|
||
name. There's no need to do this in the special friend
|
||
case mentioned above where GEN_TMPL is NULL. */
|
||
if (gen_tmpl)
|
||
{
|
||
/* The ARGVEC was built on the momentary obstack. Make it
|
||
permanent now. */
|
||
argvec = copy_to_permanent (argvec);
|
||
DECL_TEMPLATE_INFO (r)
|
||
= perm_tree_cons (gen_tmpl, argvec, NULL_TREE);
|
||
SET_DECL_IMPLICIT_INSTANTIATION (r);
|
||
register_specialization (r, gen_tmpl, argvec);
|
||
|
||
|
||
if (DECL_CONSTRUCTOR_P (r) || DECL_DESTRUCTOR_P (r))
|
||
{
|
||
maybe_retrofit_in_chrg (r);
|
||
grok_ctor_properties (ctx, r);
|
||
}
|
||
|
||
/* Set the mangled name for R. */
|
||
if (DECL_DESTRUCTOR_P (t))
|
||
DECL_ASSEMBLER_NAME (r) =
|
||
build_destructor_name (ctx, DECL_DESTRUCTOR_FOR_PVBASE_P (r));
|
||
else
|
||
{
|
||
/* Instantiations of template functions must be mangled
|
||
specially, in order to conform to 14.5.5.1
|
||
[temp.over.link]. */
|
||
tree tmpl = DECL_TI_TEMPLATE (t);
|
||
|
||
/* TMPL will be NULL if this is a specialization of a
|
||
member function of a template class. */
|
||
if (name_mangling_version < 1
|
||
|| tmpl == NULL_TREE
|
||
|| (member && !is_member_template (tmpl)
|
||
&& !DECL_TEMPLATE_INFO (tmpl)))
|
||
set_mangled_name_for_decl (r);
|
||
else
|
||
set_mangled_name_for_template_decl (r);
|
||
}
|
||
|
||
DECL_RTL (r) = 0;
|
||
make_decl_rtl (r, NULL_PTR, 1);
|
||
|
||
/* Like grokfndecl. If we don't do this, pushdecl will
|
||
mess up our TREE_CHAIN because it doesn't find a
|
||
previous decl. Sigh. */
|
||
if (member
|
||
&& ! uses_template_parms (r)
|
||
&& (IDENTIFIER_GLOBAL_VALUE (DECL_ASSEMBLER_NAME (r))
|
||
== NULL_TREE))
|
||
SET_IDENTIFIER_GLOBAL_VALUE (DECL_ASSEMBLER_NAME (r), r);
|
||
}
|
||
|
||
/* Copy the list of befriending classes. */
|
||
for (friends = &DECL_BEFRIENDING_CLASSES (r);
|
||
*friends;
|
||
friends = &TREE_CHAIN (*friends))
|
||
{
|
||
*friends = copy_node (*friends);
|
||
TREE_VALUE (*friends) = tsubst (TREE_VALUE (*friends),
|
||
args, /*complain=*/1,
|
||
in_decl);
|
||
}
|
||
|
||
#if 0
|
||
/* This has now moved further up. */
|
||
if (DECL_CONSTRUCTOR_P (r))
|
||
{
|
||
maybe_retrofit_in_chrg (r);
|
||
grok_ctor_properties (ctx, r);
|
||
}
|
||
#endif
|
||
if (IDENTIFIER_OPNAME_P (DECL_NAME (r)))
|
||
grok_op_properties (r, DECL_VIRTUAL_P (r), DECL_FRIEND_P (r));
|
||
}
|
||
break;
|
||
|
||
case PARM_DECL:
|
||
{
|
||
r = copy_node (t);
|
||
TREE_TYPE (r) = type;
|
||
c_apply_type_quals_to_decl (CP_TYPE_QUALS (type), r);
|
||
|
||
if (TREE_CODE (DECL_INITIAL (r)) != TEMPLATE_PARM_INDEX)
|
||
DECL_INITIAL (r) = TREE_TYPE (r);
|
||
else
|
||
DECL_INITIAL (r) = tsubst (DECL_INITIAL (r), args,
|
||
/*complain=*/1, in_decl);
|
||
|
||
DECL_CONTEXT (r) = NULL_TREE;
|
||
#ifdef PROMOTE_PROTOTYPES
|
||
if ((TREE_CODE (type) == INTEGER_TYPE
|
||
|| TREE_CODE (type) == ENUMERAL_TYPE)
|
||
&& TYPE_PRECISION (type) < TYPE_PRECISION (integer_type_node))
|
||
DECL_ARG_TYPE (r) = integer_type_node;
|
||
#endif
|
||
if (TREE_CHAIN (t))
|
||
TREE_CHAIN (r) = tsubst (TREE_CHAIN (t), args,
|
||
/*complain=*/1, TREE_CHAIN (t));
|
||
}
|
||
break;
|
||
|
||
case FIELD_DECL:
|
||
{
|
||
r = copy_node (t);
|
||
copy_lang_decl (r);
|
||
TREE_TYPE (r) = type;
|
||
c_apply_type_quals_to_decl (CP_TYPE_QUALS (type), r);
|
||
|
||
/* We don't have to set DECL_CONTEXT here; it is set by
|
||
finish_member_declaration. */
|
||
DECL_INITIAL (r) = tsubst_expr (DECL_INITIAL (t), args,
|
||
/*complain=*/1, in_decl);
|
||
TREE_CHAIN (r) = NULL_TREE;
|
||
if (TREE_CODE (type) == VOID_TYPE)
|
||
cp_error_at ("instantiation of `%D' as type void", r);
|
||
}
|
||
break;
|
||
|
||
case USING_DECL:
|
||
{
|
||
r = copy_node (t);
|
||
DECL_INITIAL (r)
|
||
= tsubst_copy (DECL_INITIAL (t), args, /*complain=*/1, in_decl);
|
||
TREE_CHAIN (r) = NULL_TREE;
|
||
}
|
||
break;
|
||
|
||
case VAR_DECL:
|
||
{
|
||
tree argvec;
|
||
tree gen_tmpl;
|
||
tree spec;
|
||
tree tmpl;
|
||
tree ctx = tsubst_aggr_type (DECL_CONTEXT (t), args,
|
||
/*complain=*/1,
|
||
in_decl, /*entering_scope=*/1);
|
||
|
||
/* Nobody should be tsubst'ing into non-template variables. */
|
||
my_friendly_assert (DECL_LANG_SPECIFIC (t)
|
||
&& DECL_TEMPLATE_INFO (t) != NULL_TREE, 0);
|
||
|
||
/* Check to see if we already have this specialization. */
|
||
tmpl = DECL_TI_TEMPLATE (t);
|
||
gen_tmpl = most_general_template (tmpl);
|
||
argvec = tsubst (DECL_TI_ARGS (t), args, /*complain=*/1, in_decl);
|
||
spec = retrieve_specialization (gen_tmpl, argvec);
|
||
|
||
if (spec)
|
||
{
|
||
r = spec;
|
||
break;
|
||
}
|
||
|
||
r = copy_node (t);
|
||
TREE_TYPE (r) = type;
|
||
c_apply_type_quals_to_decl (CP_TYPE_QUALS (type), r);
|
||
DECL_CONTEXT (r) = ctx;
|
||
|
||
/* Don't try to expand the initializer until someone tries to use
|
||
this variable; otherwise we run into circular dependencies. */
|
||
DECL_INITIAL (r) = NULL_TREE;
|
||
DECL_RTL (r) = 0;
|
||
DECL_SIZE (r) = 0;
|
||
copy_lang_decl (r);
|
||
DECL_CLASS_CONTEXT (r) = DECL_CONTEXT (r);
|
||
|
||
/* A static data member declaration is always marked external
|
||
when it is declared in-class, even if an initializer is
|
||
present. We mimic the non-template processing here. */
|
||
DECL_EXTERNAL (r) = 1;
|
||
|
||
DECL_TEMPLATE_INFO (r) = perm_tree_cons (tmpl, argvec, NULL_TREE);
|
||
SET_DECL_IMPLICIT_INSTANTIATION (r);
|
||
register_specialization (r, gen_tmpl, argvec);
|
||
|
||
TREE_CHAIN (r) = NULL_TREE;
|
||
if (TREE_CODE (type) == VOID_TYPE)
|
||
cp_error_at ("instantiation of `%D' as type void", r);
|
||
}
|
||
break;
|
||
|
||
case TYPE_DECL:
|
||
if (t == TYPE_NAME (TREE_TYPE (t)))
|
||
r = TYPE_NAME (type);
|
||
else
|
||
{
|
||
r = copy_node (t);
|
||
TREE_TYPE (r) = type;
|
||
DECL_CONTEXT (r) = current_class_type;
|
||
TREE_CHAIN (r) = NULL_TREE;
|
||
}
|
||
break;
|
||
|
||
default:
|
||
my_friendly_abort (0);
|
||
}
|
||
|
||
/* Restore the file and line information. */
|
||
lineno = saved_lineno;
|
||
input_filename = saved_filename;
|
||
|
||
return r;
|
||
}
|
||
|
||
/* Substitue into the ARG_TYPES of a function type. */
|
||
|
||
static tree
|
||
tsubst_arg_types (arg_types, args, complain, in_decl)
|
||
tree arg_types;
|
||
tree args;
|
||
int complain;
|
||
tree in_decl;
|
||
{
|
||
tree remaining_arg_types;
|
||
tree type;
|
||
|
||
if (!arg_types || arg_types == void_list_node)
|
||
return arg_types;
|
||
|
||
remaining_arg_types = tsubst_arg_types (TREE_CHAIN (arg_types),
|
||
args, complain, in_decl);
|
||
if (remaining_arg_types == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
type = tsubst (TREE_VALUE (arg_types), args, complain, in_decl);
|
||
if (type == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
/* Do array-to-pointer, function-to-pointer conversion, and ignore
|
||
top-level qualifiers as required. */
|
||
type = TYPE_MAIN_VARIANT (type_decays_to (type));
|
||
|
||
/* Note that we do not substitute into default arguments here. The
|
||
standard mandates that they be instantiated only when needed,
|
||
which is done in build_over_call. */
|
||
return hash_tree_cons (TREE_PURPOSE (arg_types), type,
|
||
remaining_arg_types);
|
||
|
||
}
|
||
|
||
/* Substitute into a FUNCTION_TYPE or METHOD_TYPE. This routine does
|
||
*not* handle the exception-specification for FNTYPE, because the
|
||
initial substitution of explicitly provided template parameters
|
||
during argument deduction forbids substitution into the
|
||
exception-specification:
|
||
|
||
[temp.deduct]
|
||
|
||
All references in the function type of the function template to the
|
||
corresponding template parameters are replaced by the specified tem-
|
||
plate argument values. If a substitution in a template parameter or
|
||
in the function type of the function template results in an invalid
|
||
type, type deduction fails. [Note: The equivalent substitution in
|
||
exception specifications is done only when the function is instanti-
|
||
ated, at which point a program is ill-formed if the substitution
|
||
results in an invalid type.] */
|
||
|
||
static tree
|
||
tsubst_function_type (t, args, complain, in_decl)
|
||
tree t;
|
||
tree args;
|
||
int complain;
|
||
tree in_decl;
|
||
{
|
||
tree return_type;
|
||
tree arg_types;
|
||
tree fntype;
|
||
|
||
/* The TYPE_CONTEXT is not used for function/method types. */
|
||
my_friendly_assert (TYPE_CONTEXT (t) == NULL_TREE, 0);
|
||
|
||
/* Substitue the return type. */
|
||
return_type = tsubst (TREE_TYPE (t), args, complain, in_decl);
|
||
if (return_type == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
/* Substitue the argument types. */
|
||
arg_types = tsubst_arg_types (TYPE_ARG_TYPES (t), args,
|
||
complain, in_decl);
|
||
if (arg_types == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
/* Construct a new type node and return it. */
|
||
if (TREE_CODE (t) == FUNCTION_TYPE)
|
||
fntype = build_function_type (return_type, arg_types);
|
||
else
|
||
{
|
||
tree r = TREE_TYPE (TREE_VALUE (arg_types));
|
||
if (! IS_AGGR_TYPE (r))
|
||
{
|
||
/* [temp.deduct]
|
||
|
||
Type deduction may fail for any of the following
|
||
reasons:
|
||
|
||
-- Attempting to create "pointer to member of T" when T
|
||
is not a class type. */
|
||
if (complain)
|
||
cp_error ("creating pointer to member function of non-class type `%T'",
|
||
r);
|
||
return error_mark_node;
|
||
}
|
||
|
||
fntype = build_cplus_method_type (r, return_type, TREE_CHAIN
|
||
(arg_types));
|
||
}
|
||
fntype = build_qualified_type (fntype, TYPE_QUALS (t));
|
||
fntype = build_type_attribute_variant (fntype, TYPE_ATTRIBUTES (t));
|
||
|
||
return fntype;
|
||
}
|
||
|
||
/* Substitute into the PARMS of a call-declarator. */
|
||
|
||
static tree
|
||
tsubst_call_declarator_parms (parms, args, complain, in_decl)
|
||
tree parms;
|
||
tree args;
|
||
int complain;
|
||
tree in_decl;
|
||
{
|
||
tree new_parms;
|
||
tree type;
|
||
tree defarg;
|
||
|
||
if (!parms || parms == void_list_node)
|
||
return parms;
|
||
|
||
new_parms = tsubst_call_declarator_parms (TREE_CHAIN (parms),
|
||
args, complain, in_decl);
|
||
|
||
/* Figure out the type of this parameter. */
|
||
type = tsubst (TREE_VALUE (parms), args, complain, in_decl);
|
||
|
||
/* Figure out the default argument as well. Note that we use
|
||
tsubst_expr since the default argument is really an expression. */
|
||
defarg = tsubst_expr (TREE_PURPOSE (parms), args, complain, in_decl);
|
||
|
||
/* Chain this parameter on to the front of those we have already
|
||
processed. We don't use hash_tree_cons because that function
|
||
doesn't check TREE_PARMLIST. */
|
||
new_parms = tree_cons (defarg, type, new_parms);
|
||
|
||
/* And note that these are parameters. */
|
||
TREE_PARMLIST (new_parms) = 1;
|
||
|
||
return new_parms;
|
||
}
|
||
|
||
/* Take the tree structure T and replace template parameters used
|
||
therein with the argument vector ARGS. IN_DECL is an associated
|
||
decl for diagnostics. If an error occurs, returns ERROR_MARK_NODE.
|
||
An appropriate error message is issued only if COMPLAIN is
|
||
non-zero. Note that we must be relatively non-tolerant of
|
||
extensions here, in order to preserve conformance; if we allow
|
||
substitutions that should not be allowed, we may allow argument
|
||
deductions that should not succeed, and therefore report ambiguous
|
||
overload situations where there are none. In theory, we could
|
||
allow the substitution, but indicate that it should have failed,
|
||
and allow our caller to make sure that the right thing happens, but
|
||
we don't try to do this yet.
|
||
|
||
This function is used for dealing with types, decls and the like;
|
||
for expressions, use tsubst_expr or tsubst_copy. */
|
||
|
||
tree
|
||
tsubst (t, args, complain, in_decl)
|
||
tree t, args;
|
||
int complain;
|
||
tree in_decl;
|
||
{
|
||
tree type, r;
|
||
|
||
if (t == NULL_TREE || t == error_mark_node
|
||
|| t == integer_type_node
|
||
|| t == void_type_node
|
||
|| t == char_type_node
|
||
|| TREE_CODE (t) == NAMESPACE_DECL)
|
||
return t;
|
||
|
||
if (TREE_CODE (t) == IDENTIFIER_NODE)
|
||
type = IDENTIFIER_TYPE_VALUE (t);
|
||
else
|
||
type = TREE_TYPE (t);
|
||
if (type == unknown_type_node)
|
||
my_friendly_abort (42);
|
||
|
||
if (type && TREE_CODE (t) != FUNCTION_DECL
|
||
&& TREE_CODE (t) != TYPENAME_TYPE
|
||
&& TREE_CODE (t) != TEMPLATE_DECL
|
||
&& TREE_CODE (t) != IDENTIFIER_NODE
|
||
&& TREE_CODE (t) != FUNCTION_TYPE
|
||
&& TREE_CODE (t) != METHOD_TYPE)
|
||
type = tsubst (type, args, complain, in_decl);
|
||
if (type == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
if (TREE_CODE_CLASS (TREE_CODE (t)) == 'd')
|
||
return tsubst_decl (t, args, type, in_decl);
|
||
|
||
switch (TREE_CODE (t))
|
||
{
|
||
case RECORD_TYPE:
|
||
case UNION_TYPE:
|
||
case ENUMERAL_TYPE:
|
||
return tsubst_aggr_type (t, args, complain, in_decl,
|
||
/*entering_scope=*/0);
|
||
|
||
case ERROR_MARK:
|
||
case IDENTIFIER_NODE:
|
||
case OP_IDENTIFIER:
|
||
case VOID_TYPE:
|
||
case REAL_TYPE:
|
||
case COMPLEX_TYPE:
|
||
case BOOLEAN_TYPE:
|
||
case INTEGER_CST:
|
||
case REAL_CST:
|
||
case STRING_CST:
|
||
return t;
|
||
|
||
case INTEGER_TYPE:
|
||
if (t == integer_type_node)
|
||
return t;
|
||
|
||
if (TREE_CODE (TYPE_MIN_VALUE (t)) == INTEGER_CST
|
||
&& TREE_CODE (TYPE_MAX_VALUE (t)) == INTEGER_CST)
|
||
return t;
|
||
|
||
{
|
||
tree max, omax = TREE_OPERAND (TYPE_MAX_VALUE (t), 0);
|
||
|
||
max = tsubst_expr (omax, args, complain, in_decl);
|
||
if (max == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
/* See if we can reduce this expression to something simpler. */
|
||
max = maybe_fold_nontype_arg (max);
|
||
if (!processing_template_decl && TREE_READONLY_DECL_P (max))
|
||
max = decl_constant_value (max);
|
||
|
||
if (processing_template_decl
|
||
/* When providing explicit arguments to a template
|
||
function, but leaving some arguments for subsequent
|
||
deduction, MAX may be template-dependent even if we're
|
||
not PROCESSING_TEMPLATE_DECL. */
|
||
|| TREE_CODE (max) != INTEGER_CST)
|
||
{
|
||
tree itype = make_node (INTEGER_TYPE);
|
||
TYPE_MIN_VALUE (itype) = size_zero_node;
|
||
TYPE_MAX_VALUE (itype) = build_min (MINUS_EXPR, sizetype, max,
|
||
integer_one_node);
|
||
return itype;
|
||
}
|
||
|
||
if (integer_zerop (omax))
|
||
{
|
||
/* Still allow an explicit array of size zero. */
|
||
if (pedantic)
|
||
pedwarn ("creating array with size zero");
|
||
}
|
||
else if (integer_zerop (max) || INT_CST_LT (max, integer_zero_node))
|
||
{
|
||
/* [temp.deduct]
|
||
|
||
Type deduction may fail for any of the following
|
||
reasons:
|
||
|
||
Attempting to create an array with a size that is
|
||
zero or negative. */
|
||
if (complain)
|
||
cp_error ("creating array with size `%E'", max);
|
||
|
||
return error_mark_node;
|
||
}
|
||
|
||
max = fold (build_binary_op (MINUS_EXPR, max, integer_one_node));
|
||
if (!TREE_PERMANENT (max) && !allocation_temporary_p ())
|
||
max = copy_to_permanent (max);
|
||
return build_index_type (max);
|
||
}
|
||
|
||
case TEMPLATE_TYPE_PARM:
|
||
case TEMPLATE_TEMPLATE_PARM:
|
||
case TEMPLATE_PARM_INDEX:
|
||
{
|
||
int idx;
|
||
int level;
|
||
int levels;
|
||
|
||
r = NULL_TREE;
|
||
|
||
if (TREE_CODE (t) == TEMPLATE_TYPE_PARM
|
||
|| TREE_CODE (t) == TEMPLATE_TEMPLATE_PARM)
|
||
{
|
||
idx = TEMPLATE_TYPE_IDX (t);
|
||
level = TEMPLATE_TYPE_LEVEL (t);
|
||
}
|
||
else
|
||
{
|
||
idx = TEMPLATE_PARM_IDX (t);
|
||
level = TEMPLATE_PARM_LEVEL (t);
|
||
}
|
||
|
||
if (TREE_VEC_LENGTH (args) > 0)
|
||
{
|
||
tree arg = NULL_TREE;
|
||
|
||
levels = TMPL_ARGS_DEPTH (args);
|
||
if (level <= levels)
|
||
arg = TMPL_ARG (args, level, idx);
|
||
|
||
if (arg == error_mark_node)
|
||
return error_mark_node;
|
||
else if (arg != NULL_TREE)
|
||
{
|
||
if (TREE_CODE (t) == TEMPLATE_TYPE_PARM)
|
||
{
|
||
my_friendly_assert (TREE_CODE_CLASS (TREE_CODE (arg))
|
||
== 't', 0);
|
||
return cp_build_qualified_type
|
||
(arg, CP_TYPE_QUALS (arg) | CP_TYPE_QUALS (t));
|
||
}
|
||
else if (TREE_CODE (t) == TEMPLATE_TEMPLATE_PARM)
|
||
{
|
||
if (TEMPLATE_TEMPLATE_PARM_TEMPLATE_INFO (t))
|
||
{
|
||
/* We are processing a type constructed from
|
||
a template template parameter */
|
||
tree argvec = tsubst (TYPE_TI_ARGS (t),
|
||
args, complain, in_decl);
|
||
if (argvec == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
/* We can get a TEMPLATE_TEMPLATE_PARM here when
|
||
we are resolving nested-types in the signature of
|
||
a member function templates.
|
||
Otherwise ARG is a TEMPLATE_DECL and is the real
|
||
template to be instantiated. */
|
||
if (TREE_CODE (arg) == TEMPLATE_TEMPLATE_PARM)
|
||
arg = TYPE_NAME (arg);
|
||
|
||
r = lookup_template_class (DECL_NAME (arg),
|
||
argvec, in_decl,
|
||
DECL_CONTEXT (arg),
|
||
/*entering_scope=*/0);
|
||
return cp_build_qualified_type (r, TYPE_QUALS (t));
|
||
}
|
||
else
|
||
/* We are processing a template argument list. */
|
||
return arg;
|
||
}
|
||
else
|
||
return arg;
|
||
}
|
||
}
|
||
else
|
||
my_friendly_abort (981018);
|
||
|
||
if (level == 1)
|
||
/* This can happen during the attempted tsubst'ing in
|
||
unify. This means that we don't yet have any information
|
||
about the template parameter in question. */
|
||
return t;
|
||
|
||
/* If we get here, we must have been looking at a parm for a
|
||
more deeply nested template. Make a new version of this
|
||
template parameter, but with a lower level. */
|
||
switch (TREE_CODE (t))
|
||
{
|
||
case TEMPLATE_TYPE_PARM:
|
||
case TEMPLATE_TEMPLATE_PARM:
|
||
r = copy_node (t);
|
||
TEMPLATE_TYPE_PARM_INDEX (r)
|
||
= reduce_template_parm_level (TEMPLATE_TYPE_PARM_INDEX (t),
|
||
r, levels);
|
||
TYPE_STUB_DECL (r) = TYPE_NAME (r) = TEMPLATE_TYPE_DECL (r);
|
||
TYPE_MAIN_VARIANT (r) = r;
|
||
TYPE_POINTER_TO (r) = NULL_TREE;
|
||
TYPE_REFERENCE_TO (r) = NULL_TREE;
|
||
|
||
if (TREE_CODE (t) == TEMPLATE_TEMPLATE_PARM
|
||
&& TEMPLATE_TEMPLATE_PARM_TEMPLATE_INFO (t))
|
||
{
|
||
tree argvec = tsubst (TYPE_TI_ARGS (t), args,
|
||
complain, in_decl);
|
||
if (argvec == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
TEMPLATE_TEMPLATE_PARM_TEMPLATE_INFO (r)
|
||
= perm_tree_cons (TYPE_NAME (t), argvec, NULL_TREE);
|
||
}
|
||
break;
|
||
|
||
case TEMPLATE_PARM_INDEX:
|
||
r = reduce_template_parm_level (t, type, levels);
|
||
break;
|
||
|
||
default:
|
||
my_friendly_abort (0);
|
||
}
|
||
|
||
return r;
|
||
}
|
||
|
||
case TREE_LIST:
|
||
{
|
||
tree purpose, value, chain, result;
|
||
|
||
if (t == void_list_node)
|
||
return t;
|
||
|
||
purpose = TREE_PURPOSE (t);
|
||
if (purpose)
|
||
{
|
||
purpose = tsubst (purpose, args, complain, in_decl);
|
||
if (purpose == error_mark_node)
|
||
return error_mark_node;
|
||
}
|
||
value = TREE_VALUE (t);
|
||
if (value)
|
||
{
|
||
value = tsubst (value, args, complain, in_decl);
|
||
if (value == error_mark_node)
|
||
return error_mark_node;
|
||
}
|
||
chain = TREE_CHAIN (t);
|
||
if (chain && chain != void_type_node)
|
||
{
|
||
chain = tsubst (chain, args, complain, in_decl);
|
||
if (chain == error_mark_node)
|
||
return error_mark_node;
|
||
}
|
||
if (purpose == TREE_PURPOSE (t)
|
||
&& value == TREE_VALUE (t)
|
||
&& chain == TREE_CHAIN (t))
|
||
return t;
|
||
result = hash_tree_cons (purpose, value, chain);
|
||
TREE_PARMLIST (result) = TREE_PARMLIST (t);
|
||
return result;
|
||
}
|
||
case TREE_VEC:
|
||
if (type != NULL_TREE)
|
||
{
|
||
/* A binfo node. We always need to make a copy, of the node
|
||
itself and of its BINFO_BASETYPES. */
|
||
|
||
t = copy_node (t);
|
||
|
||
/* Make sure type isn't a typedef copy. */
|
||
type = BINFO_TYPE (TYPE_BINFO (type));
|
||
|
||
TREE_TYPE (t) = complete_type (type);
|
||
if (IS_AGGR_TYPE (type))
|
||
{
|
||
BINFO_VTABLE (t) = TYPE_BINFO_VTABLE (type);
|
||
BINFO_VIRTUALS (t) = TYPE_BINFO_VIRTUALS (type);
|
||
if (TYPE_BINFO_BASETYPES (type) != NULL_TREE)
|
||
BINFO_BASETYPES (t) = copy_node (TYPE_BINFO_BASETYPES (type));
|
||
}
|
||
return t;
|
||
}
|
||
|
||
/* Otherwise, a vector of template arguments. */
|
||
return tsubst_template_arg_vector (t, args, complain);
|
||
|
||
case POINTER_TYPE:
|
||
case REFERENCE_TYPE:
|
||
{
|
||
enum tree_code code;
|
||
|
||
if (type == TREE_TYPE (t))
|
||
return t;
|
||
|
||
code = TREE_CODE (t);
|
||
|
||
|
||
/* [temp.deduct]
|
||
|
||
Type deduction may fail for any of the following
|
||
reasons:
|
||
|
||
-- Attempting to create a pointer to reference type.
|
||
-- Attempting to create a reference to a reference type or
|
||
a reference to void. */
|
||
if (TREE_CODE (type) == REFERENCE_TYPE
|
||
|| (code == REFERENCE_TYPE && TREE_CODE (type) == VOID_TYPE))
|
||
{
|
||
static int last_line = 0;
|
||
static char* last_file = 0;
|
||
|
||
/* We keep track of the last time we issued this error
|
||
message to avoid spewing a ton of messages during a
|
||
single bad template instantiation. */
|
||
if (complain && (last_line != lineno ||
|
||
last_file != input_filename))
|
||
{
|
||
if (TREE_CODE (type) == VOID_TYPE)
|
||
cp_error ("forming reference to void");
|
||
else
|
||
cp_error ("forming %s to reference type `%T'",
|
||
(code == POINTER_TYPE) ? "pointer" : "reference",
|
||
type);
|
||
last_line = lineno;
|
||
last_file = input_filename;
|
||
}
|
||
|
||
return error_mark_node;
|
||
}
|
||
else if (code == POINTER_TYPE)
|
||
r = build_pointer_type (type);
|
||
else
|
||
r = build_reference_type (type);
|
||
r = cp_build_qualified_type (r, TYPE_QUALS (t));
|
||
|
||
/* Will this ever be needed for TYPE_..._TO values? */
|
||
layout_type (r);
|
||
return r;
|
||
}
|
||
case OFFSET_TYPE:
|
||
{
|
||
r = tsubst (TYPE_OFFSET_BASETYPE (t), args, complain, in_decl);
|
||
if (r == error_mark_node || !IS_AGGR_TYPE (r))
|
||
{
|
||
/* [temp.deduct]
|
||
|
||
Type deduction may fail for any of the following
|
||
reasons:
|
||
|
||
-- Attempting to create "pointer to member of T" when T
|
||
is not a class type. */
|
||
if (complain)
|
||
cp_error ("creating pointer to member of non-class type `%T'",
|
||
r);
|
||
return error_mark_node;
|
||
}
|
||
return build_offset_type (r, type);
|
||
}
|
||
case FUNCTION_TYPE:
|
||
case METHOD_TYPE:
|
||
{
|
||
tree fntype;
|
||
tree raises;
|
||
|
||
fntype = tsubst_function_type (t, args, complain, in_decl);
|
||
if (fntype == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
/* Substitue the exception specification. */
|
||
raises = TYPE_RAISES_EXCEPTIONS (t);
|
||
if (raises)
|
||
{
|
||
raises = tsubst (raises, args, complain, in_decl);
|
||
if (raises == error_mark_node)
|
||
return raises;
|
||
fntype = build_exception_variant (fntype, raises);
|
||
}
|
||
return fntype;
|
||
}
|
||
case ARRAY_TYPE:
|
||
{
|
||
tree domain = tsubst (TYPE_DOMAIN (t), args, complain, in_decl);
|
||
if (domain == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
/* As an optimization, we avoid regenerating the array type if
|
||
it will obviously be the same as T. */
|
||
if (type == TREE_TYPE (t) && domain == TYPE_DOMAIN (t))
|
||
return t;
|
||
|
||
/* These checks should match the ones in grokdeclarator.
|
||
|
||
[temp.deduct]
|
||
|
||
The deduction may fail for any of the following reasons:
|
||
|
||
-- Attempting to create an array with an element type that
|
||
is void, a function type, or a reference type. */
|
||
if (TREE_CODE (type) == VOID_TYPE
|
||
|| TREE_CODE (type) == FUNCTION_TYPE
|
||
|| TREE_CODE (type) == REFERENCE_TYPE)
|
||
{
|
||
if (complain)
|
||
cp_error ("creating array of `%T'", type);
|
||
return error_mark_node;
|
||
}
|
||
|
||
r = build_cplus_array_type (type, domain);
|
||
return r;
|
||
}
|
||
|
||
case PLUS_EXPR:
|
||
case MINUS_EXPR:
|
||
{
|
||
tree e1 = tsubst (TREE_OPERAND (t, 0), args, complain,
|
||
in_decl);
|
||
tree e2 = tsubst (TREE_OPERAND (t, 1), args, complain,
|
||
in_decl);
|
||
|
||
if (e1 == error_mark_node || e2 == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
return fold (build (TREE_CODE (t), TREE_TYPE (t), e1, e2));
|
||
}
|
||
|
||
case NEGATE_EXPR:
|
||
case NOP_EXPR:
|
||
{
|
||
tree e = tsubst (TREE_OPERAND (t, 0), args, complain,
|
||
in_decl);
|
||
if (e == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
return fold (build (TREE_CODE (t), TREE_TYPE (t), e));
|
||
}
|
||
|
||
case TYPENAME_TYPE:
|
||
{
|
||
tree ctx = tsubst_aggr_type (TYPE_CONTEXT (t), args, complain,
|
||
in_decl, /*entering_scope=*/1);
|
||
tree f = tsubst_copy (TYPENAME_TYPE_FULLNAME (t), args,
|
||
complain, in_decl);
|
||
|
||
if (ctx == error_mark_node || f == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
if (!IS_AGGR_TYPE (ctx))
|
||
{
|
||
if (complain)
|
||
cp_error ("`%T' is not a class, struct, or union type",
|
||
ctx);
|
||
return error_mark_node;
|
||
}
|
||
else if (!uses_template_parms (ctx) && !TYPE_BEING_DEFINED (ctx))
|
||
{
|
||
/* Normally, make_typename_type does not require that the CTX
|
||
have complete type in order to allow things like:
|
||
|
||
template <class T> struct S { typename S<T>::X Y; };
|
||
|
||
But, such constructs have already been resolved by this
|
||
point, so here CTX really should have complete type, unless
|
||
it's a partial instantiation. */
|
||
ctx = complete_type (ctx);
|
||
if (!TYPE_SIZE (ctx))
|
||
{
|
||
if (complain)
|
||
incomplete_type_error (NULL_TREE, ctx);
|
||
return error_mark_node;
|
||
}
|
||
}
|
||
|
||
f = make_typename_type (ctx, f);
|
||
if (f == error_mark_node)
|
||
return f;
|
||
return cp_build_qualified_type (f,
|
||
CP_TYPE_QUALS (f)
|
||
| CP_TYPE_QUALS (t));
|
||
}
|
||
|
||
case INDIRECT_REF:
|
||
{
|
||
tree e = tsubst (TREE_OPERAND (t, 0), args, complain,
|
||
in_decl);
|
||
if (e == error_mark_node)
|
||
return error_mark_node;
|
||
return make_pointer_declarator (type, e);
|
||
}
|
||
|
||
case ADDR_EXPR:
|
||
{
|
||
tree e = tsubst (TREE_OPERAND (t, 0), args, complain,
|
||
in_decl);
|
||
if (e == error_mark_node)
|
||
return error_mark_node;
|
||
return make_reference_declarator (type, e);
|
||
}
|
||
|
||
case ARRAY_REF:
|
||
{
|
||
tree e1 = tsubst (TREE_OPERAND (t, 0), args, complain,
|
||
in_decl);
|
||
tree e2 = tsubst_expr (TREE_OPERAND (t, 1), args, complain,
|
||
in_decl);
|
||
if (e1 == error_mark_node || e2 == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
return build_parse_node (ARRAY_REF, e1, e2, tsubst_expr);
|
||
}
|
||
|
||
case CALL_EXPR:
|
||
{
|
||
tree e1 = tsubst (TREE_OPERAND (t, 0), args, complain,
|
||
in_decl);
|
||
tree e2 = tsubst_call_declarator_parms (TREE_OPERAND (t, 1), args,
|
||
complain, in_decl);
|
||
tree e3 = tsubst (TREE_TYPE (t), args, complain, in_decl);
|
||
|
||
if (e1 == error_mark_node || e2 == error_mark_node
|
||
|| e3 == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
return make_call_declarator (e1, e2, TREE_OPERAND (t, 2), e3);
|
||
}
|
||
|
||
case SCOPE_REF:
|
||
{
|
||
tree e1 = tsubst (TREE_OPERAND (t, 0), args, complain,
|
||
in_decl);
|
||
tree e2 = tsubst (TREE_OPERAND (t, 1), args, complain, in_decl);
|
||
if (e1 == error_mark_node || e2 == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
return build_parse_node (TREE_CODE (t), e1, e2);
|
||
}
|
||
|
||
case TYPEOF_TYPE:
|
||
{
|
||
tree e1 = tsubst_expr (TYPE_FIELDS (t), args, complain,
|
||
in_decl);
|
||
if (e1 == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
return TREE_TYPE (e1);
|
||
}
|
||
|
||
default:
|
||
sorry ("use of `%s' in template",
|
||
tree_code_name [(int) TREE_CODE (t)]);
|
||
return error_mark_node;
|
||
}
|
||
}
|
||
|
||
void
|
||
do_pushlevel ()
|
||
{
|
||
emit_line_note (input_filename, lineno);
|
||
pushlevel (0);
|
||
clear_last_expr ();
|
||
push_momentary ();
|
||
expand_start_bindings (0);
|
||
}
|
||
|
||
tree
|
||
do_poplevel ()
|
||
{
|
||
tree t;
|
||
int saved_warn_unused = 0;
|
||
|
||
if (processing_template_decl)
|
||
{
|
||
saved_warn_unused = warn_unused;
|
||
warn_unused = 0;
|
||
}
|
||
expand_end_bindings (getdecls (), kept_level_p (), 0);
|
||
if (processing_template_decl)
|
||
warn_unused = saved_warn_unused;
|
||
t = poplevel (kept_level_p (), 1, 0);
|
||
pop_momentary ();
|
||
return t;
|
||
}
|
||
|
||
/* Like tsubst, but deals with expressions. This function just replaces
|
||
template parms; to finish processing the resultant expression, use
|
||
tsubst_expr. */
|
||
|
||
tree
|
||
tsubst_copy (t, args, complain, in_decl)
|
||
tree t, args;
|
||
int complain;
|
||
tree in_decl;
|
||
{
|
||
enum tree_code code;
|
||
tree r;
|
||
|
||
if (t == NULL_TREE || t == error_mark_node)
|
||
return t;
|
||
|
||
code = TREE_CODE (t);
|
||
|
||
switch (code)
|
||
{
|
||
case PARM_DECL:
|
||
return do_identifier (DECL_NAME (t), 0, NULL_TREE);
|
||
|
||
case CONST_DECL:
|
||
{
|
||
tree enum_type;
|
||
tree v;
|
||
|
||
if (!DECL_CONTEXT (t))
|
||
/* This is a global enumeration constant. */
|
||
return t;
|
||
|
||
/* Unfortunately, we cannot just call lookup_name here.
|
||
Consider:
|
||
|
||
template <int I> int f() {
|
||
enum E { a = I };
|
||
struct S { void g() { E e = a; } };
|
||
};
|
||
|
||
When we instantiate f<7>::S::g(), say, lookup_name is not
|
||
clever enough to find f<7>::a. */
|
||
enum_type
|
||
= tsubst_aggr_type (TREE_TYPE (t), args, complain, in_decl,
|
||
/*entering_scope=*/0);
|
||
|
||
for (v = TYPE_VALUES (enum_type);
|
||
v != NULL_TREE;
|
||
v = TREE_CHAIN (v))
|
||
if (TREE_PURPOSE (v) == DECL_NAME (t))
|
||
return TREE_VALUE (v);
|
||
|
||
/* We didn't find the name. That should never happen; if
|
||
name-lookup found it during preliminary parsing, we
|
||
should find it again here during instantiation. */
|
||
my_friendly_abort (0);
|
||
}
|
||
return t;
|
||
|
||
case FIELD_DECL:
|
||
if (DECL_CONTEXT (t))
|
||
{
|
||
tree ctx;
|
||
|
||
ctx = tsubst_aggr_type (DECL_CONTEXT (t), args, complain, in_decl,
|
||
/*entering_scope=*/1);
|
||
if (ctx != DECL_CONTEXT (t))
|
||
return lookup_field (ctx, DECL_NAME (t), 0, 0);
|
||
}
|
||
return t;
|
||
|
||
case VAR_DECL:
|
||
case FUNCTION_DECL:
|
||
if (DECL_LANG_SPECIFIC (t) && DECL_TEMPLATE_INFO (t))
|
||
t = tsubst (t, args, complain, in_decl);
|
||
mark_used (t);
|
||
return t;
|
||
|
||
case TEMPLATE_DECL:
|
||
if (is_member_template (t))
|
||
return tsubst (t, args, complain, in_decl);
|
||
else
|
||
return t;
|
||
|
||
case LOOKUP_EXPR:
|
||
{
|
||
/* We must tsbust into a LOOKUP_EXPR in case the names to
|
||
which it refers is a conversion operator; in that case the
|
||
name will change. We avoid making unnecessary copies,
|
||
however. */
|
||
|
||
tree id = tsubst_copy (TREE_OPERAND (t, 0), args, complain, in_decl);
|
||
|
||
if (id != TREE_OPERAND (t, 0))
|
||
{
|
||
r = build_nt (LOOKUP_EXPR, id);
|
||
LOOKUP_EXPR_GLOBAL (r) = LOOKUP_EXPR_GLOBAL (t);
|
||
t = r;
|
||
}
|
||
|
||
return t;
|
||
}
|
||
|
||
case CAST_EXPR:
|
||
case REINTERPRET_CAST_EXPR:
|
||
case CONST_CAST_EXPR:
|
||
case STATIC_CAST_EXPR:
|
||
case DYNAMIC_CAST_EXPR:
|
||
case NOP_EXPR:
|
||
return build1
|
||
(code, tsubst (TREE_TYPE (t), args, complain, in_decl),
|
||
tsubst_copy (TREE_OPERAND (t, 0), args, complain, in_decl));
|
||
|
||
case INDIRECT_REF:
|
||
case PREDECREMENT_EXPR:
|
||
case PREINCREMENT_EXPR:
|
||
case POSTDECREMENT_EXPR:
|
||
case POSTINCREMENT_EXPR:
|
||
case NEGATE_EXPR:
|
||
case TRUTH_NOT_EXPR:
|
||
case BIT_NOT_EXPR:
|
||
case ADDR_EXPR:
|
||
case CONVERT_EXPR: /* Unary + */
|
||
case SIZEOF_EXPR:
|
||
case ALIGNOF_EXPR:
|
||
case ARROW_EXPR:
|
||
case THROW_EXPR:
|
||
case TYPEID_EXPR:
|
||
return build1
|
||
(code, tsubst (TREE_TYPE (t), args, complain, in_decl),
|
||
tsubst_copy (TREE_OPERAND (t, 0), args, complain, in_decl));
|
||
|
||
case PLUS_EXPR:
|
||
case MINUS_EXPR:
|
||
case MULT_EXPR:
|
||
case TRUNC_DIV_EXPR:
|
||
case CEIL_DIV_EXPR:
|
||
case FLOOR_DIV_EXPR:
|
||
case ROUND_DIV_EXPR:
|
||
case EXACT_DIV_EXPR:
|
||
case BIT_AND_EXPR:
|
||
case BIT_ANDTC_EXPR:
|
||
case BIT_IOR_EXPR:
|
||
case BIT_XOR_EXPR:
|
||
case TRUNC_MOD_EXPR:
|
||
case FLOOR_MOD_EXPR:
|
||
case TRUTH_ANDIF_EXPR:
|
||
case TRUTH_ORIF_EXPR:
|
||
case TRUTH_AND_EXPR:
|
||
case TRUTH_OR_EXPR:
|
||
case RSHIFT_EXPR:
|
||
case LSHIFT_EXPR:
|
||
case RROTATE_EXPR:
|
||
case LROTATE_EXPR:
|
||
case EQ_EXPR:
|
||
case NE_EXPR:
|
||
case MAX_EXPR:
|
||
case MIN_EXPR:
|
||
case LE_EXPR:
|
||
case GE_EXPR:
|
||
case LT_EXPR:
|
||
case GT_EXPR:
|
||
case COMPONENT_REF:
|
||
case ARRAY_REF:
|
||
case COMPOUND_EXPR:
|
||
case SCOPE_REF:
|
||
case DOTSTAR_EXPR:
|
||
case MEMBER_REF:
|
||
return build_nt
|
||
(code, tsubst_copy (TREE_OPERAND (t, 0), args, complain, in_decl),
|
||
tsubst_copy (TREE_OPERAND (t, 1), args, complain, in_decl));
|
||
|
||
case CALL_EXPR:
|
||
{
|
||
tree fn = TREE_OPERAND (t, 0);
|
||
if (is_overloaded_fn (fn))
|
||
fn = tsubst_copy (get_first_fn (fn), args, complain, in_decl);
|
||
else
|
||
/* Sometimes FN is a LOOKUP_EXPR. */
|
||
fn = tsubst_copy (fn, args, complain, in_decl);
|
||
return build_nt
|
||
(code, fn, tsubst_copy (TREE_OPERAND (t, 1), args, complain,
|
||
in_decl),
|
||
NULL_TREE);
|
||
}
|
||
|
||
case METHOD_CALL_EXPR:
|
||
{
|
||
tree name = TREE_OPERAND (t, 0);
|
||
if (TREE_CODE (name) == BIT_NOT_EXPR)
|
||
{
|
||
name = tsubst_copy (TREE_OPERAND (name, 0), args,
|
||
complain, in_decl);
|
||
name = build1 (BIT_NOT_EXPR, NULL_TREE, name);
|
||
}
|
||
else if (TREE_CODE (name) == SCOPE_REF
|
||
&& TREE_CODE (TREE_OPERAND (name, 1)) == BIT_NOT_EXPR)
|
||
{
|
||
tree base = tsubst_copy (TREE_OPERAND (name, 0), args,
|
||
complain, in_decl);
|
||
name = TREE_OPERAND (name, 1);
|
||
name = tsubst_copy (TREE_OPERAND (name, 0), args,
|
||
complain, in_decl);
|
||
name = build1 (BIT_NOT_EXPR, NULL_TREE, name);
|
||
name = build_nt (SCOPE_REF, base, name);
|
||
}
|
||
else
|
||
name = tsubst_copy (TREE_OPERAND (t, 0), args, complain, in_decl);
|
||
return build_nt
|
||
(code, name, tsubst_copy (TREE_OPERAND (t, 1), args,
|
||
complain, in_decl),
|
||
tsubst_copy (TREE_OPERAND (t, 2), args, complain, in_decl),
|
||
NULL_TREE);
|
||
}
|
||
|
||
case BIND_EXPR:
|
||
case COND_EXPR:
|
||
case MODOP_EXPR:
|
||
{
|
||
r = build_nt
|
||
(code, tsubst_copy (TREE_OPERAND (t, 0), args, complain, in_decl),
|
||
tsubst_copy (TREE_OPERAND (t, 1), args, complain, in_decl),
|
||
tsubst_copy (TREE_OPERAND (t, 2), args, complain, in_decl));
|
||
|
||
if (code == BIND_EXPR && !processing_template_decl)
|
||
{
|
||
/* This processing should really occur in tsubst_expr,
|
||
However, tsubst_expr does not recurse into expressions,
|
||
since it assumes that there aren't any statements
|
||
inside them. Instead, it simply calls
|
||
build_expr_from_tree. So, we need to expand the
|
||
BIND_EXPR here. */
|
||
tree rtl_expr = begin_stmt_expr ();
|
||
tree block = tsubst_expr (TREE_OPERAND (r, 1), args,
|
||
complain, in_decl);
|
||
r = finish_stmt_expr (rtl_expr, block);
|
||
}
|
||
|
||
return r;
|
||
}
|
||
|
||
case NEW_EXPR:
|
||
{
|
||
r = build_nt
|
||
(code, tsubst_copy (TREE_OPERAND (t, 0), args, complain, in_decl),
|
||
tsubst_copy (TREE_OPERAND (t, 1), args, complain, in_decl),
|
||
tsubst_copy (TREE_OPERAND (t, 2), args, complain, in_decl));
|
||
NEW_EXPR_USE_GLOBAL (r) = NEW_EXPR_USE_GLOBAL (t);
|
||
return r;
|
||
}
|
||
|
||
case DELETE_EXPR:
|
||
{
|
||
r = build_nt
|
||
(code, tsubst_copy (TREE_OPERAND (t, 0), args, complain, in_decl),
|
||
tsubst_copy (TREE_OPERAND (t, 1), args, complain, in_decl));
|
||
DELETE_EXPR_USE_GLOBAL (r) = DELETE_EXPR_USE_GLOBAL (t);
|
||
DELETE_EXPR_USE_VEC (r) = DELETE_EXPR_USE_VEC (t);
|
||
return r;
|
||
}
|
||
|
||
case TEMPLATE_ID_EXPR:
|
||
{
|
||
/* Substituted template arguments */
|
||
tree targs = tsubst_copy (TREE_OPERAND (t, 1), args, complain,
|
||
in_decl);
|
||
|
||
if (targs && TREE_CODE (targs) == TREE_LIST)
|
||
{
|
||
tree chain;
|
||
for (chain = targs; chain; chain = TREE_CHAIN (chain))
|
||
TREE_VALUE (chain) = maybe_fold_nontype_arg (TREE_VALUE (chain));
|
||
}
|
||
else if (targs)
|
||
{
|
||
int i;
|
||
for (i = 0; i < TREE_VEC_LENGTH (targs); ++i)
|
||
TREE_VEC_ELT (targs, i)
|
||
= maybe_fold_nontype_arg (TREE_VEC_ELT (targs, i));
|
||
}
|
||
|
||
return lookup_template_function
|
||
(tsubst_copy (TREE_OPERAND (t, 0), args, complain, in_decl), targs);
|
||
}
|
||
|
||
case TREE_LIST:
|
||
{
|
||
tree purpose, value, chain;
|
||
|
||
if (t == void_list_node)
|
||
return t;
|
||
|
||
purpose = TREE_PURPOSE (t);
|
||
if (purpose)
|
||
purpose = tsubst_copy (purpose, args, complain, in_decl);
|
||
value = TREE_VALUE (t);
|
||
if (value)
|
||
value = tsubst_copy (value, args, complain, in_decl);
|
||
chain = TREE_CHAIN (t);
|
||
if (chain && chain != void_type_node)
|
||
chain = tsubst_copy (chain, args, complain, in_decl);
|
||
if (purpose == TREE_PURPOSE (t)
|
||
&& value == TREE_VALUE (t)
|
||
&& chain == TREE_CHAIN (t))
|
||
return t;
|
||
return tree_cons (purpose, value, chain);
|
||
}
|
||
|
||
case RECORD_TYPE:
|
||
case UNION_TYPE:
|
||
case ENUMERAL_TYPE:
|
||
case INTEGER_TYPE:
|
||
case TEMPLATE_TYPE_PARM:
|
||
case TEMPLATE_TEMPLATE_PARM:
|
||
case TEMPLATE_PARM_INDEX:
|
||
case POINTER_TYPE:
|
||
case REFERENCE_TYPE:
|
||
case OFFSET_TYPE:
|
||
case FUNCTION_TYPE:
|
||
case METHOD_TYPE:
|
||
case ARRAY_TYPE:
|
||
case TYPENAME_TYPE:
|
||
case TYPE_DECL:
|
||
return tsubst (t, args, complain, in_decl);
|
||
|
||
case IDENTIFIER_NODE:
|
||
if (IDENTIFIER_TYPENAME_P (t)
|
||
/* Make sure it's not just a variable named `__opr', for instance,
|
||
which can occur in some existing code. */
|
||
&& TREE_TYPE (t))
|
||
return build_typename_overload
|
||
(tsubst (TREE_TYPE (t), args, complain, in_decl));
|
||
else
|
||
return t;
|
||
|
||
case CONSTRUCTOR:
|
||
{
|
||
r = build
|
||
(CONSTRUCTOR, tsubst (TREE_TYPE (t), args, complain, in_decl),
|
||
NULL_TREE, tsubst_copy (CONSTRUCTOR_ELTS (t), args,
|
||
complain, in_decl));
|
||
TREE_HAS_CONSTRUCTOR (r) = TREE_HAS_CONSTRUCTOR (t);
|
||
return r;
|
||
}
|
||
|
||
default:
|
||
return t;
|
||
}
|
||
}
|
||
|
||
/* Like tsubst_copy, but also does semantic processing and RTL expansion. */
|
||
|
||
tree
|
||
tsubst_expr (t, args, complain, in_decl)
|
||
tree t, args;
|
||
int complain;
|
||
tree in_decl;
|
||
{
|
||
if (t == NULL_TREE || t == error_mark_node)
|
||
return t;
|
||
|
||
if (processing_template_decl)
|
||
return tsubst_copy (t, args, complain, in_decl);
|
||
|
||
switch (TREE_CODE (t))
|
||
{
|
||
case RETURN_STMT:
|
||
lineno = TREE_COMPLEXITY (t);
|
||
finish_return_stmt (tsubst_expr (RETURN_EXPR (t),
|
||
args, complain, in_decl));
|
||
break;
|
||
|
||
case EXPR_STMT:
|
||
lineno = TREE_COMPLEXITY (t);
|
||
finish_expr_stmt (tsubst_expr (EXPR_STMT_EXPR (t),
|
||
args, complain, in_decl));
|
||
break;
|
||
|
||
case DECL_STMT:
|
||
{
|
||
int i = suspend_momentary ();
|
||
tree dcl, init;
|
||
|
||
lineno = TREE_COMPLEXITY (t);
|
||
emit_line_note (input_filename, lineno);
|
||
dcl = start_decl
|
||
(tsubst (TREE_OPERAND (t, 0), args, complain, in_decl),
|
||
tsubst (TREE_OPERAND (t, 1), args, complain, in_decl),
|
||
TREE_OPERAND (t, 2) != 0, NULL_TREE, NULL_TREE);
|
||
init = tsubst_expr (TREE_OPERAND (t, 2), args, complain, in_decl);
|
||
cp_finish_decl
|
||
(dcl, init, NULL_TREE, 1, /*init ? LOOKUP_ONLYCONVERTING :*/ 0);
|
||
resume_momentary (i);
|
||
return dcl;
|
||
}
|
||
|
||
case FOR_STMT:
|
||
{
|
||
tree tmp;
|
||
lineno = TREE_COMPLEXITY (t);
|
||
|
||
begin_for_stmt ();
|
||
for (tmp = FOR_INIT_STMT (t); tmp; tmp = TREE_CHAIN (tmp))
|
||
tsubst_expr (tmp, args, complain, in_decl);
|
||
finish_for_init_stmt (NULL_TREE);
|
||
finish_for_cond (tsubst_expr (FOR_COND (t), args,
|
||
complain, in_decl),
|
||
NULL_TREE);
|
||
tmp = tsubst_expr (FOR_EXPR (t), args, complain, in_decl);
|
||
finish_for_expr (tmp, NULL_TREE);
|
||
tsubst_expr (FOR_BODY (t), args, complain, in_decl);
|
||
finish_for_stmt (tmp, NULL_TREE);
|
||
}
|
||
break;
|
||
|
||
case WHILE_STMT:
|
||
{
|
||
lineno = TREE_COMPLEXITY (t);
|
||
begin_while_stmt ();
|
||
finish_while_stmt_cond (tsubst_expr (WHILE_COND (t),
|
||
args, complain, in_decl),
|
||
NULL_TREE);
|
||
tsubst_expr (WHILE_BODY (t), args, complain, in_decl);
|
||
finish_while_stmt (NULL_TREE);
|
||
}
|
||
break;
|
||
|
||
case DO_STMT:
|
||
{
|
||
lineno = TREE_COMPLEXITY (t);
|
||
begin_do_stmt ();
|
||
tsubst_expr (DO_BODY (t), args, complain, in_decl);
|
||
finish_do_body (NULL_TREE);
|
||
finish_do_stmt (tsubst_expr (DO_COND (t), args,
|
||
complain, in_decl),
|
||
NULL_TREE);
|
||
}
|
||
break;
|
||
|
||
case IF_STMT:
|
||
{
|
||
tree tmp;
|
||
|
||
lineno = TREE_COMPLEXITY (t);
|
||
begin_if_stmt ();
|
||
finish_if_stmt_cond (tsubst_expr (IF_COND (t),
|
||
args, complain, in_decl),
|
||
NULL_TREE);
|
||
|
||
if (tmp = THEN_CLAUSE (t), tmp)
|
||
{
|
||
tsubst_expr (tmp, args, complain, in_decl);
|
||
finish_then_clause (NULL_TREE);
|
||
}
|
||
|
||
if (tmp = ELSE_CLAUSE (t), tmp)
|
||
{
|
||
begin_else_clause ();
|
||
tsubst_expr (tmp, args, complain, in_decl);
|
||
finish_else_clause (NULL_TREE);
|
||
}
|
||
|
||
finish_if_stmt ();
|
||
}
|
||
break;
|
||
|
||
case COMPOUND_STMT:
|
||
{
|
||
tree substmt;
|
||
|
||
lineno = TREE_COMPLEXITY (t);
|
||
begin_compound_stmt (COMPOUND_STMT_NO_SCOPE (t));
|
||
for (substmt = COMPOUND_BODY (t);
|
||
substmt != NULL_TREE;
|
||
substmt = TREE_CHAIN (substmt))
|
||
tsubst_expr (substmt, args, complain, in_decl);
|
||
return finish_compound_stmt (COMPOUND_STMT_NO_SCOPE (t),
|
||
NULL_TREE);
|
||
}
|
||
break;
|
||
|
||
case BREAK_STMT:
|
||
lineno = TREE_COMPLEXITY (t);
|
||
finish_break_stmt ();
|
||
break;
|
||
|
||
case CONTINUE_STMT:
|
||
lineno = TREE_COMPLEXITY (t);
|
||
finish_continue_stmt ();
|
||
break;
|
||
|
||
case SWITCH_STMT:
|
||
{
|
||
tree val, tmp;
|
||
|
||
lineno = TREE_COMPLEXITY (t);
|
||
begin_switch_stmt ();
|
||
val = tsubst_expr (SWITCH_COND (t), args, complain, in_decl);
|
||
finish_switch_cond (val);
|
||
|
||
if (tmp = TREE_OPERAND (t, 1), tmp)
|
||
tsubst_expr (tmp, args, complain, in_decl);
|
||
|
||
finish_switch_stmt (val, NULL_TREE);
|
||
}
|
||
break;
|
||
|
||
case CASE_LABEL:
|
||
finish_case_label (tsubst_expr (CASE_LOW (t), args, complain, in_decl),
|
||
tsubst_expr (CASE_HIGH (t), args, complain, in_decl));
|
||
break;
|
||
|
||
case LABEL_DECL:
|
||
t = define_label (DECL_SOURCE_FILE (t), DECL_SOURCE_LINE (t),
|
||
DECL_NAME (t));
|
||
if (t)
|
||
expand_label (t);
|
||
break;
|
||
|
||
case GOTO_STMT:
|
||
lineno = TREE_COMPLEXITY (t);
|
||
t = GOTO_DESTINATION (t);
|
||
if (TREE_CODE (t) != IDENTIFIER_NODE)
|
||
/* Computed goto's must be tsubst'd into. On the other hand,
|
||
non-computed gotos must not be; the identifier in question
|
||
will have no binding. */
|
||
t = tsubst_expr (t, args, complain, in_decl);
|
||
finish_goto_stmt (t);
|
||
break;
|
||
|
||
case ASM_STMT:
|
||
lineno = TREE_COMPLEXITY (t);
|
||
finish_asm_stmt (tsubst_expr (ASM_CV_QUAL (t), args, complain, in_decl),
|
||
tsubst_expr (ASM_STRING (t), args, complain, in_decl),
|
||
tsubst_expr (ASM_OUTPUTS (t), args, complain, in_decl),
|
||
tsubst_expr (ASM_INPUTS (t), args, complain, in_decl),
|
||
tsubst_expr (ASM_CLOBBERS (t), args, complain,
|
||
in_decl));
|
||
break;
|
||
|
||
case TRY_BLOCK:
|
||
lineno = TREE_COMPLEXITY (t);
|
||
begin_try_block ();
|
||
tsubst_expr (TRY_STMTS (t), args, complain, in_decl);
|
||
finish_try_block (NULL_TREE);
|
||
{
|
||
tree handler = TRY_HANDLERS (t);
|
||
for (; handler; handler = TREE_CHAIN (handler))
|
||
tsubst_expr (handler, args, complain, in_decl);
|
||
}
|
||
finish_handler_sequence (NULL_TREE);
|
||
break;
|
||
|
||
case HANDLER:
|
||
lineno = TREE_COMPLEXITY (t);
|
||
begin_handler ();
|
||
if (HANDLER_PARMS (t))
|
||
{
|
||
tree d = HANDLER_PARMS (t);
|
||
expand_start_catch_block
|
||
(tsubst (TREE_OPERAND (d, 1), args, complain, in_decl),
|
||
tsubst (TREE_OPERAND (d, 0), args, complain, in_decl));
|
||
}
|
||
else
|
||
expand_start_catch_block (NULL_TREE, NULL_TREE);
|
||
finish_handler_parms (NULL_TREE);
|
||
tsubst_expr (HANDLER_BODY (t), args, complain, in_decl);
|
||
finish_handler (NULL_TREE);
|
||
break;
|
||
|
||
case TAG_DEFN:
|
||
lineno = TREE_COMPLEXITY (t);
|
||
t = TREE_TYPE (t);
|
||
if (TREE_CODE (t) == ENUMERAL_TYPE)
|
||
tsubst (t, args, complain, NULL_TREE);
|
||
break;
|
||
|
||
default:
|
||
return build_expr_from_tree (tsubst_copy (t, args, complain, in_decl));
|
||
}
|
||
return NULL_TREE;
|
||
}
|
||
|
||
/* Instantiate the indicated variable or function template TMPL with
|
||
the template arguments in TARG_PTR. */
|
||
|
||
tree
|
||
instantiate_template (tmpl, targ_ptr)
|
||
tree tmpl, targ_ptr;
|
||
{
|
||
tree fndecl;
|
||
tree gen_tmpl;
|
||
tree spec;
|
||
int i, len;
|
||
struct obstack *old_fmp_obstack;
|
||
extern struct obstack *function_maybepermanent_obstack;
|
||
tree inner_args;
|
||
|
||
if (tmpl == error_mark_node)
|
||
return error_mark_node;
|
||
|
||
my_friendly_assert (TREE_CODE (tmpl) == TEMPLATE_DECL, 283);
|
||
|
||
/* Check to see if we already have this specialization. */
|
||
spec = retrieve_specialization (tmpl, targ_ptr);
|
||
if (spec != NULL_TREE)
|
||
return spec;
|
||
|
||
if (DECL_TEMPLATE_INFO (tmpl))
|
||
{
|
||
/* The TMPL is a partial instantiation. To get a full set of
|
||
arguments we must add the arguments used to perform the
|
||
partial instantiation. */
|
||
targ_ptr = add_outermost_template_args (DECL_TI_ARGS (tmpl),
|
||
targ_ptr);
|
||
gen_tmpl = most_general_template (tmpl);
|
||
|
||
/* Check to see if we already have this specialization. */
|
||
spec = retrieve_specialization (gen_tmpl, targ_ptr);
|
||
if (spec != NULL_TREE)
|
||
return spec;
|
||
}
|
||
else
|
||
gen_tmpl = tmpl;
|
||
|
||
push_obstacks (&permanent_obstack, &permanent_obstack);
|
||
old_fmp_obstack = function_maybepermanent_obstack;
|
||
function_maybepermanent_obstack = &permanent_obstack;
|
||
|
||
len = DECL_NTPARMS (gen_tmpl);
|
||
inner_args = innermost_args (targ_ptr);
|
||
i = len;
|
||
while (i--)
|
||
{
|
||
tree t = TREE_VEC_ELT (inner_args, i);
|
||
if (TREE_CODE_CLASS (TREE_CODE (t)) == 't')
|
||
{
|
||
tree nt = target_type (t);
|
||
if (IS_AGGR_TYPE (nt) && decl_function_context (TYPE_MAIN_DECL (nt)))
|
||
{
|
||
cp_error ("type `%T' composed from a local class is not a valid template-argument", t);
|
||
cp_error (" trying to instantiate `%D'", gen_tmpl);
|
||
fndecl = error_mark_node;
|
||
goto out;
|
||
}
|
||
}
|
||
}
|
||
targ_ptr = copy_to_permanent (targ_ptr);
|
||
|
||
/* substitute template parameters */
|
||
fndecl = tsubst (DECL_RESULT (gen_tmpl), targ_ptr, /*complain=*/1, gen_tmpl);
|
||
/* The DECL_TI_TEMPLATE should always be the immediate parent
|
||
template, not the most general template. */
|
||
DECL_TI_TEMPLATE (fndecl) = tmpl;
|
||
|
||
if (flag_external_templates)
|
||
add_pending_template (fndecl);
|
||
|
||
out:
|
||
function_maybepermanent_obstack = old_fmp_obstack;
|
||
pop_obstacks ();
|
||
|
||
return fndecl;
|
||
}
|
||
|
||
/* Push the name of the class template into the scope of the instantiation. */
|
||
|
||
void
|
||
overload_template_name (type)
|
||
tree type;
|
||
{
|
||
tree id = DECL_NAME (CLASSTYPE_TI_TEMPLATE (type));
|
||
tree decl;
|
||
|
||
if (IDENTIFIER_CLASS_VALUE (id)
|
||
&& TREE_TYPE (IDENTIFIER_CLASS_VALUE (id)) == type)
|
||
return;
|
||
|
||
decl = build_decl (TYPE_DECL, id, type);
|
||
SET_DECL_ARTIFICIAL (decl);
|
||
pushdecl_class_level (decl);
|
||
}
|
||
|
||
/* The FN is a TEMPLATE_DECL for a function. The ARGS are the
|
||
arguments that are being used when calling it. TARGS is a vector
|
||
into which the deduced template arguments are placed.
|
||
|
||
Return zero for success, 2 for an incomplete match that doesn't resolve
|
||
all the types, and 1 for complete failure. An error message will be
|
||
printed only for an incomplete match.
|
||
|
||
If FN is a conversion operator, RETURN_TYPE is the type desired as
|
||
the result of the conversion operator.
|
||
|
||
TPARMS is a vector of template parameters.
|
||
|
||
The EXPLICIT_TARGS are explicit template arguments provided via a
|
||
template-id.
|
||
|
||
The parameter STRICT is one of:
|
||
|
||
DEDUCE_CALL:
|
||
We are deducing arguments for a function call, as in
|
||
[temp.deduct.call].
|
||
|
||
DEDUCE_CONV:
|
||
We are deducing arguments for a conversion function, as in
|
||
[temp.deduct.conv].
|
||
|
||
DEDUCE_EXACT:
|
||
We are deducing arguments when calculating the partial
|
||
ordering between specializations of function or class
|
||
templates, as in [temp.func.order] and [temp.class.order],
|
||
when doing an explicit instantiation as in [temp.explicit],
|
||
when determining an explicit specialization as in
|
||
[temp.expl.spec], or when taking the address of a function
|
||
template, as in [temp.deduct.funcaddr].
|
||
|
||
The other arguments are as for type_unification. */
|
||
|
||
int
|
||
fn_type_unification (fn, explicit_targs, targs, args, return_type,
|
||
strict)
|
||
tree fn, explicit_targs, targs, args, return_type;
|
||
unification_kind_t strict;
|
||
{
|
||
tree parms;
|
||
tree fntype;
|
||
|
||
my_friendly_assert (TREE_CODE (fn) == TEMPLATE_DECL, 0);
|
||
|
||
fntype = TREE_TYPE (fn);
|
||
if (explicit_targs)
|
||
{
|
||
/* [temp.deduct]
|
||
|
||
The specified template arguments must match the template
|
||
parameters in kind (i.e., type, nontype, template), and there
|
||
must not be more arguments than there are parameters;
|
||
otherwise type deduction fails.
|
||
|
||
Nontype arguments must match the types of the corresponding
|
||
nontype template parameters, or must be convertible to the
|
||
types of the corresponding nontype parameters as specified in
|
||
_temp.arg.nontype_, otherwise type deduction fails.
|
||
|
||
All references in the function type of the function template
|
||
to the corresponding template parameters are replaced by the
|
||
specified template argument values. If a substitution in a
|
||
template parameter or in the function type of the function
|
||
template results in an invalid type, type deduction fails. */
|
||
int i;
|
||
tree converted_args;
|
||
|
||
converted_args
|
||
= (coerce_template_parms (DECL_INNERMOST_TEMPLATE_PARMS (fn),
|
||
explicit_targs, NULL_TREE, /*complain=*/0,
|
||
/*require_all_arguments=*/0));
|
||
if (converted_args == error_mark_node)
|
||
return 1;
|
||
|
||
fntype = tsubst (fntype, converted_args, /*complain=*/0, NULL_TREE);
|
||
if (fntype == error_mark_node)
|
||
return 1;
|
||
|
||
/* Place the explicitly specified arguments in TARGS. */
|
||
for (i = 0; i < TREE_VEC_LENGTH (targs); i++)
|
||
TREE_VEC_ELT (targs, i) = TREE_VEC_ELT (converted_args, i);
|
||
}
|
||
|
||
parms = TYPE_ARG_TYPES (fntype);
|
||
|
||
if (DECL_CONV_FN_P (fn))
|
||
{
|
||
/* This is a template conversion operator. Use the return types
|
||
as well as the argument types. We use it instead of 'this', since
|
||
we could be comparing conversions from different classes. */
|
||
parms = scratch_tree_cons (NULL_TREE, TREE_TYPE (fntype),
|
||
TREE_CHAIN (parms));
|
||
args = scratch_tree_cons (NULL_TREE, return_type, TREE_CHAIN (args));
|
||
}
|
||
|
||
/* We allow incomplete unification without an error message here
|
||
because the standard doesn't seem to explicitly prohibit it. Our
|
||
callers must be ready to deal with unification failures in any
|
||
event. */
|
||
return type_unification_real (DECL_INNERMOST_TEMPLATE_PARMS (fn),
|
||
targs, parms, args, /*subr=*/0,
|
||
strict, /*allow_incomplete*/1);
|
||
}
|
||
|
||
/* Adjust types before performing type deduction, as described in
|
||
[temp.deduct.call] and [temp.deduct.conv]. The rules in these two
|
||
sections are symmetric. PARM is the type of a function parameter
|
||
or the return type of the conversion function. ARG is the type of
|
||
the argument passed to the call, or the type of the value
|
||
intialized with the result of the conversion function. */
|
||
|
||
static void
|
||
maybe_adjust_types_for_deduction (strict, parm, arg)
|
||
unification_kind_t strict;
|
||
tree* parm;
|
||
tree* arg;
|
||
{
|
||
switch (strict)
|
||
{
|
||
case DEDUCE_CALL:
|
||
break;
|
||
|
||
case DEDUCE_CONV:
|
||
{
|
||
/* Swap PARM and ARG throughout the remainder of this
|
||
function; the handling is precisely symmetric since PARM
|
||
will initialize ARG rather than vice versa. */
|
||
tree* temp = parm;
|
||
parm = arg;
|
||
arg = temp;
|
||
break;
|
||
}
|
||
|
||
case DEDUCE_EXACT:
|
||
/* There is nothing to do in this case. */
|
||
return;
|
||
|
||
default:
|
||
my_friendly_abort (0);
|
||
}
|
||
|
||
if (TREE_CODE (*parm) != REFERENCE_TYPE)
|
||
{
|
||
/* [temp.deduct.call]
|
||
|
||
If P is not a reference type:
|
||
|
||
--If A is an array type, the pointer type produced by the
|
||
array-to-pointer standard conversion (_conv.array_) is
|
||
used in place of A for type deduction; otherwise,
|
||
|
||
--If A is a function type, the pointer type produced by
|
||
the function-to-pointer standard conversion
|
||
(_conv.func_) is used in place of A for type deduction;
|
||
otherwise,
|
||
|
||
--If A is a cv-qualified type, the top level
|
||
cv-qualifiers of A's type are ignored for type
|
||
deduction. */
|
||
if (TREE_CODE (*arg) == ARRAY_TYPE)
|
||
*arg = build_pointer_type (TREE_TYPE (*arg));
|
||
else if (TREE_CODE (*arg) == FUNCTION_TYPE)
|
||
*arg = build_pointer_type (*arg);
|
||
else
|
||
*arg = TYPE_MAIN_VARIANT (*arg);
|
||
}
|
||
|
||
/* [temp.deduct.call]
|
||
|
||
If P is a cv-qualified type, the top level cv-qualifiers
|
||
of P's type are ignored for type deduction. If P is a
|
||
reference type, the type referred to by P is used for
|
||
type deduction. */
|
||
*parm = TYPE_MAIN_VARIANT (*parm);
|
||
if (TREE_CODE (*parm) == REFERENCE_TYPE)
|
||
*parm = TREE_TYPE (*parm);
|
||
}
|
||
|
||
/* Like type_unfication.
|
||
|
||
If SUBR is 1, we're being called recursively (to unify the
|
||
arguments of a function or method parameter of a function
|
||
template). */
|
||
|
||
static int
|
||
type_unification_real (tparms, targs, parms, args, subr,
|
||
strict, allow_incomplete)
|
||
tree tparms, targs, parms, args;
|
||
int subr;
|
||
unification_kind_t strict;
|
||
int allow_incomplete;
|
||
{
|
||
tree parm, arg;
|
||
int i;
|
||
int ntparms = TREE_VEC_LENGTH (tparms);
|
||
int sub_strict;
|
||
|
||
my_friendly_assert (TREE_CODE (tparms) == TREE_VEC, 289);
|
||
my_friendly_assert (parms == NULL_TREE
|
||
|| TREE_CODE (parms) == TREE_LIST, 290);
|
||
/* ARGS could be NULL (via a call from parse.y to
|
||
build_x_function_call). */
|
||
if (args)
|
||
my_friendly_assert (TREE_CODE (args) == TREE_LIST, 291);
|
||
my_friendly_assert (ntparms > 0, 292);
|
||
|
||
switch (strict)
|
||
{
|
||
case DEDUCE_CALL:
|
||
sub_strict = UNIFY_ALLOW_MORE_CV_QUAL | UNIFY_ALLOW_DERIVED;
|
||
break;
|
||
|
||
case DEDUCE_CONV:
|
||
sub_strict = UNIFY_ALLOW_LESS_CV_QUAL;
|
||
break;
|
||
|
||
case DEDUCE_EXACT:
|
||
sub_strict = UNIFY_ALLOW_NONE;
|
||
break;
|
||
|
||
default:
|
||
my_friendly_abort (0);
|
||
}
|
||
|
||
while (parms
|
||
&& parms != void_list_node
|
||
&& args
|
||
&& args != void_list_node)
|
||
{
|
||
parm = TREE_VALUE (parms);
|
||
parms = TREE_CHAIN (parms);
|
||
arg = TREE_VALUE (args);
|
||
args = TREE_CHAIN (args);
|
||
|
||
if (arg == error_mark_node)
|
||
return 1;
|
||
if (arg == unknown_type_node)
|
||
/* We can't deduce anything from this, but we might get all the
|
||
template args from other function args. */
|
||
continue;
|
||
|
||
/* Conversions will be performed on a function argument that
|
||
corresponds with a function parameter that contains only
|
||
non-deducible template parameters and explicitly specified
|
||
template parameters. */
|
||
if (! uses_template_parms (parm))
|
||
{
|
||
tree type;
|
||
|
||
if (TREE_CODE_CLASS (TREE_CODE (arg)) != 't')
|
||
type = TREE_TYPE (arg);
|
||
else
|
||
{
|
||
type = arg;
|
||
arg = NULL_TREE;
|
||
}
|
||
|
||
if (strict == DEDUCE_EXACT)
|
||
{
|
||
if (same_type_p (parm, type))
|
||
continue;
|
||
}
|
||
else
|
||
/* It might work; we shouldn't check now, because we might
|
||
get into infinite recursion. Overload resolution will
|
||
handle it. */
|
||
continue;
|
||
|
||
return 1;
|
||
}
|
||
|
||
if (TREE_CODE_CLASS (TREE_CODE (arg)) != 't')
|
||
{
|
||
my_friendly_assert (TREE_TYPE (arg) != NULL_TREE, 293);
|
||
if (type_unknown_p (arg))
|
||
{
|
||
/* [temp.deduct.type] A template-argument can be deduced from
|
||
a pointer to function or pointer to member function
|
||
argument if the set of overloaded functions does not
|
||
contain function templates and at most one of a set of
|
||
overloaded functions provides a unique match. */
|
||
|
||
if (resolve_overloaded_unification
|
||
(tparms, targs, parm, arg, strict, sub_strict)
|
||
!= 0)
|
||
return 1;
|
||
continue;
|
||
}
|
||
arg = TREE_TYPE (arg);
|
||
}
|
||
|
||
if (!subr)
|
||
maybe_adjust_types_for_deduction (strict, &parm, &arg);
|
||
|
||
switch (unify (tparms, targs, parm, arg, sub_strict))
|
||
{
|
||
case 0:
|
||
break;
|
||
case 1:
|
||
return 1;
|
||
}
|
||
}
|
||
/* Fail if we've reached the end of the parm list, and more args
|
||
are present, and the parm list isn't variadic. */
|
||
if (args && args != void_list_node && parms == void_list_node)
|
||
return 1;
|
||
/* Fail if parms are left and they don't have default values. */
|
||
if (parms
|
||
&& parms != void_list_node
|
||
&& TREE_PURPOSE (parms) == NULL_TREE)
|
||
return 1;
|
||
if (!subr)
|
||
for (i = 0; i < ntparms; i++)
|
||
if (TREE_VEC_ELT (targs, i) == NULL_TREE)
|
||
{
|
||
if (!allow_incomplete)
|
||
error ("incomplete type unification");
|
||
return 2;
|
||
}
|
||
return 0;
|
||
}
|
||
|
||
/* Subroutine of type_unification_real. Args are like the variables at the
|
||
call site. ARG is an overloaded function (or template-id); we try
|
||
deducing template args from each of the overloads, and if only one
|
||
succeeds, we go with that. Modifies TARGS and returns 0 on success. */
|
||
|
||
static int
|
||
resolve_overloaded_unification (tparms, targs, parm, arg, strict,
|
||
sub_strict)
|
||
tree tparms, targs, parm, arg;
|
||
unification_kind_t strict;
|
||
int sub_strict;
|
||
{
|
||
tree tempargs = copy_node (targs);
|
||
int good = 0;
|
||
|
||
if (TREE_CODE (arg) == ADDR_EXPR)
|
||
arg = TREE_OPERAND (arg, 0);
|
||
|
||
if (TREE_CODE (arg) == COMPONENT_REF)
|
||
/* Handle `&x' where `x' is some static or non-static member
|
||
function name. */
|
||
arg = TREE_OPERAND (arg, 1);
|
||
|
||
if (TREE_CODE (arg) == OFFSET_REF)
|
||
arg = TREE_OPERAND (arg, 1);
|
||
|
||
/* Strip baselink information. */
|
||
while (TREE_CODE (arg) == TREE_LIST)
|
||
arg = TREE_VALUE (arg);
|
||
|
||
if (TREE_CODE (arg) == TEMPLATE_ID_EXPR)
|
||
{
|
||
/* If we got some explicit template args, we need to plug them into
|
||
the affected templates before we try to unify, in case the
|
||
explicit args will completely resolve the templates in question. */
|
||
|
||
tree expl_subargs = TREE_OPERAND (arg, 1);
|
||
arg = TREE_OPERAND (arg, 0);
|
||
|
||
for (; arg; arg = OVL_NEXT (arg))
|
||
{
|
||
tree fn = OVL_CURRENT (arg);
|
||
tree subargs, elem;
|
||
|
||
if (TREE_CODE (fn) != TEMPLATE_DECL)
|
||
continue;
|
||
|
||
subargs = get_bindings_overload (fn, DECL_RESULT (fn), expl_subargs);
|
||
if (subargs)
|
||
{
|
||
elem = tsubst (TREE_TYPE (fn), subargs, /*complain=*/0,
|
||
NULL_TREE);
|
||
if (TREE_CODE (elem) == METHOD_TYPE)
|
||
elem = build_ptrmemfunc_type (build_pointer_type (elem));
|
||
good += try_one_overload (tparms, targs, tempargs, parm, elem,
|
||
strict, sub_strict);
|
||
}
|
||
}
|
||
}
|
||
else if (TREE_CODE (arg) == OVERLOAD)
|
||
{
|
||
for (; arg; arg = OVL_NEXT (arg))
|
||
{
|
||
tree type = TREE_TYPE (OVL_CURRENT (arg));
|
||
if (TREE_CODE (type) == METHOD_TYPE)
|
||
type = build_ptrmemfunc_type (build_pointer_type (type));
|
||
good += try_one_overload (tparms, targs, tempargs, parm,
|
||
type,
|
||
strict, sub_strict);
|
||
}
|
||
}
|
||
else
|
||
my_friendly_abort (981006);
|
||
|
||
/* [temp.deduct.type] A template-argument can be deduced from a pointer
|
||
to function or pointer to member function argument if the set of
|
||
overloaded functions does not contain function templates and at most
|
||
one of a set of overloaded functions provides a unique match.
|
||
|
||
So if we found multiple possibilities, we return success but don't
|
||
deduce anything. */
|
||
|
||
if (good == 1)
|
||
{
|
||
int i = TREE_VEC_LENGTH (targs);
|
||
for (; i--; )
|
||
if (TREE_VEC_ELT (tempargs, i))
|
||
TREE_VEC_ELT (targs, i) = TREE_VEC_ELT (tempargs, i);
|
||
}
|
||
if (good)
|
||
return 0;
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* Subroutine of resolve_overloaded_unification; does deduction for a single
|
||
overload. Fills TARGS with any deduced arguments, or error_mark_node if
|
||
different overloads deduce different arguments for a given parm.
|
||
Returns 1 on success. */
|
||
|
||
static int
|
||
try_one_overload (tparms, orig_targs, targs, parm, arg, strict,
|
||
sub_strict)
|
||
tree tparms, orig_targs, targs, parm, arg;
|
||
unification_kind_t strict;
|
||
int sub_strict;
|
||
{
|
||
int nargs;
|
||
tree tempargs;
|
||
int i;
|
||
|
||
/* [temp.deduct.type] A template-argument can be deduced from a pointer
|
||
to function or pointer to member function argument if the set of
|
||
overloaded functions does not contain function templates and at most
|
||
one of a set of overloaded functions provides a unique match.
|
||
|
||
So if this is a template, just return success. */
|
||
|
||
if (uses_template_parms (arg))
|
||
return 1;
|
||
|
||
maybe_adjust_types_for_deduction (strict, &parm, &arg);
|
||
|
||
/* We don't copy orig_targs for this because if we have already deduced
|
||
some template args from previous args, unify would complain when we
|
||
try to deduce a template parameter for the same argument, even though
|
||
there isn't really a conflict. */
|
||
nargs = TREE_VEC_LENGTH (targs);
|
||
tempargs = make_scratch_vec (nargs);
|
||
|
||
if (unify (tparms, tempargs, parm, arg, sub_strict) != 0)
|
||
return 0;
|
||
|
||
/* First make sure we didn't deduce anything that conflicts with
|
||
explicitly specified args. */
|
||
for (i = nargs; i--; )
|
||
{
|
||
tree elt = TREE_VEC_ELT (tempargs, i);
|
||
tree oldelt = TREE_VEC_ELT (orig_targs, i);
|
||
|
||
if (elt == NULL_TREE)
|
||
continue;
|
||
else if (uses_template_parms (elt))
|
||
{
|
||
/* Since we're unifying against ourselves, we will fill in template
|
||
args used in the function parm list with our own template parms.
|
||
Discard them. */
|
||
TREE_VEC_ELT (tempargs, i) = NULL_TREE;
|
||
continue;
|
||
}
|
||
else if (oldelt && ! template_args_equal (oldelt, elt))
|
||
return 0;
|
||
}
|
||
|
||
for (i = nargs; i--; )
|
||
{
|
||
tree elt = TREE_VEC_ELT (tempargs, i);
|
||
|
||
if (elt)
|
||
TREE_VEC_ELT (targs, i) = elt;
|
||
}
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* PARM is a template class (perhaps with unbound template
|
||
parameters). ARG is a fully instantiated type. If ARG can be
|
||
bound to PARM, return ARG, otherwise return NULL_TREE. TPARMS and
|
||
TARGS are as for unify. */
|
||
|
||
static tree
|
||
try_class_unification (tparms, targs, parm, arg)
|
||
tree tparms;
|
||
tree targs;
|
||
tree parm;
|
||
tree arg;
|
||
{
|
||
int i;
|
||
tree copy_of_targs;
|
||
|
||
if (!CLASSTYPE_TEMPLATE_INFO (arg)
|
||
|| CLASSTYPE_TI_TEMPLATE (arg) != CLASSTYPE_TI_TEMPLATE (parm))
|
||
return NULL_TREE;
|
||
|
||
/* We need to make a new template argument vector for the call to
|
||
unify. If we used TARGS, we'd clutter it up with the result of
|
||
the attempted unification, even if this class didn't work out.
|
||
We also don't want to commit ourselves to all the unifications
|
||
we've already done, since unification is supposed to be done on
|
||
an argument-by-argument basis. In other words, consider the
|
||
following pathological case:
|
||
|
||
template <int I, int J, int K>
|
||
struct S {};
|
||
|
||
template <int I, int J>
|
||
struct S<I, J, 2> : public S<I, I, I>, S<J, J, J> {};
|
||
|
||
template <int I, int J, int K>
|
||
void f(S<I, J, K>, S<I, I, I>);
|
||
|
||
void g() {
|
||
S<0, 0, 0> s0;
|
||
S<0, 1, 2> s2;
|
||
|
||
f(s0, s2);
|
||
}
|
||
|
||
Now, by the time we consider the unification involving `s2', we
|
||
already know that we must have `f<0, 0, 0>'. But, even though
|
||
`S<0, 1, 2>' is derived from `S<0, 0, 0>', the code is not legal
|
||
because there are two ways to unify base classes of S<0, 1, 2>
|
||
with S<I, I, I>. If we kept the already deduced knowledge, we
|
||
would reject the possibility I=1. */
|
||
push_momentary ();
|
||
copy_of_targs = make_temp_vec (TREE_VEC_LENGTH (targs));
|
||
i = unify (tparms, copy_of_targs, CLASSTYPE_TI_ARGS (parm),
|
||
CLASSTYPE_TI_ARGS (arg), UNIFY_ALLOW_NONE);
|
||
pop_momentary ();
|
||
|
||
/* If unification failed, we're done. */
|
||
if (i != 0)
|
||
return NULL_TREE;
|
||
else
|
||
return arg;
|
||
}
|
||
|
||
/* Subroutine of get_template_base. RVAL, if non-NULL, is a base we
|
||
have alreay discovered to be satisfactory. ARG_BINFO is the binfo
|
||
for the base class of ARG that we are currently examining. */
|
||
|
||
static tree
|
||
get_template_base_recursive (tparms, targs, parm,
|
||
arg_binfo, rval, flags)
|
||
tree tparms;
|
||
tree targs;
|
||
tree arg_binfo;
|
||
tree rval;
|
||
tree parm;
|
||
int flags;
|
||
{
|
||
tree binfos;
|
||
int i, n_baselinks;
|
||
tree arg = BINFO_TYPE (arg_binfo);
|
||
|
||
if (!(flags & GTB_IGNORE_TYPE))
|
||
{
|
||
tree r = try_class_unification (tparms, targs,
|
||
parm, arg);
|
||
|
||
/* If there is more than one satisfactory baseclass, then:
|
||
|
||
[temp.deduct.call]
|
||
|
||
If they yield more than one possible deduced A, the type
|
||
deduction fails.
|
||
|
||
applies. */
|
||
if (r && rval && !same_type_p (r, rval))
|
||
return error_mark_node;
|
||
else if (r)
|
||
rval = r;
|
||
}
|
||
|
||
binfos = BINFO_BASETYPES (arg_binfo);
|
||
n_baselinks = binfos ? TREE_VEC_LENGTH (binfos) : 0;
|
||
|
||
/* Process base types. */
|
||
for (i = 0; i < n_baselinks; i++)
|
||
{
|
||
tree base_binfo = TREE_VEC_ELT (binfos, i);
|
||
int this_virtual;
|
||
|
||
/* Skip this base, if we've already seen it. */
|
||
if (BINFO_MARKED (base_binfo))
|
||
continue;
|
||
|
||
this_virtual =
|
||
(flags & GTB_VIA_VIRTUAL) || TREE_VIA_VIRTUAL (base_binfo);
|
||
|
||
/* When searching for a non-virtual, we cannot mark virtually
|
||
found binfos. */
|
||
if (! this_virtual)
|
||
SET_BINFO_MARKED (base_binfo);
|
||
|
||
rval = get_template_base_recursive (tparms, targs,
|
||
parm,
|
||
base_binfo,
|
||
rval,
|
||
GTB_VIA_VIRTUAL * this_virtual);
|
||
|
||
/* If we discovered more than one matching base class, we can
|
||
stop now. */
|
||
if (rval == error_mark_node)
|
||
return error_mark_node;
|
||
}
|
||
|
||
return rval;
|
||
}
|
||
|
||
/* Given a template type PARM and a class type ARG, find the unique
|
||
base type in ARG that is an instance of PARM. We do not examine
|
||
ARG itself; only its base-classes. If there is no appropriate base
|
||
class, return NULL_TREE. If there is more than one, return
|
||
error_mark_node. PARM may be the type of a partial specialization,
|
||
as well as a plain template type. Used by unify. */
|
||
|
||
static tree
|
||
get_template_base (tparms, targs, parm, arg)
|
||
tree tparms;
|
||
tree targs;
|
||
tree parm;
|
||
tree arg;
|
||
{
|
||
tree rval;
|
||
tree arg_binfo;
|
||
|
||
my_friendly_assert (IS_AGGR_TYPE_CODE (TREE_CODE (arg)), 92);
|
||
|
||
arg_binfo = TYPE_BINFO (complete_type (arg));
|
||
rval = get_template_base_recursive (tparms, targs,
|
||
parm, arg_binfo,
|
||
NULL_TREE,
|
||
GTB_IGNORE_TYPE);
|
||
|
||
/* Since get_template_base_recursive marks the bases classes, we
|
||
must unmark them here. */
|
||
dfs_walk (arg_binfo, dfs_unmark, markedp, 0);
|
||
|
||
return rval;
|
||
}
|
||
|
||
/* Returns the level of DECL, which declares a template parameter. */
|
||
|
||
static int
|
||
template_decl_level (decl)
|
||
tree decl;
|
||
{
|
||
switch (TREE_CODE (decl))
|
||
{
|
||
case TYPE_DECL:
|
||
case TEMPLATE_DECL:
|
||
return TEMPLATE_TYPE_LEVEL (TREE_TYPE (decl));
|
||
|
||
case PARM_DECL:
|
||
return TEMPLATE_PARM_LEVEL (DECL_INITIAL (decl));
|
||
|
||
default:
|
||
my_friendly_abort (0);
|
||
return 0;
|
||
}
|
||
}
|
||
|
||
/* Decide whether ARG can be unified with PARM, considering only the
|
||
cv-qualifiers of each type, given STRICT as documented for unify.
|
||
Returns non-zero iff the unification is OK on that basis.*/
|
||
|
||
static int
|
||
check_cv_quals_for_unify (strict, arg, parm)
|
||
int strict;
|
||
tree arg;
|
||
tree parm;
|
||
{
|
||
return !((!(strict & UNIFY_ALLOW_MORE_CV_QUAL)
|
||
&& !at_least_as_qualified_p (arg, parm))
|
||
|| (!(strict & UNIFY_ALLOW_LESS_CV_QUAL)
|
||
&& (!at_least_as_qualified_p (parm, arg))));
|
||
}
|
||
|
||
/* Takes parameters as for type_unification. Returns 0 if the
|
||
type deduction suceeds, 1 otherwise. The parameter STRICT is a
|
||
bitwise or of the following flags:
|
||
|
||
UNIFY_ALLOW_NONE:
|
||
Require an exact match between PARM and ARG.
|
||
UNIFY_ALLOW_MORE_CV_QUAL:
|
||
Allow the deduced ARG to be more cv-qualified than ARG.
|
||
UNIFY_ALLOW_LESS_CV_QUAL:
|
||
Allow the deduced ARG to be less cv-qualified than ARG.
|
||
UNIFY_ALLOW_DERIVED:
|
||
Allow the deduced ARG to be a template base class of ARG,
|
||
or a pointer to a template base class of the type pointed to by
|
||
ARG.
|
||
UNIFY_ALLOW_INTEGER:
|
||
Allow any integral type to be deduced. See the TEMPLATE_PARM_INDEX
|
||
case for more information. */
|
||
|
||
static int
|
||
unify (tparms, targs, parm, arg, strict)
|
||
tree tparms, targs, parm, arg;
|
||
int strict;
|
||
{
|
||
int idx;
|
||
tree targ;
|
||
tree tparm;
|
||
|
||
/* I don't think this will do the right thing with respect to types.
|
||
But the only case I've seen it in so far has been array bounds, where
|
||
signedness is the only information lost, and I think that will be
|
||
okay. */
|
||
while (TREE_CODE (parm) == NOP_EXPR)
|
||
parm = TREE_OPERAND (parm, 0);
|
||
|
||
if (arg == error_mark_node)
|
||
return 1;
|
||
if (arg == unknown_type_node)
|
||
/* We can't deduce anything from this, but we might get all the
|
||
template args from other function args. */
|
||
return 0;
|
||
|
||
/* If PARM uses template parameters, then we can't bail out here,
|
||
even if ARG == PARM, since we won't record unifications for the
|
||
template parameters. We might need them if we're trying to
|
||
figure out which of two things is more specialized. */
|
||
if (arg == parm && !uses_template_parms (parm))
|
||
return 0;
|
||
|
||
/* Immediately reject some pairs that won't unify because of
|
||
cv-qualification mismatches. */
|
||
if (TREE_CODE (arg) == TREE_CODE (parm)
|
||
&& TREE_CODE_CLASS (TREE_CODE (arg)) == 't'
|
||
/* We check the cv-qualifiers when unifying with template type
|
||
parameters below. We want to allow ARG `const T' to unify with
|
||
PARM `T' for example, when computing which of two templates
|
||
is more specialized, for example. */
|
||
&& TREE_CODE (arg) != TEMPLATE_TYPE_PARM
|
||
&& !check_cv_quals_for_unify (strict, arg, parm))
|
||
return 1;
|
||
|
||
switch (TREE_CODE (parm))
|
||
{
|
||
case TYPENAME_TYPE:
|
||
/* In a type which contains a nested-name-specifier, template
|
||
argument values cannot be deduced for template parameters used
|
||
within the nested-name-specifier. */
|
||
return 0;
|
||
|
||
case TEMPLATE_TYPE_PARM:
|
||
case TEMPLATE_TEMPLATE_PARM:
|
||
tparm = TREE_VALUE (TREE_VEC_ELT (tparms, 0));
|
||
|
||
if (TEMPLATE_TYPE_LEVEL (parm)
|
||
!= template_decl_level (tparm))
|
||
/* The PARM is not one we're trying to unify. Just check
|
||
to see if it matches ARG. */
|
||
return (TREE_CODE (arg) == TREE_CODE (parm)
|
||
&& same_type_p (parm, arg)) ? 0 : 1;
|
||
idx = TEMPLATE_TYPE_IDX (parm);
|
||
targ = TREE_VEC_ELT (targs, idx);
|
||
tparm = TREE_VALUE (TREE_VEC_ELT (tparms, idx));
|
||
|
||
/* Check for mixed types and values. */
|
||
if ((TREE_CODE (parm) == TEMPLATE_TYPE_PARM
|
||
&& TREE_CODE (tparm) != TYPE_DECL)
|
||
|| (TREE_CODE (parm) == TEMPLATE_TEMPLATE_PARM
|
||
&& TREE_CODE (tparm) != TEMPLATE_DECL))
|
||
return 1;
|
||
|
||
if (TREE_CODE (parm) == TEMPLATE_TEMPLATE_PARM)
|
||
{
|
||
if (TEMPLATE_TEMPLATE_PARM_TEMPLATE_INFO (parm))
|
||
{
|
||
/* We arrive here when PARM does not involve template
|
||
specialization. */
|
||
|
||
/* ARG must be constructed from a template class. */
|
||
if (TREE_CODE (arg) != RECORD_TYPE || !CLASSTYPE_TEMPLATE_INFO (arg))
|
||
return 1;
|
||
|
||
{
|
||
tree parmtmpl = TYPE_TI_TEMPLATE (parm);
|
||
tree parmvec = TYPE_TI_ARGS (parm);
|
||
tree argvec = CLASSTYPE_TI_ARGS (arg);
|
||
tree argtmplvec
|
||
= DECL_INNERMOST_TEMPLATE_PARMS (CLASSTYPE_TI_TEMPLATE (arg));
|
||
int i;
|
||
|
||
/* The parameter and argument roles have to be switched here
|
||
in order to handle default arguments properly. For example,
|
||
template<template <class> class TT> void f(TT<int>)
|
||
should be able to accept vector<int> which comes from
|
||
template <class T, class Allocator = allocator>
|
||
class vector. */
|
||
|
||
if (coerce_template_parms (argtmplvec, parmvec, parmtmpl, 0, 1)
|
||
== error_mark_node)
|
||
return 1;
|
||
|
||
/* Deduce arguments T, i from TT<T> or TT<i>.
|
||
We check each element of PARMVEC and ARGVEC individually
|
||
rather than the whole TREE_VEC since they can have
|
||
different number of elements. */
|
||
|
||
for (i = 0; i < TREE_VEC_LENGTH (parmvec); ++i)
|
||
{
|
||
tree t = TREE_VEC_ELT (parmvec, i);
|
||
|
||
if (unify (tparms, targs, t,
|
||
TREE_VEC_ELT (argvec, i),
|
||
UNIFY_ALLOW_NONE))
|
||
return 1;
|
||
}
|
||
}
|
||
arg = CLASSTYPE_TI_TEMPLATE (arg);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
/* If PARM is `const T' and ARG is only `int', we don't have
|
||
a match unless we are allowing additional qualification.
|
||
If ARG is `const int' and PARM is just `T' that's OK;
|
||
that binds `const int' to `T'. */
|
||
if (!check_cv_quals_for_unify (strict | UNIFY_ALLOW_LESS_CV_QUAL,
|
||
arg, parm))
|
||
return 1;
|
||
|
||
/* Consider the case where ARG is `const volatile int' and
|
||
PARM is `const T'. Then, T should be `volatile int'. */
|
||
arg =
|
||
cp_build_qualified_type (arg,
|
||
CP_TYPE_QUALS (arg)
|
||
& ~CP_TYPE_QUALS (parm));
|
||
}
|
||
|
||
/* Simple cases: Value already set, does match or doesn't. */
|
||
if (targ != NULL_TREE && same_type_p (targ, arg))
|
||
return 0;
|
||
else if (targ)
|
||
return 1;
|
||
|
||
/* Make sure that ARG is not a variable-sized array. (Note that
|
||
were talking about variable-sized arrays (like `int[n]'),
|
||
rather than arrays of unknown size (like `int[]').) We'll
|
||
get very confused by such a type since the bound of the array
|
||
will not be computable in an instantiation. Besides, such
|
||
types are not allowed in ISO C++, so we can do as we please
|
||
here. */
|
||
if (TREE_CODE (arg) == ARRAY_TYPE
|
||
&& !uses_template_parms (arg)
|
||
&& (TREE_CODE (TYPE_MAX_VALUE (TYPE_DOMAIN (arg)))
|
||
!= INTEGER_CST))
|
||
return 1;
|
||
|
||
TREE_VEC_ELT (targs, idx) = arg;
|
||
return 0;
|
||
|
||
case TEMPLATE_PARM_INDEX:
|
||
tparm = TREE_VALUE (TREE_VEC_ELT (tparms, 0));
|
||
|
||
if (TEMPLATE_PARM_LEVEL (parm)
|
||
!= template_decl_level (tparm))
|
||
/* The PARM is not one we're trying to unify. Just check
|
||
to see if it matches ARG. */
|
||
return (TREE_CODE (arg) == TREE_CODE (parm)
|
||
&& cp_tree_equal (parm, arg) > 0) ? 0 : 1;
|
||
|
||
idx = TEMPLATE_PARM_IDX (parm);
|
||
targ = TREE_VEC_ELT (targs, idx);
|
||
|
||
if (targ)
|
||
{
|
||
int i = (cp_tree_equal (targ, arg) > 0);
|
||
if (i == 1)
|
||
return 0;
|
||
else if (i == 0)
|
||
return 1;
|
||
else
|
||
my_friendly_abort (42);
|
||
}
|
||
|
||
/* [temp.deduct.type] If, in the declaration of a function template
|
||
with a non-type template-parameter, the non-type
|
||
template-parameter is used in an expression in the function
|
||
parameter-list and, if the corresponding template-argument is
|
||
deduced, the template-argument type shall match the type of the
|
||
template-parameter exactly, except that a template-argument
|
||
deduced from an array bound may be of any integral type. */
|
||
if (same_type_p (TREE_TYPE (arg), TREE_TYPE (parm)))
|
||
/* OK */;
|
||
else if ((strict & UNIFY_ALLOW_INTEGER)
|
||
&& (TREE_CODE (TREE_TYPE (parm)) == INTEGER_TYPE
|
||
|| TREE_CODE (TREE_TYPE (parm)) == BOOLEAN_TYPE))
|
||
/* OK */;
|
||
else
|
||
return 1;
|
||
|
||
TREE_VEC_ELT (targs, idx) = copy_to_permanent (arg);
|
||
return 0;
|
||
|
||
case POINTER_TYPE:
|
||
{
|
||
int sub_strict;
|
||
|
||
if (TREE_CODE (arg) == RECORD_TYPE && TYPE_PTRMEMFUNC_FLAG (arg))
|
||
return (unify (tparms, targs, parm,
|
||
TYPE_PTRMEMFUNC_FN_TYPE (arg), strict));
|
||
|
||
if (TREE_CODE (arg) != POINTER_TYPE)
|
||
return 1;
|
||
|
||
/* [temp.deduct.call]
|
||
|
||
A can be another pointer or pointer to member type that can
|
||
be converted to the deduced A via a qualification
|
||
conversion (_conv.qual_).
|
||
|
||
We pass down STRICT here rather than UNIFY_ALLOW_NONE.
|
||
This will allow for additional cv-qualification of the
|
||
pointed-to types if appropriate. In general, this is a bit
|
||
too generous; we are only supposed to allow qualification
|
||
conversions and this method will allow an ARG of char** and
|
||
a deduced ARG of const char**. However, overload
|
||
resolution will subsequently invalidate the candidate, so
|
||
this is probably OK. */
|
||
sub_strict = strict;
|
||
|
||
if (TREE_CODE (TREE_TYPE (arg)) != RECORD_TYPE
|
||
|| TYPE_PTRMEMFUNC_FLAG (TREE_TYPE (arg)))
|
||
/* The derived-to-base conversion only persists through one
|
||
level of pointers. */
|
||
sub_strict &= ~UNIFY_ALLOW_DERIVED;
|
||
|
||
return unify (tparms, targs, TREE_TYPE (parm), TREE_TYPE
|
||
(arg), sub_strict);
|
||
}
|
||
|
||
case REFERENCE_TYPE:
|
||
if (TREE_CODE (arg) != REFERENCE_TYPE)
|
||
return 1;
|
||
return unify (tparms, targs, TREE_TYPE (parm), TREE_TYPE (arg),
|
||
UNIFY_ALLOW_NONE);
|
||
|
||
case ARRAY_TYPE:
|
||
if (TREE_CODE (arg) != ARRAY_TYPE)
|
||
return 1;
|
||
if ((TYPE_DOMAIN (parm) == NULL_TREE)
|
||
!= (TYPE_DOMAIN (arg) == NULL_TREE))
|
||
return 1;
|
||
if (TYPE_DOMAIN (parm) != NULL_TREE
|
||
&& unify (tparms, targs, TYPE_DOMAIN (parm),
|
||
TYPE_DOMAIN (arg), UNIFY_ALLOW_NONE) != 0)
|
||
return 1;
|
||
return unify (tparms, targs, TREE_TYPE (parm), TREE_TYPE (arg),
|
||
UNIFY_ALLOW_NONE);
|
||
|
||
case REAL_TYPE:
|
||
case COMPLEX_TYPE:
|
||
case INTEGER_TYPE:
|
||
case BOOLEAN_TYPE:
|
||
case VOID_TYPE:
|
||
if (TREE_CODE (arg) != TREE_CODE (parm))
|
||
return 1;
|
||
|
||
if (TREE_CODE (parm) == INTEGER_TYPE
|
||
&& TREE_CODE (TYPE_MAX_VALUE (parm)) != INTEGER_CST)
|
||
{
|
||
if (TYPE_MIN_VALUE (parm) && TYPE_MIN_VALUE (arg)
|
||
&& unify (tparms, targs, TYPE_MIN_VALUE (parm),
|
||
TYPE_MIN_VALUE (arg), UNIFY_ALLOW_INTEGER))
|
||
return 1;
|
||
if (TYPE_MAX_VALUE (parm) && TYPE_MAX_VALUE (arg)
|
||
&& unify (tparms, targs, TYPE_MAX_VALUE (parm),
|
||
TYPE_MAX_VALUE (arg), UNIFY_ALLOW_INTEGER))
|
||
return 1;
|
||
}
|
||
/* We use the TYPE_MAIN_VARIANT since we have already
|
||
checked cv-qualification at the top of the
|
||
function. */
|
||
else if (!same_type_p (TYPE_MAIN_VARIANT (arg),
|
||
TYPE_MAIN_VARIANT (parm)))
|
||
return 1;
|
||
|
||
/* As far as unification is concerned, this wins. Later checks
|
||
will invalidate it if necessary. */
|
||
return 0;
|
||
|
||
/* Types INTEGER_CST and MINUS_EXPR can come from array bounds. */
|
||
/* Type INTEGER_CST can come from ordinary constant template args. */
|
||
case INTEGER_CST:
|
||
while (TREE_CODE (arg) == NOP_EXPR)
|
||
arg = TREE_OPERAND (arg, 0);
|
||
|
||
if (TREE_CODE (arg) != INTEGER_CST)
|
||
return 1;
|
||
return !tree_int_cst_equal (parm, arg);
|
||
|
||
case TREE_VEC:
|
||
{
|
||
int i;
|
||
if (TREE_CODE (arg) != TREE_VEC)
|
||
return 1;
|
||
if (TREE_VEC_LENGTH (parm) != TREE_VEC_LENGTH (arg))
|
||
return 1;
|
||
for (i = TREE_VEC_LENGTH (parm) - 1; i >= 0; i--)
|
||
if (unify (tparms, targs,
|
||
TREE_VEC_ELT (parm, i), TREE_VEC_ELT (arg, i),
|
||
UNIFY_ALLOW_NONE))
|
||
return 1;
|
||
return 0;
|
||
}
|
||
|
||
case RECORD_TYPE:
|
||
case UNION_TYPE:
|
||
if (TYPE_PTRMEMFUNC_FLAG (parm))
|
||
return unify (tparms, targs, TYPE_PTRMEMFUNC_FN_TYPE (parm),
|
||
arg, strict);
|
||
|
||
if (TREE_CODE (arg) != TREE_CODE (parm))
|
||
return 1;
|
||
|
||
if (CLASSTYPE_TEMPLATE_INFO (parm))
|
||
{
|
||
tree t = NULL_TREE;
|
||
|
||
if (strict & UNIFY_ALLOW_DERIVED)
|
||
{
|
||
/* First, we try to unify the PARM and ARG directly. */
|
||
t = try_class_unification (tparms, targs,
|
||
parm, arg);
|
||
|
||
if (!t)
|
||
{
|
||
/* Fallback to the special case allowed in
|
||
[temp.deduct.call]:
|
||
|
||
If P is a class, and P has the form
|
||
template-id, then A can be a derived class of
|
||
the deduced A. Likewise, if P is a pointer to
|
||
a class of the form template-id, A can be a
|
||
pointer to a derived class pointed to by the
|
||
deduced A. */
|
||
t = get_template_base (tparms, targs,
|
||
parm, arg);
|
||
|
||
if (! t || t == error_mark_node)
|
||
return 1;
|
||
}
|
||
}
|
||
else if (CLASSTYPE_TEMPLATE_INFO (arg)
|
||
&& (CLASSTYPE_TI_TEMPLATE (parm)
|
||
== CLASSTYPE_TI_TEMPLATE (arg)))
|
||
/* Perhaps PARM is something like S<U> and ARG is S<int>.
|
||
Then, we should unify `int' and `U'. */
|
||
t = arg;
|
||
else
|
||
/* There's no chance of unication succeeding. */
|
||
return 1;
|
||
|
||
return unify (tparms, targs, CLASSTYPE_TI_ARGS (parm),
|
||
CLASSTYPE_TI_ARGS (t), UNIFY_ALLOW_NONE);
|
||
}
|
||
else if (!same_type_p (TYPE_MAIN_VARIANT (parm),
|
||
TYPE_MAIN_VARIANT (arg)))
|
||
return 1;
|
||
return 0;
|
||
|
||
case METHOD_TYPE:
|
||
case FUNCTION_TYPE:
|
||
if (TREE_CODE (arg) != TREE_CODE (parm))
|
||
return 1;
|
||
|
||
if (unify (tparms, targs, TREE_TYPE (parm),
|
||
TREE_TYPE (arg), UNIFY_ALLOW_NONE))
|
||
return 1;
|
||
return type_unification_real (tparms, targs, TYPE_ARG_TYPES (parm),
|
||
TYPE_ARG_TYPES (arg), 1,
|
||
DEDUCE_EXACT, 0);
|
||
|
||
case OFFSET_TYPE:
|
||
if (TREE_CODE (arg) != OFFSET_TYPE)
|
||
return 1;
|
||
if (unify (tparms, targs, TYPE_OFFSET_BASETYPE (parm),
|
||
TYPE_OFFSET_BASETYPE (arg), UNIFY_ALLOW_NONE))
|
||
return 1;
|
||
return unify (tparms, targs, TREE_TYPE (parm), TREE_TYPE (arg),
|
||
strict);
|
||
|
||
case CONST_DECL:
|
||
if (arg != decl_constant_value (parm))
|
||
return 1;
|
||
return 0;
|
||
|
||
case TEMPLATE_DECL:
|
||
/* Matched cases are handled by the ARG == PARM test above. */
|
||
return 1;
|
||
|
||
case MINUS_EXPR:
|
||
if (TREE_CODE (TREE_OPERAND (parm, 1)) == INTEGER_CST)
|
||
{
|
||
/* We handle this case specially, since it comes up with
|
||
arrays. In particular, something like:
|
||
|
||
template <int N> void f(int (&x)[N]);
|
||
|
||
Here, we are trying to unify the range type, which
|
||
looks like [0 ... (N - 1)]. */
|
||
tree t, t1, t2;
|
||
t1 = TREE_OPERAND (parm, 0);
|
||
t2 = TREE_OPERAND (parm, 1);
|
||
|
||
/* Should this be a regular fold? */
|
||
t = maybe_fold_nontype_arg (build (PLUS_EXPR,
|
||
integer_type_node,
|
||
arg, t2));
|
||
|
||
return unify (tparms, targs, t1, t, strict);
|
||
}
|
||
/* else fall through */
|
||
|
||
default:
|
||
if (IS_EXPR_CODE_CLASS (TREE_CODE_CLASS (TREE_CODE (parm))))
|
||
/* We're looking at an expression. This can happen with
|
||
something like:
|
||
|
||
template <int I>
|
||
void foo(S<I>, S<I + 2>);
|
||
|
||
This is a "nondeduced context":
|
||
|
||
[deduct.type]
|
||
|
||
The nondeduced contexts are:
|
||
|
||
--A type that is a template-id in which one or more of
|
||
the template-arguments is an expression that references
|
||
a template-parameter.
|
||
|
||
In these cases, we assume deduction succeeded, but don't
|
||
actually infer any unifications. */
|
||
return 0;
|
||
else
|
||
sorry ("use of `%s' in template type unification",
|
||
tree_code_name [(int) TREE_CODE (parm)]);
|
||
|
||
return 1;
|
||
}
|
||
}
|
||
|
||
/* Called if RESULT is explicitly instantiated, or is a member of an
|
||
explicitly instantiated class, or if using -frepo and the
|
||
instantiation of RESULT has been assigned to this file. */
|
||
|
||
void
|
||
mark_decl_instantiated (result, extern_p)
|
||
tree result;
|
||
int extern_p;
|
||
{
|
||
if (TREE_CODE (result) != FUNCTION_DECL)
|
||
/* The TREE_PUBLIC flag for function declarations will have been
|
||
set correctly by tsubst. */
|
||
TREE_PUBLIC (result) = 1;
|
||
|
||
if (! extern_p)
|
||
{
|
||
DECL_INTERFACE_KNOWN (result) = 1;
|
||
DECL_NOT_REALLY_EXTERN (result) = 1;
|
||
|
||
/* Always make artificials weak. */
|
||
if (DECL_ARTIFICIAL (result) && flag_weak)
|
||
comdat_linkage (result);
|
||
/* For WIN32 we also want to put explicit instantiations in
|
||
linkonce sections. */
|
||
else if (TREE_PUBLIC (result))
|
||
maybe_make_one_only (result);
|
||
}
|
||
else if (TREE_CODE (result) == FUNCTION_DECL)
|
||
mark_inline_for_output (result);
|
||
}
|
||
|
||
/* Given two function templates PAT1 and PAT2, and explicit template
|
||
arguments EXPLICIT_ARGS return:
|
||
|
||
1 if PAT1 is more specialized than PAT2 as described in [temp.func.order].
|
||
-1 if PAT2 is more specialized than PAT1.
|
||
0 if neither is more specialized. */
|
||
|
||
int
|
||
more_specialized (pat1, pat2, explicit_args)
|
||
tree pat1, pat2, explicit_args;
|
||
{
|
||
tree targs;
|
||
int winner = 0;
|
||
|
||
targs = get_bindings_overload (pat1, DECL_RESULT (pat2), explicit_args);
|
||
if (targs)
|
||
--winner;
|
||
|
||
targs = get_bindings_overload (pat2, DECL_RESULT (pat1), explicit_args);
|
||
if (targs)
|
||
++winner;
|
||
|
||
return winner;
|
||
}
|
||
|
||
/* Given two class template specialization list nodes PAT1 and PAT2, return:
|
||
|
||
1 if PAT1 is more specialized than PAT2 as described in [temp.class.order].
|
||
-1 if PAT2 is more specialized than PAT1.
|
||
0 if neither is more specialized. */
|
||
|
||
int
|
||
more_specialized_class (pat1, pat2)
|
||
tree pat1, pat2;
|
||
{
|
||
tree targs;
|
||
int winner = 0;
|
||
|
||
targs = get_class_bindings (TREE_VALUE (pat1), TREE_PURPOSE (pat1),
|
||
TREE_PURPOSE (pat2));
|
||
if (targs)
|
||
--winner;
|
||
|
||
targs = get_class_bindings (TREE_VALUE (pat2), TREE_PURPOSE (pat2),
|
||
TREE_PURPOSE (pat1));
|
||
if (targs)
|
||
++winner;
|
||
|
||
return winner;
|
||
}
|
||
|
||
/* Return the template arguments that will produce the function signature
|
||
DECL from the function template FN, with the explicit template
|
||
arguments EXPLICIT_ARGS. If CHECK_RETTYPE is 1, the return type must
|
||
also match. Return NULL_TREE if no satisfactory arguments could be
|
||
found. */
|
||
|
||
static tree
|
||
get_bindings_real (fn, decl, explicit_args, check_rettype)
|
||
tree fn, decl, explicit_args;
|
||
int check_rettype;
|
||
{
|
||
int ntparms = DECL_NTPARMS (fn);
|
||
tree targs = make_scratch_vec (ntparms);
|
||
tree decl_type;
|
||
tree decl_arg_types;
|
||
int i;
|
||
|
||
/* Substitute the explicit template arguments into the type of DECL.
|
||
The call to fn_type_unification will handle substitution into the
|
||
FN. */
|
||
decl_type = TREE_TYPE (decl);
|
||
if (explicit_args && uses_template_parms (decl_type))
|
||
{
|
||
tree tmpl;
|
||
tree converted_args;
|
||
|
||
if (DECL_TEMPLATE_INFO (decl))
|
||
tmpl = DECL_TI_TEMPLATE (decl);
|
||
else
|
||
/* We can get here for some illegal specializations. */
|
||
return NULL_TREE;
|
||
|
||
converted_args
|
||
= (coerce_template_parms (DECL_INNERMOST_TEMPLATE_PARMS (tmpl),
|
||
explicit_args, NULL_TREE,
|
||
/*complain=*/0,
|
||
/*require_all_arguments=*/0));
|
||
if (converted_args == error_mark_node)
|
||
return NULL_TREE;
|
||
|
||
decl_type = tsubst (decl_type, converted_args, /*complain=*/0,
|
||
NULL_TREE);
|
||
if (decl_type == error_mark_node)
|
||
return NULL_TREE;
|
||
}
|
||
|
||
/* If FN is a static member function, adjust the type of DECL
|
||
appropriately. */
|
||
decl_arg_types = TYPE_ARG_TYPES (decl_type);
|
||
if (DECL_STATIC_FUNCTION_P (fn)
|
||
&& DECL_NONSTATIC_MEMBER_FUNCTION_P (decl))
|
||
decl_arg_types = TREE_CHAIN (decl_arg_types);
|
||
|
||
i = fn_type_unification (fn, explicit_args, targs,
|
||
decl_arg_types,
|
||
TREE_TYPE (decl_type),
|
||
DEDUCE_EXACT);
|
||
|
||
if (i != 0)
|
||
return NULL_TREE;
|
||
|
||
if (check_rettype)
|
||
{
|
||
/* Check to see that the resulting return type is also OK. */
|
||
tree t = tsubst (TREE_TYPE (TREE_TYPE (fn)), targs,
|
||
/*complain=*/0, NULL_TREE);
|
||
|
||
if (!same_type_p (t, TREE_TYPE (TREE_TYPE (decl))))
|
||
return NULL_TREE;
|
||
}
|
||
|
||
return targs;
|
||
}
|
||
|
||
/* For most uses, we want to check the return type. */
|
||
|
||
tree
|
||
get_bindings (fn, decl, explicit_args)
|
||
tree fn, decl, explicit_args;
|
||
{
|
||
return get_bindings_real (fn, decl, explicit_args, 1);
|
||
}
|
||
|
||
/* But for more_specialized, we only care about the parameter types. */
|
||
|
||
static tree
|
||
get_bindings_overload (fn, decl, explicit_args)
|
||
tree fn, decl, explicit_args;
|
||
{
|
||
return get_bindings_real (fn, decl, explicit_args, 0);
|
||
}
|
||
|
||
/* Return the innermost template arguments that, when applied to a
|
||
template specialization whose innermost template parameters are
|
||
TPARMS, and whose specialization arguments are ARGS, yield the
|
||
ARGS.
|
||
|
||
For example, suppose we have:
|
||
|
||
template <class T, class U> struct S {};
|
||
template <class T> struct S<T*, int> {};
|
||
|
||
Then, suppose we want to get `S<double*, int>'. The TPARMS will be
|
||
{T}, the PARMS will be {T*, int} and the ARGS will be {double*,
|
||
int}. The resulting vector will be {double}, indicating that `T'
|
||
is bound to `double'. */
|
||
|
||
static tree
|
||
get_class_bindings (tparms, parms, args)
|
||
tree tparms, parms, args;
|
||
{
|
||
int i, ntparms = TREE_VEC_LENGTH (tparms);
|
||
tree vec = make_temp_vec (ntparms);
|
||
|
||
args = innermost_args (args);
|
||
|
||
if (unify (tparms, vec, parms, args, UNIFY_ALLOW_NONE))
|
||
return NULL_TREE;
|
||
|
||
for (i = 0; i < ntparms; ++i)
|
||
if (! TREE_VEC_ELT (vec, i))
|
||
return NULL_TREE;
|
||
|
||
return vec;
|
||
}
|
||
|
||
/* In INSTANTIATIONS is a list of <INSTANTIATION, TEMPLATE> pairs.
|
||
Pick the most specialized template, and return the corresponding
|
||
instantiation, or if there is no corresponding instantiation, the
|
||
template itself. EXPLICIT_ARGS is any template arguments explicity
|
||
mentioned in a template-id. If there is no most specialized
|
||
tempalte, error_mark_node is returned. If there are no templates
|
||
at all, NULL_TREE is returned. */
|
||
|
||
tree
|
||
most_specialized_instantiation (instantiations, explicit_args)
|
||
tree instantiations;
|
||
tree explicit_args;
|
||
{
|
||
tree fn, champ;
|
||
int fate;
|
||
|
||
if (!instantiations)
|
||
return NULL_TREE;
|
||
|
||
champ = instantiations;
|
||
for (fn = TREE_CHAIN (instantiations); fn; fn = TREE_CHAIN (fn))
|
||
{
|
||
fate = more_specialized (TREE_VALUE (champ),
|
||
TREE_VALUE (fn), explicit_args);
|
||
if (fate == 1)
|
||
;
|
||
else
|
||
{
|
||
if (fate == 0)
|
||
{
|
||
fn = TREE_CHAIN (fn);
|
||
if (! fn)
|
||
return error_mark_node;
|
||
}
|
||
champ = fn;
|
||
}
|
||
}
|
||
|
||
for (fn = instantiations; fn && fn != champ; fn = TREE_CHAIN (fn))
|
||
{
|
||
fate = more_specialized (TREE_VALUE (champ),
|
||
TREE_VALUE (fn), explicit_args);
|
||
if (fate != 1)
|
||
return error_mark_node;
|
||
}
|
||
|
||
return TREE_PURPOSE (champ) ? TREE_PURPOSE (champ) : TREE_VALUE (champ);
|
||
}
|
||
|
||
/* Return the most specialized of the list of templates in FNS that can
|
||
produce an instantiation matching DECL, given the explicit template
|
||
arguments EXPLICIT_ARGS. */
|
||
|
||
static tree
|
||
most_specialized (fns, decl, explicit_args)
|
||
tree fns, decl, explicit_args;
|
||
{
|
||
tree candidates = NULL_TREE;
|
||
tree fn, args;
|
||
|
||
for (fn = fns; fn; fn = TREE_CHAIN (fn))
|
||
{
|
||
tree candidate = TREE_VALUE (fn);
|
||
|
||
args = get_bindings (candidate, decl, explicit_args);
|
||
if (args)
|
||
candidates = scratch_tree_cons (NULL_TREE, candidate,
|
||
candidates);
|
||
}
|
||
|
||
return most_specialized_instantiation (candidates, explicit_args);
|
||
}
|
||
|
||
/* If DECL is a specialization of some template, return the most
|
||
general such template. For example, given:
|
||
|
||
template <class T> struct S { template <class U> void f(U); };
|
||
|
||
if TMPL is `template <class U> void S<int>::f(U)' this will return
|
||
the full template. This function will not trace past partial
|
||
specializations, however. For example, given in addition:
|
||
|
||
template <class T> struct S<T*> { template <class U> void f(U); };
|
||
|
||
if TMPL is `template <class U> void S<int*>::f(U)' this will return
|
||
`template <class T> template <class U> S<T*>::f(U)'. */
|
||
|
||
static tree
|
||
most_general_template (decl)
|
||
tree decl;
|
||
{
|
||
while (DECL_TEMPLATE_INFO (decl))
|
||
decl = DECL_TI_TEMPLATE (decl);
|
||
|
||
return decl;
|
||
}
|
||
|
||
/* Return the most specialized of the class template specializations
|
||
of TMPL which can produce an instantiation matching ARGS, or
|
||
error_mark_node if the choice is ambiguous. */
|
||
|
||
static tree
|
||
most_specialized_class (tmpl, args)
|
||
tree tmpl;
|
||
tree args;
|
||
{
|
||
tree list = NULL_TREE;
|
||
tree t;
|
||
tree champ;
|
||
int fate;
|
||
|
||
tmpl = most_general_template (tmpl);
|
||
for (t = DECL_TEMPLATE_SPECIALIZATIONS (tmpl); t; t = TREE_CHAIN (t))
|
||
{
|
||
tree spec_args
|
||
= get_class_bindings (TREE_VALUE (t), TREE_PURPOSE (t), args);
|
||
if (spec_args)
|
||
{
|
||
list = decl_tree_cons (TREE_PURPOSE (t), TREE_VALUE (t), list);
|
||
TREE_TYPE (list) = TREE_TYPE (t);
|
||
}
|
||
}
|
||
|
||
if (! list)
|
||
return NULL_TREE;
|
||
|
||
t = list;
|
||
champ = t;
|
||
t = TREE_CHAIN (t);
|
||
for (; t; t = TREE_CHAIN (t))
|
||
{
|
||
fate = more_specialized_class (champ, t);
|
||
if (fate == 1)
|
||
;
|
||
else
|
||
{
|
||
if (fate == 0)
|
||
{
|
||
t = TREE_CHAIN (t);
|
||
if (! t)
|
||
return error_mark_node;
|
||
}
|
||
champ = t;
|
||
}
|
||
}
|
||
|
||
for (t = list; t && t != champ; t = TREE_CHAIN (t))
|
||
{
|
||
fate = more_specialized_class (champ, t);
|
||
if (fate != 1)
|
||
return error_mark_node;
|
||
}
|
||
|
||
return champ;
|
||
}
|
||
|
||
/* called from the parser. */
|
||
|
||
void
|
||
do_decl_instantiation (declspecs, declarator, storage)
|
||
tree declspecs, declarator, storage;
|
||
{
|
||
tree decl = grokdeclarator (declarator, declspecs, NORMAL, 0, NULL_TREE);
|
||
tree result = NULL_TREE;
|
||
int extern_p = 0;
|
||
|
||
if (! DECL_LANG_SPECIFIC (decl))
|
||
{
|
||
cp_error ("explicit instantiation of non-template `%#D'", decl);
|
||
return;
|
||
}
|
||
else if (TREE_CODE (decl) == VAR_DECL)
|
||
{
|
||
/* There is an asymmetry here in the way VAR_DECLs and
|
||
FUNCTION_DECLs are handled by grokdeclarator. In the case of
|
||
the latter, the DECL we get back will be marked as a
|
||
template instantiation, and the appropriate
|
||
DECL_TEMPLATE_INFO will be set up. This does not happen for
|
||
VAR_DECLs so we do the lookup here. Probably, grokdeclarator
|
||
should handle VAR_DECLs as it currently handles
|
||
FUNCTION_DECLs. */
|
||
result = lookup_field (DECL_CONTEXT (decl), DECL_NAME (decl), 0, 0);
|
||
if (result && TREE_CODE (result) != VAR_DECL)
|
||
{
|
||
cp_error ("no matching template for `%D' found", result);
|
||
return;
|
||
}
|
||
}
|
||
else if (TREE_CODE (decl) != FUNCTION_DECL)
|
||
{
|
||
cp_error ("explicit instantiation of `%#D'", decl);
|
||
return;
|
||
}
|
||
else
|
||
result = decl;
|
||
|
||
/* Check for various error cases. Note that if the explicit
|
||
instantiation is legal the RESULT will currently be marked as an
|
||
*implicit* instantiation; DECL_EXPLICIT_INSTANTIATION is not set
|
||
until we get here. */
|
||
|
||
if (DECL_TEMPLATE_SPECIALIZATION (result))
|
||
{
|
||
/* [temp.spec]
|
||
|
||
No program shall both explicitly instantiate and explicitly
|
||
specialize a template. */
|
||
cp_pedwarn ("explicit instantiation of `%#D' after", result);
|
||
cp_pedwarn_at ("explicit specialization here", result);
|
||
return;
|
||
}
|
||
else if (DECL_EXPLICIT_INSTANTIATION (result))
|
||
{
|
||
/* [temp.spec]
|
||
|
||
No program shall explicitly instantiate any template more
|
||
than once.
|
||
|
||
We check DECL_INTERFACE_KNOWN so as not to complain when the
|
||
first instantiation was `extern' and the second is not, and
|
||
EXTERN_P for the opposite case. */
|
||
if (DECL_INTERFACE_KNOWN (result) && !extern_p && !flag_use_repository)
|
||
cp_pedwarn ("duplicate explicit instantiation of `%#D'", result);
|
||
|
||
/* If we've already instantiated the template, just return now. */
|
||
if (DECL_INTERFACE_KNOWN (result))
|
||
return;
|
||
}
|
||
else if (!DECL_IMPLICIT_INSTANTIATION (result))
|
||
{
|
||
cp_error ("no matching template for `%D' found", result);
|
||
return;
|
||
}
|
||
else if (!DECL_TEMPLATE_INFO (result))
|
||
{
|
||
cp_pedwarn ("explicit instantiation of non-template `%#D'", result);
|
||
return;
|
||
}
|
||
|
||
if (flag_external_templates)
|
||
return;
|
||
|
||
if (storage == NULL_TREE)
|
||
;
|
||
else if (storage == ridpointers[(int) RID_EXTERN])
|
||
{
|
||
if (pedantic)
|
||
cp_pedwarn ("ANSI C++ forbids the use of `extern' on explicit instantiations");
|
||
extern_p = 1;
|
||
}
|
||
else
|
||
cp_error ("storage class `%D' applied to template instantiation",
|
||
storage);
|
||
|
||
SET_DECL_EXPLICIT_INSTANTIATION (result);
|
||
mark_decl_instantiated (result, extern_p);
|
||
repo_template_instantiated (result, extern_p);
|
||
if (! extern_p)
|
||
instantiate_decl (result);
|
||
}
|
||
|
||
void
|
||
mark_class_instantiated (t, extern_p)
|
||
tree t;
|
||
int extern_p;
|
||
{
|
||
SET_CLASSTYPE_EXPLICIT_INSTANTIATION (t);
|
||
SET_CLASSTYPE_INTERFACE_KNOWN (t);
|
||
CLASSTYPE_INTERFACE_ONLY (t) = extern_p;
|
||
CLASSTYPE_VTABLE_NEEDS_WRITING (t) = ! extern_p;
|
||
TYPE_DECL_SUPPRESS_DEBUG (TYPE_NAME (t)) = extern_p;
|
||
if (! extern_p)
|
||
{
|
||
CLASSTYPE_DEBUG_REQUESTED (t) = 1;
|
||
rest_of_type_compilation (t, 1);
|
||
}
|
||
}
|
||
|
||
void
|
||
do_type_instantiation (t, storage)
|
||
tree t, storage;
|
||
{
|
||
int extern_p = 0;
|
||
int nomem_p = 0;
|
||
int static_p = 0;
|
||
|
||
if (TREE_CODE (t) == TYPE_DECL)
|
||
t = TREE_TYPE (t);
|
||
|
||
if (! CLASS_TYPE_P (t) || ! CLASSTYPE_TEMPLATE_INFO (t))
|
||
{
|
||
cp_error ("explicit instantiation of non-template type `%T'", t);
|
||
return;
|
||
}
|
||
|
||
complete_type (t);
|
||
|
||
/* With -fexternal-templates, explicit instantiations are treated the same
|
||
as implicit ones. */
|
||
if (flag_external_templates)
|
||
return;
|
||
|
||
if (TYPE_SIZE (t) == NULL_TREE)
|
||
{
|
||
cp_error ("explicit instantiation of `%#T' before definition of template",
|
||
t);
|
||
return;
|
||
}
|
||
|
||
if (storage != NULL_TREE)
|
||
{
|
||
if (pedantic)
|
||
cp_pedwarn("ANSI C++ forbids the use of `%s' on explicit instantiations",
|
||
IDENTIFIER_POINTER (storage));
|
||
|
||
if (storage == ridpointers[(int) RID_INLINE])
|
||
nomem_p = 1;
|
||
else if (storage == ridpointers[(int) RID_EXTERN])
|
||
extern_p = 1;
|
||
else if (storage == ridpointers[(int) RID_STATIC])
|
||
static_p = 1;
|
||
else
|
||
{
|
||
cp_error ("storage class `%D' applied to template instantiation",
|
||
storage);
|
||
extern_p = 0;
|
||
}
|
||
}
|
||
|
||
if (CLASSTYPE_TEMPLATE_SPECIALIZATION (t))
|
||
{
|
||
/* [temp.spec]
|
||
|
||
No program shall both explicitly instantiate and explicitly
|
||
specialize a template. */
|
||
cp_error ("explicit instantiation of `%#T' after", t);
|
||
cp_error_at ("explicit specialization here", t);
|
||
return;
|
||
}
|
||
else if (CLASSTYPE_EXPLICIT_INSTANTIATION (t))
|
||
{
|
||
/* [temp.spec]
|
||
|
||
No program shall explicitly instantiate any template more
|
||
than once.
|
||
|
||
If CLASSTYPE_INTERFACE_ONLY, then the first explicit
|
||
instantiation was `extern', and if EXTERN_P then the second
|
||
is. Both cases are OK. */
|
||
if (!CLASSTYPE_INTERFACE_ONLY (t) && !extern_p && !flag_use_repository)
|
||
cp_pedwarn ("duplicate explicit instantiation of `%#T'", t);
|
||
|
||
/* If we've already instantiated the template, just return now. */
|
||
if (!CLASSTYPE_INTERFACE_ONLY (t))
|
||
return;
|
||
}
|
||
|
||
mark_class_instantiated (t, extern_p);
|
||
repo_template_instantiated (t, extern_p);
|
||
|
||
if (nomem_p)
|
||
return;
|
||
|
||
{
|
||
tree tmp;
|
||
|
||
/* In contrast to implicit instantiation, where only the
|
||
declarations, and not the definitions, of members are
|
||
instantiated, we have here:
|
||
|
||
[temp.explicit]
|
||
|
||
The explicit instantiation of a class template specialization
|
||
implies the instantiation of all of its members not
|
||
previously explicitly specialized in the translation unit
|
||
containing the explicit instantiation.
|
||
|
||
Of course, we can't instantiate member template classes, since
|
||
we don't have any arguments for them. Note that the standard
|
||
is unclear on whether the instatiation of the members are
|
||
*explicit* instantiations or not. We choose to be generous,
|
||
and not set DECL_EXPLICIT_INSTANTIATION. Therefore, we allow
|
||
the explicit instantiation of a class where some of the members
|
||
have no definition in the current translation unit. */
|
||
|
||
if (! static_p)
|
||
for (tmp = TYPE_METHODS (t); tmp; tmp = TREE_CHAIN (tmp))
|
||
if (TREE_CODE (tmp) == FUNCTION_DECL
|
||
&& DECL_TEMPLATE_INSTANTIATION (tmp))
|
||
{
|
||
mark_decl_instantiated (tmp, extern_p);
|
||
repo_template_instantiated (tmp, extern_p);
|
||
if (! extern_p)
|
||
instantiate_decl (tmp);
|
||
}
|
||
|
||
for (tmp = TYPE_FIELDS (t); tmp; tmp = TREE_CHAIN (tmp))
|
||
if (TREE_CODE (tmp) == VAR_DECL && DECL_TEMPLATE_INSTANTIATION (tmp))
|
||
{
|
||
mark_decl_instantiated (tmp, extern_p);
|
||
repo_template_instantiated (tmp, extern_p);
|
||
if (! extern_p)
|
||
instantiate_decl (tmp);
|
||
}
|
||
|
||
for (tmp = CLASSTYPE_TAGS (t); tmp; tmp = TREE_CHAIN (tmp))
|
||
if (IS_AGGR_TYPE (TREE_VALUE (tmp))
|
||
&& !uses_template_parms (CLASSTYPE_TI_ARGS (TREE_VALUE (tmp))))
|
||
do_type_instantiation (TYPE_MAIN_DECL (TREE_VALUE (tmp)), storage);
|
||
}
|
||
}
|
||
|
||
/* Given a function DECL, which is a specialization of TMPL, modify
|
||
DECL to be a re-instantiation of TMPL with the same template
|
||
arguments. TMPL should be the template into which tsubst'ing
|
||
should occur for DECL, not the most general template.
|
||
|
||
One reason for doing this is a scenario like this:
|
||
|
||
template <class T>
|
||
void f(const T&, int i);
|
||
|
||
void g() { f(3, 7); }
|
||
|
||
template <class T>
|
||
void f(const T& t, const int i) { }
|
||
|
||
Note that when the template is first instantiated, with
|
||
instantiate_template, the resulting DECL will have no name for the
|
||
first parameter, and the wrong type for the second. So, when we go
|
||
to instantiate the DECL, we regenerate it. */
|
||
|
||
static void
|
||
regenerate_decl_from_template (decl, tmpl)
|
||
tree decl;
|
||
tree tmpl;
|
||
{
|
||
tree args;
|
||
tree code_pattern;
|
||
tree new_decl;
|
||
tree gen_tmpl;
|
||
int unregistered;
|
||
|
||
args = DECL_TI_ARGS (decl);
|
||
code_pattern = DECL_TEMPLATE_RESULT (tmpl);
|
||
|
||
/* Unregister the specialization so that when we tsubst we will not
|
||
just return DECL. We don't have to unregister DECL from TMPL
|
||
because if would only be registered there if it were a partial
|
||
instantiation of a specialization, which it isn't: it's a full
|
||
instantiation. */
|
||
gen_tmpl = most_general_template (tmpl);
|
||
unregistered = unregister_specialization (decl, gen_tmpl);
|
||
|
||
/* If the DECL was not unregistered then something peculiar is
|
||
happening: we created a specialization but did not call
|
||
register_specialization for it. */
|
||
my_friendly_assert (unregistered, 0);
|
||
|
||
if (TREE_CODE (decl) == VAR_DECL)
|
||
/* Make sure that we can see identifiers, and compute access
|
||
correctly, for the class members used in the declaration of
|
||
this static variable. */
|
||
pushclass (DECL_CONTEXT (decl), 2);
|
||
|
||
/* Do the substitution to get the new declaration. */
|
||
new_decl = tsubst (code_pattern, args, /*complain=*/1, NULL_TREE);
|
||
|
||
if (TREE_CODE (decl) == VAR_DECL)
|
||
{
|
||
/* Set up DECL_INITIAL, since tsubst doesn't. */
|
||
DECL_INITIAL (new_decl) =
|
||
tsubst_expr (DECL_INITIAL (code_pattern), args,
|
||
/*complain=*/1, DECL_TI_TEMPLATE (decl));
|
||
/* Pop the class context we pushed above. */
|
||
popclass ();
|
||
}
|
||
else if (TREE_CODE (decl) == FUNCTION_DECL)
|
||
{
|
||
/* Convince duplicate_decls to use the DECL_ARGUMENTS from the
|
||
new decl. */
|
||
DECL_INITIAL (new_decl) = error_mark_node;
|
||
/* And don't complain about a duplicate definition. */
|
||
DECL_INITIAL (decl) = NULL_TREE;
|
||
}
|
||
|
||
/* The immediate parent of the new template is still whatever it was
|
||
before, even though tsubst sets DECL_TI_TEMPLATE up as the most
|
||
general template. We also reset the DECL_ASSEMBLER_NAME since
|
||
tsubst always calculates the name as if the function in question
|
||
were really a template instance, and sometimes, with friend
|
||
functions, this is not so. See tsubst_friend_function for
|
||
details. */
|
||
DECL_TI_TEMPLATE (new_decl) = DECL_TI_TEMPLATE (decl);
|
||
DECL_ASSEMBLER_NAME (new_decl) = DECL_ASSEMBLER_NAME (decl);
|
||
DECL_RTL (new_decl) = DECL_RTL (decl);
|
||
|
||
/* Call duplicate decls to merge the old and new declarations. */
|
||
duplicate_decls (new_decl, decl);
|
||
|
||
/* Now, re-register the specialization. */
|
||
register_specialization (decl, gen_tmpl, args);
|
||
}
|
||
|
||
/* Produce the definition of D, a _DECL generated from a template. */
|
||
|
||
tree
|
||
instantiate_decl (d)
|
||
tree d;
|
||
{
|
||
tree tmpl = DECL_TI_TEMPLATE (d);
|
||
tree args = DECL_TI_ARGS (d);
|
||
tree td;
|
||
tree code_pattern;
|
||
tree spec;
|
||
tree gen_tmpl;
|
||
int nested = in_function_p ();
|
||
int pattern_defined;
|
||
int line = lineno;
|
||
char *file = input_filename;
|
||
|
||
/* This function should only be used to instantiate templates for
|
||
functions and static member variables. */
|
||
my_friendly_assert (TREE_CODE (d) == FUNCTION_DECL
|
||
|| TREE_CODE (d) == VAR_DECL, 0);
|
||
|
||
if (DECL_TEMPLATE_INSTANTIATED (d))
|
||
/* D has already been instantiated. It might seem reasonable to
|
||
check whether or not D is an explict instantiation, and, if so,
|
||
stop here. But when an explicit instantiation is deferred
|
||
until the end of the compilation, DECL_EXPLICIT_INSTANTIATION
|
||
is set, even though we still need to do the instantiation. */
|
||
return d;
|
||
|
||
/* If we already have a specialization of this declaration, then
|
||
there's no reason to instantiate it. Note that
|
||
retrieve_specialization gives us both instantiations and
|
||
specializations, so we must explicitly check
|
||
DECL_TEMPLATE_SPECIALIZATION. */
|
||
gen_tmpl = most_general_template (tmpl);
|
||
spec = retrieve_specialization (gen_tmpl, args);
|
||
if (spec != NULL_TREE && DECL_TEMPLATE_SPECIALIZATION (spec))
|
||
return spec;
|
||
|
||
/* This needs to happen before any tsubsting. */
|
||
if (! push_tinst_level (d))
|
||
return d;
|
||
|
||
/* Set TD to the template whose DECL_TEMPLATE_RESULT is the pattern
|
||
for the instantiation. This is not always the most general
|
||
template. Consider, for example:
|
||
|
||
template <class T>
|
||
struct S { template <class U> void f();
|
||
template <> void f<int>(); };
|
||
|
||
and an instantiation of S<double>::f<int>. We want TD to be the
|
||
specialization S<T>::f<int>, not the more general S<T>::f<U>. */
|
||
td = tmpl;
|
||
for (td = tmpl;
|
||
/* An instantiation cannot have a definition, so we need a
|
||
more general template. */
|
||
DECL_TEMPLATE_INSTANTIATION (td)
|
||
/* We must also deal with friend templates. Given:
|
||
|
||
template <class T> struct S {
|
||
template <class U> friend void f() {};
|
||
};
|
||
|
||
S<int>::f<U> say, is not an instantiation of S<T>::f<U>,
|
||
so far as the language is concerned, but that's still
|
||
where we get the pattern for the instantiation from. On
|
||
ther hand, if the definition comes outside the class, say:
|
||
|
||
template <class T> struct S {
|
||
template <class U> friend void f();
|
||
};
|
||
template <class U> friend void f() {}
|
||
|
||
we don't need to look any further. That's what the check for
|
||
DECL_INITIAL is for. */
|
||
|| (TREE_CODE (d) == FUNCTION_DECL
|
||
&& DECL_FRIEND_PSEUDO_TEMPLATE_INSTANTIATION (td)
|
||
&& !DECL_INITIAL (DECL_TEMPLATE_RESULT (td)));
|
||
)
|
||
{
|
||
/* The present template, TD, should not be a definition. If it
|
||
were a definition, we should be using it! Note that we
|
||
cannot restructure the loop to just keep going until we find
|
||
a template with a definition, since that might go too far if
|
||
a specialization was declared, but not defined. */
|
||
my_friendly_assert (!(TREE_CODE (d) == VAR_DECL
|
||
&& !DECL_IN_AGGR_P (DECL_TEMPLATE_RESULT (td))),
|
||
0);
|
||
|
||
/* Fetch the more general template. */
|
||
td = DECL_TI_TEMPLATE (td);
|
||
}
|
||
|
||
code_pattern = DECL_TEMPLATE_RESULT (td);
|
||
|
||
if (TREE_CODE (d) == FUNCTION_DECL)
|
||
pattern_defined = (DECL_INITIAL (code_pattern) != NULL_TREE);
|
||
else
|
||
pattern_defined = ! DECL_IN_AGGR_P (code_pattern);
|
||
|
||
push_to_top_level ();
|
||
lineno = DECL_SOURCE_LINE (d);
|
||
input_filename = DECL_SOURCE_FILE (d);
|
||
|
||
if (pattern_defined)
|
||
{
|
||
repo_template_used (d);
|
||
|
||
if (flag_external_templates && ! DECL_INTERFACE_KNOWN (d))
|
||
{
|
||
if (flag_alt_external_templates)
|
||
{
|
||
if (interface_unknown)
|
||
warn_if_unknown_interface (d);
|
||
}
|
||
else if (DECL_INTERFACE_KNOWN (code_pattern))
|
||
{
|
||
DECL_INTERFACE_KNOWN (d) = 1;
|
||
DECL_NOT_REALLY_EXTERN (d) = ! DECL_EXTERNAL (code_pattern);
|
||
}
|
||
else
|
||
warn_if_unknown_interface (code_pattern);
|
||
}
|
||
|
||
if (at_eof)
|
||
import_export_decl (d);
|
||
}
|
||
|
||
/* Reject all external templates except inline functions. */
|
||
if (DECL_INTERFACE_KNOWN (d)
|
||
&& ! DECL_NOT_REALLY_EXTERN (d)
|
||
&& ! (TREE_CODE (d) == FUNCTION_DECL && DECL_INLINE (d)))
|
||
goto out;
|
||
|
||
if (TREE_CODE (d) == VAR_DECL
|
||
&& TREE_READONLY (d)
|
||
&& DECL_INITIAL (d) == NULL_TREE
|
||
&& DECL_INITIAL (code_pattern) != NULL_TREE)
|
||
/* We need to set up DECL_INITIAL regardless of pattern_defined if
|
||
the variable is a static const initialized in the class body. */;
|
||
else if (! pattern_defined
|
||
|| (! (TREE_CODE (d) == FUNCTION_DECL && DECL_INLINE (d) && nested)
|
||
&& ! at_eof))
|
||
{
|
||
/* Defer all templates except inline functions used in another
|
||
function. We restore the source position here because it's used
|
||
by add_pending_template. */
|
||
lineno = line;
|
||
input_filename = file;
|
||
|
||
if (at_eof && !pattern_defined
|
||
&& DECL_EXPLICIT_INSTANTIATION (d))
|
||
/* [temp.explicit]
|
||
|
||
The definition of a non-exported function template, a
|
||
non-exported member function template, or a non-exported
|
||
member function or static data member of a class template
|
||
shall be present in every translation unit in which it is
|
||
explicitly instantiated. */
|
||
cp_error ("explicit instantiation of `%D' but no definition available",
|
||
d);
|
||
|
||
add_pending_template (d);
|
||
goto out;
|
||
}
|
||
|
||
/* We're now committed to instantiating this template. Mark it as
|
||
instantiated so that recursive calls to instantiate_decl do not
|
||
try to instantiate it again. */
|
||
DECL_TEMPLATE_INSTANTIATED (d) = 1;
|
||
|
||
/* Regenerate the declaration in case the template has been modified
|
||
by a subsequent redeclaration. */
|
||
regenerate_decl_from_template (d, td);
|
||
|
||
/* We already set the file and line above. Reset them now in case
|
||
they changed as a result of calling regenerate_decl_from_template. */
|
||
lineno = DECL_SOURCE_LINE (d);
|
||
input_filename = DECL_SOURCE_FILE (d);
|
||
|
||
if (TREE_CODE (d) == VAR_DECL)
|
||
{
|
||
DECL_IN_AGGR_P (d) = 0;
|
||
if (DECL_INTERFACE_KNOWN (d))
|
||
DECL_EXTERNAL (d) = ! DECL_NOT_REALLY_EXTERN (d);
|
||
else
|
||
{
|
||
DECL_EXTERNAL (d) = 1;
|
||
DECL_NOT_REALLY_EXTERN (d) = 1;
|
||
}
|
||
cp_finish_decl (d, DECL_INITIAL (d), NULL_TREE, 0, 0);
|
||
}
|
||
else if (TREE_CODE (d) == FUNCTION_DECL)
|
||
{
|
||
tree t = DECL_SAVED_TREE (code_pattern);
|
||
|
||
start_function (NULL_TREE, d, NULL_TREE, 1);
|
||
store_parm_decls ();
|
||
|
||
if (t && TREE_CODE (t) == RETURN_INIT)
|
||
{
|
||
store_return_init
|
||
(TREE_OPERAND (t, 0),
|
||
tsubst_expr (TREE_OPERAND (t, 1), args, /*complain=*/1, tmpl));
|
||
t = TREE_CHAIN (t);
|
||
}
|
||
|
||
if (t && TREE_CODE (t) == CTOR_INITIALIZER)
|
||
{
|
||
current_member_init_list
|
||
= tsubst_expr_values (TREE_OPERAND (t, 0), args);
|
||
current_base_init_list
|
||
= tsubst_expr_values (TREE_OPERAND (t, 1), args);
|
||
t = TREE_CHAIN (t);
|
||
}
|
||
|
||
setup_vtbl_ptr ();
|
||
/* Always keep the BLOCK node associated with the outermost
|
||
pair of curly braces of a function. These are needed
|
||
for correct operation of dwarfout.c. */
|
||
keep_next_level ();
|
||
|
||
my_friendly_assert (TREE_CODE (t) == COMPOUND_STMT, 42);
|
||
tsubst_expr (t, args, /*complain=*/1, tmpl);
|
||
|
||
finish_function (lineno, 0, nested);
|
||
}
|
||
|
||
out:
|
||
lineno = line;
|
||
input_filename = file;
|
||
|
||
pop_from_top_level ();
|
||
pop_tinst_level ();
|
||
|
||
return d;
|
||
}
|
||
|
||
/* Run through the list of templates that we wish we could
|
||
instantiate, and instantiate any we can. */
|
||
|
||
int
|
||
instantiate_pending_templates ()
|
||
{
|
||
tree *t;
|
||
int instantiated_something = 0;
|
||
int reconsider;
|
||
|
||
do
|
||
{
|
||
reconsider = 0;
|
||
|
||
t = &pending_templates;
|
||
while (*t)
|
||
{
|
||
tree srcloc = TREE_PURPOSE (*t);
|
||
tree instantiation = TREE_VALUE (*t);
|
||
|
||
input_filename = SRCLOC_FILE (srcloc);
|
||
lineno = SRCLOC_LINE (srcloc);
|
||
|
||
if (TREE_CODE_CLASS (TREE_CODE (instantiation)) == 't')
|
||
{
|
||
tree fn;
|
||
|
||
if (!TYPE_SIZE (instantiation))
|
||
{
|
||
instantiate_class_template (instantiation);
|
||
if (CLASSTYPE_TEMPLATE_INSTANTIATION (instantiation))
|
||
for (fn = TYPE_METHODS (instantiation);
|
||
fn;
|
||
fn = TREE_CHAIN (fn))
|
||
if (! DECL_ARTIFICIAL (fn))
|
||
instantiate_decl (fn);
|
||
if (TYPE_SIZE (instantiation))
|
||
{
|
||
instantiated_something = 1;
|
||
reconsider = 1;
|
||
}
|
||
}
|
||
|
||
if (TYPE_SIZE (instantiation))
|
||
/* If INSTANTIATION has been instantiated, then we don't
|
||
need to consider it again in the future. */
|
||
*t = TREE_CHAIN (*t);
|
||
else
|
||
t = &TREE_CHAIN (*t);
|
||
}
|
||
else
|
||
{
|
||
if (DECL_TEMPLATE_INSTANTIATION (instantiation)
|
||
&& !DECL_TEMPLATE_INSTANTIATED (instantiation))
|
||
{
|
||
instantiation = instantiate_decl (instantiation);
|
||
if (DECL_TEMPLATE_INSTANTIATED (instantiation))
|
||
{
|
||
instantiated_something = 1;
|
||
reconsider = 1;
|
||
}
|
||
}
|
||
|
||
if (!DECL_TEMPLATE_INSTANTIATION (instantiation)
|
||
|| DECL_TEMPLATE_INSTANTIATED (instantiation))
|
||
/* If INSTANTIATION has been instantiated, then we don't
|
||
need to consider it again in the future. */
|
||
*t = TREE_CHAIN (*t);
|
||
else
|
||
t = &TREE_CHAIN (*t);
|
||
}
|
||
}
|
||
template_tail = t;
|
||
|
||
/* Go through the things that are template instantiations if we are
|
||
using guiding declarations. */
|
||
t = &maybe_templates;
|
||
while (*t)
|
||
{
|
||
tree template;
|
||
tree fn;
|
||
tree args;
|
||
|
||
fn = TREE_VALUE (*t);
|
||
|
||
if (DECL_INITIAL (fn))
|
||
/* If the FN is already defined, then it was either already
|
||
instantiated or, even though guiding declarations were
|
||
allowed, a non-template definition was provided. */
|
||
;
|
||
else
|
||
{
|
||
template = TREE_PURPOSE (*t);
|
||
args = get_bindings (template, fn, NULL_TREE);
|
||
fn = instantiate_template (template, args);
|
||
instantiate_decl (fn);
|
||
reconsider = 1;
|
||
}
|
||
|
||
/* Remove this entry from the chain. */
|
||
*t = TREE_CHAIN (*t);
|
||
}
|
||
maybe_template_tail = t;
|
||
}
|
||
while (reconsider);
|
||
|
||
return instantiated_something;
|
||
}
|
||
|
||
/* Substitute ARGVEC into T, which is a TREE_LIST. In particular, it
|
||
is an initializer list: the TREE_PURPOSEs are DECLs, and the
|
||
TREE_VALUEs are initializer values. Used by instantiate_decl. */
|
||
|
||
static tree
|
||
tsubst_expr_values (t, argvec)
|
||
tree t, argvec;
|
||
{
|
||
tree first = NULL_TREE;
|
||
tree *p = &first;
|
||
|
||
for (; t; t = TREE_CHAIN (t))
|
||
{
|
||
tree pur = tsubst_copy (TREE_PURPOSE (t), argvec,
|
||
/*complain=*/1, NULL_TREE);
|
||
tree val = tsubst_expr (TREE_VALUE (t), argvec, /*complain=*/1,
|
||
NULL_TREE);
|
||
*p = build_tree_list (pur, val);
|
||
p = &TREE_CHAIN (*p);
|
||
}
|
||
return first;
|
||
}
|
||
|
||
tree last_tree;
|
||
|
||
void
|
||
add_tree (t)
|
||
tree t;
|
||
{
|
||
last_tree = TREE_CHAIN (last_tree) = t;
|
||
}
|
||
|
||
|
||
void
|
||
begin_tree ()
|
||
{
|
||
saved_trees = tree_cons (NULL_TREE, last_tree, saved_trees);
|
||
last_tree = NULL_TREE;
|
||
}
|
||
|
||
|
||
void
|
||
end_tree ()
|
||
{
|
||
my_friendly_assert (saved_trees != NULL_TREE, 0);
|
||
|
||
last_tree = TREE_VALUE (saved_trees);
|
||
saved_trees = TREE_CHAIN (saved_trees);
|
||
}
|
||
|
||
/* D is an undefined function declaration in the presence of templates with
|
||
the same name, listed in FNS. If one of them can produce D as an
|
||
instantiation, remember this so we can instantiate it at EOF if D has
|
||
not been defined by that time. */
|
||
|
||
void
|
||
add_maybe_template (d, fns)
|
||
tree d, fns;
|
||
{
|
||
tree t;
|
||
|
||
if (DECL_MAYBE_TEMPLATE (d))
|
||
return;
|
||
|
||
t = most_specialized (fns, d, NULL_TREE);
|
||
if (! t)
|
||
return;
|
||
if (t == error_mark_node)
|
||
{
|
||
cp_error ("ambiguous template instantiation for `%D'", d);
|
||
return;
|
||
}
|
||
|
||
*maybe_template_tail = perm_tree_cons (t, d, NULL_TREE);
|
||
maybe_template_tail = &TREE_CHAIN (*maybe_template_tail);
|
||
DECL_MAYBE_TEMPLATE (d) = 1;
|
||
}
|
||
|
||
/* Set CURRENT_ACCESS_SPECIFIER based on the protection of DECL. */
|
||
|
||
static void
|
||
set_current_access_from_decl (decl)
|
||
tree decl;
|
||
{
|
||
if (TREE_PRIVATE (decl))
|
||
current_access_specifier = access_private_node;
|
||
else if (TREE_PROTECTED (decl))
|
||
current_access_specifier = access_protected_node;
|
||
else
|
||
current_access_specifier = access_public_node;
|
||
}
|
||
|
||
/* Instantiate an enumerated type. TAG is the template type, NEWTAG
|
||
is the instantiation (which should have been created with
|
||
start_enum) and ARGS are the template arguments to use. */
|
||
|
||
static void
|
||
tsubst_enum (tag, newtag, args)
|
||
tree tag;
|
||
tree newtag;
|
||
tree args;
|
||
{
|
||
tree e;
|
||
|
||
for (e = TYPE_VALUES (tag); e; e = TREE_CHAIN (e))
|
||
{
|
||
tree value;
|
||
tree elt;
|
||
|
||
/* Note that in a template enum, the TREE_VALUE is the
|
||
CONST_DECL, not the corresponding INTEGER_CST. */
|
||
value = tsubst_expr (DECL_INITIAL (TREE_VALUE (e)),
|
||
args, /*complain=*/1,
|
||
NULL_TREE);
|
||
|
||
/* Give this enumeration constant the correct access. */
|
||
set_current_access_from_decl (TREE_VALUE (e));
|
||
|
||
/* Actually build the enumerator itself. */
|
||
elt = build_enumerator (TREE_PURPOSE (e), value, newtag);
|
||
|
||
/* We save the enumerators we have built so far in the
|
||
TYPE_VALUES so that if the enumeration constants for
|
||
subsequent enumerators involve those for previous ones,
|
||
tsubst_copy will be able to find them. */
|
||
TREE_CHAIN (elt) = TYPE_VALUES (newtag);
|
||
TYPE_VALUES (newtag) = elt;
|
||
}
|
||
|
||
finish_enum (newtag);
|
||
}
|
||
|
||
/* Set the DECL_ASSEMBLER_NAME for DECL, which is a FUNCTION_DECL that
|
||
is either an instantiation or specialization of a template
|
||
function. */
|
||
|
||
static void
|
||
set_mangled_name_for_template_decl (decl)
|
||
tree decl;
|
||
{
|
||
tree saved_namespace;
|
||
tree context = NULL_TREE;
|
||
tree fn_type;
|
||
tree ret_type;
|
||
tree parm_types;
|
||
tree tparms;
|
||
tree targs;
|
||
tree tmpl;
|
||
int parm_depth;
|
||
|
||
my_friendly_assert (TREE_CODE (decl) == FUNCTION_DECL, 0);
|
||
my_friendly_assert (DECL_TEMPLATE_INFO (decl) != NULL_TREE, 0);
|
||
|
||
/* The names of template functions must be mangled so as to indicate
|
||
what template is being specialized with what template arguments.
|
||
For example, each of the following three functions must get
|
||
different mangled names:
|
||
|
||
void f(int);
|
||
template <> void f<7>(int);
|
||
template <> void f<8>(int); */
|
||
|
||
targs = DECL_TI_ARGS (decl);
|
||
if (uses_template_parms (targs))
|
||
/* This DECL is for a partial instantiation. There's no need to
|
||
mangle the name of such an entity. */
|
||
return;
|
||
|
||
tmpl = most_general_template (DECL_TI_TEMPLATE (decl));
|
||
tparms = DECL_TEMPLATE_PARMS (tmpl);
|
||
parm_depth = TMPL_PARMS_DEPTH (tparms);
|
||
|
||
/* There should be as many levels of arguments as there are levels
|
||
of parameters. */
|
||
my_friendly_assert (parm_depth == TMPL_ARGS_DEPTH (targs), 0);
|
||
|
||
/* We now compute the PARMS and RET_TYPE to give to
|
||
build_decl_overload_real. The PARMS and RET_TYPE are the
|
||
parameter and return types of the template, after all but the
|
||
innermost template arguments have been substituted, not the
|
||
parameter and return types of the function DECL. For example,
|
||
given:
|
||
|
||
template <class T> T f(T);
|
||
|
||
both PARMS and RET_TYPE should be `T' even if DECL is `int f(int)'.
|
||
A more subtle example is:
|
||
|
||
template <class T> struct S { template <class U> void f(T, U); }
|
||
|
||
Here, if DECL is `void S<int>::f(int, double)', PARMS should be
|
||
{int, U}. Thus, the args that we want to subsitute into the
|
||
return and parameter type for the function are those in TARGS,
|
||
with the innermost level omitted. */
|
||
fn_type = TREE_TYPE (tmpl);
|
||
if (DECL_STATIC_FUNCTION_P (decl))
|
||
context = DECL_CLASS_CONTEXT (decl);
|
||
|
||
if (parm_depth == 1)
|
||
/* No substitution is necessary. */
|
||
;
|
||
else
|
||
{
|
||
int i;
|
||
tree partial_args;
|
||
|
||
/* Replace the innermost level of the TARGS with NULL_TREEs to
|
||
let tsubst know not to subsitute for those parameters. */
|
||
partial_args = make_temp_vec (TREE_VEC_LENGTH (targs));
|
||
for (i = 1; i < TMPL_ARGS_DEPTH (targs); ++i)
|
||
SET_TMPL_ARGS_LEVEL (partial_args, i,
|
||
TMPL_ARGS_LEVEL (targs, i));
|
||
SET_TMPL_ARGS_LEVEL (partial_args,
|
||
TMPL_ARGS_DEPTH (targs),
|
||
make_temp_vec (DECL_NTPARMS (tmpl)));
|
||
|
||
/* Now, do the (partial) substitution to figure out the
|
||
appropriate function type. */
|
||
fn_type = tsubst (fn_type, partial_args, /*complain=*/1, NULL_TREE);
|
||
if (DECL_STATIC_FUNCTION_P (decl))
|
||
context = tsubst (context, partial_args, /*complain=*/1, NULL_TREE);
|
||
|
||
/* Substitute into the template parameters to obtain the real
|
||
innermost set of parameters. This step is important if the
|
||
innermost set of template parameters contains value
|
||
parameters whose types depend on outer template parameters. */
|
||
TREE_VEC_LENGTH (partial_args)--;
|
||
tparms = tsubst_template_parms (tparms, partial_args, /*complain=*/1);
|
||
}
|
||
|
||
/* Now, get the innermost parameters and arguments, and figure out
|
||
the parameter and return types. */
|
||
tparms = INNERMOST_TEMPLATE_PARMS (tparms);
|
||
targs = innermost_args (targs);
|
||
ret_type = TREE_TYPE (fn_type);
|
||
parm_types = TYPE_ARG_TYPES (fn_type);
|
||
|
||
/* For a static member function, we generate a fake `this' pointer,
|
||
for the purposes of mangling. This indicates of which class the
|
||
function is a member. Because of:
|
||
|
||
[class.static]
|
||
|
||
There shall not be a static and a nonstatic member function
|
||
with the same name and the same parameter types
|
||
|
||
we don't have to worry that this will result in a clash with a
|
||
non-static member function. */
|
||
if (DECL_STATIC_FUNCTION_P (decl))
|
||
parm_types = hash_tree_chain (build_pointer_type (context), parm_types);
|
||
|
||
/* There should be the same number of template parameters as
|
||
template arguments. */
|
||
my_friendly_assert (TREE_VEC_LENGTH (tparms) == TREE_VEC_LENGTH (targs),
|
||
0);
|
||
|
||
/* If the template is in a namespace, we need to put that into the
|
||
mangled name. Unfortunately, build_decl_overload_real does not
|
||
get the decl to mangle, so it relies on the current
|
||
namespace. Therefore, we set that here temporarily. */
|
||
my_friendly_assert (TREE_CODE_CLASS (TREE_CODE (decl)) == 'd', 980702);
|
||
saved_namespace = current_namespace;
|
||
current_namespace = CP_DECL_CONTEXT (decl);
|
||
|
||
/* Actually set the DCL_ASSEMBLER_NAME. */
|
||
DECL_ASSEMBLER_NAME (decl)
|
||
= build_decl_overload_real (DECL_NAME (decl), parm_types, ret_type,
|
||
tparms, targs,
|
||
DECL_FUNCTION_MEMBER_P (decl)
|
||
+ DECL_CONSTRUCTOR_P (decl));
|
||
|
||
/* Restore the previously active namespace. */
|
||
current_namespace = saved_namespace;
|
||
}
|