mirror of
https://github.com/pmret/gcc-papermario.git
synced 2024-11-08 20:02:47 +01:00
4751 lines
124 KiB
C
4751 lines
124 KiB
C
|
/* Handle parameterized types (templates) for GNU C++.
|
|||
|
Copyright (C) 1992, 93, 94, 95, 96, 1997 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 <stdio.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"
|
|||
|
|
|||
|
#ifdef HAVE_STDLIB_H
|
|||
|
#include <stdlib.h>
|
|||
|
#endif
|
|||
|
|
|||
|
extern struct obstack permanent_obstack;
|
|||
|
|
|||
|
extern int lineno;
|
|||
|
extern char *input_filename;
|
|||
|
struct pending_inline *pending_template_expansions;
|
|||
|
|
|||
|
tree current_template_parms;
|
|||
|
HOST_WIDE_INT processing_template_decl;
|
|||
|
|
|||
|
tree pending_templates;
|
|||
|
static tree *template_tail = &pending_templates;
|
|||
|
|
|||
|
tree maybe_templates;
|
|||
|
static tree *maybe_template_tail = &maybe_templates;
|
|||
|
|
|||
|
int minimal_parse_mode;
|
|||
|
|
|||
|
int processing_specialization;
|
|||
|
static int template_header_count;
|
|||
|
|
|||
|
#define obstack_chunk_alloc xmalloc
|
|||
|
#define obstack_chunk_free free
|
|||
|
|
|||
|
static int unify PROTO((tree, tree *, int, tree, tree, int *, 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 comp_template_args 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));
|
|||
|
static tree tsubst_enum PROTO((tree, tree, int, tree *));
|
|||
|
static tree add_to_template_args PROTO((tree, tree));
|
|||
|
static int type_unification_real PROTO((tree, tree *, tree, tree, int*,
|
|||
|
int, int, int));
|
|||
|
static int processing_explicit_specialization PROTO((int));
|
|||
|
static void note_template_header PROTO((int));
|
|||
|
|
|||
|
/* Restore the template parameter context. */
|
|||
|
|
|||
|
void
|
|||
|
begin_member_template_processing (decl)
|
|||
|
tree decl;
|
|||
|
{
|
|||
|
tree parms;
|
|||
|
int i;
|
|||
|
|
|||
|
parms = DECL_INNERMOST_TEMPLATE_PARMS (DECL_TI_TEMPLATE (decl));
|
|||
|
|
|||
|
++processing_template_decl;
|
|||
|
current_template_parms
|
|||
|
= tree_cons (build_int_2 (0, processing_template_decl),
|
|||
|
parms, current_template_parms);
|
|||
|
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:
|
|||
|
pushdecl (parm);
|
|||
|
break;
|
|||
|
|
|||
|
case PARM_DECL:
|
|||
|
{
|
|||
|
/* Make a CONST_DECL as is done in process_template_parm. */
|
|||
|
tree decl = build_decl (CONST_DECL, DECL_NAME (parm),
|
|||
|
TREE_TYPE (parm));
|
|||
|
DECL_INITIAL (decl) = DECL_INITIAL (parm);
|
|||
|
pushdecl (decl);
|
|||
|
}
|
|||
|
break;
|
|||
|
|
|||
|
default:
|
|||
|
my_friendly_abort (0);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/* Undo the effects of begin_member_template_processing. */
|
|||
|
|
|||
|
void
|
|||
|
end_member_template_processing ()
|
|||
|
{
|
|||
|
if (! processing_template_decl)
|
|||
|
return;
|
|||
|
|
|||
|
--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. Works if T
|
|||
|
is either a FUNCTION_DECL or a TEMPLATE_DECL. */
|
|||
|
|
|||
|
int
|
|||
|
is_member_template (t)
|
|||
|
tree t;
|
|||
|
{
|
|||
|
int r = 0;
|
|||
|
|
|||
|
if (TREE_CODE (t) != FUNCTION_DECL
|
|||
|
&& !DECL_FUNCTION_TEMPLATE_P (t))
|
|||
|
/* Anything that isn't a template or a template function is
|
|||
|
certainly not a member template. */
|
|||
|
return 0;
|
|||
|
|
|||
|
if ((DECL_FUNCTION_MEMBER_P (t)
|
|||
|
&& !DECL_TEMPLATE_SPECIALIZATION (t))
|
|||
|
|| (TREE_CODE (t) == TEMPLATE_DECL &&
|
|||
|
DECL_FUNCTION_MEMBER_P (DECL_TEMPLATE_RESULT (t))))
|
|||
|
{
|
|||
|
tree tmpl = NULL_TREE;
|
|||
|
|
|||
|
if (DECL_FUNCTION_TEMPLATE_P (t))
|
|||
|
tmpl = t;
|
|||
|
else if (DECL_TEMPLATE_INFO (t)
|
|||
|
&& DECL_FUNCTION_TEMPLATE_P (DECL_TI_TEMPLATE (t)))
|
|||
|
tmpl = DECL_TI_TEMPLATE (t);
|
|||
|
|
|||
|
if (tmpl)
|
|||
|
{
|
|||
|
tree parms = DECL_TEMPLATE_PARMS (tmpl);
|
|||
|
int parm_levels = list_length (parms);
|
|||
|
int template_class_levels = 0;
|
|||
|
tree ctx = DECL_CLASS_CONTEXT (t);
|
|||
|
|
|||
|
if (CLASSTYPE_TEMPLATE_INFO (ctx))
|
|||
|
{
|
|||
|
tree args;
|
|||
|
|
|||
|
/* Here, we should really count the number of levels
|
|||
|
deep ctx is, making sure not to count any levels that
|
|||
|
are just specializations. Since there are no member
|
|||
|
template classes yet, we don't have to do all that. */
|
|||
|
|
|||
|
if (!CLASSTYPE_TEMPLATE_SPECIALIZATION (ctx))
|
|||
|
template_class_levels = 1;
|
|||
|
else
|
|||
|
{
|
|||
|
int i;
|
|||
|
|
|||
|
args = CLASSTYPE_TI_ARGS (ctx);
|
|||
|
|
|||
|
if (args == NULL_TREE)
|
|||
|
template_class_levels = 1;
|
|||
|
else
|
|||
|
for (i = 0; i < TREE_VEC_LENGTH (args); ++i)
|
|||
|
if (uses_template_parms (TREE_VEC_ELT (args, i)))
|
|||
|
{
|
|||
|
template_class_levels++;
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (parm_levels > template_class_levels)
|
|||
|
r = 1;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return r;
|
|||
|
}
|
|||
|
|
|||
|
/* Return a new template argument vector which contains all of ARGS,
|
|||
|
but has as its innermost set of arguments the EXTRA_ARGS. */
|
|||
|
|
|||
|
static tree
|
|||
|
add_to_template_args (args, extra_args)
|
|||
|
tree args;
|
|||
|
tree extra_args;
|
|||
|
{
|
|||
|
tree new_args;
|
|||
|
|
|||
|
if (TREE_CODE (TREE_VEC_ELT (args, 0)) != TREE_VEC)
|
|||
|
{
|
|||
|
new_args = make_tree_vec (2);
|
|||
|
TREE_VEC_ELT (new_args, 0) = args;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
int i;
|
|||
|
|
|||
|
new_args = make_tree_vec (TREE_VEC_LENGTH (args) - 1);
|
|||
|
|
|||
|
for (i = 0; i < TREE_VEC_LENGTH (args); ++i)
|
|||
|
TREE_VEC_ELT (new_args, i) = TREE_VEC_ELT (args, i);
|
|||
|
}
|
|||
|
|
|||
|
TREE_VEC_ELT (new_args,
|
|||
|
TREE_VEC_LENGTH (new_args) - 1) = 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 ()
|
|||
|
{
|
|||
|
pushlevel (0);
|
|||
|
declare_pseudo_global_level ();
|
|||
|
++processing_template_decl;
|
|||
|
note_template_header (0);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* We've just seen template <>. */
|
|||
|
|
|||
|
void
|
|||
|
begin_specialization ()
|
|||
|
{
|
|||
|
note_template_header (1);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* Called at then end of processing a declaration preceded 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++;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* Returns non-zero iff a declarator, in which the number of template
|
|||
|
types that appeared was TEMPLATE_COUNT, is an explicit
|
|||
|
specialization. */
|
|||
|
|
|||
|
static int
|
|||
|
processing_explicit_specialization (template_count)
|
|||
|
int template_count;
|
|||
|
{
|
|||
|
/* A function declaration is an explicit specialization of a member
|
|||
|
template if all of the following conditions hold:
|
|||
|
|
|||
|
o There was a template <...> preceding the declaration.
|
|||
|
o The last template <...> was in fact template <>.
|
|||
|
o The number of template <...>'s preceding the declaration, less
|
|||
|
the number of template classes with arguments specified used to
|
|||
|
qualify the function name, is 1.
|
|||
|
|
|||
|
For example:
|
|||
|
|
|||
|
template <> void S<int>::foo();
|
|||
|
template <class T> template <> void S<T>::foo();
|
|||
|
template <> struct S<int> { ... template <> void foo(); }
|
|||
|
|
|||
|
The first of these is not a specialization of S<int>::foo() (it
|
|||
|
is instead a specialization of S<T>::foo), while the next two are
|
|||
|
specializations of member template functions. */
|
|||
|
|
|||
|
return processing_specialization
|
|||
|
&& template_header_count > template_count;
|
|||
|
}
|
|||
|
|
|||
|
/* Returns the template function specialized by TEMPLATE_ID, or
|
|||
|
NULL_TREE if there is none.
|
|||
|
|
|||
|
The TEMPLATE_ID is a TEMPLATE_ID_EXPR. The TYPE is
|
|||
|
the type it has been declared to have. Return the TEMPLATE_DECL
|
|||
|
that is being specialized, and put the specialization arguments in
|
|||
|
*TARGS. If no appropriate specialization can be found, NULL_TREE is
|
|||
|
returned, and *TARGS is assigned NULL_TREE. If complain is
|
|||
|
non-zero, error messages are printed where appropriate. */
|
|||
|
|
|||
|
tree
|
|||
|
determine_explicit_specialization (template_id, type, targs_out,
|
|||
|
need_member_template,
|
|||
|
complain)
|
|||
|
tree template_id;
|
|||
|
tree type;
|
|||
|
tree* targs_out;
|
|||
|
int need_member_template;
|
|||
|
int complain;
|
|||
|
{
|
|||
|
int i;
|
|||
|
int overloaded;
|
|||
|
tree fns;
|
|||
|
tree matching_fns = NULL_TREE;
|
|||
|
tree name = NULL_TREE;
|
|||
|
tree result;
|
|||
|
tree fn;
|
|||
|
|
|||
|
my_friendly_assert (TREE_CODE (template_id) == TEMPLATE_ID_EXPR,
|
|||
|
0);
|
|||
|
|
|||
|
fns = TREE_OPERAND (template_id, 0);
|
|||
|
|
|||
|
overloaded = fns != NULL_TREE && really_overloaded_fn (fns);
|
|||
|
|
|||
|
for (fn = (fns != NULL_TREE) ? get_first_fn (fns) : NULL_TREE;
|
|||
|
fn != NULL_TREE;
|
|||
|
fn = overloaded ? DECL_CHAIN (fn) : NULL_TREE)
|
|||
|
{
|
|||
|
int dummy = 0;
|
|||
|
tree targs;
|
|||
|
|
|||
|
if (name == NULL_TREE)
|
|||
|
name = DECL_NAME (fn);
|
|||
|
|
|||
|
if (TREE_CODE (fn) != TEMPLATE_DECL
|
|||
|
|| (need_member_template && !is_member_template (fn)))
|
|||
|
continue;
|
|||
|
|
|||
|
if (list_length (TREE_OPERAND (template_id, 1)) > DECL_NTPARMS (fn))
|
|||
|
continue;
|
|||
|
|
|||
|
targs = make_scratch_vec (DECL_NTPARMS (fn));
|
|||
|
|
|||
|
/* We allow incomplete unification here, because we are going to
|
|||
|
check all the functions. */
|
|||
|
i = type_unification (DECL_INNERMOST_TEMPLATE_PARMS (fn),
|
|||
|
&TREE_VEC_ELT (targs, 0),
|
|||
|
type
|
|||
|
? TYPE_ARG_TYPES (TREE_TYPE (fn)) : NULL_TREE,
|
|||
|
type ? TYPE_ARG_TYPES (type) : NULL_TREE,
|
|||
|
TREE_OPERAND (template_id, 1),
|
|||
|
&dummy, 1, 1);
|
|||
|
|
|||
|
if (i == 0)
|
|||
|
{
|
|||
|
/* Unification was successful. See if the return types
|
|||
|
match. */
|
|||
|
if (type != NULL_TREE)
|
|||
|
{
|
|||
|
tree tmpl_return_type = tsubst (TREE_TYPE (TREE_TYPE (fn)),
|
|||
|
targs,
|
|||
|
DECL_NTPARMS (fn),
|
|||
|
NULL_TREE);
|
|||
|
|
|||
|
if (tmpl_return_type != TREE_TYPE (type))
|
|||
|
{
|
|||
|
/* Always complain about this. With ambiguity, some
|
|||
|
other context, might resolve things. But, a
|
|||
|
non-matching return type will always be a
|
|||
|
problem. */
|
|||
|
cp_error ("Return type of explicit specialization of");
|
|||
|
cp_error ("`%D' is `%T', but should be `%T'.",
|
|||
|
fn, TREE_TYPE (type), tmpl_return_type);
|
|||
|
*targs_out = NULL_TREE;
|
|||
|
return NULL_TREE;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
matching_fns = scratch_tree_cons (fn, targs, matching_fns);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (matching_fns == NULL_TREE)
|
|||
|
{
|
|||
|
if (complain)
|
|||
|
cp_error ("Specialization of `%s' does not match any template declaration.",
|
|||
|
IDENTIFIER_POINTER (name));
|
|||
|
*targs_out = NULL_TREE;
|
|||
|
return NULL_TREE;
|
|||
|
}
|
|||
|
|
|||
|
if (TREE_CHAIN (matching_fns) != NULL_TREE)
|
|||
|
{
|
|||
|
if (complain)
|
|||
|
{
|
|||
|
tree fn;
|
|||
|
|
|||
|
cp_error ("Ambiguous explicit specialization. Candidates are:");
|
|||
|
for (fn = matching_fns; fn != NULL_TREE; fn = TREE_CHAIN (fn))
|
|||
|
cp_error (" %D", TREE_PURPOSE (fn));
|
|||
|
}
|
|||
|
|
|||
|
*targs_out = NULL_TREE;
|
|||
|
return NULL_TREE;
|
|||
|
}
|
|||
|
|
|||
|
/* We have one, and exactly one, match. */
|
|||
|
*targs_out = TREE_VALUE (matching_fns);
|
|||
|
return TREE_PURPOSE (matching_fns);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* Check to see if the function just declared, as indicated in
|
|||
|
DECLARATOR, and in DECL, is a specialization. Check that the
|
|||
|
specialization is OK. If FLAGS == 1, we are being called by
|
|||
|
finish_struct_methods. If FLAGS == 2, we are being called by
|
|||
|
grokfndecl, and the function has a definition, or is a friend. If
|
|||
|
FLAGS == 3, this is a friend declaration.
|
|||
|
Returns 0 if the decl is not an explicit specialization or
|
|||
|
instantiation, 1 if it is an explicit specialization, and 2 if it
|
|||
|
is an explicit instantiation. */
|
|||
|
|
|||
|
int
|
|||
|
check_explicit_specialization (declarator, decl, template_count, flags)
|
|||
|
tree declarator;
|
|||
|
tree decl;
|
|||
|
int template_count;
|
|||
|
int flags;
|
|||
|
{
|
|||
|
int finish_member = flags == 1;
|
|||
|
int have_def = flags == 2;
|
|||
|
int is_friend = flags == 3;
|
|||
|
|
|||
|
if (processing_explicit_specialization (template_count)
|
|||
|
|| finish_member
|
|||
|
|| TREE_CODE (declarator) == TEMPLATE_ID_EXPR)
|
|||
|
{
|
|||
|
tree tmpl = NULL_TREE;
|
|||
|
tree dname = DECL_NAME (decl);
|
|||
|
tree ctype = DECL_CLASS_CONTEXT (decl);
|
|||
|
tree targs;
|
|||
|
|
|||
|
/* We've come across a declarator that looks like: U f<T1,
|
|||
|
T2, ...>(A1, A2, ..). This is an explicit template
|
|||
|
specialization. Check that:
|
|||
|
|
|||
|
o The explicitly specified parameters together with those
|
|||
|
that can be deduced by template argument deduction
|
|||
|
uniquely determine a particular specialization.
|
|||
|
|
|||
|
See [temp.expl.spec]. */
|
|||
|
|
|||
|
if (!finish_member
|
|||
|
&& TREE_CODE (declarator) == TEMPLATE_ID_EXPR
|
|||
|
&& !processing_explicit_specialization (template_count)
|
|||
|
&& !is_friend)
|
|||
|
{
|
|||
|
if (!have_def)
|
|||
|
/* This is not an explicit specialization. It must be
|
|||
|
an explicit instantiation. */
|
|||
|
return 2;
|
|||
|
else if (pedantic)
|
|||
|
pedwarn ("Explicit specialization not preceded by `template <>'");
|
|||
|
}
|
|||
|
|
|||
|
if (TREE_CODE (declarator) != TEMPLATE_ID_EXPR)
|
|||
|
{
|
|||
|
tree fns;
|
|||
|
|
|||
|
my_friendly_assert (TREE_CODE (declarator) == IDENTIFIER_NODE,
|
|||
|
0);
|
|||
|
if (!ctype)
|
|||
|
fns = IDENTIFIER_GLOBAL_VALUE (dname);
|
|||
|
else
|
|||
|
fns = dname;
|
|||
|
|
|||
|
declarator = lookup_template_function (fns, NULL_TREE);
|
|||
|
}
|
|||
|
|
|||
|
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);
|
|||
|
SET_DECL_IMPLICIT_INSTANTIATION (decl);
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
if (ctype
|
|||
|
&& TREE_CODE (TREE_OPERAND (declarator, 0)) == IDENTIFIER_NODE)
|
|||
|
{
|
|||
|
tree fns;
|
|||
|
|
|||
|
if (TYPE_BEING_DEFINED (ctype) && !finish_member)
|
|||
|
{
|
|||
|
/* Since finish_struct_1 has not been called yet, we
|
|||
|
can't call lookup_fnfields. We note that this
|
|||
|
template is a specialization, and proceed, letting
|
|||
|
finish_struct_methods fix this up later. */
|
|||
|
SET_DECL_TEMPLATE_SPECIALIZATION (decl);
|
|||
|
DECL_TEMPLATE_INFO (decl)
|
|||
|
= perm_tree_cons (NULL_TREE,
|
|||
|
TREE_OPERAND (declarator, 1),
|
|||
|
NULL_TREE);
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
fns = lookup_fnfields (TYPE_BINFO (ctype),
|
|||
|
TREE_OPERAND (declarator, 0),
|
|||
|
1);
|
|||
|
|
|||
|
if (fns == NULL_TREE)
|
|||
|
{
|
|||
|
cp_error ("No member template `%s' declared in `%T'",
|
|||
|
IDENTIFIER_POINTER (TREE_OPERAND (declarator,
|
|||
|
0)),
|
|||
|
ctype);
|
|||
|
return 1;
|
|||
|
}
|
|||
|
else
|
|||
|
TREE_OPERAND (declarator, 0) = fns;
|
|||
|
}
|
|||
|
|
|||
|
tmpl =
|
|||
|
determine_explicit_specialization
|
|||
|
(declarator, TREE_TYPE (decl), &targs,
|
|||
|
TREE_CODE (decl) == TEMPLATE_DECL, 1);
|
|||
|
|
|||
|
if (tmpl)
|
|||
|
{
|
|||
|
/* Mangle the function name appropriately. */
|
|||
|
if (name_mangling_version >= 1)
|
|||
|
{
|
|||
|
tree arg_types = TYPE_ARG_TYPES (TREE_TYPE (tmpl));
|
|||
|
|
|||
|
if (ctype
|
|||
|
&& TREE_CODE (TREE_TYPE (tmpl)) == FUNCTION_TYPE)
|
|||
|
arg_types =
|
|||
|
hash_tree_chain (build_pointer_type (ctype),
|
|||
|
arg_types);
|
|||
|
|
|||
|
DECL_ASSEMBLER_NAME (decl)
|
|||
|
= build_template_decl_overload
|
|||
|
(DECL_NAME (decl),
|
|||
|
arg_types,
|
|||
|
TREE_TYPE (TREE_TYPE (tmpl)),
|
|||
|
DECL_INNERMOST_TEMPLATE_PARMS (tmpl),
|
|||
|
targs, ctype != NULL_TREE);
|
|||
|
}
|
|||
|
|
|||
|
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);
|
|||
|
DECL_TEMPLATE_INFO (decl)
|
|||
|
= perm_tree_cons (tmpl, targs, NULL_TREE);
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
/* This function declaration is a template specialization.
|
|||
|
Record that fact. */
|
|||
|
SET_DECL_TEMPLATE_SPECIALIZATION (decl);
|
|||
|
DECL_TEMPLATE_SPECIALIZATIONS (tmpl)
|
|||
|
= perm_tree_cons (targs, decl,
|
|||
|
DECL_TEMPLATE_SPECIALIZATIONS
|
|||
|
(tmpl));
|
|||
|
/* If DECL_TI_TEMPLATE (decl), the decl is an
|
|||
|
instantiation of a specialization of a member template.
|
|||
|
(In other words, there was a member template, in a
|
|||
|
class template. That member template was specialized.
|
|||
|
We then instantiated the class, so there is now an
|
|||
|
instance of that specialization.)
|
|||
|
|
|||
|
According to the CD2,
|
|||
|
|
|||
|
14.7.3.13 [tmpl.expl.spec]
|
|||
|
|
|||
|
A specialization of a member function template or
|
|||
|
member class template of a non-specialized class
|
|||
|
template is itself a template.
|
|||
|
|
|||
|
So, we just leave the template info alone in this case. */
|
|||
|
if (!(DECL_TEMPLATE_INFO (decl) && DECL_TI_TEMPLATE (decl)))
|
|||
|
DECL_TEMPLATE_INFO (decl)
|
|||
|
= perm_tree_cons (tmpl, targs, NULL_TREE);
|
|||
|
return 1;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|
|||
|
/* 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
|
|||
|
idx = TEMPLATE_CONST_IDX (DECL_INITIAL (p));
|
|||
|
++idx;
|
|||
|
}
|
|||
|
else
|
|||
|
idx = 0;
|
|||
|
|
|||
|
if (!is_type)
|
|||
|
{
|
|||
|
tree tinfo = 0;
|
|||
|
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);
|
|||
|
/* 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)
|
|||
|
{
|
|||
|
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));
|
|||
|
tinfo = make_node (TEMPLATE_CONST_PARM);
|
|||
|
my_friendly_assert (TREE_PERMANENT (tinfo), 260.5);
|
|||
|
if (TREE_PERMANENT (parm) == 0)
|
|||
|
{
|
|||
|
parm = copy_node (parm);
|
|||
|
TREE_PERMANENT (parm) = 1;
|
|||
|
}
|
|||
|
TREE_TYPE (tinfo) = TREE_TYPE (parm);
|
|||
|
decl = build_decl (CONST_DECL, DECL_NAME (parm), TREE_TYPE (parm));
|
|||
|
DECL_INITIAL (decl) = tinfo;
|
|||
|
DECL_INITIAL (parm) = tinfo;
|
|||
|
TEMPLATE_CONST_SET_INFO (tinfo, idx, processing_template_decl);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
tree t = make_lang_type (TEMPLATE_TYPE_PARM);
|
|||
|
CLASSTYPE_GOT_SEMICOLON (t) = 1;
|
|||
|
decl = build_decl (TYPE_DECL, TREE_VALUE (parm), t);
|
|||
|
TYPE_NAME (t) = decl;
|
|||
|
TYPE_STUB_DECL (t) = decl;
|
|||
|
parm = decl;
|
|||
|
TEMPLATE_TYPE_SET_INFO (t, idx, processing_template_decl);
|
|||
|
}
|
|||
|
SET_DECL_ARTIFICIAL (decl);
|
|||
|
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;
|
|||
|
|
|||
|
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? */
|
|||
|
}
|
|||
|
|
|||
|
/* Generate a valid set of template args from current_template_parms. */
|
|||
|
|
|||
|
tree
|
|||
|
current_template_args ()
|
|||
|
{
|
|||
|
tree header = current_template_parms;
|
|||
|
int length = list_length (header);
|
|||
|
tree args = make_tree_vec (length);
|
|||
|
int l = length;
|
|||
|
|
|||
|
while (header)
|
|||
|
{
|
|||
|
tree a = copy_node (TREE_VALUE (header));
|
|||
|
int i = TREE_VEC_LENGTH (a);
|
|||
|
TREE_TYPE (a) = NULL_TREE;
|
|||
|
while (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)
|
|||
|
t = TREE_TYPE (t);
|
|||
|
else
|
|||
|
t = DECL_INITIAL (t);
|
|||
|
}
|
|||
|
|
|||
|
TREE_VEC_ELT (a, i) = t;
|
|||
|
}
|
|||
|
TREE_VEC_ELT (args, --l) = a;
|
|||
|
header = TREE_CHAIN (header);
|
|||
|
}
|
|||
|
|
|||
|
return args;
|
|||
|
}
|
|||
|
|
|||
|
void
|
|||
|
push_template_decl (decl)
|
|||
|
tree decl;
|
|||
|
{
|
|||
|
tree tmpl;
|
|||
|
tree args = NULL_TREE;
|
|||
|
tree info;
|
|||
|
tree ctx = DECL_CONTEXT (decl) ? DECL_CONTEXT (decl) : current_class_type;
|
|||
|
int primary = 0;
|
|||
|
|
|||
|
/* Kludge! */
|
|||
|
if (TREE_CODE (decl) == FUNCTION_DECL && DECL_FRIEND_P (decl)
|
|||
|
&& DECL_CLASS_CONTEXT (decl))
|
|||
|
;
|
|||
|
/* Note that this template is a "primary template" */
|
|||
|
else if (! ctx || ! CLASSTYPE_TEMPLATE_INFO (ctx)
|
|||
|
/* || (processing_template_decl > CLASSTYPE_TEMPLATE_LEVEL (ctx)) */)
|
|||
|
primary = 1;
|
|||
|
|
|||
|
/* Partial specialization. */
|
|||
|
if (TREE_CODE (decl) == TYPE_DECL && DECL_ARTIFICIAL (decl)
|
|||
|
&& CLASSTYPE_TEMPLATE_SPECIALIZATION (TREE_TYPE (decl)))
|
|||
|
{
|
|||
|
tree type = TREE_TYPE (decl);
|
|||
|
tree maintmpl = CLASSTYPE_TI_TEMPLATE (type);
|
|||
|
tree mainargs = CLASSTYPE_TI_ARGS (type);
|
|||
|
tree spec = DECL_TEMPLATE_SPECIALIZATIONS (maintmpl);
|
|||
|
|
|||
|
for (; spec; spec = TREE_CHAIN (spec))
|
|||
|
{
|
|||
|
/* purpose: args to main template
|
|||
|
value: spec template */
|
|||
|
if (comp_template_args (TREE_PURPOSE (spec), mainargs))
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
DECL_TEMPLATE_SPECIALIZATIONS (maintmpl) = CLASSTYPE_TI_SPEC_INFO (type)
|
|||
|
= perm_tree_cons (mainargs, TREE_VALUE (current_template_parms),
|
|||
|
DECL_TEMPLATE_SPECIALIZATIONS (maintmpl));
|
|||
|
TREE_TYPE (DECL_TEMPLATE_SPECIALIZATIONS (maintmpl)) = type;
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
args = current_template_args ();
|
|||
|
|
|||
|
if (! ctx || TYPE_BEING_DEFINED (ctx))
|
|||
|
{
|
|||
|
tmpl = build_lang_decl (TEMPLATE_DECL, DECL_NAME (decl), NULL_TREE);
|
|||
|
DECL_TEMPLATE_PARMS (tmpl) = current_template_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);
|
|||
|
|
|||
|
if (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 t;
|
|||
|
tree a;
|
|||
|
|
|||
|
if (CLASSTYPE_TEMPLATE_INSTANTIATION (ctx))
|
|||
|
cp_error ("must specialize `%#T' before defining member `%#D'",
|
|||
|
ctx, decl);
|
|||
|
if (TREE_CODE (decl) == TYPE_DECL && DECL_ARTIFICIAL (decl))
|
|||
|
tmpl = CLASSTYPE_TI_TEMPLATE (TREE_TYPE (decl));
|
|||
|
else if (! DECL_TEMPLATE_INFO (decl))
|
|||
|
{
|
|||
|
cp_error ("template definition of non-template `%#D'", decl);
|
|||
|
return;
|
|||
|
}
|
|||
|
else
|
|||
|
tmpl = DECL_TI_TEMPLATE (decl);
|
|||
|
|
|||
|
if (is_member_template (tmpl))
|
|||
|
{
|
|||
|
a = TREE_VEC_ELT (args, TREE_VEC_LENGTH (args) - 1);
|
|||
|
t = DECL_INNERMOST_TEMPLATE_PARMS (DECL_TI_TEMPLATE (decl));
|
|||
|
if (TREE_VEC_LENGTH (t)
|
|||
|
!= TREE_VEC_LENGTH (a))
|
|||
|
{
|
|||
|
cp_error ("got %d template parameters for `%#D'",
|
|||
|
TREE_VEC_LENGTH (a), decl);
|
|||
|
cp_error (" but %d required", TREE_VEC_LENGTH (t));
|
|||
|
}
|
|||
|
if (TREE_VEC_LENGTH (args) > 1)
|
|||
|
/* Get the template parameters for the enclosing template
|
|||
|
class. */
|
|||
|
a = TREE_VEC_ELT (args, TREE_VEC_LENGTH (args) - 2);
|
|||
|
else
|
|||
|
a = NULL_TREE;
|
|||
|
}
|
|||
|
else
|
|||
|
a = TREE_VEC_ELT (args, TREE_VEC_LENGTH (args) - 1);
|
|||
|
|
|||
|
t = NULL_TREE;
|
|||
|
|
|||
|
if (CLASSTYPE_TEMPLATE_SPECIALIZATION (ctx))
|
|||
|
{
|
|||
|
/* When processing an inline member template of a
|
|||
|
specialized class, there is no CLASSTYPE_TI_SPEC_INFO. */
|
|||
|
if (CLASSTYPE_TI_SPEC_INFO (ctx))
|
|||
|
t = TREE_VALUE (CLASSTYPE_TI_SPEC_INFO (ctx));
|
|||
|
}
|
|||
|
else if (CLASSTYPE_TEMPLATE_INFO (ctx))
|
|||
|
t = DECL_INNERMOST_TEMPLATE_PARMS (CLASSTYPE_TI_TEMPLATE (ctx));
|
|||
|
|
|||
|
/* There should be template arguments if and only if there is a
|
|||
|
template class. */
|
|||
|
my_friendly_assert((a != NULL_TREE) == (t != NULL_TREE), 0);
|
|||
|
|
|||
|
if (t != NULL_TREE
|
|||
|
&& TREE_VEC_LENGTH (t) != TREE_VEC_LENGTH (a))
|
|||
|
{
|
|||
|
cp_error ("got %d template parameters for `%#D'",
|
|||
|
TREE_VEC_LENGTH (a), decl);
|
|||
|
cp_error (" but `%#T' has %d", ctx, TREE_VEC_LENGTH (t));
|
|||
|
}
|
|||
|
}
|
|||
|
/* Get the innermost set of template arguments. */
|
|||
|
args = TREE_VEC_ELT (args, TREE_VEC_LENGTH (args) - 1);
|
|||
|
|
|||
|
DECL_TEMPLATE_RESULT (tmpl) = decl;
|
|||
|
TREE_TYPE (tmpl) = TREE_TYPE (decl);
|
|||
|
|
|||
|
if (! ctx)
|
|||
|
tmpl = pushdecl_top_level (tmpl);
|
|||
|
|
|||
|
if (primary)
|
|||
|
TREE_TYPE (DECL_INNERMOST_TEMPLATE_PARMS (tmpl)) = tmpl;
|
|||
|
|
|||
|
info = perm_tree_cons (tmpl, args, NULL_TREE);
|
|||
|
|
|||
|
if (TREE_CODE (decl) == TYPE_DECL && DECL_ARTIFICIAL (decl))
|
|||
|
{
|
|||
|
CLASSTYPE_TEMPLATE_INFO (TREE_TYPE (tmpl)) = info;
|
|||
|
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;
|
|||
|
}
|
|||
|
|
|||
|
/* Convert all template arguments to their appropriate types, and return
|
|||
|
a vector containing the resulting values. If any error occurs, return
|
|||
|
error_mark_node. */
|
|||
|
|
|||
|
static tree
|
|||
|
coerce_template_parms (parms, arglist, in_decl)
|
|||
|
tree parms, arglist;
|
|||
|
tree in_decl;
|
|||
|
{
|
|||
|
int nparms, nargs, i, lost = 0;
|
|||
|
tree vec;
|
|||
|
|
|||
|
if (arglist == NULL_TREE)
|
|||
|
nargs = 0;
|
|||
|
else if (TREE_CODE (arglist) == TREE_VEC)
|
|||
|
nargs = TREE_VEC_LENGTH (arglist);
|
|||
|
else
|
|||
|
nargs = list_length (arglist);
|
|||
|
|
|||
|
nparms = TREE_VEC_LENGTH (parms);
|
|||
|
|
|||
|
if (nargs > nparms
|
|||
|
|| (nargs < nparms
|
|||
|
&& TREE_PURPOSE (TREE_VEC_ELT (parms, nargs)) == NULL_TREE))
|
|||
|
{
|
|||
|
error ("incorrect number of parameters (%d, should be %d)",
|
|||
|
nargs, nparms);
|
|||
|
if (in_decl)
|
|||
|
cp_error_at ("in template expansion for decl `%D'", in_decl);
|
|||
|
return error_mark_node;
|
|||
|
}
|
|||
|
|
|||
|
if (arglist && TREE_CODE (arglist) == TREE_VEC)
|
|||
|
vec = copy_node (arglist);
|
|||
|
else
|
|||
|
{
|
|||
|
vec = make_tree_vec (nparms);
|
|||
|
for (i = 0; i < nparms; i++)
|
|||
|
{
|
|||
|
tree arg;
|
|||
|
|
|||
|
if (arglist)
|
|||
|
{
|
|||
|
arg = arglist;
|
|||
|
arglist = TREE_CHAIN (arglist);
|
|||
|
|
|||
|
if (arg == error_mark_node)
|
|||
|
lost++;
|
|||
|
else
|
|||
|
arg = TREE_VALUE (arg);
|
|||
|
}
|
|||
|
else if (TREE_CODE (TREE_VALUE (TREE_VEC_ELT (parms, i)))
|
|||
|
== TYPE_DECL)
|
|||
|
arg = tsubst (TREE_PURPOSE (TREE_VEC_ELT (parms, i)),
|
|||
|
vec, i, in_decl);
|
|||
|
else
|
|||
|
arg = tsubst_expr (TREE_PURPOSE (TREE_VEC_ELT (parms, i)),
|
|||
|
vec, i, in_decl);
|
|||
|
|
|||
|
TREE_VEC_ELT (vec, i) = arg;
|
|||
|
}
|
|||
|
}
|
|||
|
for (i = 0; i < nparms; i++)
|
|||
|
{
|
|||
|
tree arg = TREE_VEC_ELT (vec, i);
|
|||
|
tree parm = TREE_VALUE (TREE_VEC_ELT (parms, i));
|
|||
|
tree val = 0;
|
|||
|
int is_type, requires_type;
|
|||
|
|
|||
|
is_type = TREE_CODE_CLASS (TREE_CODE (arg)) == 't';
|
|||
|
requires_type = TREE_CODE (parm) == TYPE_DECL;
|
|||
|
|
|||
|
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)
|
|||
|
{
|
|||
|
cp_error ("type/value mismatch at argument %d in template parameter list for `%D'",
|
|||
|
i, in_decl);
|
|||
|
if (is_type)
|
|||
|
cp_error (" expected a constant of type `%T', got `%T'",
|
|||
|
TREE_TYPE (parm), arg);
|
|||
|
else
|
|||
|
cp_error (" expected a type, got `%E'", arg);
|
|||
|
}
|
|||
|
lost++;
|
|||
|
TREE_VEC_ELT (vec, i) = error_mark_node;
|
|||
|
continue;
|
|||
|
}
|
|||
|
if (is_type)
|
|||
|
{
|
|||
|
val = groktypename (arg);
|
|||
|
if (! processing_template_decl)
|
|||
|
{
|
|||
|
tree t = target_type (val);
|
|||
|
if (TREE_CODE (t) != TYPENAME_TYPE
|
|||
|
&& IS_AGGR_TYPE (t)
|
|||
|
&& decl_function_context (TYPE_MAIN_DECL (t)))
|
|||
|
{
|
|||
|
cp_error ("type `%T' composed from a local class is not a valid template-argument", val);
|
|||
|
return error_mark_node;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
tree t = tsubst (TREE_TYPE (parm), vec,
|
|||
|
TREE_VEC_LENGTH (vec), in_decl);
|
|||
|
if (processing_template_decl)
|
|||
|
val = arg;
|
|||
|
else
|
|||
|
val = digest_init (t, arg, (tree *) 0);
|
|||
|
|
|||
|
if (val == error_mark_node || processing_template_decl)
|
|||
|
;
|
|||
|
|
|||
|
/* 14.2: Other template-arguments must be constant-expressions,
|
|||
|
addresses of objects or functions with external linkage, or of
|
|||
|
static class members. */
|
|||
|
else if (IS_AGGR_TYPE (TREE_TYPE (val)))
|
|||
|
{
|
|||
|
cp_error ("object `%E' cannot be used as template argument", arg);
|
|||
|
val = error_mark_node;
|
|||
|
}
|
|||
|
else if (!TREE_CONSTANT (val))
|
|||
|
{
|
|||
|
cp_error ("non-constant `%E' cannot be used as template argument",
|
|||
|
arg);
|
|||
|
val = error_mark_node;
|
|||
|
}
|
|||
|
else if (POINTER_TYPE_P (TREE_TYPE (val))
|
|||
|
&& ! integer_zerop (val)
|
|||
|
&& TREE_CODE (TREE_TYPE (TREE_TYPE (val))) != OFFSET_TYPE
|
|||
|
&& TREE_CODE (TREE_TYPE (TREE_TYPE (val))) != METHOD_TYPE)
|
|||
|
{
|
|||
|
t = val;
|
|||
|
STRIP_NOPS (t);
|
|||
|
if (TREE_CODE (t) == ADDR_EXPR)
|
|||
|
{
|
|||
|
tree a = TREE_OPERAND (t, 0);
|
|||
|
STRIP_NOPS (a);
|
|||
|
if (TREE_CODE (a) == STRING_CST)
|
|||
|
{
|
|||
|
cp_error ("string literal %E is not a valid template argument", a);
|
|||
|
error ("because it is the address of an object with static linkage");
|
|||
|
val = error_mark_node;
|
|||
|
}
|
|||
|
else if (TREE_CODE (a) != VAR_DECL
|
|||
|
&& TREE_CODE (a) != FUNCTION_DECL)
|
|||
|
goto bad;
|
|||
|
else if (! TREE_PUBLIC (a))
|
|||
|
{
|
|||
|
cp_error ("address of non-extern `%E' cannot be used as template argument", a);
|
|||
|
val = error_mark_node;
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
bad:
|
|||
|
cp_error ("`%E' is not a valid template argument", t);
|
|||
|
error ("it must be %s%s with external linkage",
|
|||
|
TREE_CODE (TREE_TYPE (val)) == POINTER_TYPE
|
|||
|
? "a pointer to " : "",
|
|||
|
TREE_CODE (TREE_TYPE (TREE_TYPE (val))) == FUNCTION_TYPE
|
|||
|
? "a function" : "an object");
|
|||
|
val = error_mark_node;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (val == error_mark_node)
|
|||
|
lost++;
|
|||
|
|
|||
|
TREE_VEC_ELT (vec, i) = val;
|
|||
|
}
|
|||
|
if (lost)
|
|||
|
return error_mark_node;
|
|||
|
return vec;
|
|||
|
}
|
|||
|
|
|||
|
static 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 (nt == ot)
|
|||
|
continue;
|
|||
|
if (TREE_CODE (nt) != TREE_CODE (ot))
|
|||
|
return 0;
|
|||
|
if (TREE_CODE (nt) == TREE_VEC)
|
|||
|
{
|
|||
|
/* For member templates */
|
|||
|
if (comp_template_args (nt, ot))
|
|||
|
continue;
|
|||
|
}
|
|||
|
else if (TREE_CODE_CLASS (TREE_CODE (ot)) == 't')
|
|||
|
{
|
|||
|
if (comptypes (ot, nt, 1))
|
|||
|
continue;
|
|||
|
}
|
|||
|
else if (cp_tree_equal (ot, nt) > 0)
|
|||
|
continue;
|
|||
|
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);
|
|||
|
|
|||
|
#if 0
|
|||
|
#define buflen sizeof(buf)
|
|||
|
#define check if (bufp >= buf+buflen-1) goto too_long
|
|||
|
#define ccat(c) *bufp++=(c); check
|
|||
|
#define advance bufp+=strlen(bufp); check
|
|||
|
#define cat(s) strncpy(bufp, s, buf+buflen-bufp-1); advance
|
|||
|
#else
|
|||
|
#define check
|
|||
|
#define ccat(c) obstack_1grow (&scratch_obstack, (c));
|
|||
|
#define advance
|
|||
|
#define cat(s) obstack_grow (&scratch_obstack, (s), strlen (s))
|
|||
|
#endif
|
|||
|
|
|||
|
cat (name);
|
|||
|
ccat ('<');
|
|||
|
nparms = TREE_VEC_LENGTH (parms);
|
|||
|
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 (arg, 0));
|
|||
|
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);
|
|||
|
|
|||
|
#if 0
|
|||
|
too_long:
|
|||
|
#endif
|
|||
|
fatal ("out of (preallocated) string space creating template instantiation name");
|
|||
|
/* NOTREACHED */
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
static tree
|
|||
|
classtype_mangled_name (t)
|
|||
|
tree t;
|
|||
|
{
|
|||
|
if (CLASSTYPE_TEMPLATE_INFO (t)
|
|||
|
&& PRIMARY_TEMPLATE_P (CLASSTYPE_TI_TEMPLATE (t)))
|
|||
|
{
|
|||
|
tree name = DECL_NAME (CLASSTYPE_TI_TEMPLATE (t));
|
|||
|
char *mangled_name = mangle_class_name_for_template
|
|||
|
(IDENTIFIER_POINTER (name),
|
|||
|
DECL_INNERMOST_TEMPLATE_PARMS (CLASSTYPE_TI_TEMPLATE (t)),
|
|||
|
CLASSTYPE_TI_ARGS (t));
|
|||
|
tree id = get_identifier (mangled_name);
|
|||
|
IDENTIFIER_TEMPLATE (id) = name;
|
|||
|
return id;
|
|||
|
}
|
|||
|
else
|
|||
|
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
|
|||
|
(current_function_decl, 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;
|
|||
|
{
|
|||
|
if (fns == NULL_TREE)
|
|||
|
{
|
|||
|
cp_error ("non-template used as template");
|
|||
|
return error_mark_node;
|
|||
|
}
|
|||
|
|
|||
|
if (arglist != NULL_TREE && !TREE_PERMANENT (arglist))
|
|||
|
{
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
arglist = copy_list (arglist);
|
|||
|
pop_obstacks ();
|
|||
|
}
|
|||
|
|
|||
|
return build_min (TEMPLATE_ID_EXPR,
|
|||
|
TREE_TYPE (fns)
|
|||
|
? TREE_TYPE (fns) : unknown_type_node,
|
|||
|
fns, arglist);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* 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.
|
|||
|
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. */
|
|||
|
|
|||
|
tree
|
|||
|
lookup_template_class (d1, arglist, in_decl)
|
|||
|
tree d1, arglist;
|
|||
|
tree in_decl;
|
|||
|
{
|
|||
|
tree template, parmlist;
|
|||
|
char *mangled_name;
|
|||
|
tree id, t;
|
|||
|
|
|||
|
if (TREE_CODE (d1) == IDENTIFIER_NODE)
|
|||
|
{
|
|||
|
template = IDENTIFIER_GLOBAL_VALUE (d1); /* XXX */
|
|||
|
if (! template)
|
|||
|
template = IDENTIFIER_CLASS_VALUE (d1);
|
|||
|
}
|
|||
|
else if (TREE_CODE (d1) == TYPE_DECL && IS_AGGR_TYPE (TREE_TYPE (d1)))
|
|||
|
{
|
|||
|
template = CLASSTYPE_TI_TEMPLATE (TREE_TYPE (d1));
|
|||
|
d1 = DECL_NAME (template);
|
|||
|
}
|
|||
|
else if (TREE_CODE_CLASS (TREE_CODE (d1)) == 't' && IS_AGGR_TYPE (d1))
|
|||
|
{
|
|||
|
template = CLASSTYPE_TI_TEMPLATE (d1);
|
|||
|
d1 = DECL_NAME (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_LOCAL_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)
|
|||
|
return error_mark_node;
|
|||
|
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 (PRIMARY_TEMPLATE_P (template))
|
|||
|
{
|
|||
|
parmlist = DECL_INNERMOST_TEMPLATE_PARMS (template);
|
|||
|
|
|||
|
arglist = coerce_template_parms (parmlist, arglist, template);
|
|||
|
if (arglist == error_mark_node)
|
|||
|
return error_mark_node;
|
|||
|
if (uses_template_parms (arglist))
|
|||
|
{
|
|||
|
tree found;
|
|||
|
if (comp_template_args
|
|||
|
(CLASSTYPE_TI_ARGS (TREE_TYPE (template)), arglist))
|
|||
|
found = TREE_TYPE (template);
|
|||
|
else
|
|||
|
{
|
|||
|
for (found = DECL_TEMPLATE_INSTANTIATIONS (template);
|
|||
|
found; found = TREE_CHAIN (found))
|
|||
|
{
|
|||
|
if (TI_USES_TEMPLATE_PARMS (found)
|
|||
|
&& comp_template_args (TREE_PURPOSE (found), arglist))
|
|||
|
break;
|
|||
|
}
|
|||
|
if (found)
|
|||
|
found = TREE_VALUE (found);
|
|||
|
}
|
|||
|
|
|||
|
if (found)
|
|||
|
{
|
|||
|
if (can_free (&permanent_obstack, arglist))
|
|||
|
obstack_free (&permanent_obstack, arglist);
|
|||
|
return found;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
mangled_name = mangle_class_name_for_template (IDENTIFIER_POINTER (d1),
|
|||
|
parmlist, arglist);
|
|||
|
id = get_identifier (mangled_name);
|
|||
|
IDENTIFIER_TEMPLATE (id) = d1;
|
|||
|
|
|||
|
maybe_push_to_top_level (uses_template_parms (arglist));
|
|||
|
t = xref_tag_from_type (TREE_TYPE (template), id, 1);
|
|||
|
pop_from_top_level ();
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
tree ctx = lookup_template_class (TYPE_CONTEXT (TREE_TYPE (template)),
|
|||
|
arglist, in_decl);
|
|||
|
id = d1;
|
|||
|
arglist = CLASSTYPE_TI_ARGS (ctx);
|
|||
|
|
|||
|
if (TYPE_BEING_DEFINED (ctx) && ctx == current_class_type)
|
|||
|
{
|
|||
|
int save_temp = processing_template_decl;
|
|||
|
processing_template_decl = 0;
|
|||
|
t = xref_tag_from_type (TREE_TYPE (template), id, 0);
|
|||
|
processing_template_decl = save_temp;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
t = lookup_nested_type_by_name (ctx, id);
|
|||
|
my_friendly_assert (t != NULL_TREE, 42);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/* Seems to be wanted. */
|
|||
|
CLASSTYPE_GOT_SEMICOLON (t) = 1;
|
|||
|
|
|||
|
if (! CLASSTYPE_TEMPLATE_INFO (t))
|
|||
|
{
|
|||
|
arglist = copy_to_permanent (arglist);
|
|||
|
CLASSTYPE_TEMPLATE_INFO (t)
|
|||
|
= perm_tree_cons (template, arglist, NULL_TREE);
|
|||
|
DECL_TEMPLATE_INSTANTIATIONS (template) = perm_tree_cons
|
|||
|
(arglist, t, DECL_TEMPLATE_INSTANTIATIONS (template));
|
|||
|
TI_USES_TEMPLATE_PARMS (DECL_TEMPLATE_INSTANTIATIONS (template))
|
|||
|
= uses_template_parms (arglist);
|
|||
|
|
|||
|
SET_CLASSTYPE_IMPLICIT_INSTANTIATION (t);
|
|||
|
|
|||
|
/* We need to set this again after CLASSTYPE_TEMPLATE_INFO is set up. */
|
|||
|
DECL_ASSEMBLER_NAME (TYPE_MAIN_DECL (t)) = id;
|
|||
|
/* if (! uses_template_parms (arglist)) */
|
|||
|
DECL_ASSEMBLER_NAME (TYPE_MAIN_DECL (t))
|
|||
|
= get_identifier (build_overload_name (t, 1, 1));
|
|||
|
|
|||
|
if (flag_external_templates && ! uses_template_parms (arglist)
|
|||
|
&& CLASSTYPE_INTERFACE_KNOWN (TREE_TYPE (template))
|
|||
|
&& ! CLASSTYPE_INTERFACE_ONLY (TREE_TYPE (template)))
|
|||
|
add_pending_template (t);
|
|||
|
}
|
|||
|
|
|||
|
return t;
|
|||
|
}
|
|||
|
|
|||
|
/* Should be defined in parse.h. */
|
|||
|
extern int yychar;
|
|||
|
|
|||
|
int
|
|||
|
uses_template_parms (t)
|
|||
|
tree t;
|
|||
|
{
|
|||
|
if (!t)
|
|||
|
return 0;
|
|||
|
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 uses_template_parms (TREE_TYPE (t));
|
|||
|
|
|||
|
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 (uses_template_parms (TREE_VEC_ELT (t, i)))
|
|||
|
return 1;
|
|||
|
return 0;
|
|||
|
}
|
|||
|
case TREE_LIST:
|
|||
|
if (uses_template_parms (TREE_PURPOSE (t))
|
|||
|
|| uses_template_parms (TREE_VALUE (t)))
|
|||
|
return 1;
|
|||
|
return uses_template_parms (TREE_CHAIN (t));
|
|||
|
|
|||
|
/* constructed type nodes */
|
|||
|
case POINTER_TYPE:
|
|||
|
case REFERENCE_TYPE:
|
|||
|
return uses_template_parms (TREE_TYPE (t));
|
|||
|
case RECORD_TYPE:
|
|||
|
if (TYPE_PTRMEMFUNC_FLAG (t))
|
|||
|
return uses_template_parms (TYPE_PTRMEMFUNC_FN_TYPE (t));
|
|||
|
case UNION_TYPE:
|
|||
|
if (! CLASSTYPE_TEMPLATE_INFO (t))
|
|||
|
return 0;
|
|||
|
return uses_template_parms (TREE_VALUE (CLASSTYPE_TEMPLATE_INFO (t)));
|
|||
|
case FUNCTION_TYPE:
|
|||
|
if (uses_template_parms (TYPE_ARG_TYPES (t)))
|
|||
|
return 1;
|
|||
|
return uses_template_parms (TREE_TYPE (t));
|
|||
|
case ARRAY_TYPE:
|
|||
|
if (uses_template_parms (TYPE_DOMAIN (t)))
|
|||
|
return 1;
|
|||
|
return uses_template_parms (TREE_TYPE (t));
|
|||
|
case OFFSET_TYPE:
|
|||
|
if (uses_template_parms (TYPE_OFFSET_BASETYPE (t)))
|
|||
|
return 1;
|
|||
|
return uses_template_parms (TREE_TYPE (t));
|
|||
|
case METHOD_TYPE:
|
|||
|
if (uses_template_parms (TYPE_METHOD_BASETYPE (t)))
|
|||
|
return 1;
|
|||
|
if (uses_template_parms (TYPE_ARG_TYPES (t)))
|
|||
|
return 1;
|
|||
|
return uses_template_parms (TREE_TYPE (t));
|
|||
|
|
|||
|
/* decl nodes */
|
|||
|
case TYPE_DECL:
|
|||
|
return uses_template_parms (TREE_TYPE (t));
|
|||
|
|
|||
|
case FUNCTION_DECL:
|
|||
|
case VAR_DECL:
|
|||
|
/* ??? What about FIELD_DECLs? */
|
|||
|
if (DECL_LANG_SPECIFIC (t) && DECL_TEMPLATE_INFO (t)
|
|||
|
&& uses_template_parms (DECL_TI_ARGS (t)))
|
|||
|
return 1;
|
|||
|
/* fall through */
|
|||
|
case CONST_DECL:
|
|||
|
case PARM_DECL:
|
|||
|
if (uses_template_parms (TREE_TYPE (t)))
|
|||
|
return 1;
|
|||
|
if (DECL_CONTEXT (t) && uses_template_parms (DECL_CONTEXT (t)))
|
|||
|
return 1;
|
|||
|
return 0;
|
|||
|
|
|||
|
case CALL_EXPR:
|
|||
|
return uses_template_parms (TREE_TYPE (t));
|
|||
|
case ADDR_EXPR:
|
|||
|
return uses_template_parms (TREE_OPERAND (t, 0));
|
|||
|
|
|||
|
/* template parm nodes */
|
|||
|
case TEMPLATE_TYPE_PARM:
|
|||
|
case TEMPLATE_CONST_PARM:
|
|||
|
return 1;
|
|||
|
|
|||
|
/* simple type nodes */
|
|||
|
case INTEGER_TYPE:
|
|||
|
if (uses_template_parms (TYPE_MIN_VALUE (t)))
|
|||
|
return 1;
|
|||
|
return uses_template_parms (TYPE_MAX_VALUE (t));
|
|||
|
|
|||
|
case REAL_TYPE:
|
|||
|
case COMPLEX_TYPE:
|
|||
|
case VOID_TYPE:
|
|||
|
case BOOLEAN_TYPE:
|
|||
|
return 0;
|
|||
|
|
|||
|
case ENUMERAL_TYPE:
|
|||
|
{
|
|||
|
tree v;
|
|||
|
|
|||
|
for (v = TYPE_VALUES (t); v != NULL_TREE; v = TREE_CHAIN (v))
|
|||
|
if (uses_template_parms (TREE_VALUE (v)))
|
|||
|
return 1;
|
|||
|
}
|
|||
|
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 SCOPE_REF:
|
|||
|
return uses_template_parms (TREE_OPERAND (t, 0));
|
|||
|
|
|||
|
case CONSTRUCTOR:
|
|||
|
if (TREE_TYPE (t) && TYPE_PTRMEMFUNC_P (TREE_TYPE (t)))
|
|||
|
return uses_template_parms (TYPE_PTRMEMFUNC_FN_TYPE (TREE_TYPE (t)));
|
|||
|
return uses_template_parms (TREE_OPERAND (t, 1));
|
|||
|
|
|||
|
case MODOP_EXPR:
|
|||
|
case CAST_EXPR:
|
|||
|
case REINTERPRET_CAST_EXPR:
|
|||
|
case CONST_CAST_EXPR:
|
|||
|
case STATIC_CAST_EXPR:
|
|||
|
case DYNAMIC_CAST_EXPR:
|
|||
|
case SIZEOF_EXPR:
|
|||
|
case ARROW_EXPR:
|
|||
|
case DOTSTAR_EXPR:
|
|||
|
case TYPEID_EXPR:
|
|||
|
return 1;
|
|||
|
|
|||
|
default:
|
|||
|
switch (TREE_CODE_CLASS (TREE_CODE (t)))
|
|||
|
{
|
|||
|
case '1':
|
|||
|
case '2':
|
|||
|
case 'e':
|
|||
|
case '<':
|
|||
|
{
|
|||
|
int i;
|
|||
|
for (i = tree_code_length[(int) TREE_CODE (t)]; --i >= 0;)
|
|||
|
if (uses_template_parms (TREE_OPERAND (t, i)))
|
|||
|
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;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
static struct tinst_level *current_tinst_level = 0;
|
|||
|
static struct tinst_level *free_tinst_level = 0;
|
|||
|
static int tinst_depth = 0;
|
|||
|
extern int max_tinst_depth;
|
|||
|
#ifdef GATHER_STATISTICS
|
|||
|
int depth_reached = 0;
|
|||
|
#endif
|
|||
|
|
|||
|
static int
|
|||
|
push_tinst_level (d)
|
|||
|
tree d;
|
|||
|
{
|
|||
|
struct tinst_level *new;
|
|||
|
|
|||
|
if (tinst_depth >= max_tinst_depth)
|
|||
|
{
|
|||
|
struct tinst_level *p = current_tinst_level;
|
|||
|
int line = lineno;
|
|||
|
char *file = input_filename;
|
|||
|
|
|||
|
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);
|
|||
|
|
|||
|
for (; p; p = p->next)
|
|||
|
{
|
|||
|
cp_error (" instantiated from `%D'", p->decl);
|
|||
|
lineno = p->line;
|
|||
|
input_filename = p->file;
|
|||
|
}
|
|||
|
error (" instantiated from here");
|
|||
|
|
|||
|
lineno = line;
|
|||
|
input_filename = file;
|
|||
|
|
|||
|
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
|
|||
|
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
void
|
|||
|
pop_tinst_level ()
|
|||
|
{
|
|||
|
struct tinst_level *old = current_tinst_level;
|
|||
|
|
|||
|
current_tinst_level = old->next;
|
|||
|
old->next = free_tinst_level;
|
|||
|
free_tinst_level = old;
|
|||
|
--tinst_depth;
|
|||
|
}
|
|||
|
|
|||
|
struct tinst_level *
|
|||
|
tinst_for_decl ()
|
|||
|
{
|
|||
|
struct tinst_level *p = current_tinst_level;
|
|||
|
|
|||
|
if (p)
|
|||
|
for (; p->next ; p = p->next )
|
|||
|
;
|
|||
|
return p;
|
|||
|
}
|
|||
|
|
|||
|
tree
|
|||
|
instantiate_class_template (type)
|
|||
|
tree type;
|
|||
|
{
|
|||
|
tree template, template_info, args, pattern, t, *field_chain;
|
|||
|
|
|||
|
if (type == error_mark_node)
|
|||
|
return error_mark_node;
|
|||
|
|
|||
|
template_info = CLASSTYPE_TEMPLATE_INFO (type);
|
|||
|
|
|||
|
if (TYPE_BEING_DEFINED (type) || TYPE_SIZE (type))
|
|||
|
return type;
|
|||
|
|
|||
|
template = TI_TEMPLATE (template_info);
|
|||
|
my_friendly_assert (TREE_CODE (template) == TEMPLATE_DECL, 279);
|
|||
|
args = TI_ARGS (template_info);
|
|||
|
|
|||
|
t = most_specialized_class
|
|||
|
(DECL_TEMPLATE_SPECIALIZATIONS (template), args);
|
|||
|
|
|||
|
if (t == error_mark_node)
|
|||
|
{
|
|||
|
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;
|
|||
|
return error_mark_node;
|
|||
|
}
|
|||
|
else if (t)
|
|||
|
pattern = TREE_TYPE (t);
|
|||
|
else
|
|||
|
pattern = TREE_TYPE (template);
|
|||
|
|
|||
|
if (TYPE_SIZE (pattern) == NULL_TREE)
|
|||
|
return type;
|
|||
|
|
|||
|
if (t)
|
|||
|
args = get_class_bindings (TREE_VALUE (t), TREE_PURPOSE (t), args);
|
|||
|
|
|||
|
TYPE_BEING_DEFINED (type) = 1;
|
|||
|
|
|||
|
if (! push_tinst_level (type))
|
|||
|
return type;
|
|||
|
|
|||
|
maybe_push_to_top_level (uses_template_parms (type));
|
|||
|
pushclass (type, 0);
|
|||
|
|
|||
|
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_HAS_ASSIGNMENT (type) = TYPE_HAS_ASSIGNMENT (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_GETS_INIT_AGGR (type) = TYPE_GETS_INIT_AGGR (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);
|
|||
|
|
|||
|
{
|
|||
|
tree binfo = TYPE_BINFO (type);
|
|||
|
tree pbases = TYPE_BINFO_BASETYPES (pattern);
|
|||
|
|
|||
|
if (pbases)
|
|||
|
{
|
|||
|
tree bases;
|
|||
|
int i;
|
|||
|
int len = TREE_VEC_LENGTH (pbases);
|
|||
|
bases = make_tree_vec (len);
|
|||
|
for (i = 0; i < len; ++i)
|
|||
|
{
|
|||
|
tree elt;
|
|||
|
|
|||
|
TREE_VEC_ELT (bases, i) = elt
|
|||
|
= tsubst (TREE_VEC_ELT (pbases, i), args,
|
|||
|
TREE_VEC_LENGTH (args), NULL_TREE);
|
|||
|
BINFO_INHERITANCE_CHAIN (elt) = binfo;
|
|||
|
|
|||
|
if (! IS_AGGR_TYPE (TREE_TYPE (elt)))
|
|||
|
cp_error
|
|||
|
("base type `%T' of `%T' fails to be a struct or class type",
|
|||
|
TREE_TYPE (elt), type);
|
|||
|
else if (! uses_template_parms (type)
|
|||
|
&& (TYPE_SIZE (complete_type (TREE_TYPE (elt)))
|
|||
|
== NULL_TREE))
|
|||
|
cp_error ("base class `%T' of `%T' has incomplete type",
|
|||
|
TREE_TYPE (elt), type);
|
|||
|
}
|
|||
|
/* Don't initialize this until the vector is filled out, or
|
|||
|
lookups will crash. */
|
|||
|
BINFO_BASETYPES (binfo) = bases;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
CLASSTYPE_LOCAL_TYPEDECLS (type) = CLASSTYPE_LOCAL_TYPEDECLS (pattern);
|
|||
|
|
|||
|
field_chain = &TYPE_FIELDS (type);
|
|||
|
|
|||
|
for (t = CLASSTYPE_TAGS (pattern); t; t = TREE_CHAIN (t))
|
|||
|
{
|
|||
|
tree name = TREE_PURPOSE (t);
|
|||
|
tree tag = TREE_VALUE (t);
|
|||
|
|
|||
|
/* These will add themselves to CLASSTYPE_TAGS for the new type. */
|
|||
|
if (TREE_CODE (tag) == ENUMERAL_TYPE)
|
|||
|
{
|
|||
|
tree e, newtag = tsubst_enum (tag, args,
|
|||
|
TREE_VEC_LENGTH (args), field_chain);
|
|||
|
|
|||
|
while (*field_chain)
|
|||
|
{
|
|||
|
DECL_FIELD_CONTEXT (*field_chain) = type;
|
|||
|
field_chain = &TREE_CHAIN (*field_chain);
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
tsubst (tag, args,
|
|||
|
TREE_VEC_LENGTH (args), NULL_TREE);
|
|||
|
}
|
|||
|
|
|||
|
/* Don't replace enum constants here. */
|
|||
|
for (t = TYPE_FIELDS (pattern); t; t = TREE_CHAIN (t))
|
|||
|
if (TREE_CODE (t) != CONST_DECL)
|
|||
|
{
|
|||
|
tree r = tsubst (t, args,
|
|||
|
TREE_VEC_LENGTH (args), NULL_TREE);
|
|||
|
if (TREE_CODE (r) == VAR_DECL)
|
|||
|
{
|
|||
|
if (! uses_template_parms (r))
|
|||
|
pending_statics = perm_tree_cons (NULL_TREE, r, pending_statics);
|
|||
|
/* Perhaps I should do more of grokfield here. */
|
|||
|
start_decl_1 (r);
|
|||
|
DECL_IN_AGGR_P (r) = 1;
|
|||
|
DECL_EXTERNAL (r) = 1;
|
|||
|
cp_finish_decl (r, DECL_INITIAL (r), NULL_TREE, 0, 0);
|
|||
|
}
|
|||
|
|
|||
|
*field_chain = r;
|
|||
|
field_chain = &TREE_CHAIN (r);
|
|||
|
}
|
|||
|
|
|||
|
TYPE_METHODS (type) = tsubst_chain (TYPE_METHODS (pattern), args);
|
|||
|
for (t = TYPE_METHODS (type); t; t = TREE_CHAIN (t))
|
|||
|
{
|
|||
|
if (DECL_CONSTRUCTOR_P (t))
|
|||
|
grok_ctor_properties (type, t);
|
|||
|
else if (IDENTIFIER_OPNAME_P (DECL_NAME (t)))
|
|||
|
grok_op_properties (t, DECL_VIRTUAL_P (t), 0);
|
|||
|
}
|
|||
|
|
|||
|
DECL_FRIENDLIST (TYPE_MAIN_DECL (type))
|
|||
|
= tsubst (DECL_FRIENDLIST (TYPE_MAIN_DECL (pattern)),
|
|||
|
args, TREE_VEC_LENGTH (args), NULL_TREE);
|
|||
|
|
|||
|
{
|
|||
|
tree d = CLASSTYPE_FRIEND_CLASSES (type)
|
|||
|
= tsubst (CLASSTYPE_FRIEND_CLASSES (pattern), args,
|
|||
|
TREE_VEC_LENGTH (args), NULL_TREE);
|
|||
|
|
|||
|
/* This does injection for friend classes. */
|
|||
|
for (; d; d = TREE_CHAIN (d))
|
|||
|
TREE_VALUE (d) = xref_tag_from_type (TREE_VALUE (d), NULL_TREE, 1);
|
|||
|
|
|||
|
d = tsubst (DECL_TEMPLATE_INJECT (template), args,
|
|||
|
TREE_VEC_LENGTH (args), NULL_TREE);
|
|||
|
|
|||
|
for (; d; d = TREE_CHAIN (d))
|
|||
|
{
|
|||
|
tree t = TREE_VALUE (d);
|
|||
|
|
|||
|
if (TREE_CODE (t) == TYPE_DECL)
|
|||
|
/* Already injected. */;
|
|||
|
else
|
|||
|
pushdecl (t);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (! uses_template_parms (type))
|
|||
|
{
|
|||
|
tree tmp;
|
|||
|
for (tmp = TYPE_FIELDS (type); tmp; tmp = TREE_CHAIN (tmp))
|
|||
|
if (TREE_CODE (tmp) == FIELD_DECL)
|
|||
|
{
|
|||
|
TREE_TYPE (tmp) = complete_type (TREE_TYPE (tmp));
|
|||
|
require_complete_type (tmp);
|
|||
|
}
|
|||
|
|
|||
|
type = finish_struct_1 (type, 0);
|
|||
|
CLASSTYPE_GOT_SEMICOLON (type) = 1;
|
|||
|
|
|||
|
repo_template_used (type);
|
|||
|
if (at_eof && TYPE_BINFO_VTABLE (type) != NULL_TREE)
|
|||
|
finish_prevtable_vardecl (NULL, TYPE_BINFO_VTABLE (type));
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
TYPE_SIZE (type) = integer_zero_node;
|
|||
|
CLASSTYPE_METHOD_VEC (type)
|
|||
|
= finish_struct_methods (type, TYPE_METHODS (type), 1);
|
|||
|
}
|
|||
|
|
|||
|
TYPE_BEING_DEFINED (type) = 0;
|
|||
|
popclass (0);
|
|||
|
|
|||
|
pop_from_top_level ();
|
|||
|
pop_tinst_level ();
|
|||
|
|
|||
|
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));
|
|||
|
}
|
|||
|
|
|||
|
tree
|
|||
|
lookup_nested_type_by_name (ctype, name)
|
|||
|
tree ctype, name;
|
|||
|
{
|
|||
|
tree t;
|
|||
|
|
|||
|
complete_type (ctype);
|
|||
|
|
|||
|
for (t = CLASSTYPE_TAGS (ctype); t; t = TREE_CHAIN (t))
|
|||
|
{
|
|||
|
if (name == TREE_PURPOSE (t)
|
|||
|
/* this catches typedef enum { foo } bar; */
|
|||
|
|| name == TYPE_IDENTIFIER (TREE_VALUE (t)))
|
|||
|
return TREE_VALUE (t);
|
|||
|
}
|
|||
|
return NULL_TREE;
|
|||
|
}
|
|||
|
|
|||
|
/* 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;
|
|||
|
}
|
|||
|
|
|||
|
/* Take the tree structure T and replace template parameters used therein
|
|||
|
with the argument vector ARGS. NARGS is the number of args; should
|
|||
|
be removed. IN_DECL is an associated decl for diagnostics.
|
|||
|
|
|||
|
tsubst is used for dealing with types, decls and the like; for
|
|||
|
expressions, use tsubst_expr or tsubst_copy. */
|
|||
|
|
|||
|
tree
|
|||
|
tsubst (t, args, nargs, in_decl)
|
|||
|
tree t, args;
|
|||
|
int nargs;
|
|||
|
tree in_decl;
|
|||
|
{
|
|||
|
tree type;
|
|||
|
|
|||
|
if (t == NULL_TREE || t == error_mark_node
|
|||
|
|| t == integer_type_node
|
|||
|
|| t == void_type_node
|
|||
|
|| t == char_type_node)
|
|||
|
return t;
|
|||
|
|
|||
|
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)
|
|||
|
type = tsubst (type, args, nargs, in_decl);
|
|||
|
|
|||
|
switch (TREE_CODE (t))
|
|||
|
{
|
|||
|
case RECORD_TYPE:
|
|||
|
if (TYPE_PTRMEMFUNC_P (t))
|
|||
|
{
|
|||
|
tree r = build_ptrmemfunc_type
|
|||
|
(tsubst (TYPE_PTRMEMFUNC_FN_TYPE (t), args, nargs, in_decl));
|
|||
|
return cp_build_type_variant (r, TYPE_READONLY (t),
|
|||
|
TYPE_VOLATILE (t));
|
|||
|
}
|
|||
|
|
|||
|
/* else fall through */
|
|||
|
case UNION_TYPE:
|
|||
|
if (uses_template_parms (t))
|
|||
|
{
|
|||
|
tree argvec = tsubst (CLASSTYPE_TI_ARGS (t), args, nargs, in_decl);
|
|||
|
tree r = lookup_template_class (t, argvec, in_decl);
|
|||
|
return cp_build_type_variant (r, TYPE_READONLY (t),
|
|||
|
TYPE_VOLATILE (t));
|
|||
|
}
|
|||
|
|
|||
|
/* else fall through */
|
|||
|
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 ENUMERAL_TYPE:
|
|||
|
{
|
|||
|
tree ctx = tsubst (TYPE_CONTEXT (t), args, nargs, in_decl);
|
|||
|
if (ctx == NULL_TREE)
|
|||
|
return t;
|
|||
|
else if (ctx == current_function_decl)
|
|||
|
return lookup_name (TYPE_IDENTIFIER (t), 1);
|
|||
|
else
|
|||
|
return lookup_nested_type_by_name (ctx, TYPE_IDENTIFIER (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 = TREE_OPERAND (TYPE_MAX_VALUE (t), 0);
|
|||
|
max = tsubst_expr (max, args, nargs, in_decl);
|
|||
|
if (processing_template_decl)
|
|||
|
{
|
|||
|
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;
|
|||
|
}
|
|||
|
|
|||
|
max = fold (build_binary_op (MINUS_EXPR, max, integer_one_node, 1));
|
|||
|
return build_index_2_type (size_zero_node, max);
|
|||
|
}
|
|||
|
|
|||
|
case TEMPLATE_TYPE_PARM:
|
|||
|
case TEMPLATE_CONST_PARM:
|
|||
|
{
|
|||
|
int idx;
|
|||
|
int level;
|
|||
|
|
|||
|
if (TREE_CODE (t) == TEMPLATE_TYPE_PARM)
|
|||
|
{
|
|||
|
idx = TEMPLATE_TYPE_IDX (t);
|
|||
|
level = TEMPLATE_TYPE_LEVEL (t);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
idx = TEMPLATE_CONST_IDX (t);
|
|||
|
level = TEMPLATE_CONST_LEVEL (t);
|
|||
|
}
|
|||
|
|
|||
|
if (TREE_VEC_LENGTH (args) > 0)
|
|||
|
{
|
|||
|
tree arg = NULL_TREE;
|
|||
|
|
|||
|
if (TREE_CODE (TREE_VEC_ELT (args, 0)) == TREE_VEC)
|
|||
|
{
|
|||
|
if (TREE_VEC_LENGTH (args) >= level - 1)
|
|||
|
arg = TREE_VEC_ELT
|
|||
|
(TREE_VEC_ELT (args, level - 1), idx);
|
|||
|
}
|
|||
|
else if (level == 1)
|
|||
|
arg = TREE_VEC_ELT (args, idx);
|
|||
|
|
|||
|
if (arg != NULL_TREE)
|
|||
|
{
|
|||
|
if (TREE_CODE (t) == TEMPLATE_TYPE_PARM)
|
|||
|
return cp_build_type_variant
|
|||
|
(arg, TYPE_READONLY (arg) || TYPE_READONLY (t),
|
|||
|
TYPE_VOLATILE (arg) || TYPE_VOLATILE (t));
|
|||
|
else
|
|||
|
return arg;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/* If we get here, we must have been looking at a parm for a
|
|||
|
more deeply nested template. */
|
|||
|
my_friendly_assert((TREE_CODE (t) == TEMPLATE_CONST_PARM
|
|||
|
&& TEMPLATE_CONST_LEVEL (t) > 1)
|
|||
|
|| (TREE_CODE (t) == TEMPLATE_TYPE_PARM
|
|||
|
&& TEMPLATE_TYPE_LEVEL (t) > 1),
|
|||
|
0);
|
|||
|
return t;
|
|||
|
}
|
|||
|
|
|||
|
case TEMPLATE_DECL:
|
|||
|
{
|
|||
|
/* We can get here when processing a member template function
|
|||
|
of a template class. */
|
|||
|
tree tmpl;
|
|||
|
tree decl = DECL_TEMPLATE_RESULT (t);
|
|||
|
tree new_decl;
|
|||
|
tree parms;
|
|||
|
tree spec;
|
|||
|
int i;
|
|||
|
|
|||
|
/* We might already have an instance of this template. */
|
|||
|
tree instances = DECL_TEMPLATE_INSTANTIATIONS (t);
|
|||
|
tree ctx = tsubst (DECL_CLASS_CONTEXT (t), args, nargs, in_decl);
|
|||
|
|
|||
|
for (; instances; instances = TREE_CHAIN (instances))
|
|||
|
if (DECL_CLASS_CONTEXT (TREE_VALUE (instances)) == ctx)
|
|||
|
return TREE_VALUE (instances);
|
|||
|
|
|||
|
/* 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. */
|
|||
|
tmpl = copy_node (t);
|
|||
|
copy_lang_decl (tmpl);
|
|||
|
my_friendly_assert (DECL_LANG_SPECIFIC (tmpl) != 0, 0);
|
|||
|
DECL_CHAIN (tmpl) = NULL_TREE;
|
|||
|
TREE_CHAIN (tmpl) = NULL_TREE;
|
|||
|
DECL_TEMPLATE_INFO (tmpl) = build_tree_list (t, args);
|
|||
|
new_decl = tsubst (decl, args, nargs, in_decl);
|
|||
|
DECL_RESULT (tmpl) = new_decl;
|
|||
|
DECL_TI_TEMPLATE (new_decl) = tmpl;
|
|||
|
TREE_TYPE (tmpl) = TREE_TYPE (new_decl);
|
|||
|
DECL_TEMPLATE_INSTANTIATIONS (tmpl) = NULL_TREE;
|
|||
|
SET_DECL_IMPLICIT_INSTANTIATION (tmpl);
|
|||
|
|
|||
|
/* 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 (tmpl)
|
|||
|
= copy_node (DECL_TEMPLATE_PARMS (tmpl));
|
|||
|
for (parms = DECL_TEMPLATE_PARMS (tmpl);
|
|||
|
TREE_CHAIN (parms) != NULL_TREE;
|
|||
|
parms = TREE_CHAIN (parms))
|
|||
|
TREE_CHAIN (parms) = copy_node (TREE_CHAIN (parms));
|
|||
|
|
|||
|
/* Record this partial instantiation. */
|
|||
|
DECL_TEMPLATE_INSTANTIATIONS (t)
|
|||
|
= perm_tree_cons (NULL_TREE, tmpl,
|
|||
|
DECL_TEMPLATE_INSTANTIATIONS (t));
|
|||
|
|
|||
|
DECL_TEMPLATE_SPECIALIZATIONS (tmpl) = NULL_TREE;
|
|||
|
return tmpl;
|
|||
|
}
|
|||
|
|
|||
|
case FUNCTION_DECL:
|
|||
|
{
|
|||
|
tree r = NULL_TREE;
|
|||
|
tree ctx;
|
|||
|
|
|||
|
int member;
|
|||
|
|
|||
|
if (DECL_CONTEXT (t) != NULL_TREE
|
|||
|
&& TREE_CODE_CLASS (TREE_CODE (DECL_CONTEXT (t))) == 't')
|
|||
|
{
|
|||
|
if (DECL_NAME (t) == constructor_name (DECL_CONTEXT (t)))
|
|||
|
member = 2;
|
|||
|
else
|
|||
|
member = 1;
|
|||
|
ctx = tsubst (DECL_CLASS_CONTEXT (t), args, nargs, t);
|
|||
|
type = tsubst (type, args, nargs, in_decl);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
member = 0;
|
|||
|
ctx = NULL_TREE;
|
|||
|
type = tsubst (type, args, nargs, in_decl);
|
|||
|
}
|
|||
|
|
|||
|
/* Do we already have this instantiation? */
|
|||
|
if (DECL_TEMPLATE_INFO (t) != NULL_TREE)
|
|||
|
{
|
|||
|
tree tmpl = DECL_TI_TEMPLATE (t);
|
|||
|
tree decls = DECL_TEMPLATE_INSTANTIATIONS (tmpl);
|
|||
|
|
|||
|
for (; decls; decls = TREE_CHAIN (decls))
|
|||
|
if (TREE_TYPE (TREE_VALUE (decls)) == type
|
|||
|
&& DECL_CLASS_CONTEXT (TREE_VALUE (decls)) == ctx
|
|||
|
&& comp_template_args (TREE_PURPOSE (decls), args))
|
|||
|
return TREE_VALUE (decls);
|
|||
|
}
|
|||
|
|
|||
|
/* 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);
|
|||
|
TREE_TYPE (r) = type;
|
|||
|
|
|||
|
DECL_CONTEXT (r)
|
|||
|
= tsubst (DECL_CONTEXT (t), args, nargs, t);
|
|||
|
DECL_CLASS_CONTEXT (r) = ctx;
|
|||
|
|
|||
|
if (member && !strncmp (OPERATOR_TYPENAME_FORMAT,
|
|||
|
IDENTIFIER_POINTER (DECL_NAME (r)),
|
|||
|
sizeof (OPERATOR_TYPENAME_FORMAT) - 1))
|
|||
|
{
|
|||
|
/* 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));
|
|||
|
}
|
|||
|
|
|||
|
if (DESTRUCTOR_NAME_P (DECL_ASSEMBLER_NAME (t)))
|
|||
|
{
|
|||
|
char *buf, *dbuf = build_overload_name (ctx, 1, 1);
|
|||
|
int len = sizeof (DESTRUCTOR_DECL_PREFIX) - 1;
|
|||
|
buf = (char *) alloca (strlen (dbuf)
|
|||
|
+ sizeof (DESTRUCTOR_DECL_PREFIX));
|
|||
|
bcopy (DESTRUCTOR_DECL_PREFIX, buf, len);
|
|||
|
buf[len] = '\0';
|
|||
|
strcat (buf, dbuf);
|
|||
|
DECL_ASSEMBLER_NAME (r) = get_identifier (buf);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
/* Instantiations of template functions must be mangled
|
|||
|
specially, in order to conform to 14.5.5.1
|
|||
|
[temp.over.link]. We use in_decl below rather than
|
|||
|
DECL_TI_TEMPLATE (r) because the latter is set to
|
|||
|
NULL_TREE in instantiate_decl. */
|
|||
|
tree tmpl;
|
|||
|
tree arg_types;
|
|||
|
|
|||
|
if (DECL_TEMPLATE_INFO (r))
|
|||
|
tmpl = DECL_TI_TEMPLATE (r);
|
|||
|
else
|
|||
|
tmpl = in_decl;
|
|||
|
|
|||
|
/* tmpl will be NULL if this is a specialization of a
|
|||
|
member template of a template class. */
|
|||
|
if (name_mangling_version < 1
|
|||
|
|| tmpl == NULL_TREE
|
|||
|
|| (member && !is_member_template (tmpl)
|
|||
|
&& !DECL_TEMPLATE_INFO (tmpl)))
|
|||
|
{
|
|||
|
arg_types = TYPE_ARG_TYPES (type);
|
|||
|
if (member && TREE_CODE (type) == FUNCTION_TYPE)
|
|||
|
arg_types = hash_tree_chain
|
|||
|
(build_pointer_type (DECL_CONTEXT (r)),
|
|||
|
arg_types);
|
|||
|
|
|||
|
DECL_ASSEMBLER_NAME (r)
|
|||
|
= build_decl_overload (DECL_NAME (r), arg_types,
|
|||
|
member);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
/* We pass the outermost template parameters to
|
|||
|
build_template_decl_overload since the innermost
|
|||
|
template parameters are still just template
|
|||
|
parameters; there are no corresponding substitution
|
|||
|
arguments. */
|
|||
|
/* FIXME The messed up thing here is that we get here with
|
|||
|
full args and only one level of parms. This is necessary
|
|||
|
because when we partially instantiate a member template,
|
|||
|
even though there's really only one level of parms left
|
|||
|
we re-use the parms from the original template, which
|
|||
|
have level 2. When this is fixed we can remove the
|
|||
|
add_to_template_args from instantiate_template. */
|
|||
|
tree tparms = DECL_TEMPLATE_PARMS (tmpl);
|
|||
|
|
|||
|
while (tparms && TREE_CHAIN (tparms) != NULL_TREE)
|
|||
|
tparms = TREE_CHAIN (tparms);
|
|||
|
|
|||
|
my_friendly_assert (tparms != NULL_TREE
|
|||
|
&& TREE_CODE (tparms) == TREE_LIST,
|
|||
|
0);
|
|||
|
tparms = TREE_VALUE (tparms);
|
|||
|
|
|||
|
arg_types = TYPE_ARG_TYPES (TREE_TYPE (tmpl));
|
|||
|
if (member && TREE_CODE (type) == FUNCTION_TYPE)
|
|||
|
arg_types = hash_tree_chain
|
|||
|
(build_pointer_type (DECL_CONTEXT (r)),
|
|||
|
arg_types);
|
|||
|
|
|||
|
DECL_ASSEMBLER_NAME (r)
|
|||
|
= build_template_decl_overload
|
|||
|
(DECL_NAME (r), arg_types,
|
|||
|
TREE_TYPE (TREE_TYPE (tmpl)),
|
|||
|
tparms,
|
|||
|
TREE_CODE (TREE_VEC_ELT (args, 0)) == TREE_VEC
|
|||
|
? TREE_VEC_ELT (args, TREE_VEC_LENGTH (args) - 1) :
|
|||
|
args,
|
|||
|
member);
|
|||
|
}
|
|||
|
}
|
|||
|
DECL_RTL (r) = 0;
|
|||
|
make_decl_rtl (r, NULL_PTR, 1);
|
|||
|
|
|||
|
DECL_ARGUMENTS (r) = tsubst (DECL_ARGUMENTS (t), args, nargs, t);
|
|||
|
DECL_MAIN_VARIANT (r) = r;
|
|||
|
DECL_RESULT (r) = NULL_TREE;
|
|||
|
DECL_INITIAL (r) = NULL_TREE;
|
|||
|
|
|||
|
TREE_STATIC (r) = 0;
|
|||
|
TREE_PUBLIC (r) = 1;
|
|||
|
DECL_EXTERNAL (r) = 1;
|
|||
|
DECL_INTERFACE_KNOWN (r) = 0;
|
|||
|
DECL_DEFER_OUTPUT (r) = 0;
|
|||
|
TREE_CHAIN (r) = NULL_TREE;
|
|||
|
DECL_CHAIN (r) = NULL_TREE;
|
|||
|
|
|||
|
if (IDENTIFIER_OPNAME_P (DECL_NAME (r)))
|
|||
|
grok_op_properties (r, DECL_VIRTUAL_P (r), DECL_FRIEND_P (r));
|
|||
|
|
|||
|
/* Look for matching decls for the moment. */
|
|||
|
if (! member && ! flag_ansi_overloading)
|
|||
|
{
|
|||
|
tree decls = lookup_name_nonclass (DECL_NAME (t));
|
|||
|
tree d = NULL_TREE;
|
|||
|
|
|||
|
if (decls == NULL_TREE)
|
|||
|
/* no match */;
|
|||
|
else if (is_overloaded_fn (decls))
|
|||
|
for (decls = get_first_fn (decls); decls;
|
|||
|
decls = DECL_CHAIN (decls))
|
|||
|
{
|
|||
|
if (TREE_CODE (decls) == FUNCTION_DECL
|
|||
|
&& TREE_TYPE (decls) == type)
|
|||
|
{
|
|||
|
d = decls;
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (d)
|
|||
|
{
|
|||
|
int dcl_only = ! DECL_INITIAL (d);
|
|||
|
if (dcl_only)
|
|||
|
DECL_INITIAL (r) = error_mark_node;
|
|||
|
duplicate_decls (r, d);
|
|||
|
r = d;
|
|||
|
if (dcl_only)
|
|||
|
DECL_INITIAL (r) = 0;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (DECL_TEMPLATE_INFO (t) != NULL_TREE)
|
|||
|
{
|
|||
|
tree tmpl = DECL_TI_TEMPLATE (t);
|
|||
|
tree *declsp = &DECL_TEMPLATE_INSTANTIATIONS (tmpl);
|
|||
|
tree argvec = tsubst (DECL_TI_ARGS (t), args, nargs, in_decl);
|
|||
|
|
|||
|
if (DECL_TEMPLATE_INFO (tmpl) && DECL_TI_ARGS (tmpl))
|
|||
|
argvec = add_to_template_args (DECL_TI_ARGS (tmpl), argvec);
|
|||
|
|
|||
|
DECL_TEMPLATE_INFO (r) = perm_tree_cons (tmpl, argvec, NULL_TREE);
|
|||
|
*declsp = perm_tree_cons (argvec, r, *declsp);
|
|||
|
|
|||
|
/* If we have a preexisting version of this function, don't expand
|
|||
|
the template version, use the other instead. */
|
|||
|
if (TREE_STATIC (r) || DECL_TEMPLATE_SPECIALIZATION (r))
|
|||
|
SET_DECL_TEMPLATE_SPECIALIZATION (r);
|
|||
|
else
|
|||
|
SET_DECL_IMPLICIT_INSTANTIATION (r);
|
|||
|
|
|||
|
DECL_TEMPLATE_INSTANTIATIONS (tmpl)
|
|||
|
= tree_cons (argvec, r, DECL_TEMPLATE_INSTANTIATIONS (tmpl));
|
|||
|
}
|
|||
|
|
|||
|
/* 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
|
|||
|
&& IDENTIFIER_GLOBAL_VALUE (DECL_ASSEMBLER_NAME (r)) == NULL_TREE)
|
|||
|
IDENTIFIER_GLOBAL_VALUE (DECL_ASSEMBLER_NAME (r)) = r;
|
|||
|
|
|||
|
return r;
|
|||
|
}
|
|||
|
|
|||
|
case PARM_DECL:
|
|||
|
{
|
|||
|
tree r = copy_node (t);
|
|||
|
TREE_TYPE (r) = type;
|
|||
|
DECL_INITIAL (r) = TREE_TYPE (r);
|
|||
|
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, nargs, TREE_CHAIN (t));
|
|||
|
return r;
|
|||
|
}
|
|||
|
|
|||
|
case FIELD_DECL:
|
|||
|
{
|
|||
|
tree r = copy_node (t);
|
|||
|
TREE_TYPE (r) = type;
|
|||
|
copy_lang_decl (r);
|
|||
|
#if 0
|
|||
|
DECL_FIELD_CONTEXT (r) = tsubst (DECL_FIELD_CONTEXT (t), args, nargs, in_decl);
|
|||
|
#endif
|
|||
|
DECL_INITIAL (r) = tsubst_expr (DECL_INITIAL (t), args, nargs, in_decl);
|
|||
|
TREE_CHAIN (r) = NULL_TREE;
|
|||
|
return r;
|
|||
|
}
|
|||
|
|
|||
|
case USING_DECL:
|
|||
|
{
|
|||
|
tree r = copy_node (t);
|
|||
|
DECL_INITIAL (r)
|
|||
|
= tsubst_copy (DECL_INITIAL (t), args, nargs, in_decl);
|
|||
|
TREE_CHAIN (r) = NULL_TREE;
|
|||
|
return r;
|
|||
|
}
|
|||
|
|
|||
|
case VAR_DECL:
|
|||
|
{
|
|||
|
tree r;
|
|||
|
tree ctx = tsubst_copy (DECL_CONTEXT (t), args, nargs, in_decl);
|
|||
|
|
|||
|
/* Do we already have this instantiation? */
|
|||
|
if (DECL_LANG_SPECIFIC (t) && DECL_TEMPLATE_INFO (t))
|
|||
|
{
|
|||
|
tree tmpl = DECL_TI_TEMPLATE (t);
|
|||
|
tree decls = DECL_TEMPLATE_INSTANTIATIONS (tmpl);
|
|||
|
|
|||
|
for (; decls; decls = TREE_CHAIN (decls))
|
|||
|
if (DECL_CONTEXT (TREE_VALUE (decls)) == ctx)
|
|||
|
return TREE_VALUE (decls);
|
|||
|
}
|
|||
|
|
|||
|
r = copy_node (t);
|
|||
|
TREE_TYPE (r) = type;
|
|||
|
DECL_CONTEXT (r) = ctx;
|
|||
|
if (TREE_STATIC (r))
|
|||
|
DECL_ASSEMBLER_NAME (r)
|
|||
|
= build_static_name (DECL_CONTEXT (r), DECL_NAME (r));
|
|||
|
|
|||
|
/* 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;
|
|||
|
|
|||
|
if (DECL_LANG_SPECIFIC (r))
|
|||
|
{
|
|||
|
copy_lang_decl (r);
|
|||
|
DECL_CLASS_CONTEXT (r) = DECL_CONTEXT (r);
|
|||
|
}
|
|||
|
|
|||
|
if (DECL_LANG_SPECIFIC (t) && DECL_TEMPLATE_INFO (t))
|
|||
|
{
|
|||
|
tree tmpl = DECL_TI_TEMPLATE (t);
|
|||
|
tree *declsp = &DECL_TEMPLATE_INSTANTIATIONS (tmpl);
|
|||
|
tree argvec = tsubst (DECL_TI_ARGS (t), args, nargs, in_decl);
|
|||
|
|
|||
|
DECL_TEMPLATE_INFO (r) = perm_tree_cons (tmpl, argvec, NULL_TREE);
|
|||
|
*declsp = perm_tree_cons (argvec, r, *declsp);
|
|||
|
SET_DECL_IMPLICIT_INSTANTIATION (r);
|
|||
|
}
|
|||
|
TREE_CHAIN (r) = NULL_TREE;
|
|||
|
return r;
|
|||
|
}
|
|||
|
|
|||
|
case TYPE_DECL:
|
|||
|
if (t == TYPE_NAME (TREE_TYPE (t)))
|
|||
|
return TYPE_NAME (type);
|
|||
|
|
|||
|
{
|
|||
|
tree r = copy_node (t);
|
|||
|
TREE_TYPE (r) = type;
|
|||
|
DECL_CONTEXT (r) = current_class_type;
|
|||
|
TREE_CHAIN (r) = NULL_TREE;
|
|||
|
return r;
|
|||
|
}
|
|||
|
|
|||
|
case TREE_LIST:
|
|||
|
{
|
|||
|
tree purpose, value, chain, result;
|
|||
|
int via_public, via_virtual, via_protected;
|
|||
|
|
|||
|
if (t == void_list_node)
|
|||
|
return t;
|
|||
|
|
|||
|
via_public = TREE_VIA_PUBLIC (t);
|
|||
|
via_protected = TREE_VIA_PROTECTED (t);
|
|||
|
via_virtual = TREE_VIA_VIRTUAL (t);
|
|||
|
|
|||
|
purpose = TREE_PURPOSE (t);
|
|||
|
if (purpose)
|
|||
|
purpose = tsubst (purpose, args, nargs, in_decl);
|
|||
|
value = TREE_VALUE (t);
|
|||
|
if (value)
|
|||
|
value = tsubst (value, args, nargs, in_decl);
|
|||
|
chain = TREE_CHAIN (t);
|
|||
|
if (chain && chain != void_type_node)
|
|||
|
chain = tsubst (chain, args, nargs, in_decl);
|
|||
|
if (purpose == TREE_PURPOSE (t)
|
|||
|
&& value == TREE_VALUE (t)
|
|||
|
&& chain == TREE_CHAIN (t))
|
|||
|
return t;
|
|||
|
result = hash_tree_cons (via_public, via_virtual, via_protected,
|
|||
|
purpose, value, chain);
|
|||
|
TREE_PARMLIST (result) = TREE_PARMLIST (t);
|
|||
|
return result;
|
|||
|
}
|
|||
|
case TREE_VEC:
|
|||
|
if (type != NULL_TREE)
|
|||
|
{
|
|||
|
/* A binfo node. */
|
|||
|
|
|||
|
t = copy_node (t);
|
|||
|
|
|||
|
if (type == TREE_TYPE (t))
|
|||
|
return t;
|
|||
|
|
|||
|
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. */
|
|||
|
{
|
|||
|
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++)
|
|||
|
{
|
|||
|
elts[i] = maybe_fold_nontype_arg
|
|||
|
(tsubst_expr (TREE_VEC_ELT (t, i), args, nargs, in_decl));
|
|||
|
|
|||
|
if (elts[i] != TREE_VEC_ELT (t, i))
|
|||
|
need_new = 1;
|
|||
|
}
|
|||
|
|
|||
|
if (!need_new)
|
|||
|
return t;
|
|||
|
|
|||
|
t = make_tree_vec (len);
|
|||
|
for (i = 0; i < len; i++)
|
|||
|
TREE_VEC_ELT (t, i) = elts[i];
|
|||
|
|
|||
|
return t;
|
|||
|
}
|
|||
|
case POINTER_TYPE:
|
|||
|
case REFERENCE_TYPE:
|
|||
|
{
|
|||
|
tree r;
|
|||
|
enum tree_code code;
|
|||
|
if (type == TREE_TYPE (t))
|
|||
|
return t;
|
|||
|
|
|||
|
code = TREE_CODE (t);
|
|||
|
if (code == POINTER_TYPE)
|
|||
|
r = build_pointer_type (type);
|
|||
|
else
|
|||
|
r = build_reference_type (type);
|
|||
|
r = cp_build_type_variant (r, TYPE_READONLY (t), TYPE_VOLATILE (t));
|
|||
|
/* Will this ever be needed for TYPE_..._TO values? */
|
|||
|
layout_type (r);
|
|||
|
return r;
|
|||
|
}
|
|||
|
case OFFSET_TYPE:
|
|||
|
return build_offset_type
|
|||
|
(tsubst (TYPE_OFFSET_BASETYPE (t), args, nargs, in_decl), type);
|
|||
|
case FUNCTION_TYPE:
|
|||
|
case METHOD_TYPE:
|
|||
|
{
|
|||
|
tree values = TYPE_ARG_TYPES (t);
|
|||
|
tree context = TYPE_CONTEXT (t);
|
|||
|
tree raises = TYPE_RAISES_EXCEPTIONS (t);
|
|||
|
tree fntype;
|
|||
|
|
|||
|
/* Don't bother recursing if we know it won't change anything. */
|
|||
|
if (values != void_list_node)
|
|||
|
{
|
|||
|
/* This should probably be rewritten to use hash_tree_cons for
|
|||
|
the memory savings. */
|
|||
|
tree first = NULL_TREE;
|
|||
|
tree last;
|
|||
|
|
|||
|
for (; values && values != void_list_node;
|
|||
|
values = TREE_CHAIN (values))
|
|||
|
{
|
|||
|
tree value = TYPE_MAIN_VARIANT (type_decays_to
|
|||
|
(tsubst (TREE_VALUE (values), args, nargs, in_decl)));
|
|||
|
/* Don't instantiate default args unless they are used.
|
|||
|
Handle it in build_over_call instead. */
|
|||
|
tree purpose = TREE_PURPOSE (values);
|
|||
|
tree x = build_tree_list (purpose, value);
|
|||
|
|
|||
|
if (first)
|
|||
|
TREE_CHAIN (last) = x;
|
|||
|
else
|
|||
|
first = x;
|
|||
|
last = x;
|
|||
|
}
|
|||
|
|
|||
|
if (values == void_list_node)
|
|||
|
TREE_CHAIN (last) = void_list_node;
|
|||
|
|
|||
|
values = first;
|
|||
|
}
|
|||
|
if (context)
|
|||
|
context = tsubst (context, args, nargs, in_decl);
|
|||
|
/* Could also optimize cases where return value and
|
|||
|
values have common elements (e.g., T min(const &T, const T&). */
|
|||
|
|
|||
|
/* If the above parameters haven't changed, just return the type. */
|
|||
|
if (type == TREE_TYPE (t)
|
|||
|
&& values == TYPE_VALUES (t)
|
|||
|
&& context == TYPE_CONTEXT (t))
|
|||
|
return t;
|
|||
|
|
|||
|
/* Construct a new type node and return it. */
|
|||
|
if (TREE_CODE (t) == FUNCTION_TYPE
|
|||
|
&& context == NULL_TREE)
|
|||
|
{
|
|||
|
fntype = build_function_type (type, values);
|
|||
|
}
|
|||
|
else if (context == NULL_TREE)
|
|||
|
{
|
|||
|
tree base = tsubst (TREE_TYPE (TREE_VALUE (TYPE_ARG_TYPES (t))),
|
|||
|
args, nargs, in_decl);
|
|||
|
fntype = build_cplus_method_type (base, type,
|
|||
|
TREE_CHAIN (values));
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
fntype = make_node (TREE_CODE (t));
|
|||
|
TREE_TYPE (fntype) = type;
|
|||
|
TYPE_CONTEXT (fntype) = context;
|
|||
|
TYPE_VALUES (fntype) = values;
|
|||
|
TYPE_SIZE (fntype) = TYPE_SIZE (t);
|
|||
|
TYPE_ALIGN (fntype) = TYPE_ALIGN (t);
|
|||
|
TYPE_MODE (fntype) = TYPE_MODE (t);
|
|||
|
if (TYPE_METHOD_BASETYPE (t))
|
|||
|
TYPE_METHOD_BASETYPE (fntype) = tsubst (TYPE_METHOD_BASETYPE (t),
|
|||
|
args, nargs, in_decl);
|
|||
|
/* Need to generate hash value. */
|
|||
|
my_friendly_abort (84);
|
|||
|
}
|
|||
|
fntype = build_type_variant (fntype,
|
|||
|
TYPE_READONLY (t),
|
|||
|
TYPE_VOLATILE (t));
|
|||
|
if (raises)
|
|||
|
{
|
|||
|
raises = tsubst (raises, args, nargs, in_decl);
|
|||
|
fntype = build_exception_variant (fntype, raises);
|
|||
|
}
|
|||
|
return fntype;
|
|||
|
}
|
|||
|
case ARRAY_TYPE:
|
|||
|
{
|
|||
|
tree domain = tsubst (TYPE_DOMAIN (t), args, nargs, in_decl);
|
|||
|
tree r;
|
|||
|
if (type == TREE_TYPE (t) && domain == TYPE_DOMAIN (t))
|
|||
|
return t;
|
|||
|
r = build_cplus_array_type (type, domain);
|
|||
|
return r;
|
|||
|
}
|
|||
|
|
|||
|
case PLUS_EXPR:
|
|||
|
case MINUS_EXPR:
|
|||
|
return fold (build (TREE_CODE (t), TREE_TYPE (t),
|
|||
|
tsubst (TREE_OPERAND (t, 0), args, nargs, in_decl),
|
|||
|
tsubst (TREE_OPERAND (t, 1), args, nargs, in_decl)));
|
|||
|
|
|||
|
case NEGATE_EXPR:
|
|||
|
case NOP_EXPR:
|
|||
|
return fold (build1 (TREE_CODE (t), TREE_TYPE (t),
|
|||
|
tsubst (TREE_OPERAND (t, 0), args, nargs, in_decl)));
|
|||
|
|
|||
|
case TYPENAME_TYPE:
|
|||
|
{
|
|||
|
tree ctx = tsubst (TYPE_CONTEXT (t), args, nargs, in_decl);
|
|||
|
tree f = make_typename_type (ctx, TYPE_IDENTIFIER (t));
|
|||
|
return cp_build_type_variant
|
|||
|
(f, TYPE_READONLY (f) || TYPE_READONLY (t),
|
|||
|
TYPE_VOLATILE (f) || TYPE_VOLATILE (t));
|
|||
|
}
|
|||
|
|
|||
|
case INDIRECT_REF:
|
|||
|
return make_pointer_declarator
|
|||
|
(type, tsubst (TREE_OPERAND (t, 0), args, nargs, in_decl));
|
|||
|
|
|||
|
case ADDR_EXPR:
|
|||
|
return make_reference_declarator
|
|||
|
(type, tsubst (TREE_OPERAND (t, 0), args, nargs, in_decl));
|
|||
|
|
|||
|
case ARRAY_REF:
|
|||
|
return build_parse_node
|
|||
|
(ARRAY_REF, tsubst (TREE_OPERAND (t, 0), args, nargs, in_decl),
|
|||
|
tsubst_expr (TREE_OPERAND (t, 1), args, nargs, in_decl));
|
|||
|
|
|||
|
case CALL_EXPR:
|
|||
|
return make_call_declarator
|
|||
|
(tsubst (TREE_OPERAND (t, 0), args, nargs, in_decl),
|
|||
|
tsubst (TREE_OPERAND (t, 1), args, nargs, in_decl),
|
|||
|
TREE_OPERAND (t, 2),
|
|||
|
tsubst (TREE_TYPE (t), args, nargs, in_decl));
|
|||
|
|
|||
|
case SCOPE_REF:
|
|||
|
return build_parse_node
|
|||
|
(TREE_CODE (t), tsubst (TREE_OPERAND (t, 0), args, nargs, in_decl),
|
|||
|
tsubst (TREE_OPERAND (t, 1), args, nargs, in_decl));
|
|||
|
|
|||
|
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;
|
|||
|
|
|||
|
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, nargs, in_decl)
|
|||
|
tree t, args;
|
|||
|
int nargs;
|
|||
|
tree in_decl;
|
|||
|
{
|
|||
|
enum tree_code code;
|
|||
|
|
|||
|
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);
|
|||
|
|
|||
|
case CONST_DECL:
|
|||
|
case FIELD_DECL:
|
|||
|
if (DECL_CONTEXT (t))
|
|||
|
{
|
|||
|
tree ctx = tsubst (DECL_CONTEXT (t), args, nargs, in_decl);
|
|||
|
if (ctx == current_function_decl)
|
|||
|
return lookup_name (DECL_NAME (t), 0);
|
|||
|
else 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, nargs, in_decl);
|
|||
|
mark_used (t);
|
|||
|
return t;
|
|||
|
|
|||
|
case TEMPLATE_DECL:
|
|||
|
if (is_member_template (t))
|
|||
|
return tsubst (t, args, nargs, in_decl);
|
|||
|
else
|
|||
|
return t;
|
|||
|
|
|||
|
#if 0
|
|||
|
case IDENTIFIER_NODE:
|
|||
|
return do_identifier (t, 0);
|
|||
|
#endif
|
|||
|
|
|||
|
case CAST_EXPR:
|
|||
|
case REINTERPRET_CAST_EXPR:
|
|||
|
case CONST_CAST_EXPR:
|
|||
|
case STATIC_CAST_EXPR:
|
|||
|
case DYNAMIC_CAST_EXPR:
|
|||
|
return build1
|
|||
|
(code, tsubst (TREE_TYPE (t), args, nargs, in_decl),
|
|||
|
tsubst_copy (TREE_OPERAND (t, 0), args, nargs, 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 ARROW_EXPR:
|
|||
|
case THROW_EXPR:
|
|||
|
case TYPEID_EXPR:
|
|||
|
return build1
|
|||
|
(code, NULL_TREE,
|
|||
|
tsubst_copy (TREE_OPERAND (t, 0), args, nargs, 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, nargs, in_decl),
|
|||
|
tsubst_copy (TREE_OPERAND (t, 1), args, nargs, in_decl));
|
|||
|
|
|||
|
case CALL_EXPR:
|
|||
|
{
|
|||
|
tree fn = TREE_OPERAND (t, 0);
|
|||
|
if (really_overloaded_fn (fn))
|
|||
|
fn = tsubst_copy (get_first_fn (fn), args, nargs, in_decl);
|
|||
|
else
|
|||
|
fn = tsubst_copy (fn, args, nargs, in_decl);
|
|||
|
return build_nt
|
|||
|
(code, fn, tsubst_copy (TREE_OPERAND (t, 1), args, nargs, 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, nargs, in_decl);
|
|||
|
name = build1 (BIT_NOT_EXPR, NULL_TREE, TYPE_MAIN_VARIANT (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, nargs, in_decl);
|
|||
|
name = TREE_OPERAND (name, 1);
|
|||
|
name = tsubst_copy (TREE_OPERAND (name, 0), args, nargs, in_decl);
|
|||
|
name = build1 (BIT_NOT_EXPR, NULL_TREE, TYPE_MAIN_VARIANT (name));
|
|||
|
name = build_nt (SCOPE_REF, base, name);
|
|||
|
}
|
|||
|
else
|
|||
|
name = tsubst_copy (TREE_OPERAND (t, 0), args, nargs, in_decl);
|
|||
|
return build_nt
|
|||
|
(code, name, tsubst_copy (TREE_OPERAND (t, 1), args, nargs, in_decl),
|
|||
|
tsubst_copy (TREE_OPERAND (t, 2), args, nargs, in_decl),
|
|||
|
NULL_TREE);
|
|||
|
}
|
|||
|
|
|||
|
case COND_EXPR:
|
|||
|
case MODOP_EXPR:
|
|||
|
return build_nt
|
|||
|
(code, tsubst_copy (TREE_OPERAND (t, 0), args, nargs, in_decl),
|
|||
|
tsubst_copy (TREE_OPERAND (t, 1), args, nargs, in_decl),
|
|||
|
tsubst_copy (TREE_OPERAND (t, 2), args, nargs, in_decl));
|
|||
|
|
|||
|
case NEW_EXPR:
|
|||
|
{
|
|||
|
tree r = build_nt
|
|||
|
(code, tsubst_copy (TREE_OPERAND (t, 0), args, nargs, in_decl),
|
|||
|
tsubst_copy (TREE_OPERAND (t, 1), args, nargs, in_decl),
|
|||
|
tsubst_copy (TREE_OPERAND (t, 2), args, nargs, in_decl));
|
|||
|
NEW_EXPR_USE_GLOBAL (r) = NEW_EXPR_USE_GLOBAL (t);
|
|||
|
return r;
|
|||
|
}
|
|||
|
|
|||
|
case DELETE_EXPR:
|
|||
|
{
|
|||
|
tree r = build_nt
|
|||
|
(code, tsubst_copy (TREE_OPERAND (t, 0), args, nargs, in_decl),
|
|||
|
tsubst_copy (TREE_OPERAND (t, 1), args, nargs, 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, nargs, in_decl);
|
|||
|
tree chain;
|
|||
|
for (chain = targs; chain; chain = TREE_CHAIN (chain))
|
|||
|
TREE_VALUE (chain) = maybe_fold_nontype_arg (TREE_VALUE (chain));
|
|||
|
|
|||
|
return lookup_template_function
|
|||
|
(tsubst_copy (TREE_OPERAND (t, 0), args, nargs, 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, nargs, in_decl);
|
|||
|
value = TREE_VALUE (t);
|
|||
|
if (value)
|
|||
|
value = tsubst_copy (value, args, nargs, in_decl);
|
|||
|
chain = TREE_CHAIN (t);
|
|||
|
if (chain && chain != void_type_node)
|
|||
|
chain = tsubst_copy (chain, args, nargs, 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_CONST_PARM:
|
|||
|
case POINTER_TYPE:
|
|||
|
case REFERENCE_TYPE:
|
|||
|
case OFFSET_TYPE:
|
|||
|
case FUNCTION_TYPE:
|
|||
|
case METHOD_TYPE:
|
|||
|
case ARRAY_TYPE:
|
|||
|
case TYPENAME_TYPE:
|
|||
|
return tsubst (t, args, nargs, in_decl);
|
|||
|
|
|||
|
case IDENTIFIER_NODE:
|
|||
|
if (IDENTIFIER_TYPENAME_P (t))
|
|||
|
return build_typename_overload
|
|||
|
(tsubst (TREE_TYPE (t), args, nargs, in_decl));
|
|||
|
else
|
|||
|
return t;
|
|||
|
|
|||
|
case CONSTRUCTOR:
|
|||
|
return build
|
|||
|
(CONSTRUCTOR, tsubst (TREE_TYPE (t), args, nargs, in_decl), NULL_TREE,
|
|||
|
tsubst_copy (CONSTRUCTOR_ELTS (t), args, nargs, in_decl));
|
|||
|
|
|||
|
default:
|
|||
|
return t;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/* Like tsubst_copy, but also does semantic processing and RTL expansion. */
|
|||
|
|
|||
|
tree
|
|||
|
tsubst_expr (t, args, nargs, in_decl)
|
|||
|
tree t, args;
|
|||
|
int nargs;
|
|||
|
tree in_decl;
|
|||
|
{
|
|||
|
if (t == NULL_TREE || t == error_mark_node)
|
|||
|
return t;
|
|||
|
|
|||
|
if (processing_template_decl)
|
|||
|
return tsubst_copy (t, args, nargs, in_decl);
|
|||
|
|
|||
|
switch (TREE_CODE (t))
|
|||
|
{
|
|||
|
case RETURN_STMT:
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
c_expand_return
|
|||
|
(tsubst_expr (TREE_OPERAND (t, 0), args, nargs, in_decl));
|
|||
|
finish_stmt ();
|
|||
|
break;
|
|||
|
|
|||
|
case EXPR_STMT:
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
t = tsubst_expr (TREE_OPERAND (t, 0), args, nargs, in_decl);
|
|||
|
/* Do default conversion if safe and possibly important,
|
|||
|
in case within ({...}). */
|
|||
|
if ((TREE_CODE (TREE_TYPE (t)) == ARRAY_TYPE && lvalue_p (t))
|
|||
|
|| TREE_CODE (TREE_TYPE (t)) == FUNCTION_TYPE)
|
|||
|
t = default_conversion (t);
|
|||
|
cplus_expand_expr_stmt (t);
|
|||
|
clear_momentary ();
|
|||
|
finish_stmt ();
|
|||
|
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, nargs, in_decl),
|
|||
|
tsubst (TREE_OPERAND (t, 1), args, nargs, in_decl),
|
|||
|
TREE_OPERAND (t, 2) != 0);
|
|||
|
init = tsubst_expr (TREE_OPERAND (t, 2), args, nargs, in_decl);
|
|||
|
cp_finish_decl
|
|||
|
(dcl, init, NULL_TREE, 1, /*init ? LOOKUP_ONLYCONVERTING :*/ 0);
|
|||
|
resume_momentary (i);
|
|||
|
return dcl;
|
|||
|
}
|
|||
|
|
|||
|
case FOR_STMT:
|
|||
|
{
|
|||
|
tree tmp;
|
|||
|
int init_scope = (flag_new_for_scope > 0 && TREE_OPERAND (t, 0)
|
|||
|
&& TREE_CODE (TREE_OPERAND (t, 0)) == DECL_STMT);
|
|||
|
int cond_scope = (TREE_OPERAND (t, 1)
|
|||
|
&& TREE_CODE (TREE_OPERAND (t, 1)) == DECL_STMT);
|
|||
|
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
if (init_scope)
|
|||
|
do_pushlevel ();
|
|||
|
for (tmp = TREE_OPERAND (t, 0); tmp; tmp = TREE_CHAIN (tmp))
|
|||
|
tsubst_expr (tmp, args, nargs, in_decl);
|
|||
|
emit_nop ();
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
expand_start_loop_continue_elsewhere (1);
|
|||
|
|
|||
|
if (cond_scope)
|
|||
|
do_pushlevel ();
|
|||
|
tmp = tsubst_expr (TREE_OPERAND (t, 1), args, nargs, in_decl);
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
if (tmp)
|
|||
|
expand_exit_loop_if_false (0, condition_conversion (tmp));
|
|||
|
|
|||
|
if (! cond_scope)
|
|||
|
do_pushlevel ();
|
|||
|
tsubst_expr (TREE_OPERAND (t, 3), args, nargs, in_decl);
|
|||
|
do_poplevel ();
|
|||
|
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
expand_loop_continue_here ();
|
|||
|
tmp = tsubst_expr (TREE_OPERAND (t, 2), args, nargs, in_decl);
|
|||
|
if (tmp)
|
|||
|
cplus_expand_expr_stmt (tmp);
|
|||
|
|
|||
|
expand_end_loop ();
|
|||
|
if (init_scope)
|
|||
|
do_poplevel ();
|
|||
|
finish_stmt ();
|
|||
|
}
|
|||
|
break;
|
|||
|
|
|||
|
case WHILE_STMT:
|
|||
|
{
|
|||
|
tree cond;
|
|||
|
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
emit_nop ();
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
expand_start_loop (1);
|
|||
|
|
|||
|
cond = TREE_OPERAND (t, 0);
|
|||
|
if (TREE_CODE (cond) == DECL_STMT)
|
|||
|
do_pushlevel ();
|
|||
|
cond = tsubst_expr (cond, args, nargs, in_decl);
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
expand_exit_loop_if_false (0, condition_conversion (cond));
|
|||
|
|
|||
|
if (TREE_CODE (TREE_OPERAND (t, 0)) != DECL_STMT)
|
|||
|
do_pushlevel ();
|
|||
|
tsubst_expr (TREE_OPERAND (t, 1), args, nargs, in_decl);
|
|||
|
do_poplevel ();
|
|||
|
|
|||
|
expand_end_loop ();
|
|||
|
finish_stmt ();
|
|||
|
}
|
|||
|
break;
|
|||
|
|
|||
|
case DO_STMT:
|
|||
|
{
|
|||
|
tree cond;
|
|||
|
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
emit_nop ();
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
expand_start_loop_continue_elsewhere (1);
|
|||
|
|
|||
|
tsubst_expr (TREE_OPERAND (t, 0), args, nargs, in_decl);
|
|||
|
expand_loop_continue_here ();
|
|||
|
|
|||
|
cond = tsubst_expr (TREE_OPERAND (t, 1), args, nargs, in_decl);
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
expand_exit_loop_if_false (0, condition_conversion (cond));
|
|||
|
expand_end_loop ();
|
|||
|
|
|||
|
clear_momentary ();
|
|||
|
finish_stmt ();
|
|||
|
}
|
|||
|
break;
|
|||
|
|
|||
|
case IF_STMT:
|
|||
|
{
|
|||
|
tree tmp;
|
|||
|
int cond_scope = (TREE_CODE (TREE_OPERAND (t, 0)) == DECL_STMT);
|
|||
|
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
if (cond_scope)
|
|||
|
do_pushlevel ();
|
|||
|
tmp = tsubst_expr (TREE_OPERAND (t, 0), args, nargs, in_decl);
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
expand_start_cond (condition_conversion (tmp), 0);
|
|||
|
|
|||
|
if (tmp = TREE_OPERAND (t, 1), tmp)
|
|||
|
tsubst_expr (tmp, args, nargs, in_decl);
|
|||
|
|
|||
|
if (tmp = TREE_OPERAND (t, 2), tmp)
|
|||
|
{
|
|||
|
expand_start_else ();
|
|||
|
tsubst_expr (tmp, args, nargs, in_decl);
|
|||
|
}
|
|||
|
|
|||
|
expand_end_cond ();
|
|||
|
|
|||
|
if (cond_scope)
|
|||
|
do_poplevel ();
|
|||
|
|
|||
|
finish_stmt ();
|
|||
|
}
|
|||
|
break;
|
|||
|
|
|||
|
case COMPOUND_STMT:
|
|||
|
{
|
|||
|
tree substmt = TREE_OPERAND (t, 0);
|
|||
|
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
|
|||
|
if (COMPOUND_STMT_NO_SCOPE (t) == 0)
|
|||
|
do_pushlevel ();
|
|||
|
|
|||
|
for (; substmt; substmt = TREE_CHAIN (substmt))
|
|||
|
tsubst_expr (substmt, args, nargs, in_decl);
|
|||
|
|
|||
|
if (COMPOUND_STMT_NO_SCOPE (t) == 0)
|
|||
|
do_poplevel ();
|
|||
|
}
|
|||
|
break;
|
|||
|
|
|||
|
case BREAK_STMT:
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
if (! expand_exit_something ())
|
|||
|
error ("break statement not within loop or switch");
|
|||
|
break;
|
|||
|
|
|||
|
case CONTINUE_STMT:
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
if (! expand_continue_loop (0))
|
|||
|
error ("continue statement not within a loop");
|
|||
|
break;
|
|||
|
|
|||
|
case SWITCH_STMT:
|
|||
|
{
|
|||
|
tree val, tmp;
|
|||
|
int cond_scope = (TREE_CODE (TREE_OPERAND (t, 0)) == DECL_STMT);
|
|||
|
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
if (cond_scope)
|
|||
|
do_pushlevel ();
|
|||
|
val = tsubst_expr (TREE_OPERAND (t, 0), args, nargs, in_decl);
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
c_expand_start_case (val);
|
|||
|
push_switch ();
|
|||
|
|
|||
|
if (tmp = TREE_OPERAND (t, 1), tmp)
|
|||
|
tsubst_expr (tmp, args, nargs, in_decl);
|
|||
|
|
|||
|
expand_end_case (val);
|
|||
|
pop_switch ();
|
|||
|
|
|||
|
if (cond_scope)
|
|||
|
do_poplevel ();
|
|||
|
|
|||
|
finish_stmt ();
|
|||
|
}
|
|||
|
break;
|
|||
|
|
|||
|
case CASE_LABEL:
|
|||
|
do_case (tsubst_expr (TREE_OPERAND (t, 0), args, nargs, in_decl),
|
|||
|
tsubst_expr (TREE_OPERAND (t, 1), args, nargs, 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);
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
if (TREE_CODE (TREE_OPERAND (t, 0)) == IDENTIFIER_NODE)
|
|||
|
{
|
|||
|
tree decl = lookup_label (TREE_OPERAND (t, 0));
|
|||
|
TREE_USED (decl) = 1;
|
|||
|
expand_goto (decl);
|
|||
|
}
|
|||
|
else
|
|||
|
expand_computed_goto
|
|||
|
(tsubst_expr (TREE_OPERAND (t, 0), args, nargs, in_decl));
|
|||
|
break;
|
|||
|
|
|||
|
case TRY_BLOCK:
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
emit_line_note (input_filename, lineno);
|
|||
|
expand_start_try_stmts ();
|
|||
|
tsubst_expr (TREE_OPERAND (t, 0), args, nargs, in_decl);
|
|||
|
expand_start_all_catch ();
|
|||
|
{
|
|||
|
tree handler = TREE_OPERAND (t, 1);
|
|||
|
for (; handler; handler = TREE_CHAIN (handler))
|
|||
|
tsubst_expr (handler, args, nargs, in_decl);
|
|||
|
}
|
|||
|
expand_end_all_catch ();
|
|||
|
break;
|
|||
|
|
|||
|
case HANDLER:
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
do_pushlevel ();
|
|||
|
if (TREE_OPERAND (t, 0))
|
|||
|
{
|
|||
|
tree d = TREE_OPERAND (t, 0);
|
|||
|
expand_start_catch_block
|
|||
|
(tsubst (TREE_OPERAND (d, 1), args, nargs, in_decl),
|
|||
|
tsubst (TREE_OPERAND (d, 0), args, nargs, in_decl));
|
|||
|
}
|
|||
|
else
|
|||
|
expand_start_catch_block (NULL_TREE, NULL_TREE);
|
|||
|
tsubst_expr (TREE_OPERAND (t, 1), args, nargs, in_decl);
|
|||
|
expand_end_catch_block ();
|
|||
|
do_poplevel ();
|
|||
|
break;
|
|||
|
|
|||
|
case TAG_DEFN:
|
|||
|
lineno = TREE_COMPLEXITY (t);
|
|||
|
t = TREE_TYPE (t);
|
|||
|
if (TREE_CODE (t) == ENUMERAL_TYPE)
|
|||
|
tsubst_enum (t, args, nargs, NULL);
|
|||
|
break;
|
|||
|
|
|||
|
default:
|
|||
|
return build_expr_from_tree (tsubst_copy (t, args, nargs, in_decl));
|
|||
|
}
|
|||
|
return NULL_TREE;
|
|||
|
}
|
|||
|
|
|||
|
tree
|
|||
|
instantiate_template (tmpl, targ_ptr)
|
|||
|
tree tmpl, targ_ptr;
|
|||
|
{
|
|||
|
tree fndecl;
|
|||
|
int i, len;
|
|||
|
struct obstack *old_fmp_obstack;
|
|||
|
extern struct obstack *function_maybepermanent_obstack;
|
|||
|
|
|||
|
my_friendly_assert (TREE_CODE (tmpl) == TEMPLATE_DECL, 283);
|
|||
|
|
|||
|
if (DECL_FUNCTION_TEMPLATE_P (tmpl))
|
|||
|
{
|
|||
|
tree specs;
|
|||
|
|
|||
|
/* Check to see if there is a matching specialization. */
|
|||
|
for (specs = DECL_TEMPLATE_SPECIALIZATIONS (tmpl);
|
|||
|
specs != NULL_TREE;
|
|||
|
specs = TREE_CHAIN (specs))
|
|||
|
if (comp_template_args (TREE_PURPOSE (specs), targ_ptr))
|
|||
|
return TREE_VALUE (specs);
|
|||
|
}
|
|||
|
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
old_fmp_obstack = function_maybepermanent_obstack;
|
|||
|
function_maybepermanent_obstack = &permanent_obstack;
|
|||
|
|
|||
|
len = DECL_NTPARMS (tmpl);
|
|||
|
|
|||
|
i = len;
|
|||
|
while (i--)
|
|||
|
{
|
|||
|
tree t = TREE_VEC_ELT (targ_ptr, 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'", tmpl);
|
|||
|
fndecl = error_mark_node;
|
|||
|
goto out;
|
|||
|
}
|
|||
|
}
|
|||
|
TREE_VEC_ELT (targ_ptr, i) = copy_to_permanent (t);
|
|||
|
}
|
|||
|
targ_ptr = copy_to_permanent (targ_ptr);
|
|||
|
|
|||
|
if (DECL_TEMPLATE_INFO (tmpl) && DECL_TI_ARGS (tmpl))
|
|||
|
targ_ptr = add_to_template_args (DECL_TI_ARGS (tmpl), targ_ptr);
|
|||
|
|
|||
|
/* substitute template parameters */
|
|||
|
fndecl = tsubst (DECL_RESULT (tmpl), targ_ptr, len, 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);
|
|||
|
}
|
|||
|
|
|||
|
/* Like type_unification but designed specially to handle conversion
|
|||
|
operators. */
|
|||
|
|
|||
|
int
|
|||
|
fn_type_unification (fn, explicit_targs, targs, args, return_type, strict)
|
|||
|
tree fn, explicit_targs, targs, args, return_type;
|
|||
|
int strict;
|
|||
|
{
|
|||
|
int i, dummy = 0;
|
|||
|
tree fn_arg_types = TYPE_ARG_TYPES (TREE_TYPE (fn));
|
|||
|
tree decl_arg_types = args;
|
|||
|
|
|||
|
my_friendly_assert (TREE_CODE (fn) == TEMPLATE_DECL, 0);
|
|||
|
|
|||
|
if (IDENTIFIER_TYPENAME_P (DECL_NAME (fn)))
|
|||
|
{
|
|||
|
/* This is a template conversion operator. Use the return types
|
|||
|
as well as the argument types. */
|
|||
|
fn_arg_types = scratch_tree_cons (NULL_TREE,
|
|||
|
TREE_TYPE (TREE_TYPE (fn)),
|
|||
|
fn_arg_types);
|
|||
|
decl_arg_types = scratch_tree_cons (NULL_TREE,
|
|||
|
return_type,
|
|||
|
decl_arg_types);
|
|||
|
}
|
|||
|
|
|||
|
i = type_unification (DECL_INNERMOST_TEMPLATE_PARMS (fn),
|
|||
|
&TREE_VEC_ELT (targs, 0),
|
|||
|
fn_arg_types,
|
|||
|
decl_arg_types,
|
|||
|
explicit_targs,
|
|||
|
&dummy, strict, 0);
|
|||
|
|
|||
|
return i;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/* Type unification.
|
|||
|
|
|||
|
We have a function template signature with one or more references to
|
|||
|
template parameters, and a parameter list we wish to fit to this
|
|||
|
template. If possible, produce a list of parameters for the template
|
|||
|
which will cause it to fit the supplied parameter list.
|
|||
|
|
|||
|
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.
|
|||
|
|
|||
|
TPARMS[NTPARMS] is an array of template parameter types;
|
|||
|
TARGS[NTPARMS] is the array of template parameter values. PARMS is
|
|||
|
the function template's signature (using TEMPLATE_PARM_IDX nodes),
|
|||
|
and ARGS is the argument list we're trying to match against it.
|
|||
|
|
|||
|
If SUBR is 1, we're being called recursively (to unify the arguments of
|
|||
|
a function or method parameter of a function template), so don't zero
|
|||
|
out targs and don't fail on an incomplete match.
|
|||
|
|
|||
|
If STRICT is 1, the match must be exact (for casts of overloaded
|
|||
|
addresses, explicit instantiation, and more_specialized). */
|
|||
|
|
|||
|
int
|
|||
|
type_unification (tparms, targs, parms, args, targs_in, nsubsts,
|
|||
|
strict, allow_incomplete)
|
|||
|
tree tparms, *targs, parms, args, targs_in;
|
|||
|
int *nsubsts, strict, allow_incomplete;
|
|||
|
{
|
|||
|
int ntparms = TREE_VEC_LENGTH (tparms);
|
|||
|
tree t;
|
|||
|
int i;
|
|||
|
int r;
|
|||
|
|
|||
|
bzero ((char *) targs, sizeof (tree) * ntparms);
|
|||
|
|
|||
|
/* Insert any explicit template arguments. They are encoded as the
|
|||
|
operands of NOP_EXPRs so that unify can tell that they are
|
|||
|
explicit arguments. */
|
|||
|
for (i = 0, t = targs_in; t != NULL_TREE; t = TREE_CHAIN (t), ++i)
|
|||
|
targs[i] = build1 (NOP_EXPR, NULL_TREE, TREE_VALUE (t));
|
|||
|
|
|||
|
r = type_unification_real (tparms, targs, parms, args, nsubsts, 0,
|
|||
|
strict, allow_incomplete);
|
|||
|
|
|||
|
for (i = 0, t = targs_in; t != NULL_TREE; t = TREE_CHAIN (t), ++i)
|
|||
|
if (TREE_CODE (targs[i]) == NOP_EXPR)
|
|||
|
targs[i] = TREE_OPERAND (targs[i], 0);
|
|||
|
|
|||
|
return r;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
static int
|
|||
|
type_unification_real (tparms, targs, parms, args, nsubsts, subr,
|
|||
|
strict, allow_incomplete)
|
|||
|
tree tparms, *targs, parms, args;
|
|||
|
int *nsubsts, subr, strict, allow_incomplete;
|
|||
|
{
|
|||
|
tree parm, arg;
|
|||
|
int i;
|
|||
|
int ntparms = TREE_VEC_LENGTH (tparms);
|
|||
|
|
|||
|
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);
|
|||
|
|
|||
|
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)
|
|||
|
return 1;
|
|||
|
|
|||
|
/* 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)
|
|||
|
{
|
|||
|
if (comptypes (parm, type, 1))
|
|||
|
continue;
|
|||
|
}
|
|||
|
else if (arg)
|
|||
|
{
|
|||
|
if (can_convert_arg (parm, type, arg))
|
|||
|
continue;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
if (can_convert (parm, type))
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
return 1;
|
|||
|
}
|
|||
|
|
|||
|
#if 0
|
|||
|
if (TREE_CODE (arg) == VAR_DECL)
|
|||
|
arg = TREE_TYPE (arg);
|
|||
|
else if (TREE_CODE_CLASS (TREE_CODE (arg)) == 'e')
|
|||
|
arg = TREE_TYPE (arg);
|
|||
|
#else
|
|||
|
if (TREE_CODE_CLASS (TREE_CODE (arg)) != 't')
|
|||
|
{
|
|||
|
my_friendly_assert (TREE_TYPE (arg) != NULL_TREE, 293);
|
|||
|
if (TREE_CODE (arg) == TREE_LIST
|
|||
|
&& TREE_TYPE (arg) == unknown_type_node
|
|||
|
&& TREE_CODE (TREE_VALUE (arg)) == TEMPLATE_DECL)
|
|||
|
{
|
|||
|
int nsubsts, ntparms;
|
|||
|
tree *targs;
|
|||
|
|
|||
|
/* Have to back unify here */
|
|||
|
arg = TREE_VALUE (arg);
|
|||
|
nsubsts = 0;
|
|||
|
ntparms = DECL_NTPARMS (arg);
|
|||
|
targs = (tree *) alloca (sizeof (tree) * ntparms);
|
|||
|
parm = expr_tree_cons (NULL_TREE, parm, NULL_TREE);
|
|||
|
return
|
|||
|
type_unification (DECL_INNERMOST_TEMPLATE_PARMS (arg),
|
|||
|
targs,
|
|||
|
TYPE_ARG_TYPES (TREE_TYPE (arg)),
|
|||
|
parm, NULL_TREE, &nsubsts, strict,
|
|||
|
allow_incomplete);
|
|||
|
}
|
|||
|
arg = TREE_TYPE (arg);
|
|||
|
}
|
|||
|
#endif
|
|||
|
if (! subr && TREE_CODE (arg) == REFERENCE_TYPE)
|
|||
|
arg = TREE_TYPE (arg);
|
|||
|
|
|||
|
if (! subr && TREE_CODE (parm) != REFERENCE_TYPE)
|
|||
|
{
|
|||
|
if (TREE_CODE (arg) == FUNCTION_TYPE
|
|||
|
|| TREE_CODE (arg) == METHOD_TYPE)
|
|||
|
arg = build_pointer_type (arg);
|
|||
|
else if (TREE_CODE (arg) == ARRAY_TYPE)
|
|||
|
arg = build_pointer_type (TREE_TYPE (arg));
|
|||
|
else
|
|||
|
arg = TYPE_MAIN_VARIANT (arg);
|
|||
|
}
|
|||
|
|
|||
|
switch (unify (tparms, targs, ntparms, parm, arg, nsubsts, 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 (!targs[i])
|
|||
|
{
|
|||
|
if (!allow_incomplete)
|
|||
|
error ("incomplete type unification");
|
|||
|
return 2;
|
|||
|
}
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|
|||
|
/* Tail recursion is your friend. */
|
|||
|
|
|||
|
static int
|
|||
|
unify (tparms, targs, ntparms, parm, arg, nsubsts, strict)
|
|||
|
tree tparms, *targs, parm, arg;
|
|||
|
int *nsubsts, ntparms, strict;
|
|||
|
{
|
|||
|
int idx;
|
|||
|
|
|||
|
/* 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)
|
|||
|
return 1;
|
|||
|
if (arg == parm)
|
|||
|
return 0;
|
|||
|
|
|||
|
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:
|
|||
|
(*nsubsts)++;
|
|||
|
idx = TEMPLATE_TYPE_IDX (parm);
|
|||
|
/* Check for mixed types and values. */
|
|||
|
if (TREE_CODE (TREE_VALUE (TREE_VEC_ELT (tparms, idx))) != TYPE_DECL)
|
|||
|
return 1;
|
|||
|
|
|||
|
if (!strict && targs[idx] != NULL_TREE &&
|
|||
|
TREE_CODE (targs[idx]) == NOP_EXPR)
|
|||
|
/* An explicit template argument. Don't even try to match
|
|||
|
here; the overload resolution code will manage check to
|
|||
|
see whether the call is legal. */
|
|||
|
return 0;
|
|||
|
|
|||
|
if (strict && (TYPE_READONLY (arg) < TYPE_READONLY (parm)
|
|||
|
|| TYPE_VOLATILE (arg) < TYPE_VOLATILE (parm)))
|
|||
|
return 1;
|
|||
|
#if 0
|
|||
|
/* Template type parameters cannot contain cv-quals; i.e.
|
|||
|
template <class T> void f (T& a, T& b) will not generate
|
|||
|
void f (const int& a, const int& b). */
|
|||
|
if (TYPE_READONLY (arg) > TYPE_READONLY (parm)
|
|||
|
|| TYPE_VOLATILE (arg) > TYPE_VOLATILE (parm))
|
|||
|
return 1;
|
|||
|
arg = TYPE_MAIN_VARIANT (arg);
|
|||
|
#else
|
|||
|
{
|
|||
|
int constp = TYPE_READONLY (arg) > TYPE_READONLY (parm);
|
|||
|
int volatilep = TYPE_VOLATILE (arg) > TYPE_VOLATILE (parm);
|
|||
|
arg = cp_build_type_variant (arg, constp, volatilep);
|
|||
|
}
|
|||
|
#endif
|
|||
|
/* Simple cases: Value already set, does match or doesn't. */
|
|||
|
if (targs[idx] == arg
|
|||
|
|| (targs[idx]
|
|||
|
&& TREE_CODE (targs[idx]) == NOP_EXPR
|
|||
|
&& TREE_OPERAND (targs[idx], 0) == arg))
|
|||
|
return 0;
|
|||
|
else if (targs[idx])
|
|||
|
return 1;
|
|||
|
targs[idx] = arg;
|
|||
|
return 0;
|
|||
|
case TEMPLATE_CONST_PARM:
|
|||
|
(*nsubsts)++;
|
|||
|
idx = TEMPLATE_CONST_IDX (parm);
|
|||
|
if (targs[idx])
|
|||
|
{
|
|||
|
int i = cp_tree_equal (targs[idx], arg);
|
|||
|
if (i == 1)
|
|||
|
return 0;
|
|||
|
else if (i == 0)
|
|||
|
return 1;
|
|||
|
else
|
|||
|
my_friendly_abort (42);
|
|||
|
}
|
|||
|
|
|||
|
targs[idx] = copy_to_permanent (arg);
|
|||
|
return 0;
|
|||
|
|
|||
|
case POINTER_TYPE:
|
|||
|
if (TREE_CODE (arg) == RECORD_TYPE && TYPE_PTRMEMFUNC_FLAG (arg))
|
|||
|
return unify (tparms, targs, ntparms, parm,
|
|||
|
TYPE_PTRMEMFUNC_FN_TYPE (arg), nsubsts, strict);
|
|||
|
|
|||
|
if (TREE_CODE (arg) != POINTER_TYPE)
|
|||
|
return 1;
|
|||
|
return unify (tparms, targs, ntparms, TREE_TYPE (parm), TREE_TYPE (arg),
|
|||
|
nsubsts, strict);
|
|||
|
|
|||
|
case REFERENCE_TYPE:
|
|||
|
if (TREE_CODE (arg) == REFERENCE_TYPE)
|
|||
|
arg = TREE_TYPE (arg);
|
|||
|
return unify (tparms, targs, ntparms, TREE_TYPE (parm), arg,
|
|||
|
nsubsts, strict);
|
|||
|
|
|||
|
case ARRAY_TYPE:
|
|||
|
if (TREE_CODE (arg) != ARRAY_TYPE)
|
|||
|
return 1;
|
|||
|
if (unify (tparms, targs, ntparms, TYPE_DOMAIN (parm), TYPE_DOMAIN (arg),
|
|||
|
nsubsts, strict) != 0)
|
|||
|
return 1;
|
|||
|
return unify (tparms, targs, ntparms, TREE_TYPE (parm), TREE_TYPE (arg),
|
|||
|
nsubsts, strict);
|
|||
|
|
|||
|
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)
|
|||
|
{
|
|||
|
if (TYPE_MIN_VALUE (parm) && TYPE_MIN_VALUE (arg)
|
|||
|
&& unify (tparms, targs, ntparms, TYPE_MIN_VALUE (parm),
|
|||
|
TYPE_MIN_VALUE (arg), nsubsts, strict))
|
|||
|
return 1;
|
|||
|
if (TYPE_MAX_VALUE (parm) && TYPE_MAX_VALUE (arg)
|
|||
|
&& unify (tparms, targs, ntparms, TYPE_MAX_VALUE (parm),
|
|||
|
TYPE_MAX_VALUE (arg), nsubsts, strict))
|
|||
|
return 1;
|
|||
|
}
|
|||
|
else if (TREE_CODE (parm) == REAL_TYPE
|
|||
|
&& 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 MINUS_EXPR:
|
|||
|
{
|
|||
|
tree t1, t2;
|
|||
|
t1 = TREE_OPERAND (parm, 0);
|
|||
|
t2 = TREE_OPERAND (parm, 1);
|
|||
|
return unify (tparms, targs, ntparms, t1,
|
|||
|
fold (build (PLUS_EXPR, integer_type_node, arg, t2)),
|
|||
|
nsubsts, strict);
|
|||
|
}
|
|||
|
|
|||
|
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, ntparms,
|
|||
|
TREE_VEC_ELT (parm, i), TREE_VEC_ELT (arg, i),
|
|||
|
nsubsts, strict))
|
|||
|
return 1;
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|
|||
|
case RECORD_TYPE:
|
|||
|
if (TYPE_PTRMEMFUNC_FLAG (parm))
|
|||
|
return unify (tparms, targs, ntparms, TYPE_PTRMEMFUNC_FN_TYPE (parm),
|
|||
|
arg, nsubsts, strict);
|
|||
|
|
|||
|
/* Allow trivial conversions. */
|
|||
|
if (TREE_CODE (arg) != RECORD_TYPE
|
|||
|
|| TYPE_READONLY (parm) < TYPE_READONLY (arg)
|
|||
|
|| TYPE_VOLATILE (parm) < TYPE_VOLATILE (arg))
|
|||
|
return 1;
|
|||
|
|
|||
|
if (CLASSTYPE_TEMPLATE_INFO (parm) && uses_template_parms (parm))
|
|||
|
{
|
|||
|
tree t = NULL_TREE;
|
|||
|
if (flag_ansi_overloading && ! strict)
|
|||
|
t = get_template_base (CLASSTYPE_TI_TEMPLATE (parm), arg);
|
|||
|
else if
|
|||
|
(CLASSTYPE_TEMPLATE_INFO (arg)
|
|||
|
&& CLASSTYPE_TI_TEMPLATE (parm) == CLASSTYPE_TI_TEMPLATE (arg))
|
|||
|
t = arg;
|
|||
|
if (! t || t == error_mark_node)
|
|||
|
return 1;
|
|||
|
|
|||
|
return unify (tparms, targs, ntparms, CLASSTYPE_TI_ARGS (parm),
|
|||
|
CLASSTYPE_TI_ARGS (t), nsubsts, strict);
|
|||
|
}
|
|||
|
else if (TYPE_MAIN_VARIANT (parm) != TYPE_MAIN_VARIANT (arg))
|
|||
|
return 1;
|
|||
|
return 0;
|
|||
|
|
|||
|
case METHOD_TYPE:
|
|||
|
if (TREE_CODE (arg) != METHOD_TYPE)
|
|||
|
return 1;
|
|||
|
goto check_args;
|
|||
|
|
|||
|
case FUNCTION_TYPE:
|
|||
|
if (TREE_CODE (arg) != FUNCTION_TYPE)
|
|||
|
return 1;
|
|||
|
check_args:
|
|||
|
if (unify (tparms, targs, ntparms, TREE_TYPE (parm),
|
|||
|
TREE_TYPE (arg), nsubsts, strict))
|
|||
|
return 1;
|
|||
|
return type_unification_real (tparms, targs, TYPE_ARG_TYPES (parm),
|
|||
|
TYPE_ARG_TYPES (arg), nsubsts, 1,
|
|||
|
strict, 0);
|
|||
|
|
|||
|
case OFFSET_TYPE:
|
|||
|
if (TREE_CODE (arg) != OFFSET_TYPE)
|
|||
|
return 1;
|
|||
|
if (unify (tparms, targs, ntparms, TYPE_OFFSET_BASETYPE (parm),
|
|||
|
TYPE_OFFSET_BASETYPE (arg), nsubsts, strict))
|
|||
|
return 1;
|
|||
|
return unify (tparms, targs, ntparms, TREE_TYPE (parm),
|
|||
|
TREE_TYPE (arg), nsubsts, strict);
|
|||
|
|
|||
|
case CONST_DECL:
|
|||
|
if (arg != decl_constant_value (parm))
|
|||
|
return 1;
|
|||
|
return 0;
|
|||
|
|
|||
|
default:
|
|||
|
sorry ("use of `%s' in template type unification",
|
|||
|
tree_code_name [(int) TREE_CODE (parm)]);
|
|||
|
return 1;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
void
|
|||
|
mark_decl_instantiated (result, extern_p)
|
|||
|
tree result;
|
|||
|
int extern_p;
|
|||
|
{
|
|||
|
if (DECL_TEMPLATE_INSTANTIATION (result))
|
|||
|
SET_DECL_EXPLICIT_INSTANTIATION (result);
|
|||
|
TREE_PUBLIC (result) = 1;
|
|||
|
|
|||
|
if (! extern_p)
|
|||
|
{
|
|||
|
DECL_INTERFACE_KNOWN (result) = 1;
|
|||
|
DECL_NOT_REALLY_EXTERN (result) = 1;
|
|||
|
|
|||
|
/* For WIN32 we also want to put explicit instantiations in
|
|||
|
linkonce sections. */
|
|||
|
if (supports_one_only () && ! SUPPORTS_WEAK)
|
|||
|
comdat_linkage (result);
|
|||
|
}
|
|||
|
else if (TREE_CODE (result) == FUNCTION_DECL)
|
|||
|
mark_inline_for_output (result);
|
|||
|
}
|
|||
|
|
|||
|
/* Given two function templates PAT1 and PAT2, 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)
|
|||
|
tree pat1, pat2;
|
|||
|
{
|
|||
|
tree targs;
|
|||
|
int winner = 0;
|
|||
|
|
|||
|
targs = get_bindings (pat1, pat2);
|
|||
|
if (targs)
|
|||
|
{
|
|||
|
--winner;
|
|||
|
}
|
|||
|
|
|||
|
targs = get_bindings (pat2, pat1);
|
|||
|
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. */
|
|||
|
|
|||
|
tree
|
|||
|
get_bindings (fn, decl)
|
|||
|
tree fn, decl;
|
|||
|
{
|
|||
|
int ntparms = DECL_NTPARMS (fn);
|
|||
|
tree targs = make_scratch_vec (ntparms);
|
|||
|
int i;
|
|||
|
|
|||
|
i = fn_type_unification (fn, NULL_TREE, targs,
|
|||
|
TYPE_ARG_TYPES (TREE_TYPE (decl)),
|
|||
|
TREE_TYPE (TREE_TYPE (decl)),
|
|||
|
1);
|
|||
|
|
|||
|
if (i == 0)
|
|||
|
return targs;
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|
|||
|
static tree
|
|||
|
get_class_bindings (tparms, parms, args)
|
|||
|
tree tparms, parms, args;
|
|||
|
{
|
|||
|
int i, dummy, ntparms = TREE_VEC_LENGTH (tparms);
|
|||
|
tree vec = make_temp_vec (ntparms);
|
|||
|
|
|||
|
for (i = 0; i < TREE_VEC_LENGTH (parms); ++i)
|
|||
|
{
|
|||
|
switch (unify (tparms, &TREE_VEC_ELT (vec, 0), ntparms,
|
|||
|
TREE_VEC_ELT (parms, i), TREE_VEC_ELT (args, i),
|
|||
|
&dummy, 1))
|
|||
|
{
|
|||
|
case 0:
|
|||
|
break;
|
|||
|
case 1:
|
|||
|
return NULL_TREE;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
for (i = 0; i < ntparms; ++i)
|
|||
|
if (! TREE_VEC_ELT (vec, i))
|
|||
|
return NULL_TREE;
|
|||
|
|
|||
|
return vec;
|
|||
|
}
|
|||
|
|
|||
|
/* Return the most specialized of the list of templates in FNS that can
|
|||
|
produce an instantiation matching DECL. */
|
|||
|
|
|||
|
tree
|
|||
|
most_specialized (fns, decl)
|
|||
|
tree fns, decl;
|
|||
|
{
|
|||
|
tree fn, champ, args, *p;
|
|||
|
int fate;
|
|||
|
|
|||
|
for (p = &fns; *p; )
|
|||
|
{
|
|||
|
args = get_bindings (TREE_VALUE (*p), decl);
|
|||
|
if (args)
|
|||
|
{
|
|||
|
p = &TREE_CHAIN (*p);
|
|||
|
}
|
|||
|
else
|
|||
|
*p = TREE_CHAIN (*p);
|
|||
|
}
|
|||
|
|
|||
|
if (! fns)
|
|||
|
return NULL_TREE;
|
|||
|
|
|||
|
fn = fns;
|
|||
|
champ = TREE_VALUE (fn);
|
|||
|
fn = TREE_CHAIN (fn);
|
|||
|
for (; fn; fn = TREE_CHAIN (fn))
|
|||
|
{
|
|||
|
fate = more_specialized (champ, TREE_VALUE (fn));
|
|||
|
if (fate == 1)
|
|||
|
;
|
|||
|
else
|
|||
|
{
|
|||
|
if (fate == 0)
|
|||
|
{
|
|||
|
fn = TREE_CHAIN (fn);
|
|||
|
if (! fn)
|
|||
|
return error_mark_node;
|
|||
|
}
|
|||
|
champ = TREE_VALUE (fn);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
for (fn = fns; fn && TREE_VALUE (fn) != champ; fn = TREE_CHAIN (fn))
|
|||
|
{
|
|||
|
fate = more_specialized (champ, TREE_VALUE (fn));
|
|||
|
if (fate != 1)
|
|||
|
return error_mark_node;
|
|||
|
}
|
|||
|
|
|||
|
return champ;
|
|||
|
}
|
|||
|
|
|||
|
/* Return the most specialized of the class template specializations in
|
|||
|
SPECS that can produce an instantiation matching ARGS. */
|
|||
|
|
|||
|
tree
|
|||
|
most_specialized_class (specs, mainargs)
|
|||
|
tree specs, mainargs;
|
|||
|
{
|
|||
|
tree list = NULL_TREE, t, args, champ;
|
|||
|
int fate;
|
|||
|
|
|||
|
for (t = specs; t; t = TREE_CHAIN (t))
|
|||
|
{
|
|||
|
args = get_class_bindings (TREE_VALUE (t), TREE_PURPOSE (t), mainargs);
|
|||
|
if (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 name;
|
|||
|
tree fn;
|
|||
|
tree result = NULL_TREE;
|
|||
|
int extern_p = 0;
|
|||
|
tree templates = NULL_TREE;
|
|||
|
|
|||
|
if (! DECL_LANG_SPECIFIC (decl))
|
|||
|
{
|
|||
|
cp_error ("explicit instantiation of non-template `%#D'", decl);
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
/* If we've already seen this template instance, use it. */
|
|||
|
if (TREE_CODE (decl) == VAR_DECL)
|
|||
|
{
|
|||
|
result = lookup_field (DECL_CONTEXT (decl), DECL_NAME (decl), 0, 0);
|
|||
|
if (result && TREE_CODE (result) != VAR_DECL)
|
|||
|
result = NULL_TREE;
|
|||
|
}
|
|||
|
else if (TREE_CODE (decl) != FUNCTION_DECL)
|
|||
|
{
|
|||
|
cp_error ("explicit instantiation of `%#D'", decl);
|
|||
|
return;
|
|||
|
}
|
|||
|
else if (DECL_FUNCTION_MEMBER_P (decl))
|
|||
|
{
|
|||
|
if (DECL_TEMPLATE_INSTANTIATION (decl))
|
|||
|
result = decl;
|
|||
|
else if (name = DECL_ASSEMBLER_NAME (decl),
|
|||
|
fn = IDENTIFIER_GLOBAL_VALUE (name),
|
|||
|
fn && DECL_TEMPLATE_INSTANTIATION (fn))
|
|||
|
result = fn;
|
|||
|
else
|
|||
|
{
|
|||
|
/* Maybe this is an instantiation of a member template
|
|||
|
function. */
|
|||
|
tree ctype = DECL_CONTEXT (decl);
|
|||
|
|
|||
|
name = DECL_NAME (decl);
|
|||
|
fn = lookup_fnfields (TYPE_BINFO (ctype), name, 1);
|
|||
|
if (fn)
|
|||
|
fn = TREE_VALUE (fn);
|
|||
|
|
|||
|
for (; fn; fn = DECL_CHAIN (fn))
|
|||
|
if (TREE_CODE (fn) == TEMPLATE_DECL)
|
|||
|
templates = decl_tree_cons (NULL_TREE, fn, templates);
|
|||
|
}
|
|||
|
}
|
|||
|
else if (name = DECL_NAME (decl), fn = IDENTIFIER_GLOBAL_VALUE (name), fn)
|
|||
|
{
|
|||
|
for (fn = get_first_fn (fn); fn; fn = DECL_CHAIN (fn))
|
|||
|
if (TREE_CODE (fn) == TEMPLATE_DECL)
|
|||
|
templates = decl_tree_cons (NULL_TREE, fn, templates);
|
|||
|
}
|
|||
|
|
|||
|
if (templates && !result)
|
|||
|
{
|
|||
|
tree args;
|
|||
|
result = most_specialized (templates, decl);
|
|||
|
if (result == error_mark_node)
|
|||
|
{
|
|||
|
char *str = "candidates are:";
|
|||
|
cp_error ("ambiguous template instantiation for `%D' requested", decl);
|
|||
|
for (fn = templates; fn; fn = TREE_CHAIN (fn))
|
|||
|
{
|
|||
|
cp_error_at ("%s %+#D", str, TREE_VALUE (fn));
|
|||
|
str = " ";
|
|||
|
}
|
|||
|
return;
|
|||
|
}
|
|||
|
else if (result)
|
|||
|
{
|
|||
|
args = get_bindings (result, decl);
|
|||
|
result = instantiate_template (result, args);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (! result)
|
|||
|
{
|
|||
|
cp_error ("no matching template for `%D' found", decl);
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
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])
|
|||
|
extern_p = 1;
|
|||
|
else
|
|||
|
cp_error ("storage class `%D' applied to template instantiation",
|
|||
|
storage);
|
|||
|
|
|||
|
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);
|
|||
|
|
|||
|
if (supports_one_only () && ! SUPPORTS_WEAK)
|
|||
|
/* For WIN32 we also want to put explicit instantiations in
|
|||
|
linkonce sections. */;
|
|||
|
else
|
|||
|
{
|
|||
|
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 (! IS_AGGR_TYPE (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)
|
|||
|
/* OK */;
|
|||
|
else 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;
|
|||
|
}
|
|||
|
|
|||
|
/* We've already instantiated this. */
|
|||
|
if (CLASSTYPE_EXPLICIT_INSTANTIATION (t) && ! CLASSTYPE_INTERFACE_ONLY (t)
|
|||
|
&& extern_p)
|
|||
|
return;
|
|||
|
|
|||
|
if (! CLASSTYPE_TEMPLATE_SPECIALIZATION (t))
|
|||
|
{
|
|||
|
mark_class_instantiated (t, extern_p);
|
|||
|
repo_template_instantiated (t, extern_p);
|
|||
|
}
|
|||
|
|
|||
|
if (nomem_p)
|
|||
|
return;
|
|||
|
|
|||
|
{
|
|||
|
tree tmp;
|
|||
|
|
|||
|
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)))
|
|||
|
do_type_instantiation (TYPE_MAIN_DECL (TREE_VALUE (tmp)), storage);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
tree
|
|||
|
instantiate_decl (d)
|
|||
|
tree d;
|
|||
|
{
|
|||
|
tree ti = DECL_TEMPLATE_INFO (d);
|
|||
|
tree tmpl = TI_TEMPLATE (ti);
|
|||
|
tree args = TI_ARGS (ti);
|
|||
|
tree td;
|
|||
|
tree decl_pattern, code_pattern;
|
|||
|
tree save_ti;
|
|||
|
int nested = in_function_p ();
|
|||
|
int d_defined;
|
|||
|
int pattern_defined;
|
|||
|
int line = lineno;
|
|||
|
char *file = input_filename;
|
|||
|
|
|||
|
for (td = tmpl; DECL_TEMPLATE_INSTANTIATION (td); )
|
|||
|
td = DECL_TI_TEMPLATE (td);
|
|||
|
|
|||
|
/* In the case of a member template, decl_pattern is the partially
|
|||
|
instantiated declaration (in the instantiated class), and code_pattern
|
|||
|
is the original template definition. */
|
|||
|
decl_pattern = DECL_TEMPLATE_RESULT (tmpl);
|
|||
|
code_pattern = DECL_TEMPLATE_RESULT (td);
|
|||
|
|
|||
|
if (TREE_CODE (d) == FUNCTION_DECL)
|
|||
|
{
|
|||
|
d_defined = (DECL_INITIAL (d) != NULL_TREE);
|
|||
|
pattern_defined = (DECL_INITIAL (code_pattern) != NULL_TREE);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
d_defined = ! DECL_IN_AGGR_P (d);
|
|||
|
pattern_defined = ! DECL_IN_AGGR_P (code_pattern);
|
|||
|
}
|
|||
|
|
|||
|
if (d_defined)
|
|||
|
return d;
|
|||
|
|
|||
|
if (TREE_CODE (d) == FUNCTION_DECL)
|
|||
|
{
|
|||
|
tree specs;
|
|||
|
|
|||
|
/* Check to see if there is a matching specialization. */
|
|||
|
for (specs = DECL_TEMPLATE_SPECIALIZATIONS (tmpl);
|
|||
|
specs != NULL_TREE;
|
|||
|
specs = TREE_CHAIN (specs))
|
|||
|
if (comp_template_args (TREE_PURPOSE (specs), args))
|
|||
|
return TREE_VALUE (specs);
|
|||
|
}
|
|||
|
|
|||
|
/* This needs to happen before any tsubsting. */
|
|||
|
if (! push_tinst_level (d))
|
|||
|
return d;
|
|||
|
|
|||
|
push_to_top_level ();
|
|||
|
lineno = DECL_SOURCE_LINE (d);
|
|||
|
input_filename = DECL_SOURCE_FILE (d);
|
|||
|
|
|||
|
/* We need to set up DECL_INITIAL regardless of pattern_defined if the
|
|||
|
variable is a static const initialized in the class body. */
|
|||
|
if (TREE_CODE (d) == VAR_DECL
|
|||
|
&& ! DECL_INITIAL (d) && DECL_INITIAL (code_pattern))
|
|||
|
{
|
|||
|
pushclass (DECL_CONTEXT (d), 2);
|
|||
|
DECL_INITIAL (d) = tsubst_expr (DECL_INITIAL (code_pattern), args,
|
|||
|
TREE_VEC_LENGTH (args), tmpl);
|
|||
|
popclass (1);
|
|||
|
}
|
|||
|
|
|||
|
/* import_export_decl has to happen after DECL_INITIAL is set up. */
|
|||
|
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;
|
|||
|
|
|||
|
/* Defer all templates except inline functions used in another function. */
|
|||
|
if (! pattern_defined
|
|||
|
|| (! (TREE_CODE (d) == FUNCTION_DECL && DECL_INLINE (d) && nested)
|
|||
|
&& ! at_eof))
|
|||
|
{
|
|||
|
add_pending_template (d);
|
|||
|
goto out;
|
|||
|
}
|
|||
|
|
|||
|
lineno = DECL_SOURCE_LINE (d);
|
|||
|
input_filename = DECL_SOURCE_FILE (d);
|
|||
|
|
|||
|
/* Trick tsubst into giving us a new decl in case the template changed. */
|
|||
|
save_ti = DECL_TEMPLATE_INFO (decl_pattern);
|
|||
|
DECL_TEMPLATE_INFO (decl_pattern) = NULL_TREE;
|
|||
|
td = tsubst (decl_pattern, args, TREE_VEC_LENGTH (args), tmpl);
|
|||
|
SET_DECL_IMPLICIT_INSTANTIATION (td);
|
|||
|
DECL_TEMPLATE_INFO (decl_pattern) = save_ti;
|
|||
|
|
|||
|
/* And set up DECL_INITIAL, since tsubst doesn't. */
|
|||
|
if (TREE_CODE (td) == VAR_DECL)
|
|||
|
{
|
|||
|
pushclass (DECL_CONTEXT (d), 2);
|
|||
|
DECL_INITIAL (td) = tsubst_expr (DECL_INITIAL (code_pattern), args,
|
|||
|
TREE_VEC_LENGTH (args), tmpl);
|
|||
|
popclass (1);
|
|||
|
}
|
|||
|
|
|||
|
if (TREE_CODE (d) == FUNCTION_DECL)
|
|||
|
{
|
|||
|
/* Convince duplicate_decls to use the DECL_ARGUMENTS from the
|
|||
|
new decl. */
|
|||
|
DECL_INITIAL (td) = error_mark_node;
|
|||
|
|
|||
|
if (DECL_TEMPLATE_SPECIALIZATION (td) && !DECL_TEMPLATE_INFO (td))
|
|||
|
/* Set up the information about what is being specialized. */
|
|||
|
DECL_TEMPLATE_INFO (td) = DECL_TEMPLATE_INFO (d);
|
|||
|
}
|
|||
|
duplicate_decls (td, d);
|
|||
|
if (TREE_CODE (d) == FUNCTION_DECL)
|
|||
|
DECL_INITIAL (td) = 0;
|
|||
|
|
|||
|
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,
|
|||
|
TREE_VEC_LENGTH (args), 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, TREE_VEC_LENGTH (args), tmpl);
|
|||
|
|
|||
|
finish_function (lineno, 0, nested);
|
|||
|
}
|
|||
|
|
|||
|
out:
|
|||
|
lineno = line;
|
|||
|
input_filename = file;
|
|||
|
|
|||
|
pop_from_top_level ();
|
|||
|
pop_tinst_level ();
|
|||
|
|
|||
|
return d;
|
|||
|
}
|
|||
|
|
|||
|
tree
|
|||
|
tsubst_chain (t, argvec)
|
|||
|
tree t, argvec;
|
|||
|
{
|
|||
|
if (t)
|
|||
|
{
|
|||
|
tree first = tsubst (t, argvec,
|
|||
|
TREE_VEC_LENGTH (argvec), NULL_TREE);
|
|||
|
tree last = first;
|
|||
|
|
|||
|
for (t = TREE_CHAIN (t); t; t = TREE_CHAIN (t))
|
|||
|
{
|
|||
|
tree x = tsubst (t, argvec, TREE_VEC_LENGTH (argvec), NULL_TREE);
|
|||
|
TREE_CHAIN (last) = x;
|
|||
|
last = x;
|
|||
|
}
|
|||
|
|
|||
|
return first;
|
|||
|
}
|
|||
|
return NULL_TREE;
|
|||
|
}
|
|||
|
|
|||
|
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,
|
|||
|
TREE_VEC_LENGTH (argvec), NULL_TREE);
|
|||
|
tree val = tsubst_expr (TREE_VALUE (t), argvec,
|
|||
|
TREE_VEC_LENGTH (argvec), 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;
|
|||
|
}
|
|||
|
|
|||
|
/* 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);
|
|||
|
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;
|
|||
|
}
|
|||
|
|
|||
|
/* Instantiate an enumerated type. Used by instantiate_class_template and
|
|||
|
tsubst_expr. */
|
|||
|
|
|||
|
static tree
|
|||
|
tsubst_enum (tag, args, nargs, field_chain)
|
|||
|
tree tag, args;
|
|||
|
int nargs;
|
|||
|
tree * field_chain;
|
|||
|
{
|
|||
|
extern tree current_local_enum;
|
|||
|
tree prev_local_enum = current_local_enum;
|
|||
|
|
|||
|
tree newtag = start_enum (TYPE_IDENTIFIER (tag));
|
|||
|
tree e, values = NULL_TREE;
|
|||
|
|
|||
|
for (e = TYPE_VALUES (tag); e; e = TREE_CHAIN (e))
|
|||
|
{
|
|||
|
tree elt = build_enumerator (TREE_PURPOSE (e),
|
|||
|
tsubst_expr (TREE_VALUE (e), args,
|
|||
|
nargs, NULL_TREE));
|
|||
|
TREE_CHAIN (elt) = values;
|
|||
|
values = elt;
|
|||
|
}
|
|||
|
|
|||
|
finish_enum (newtag, values);
|
|||
|
|
|||
|
if (NULL != field_chain)
|
|||
|
*field_chain = grok_enum_decls (newtag, NULL_TREE);
|
|||
|
|
|||
|
current_local_enum = prev_local_enum;
|
|||
|
|
|||
|
return newtag;
|
|||
|
}
|