1136 lines
32 KiB
C
1136 lines
32 KiB
C
|
/* RunTime Type Identification
|
|||
|
Copyright (C) 1995, 1996 Free Software Foundation, Inc.
|
|||
|
Mostly written 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. */
|
|||
|
|
|||
|
|
|||
|
#include "config.h"
|
|||
|
#include "system.h"
|
|||
|
#include "tree.h"
|
|||
|
#include "cp-tree.h"
|
|||
|
#include "flags.h"
|
|||
|
#include "output.h"
|
|||
|
#include "assert.h"
|
|||
|
#include "toplev.h"
|
|||
|
|
|||
|
#ifndef INT_TYPE_SIZE
|
|||
|
#define INT_TYPE_SIZE BITS_PER_WORD
|
|||
|
#endif
|
|||
|
|
|||
|
extern struct obstack permanent_obstack;
|
|||
|
|
|||
|
static tree call_void_fn PROTO((char *));
|
|||
|
static tree build_headof_sub PROTO((tree));
|
|||
|
static tree build_headof PROTO((tree));
|
|||
|
static tree get_tinfo_var PROTO((tree));
|
|||
|
static tree get_typeid_1 PROTO((tree));
|
|||
|
static tree ifnonnull PROTO((tree, tree));
|
|||
|
static tree build_dynamic_cast_1 PROTO((tree, tree));
|
|||
|
static void expand_si_desc PROTO((tree, tree));
|
|||
|
static void expand_class_desc PROTO((tree, tree));
|
|||
|
static void expand_attr_desc PROTO((tree, tree));
|
|||
|
static void expand_ptr_desc PROTO((tree, tree));
|
|||
|
static void expand_generic_desc PROTO((tree, tree, char *));
|
|||
|
static tree throw_bad_cast PROTO((void));
|
|||
|
static tree throw_bad_typeid PROTO((void));
|
|||
|
|
|||
|
tree type_info_type_node;
|
|||
|
tree tinfo_fn_id;
|
|||
|
tree tinfo_fn_type;
|
|||
|
|
|||
|
void
|
|||
|
init_rtti_processing ()
|
|||
|
{
|
|||
|
if (flag_honor_std)
|
|||
|
push_namespace (get_identifier ("std"));
|
|||
|
type_info_type_node = xref_tag
|
|||
|
(class_type_node, get_identifier ("type_info"), NULL_TREE, 1);
|
|||
|
if (flag_honor_std)
|
|||
|
pop_namespace ();
|
|||
|
tinfo_fn_id = get_identifier ("__tf");
|
|||
|
tinfo_fn_type = build_function_type
|
|||
|
(build_reference_type (build_type_variant (type_info_type_node, 1, 0)),
|
|||
|
void_list_node);
|
|||
|
}
|
|||
|
|
|||
|
/* Given a pointer to an object with at least one virtual table
|
|||
|
pointer somewhere, return a pointer to a possible sub-object that
|
|||
|
has a virtual table pointer in it that is the vtable parent for
|
|||
|
that sub-object. */
|
|||
|
|
|||
|
static tree
|
|||
|
build_headof_sub (exp)
|
|||
|
tree exp;
|
|||
|
{
|
|||
|
tree type = TREE_TYPE (TREE_TYPE (exp));
|
|||
|
tree basetype = CLASSTYPE_RTTI (type);
|
|||
|
tree binfo = get_binfo (basetype, type, 0);
|
|||
|
|
|||
|
exp = convert_pointer_to_real (binfo, exp);
|
|||
|
return exp;
|
|||
|
}
|
|||
|
|
|||
|
/* Given the expression EXP of type `class *', return the head of the
|
|||
|
object pointed to by EXP with type cv void*, if the class has any
|
|||
|
virtual functions (TYPE_VIRTUAL_P), else just return the
|
|||
|
expression. */
|
|||
|
|
|||
|
static tree
|
|||
|
build_headof (exp)
|
|||
|
tree exp;
|
|||
|
{
|
|||
|
tree type = TREE_TYPE (exp);
|
|||
|
tree aref;
|
|||
|
tree offset;
|
|||
|
|
|||
|
if (TREE_CODE (type) != POINTER_TYPE)
|
|||
|
{
|
|||
|
error ("`headof' applied to non-pointer type");
|
|||
|
return error_mark_node;
|
|||
|
}
|
|||
|
type = TREE_TYPE (type);
|
|||
|
|
|||
|
if (!TYPE_VIRTUAL_P (type))
|
|||
|
return exp;
|
|||
|
|
|||
|
/* If we don't have rtti stuff, get to a sub-object that does. */
|
|||
|
if (!CLASSTYPE_VFIELDS (TREE_TYPE (TREE_TYPE (exp))))
|
|||
|
exp = build_headof_sub (exp);
|
|||
|
|
|||
|
/* We use this a couple of times below, protect it. */
|
|||
|
exp = save_expr (exp);
|
|||
|
|
|||
|
aref = build_vtbl_ref (build_indirect_ref (exp, NULL_PTR), integer_zero_node);
|
|||
|
|
|||
|
if (flag_vtable_thunks)
|
|||
|
offset = aref;
|
|||
|
else
|
|||
|
offset = build_component_ref (aref, delta_identifier, NULL_TREE, 0);
|
|||
|
|
|||
|
type = build_type_variant (ptr_type_node, TREE_READONLY (exp),
|
|||
|
TREE_THIS_VOLATILE (exp));
|
|||
|
return build (PLUS_EXPR, type, exp,
|
|||
|
cp_convert (ptrdiff_type_node, offset));
|
|||
|
}
|
|||
|
|
|||
|
/* Build a call to a generic entry point taking and returning void. */
|
|||
|
|
|||
|
static tree
|
|||
|
call_void_fn (name)
|
|||
|
char *name;
|
|||
|
{
|
|||
|
tree d = get_identifier (name);
|
|||
|
tree type;
|
|||
|
|
|||
|
if (IDENTIFIER_GLOBAL_VALUE (d))
|
|||
|
d = IDENTIFIER_GLOBAL_VALUE (d);
|
|||
|
else
|
|||
|
{
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
|
|||
|
type = build_function_type (void_type_node, void_list_node);
|
|||
|
d = build_lang_decl (FUNCTION_DECL, d, type);
|
|||
|
DECL_EXTERNAL (d) = 1;
|
|||
|
TREE_PUBLIC (d) = 1;
|
|||
|
DECL_ARTIFICIAL (d) = 1;
|
|||
|
pushdecl_top_level (d);
|
|||
|
make_function_rtl (d);
|
|||
|
assemble_external (d);
|
|||
|
|
|||
|
pop_obstacks ();
|
|||
|
}
|
|||
|
|
|||
|
return build_call (d, void_type_node, NULL_TREE);
|
|||
|
}
|
|||
|
|
|||
|
/* Get a bad_cast node for the program to throw...
|
|||
|
|
|||
|
See libstdc++/exception.cc for __throw_bad_cast */
|
|||
|
|
|||
|
static tree
|
|||
|
throw_bad_cast ()
|
|||
|
{
|
|||
|
return call_void_fn ("__throw_bad_cast");
|
|||
|
}
|
|||
|
|
|||
|
static tree
|
|||
|
throw_bad_typeid ()
|
|||
|
{
|
|||
|
return call_void_fn ("__throw_bad_typeid");
|
|||
|
}
|
|||
|
|
|||
|
/* Return the type_info function associated with the expression EXP. If
|
|||
|
EXP is a reference to a polymorphic class, return the dynamic type;
|
|||
|
otherwise return the static type of the expression. */
|
|||
|
|
|||
|
tree
|
|||
|
get_tinfo_fn_dynamic (exp)
|
|||
|
tree exp;
|
|||
|
{
|
|||
|
tree type;
|
|||
|
|
|||
|
if (exp == error_mark_node)
|
|||
|
return error_mark_node;
|
|||
|
|
|||
|
if (type_unknown_p (exp))
|
|||
|
{
|
|||
|
error ("typeid of overloaded function");
|
|||
|
return error_mark_node;
|
|||
|
}
|
|||
|
|
|||
|
type = TREE_TYPE (exp);
|
|||
|
|
|||
|
/* peel back references, so they match. */
|
|||
|
if (TREE_CODE (type) == REFERENCE_TYPE)
|
|||
|
type = TREE_TYPE (type);
|
|||
|
|
|||
|
/* Peel off cv qualifiers. */
|
|||
|
type = TYPE_MAIN_VARIANT (type);
|
|||
|
|
|||
|
/* If exp is a reference to polymorphic type, get the real type_info. */
|
|||
|
if (TYPE_VIRTUAL_P (type) && ! resolves_to_fixed_type_p (exp, 0))
|
|||
|
{
|
|||
|
/* build reference to type_info from vtable. */
|
|||
|
tree t;
|
|||
|
|
|||
|
if (! flag_rtti)
|
|||
|
{
|
|||
|
warning ("taking dynamic typeid of object without -frtti");
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
init_rtti_processing ();
|
|||
|
pop_obstacks ();
|
|||
|
flag_rtti = 1;
|
|||
|
}
|
|||
|
|
|||
|
/* If we don't have rtti stuff, get to a sub-object that does. */
|
|||
|
if (! CLASSTYPE_VFIELDS (type))
|
|||
|
{
|
|||
|
exp = build_unary_op (ADDR_EXPR, exp, 0);
|
|||
|
exp = build_headof_sub (exp);
|
|||
|
exp = build_indirect_ref (exp, NULL_PTR);
|
|||
|
}
|
|||
|
|
|||
|
if (flag_vtable_thunks)
|
|||
|
t = build_vfn_ref ((tree *) 0, exp, integer_one_node);
|
|||
|
else
|
|||
|
t = build_vfn_ref ((tree *) 0, exp, integer_zero_node);
|
|||
|
TREE_TYPE (t) = build_pointer_type (tinfo_fn_type);
|
|||
|
return t;
|
|||
|
}
|
|||
|
|
|||
|
/* otherwise return the type_info for the static type of the expr. */
|
|||
|
return get_tinfo_fn (TYPE_MAIN_VARIANT (type));
|
|||
|
}
|
|||
|
|
|||
|
tree
|
|||
|
build_typeid (exp)
|
|||
|
tree exp;
|
|||
|
{
|
|||
|
exp = get_tinfo_fn_dynamic (exp);
|
|||
|
exp = build_call (exp, TREE_TYPE (tinfo_fn_type), NULL_TREE);
|
|||
|
return convert_from_reference (exp);
|
|||
|
}
|
|||
|
|
|||
|
tree
|
|||
|
build_x_typeid (exp)
|
|||
|
tree exp;
|
|||
|
{
|
|||
|
tree cond = NULL_TREE;
|
|||
|
tree type = TREE_TYPE (tinfo_fn_type);
|
|||
|
int nonnull;
|
|||
|
|
|||
|
if (processing_template_decl)
|
|||
|
return build_min_nt (TYPEID_EXPR, exp);
|
|||
|
|
|||
|
if (TREE_CODE (exp) == INDIRECT_REF
|
|||
|
&& TREE_CODE (TREE_TYPE (TREE_OPERAND (exp, 0))) == POINTER_TYPE
|
|||
|
&& TYPE_VIRTUAL_P (TREE_TYPE (exp))
|
|||
|
&& ! resolves_to_fixed_type_p (exp, &nonnull)
|
|||
|
&& ! nonnull)
|
|||
|
{
|
|||
|
exp = stabilize_reference (exp);
|
|||
|
cond = cp_convert (boolean_type_node, TREE_OPERAND (exp, 0));
|
|||
|
}
|
|||
|
|
|||
|
exp = get_tinfo_fn_dynamic (exp);
|
|||
|
|
|||
|
if (exp == error_mark_node)
|
|||
|
return error_mark_node;
|
|||
|
|
|||
|
exp = build_call (exp, type, NULL_TREE);
|
|||
|
|
|||
|
if (cond)
|
|||
|
{
|
|||
|
tree bad = throw_bad_typeid ();
|
|||
|
|
|||
|
bad = build_compound_expr
|
|||
|
(expr_tree_cons (NULL_TREE, bad, build_expr_list
|
|||
|
(NULL_TREE, cp_convert (type, integer_zero_node))));
|
|||
|
exp = build (COND_EXPR, type, cond, exp, bad);
|
|||
|
}
|
|||
|
|
|||
|
return convert_from_reference (exp);
|
|||
|
}
|
|||
|
|
|||
|
static tree
|
|||
|
get_tinfo_var (type)
|
|||
|
tree type;
|
|||
|
{
|
|||
|
tree tname = build_overload_with_type (get_identifier ("__ti"), type);
|
|||
|
tree tdecl, arrtype;
|
|||
|
int size;
|
|||
|
|
|||
|
if (IDENTIFIER_GLOBAL_VALUE (tname))
|
|||
|
return IDENTIFIER_GLOBAL_VALUE (tname);
|
|||
|
|
|||
|
/* Figure out how much space we need to allocate for the type_info object.
|
|||
|
If our struct layout or the type_info classes are changed, this will
|
|||
|
need to be modified. */
|
|||
|
if (TYPE_VOLATILE (type) || TYPE_READONLY (type))
|
|||
|
size = 3 * POINTER_SIZE + INT_TYPE_SIZE;
|
|||
|
else if (TREE_CODE (type) == POINTER_TYPE
|
|||
|
&& ! (TREE_CODE (TREE_TYPE (type)) == OFFSET_TYPE
|
|||
|
|| TREE_CODE (TREE_TYPE (type)) == METHOD_TYPE))
|
|||
|
size = 3 * POINTER_SIZE;
|
|||
|
else if (IS_AGGR_TYPE (type))
|
|||
|
{
|
|||
|
if (CLASSTYPE_N_BASECLASSES (type) == 0)
|
|||
|
size = 2 * POINTER_SIZE;
|
|||
|
else if (! TYPE_USES_COMPLEX_INHERITANCE (type)
|
|||
|
&& (TREE_VIA_PUBLIC
|
|||
|
(TREE_VEC_ELT (TYPE_BINFO_BASETYPES (type), 0))))
|
|||
|
size = 3 * POINTER_SIZE;
|
|||
|
else
|
|||
|
size = 3 * POINTER_SIZE + TYPE_PRECISION (sizetype);
|
|||
|
}
|
|||
|
else
|
|||
|
size = 2 * POINTER_SIZE;
|
|||
|
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
|
|||
|
/* The type for a character array of the appropriate size. */
|
|||
|
arrtype = build_cplus_array_type
|
|||
|
(unsigned_char_type_node,
|
|||
|
build_index_type (size_int (size / BITS_PER_UNIT - 1)));
|
|||
|
|
|||
|
tdecl = build_decl (VAR_DECL, tname, arrtype);
|
|||
|
TREE_PUBLIC (tdecl) = 1;
|
|||
|
DECL_EXTERNAL (tdecl) = 1;
|
|||
|
DECL_ARTIFICIAL (tdecl) = 1;
|
|||
|
pushdecl_top_level (tdecl);
|
|||
|
cp_finish_decl (tdecl, NULL_TREE, NULL_TREE, 0, 0);
|
|||
|
|
|||
|
pop_obstacks ();
|
|||
|
|
|||
|
return tdecl;
|
|||
|
}
|
|||
|
|
|||
|
tree
|
|||
|
get_tinfo_fn (type)
|
|||
|
tree type;
|
|||
|
{
|
|||
|
tree name;
|
|||
|
tree d;
|
|||
|
|
|||
|
if (TREE_CODE (type) == OFFSET_TYPE)
|
|||
|
type = TREE_TYPE (type);
|
|||
|
if (TREE_CODE (type) == METHOD_TYPE)
|
|||
|
type = build_function_type (TREE_TYPE (type),
|
|||
|
TREE_CHAIN (TYPE_ARG_TYPES (type)));
|
|||
|
|
|||
|
name = build_overload_with_type (tinfo_fn_id, type);
|
|||
|
|
|||
|
if (IDENTIFIER_GLOBAL_VALUE (name))
|
|||
|
return IDENTIFIER_GLOBAL_VALUE (name);
|
|||
|
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
|
|||
|
d = build_lang_decl (FUNCTION_DECL, name, tinfo_fn_type);
|
|||
|
DECL_EXTERNAL (d) = 1;
|
|||
|
TREE_PUBLIC (d) = 1;
|
|||
|
DECL_ARTIFICIAL (d) = 1;
|
|||
|
DECL_NOT_REALLY_EXTERN (d) = 1;
|
|||
|
DECL_MUTABLE_P (d) = 1;
|
|||
|
TREE_TYPE (name) = copy_to_permanent (type);
|
|||
|
|
|||
|
pushdecl_top_level (d);
|
|||
|
make_function_rtl (d);
|
|||
|
assemble_external (d);
|
|||
|
mark_inline_for_output (d);
|
|||
|
pop_obstacks ();
|
|||
|
|
|||
|
return d;
|
|||
|
}
|
|||
|
|
|||
|
static tree
|
|||
|
get_typeid_1 (type)
|
|||
|
tree type;
|
|||
|
{
|
|||
|
tree t = build_call
|
|||
|
(get_tinfo_fn (type), TREE_TYPE (tinfo_fn_type), NULL_TREE);
|
|||
|
return convert_from_reference (t);
|
|||
|
}
|
|||
|
|
|||
|
/* Return the type_info object for TYPE, creating it if necessary. */
|
|||
|
|
|||
|
tree
|
|||
|
get_typeid (type)
|
|||
|
tree type;
|
|||
|
{
|
|||
|
if (type == error_mark_node)
|
|||
|
return error_mark_node;
|
|||
|
|
|||
|
if (! flag_rtti)
|
|||
|
{
|
|||
|
warning ("requesting typeid of object without -frtti");
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
init_rtti_processing ();
|
|||
|
pop_obstacks ();
|
|||
|
flag_rtti = 1;
|
|||
|
}
|
|||
|
|
|||
|
if (processing_template_decl)
|
|||
|
return build_min_nt (TYPEID_EXPR, type);
|
|||
|
|
|||
|
/* If the type of the type-id is a reference type, the result of the
|
|||
|
typeid expression refers to a type_info object representing the
|
|||
|
referenced type. */
|
|||
|
if (TREE_CODE (type) == REFERENCE_TYPE)
|
|||
|
type = TREE_TYPE (type);
|
|||
|
|
|||
|
/* The top-level cv-qualifiers of the lvalue expression or the type-id
|
|||
|
that is the operand of typeid are always ignored. */
|
|||
|
type = TYPE_MAIN_VARIANT (type);
|
|||
|
|
|||
|
return get_typeid_1 (type);
|
|||
|
}
|
|||
|
|
|||
|
/* Check whether TEST is null before returning RESULT. If TEST is used in
|
|||
|
RESULT, it must have previously had a save_expr applied to it. */
|
|||
|
|
|||
|
static tree
|
|||
|
ifnonnull (test, result)
|
|||
|
tree test, result;
|
|||
|
{
|
|||
|
return build (COND_EXPR, TREE_TYPE (result),
|
|||
|
build (EQ_EXPR, boolean_type_node, test, integer_zero_node),
|
|||
|
cp_convert (TREE_TYPE (result), integer_zero_node),
|
|||
|
result);
|
|||
|
}
|
|||
|
|
|||
|
/* Execute a dynamic cast, as described in section 5.2.6 of the 9/93 working
|
|||
|
paper. */
|
|||
|
|
|||
|
static tree
|
|||
|
build_dynamic_cast_1 (type, expr)
|
|||
|
tree type, expr;
|
|||
|
{
|
|||
|
enum tree_code tc = TREE_CODE (type);
|
|||
|
tree exprtype = TREE_TYPE (expr);
|
|||
|
enum tree_code ec;
|
|||
|
tree dcast_fn;
|
|||
|
|
|||
|
assert (exprtype != NULL_TREE);
|
|||
|
ec = TREE_CODE (exprtype);
|
|||
|
|
|||
|
switch (tc)
|
|||
|
{
|
|||
|
case POINTER_TYPE:
|
|||
|
if (ec == REFERENCE_TYPE)
|
|||
|
{
|
|||
|
expr = convert_from_reference (expr);
|
|||
|
exprtype = TREE_TYPE (expr);
|
|||
|
ec = TREE_CODE (exprtype);
|
|||
|
}
|
|||
|
if (ec != POINTER_TYPE)
|
|||
|
goto fail;
|
|||
|
if (TREE_CODE (TREE_TYPE (exprtype)) != RECORD_TYPE)
|
|||
|
goto fail;
|
|||
|
if (TYPE_SIZE (complete_type (TREE_TYPE (exprtype))) == NULL_TREE)
|
|||
|
goto fail;
|
|||
|
if (TREE_READONLY (TREE_TYPE (exprtype))
|
|||
|
&& ! TYPE_READONLY (TREE_TYPE (type)))
|
|||
|
goto fail;
|
|||
|
if (TYPE_MAIN_VARIANT (TREE_TYPE (type)) == void_type_node)
|
|||
|
break;
|
|||
|
/* else fall through */
|
|||
|
case REFERENCE_TYPE:
|
|||
|
if (TREE_CODE (TREE_TYPE (type)) != RECORD_TYPE)
|
|||
|
goto fail;
|
|||
|
if (TYPE_SIZE (complete_type (TREE_TYPE (type))) == NULL_TREE)
|
|||
|
goto fail;
|
|||
|
break;
|
|||
|
/* else fall through */
|
|||
|
default:
|
|||
|
goto fail;
|
|||
|
}
|
|||
|
|
|||
|
/* Apply trivial conversion T -> T& for dereferenced ptrs. */
|
|||
|
if (ec == RECORD_TYPE)
|
|||
|
{
|
|||
|
exprtype = build_type_variant (exprtype, TREE_READONLY (expr),
|
|||
|
TREE_THIS_VOLATILE (expr));
|
|||
|
exprtype = build_reference_type (exprtype);
|
|||
|
expr = convert_to_reference (exprtype, expr, CONV_IMPLICIT,
|
|||
|
LOOKUP_NORMAL, NULL_TREE);
|
|||
|
ec = REFERENCE_TYPE;
|
|||
|
}
|
|||
|
|
|||
|
if (tc == REFERENCE_TYPE)
|
|||
|
{
|
|||
|
if (ec != REFERENCE_TYPE)
|
|||
|
goto fail;
|
|||
|
if (TREE_CODE (TREE_TYPE (exprtype)) != RECORD_TYPE)
|
|||
|
goto fail;
|
|||
|
if (TYPE_SIZE (complete_type (TREE_TYPE (exprtype))) == NULL_TREE)
|
|||
|
goto fail;
|
|||
|
if (TREE_READONLY (TREE_TYPE (exprtype))
|
|||
|
&& ! TYPE_READONLY (TREE_TYPE (type)))
|
|||
|
goto fail;
|
|||
|
}
|
|||
|
|
|||
|
/* If *type is an unambiguous accessible base class of *exprtype,
|
|||
|
convert statically. */
|
|||
|
{
|
|||
|
int distance;
|
|||
|
tree path;
|
|||
|
|
|||
|
distance = get_base_distance (TREE_TYPE (type), TREE_TYPE (exprtype), 1,
|
|||
|
&path);
|
|||
|
if (distance >= 0)
|
|||
|
return build_vbase_path (PLUS_EXPR, type, expr, path, 0);
|
|||
|
}
|
|||
|
|
|||
|
/* Otherwise *exprtype must be a polymorphic class (have a vtbl). */
|
|||
|
if (TYPE_VIRTUAL_P (TREE_TYPE (exprtype)))
|
|||
|
{
|
|||
|
tree expr1;
|
|||
|
/* if TYPE is `void *', return pointer to complete object. */
|
|||
|
if (tc == POINTER_TYPE
|
|||
|
&& TYPE_MAIN_VARIANT (TREE_TYPE (type)) == void_type_node)
|
|||
|
{
|
|||
|
/* if b is an object, dynamic_cast<void *>(&b) == (void *)&b. */
|
|||
|
if (TREE_CODE (expr) == ADDR_EXPR
|
|||
|
&& TREE_CODE (TREE_OPERAND (expr, 0)) == VAR_DECL
|
|||
|
&& TREE_CODE (TREE_TYPE (TREE_OPERAND (expr, 0))) == RECORD_TYPE)
|
|||
|
return build1 (NOP_EXPR, type, expr);
|
|||
|
|
|||
|
/* Since expr is used twice below, save it. */
|
|||
|
expr = save_expr (expr);
|
|||
|
|
|||
|
expr1 = build_headof (expr);
|
|||
|
if (TREE_TYPE (expr1) != type)
|
|||
|
expr1 = build1 (NOP_EXPR, type, expr1);
|
|||
|
return ifnonnull (expr, expr1);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
tree retval;
|
|||
|
tree result, td1, td2, td3, elems, expr2;
|
|||
|
|
|||
|
/* If we got here, we can't convert statically. Therefore,
|
|||
|
dynamic_cast<D&>(b) (b an object) cannot succeed. */
|
|||
|
if (ec == REFERENCE_TYPE)
|
|||
|
{
|
|||
|
if (TREE_CODE (expr) == VAR_DECL
|
|||
|
&& TREE_CODE (TREE_TYPE (expr)) == RECORD_TYPE)
|
|||
|
{
|
|||
|
cp_warning ("dynamic_cast of `%#D' to `%#T' can never succeed",
|
|||
|
expr, type);
|
|||
|
return throw_bad_cast ();
|
|||
|
}
|
|||
|
}
|
|||
|
/* Ditto for dynamic_cast<D*>(&b). */
|
|||
|
else if (TREE_CODE (expr) == ADDR_EXPR)
|
|||
|
{
|
|||
|
tree op = TREE_OPERAND (expr, 0);
|
|||
|
if (TREE_CODE (op) == VAR_DECL
|
|||
|
&& TREE_CODE (TREE_TYPE (op)) == RECORD_TYPE)
|
|||
|
{
|
|||
|
cp_warning ("dynamic_cast of `%#D' to `%#T' can never succeed",
|
|||
|
expr, type);
|
|||
|
retval = build_int_2 (0, 0);
|
|||
|
TREE_TYPE (retval) = type;
|
|||
|
return retval;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/* Since expr is used twice below, save it. */
|
|||
|
expr = save_expr (expr);
|
|||
|
|
|||
|
expr1 = expr;
|
|||
|
if (tc == REFERENCE_TYPE)
|
|||
|
expr1 = build_unary_op (ADDR_EXPR, expr1, 0);
|
|||
|
|
|||
|
/* Build run-time conversion. */
|
|||
|
expr2 = build_headof (expr1);
|
|||
|
|
|||
|
if (ec == POINTER_TYPE)
|
|||
|
td1 = get_tinfo_fn_dynamic (build_indirect_ref (expr, NULL_PTR));
|
|||
|
else
|
|||
|
td1 = get_tinfo_fn_dynamic (expr);
|
|||
|
td1 = decay_conversion (td1);
|
|||
|
|
|||
|
td2 = decay_conversion
|
|||
|
(get_tinfo_fn (TYPE_MAIN_VARIANT (TREE_TYPE (type))));
|
|||
|
td3 = decay_conversion
|
|||
|
(get_tinfo_fn (TYPE_MAIN_VARIANT (TREE_TYPE (exprtype))));
|
|||
|
|
|||
|
elems = tree_cons
|
|||
|
(NULL_TREE, td1, tree_cons
|
|||
|
(NULL_TREE, td2, tree_cons
|
|||
|
(NULL_TREE, build_int_2 (1, 0), tree_cons
|
|||
|
(NULL_TREE, expr2, tree_cons
|
|||
|
(NULL_TREE, td3, tree_cons
|
|||
|
(NULL_TREE, expr1, NULL_TREE))))));
|
|||
|
|
|||
|
dcast_fn = get_identifier ("__dynamic_cast");
|
|||
|
if (IDENTIFIER_GLOBAL_VALUE (dcast_fn))
|
|||
|
dcast_fn = IDENTIFIER_GLOBAL_VALUE (dcast_fn);
|
|||
|
else
|
|||
|
{
|
|||
|
tree tmp;
|
|||
|
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
tmp = tree_cons
|
|||
|
(NULL_TREE, TREE_TYPE (td1), tree_cons
|
|||
|
(NULL_TREE, TREE_TYPE (td1), tree_cons
|
|||
|
(NULL_TREE, integer_type_node, tree_cons
|
|||
|
(NULL_TREE, ptr_type_node, tree_cons
|
|||
|
(NULL_TREE, TREE_TYPE (td1), tree_cons
|
|||
|
(NULL_TREE, ptr_type_node, void_list_node))))));
|
|||
|
tmp = build_function_type (ptr_type_node, tmp);
|
|||
|
dcast_fn = build_lang_decl (FUNCTION_DECL, dcast_fn, tmp);
|
|||
|
DECL_EXTERNAL (dcast_fn) = 1;
|
|||
|
TREE_PUBLIC (dcast_fn) = 1;
|
|||
|
DECL_ARTIFICIAL (dcast_fn) = 1;
|
|||
|
pushdecl_top_level (dcast_fn);
|
|||
|
make_function_rtl (dcast_fn);
|
|||
|
assemble_external (dcast_fn);
|
|||
|
pop_obstacks ();
|
|||
|
}
|
|||
|
|
|||
|
result = build_call
|
|||
|
(dcast_fn, TREE_TYPE (TREE_TYPE (dcast_fn)), elems);
|
|||
|
|
|||
|
if (tc == REFERENCE_TYPE)
|
|||
|
{
|
|||
|
expr1 = throw_bad_cast ();
|
|||
|
expr1 = build_compound_expr
|
|||
|
(expr_tree_cons (NULL_TREE, expr1,
|
|||
|
build_expr_list (NULL_TREE, cp_convert (type, integer_zero_node))));
|
|||
|
TREE_TYPE (expr1) = type;
|
|||
|
result = save_expr (result);
|
|||
|
return build (COND_EXPR, type, result, result, expr1);
|
|||
|
}
|
|||
|
|
|||
|
/* Now back to the type we want from a void*. */
|
|||
|
result = cp_convert (type, result);
|
|||
|
return ifnonnull (expr, result);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
fail:
|
|||
|
cp_error ("cannot dynamic_cast `%E' (of type `%#T') to type `%#T'",
|
|||
|
expr, exprtype, type);
|
|||
|
return error_mark_node;
|
|||
|
}
|
|||
|
|
|||
|
tree
|
|||
|
build_dynamic_cast (type, expr)
|
|||
|
tree type, expr;
|
|||
|
{
|
|||
|
if (type == error_mark_node || expr == error_mark_node)
|
|||
|
return error_mark_node;
|
|||
|
|
|||
|
if (processing_template_decl)
|
|||
|
return build_min (DYNAMIC_CAST_EXPR, copy_to_permanent (type), expr);
|
|||
|
|
|||
|
return convert_from_reference (build_dynamic_cast_1 (type, expr));
|
|||
|
}
|
|||
|
|
|||
|
/* Build and initialize various sorts of descriptors. Every descriptor
|
|||
|
node has a name associated with it (the name created by mangling).
|
|||
|
For this reason, we use the identifier as our access to the __*_desc
|
|||
|
nodes, instead of sticking them directly in the types. Otherwise we
|
|||
|
would burden all built-in types (and pointer types) with slots that
|
|||
|
we don't necessarily want to use.
|
|||
|
|
|||
|
For each descriptor we build, we build a variable that contains
|
|||
|
the descriptor's information. When we need this info at runtime,
|
|||
|
all we need is access to these variables.
|
|||
|
|
|||
|
Note: these constructors always return the address of the descriptor
|
|||
|
info, since that is simplest for their mutual interaction. */
|
|||
|
|
|||
|
extern tree const_string_type_node;
|
|||
|
|
|||
|
/* Build an initializer for a __si_type_info node. */
|
|||
|
|
|||
|
static void
|
|||
|
expand_si_desc (tdecl, type)
|
|||
|
tree tdecl;
|
|||
|
tree type;
|
|||
|
{
|
|||
|
tree t, elems, fn;
|
|||
|
char *name = build_overload_name (type, 1, 1);
|
|||
|
tree name_string = combine_strings (build_string (strlen (name)+1, name));
|
|||
|
|
|||
|
type = BINFO_TYPE (TREE_VEC_ELT (TYPE_BINFO_BASETYPES (type), 0));
|
|||
|
expand_expr_stmt (get_typeid_1 (type));
|
|||
|
t = decay_conversion (get_tinfo_var (type));
|
|||
|
elems = tree_cons
|
|||
|
(NULL_TREE, decay_conversion (tdecl), tree_cons
|
|||
|
(NULL_TREE, decay_conversion (name_string), tree_cons
|
|||
|
(NULL_TREE, t, NULL_TREE)));
|
|||
|
|
|||
|
fn = get_identifier ("__rtti_si");
|
|||
|
if (IDENTIFIER_GLOBAL_VALUE (fn))
|
|||
|
fn = IDENTIFIER_GLOBAL_VALUE (fn);
|
|||
|
else
|
|||
|
{
|
|||
|
tree tmp;
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
tmp = tree_cons
|
|||
|
(NULL_TREE, ptr_type_node, tree_cons
|
|||
|
(NULL_TREE, const_string_type_node, tree_cons
|
|||
|
(NULL_TREE, build_pointer_type (type_info_type_node),
|
|||
|
void_list_node)));
|
|||
|
tmp = build_function_type (void_type_node, tmp);
|
|||
|
|
|||
|
fn = build_lang_decl (FUNCTION_DECL, fn, tmp);
|
|||
|
DECL_EXTERNAL (fn) = 1;
|
|||
|
TREE_PUBLIC (fn) = 1;
|
|||
|
DECL_ARTIFICIAL (fn) = 1;
|
|||
|
pushdecl_top_level (fn);
|
|||
|
make_function_rtl (fn);
|
|||
|
assemble_external (fn);
|
|||
|
pop_obstacks ();
|
|||
|
}
|
|||
|
|
|||
|
fn = build_call (fn, TREE_TYPE (TREE_TYPE (fn)), elems);
|
|||
|
expand_expr_stmt (fn);
|
|||
|
}
|
|||
|
|
|||
|
/* Build an initializer for a __class_type_info node. */
|
|||
|
|
|||
|
static void
|
|||
|
expand_class_desc (tdecl, type)
|
|||
|
tree tdecl;
|
|||
|
tree type;
|
|||
|
{
|
|||
|
tree name_string;
|
|||
|
tree fn, tmp;
|
|||
|
char *name;
|
|||
|
|
|||
|
int i = CLASSTYPE_N_BASECLASSES (type);
|
|||
|
int base_cnt = 0;
|
|||
|
tree binfos = TYPE_BINFO_BASETYPES (type);
|
|||
|
#if 0
|
|||
|
/* See code below that used these. */
|
|||
|
tree vb = CLASSTYPE_VBASECLASSES (type);
|
|||
|
int n_base = i;
|
|||
|
#endif
|
|||
|
tree base, elems, access, offset, isvir;
|
|||
|
tree elt, elts = NULL_TREE;
|
|||
|
static tree base_info_type_node;
|
|||
|
|
|||
|
if (base_info_type_node == NULL_TREE)
|
|||
|
{
|
|||
|
tree fields [4];
|
|||
|
|
|||
|
/* A reasonably close approximation of __class_type_info::base_info */
|
|||
|
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
base_info_type_node = make_lang_type (RECORD_TYPE);
|
|||
|
|
|||
|
/* Actually const __user_type_info * */
|
|||
|
fields [0] = build_lang_field_decl
|
|||
|
(FIELD_DECL, NULL_TREE,
|
|||
|
build_pointer_type (build_type_variant (type_info_type_node, 1, 0)));
|
|||
|
fields [1] = build_lang_field_decl
|
|||
|
(FIELD_DECL, NULL_TREE, unsigned_intSI_type_node);
|
|||
|
DECL_BIT_FIELD (fields[1]) = 1;
|
|||
|
DECL_FIELD_SIZE (fields[1]) = 29;
|
|||
|
|
|||
|
fields [2] = build_lang_field_decl
|
|||
|
(FIELD_DECL, NULL_TREE, boolean_type_node);
|
|||
|
DECL_BIT_FIELD (fields[2]) = 1;
|
|||
|
DECL_FIELD_SIZE (fields[2]) = 1;
|
|||
|
|
|||
|
/* Actually enum access */
|
|||
|
fields [3] = build_lang_field_decl
|
|||
|
(FIELD_DECL, NULL_TREE, integer_type_node);
|
|||
|
DECL_BIT_FIELD (fields[3]) = 1;
|
|||
|
DECL_FIELD_SIZE (fields[3]) = 2;
|
|||
|
|
|||
|
finish_builtin_type (base_info_type_node, "__base_info", fields,
|
|||
|
3, ptr_type_node);
|
|||
|
pop_obstacks ();
|
|||
|
}
|
|||
|
|
|||
|
while (--i >= 0)
|
|||
|
{
|
|||
|
tree binfo = TREE_VEC_ELT (binfos, i);
|
|||
|
|
|||
|
expand_expr_stmt (get_typeid_1 (BINFO_TYPE (binfo)));
|
|||
|
base = decay_conversion (get_tinfo_var (BINFO_TYPE (binfo)));
|
|||
|
|
|||
|
if (TREE_VIA_VIRTUAL (binfo))
|
|||
|
{
|
|||
|
tree t = BINFO_TYPE (binfo);
|
|||
|
char *name;
|
|||
|
tree field;
|
|||
|
|
|||
|
name = (char *) alloca (TYPE_NAME_LENGTH (t)+sizeof (VBASE_NAME)+1);
|
|||
|
sprintf (name, VBASE_NAME_FORMAT, TYPE_NAME_STRING (t));
|
|||
|
field = lookup_field (type, get_identifier (name), 0, 0);
|
|||
|
offset = size_binop (FLOOR_DIV_EXPR,
|
|||
|
DECL_FIELD_BITPOS (field), size_int (BITS_PER_UNIT));
|
|||
|
offset = convert (sizetype, offset);
|
|||
|
}
|
|||
|
else
|
|||
|
offset = BINFO_OFFSET (binfo);
|
|||
|
|
|||
|
if (TREE_VIA_PUBLIC (binfo))
|
|||
|
access = access_public_node;
|
|||
|
else if (TREE_VIA_PROTECTED (binfo))
|
|||
|
access = access_protected_node;
|
|||
|
else
|
|||
|
access = access_private_node;
|
|||
|
if (TREE_VIA_VIRTUAL (binfo))
|
|||
|
isvir = boolean_true_node;
|
|||
|
else
|
|||
|
isvir = boolean_false_node;
|
|||
|
|
|||
|
elt = build
|
|||
|
(CONSTRUCTOR, base_info_type_node, NULL_TREE, tree_cons
|
|||
|
(NULL_TREE, base, tree_cons
|
|||
|
(NULL_TREE, offset, tree_cons
|
|||
|
(NULL_TREE, isvir, tree_cons
|
|||
|
(NULL_TREE, access, NULL_TREE)))));
|
|||
|
TREE_HAS_CONSTRUCTOR (elt) = TREE_CONSTANT (elt) = TREE_STATIC (elt) = 1;
|
|||
|
elts = expr_tree_cons (NULL_TREE, elt, elts);
|
|||
|
base_cnt++;
|
|||
|
}
|
|||
|
#if 0
|
|||
|
i = n_base;
|
|||
|
while (vb)
|
|||
|
{
|
|||
|
tree b;
|
|||
|
access = access_public_node;
|
|||
|
while (--i >= 0)
|
|||
|
{
|
|||
|
b = TREE_VEC_ELT (binfos, i);
|
|||
|
if (BINFO_TYPE (vb) == BINFO_TYPE (b) && TREE_VIA_VIRTUAL (b))
|
|||
|
{
|
|||
|
if (TREE_VIA_PUBLIC (b))
|
|||
|
access = access_public_node;
|
|||
|
else if (TREE_VIA_PROTECTED (b))
|
|||
|
access = access_protected_node;
|
|||
|
else
|
|||
|
access = access_private_node;
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
base = build_t_desc (BINFO_TYPE (vb), 1);
|
|||
|
offset = BINFO_OFFSET (vb);
|
|||
|
isvir = build_int_2 (1, 0);
|
|||
|
|
|||
|
base_list = expr_tree_cons (NULL_TREE, base, base_list);
|
|||
|
isvir_list = expr_tree_cons (NULL_TREE, isvir, isvir_list);
|
|||
|
acc_list = expr_tree_cons (NULL_TREE, access, acc_list);
|
|||
|
off_list = expr_tree_cons (NULL_TREE, offset, off_list);
|
|||
|
|
|||
|
base_cnt++;
|
|||
|
vb = TREE_CHAIN (vb);
|
|||
|
}
|
|||
|
#endif
|
|||
|
|
|||
|
name = build_overload_name (type, 1, 1);
|
|||
|
name_string = combine_strings (build_string (strlen (name)+1, name));
|
|||
|
|
|||
|
{
|
|||
|
tree arrtype = build_array_type (base_info_type_node, NULL_TREE);
|
|||
|
elts = build (CONSTRUCTOR, arrtype, NULL_TREE, elts);
|
|||
|
TREE_HAS_CONSTRUCTOR (elts) = TREE_CONSTANT (elts)
|
|||
|
= TREE_STATIC (elts) = 1;
|
|||
|
complete_array_type (arrtype, elts, 1);
|
|||
|
}
|
|||
|
|
|||
|
elems = tree_cons
|
|||
|
(NULL_TREE, decay_conversion (tdecl), tree_cons
|
|||
|
(NULL_TREE, decay_conversion (name_string), tree_cons
|
|||
|
(NULL_TREE, decay_conversion (elts), tree_cons
|
|||
|
(NULL_TREE, cp_convert (sizetype, build_int_2 (base_cnt, 0)),
|
|||
|
NULL_TREE))));
|
|||
|
|
|||
|
fn = get_identifier ("__rtti_class");
|
|||
|
if (IDENTIFIER_GLOBAL_VALUE (fn))
|
|||
|
fn = IDENTIFIER_GLOBAL_VALUE (fn);
|
|||
|
else
|
|||
|
{
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
tmp = tree_cons
|
|||
|
(NULL_TREE, ptr_type_node, tree_cons
|
|||
|
(NULL_TREE, const_string_type_node, tree_cons
|
|||
|
(NULL_TREE, build_pointer_type (base_info_type_node), tree_cons
|
|||
|
(NULL_TREE, sizetype, void_list_node))));
|
|||
|
tmp = build_function_type (void_type_node, tmp);
|
|||
|
|
|||
|
fn = build_lang_decl (FUNCTION_DECL, fn, tmp);
|
|||
|
DECL_EXTERNAL (fn) = 1;
|
|||
|
TREE_PUBLIC (fn) = 1;
|
|||
|
DECL_ARTIFICIAL (fn) = 1;
|
|||
|
pushdecl_top_level (fn);
|
|||
|
make_function_rtl (fn);
|
|||
|
assemble_external (fn);
|
|||
|
pop_obstacks ();
|
|||
|
}
|
|||
|
|
|||
|
fn = build_call (fn, TREE_TYPE (TREE_TYPE (fn)), elems);
|
|||
|
expand_expr_stmt (fn);
|
|||
|
}
|
|||
|
|
|||
|
/* Build an initializer for a __pointer_type_info node. */
|
|||
|
|
|||
|
static void
|
|||
|
expand_ptr_desc (tdecl, type)
|
|||
|
tree tdecl;
|
|||
|
tree type;
|
|||
|
{
|
|||
|
tree t, elems, fn;
|
|||
|
char *name = build_overload_name (type, 1, 1);
|
|||
|
tree name_string = combine_strings (build_string (strlen (name)+1, name));
|
|||
|
|
|||
|
type = TREE_TYPE (type);
|
|||
|
expand_expr_stmt (get_typeid_1 (type));
|
|||
|
t = decay_conversion (get_tinfo_var (type));
|
|||
|
elems = tree_cons
|
|||
|
(NULL_TREE, decay_conversion (tdecl), tree_cons
|
|||
|
(NULL_TREE, decay_conversion (name_string), tree_cons
|
|||
|
(NULL_TREE, t, NULL_TREE)));
|
|||
|
|
|||
|
fn = get_identifier ("__rtti_ptr");
|
|||
|
if (IDENTIFIER_GLOBAL_VALUE (fn))
|
|||
|
fn = IDENTIFIER_GLOBAL_VALUE (fn);
|
|||
|
else
|
|||
|
{
|
|||
|
tree tmp;
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
tmp = tree_cons
|
|||
|
(NULL_TREE, ptr_type_node, tree_cons
|
|||
|
(NULL_TREE, const_string_type_node, tree_cons
|
|||
|
(NULL_TREE, build_pointer_type (type_info_type_node),
|
|||
|
void_list_node)));
|
|||
|
tmp = build_function_type (void_type_node, tmp);
|
|||
|
|
|||
|
fn = build_lang_decl (FUNCTION_DECL, fn, tmp);
|
|||
|
DECL_EXTERNAL (fn) = 1;
|
|||
|
TREE_PUBLIC (fn) = 1;
|
|||
|
DECL_ARTIFICIAL (fn) = 1;
|
|||
|
pushdecl_top_level (fn);
|
|||
|
make_function_rtl (fn);
|
|||
|
assemble_external (fn);
|
|||
|
pop_obstacks ();
|
|||
|
}
|
|||
|
|
|||
|
fn = build_call (fn, TREE_TYPE (TREE_TYPE (fn)), elems);
|
|||
|
expand_expr_stmt (fn);
|
|||
|
}
|
|||
|
|
|||
|
/* Build an initializer for a __attr_type_info node. */
|
|||
|
|
|||
|
static void
|
|||
|
expand_attr_desc (tdecl, type)
|
|||
|
tree tdecl;
|
|||
|
tree type;
|
|||
|
{
|
|||
|
tree elems, t, fn;
|
|||
|
char *name = build_overload_name (type, 1, 1);
|
|||
|
tree name_string = combine_strings (build_string (strlen (name)+1, name));
|
|||
|
tree attrval = build_int_2
|
|||
|
(TYPE_READONLY (type) | TYPE_VOLATILE (type) * 2, 0);
|
|||
|
|
|||
|
expand_expr_stmt (get_typeid_1 (TYPE_MAIN_VARIANT (type)));
|
|||
|
t = decay_conversion (get_tinfo_var (TYPE_MAIN_VARIANT (type)));
|
|||
|
elems = tree_cons
|
|||
|
(NULL_TREE, decay_conversion (tdecl), tree_cons
|
|||
|
(NULL_TREE, decay_conversion (name_string), tree_cons
|
|||
|
(NULL_TREE, attrval, expr_tree_cons (NULL_TREE, t, NULL_TREE))));
|
|||
|
|
|||
|
fn = get_identifier ("__rtti_attr");
|
|||
|
if (IDENTIFIER_GLOBAL_VALUE (fn))
|
|||
|
fn = IDENTIFIER_GLOBAL_VALUE (fn);
|
|||
|
else
|
|||
|
{
|
|||
|
tree tmp;
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
tmp = tree_cons
|
|||
|
(NULL_TREE, ptr_type_node, tree_cons
|
|||
|
(NULL_TREE, const_string_type_node, tree_cons
|
|||
|
(NULL_TREE, integer_type_node, tree_cons
|
|||
|
(NULL_TREE, build_pointer_type (type_info_type_node),
|
|||
|
void_list_node))));
|
|||
|
tmp = build_function_type (void_type_node, tmp);
|
|||
|
|
|||
|
fn = build_lang_decl (FUNCTION_DECL, fn, tmp);
|
|||
|
DECL_EXTERNAL (fn) = 1;
|
|||
|
TREE_PUBLIC (fn) = 1;
|
|||
|
DECL_ARTIFICIAL (fn) = 1;
|
|||
|
pushdecl_top_level (fn);
|
|||
|
make_function_rtl (fn);
|
|||
|
assemble_external (fn);
|
|||
|
pop_obstacks ();
|
|||
|
}
|
|||
|
|
|||
|
fn = build_call (fn, TREE_TYPE (TREE_TYPE (fn)), elems);
|
|||
|
expand_expr_stmt (fn);
|
|||
|
}
|
|||
|
|
|||
|
/* Build an initializer for a type_info node that just has a name. */
|
|||
|
|
|||
|
static void
|
|||
|
expand_generic_desc (tdecl, type, fnname)
|
|||
|
tree tdecl;
|
|||
|
tree type;
|
|||
|
char *fnname;
|
|||
|
{
|
|||
|
char *name = build_overload_name (type, 1, 1);
|
|||
|
tree name_string = combine_strings (build_string (strlen (name)+1, name));
|
|||
|
tree elems = tree_cons
|
|||
|
(NULL_TREE, decay_conversion (tdecl), tree_cons
|
|||
|
(NULL_TREE, decay_conversion (name_string), NULL_TREE));
|
|||
|
|
|||
|
tree fn = get_identifier (fnname);
|
|||
|
if (IDENTIFIER_GLOBAL_VALUE (fn))
|
|||
|
fn = IDENTIFIER_GLOBAL_VALUE (fn);
|
|||
|
else
|
|||
|
{
|
|||
|
tree tmp;
|
|||
|
push_obstacks (&permanent_obstack, &permanent_obstack);
|
|||
|
tmp = tree_cons
|
|||
|
(NULL_TREE, ptr_type_node, tree_cons
|
|||
|
(NULL_TREE, const_string_type_node, void_list_node));
|
|||
|
tmp = build_function_type (void_type_node, tmp);
|
|||
|
|
|||
|
fn = build_lang_decl (FUNCTION_DECL, fn, tmp);
|
|||
|
DECL_EXTERNAL (fn) = 1;
|
|||
|
TREE_PUBLIC (fn) = 1;
|
|||
|
DECL_ARTIFICIAL (fn) = 1;
|
|||
|
pushdecl_top_level (fn);
|
|||
|
make_function_rtl (fn);
|
|||
|
assemble_external (fn);
|
|||
|
pop_obstacks ();
|
|||
|
}
|
|||
|
|
|||
|
fn = build_call (fn, TREE_TYPE (TREE_TYPE (fn)), elems);
|
|||
|
expand_expr_stmt (fn);
|
|||
|
}
|
|||
|
|
|||
|
/* Generate the code for a type_info initialization function.
|
|||
|
Note that we take advantage of the passage
|
|||
|
|
|||
|
5.2.7 Type identification [expr.typeid]
|
|||
|
|
|||
|
Whether or not the destructor is called for the type_info object at the
|
|||
|
end of the program is unspecified.
|
|||
|
|
|||
|
and don't bother to arrange for these objects to be destroyed. It
|
|||
|
doesn't matter, anyway, since the destructors don't do anything.
|
|||
|
|
|||
|
This must only be called from toplevel (i.e. from finish_file)! */
|
|||
|
|
|||
|
void
|
|||
|
synthesize_tinfo_fn (fndecl)
|
|||
|
tree fndecl;
|
|||
|
{
|
|||
|
tree type = TREE_TYPE (DECL_NAME (fndecl));
|
|||
|
tree tmp, addr, tdecl;
|
|||
|
|
|||
|
if (at_eof)
|
|||
|
{
|
|||
|
import_export_decl (fndecl);
|
|||
|
if (DECL_REALLY_EXTERN (fndecl))
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
tdecl = get_tinfo_var (type);
|
|||
|
DECL_EXTERNAL (tdecl) = 0;
|
|||
|
TREE_STATIC (tdecl) = 1;
|
|||
|
DECL_COMMON (tdecl) = 1;
|
|||
|
TREE_USED (tdecl) = 1;
|
|||
|
DECL_ALIGN (tdecl) = TYPE_ALIGN (ptr_type_node);
|
|||
|
cp_finish_decl (tdecl, NULL_TREE, NULL_TREE, 0, 0);
|
|||
|
|
|||
|
start_function (NULL_TREE, fndecl, NULL_TREE, 1);
|
|||
|
store_parm_decls ();
|
|||
|
clear_last_expr ();
|
|||
|
push_momentary ();
|
|||
|
|
|||
|
/* If the first word of the array (the vtable) is non-zero, we've already
|
|||
|
initialized the object, so don't do it again. */
|
|||
|
addr = decay_conversion (tdecl);
|
|||
|
tmp = cp_convert (build_pointer_type (ptr_type_node), addr);
|
|||
|
tmp = build_indirect_ref (tmp, 0);
|
|||
|
tmp = build_binary_op (EQ_EXPR, tmp, integer_zero_node, 1);
|
|||
|
expand_start_cond (tmp, 0);
|
|||
|
|
|||
|
if (TREE_CODE (type) == FUNCTION_TYPE)
|
|||
|
expand_generic_desc (tdecl, type, "__rtti_func");
|
|||
|
else if (TREE_CODE (type) == ARRAY_TYPE)
|
|||
|
expand_generic_desc (tdecl, type, "__rtti_array");
|
|||
|
else if (TYPE_VOLATILE (type) || TYPE_READONLY (type))
|
|||
|
expand_attr_desc (tdecl, type);
|
|||
|
else if (TREE_CODE (type) == POINTER_TYPE)
|
|||
|
{
|
|||
|
if (TREE_CODE (TREE_TYPE (type)) == OFFSET_TYPE)
|
|||
|
expand_generic_desc (tdecl, type, "__rtti_ptmd");
|
|||
|
else if (TREE_CODE (TREE_TYPE (type)) == METHOD_TYPE)
|
|||
|
expand_generic_desc (tdecl, type, "__rtti_ptmf");
|
|||
|
else
|
|||
|
expand_ptr_desc (tdecl, type);
|
|||
|
}
|
|||
|
else if (TYPE_PTRMEMFUNC_P (type))
|
|||
|
expand_generic_desc (tdecl, type, "__rtti_ptmf");
|
|||
|
else if (IS_AGGR_TYPE (type))
|
|||
|
{
|
|||
|
if (CLASSTYPE_N_BASECLASSES (type) == 0)
|
|||
|
expand_generic_desc (tdecl, type, "__rtti_user");
|
|||
|
else if (! TYPE_USES_COMPLEX_INHERITANCE (type)
|
|||
|
&& (TREE_VIA_PUBLIC
|
|||
|
(TREE_VEC_ELT (TYPE_BINFO_BASETYPES (type), 0))))
|
|||
|
expand_si_desc (tdecl, type);
|
|||
|
else
|
|||
|
expand_class_desc (tdecl, type);
|
|||
|
}
|
|||
|
else if (TREE_CODE (type) == ENUMERAL_TYPE)
|
|||
|
expand_generic_desc (tdecl, type, "__rtti_user");
|
|||
|
else
|
|||
|
my_friendly_abort (252);
|
|||
|
|
|||
|
expand_end_cond ();
|
|||
|
|
|||
|
/* OK, now return the type_info object. */
|
|||
|
tmp = cp_convert (build_pointer_type (type_info_type_node), addr);
|
|||
|
tmp = build_indirect_ref (tmp, 0);
|
|||
|
c_expand_return (tmp);
|
|||
|
finish_function (lineno, 0, 0);
|
|||
|
}
|