[comp.lang.c++] Problems with multiple inheritance

piero@bnrmtl.bnr.ca (Piero Colagrosso) (06/19/91)

I have been experiencing a problem with the use of multiple 
inheritance in C++. I am using the Sun (AT&T) compiler on a Sun 
Sparc II running SunOS 4.1.1. The problem manifests itself with
both the 2.0 and 2.1 compilers.


Consider the following class hierarchy: 

                  A 
                 / \
                B   C
                 \ /   
                  D1
                  | 
                  D2

The problem seems to occur when you derive from a class (i.e. D1 in 
the example above) which is multiply derived from two other classes 
(i.e. B and C) and the following conditions hold:

       1) B and C are derived virtually from their parent class (A).
       2) A and C implement a virtual function m1.
       3) A, D1, and D2 implement a virtual function m2.

The problem is the following:
   1) You create a D2 object and assign it to an A* pointer.
   2) You subsequently invoke the m1 member function on this object 
      via the A* pointer.

The result: It seems that the calculation of the "this" pointer 
which is passed to m1 (i.e. C::m1) is done incorrectly, resulting in 
defective code (an unexpected crash was the orginal symptom
which led to the identification of this problem). 

The following example illustrates inconsistent values of C's "this"
pointer in the constructor and subsequent invocation of m1 (see
program output):


--------------------------------------------------------------------

class A
{
public:
	A(); 
	virtual void m1();
	virtual void m2();
private:
	int a;
};


class B : public virtual A
{
public:
	B();
	virtual void m2();
private:
	int b;
};


class C : public virtual A
{
public:
	C();
	virtual void m1();
private:
	int c;
};


class D1: public B, public C
{
public:
	D1();
	virtual void m2();
private:
	int d1;
};

class D2: public D1
{
public:
	D2();
	virtual void m2();
private:
	int d2;	
};

#include <iostream.h>


main()
{
	A* pA = new D2;
	pA->m1();  
}

A::A() 
{ 
	cerr << "A::A; this = " << this << endl;
}

void A::m1()
{
	cerr << "A::m1; this = " << this << endl;
}

void A::m2()
{
	cerr << "A::m2; this = " << this << endl;
}

B::B()
{
	cerr << "B::B; this = " << this << endl;
}

void B::m2()
{
	cerr << "B::m2; this = " << this << endl;
}

C::C()
{
	cerr << "C::C; this = " << this << endl;
}

void C::m1()
{
	cerr << "C::m1; this = " << this << endl;
}

D1::D1()
{
	cerr << "D1::D1; this = " << this << endl;
}

void D1::m2()
{
	cerr << "D1::m2; this = " << this << endl;
}

D2::D2()
{
	cerr << "D2::D2; this = " << this << endl;
}

void D2::m2()
{
	cerr << "D2::m2; this = " << this << endl;
}

--------------------------------------------------------------------

The output generated from this program:

A::A; this = 0x6490
B::B; this = 0x6468
C::C; this = 0x6470      <-------
D1::D1; this = 0x6468           |   These aren't the same !!
D2::D2; this = 0x6468           |  
C::m1; this = 0x6468     <-------


Am I doing anything wrong or is this really a compiler problem ?

I have noticed that the problem does not occur if:

 i) Either D1 or D2 (or both) does not implement the virtual function 
    m1. 
ii) There is no extra level of inheritance after the multiple 
    inheritance, i.e. if the above class hierarchy stops at D1, 
    so that we instantiate a D1 object instead of a D2 object. 

Based on my understanding of the implementation of MI, as 
described in the ARM, I think that the "this" pointer offset is 
not being set properly in D2's virtual table (vtbl). 

If this really is a compiler problem, is it a known one and are 
there any workarounds ?


#######################################################
# Piero Colagrosso              Dept. 6s52            #
# Bell-Northern Research, Montreal, Canada.           #
#                                                     #
# INTERNET:  bnrmtl!piero@Larry.McRCIM.McGill.EDU     #
#######################################################