cp-tree.def (THUNK_DECL): Discuss vcall indices.
* cp-tree.def (THUNK_DECL): Discuss vcall indices. * cp-tree.h (BINFO_VIRTUALS): Update documentation. (BF_DELTA): New macro. (BF_VCALL_INDEX): Likewise. (BF_FN): Likewise. (THUNK_VCALL_OFFSET): Likewise. (make_thunk): Change prototype. * class.c (build_vtable_entry): Integrate build_vtable_entry_for_fn. Handle vcall indices. (build_vtable_entry_for_fn): Remove. (set_rtti_entry): Handle vcall indices. Use BF_DELTA, BF_VCALL_INDEX, BF_FN. (modify_vtable_entry): Integrate common code from modify_one_vtable and dfs_fixup_vtable_deltas. (add_virtual_function): Set BF_VCALL_INDEX. (build_vtbl_initializer): Simplify. Use BF_DELTA, BF_VCALL_INDEX, and BF_FN. (modify_one_vtable): Simplify. (dfs_fixup_vtable_deltas): Likewise. (override_one_vtable): Use BF_DELTA, BF_VCALL_INDEX, BF_FN. * method.c (make_thunk): Handle vcall indices. From-SVN: r31684
This commit is contained in:
parent
4fda2521e9
commit
c0bbf652c5
@ -1,3 +1,27 @@
|
|||||||
|
2000-01-27 Mark Mitchell <mark@codesourcery.com>
|
||||||
|
|
||||||
|
* cp-tree.def (THUNK_DECL): Discuss vcall indices.
|
||||||
|
* cp-tree.h (BINFO_VIRTUALS): Update documentation.
|
||||||
|
(BF_DELTA): New macro.
|
||||||
|
(BF_VCALL_INDEX): Likewise.
|
||||||
|
(BF_FN): Likewise.
|
||||||
|
(THUNK_VCALL_OFFSET): Likewise.
|
||||||
|
(make_thunk): Change prototype.
|
||||||
|
* class.c (build_vtable_entry): Integrate
|
||||||
|
build_vtable_entry_for_fn. Handle vcall indices.
|
||||||
|
(build_vtable_entry_for_fn): Remove.
|
||||||
|
(set_rtti_entry): Handle vcall indices. Use BF_DELTA,
|
||||||
|
BF_VCALL_INDEX, BF_FN.
|
||||||
|
(modify_vtable_entry): Integrate common code from
|
||||||
|
modify_one_vtable and dfs_fixup_vtable_deltas.
|
||||||
|
(add_virtual_function): Set BF_VCALL_INDEX.
|
||||||
|
(build_vtbl_initializer): Simplify. Use BF_DELTA, BF_VCALL_INDEX,
|
||||||
|
and BF_FN.
|
||||||
|
(modify_one_vtable): Simplify.
|
||||||
|
(dfs_fixup_vtable_deltas): Likewise.
|
||||||
|
(override_one_vtable): Use BF_DELTA, BF_VCALL_INDEX, BF_FN.
|
||||||
|
* method.c (make_thunk): Handle vcall indices.
|
||||||
|
|
||||||
2000-01-28 Nathan Sidwell <sidwell@codesourcery.com>
|
2000-01-28 Nathan Sidwell <sidwell@codesourcery.com>
|
||||||
|
|
||||||
Compiler side new abi rtti (not enabled).
|
Compiler side new abi rtti (not enabled).
|
||||||
|
306
gcc/cp/class.c
306
gcc/cp/class.c
@ -75,7 +75,7 @@ static class_stack_node_t current_class_stack;
|
|||||||
static tree get_vfield_name PARAMS ((tree));
|
static tree get_vfield_name PARAMS ((tree));
|
||||||
static void finish_struct_anon PARAMS ((tree));
|
static void finish_struct_anon PARAMS ((tree));
|
||||||
static tree build_vbase_pointer PARAMS ((tree, tree));
|
static tree build_vbase_pointer PARAMS ((tree, tree));
|
||||||
static tree build_vtable_entry PARAMS ((tree, tree));
|
static tree build_vtable_entry PARAMS ((tree, tree, tree));
|
||||||
static tree get_vtable_name PARAMS ((tree));
|
static tree get_vtable_name PARAMS ((tree));
|
||||||
static tree get_derived_offset PARAMS ((tree, tree));
|
static tree get_derived_offset PARAMS ((tree, tree));
|
||||||
static tree get_basefndecls PARAMS ((tree, tree));
|
static tree get_basefndecls PARAMS ((tree, tree));
|
||||||
@ -85,8 +85,7 @@ static void prepare_fresh_vtable PARAMS ((tree, tree));
|
|||||||
static tree dfs_fixup_vtable_deltas PARAMS ((tree, void *));
|
static tree dfs_fixup_vtable_deltas PARAMS ((tree, void *));
|
||||||
static tree dfs_finish_vtbls PARAMS ((tree, void *));
|
static tree dfs_finish_vtbls PARAMS ((tree, void *));
|
||||||
static void finish_vtbls PARAMS ((tree));
|
static void finish_vtbls PARAMS ((tree));
|
||||||
static void modify_vtable_entry PARAMS ((tree, tree, tree));
|
static void modify_vtable_entry PARAMS ((tree, tree, tree, tree));
|
||||||
static tree get_vtable_entry_n PARAMS ((tree, unsigned HOST_WIDE_INT));
|
|
||||||
static void add_virtual_function PARAMS ((tree *, tree *, int *, tree, tree));
|
static void add_virtual_function PARAMS ((tree *, tree *, int *, tree, tree));
|
||||||
static tree delete_duplicate_fields_1 PARAMS ((tree, tree));
|
static tree delete_duplicate_fields_1 PARAMS ((tree, tree));
|
||||||
static void delete_duplicate_fields PARAMS ((tree));
|
static void delete_duplicate_fields PARAMS ((tree));
|
||||||
@ -114,7 +113,6 @@ static tree fixed_type_or_null PARAMS ((tree, int *));
|
|||||||
static tree resolve_address_of_overloaded_function PARAMS ((tree, tree, int,
|
static tree resolve_address_of_overloaded_function PARAMS ((tree, tree, int,
|
||||||
int, tree));
|
int, tree));
|
||||||
static void build_vtable_entry_ref PARAMS ((tree, tree, tree));
|
static void build_vtable_entry_ref PARAMS ((tree, tree, tree));
|
||||||
static tree build_vtable_entry_for_fn PARAMS ((tree, tree));
|
|
||||||
static tree build_vtbl_initializer PARAMS ((tree, tree));
|
static tree build_vtbl_initializer PARAMS ((tree, tree));
|
||||||
static int count_fields PARAMS ((tree));
|
static int count_fields PARAMS ((tree));
|
||||||
static int add_fields_to_vec PARAMS ((tree, tree, int));
|
static int add_fields_to_vec PARAMS ((tree, tree, int));
|
||||||
@ -683,23 +681,37 @@ build_vbase_path (code, type, expr, path, nonnull)
|
|||||||
|
|
||||||
/* Virtual function things. */
|
/* Virtual function things. */
|
||||||
|
|
||||||
/* Build an entry in the virtual function table.
|
/* Build an entry in the virtual function table. DELTA is the offset
|
||||||
DELTA is the offset for the `this' pointer.
|
for the `this' pointer. VCALL_INDEX is the vtable index containing
|
||||||
PFN is an ADDR_EXPR containing a pointer to the virtual function.
|
the vcall offset; zero if none. FNDECL is the virtual function
|
||||||
Note that the index (DELTA2) in the virtual function table
|
itself. */
|
||||||
is always 0. */
|
|
||||||
|
|
||||||
static tree
|
static tree
|
||||||
build_vtable_entry (delta, pfn)
|
build_vtable_entry (delta, vcall_index, fndecl)
|
||||||
tree delta, pfn;
|
tree delta;
|
||||||
|
tree vcall_index;
|
||||||
|
tree fndecl;
|
||||||
{
|
{
|
||||||
|
tree pfn;
|
||||||
|
|
||||||
|
/* Take the address of the function, considering it to be of an
|
||||||
|
appropriate generic type. */
|
||||||
|
pfn = build1 (ADDR_EXPR, vfunc_ptr_type_node, fndecl);
|
||||||
|
/* The address of a function can't change. */
|
||||||
|
TREE_CONSTANT (pfn) = 1;
|
||||||
|
|
||||||
if (flag_vtable_thunks)
|
if (flag_vtable_thunks)
|
||||||
{
|
{
|
||||||
HOST_WIDE_INT idelta = TREE_INT_CST_LOW (delta);
|
HOST_WIDE_INT idelta;
|
||||||
if (idelta && ! DECL_PURE_VIRTUAL_P (TREE_OPERAND (pfn, 0)))
|
HOST_WIDE_INT ivindex;
|
||||||
|
|
||||||
|
idelta = TREE_INT_CST_LOW (delta);
|
||||||
|
ivindex = TREE_INT_CST_LOW (vcall_index);
|
||||||
|
if ((idelta || ivindex)
|
||||||
|
&& ! DECL_PURE_VIRTUAL_P (TREE_OPERAND (pfn, 0)))
|
||||||
{
|
{
|
||||||
pfn = build1 (ADDR_EXPR, vtable_entry_type,
|
pfn = make_thunk (pfn, idelta, ivindex);
|
||||||
make_thunk (pfn, idelta));
|
pfn = build1 (ADDR_EXPR, vtable_entry_type, pfn);
|
||||||
TREE_READONLY (pfn) = 1;
|
TREE_READONLY (pfn) = 1;
|
||||||
TREE_CONSTANT (pfn) = 1;
|
TREE_CONSTANT (pfn) = 1;
|
||||||
}
|
}
|
||||||
@ -716,6 +728,9 @@ build_vtable_entry (delta, pfn)
|
|||||||
build_tree_list (NULL_TREE, pfn)));
|
build_tree_list (NULL_TREE, pfn)));
|
||||||
tree entry = build (CONSTRUCTOR, vtable_entry_type, NULL_TREE, elems);
|
tree entry = build (CONSTRUCTOR, vtable_entry_type, NULL_TREE, elems);
|
||||||
|
|
||||||
|
/* We don't use vcall offsets when not using vtable thunks. */
|
||||||
|
my_friendly_assert (integer_zerop (vcall_index), 20000125);
|
||||||
|
|
||||||
/* DELTA used to be constructed by `size_int' and/or size_binop,
|
/* DELTA used to be constructed by `size_int' and/or size_binop,
|
||||||
which caused overflow problems when it was negative. That should
|
which caused overflow problems when it was negative. That should
|
||||||
be fixed now. */
|
be fixed now. */
|
||||||
@ -740,25 +755,6 @@ build_vtable_entry (delta, pfn)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Build a vtable entry for FNDECL. DELTA is the amount by which we
|
|
||||||
must adjust the this pointer when calling F. */
|
|
||||||
|
|
||||||
static tree
|
|
||||||
build_vtable_entry_for_fn (delta, fndecl)
|
|
||||||
tree delta;
|
|
||||||
tree fndecl;
|
|
||||||
{
|
|
||||||
tree pfn;
|
|
||||||
|
|
||||||
/* Take the address of the function, considering it to be of an
|
|
||||||
appropriate generic type. */
|
|
||||||
pfn = build1 (ADDR_EXPR, vfunc_ptr_type_node, fndecl);
|
|
||||||
/* The address of a function can't change. */
|
|
||||||
TREE_CONSTANT (pfn) = 1;
|
|
||||||
/* Now build the vtable entry itself. */
|
|
||||||
return build_vtable_entry (delta, pfn);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* We want to give the assembler the vtable identifier as well as
|
/* We want to give the assembler the vtable identifier as well as
|
||||||
the offset to the function pointer. So we generate
|
the offset to the function pointer. So we generate
|
||||||
|
|
||||||
@ -984,16 +980,18 @@ set_rtti_entry (virtuals, offset, type)
|
|||||||
if (flag_vtable_thunks)
|
if (flag_vtable_thunks)
|
||||||
{
|
{
|
||||||
/* The first slot holds the offset. */
|
/* The first slot holds the offset. */
|
||||||
TREE_PURPOSE (virtuals) = offset;
|
BF_DELTA (virtuals) = offset;
|
||||||
|
BF_VCALL_INDEX (virtuals) = integer_zero_node;
|
||||||
|
|
||||||
/* The next node holds the decl. */
|
/* The next node holds the decl. */
|
||||||
virtuals = TREE_CHAIN (virtuals);
|
virtuals = TREE_CHAIN (virtuals);
|
||||||
offset = integer_zero_node;
|
offset = integer_zero_node;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This slot holds the decl. */
|
/* This slot holds the function to call. */
|
||||||
TREE_PURPOSE (virtuals) = offset;
|
BF_DELTA (virtuals) = offset;
|
||||||
TREE_VALUE (virtuals) = decl;
|
BF_VCALL_INDEX (virtuals) = integer_zero_node;
|
||||||
|
BF_FN (virtuals) = decl;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get the VAR_DECL of the vtable for TYPE. TYPE need not be polymorphic,
|
/* Get the VAR_DECL of the vtable for TYPE. TYPE need not be polymorphic,
|
||||||
@ -1263,42 +1261,71 @@ prepare_fresh_vtable (binfo, for_type)
|
|||||||
SET_BINFO_NEW_VTABLE_MARKED (binfo);
|
SET_BINFO_NEW_VTABLE_MARKED (binfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Change the offset for the FNDECL entry to NEW_OFFSET. Also update
|
/* Make V, an entry on the BINFO_VIRTUALS list for BINFO (which is in
|
||||||
DECL_VINDEX (FNDECL). */
|
the hierarchy dominated by T) list FNDECL as its BF_FN. */
|
||||||
|
|
||||||
static void
|
static void
|
||||||
modify_vtable_entry (old_entry_in_list, new_offset, fndecl)
|
modify_vtable_entry (t, binfo, fndecl, v)
|
||||||
tree old_entry_in_list, new_offset, fndecl;
|
tree t;
|
||||||
|
tree binfo;
|
||||||
|
tree fndecl;
|
||||||
|
tree v;
|
||||||
{
|
{
|
||||||
tree base_fndecl = TREE_VALUE (old_entry_in_list);
|
tree base_offset, offset;
|
||||||
|
tree context = DECL_CLASS_CONTEXT (fndecl);
|
||||||
|
tree vfield = TYPE_VFIELD (t);
|
||||||
|
tree this_offset;
|
||||||
|
tree vcall_index;
|
||||||
|
|
||||||
/* Update the entry. */
|
offset = get_class_offset (context, t, binfo, fndecl);
|
||||||
TREE_PURPOSE (old_entry_in_list) = new_offset;
|
|
||||||
TREE_VALUE (old_entry_in_list) = fndecl;
|
|
||||||
|
|
||||||
/* Now assign virtual dispatch information, if unset. We can
|
/* Find the right offset for ythe this pointer based on the
|
||||||
dispatch this, through any overridden base function. */
|
base class we just found. We have to take into
|
||||||
if (TREE_CODE (DECL_VINDEX (fndecl)) != INTEGER_CST)
|
consideration the virtual base class pointers that we
|
||||||
|
stick in before the virtual function table pointer.
|
||||||
|
|
||||||
|
Also, we want just the delta between the most base class
|
||||||
|
that we derived this vfield from and us. */
|
||||||
|
base_offset
|
||||||
|
= size_binop (PLUS_EXPR,
|
||||||
|
get_derived_offset (binfo,
|
||||||
|
DECL_VIRTUAL_CONTEXT (BF_FN (v))),
|
||||||
|
BINFO_OFFSET (binfo));
|
||||||
|
this_offset = ssize_binop (MINUS_EXPR, offset, base_offset);
|
||||||
|
vcall_index = integer_zero_node;
|
||||||
|
|
||||||
|
if (fndecl != BF_FN (v)
|
||||||
|
|| !tree_int_cst_equal (this_offset, BF_DELTA (v))
|
||||||
|
|| !tree_int_cst_equal (vcall_index, BF_VCALL_INDEX (v)))
|
||||||
{
|
{
|
||||||
DECL_VINDEX (fndecl) = DECL_VINDEX (base_fndecl);
|
tree base_fndecl;
|
||||||
DECL_VIRTUAL_CONTEXT (fndecl) = DECL_VIRTUAL_CONTEXT (base_fndecl);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Access the virtual function table entry N. VIRTUALS is the virtual
|
/* Make sure we can modify the derived association with immunity. */
|
||||||
function table's initializer. */
|
if (binfo == TYPE_BINFO (t))
|
||||||
|
/* In this case, it is *type*'s vtable we are modifying. We
|
||||||
|
start with the approximation that it's vtable is that of
|
||||||
|
the immediate base class. */
|
||||||
|
build_vtable (TYPE_BINFO (DECL_CONTEXT (vfield)), t);
|
||||||
|
else
|
||||||
|
/* This is our very own copy of `basetype' to play with.
|
||||||
|
Later, we will fill in all the virtual functions that
|
||||||
|
override the virtual functions in these base classes which
|
||||||
|
are not defined by the current type. */
|
||||||
|
prepare_fresh_vtable (binfo, t);
|
||||||
|
|
||||||
static tree
|
base_fndecl = BF_FN (v);
|
||||||
get_vtable_entry_n (virtuals, n)
|
BF_DELTA (v) = this_offset;
|
||||||
tree virtuals;
|
BF_VCALL_INDEX (v) = vcall_index;
|
||||||
unsigned HOST_WIDE_INT n;
|
BF_FN (v) = fndecl;
|
||||||
{
|
|
||||||
while (n > 0)
|
/* Now assign virtual dispatch information, if unset. We can
|
||||||
{
|
dispatch this, through any overridden base function. */
|
||||||
--n;
|
if (TREE_CODE (DECL_VINDEX (fndecl)) != INTEGER_CST)
|
||||||
virtuals = TREE_CHAIN (virtuals);
|
{
|
||||||
|
DECL_VINDEX (fndecl) = DECL_VINDEX (base_fndecl);
|
||||||
|
DECL_VIRTUAL_CONTEXT (fndecl) = DECL_VIRTUAL_CONTEXT (base_fndecl);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return virtuals;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Call this function whenever its known that a vtable for T is going
|
/* Call this function whenever its known that a vtable for T is going
|
||||||
@ -1340,14 +1367,20 @@ add_virtual_function (new_virtuals_p, overridden_virtuals_p,
|
|||||||
tree fndecl;
|
tree fndecl;
|
||||||
tree t; /* Structure type. */
|
tree t; /* Structure type. */
|
||||||
{
|
{
|
||||||
|
tree new_virtual;
|
||||||
|
|
||||||
/* If this function doesn't override anything from a base class, we
|
/* If this function doesn't override anything from a base class, we
|
||||||
can just assign it a new DECL_VINDEX now. Otherwise, if it does
|
can just assign it a new DECL_VINDEX now. Otherwise, if it does
|
||||||
override something, we keep it around and assign its DECL_VINDEX
|
override something, we keep it around and assign its DECL_VINDEX
|
||||||
later, in modify_all_vtables. */
|
later, in modify_all_vtables. */
|
||||||
if (TREE_CODE (DECL_VINDEX (fndecl)) == INTEGER_CST)
|
if (TREE_CODE (DECL_VINDEX (fndecl)) == INTEGER_CST)
|
||||||
/* We've already dealt with this function. */
|
/* We've already dealt with this function. */
|
||||||
;
|
return;
|
||||||
else if (DECL_VINDEX (fndecl) == error_mark_node)
|
|
||||||
|
new_virtual = build_tree_list (integer_zero_node, fndecl);
|
||||||
|
BF_VCALL_INDEX (new_virtual) = integer_zero_node;
|
||||||
|
|
||||||
|
if (DECL_VINDEX (fndecl) == error_mark_node)
|
||||||
{
|
{
|
||||||
/* FNDECL is a new virtual function; it doesn't override any
|
/* FNDECL is a new virtual function; it doesn't override any
|
||||||
virtual function in a base class. */
|
virtual function in a base class. */
|
||||||
@ -1362,15 +1395,15 @@ add_virtual_function (new_virtuals_p, overridden_virtuals_p,
|
|||||||
DECL_VIRTUAL_CONTEXT (fndecl) = t;
|
DECL_VIRTUAL_CONTEXT (fndecl) = t;
|
||||||
|
|
||||||
/* Save the state we've computed on the NEW_VIRTUALS list. */
|
/* Save the state we've computed on the NEW_VIRTUALS list. */
|
||||||
*new_virtuals_p = tree_cons (integer_zero_node,
|
TREE_CHAIN (new_virtual) = *new_virtuals_p;
|
||||||
fndecl,
|
*new_virtuals_p = new_virtual;
|
||||||
*new_virtuals_p);
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* FNDECL overrides a function from a base class. */
|
||||||
|
TREE_CHAIN (new_virtual) = *overridden_virtuals_p;
|
||||||
|
*overridden_virtuals_p = new_virtual;
|
||||||
}
|
}
|
||||||
else if (TREE_CODE (DECL_VINDEX (fndecl)) != INTEGER_CST)
|
|
||||||
/* FNDECL overrides a function from a base class. */
|
|
||||||
*overridden_virtuals_p = tree_cons (NULL_TREE,
|
|
||||||
fndecl,
|
|
||||||
*overridden_virtuals_p);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
extern struct obstack *current_obstack;
|
extern struct obstack *current_obstack;
|
||||||
@ -2615,7 +2648,6 @@ build_vtbl_initializer (binfo, t)
|
|||||||
we can put it in the vtable. */
|
we can put it in the vtable. */
|
||||||
init = build1 (NOP_EXPR, vfunc_ptr_type_node, offset);
|
init = build1 (NOP_EXPR, vfunc_ptr_type_node, offset);
|
||||||
TREE_CONSTANT (init) = 1;
|
TREE_CONSTANT (init) = 1;
|
||||||
init = build_vtable_entry (integer_zero_node, init);
|
|
||||||
inits = tree_cons (NULL_TREE, init, inits);
|
inits = tree_cons (NULL_TREE, init, inits);
|
||||||
|
|
||||||
v = TREE_CHAIN (v);
|
v = TREE_CHAIN (v);
|
||||||
@ -2645,13 +2677,15 @@ build_vtbl_initializer (binfo, t)
|
|||||||
while (v)
|
while (v)
|
||||||
{
|
{
|
||||||
tree delta;
|
tree delta;
|
||||||
|
tree vcall_index;
|
||||||
tree fn;
|
tree fn;
|
||||||
tree init;
|
tree init;
|
||||||
|
|
||||||
/* Pull the offset for `this', and the function to call, out of
|
/* Pull the offset for `this', and the function to call, out of
|
||||||
the list. */
|
the list. */
|
||||||
delta = TREE_PURPOSE (v);
|
delta = BF_DELTA (v);
|
||||||
fn = TREE_VALUE (v);
|
vcall_index = BF_VCALL_INDEX (v);
|
||||||
|
fn = BF_FN (v);
|
||||||
my_friendly_assert (TREE_CODE (delta) == INTEGER_CST, 19990727);
|
my_friendly_assert (TREE_CODE (delta) == INTEGER_CST, 19990727);
|
||||||
my_friendly_assert (TREE_CODE (fn) == FUNCTION_DECL, 19990727);
|
my_friendly_assert (TREE_CODE (fn) == FUNCTION_DECL, 19990727);
|
||||||
|
|
||||||
@ -2661,7 +2695,7 @@ build_vtbl_initializer (binfo, t)
|
|||||||
fn = abort_fndecl;
|
fn = abort_fndecl;
|
||||||
|
|
||||||
/* Package up that information for the vtable. */
|
/* Package up that information for the vtable. */
|
||||||
init = build_vtable_entry_for_fn (delta, fn);
|
init = build_vtable_entry (delta, vcall_index, fn);
|
||||||
/* And add it to the chain of initializers. */
|
/* And add it to the chain of initializers. */
|
||||||
inits = tree_cons (NULL_TREE, init, inits);
|
inits = tree_cons (NULL_TREE, init, inits);
|
||||||
|
|
||||||
@ -2928,11 +2962,11 @@ modify_one_vtable (binfo, t, fndecl)
|
|||||||
if (fndecl == NULL_TREE)
|
if (fndecl == NULL_TREE)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
virtuals = skip_rtti_stuff (binfo, BINFO_TYPE (binfo), &n);
|
for (virtuals = skip_rtti_stuff (binfo, BINFO_TYPE (binfo), &n);
|
||||||
|
virtuals;
|
||||||
while (virtuals)
|
virtuals = TREE_CHAIN (virtuals), ++n)
|
||||||
{
|
{
|
||||||
tree current_fndecl = TREE_VALUE (virtuals);
|
tree current_fndecl = BF_FN (virtuals);
|
||||||
|
|
||||||
/* We should never have an instance of __pure_virtual on the
|
/* We should never have an instance of __pure_virtual on the
|
||||||
BINFO_VIRTUALS list. If we do, then we will never notice
|
BINFO_VIRTUALS list. If we do, then we will never notice
|
||||||
@ -2942,47 +2976,7 @@ modify_one_vtable (binfo, t, fndecl)
|
|||||||
19990727);
|
19990727);
|
||||||
|
|
||||||
if (current_fndecl && overrides (fndecl, current_fndecl))
|
if (current_fndecl && overrides (fndecl, current_fndecl))
|
||||||
{
|
modify_vtable_entry (t, binfo, fndecl, virtuals);
|
||||||
tree base_offset, offset;
|
|
||||||
tree context = DECL_CLASS_CONTEXT (fndecl);
|
|
||||||
tree vfield = TYPE_VFIELD (t);
|
|
||||||
tree this_offset;
|
|
||||||
|
|
||||||
offset = get_class_offset (context, t, binfo, fndecl);
|
|
||||||
|
|
||||||
/* Find the right offset for the this pointer based on the
|
|
||||||
base class we just found. We have to take into
|
|
||||||
consideration the virtual base class pointers that we
|
|
||||||
stick in before the virtual function table pointer.
|
|
||||||
|
|
||||||
Also, we want just the delta between the most base class
|
|
||||||
that we derived this vfield from and us. */
|
|
||||||
base_offset = size_binop (PLUS_EXPR,
|
|
||||||
get_derived_offset (binfo, DECL_CONTEXT (current_fndecl)),
|
|
||||||
BINFO_OFFSET (binfo));
|
|
||||||
this_offset = ssize_binop (MINUS_EXPR, offset, base_offset);
|
|
||||||
|
|
||||||
if (binfo == TYPE_BINFO (t))
|
|
||||||
/* In this case, it is *type*'s vtable we are modifying.
|
|
||||||
We start with the approximation that it's vtable is
|
|
||||||
that of the immediate base class. */
|
|
||||||
build_vtable (TYPE_BINFO (DECL_CONTEXT (vfield)), t);
|
|
||||||
else
|
|
||||||
/* This is our very own copy of `basetype' to play with.
|
|
||||||
Later, we will fill in all the virtual functions that
|
|
||||||
override the virtual functions in these base classes
|
|
||||||
which are not defined by the current type. */
|
|
||||||
prepare_fresh_vtable (binfo, t);
|
|
||||||
|
|
||||||
#ifdef NOTQUITE
|
|
||||||
cp_warning ("in %D", DECL_NAME (BINFO_VTABLE (binfo)));
|
|
||||||
#endif
|
|
||||||
modify_vtable_entry (get_vtable_entry_n (BINFO_VIRTUALS (binfo), n),
|
|
||||||
this_offset,
|
|
||||||
fndecl);
|
|
||||||
}
|
|
||||||
++n;
|
|
||||||
virtuals = TREE_CHAIN (virtuals);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3104,58 +3098,14 @@ dfs_fixup_vtable_deltas (binfo, data)
|
|||||||
return NULL_TREE;
|
return NULL_TREE;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtuals = skip_rtti_stuff (binfo, BINFO_TYPE (binfo), &n);
|
for (virtuals = skip_rtti_stuff (binfo, BINFO_TYPE (binfo), &n);
|
||||||
|
virtuals;
|
||||||
while (virtuals)
|
virtuals = TREE_CHAIN (virtuals), ++n)
|
||||||
{
|
{
|
||||||
tree fndecl = TREE_VALUE (virtuals);
|
tree fndecl = BF_FN (virtuals);
|
||||||
tree delta = TREE_PURPOSE (virtuals);
|
|
||||||
|
|
||||||
if (fndecl)
|
if (fndecl)
|
||||||
{
|
modify_vtable_entry (t, binfo, fndecl, virtuals);
|
||||||
tree base_offset, offset;
|
|
||||||
tree context = DECL_CLASS_CONTEXT (fndecl);
|
|
||||||
tree vfield = TYPE_VFIELD (t);
|
|
||||||
tree this_offset;
|
|
||||||
|
|
||||||
offset = get_class_offset (context, t, binfo, fndecl);
|
|
||||||
|
|
||||||
/* Find the right offset for the this pointer based on the
|
|
||||||
base class we just found. We have to take into
|
|
||||||
consideration the virtual base class pointers that we
|
|
||||||
stick in before the virtual function table pointer.
|
|
||||||
|
|
||||||
Also, we want just the delta between the most base class
|
|
||||||
that we derived this vfield from and us. */
|
|
||||||
base_offset = size_binop (PLUS_EXPR,
|
|
||||||
get_derived_offset (binfo,
|
|
||||||
DECL_CONTEXT (fndecl)),
|
|
||||||
BINFO_OFFSET (binfo));
|
|
||||||
this_offset = ssize_binop (MINUS_EXPR, offset, base_offset);
|
|
||||||
|
|
||||||
if (! tree_int_cst_equal (this_offset, delta))
|
|
||||||
{
|
|
||||||
/* Make sure we can modify the derived association with immunity. */
|
|
||||||
if (binfo == TYPE_BINFO (t))
|
|
||||||
/* In this case, it is *type*'s vtable we are modifying.
|
|
||||||
We start with the approximation that it's vtable is that
|
|
||||||
of the immediate base class. */
|
|
||||||
build_vtable (TYPE_BINFO (DECL_CONTEXT (vfield)), t);
|
|
||||||
else
|
|
||||||
/* This is our very own copy of `basetype' to play
|
|
||||||
with. Later, we will fill in all the virtual
|
|
||||||
functions that override the virtual functions in
|
|
||||||
these base classes which are not defined by the
|
|
||||||
current type. */
|
|
||||||
prepare_fresh_vtable (binfo, t);
|
|
||||||
|
|
||||||
modify_vtable_entry (get_vtable_entry_n (BINFO_VIRTUALS (binfo), n),
|
|
||||||
this_offset,
|
|
||||||
fndecl);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
++n;
|
|
||||||
virtuals = TREE_CHAIN (virtuals);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return NULL_TREE;
|
return NULL_TREE;
|
||||||
@ -3226,8 +3176,8 @@ override_one_vtable (binfo, old, t)
|
|||||||
|
|
||||||
while (orig_virtuals)
|
while (orig_virtuals)
|
||||||
{
|
{
|
||||||
tree fndecl = TREE_VALUE (virtuals);
|
tree fndecl = BF_FN (virtuals);
|
||||||
tree old_fndecl = TREE_VALUE (old_virtuals);
|
tree old_fndecl = BF_FN (old_virtuals);
|
||||||
|
|
||||||
/* First check to see if they are the same. */
|
/* First check to see if they are the same. */
|
||||||
if (DECL_ASSEMBLER_NAME (fndecl) == DECL_ASSEMBLER_NAME (old_fndecl))
|
if (DECL_ASSEMBLER_NAME (fndecl) == DECL_ASSEMBLER_NAME (old_fndecl))
|
||||||
@ -3280,7 +3230,7 @@ override_one_vtable (binfo, old, t)
|
|||||||
}
|
}
|
||||||
{
|
{
|
||||||
/* This MUST be overridden, or the class is ill-formed. */
|
/* This MUST be overridden, or the class is ill-formed. */
|
||||||
tree fndecl = TREE_VALUE (virtuals);
|
tree fndecl = BF_FN (virtuals);
|
||||||
|
|
||||||
fndecl = copy_node (fndecl);
|
fndecl = copy_node (fndecl);
|
||||||
copy_lang_decl (fndecl);
|
copy_lang_decl (fndecl);
|
||||||
@ -3291,8 +3241,8 @@ override_one_vtable (binfo, old, t)
|
|||||||
|
|
||||||
/* We can use integer_zero_node, as we will core dump
|
/* We can use integer_zero_node, as we will core dump
|
||||||
if this is used anyway. */
|
if this is used anyway. */
|
||||||
TREE_PURPOSE (virtuals) = integer_zero_node;
|
BF_DELTA (virtuals) = integer_zero_node;
|
||||||
TREE_VALUE (virtuals) = fndecl;
|
BF_FN (virtuals) = fndecl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
virtuals = TREE_CHAIN (virtuals);
|
virtuals = TREE_CHAIN (virtuals);
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
/* This file contains the definitions and documentation for the
|
/* This file contains the definitions and documentation for the
|
||||||
additional tree codes used in the GNU C++ compiler (see tree.def
|
additional tree codes used in the GNU C++ compiler (see tree.def
|
||||||
for the standard codes).
|
for the standard codes).
|
||||||
Copyright (C) 1987,88,90,93,97-8,1999 Free Software Foundation, Inc.
|
Copyright (C) 1987,88,90,93,97-8,1999, 2000 Free Software Foundation, Inc.
|
||||||
Hacked by Michael Tiemann (tiemann@cygnus.com)
|
Hacked by Michael Tiemann (tiemann@cygnus.com)
|
||||||
|
|
||||||
This file is part of GNU CC.
|
This file is part of GNU CC.
|
||||||
@ -159,12 +159,22 @@ DEFTREECODE (TYPEOF_TYPE, "typeof_type", 't', 0)
|
|||||||
|
|
||||||
/* A thunk is a stub function.
|
/* A thunk is a stub function.
|
||||||
|
|
||||||
Thunks are used to implement multiple inheritance:
|
A THUNK_DECL is an alternate entry point for an ordinary
|
||||||
At run-time, such a thunk subtracts THUNK_DELTA (an int, not a tree)
|
FUNCTION_DECL. It's job is to adjust the `this' poitner before
|
||||||
from the this pointer, and then jumps to DECL_INITIAL
|
transferring control to the FUNCTION_DECL.
|
||||||
(which is an ADDR_EXPR whose operand is a FUNCTION_DECL).
|
|
||||||
|
|
||||||
Other kinds of thunks may be defined later. */
|
A thunk may perform either, or both, of the following operations:
|
||||||
|
|
||||||
|
o Adjust the `this' pointer by a constant offset.
|
||||||
|
o Adjust the `this' pointer by looking up a vcall-offset
|
||||||
|
in the vtable.
|
||||||
|
|
||||||
|
If both operations are performed, then the constant adjument to
|
||||||
|
`this' is performed first.
|
||||||
|
|
||||||
|
The constant adjustment is given by THUNK_DELTA. If the
|
||||||
|
vcall-offset is required, the index into the vtable is given by
|
||||||
|
THUNK_VCALL_OFFSET. */
|
||||||
DEFTREECODE (THUNK_DECL, "thunk_decl", 'd', 0)
|
DEFTREECODE (THUNK_DECL, "thunk_decl", 'd', 0)
|
||||||
|
|
||||||
/* A using declaration. DECL_INITIAL contains the specified scope.
|
/* A using declaration. DECL_INITIAL contains the specified scope.
|
||||||
|
@ -115,15 +115,18 @@ Boston, MA 02111-1307, USA. */
|
|||||||
For a static VAR_DECL, this is DECL_INIT_PRIORITY.
|
For a static VAR_DECL, this is DECL_INIT_PRIORITY.
|
||||||
|
|
||||||
BINFO_VIRTUALS
|
BINFO_VIRTUALS
|
||||||
For a binfo, this is a TREE_LIST. The TREE_PURPOSE of each node
|
For a binfo, this is a TREE_LIST. The BF_DELTA of each node
|
||||||
gives the amount by which to adjust the `this' pointer when
|
gives the amount by which to adjust the `this' pointer when
|
||||||
calling the function. If the method is an overriden version of a
|
calling the function. If the method is an overriden version of a
|
||||||
base class method, then it is assumed that, prior to adjustment,
|
base class method, then it is assumed that, prior to adjustment,
|
||||||
the this pointer points to an object of the base class.
|
the this pointer points to an object of the base class.
|
||||||
|
|
||||||
The TREE_VALUE is the declaration for the virtual function
|
The BF_VCALL_INDEX of each node, if non-NULL, gives the vtable
|
||||||
itself. When CLASSTYPE_COM_INTERFACE_P does not hold, the first
|
index of the vcall offset for this entry.
|
||||||
entry does not have a TREE_VALUE; it is just an offset.
|
|
||||||
|
The BF_FN is the declaration for the virtual function itself.
|
||||||
|
When CLASSTYPE_COM_INTERFACE_P does not hold, the first entry
|
||||||
|
does not have a BF_FN; it is just an offset.
|
||||||
|
|
||||||
DECL_ARGUMENTS
|
DECL_ARGUMENTS
|
||||||
For a VAR_DECL this is DECL_ANON_UNION_ELEMS.
|
For a VAR_DECL this is DECL_ANON_UNION_ELEMS.
|
||||||
@ -1720,6 +1723,18 @@ struct lang_type
|
|||||||
|
|
||||||
/* Get the value of the top-most type dominating the non-`normal' vfields. */
|
/* Get the value of the top-most type dominating the non-`normal' vfields. */
|
||||||
#define VF_DERIVED_VALUE(NODE) (VF_BINFO_VALUE (NODE) ? BINFO_TYPE (VF_BINFO_VALUE (NODE)) : NULL_TREE)
|
#define VF_DERIVED_VALUE(NODE) (VF_BINFO_VALUE (NODE) ? BINFO_TYPE (VF_BINFO_VALUE (NODE)) : NULL_TREE)
|
||||||
|
|
||||||
|
/* The number of bytes by which to adjust the `this' pointer when
|
||||||
|
calling this virtual function. */
|
||||||
|
#define BF_DELTA(NODE) (TREE_PURPOSE (NODE))
|
||||||
|
|
||||||
|
/* If non-NULL, the vtable index at which to find the vcall offset
|
||||||
|
when calling this virtual function. */
|
||||||
|
#define BF_VCALL_INDEX(NODE) (TREE_TYPE (NODE))
|
||||||
|
|
||||||
|
/* The function to call. */
|
||||||
|
#define BF_FN(NODE) (TREE_VALUE (NODE))
|
||||||
|
|
||||||
|
|
||||||
/* Nonzero for TREE_LIST node means that this list of things
|
/* Nonzero for TREE_LIST node means that this list of things
|
||||||
is a list of parameters, as opposed to a list of expressions. */
|
is a list of parameters, as opposed to a list of expressions. */
|
||||||
@ -2789,8 +2804,16 @@ extern int flag_new_for_scope;
|
|||||||
#define DECL_REALLY_EXTERN(NODE) \
|
#define DECL_REALLY_EXTERN(NODE) \
|
||||||
(DECL_EXTERNAL (NODE) && ! DECL_NOT_REALLY_EXTERN (NODE))
|
(DECL_EXTERNAL (NODE) && ! DECL_NOT_REALLY_EXTERN (NODE))
|
||||||
|
|
||||||
|
/* An integer indicating how many bytes should be subtracted from the
|
||||||
|
`this' pointer when this function is called. */
|
||||||
#define THUNK_DELTA(DECL) ((DECL)->decl.frame_size.i)
|
#define THUNK_DELTA(DECL) ((DECL)->decl.frame_size.i)
|
||||||
|
|
||||||
|
/* An integer indicating how many bytes should be subtracted from the
|
||||||
|
vtable for the `this' pointer to find the vcall offset. (The vptr
|
||||||
|
is always located at offset zero from the `this' pointer.) If
|
||||||
|
zero, then there is no vcall offset. */
|
||||||
|
#define THUNK_VCALL_OFFSET(DECL) (DECL_FIELD_SIZE (DECL))
|
||||||
|
|
||||||
/* DECL_NEEDED_P holds of a declaration when we need to emit its
|
/* DECL_NEEDED_P holds of a declaration when we need to emit its
|
||||||
definition. This is true when the back-end tells us that
|
definition. This is true when the back-end tells us that
|
||||||
the symbol has been referenced in the generated code. If, however,
|
the symbol has been referenced in the generated code. If, however,
|
||||||
@ -3918,7 +3941,7 @@ extern tree build_overload_with_type PARAMS ((tree, tree));
|
|||||||
extern tree build_destructor_name PARAMS ((tree));
|
extern tree build_destructor_name PARAMS ((tree));
|
||||||
extern tree build_opfncall PARAMS ((enum tree_code, int, tree, tree, tree));
|
extern tree build_opfncall PARAMS ((enum tree_code, int, tree, tree, tree));
|
||||||
extern tree hack_identifier PARAMS ((tree, tree));
|
extern tree hack_identifier PARAMS ((tree, tree));
|
||||||
extern tree make_thunk PARAMS ((tree, int));
|
extern tree make_thunk PROTO((tree, int, int));
|
||||||
extern void emit_thunk PARAMS ((tree));
|
extern void emit_thunk PARAMS ((tree));
|
||||||
extern void synthesize_method PARAMS ((tree));
|
extern void synthesize_method PARAMS ((tree));
|
||||||
extern tree get_id_2 PARAMS ((const char *, tree));
|
extern tree get_id_2 PARAMS ((const char *, tree));
|
||||||
|
@ -2008,9 +2008,10 @@ hack_identifier (value, name)
|
|||||||
|
|
||||||
|
|
||||||
tree
|
tree
|
||||||
make_thunk (function, delta)
|
make_thunk (function, delta, vcall_index)
|
||||||
tree function;
|
tree function;
|
||||||
int delta;
|
int delta;
|
||||||
|
int vcall_index;
|
||||||
{
|
{
|
||||||
tree thunk_id;
|
tree thunk_id;
|
||||||
tree thunk;
|
tree thunk;
|
||||||
@ -2033,6 +2034,11 @@ make_thunk (function, delta)
|
|||||||
icat (-delta);
|
icat (-delta);
|
||||||
OB_PUTC ('_');
|
OB_PUTC ('_');
|
||||||
OB_PUTID (DECL_ASSEMBLER_NAME (func_decl));
|
OB_PUTID (DECL_ASSEMBLER_NAME (func_decl));
|
||||||
|
if (vcall_index)
|
||||||
|
{
|
||||||
|
OB_PUTC ('_');
|
||||||
|
icat (vcall_index);
|
||||||
|
}
|
||||||
OB_FINISH ();
|
OB_FINISH ();
|
||||||
thunk_id = get_identifier (obstack_base (&scratch_obstack));
|
thunk_id = get_identifier (obstack_base (&scratch_obstack));
|
||||||
|
|
||||||
@ -2052,6 +2058,8 @@ make_thunk (function, delta)
|
|||||||
TREE_SET_CODE (thunk, THUNK_DECL);
|
TREE_SET_CODE (thunk, THUNK_DECL);
|
||||||
DECL_INITIAL (thunk) = function;
|
DECL_INITIAL (thunk) = function;
|
||||||
THUNK_DELTA (thunk) = delta;
|
THUNK_DELTA (thunk) = delta;
|
||||||
|
THUNK_VCALL_OFFSET (thunk)
|
||||||
|
= vcall_index * TREE_INT_CST_LOW (TYPE_SIZE (vtable_entry_type));
|
||||||
DECL_EXTERNAL (thunk) = 1;
|
DECL_EXTERNAL (thunk) = 1;
|
||||||
DECL_ARTIFICIAL (thunk) = 1;
|
DECL_ARTIFICIAL (thunk) = 1;
|
||||||
/* So that finish_file can write out any thunks that need to be: */
|
/* So that finish_file can write out any thunks that need to be: */
|
||||||
|
Loading…
Reference in New Issue
Block a user