[comp.lang.c++] Multiple Inheritance Problem

shi_fuchs@dkauni5t.bitnet (Harald Fuchs) (06/15/89)

I'm trying to write a simple tree class, but I can't get it working
using G++ 1.35.0. It goes like that:

class link {
  // Element of a doubly-linked list
  friend class list;
  friend class list_iterator;
  link* p;
  link* n;
protected:
  link () { p = n = 0; }
public:
  // ...
};

class list: link {
  // Circular list: dummy header with forward and backward pointers
public:
  list () { n = p = this; }
  // ...
};

class list_iterator {
  list* l;
  link* x;
public:
  list_iterator (list* a): l (a) { x = (link*) l; }
  link* operator() () { if (x) { x = x->n; if (x == l) x = 0; } return x; }
  void reset (list* a = 0) { if (a) l = a; x = (link*) l; }
};

class tree: list, link {
  // A tree node has a list of sons (if not leaf)
  // and is an element of such a list (if not root)
public:
  // ...
};

A tree node must have TWO prev/next pointers to work. I think this is
the case because I didn't declare the ``link'' base class virtual. So
what's wrong here: G++ or my brain?
--

Harald Fuchs    shi_fuchs@dkauni5t.bitnet

sorensen@athena.mit.edu (Alma G. Sorensen) (07/23/89)

p 199 of Stroustrup tells how to specify arguments for the
base class's constructor when deriving one class from another.

What is the syntax for specifying arguments to more than one
derived class (e.g. multiple inheritence in g++ 1.35.0)?

I tried just listing the different arguments ala BS, and get
errors like:

image.h:252: initializer for unnamed base class ambiguous
image.h:252: (type `int2D' uses multiple inheritance)
image.h:252: initializer for unnamed base class ambiguous
image.h:252: (type `int2D' uses multiple inheritance)


Thanks in advance.

Greg Sorensen
sorensen@imager.mit.edu  or
sorensen@athena.mit.edu

roman@hri.com (Roman Budzianowski) (10/12/90)

I am having a hard time with multiple inheritance (cfront 2.0 on sparc).

Basically, when in the constructor of a derived class, parts of the
object are not initialized and as a result I get 'no mapping' errors
when I call methods of this object.
I can get it to work, but any minor change brings the problem again.

Here is the class hierarchy:


		class Schema { 
			public:
			 virtual void f() {}
			 ...a lot of virtuals 
			}

class WithSimpleSlot : virtual public Schema | class WithMethod :
virtual public Schema {}
{					     |
	void f() {}			     |
}                                            |


...



class SystemSchema : public WithSimpleSlot, public WithMethod, public
WithNoRelation {}


class SchemaStream : public SystemSchema 
{
...

SchemaStream() { f(); } // here is the problem
}


An instance of SchemaStream is created as a static object.


It appears that the cfront actually creates three copies of Schema, and
the methods called in the constructor of SchemaStream refer to a wrong copy.

Can somebody help me ? Is this a cfront bug, or a problem with the design.
Actually I got it to work, but it broke again when I tryed to make the
destructor for Schema virtual.

Below is an ouput from dbx, the 'no mapping' problem is due to
myNode.ptr being null, which is not.

(dbx) p *this
*`extstream`StreamSchema::StreamSchema`this = {
        slotTable                = 0x1aff20
        [__vptr__14WithSimpleSlot = 0x1a278c]
        PSchema                  = 0x1a22e0
        OWithMethod              = {
                __vptr__10WithMethod = 0x1a25ac
                PSchema              = 0x1a22e0
                OSchema              = {
                        __W68__13SlotOwnerType = '\0'
                        schemaNameID  = (nil)
                        myNode        = {
                                ptr = (nil)
                        }
                        state         = 0
                        lock          = '\0'
                        __vptr__6Schema        = (nil)
                }
        }
        OWithNoRelation          = {
                __vptr__14WithNoRelation = 0x1a2500
                PSchema                  = 0x1a22e0
                OSchema                  = {
                        __W68__13SlotOwnerType = '\0'
                        schemaNameID  = (nil)
                        myNode        = {
                                ptr = (nil)
                        }
                        state         = 0
                        lock          = '\0'
                        __vptr__6Schema        = (nil)
                }
        }
        stream                   = 0x1a521c
        OSchema                  = {
                __W68__13SlotOwnerType = '\0'
                schemaNameID  = 0x1affb8
                myNode        = {
                        ptr = 0x1ad310
                }
                state         = 0
                lock          = '\0'
                __vptr__6Schema        = 0x1a25e0
        }
}


Thanks.

thc@cs.brown.edu (Thomas Colthurst) (05/04/91)

I've got the following object inheritance hierarchy:


                                Object
                                /    \
                               /      \
                             Line    Dependent
                               \       /
                                \     /
                              DependentLine

or in code,

class Object {
public:

  Object();
  ~Object();

   virtual void Draw() = 0;

   virtual void Update() = 0;

   virtual Object* Dup() = 0;

}

class Line: public virtual Object {
public:

   Line();
   ~Line();

   void Draw();

}

class Dependent: public virtual Object {
public:

  Dependent();
  ~Dependent();

  void Update();

}


class Dependent: public Line, public Dependent {
public:

  DependentLine();
  ~DependentLine();

  Object *Dup() { return( (new DependentLine) ); }; // error here

}



Sun CC version 2.1 gives me an error on the line
marked above, saying

"./ChildrenObjs.H", line 30: error: `new' of abstract class DependentLine

What am I doing wrong?

Thanks,
-Thomas C

robert@kohlrabi.tcs.com (Robert Blumen) (05/07/91)

In article <74462@brunix.UUCP>, thc@cs.brown.edu (Thomas Colthurst) writes:
|> class Object {
|> public:
|>   Object();
|>   ~Object();
|>    virtual void Draw() = 0;
|>    virtual void Update() = 0;
|>    virtual Object* Dup() = 0;
|> }
|> class Line: public virtual Object {
|> public:
|>    Line();
|>    ~Line();
|>    void Draw();
|> }
|> class Dependent: public virtual Object {
|> public:
|>   Dependent();
|>   ~Dependent();
|>   void Update();
|> }
|> class DependentLine : public Line, public Dependent {
|> public:
|>   DependentLine();
|>   ~DependentLine();
|>   Object *Dup() { return( (new DependentLine) ); }; // error here
|> }
|> Sun CC version 2.1 gives me an error on the line
|> marked above, saying
|> "./ChildrenObjs.H", line 30: error: `new' of abstract class DependentLine

For one thing, a class that inherits a pure virtual function must define it
or redefine it to be pure virtual.

-----------------------------------------------------------------------------
Robert Blumen                          | rblumen@tcs.com
Senior Software Engineer               | 2121 Allston Way, Berkeley, CA 94704
Teknekron Communications Systems, Inc. | (415) 649-3759

ilanc@microsoft.UUCP (Ilan CARON) (05/08/91)

C++ 2.1 now does for you the service of not requiring you to
redeclare your pure virtual functions in derived classes (i.e. it
automatically redeclares them for you as pure virtual).  (2.0 made
you redeclare them explicitly).

|class Object {
|public:
|
|  Object();
|  ~Object();
|
|   virtual void Draw() = 0;
|
|   virtual void Update() = 0;
|
|   virtual Object* Dup() = 0;
|
|}
|
|class Line: public virtual Object {
|public:
|
|   Line();
|   ~Line();
|
|   void Draw();

    // The 2.1 compiler effectively adds these decls.
    virtual void Update() = 0;
    virtual Object* Dup() = 0;

|
|}
|
|class Dependent: public virtual Object {
|public:
|
|  Dependent();
|  ~Dependent();
|
|  void Update();

    // The 2.1 compiler effectively adds these decls.
    virtual void Draw() = 0;
    virtual Object* Dup() = 0;
|
|}
|
|
|class DependentLine: public Line, public Dependent {
		^^^^
	You forgot this
|public:
|
|  DependentLine();
|  ~DependentLine();
|
|  Object *Dup() { return( (new DependentLine) ); }; // error here
|
|}
|
|Sun CC version 2.1 gives me an error on the line
|marked above, saying
|
|"./ChildrenObjs.H", line 30: error: `new' of abstract class DependentLine
|
|What am I doing wrong?


In order for a class to be concrete (non-abstract), so that
you can create instances of it with operator 'new', it must be
the case that it has no pure virtual functions.

In above hierarchy, Object, Dependent and Line are clearly abstract
classes.  The question is: why is DependentLine also abstract?

Well, DependentLine only explicitly overrides Dup().  What about
Draw() and Update()?  From one base class it inherits a pure virtual
version and from the other an implementation.  In any event, in the
derived class it is not clear which version to "use".

You might think that it should prefer the implementation as opposed to
the pure virtual definition -- but on reflection that doesn't make
much sense.  Consider this fragment:

    Line *pLine = new DependentLine;
    pLine->Update();			// ??? use Dependent::Update() ???


So, in conclusion, DependentLine is an abstract class because it
doesn't provide an explicit implementation for Draw() and Update().

Hope this makes sense.

--ilan caron (uunet!microsoft!ilanc)

purtill@morley.rutgers.edu (Mark Purtill) (05/10/91)

robert@kohlrabi.tcs.com (Robert Blumen) writes:
>For one thing, a class that inherits a pure virtual function must define it
>or redefine it to be pure virtual.
	Not any more (see: E&S, sec 10.3, page 215, first commentary
paragraph).  Unless this has been changed again.

^.-.^ Mark Purtill         purtill@dimacs.rutgers.edu         (908)932-4580 (O)
((")) DIMACS, P.O. Box 1179, Rutgers U., Piscataway, NJ 08855 (908)220-6905 (H)
********** Note new area code!  We are now (908) rather than (201)!  **********

ilanc@microsoft.UUCP (Ilan CARON) (05/14/91)

In article <2153@godzilla.tcs.com> robert@kohlrabi.tcs.com (Robert Blumen) writes:

>For one thing, a class that inherits a pure virtual function must define it
>or redefine it to be pure virtual.
>

Actually this is no longer the case -- as of the ARM (p. 215), a pure
virtual function is in fact inherited as a pure virtual.  I.e. you 
don't need to redeclare it explicitly (if you're not defining
it in the derived class).  However, I believe this is only implemented
in 2.1 conformant compilers.

[This is a good design decision since otherwise "demoting" a pure
virtual to an "impure" virtual in a base class would require mucking
with all derived classes].

--ilan caron (uunet!microsoft!ilanc)

mgates@entiat.boeing.com (Michael Gates) (05/16/91)

>In article <2153@godzilla.tcs.com> robert@kohlrabi.tcs.com (Robert Blumen) writes:
> In article <74462@brunix.UUCP>, thc@cs.brown.edu (Thomas Colthurst) writes:
> |> class Object {
> |> public:
> |>   Object();
> |>   ~Object();
> |>    virtual void Draw() = 0;
> |>    virtual void Update() = 0;
> |>    virtual Object* Dup() = 0;
> |> }
> |> class Line: public virtual Object {
> |> public:
> |>    Line();
> |>    ~Line();
> |>    void Draw();
> |> }
> |> class Dependent: public virtual Object {
> |> public:
> |>   Dependent();
> |>   ~Dependent();
> |>   void Update();
> |> }
> |> class DependentLine : public Line, public Dependent {
> |> public:
> |>   DependentLine();
> |>   ~DependentLine();
> |>   Object *Dup() { return( (new DependentLine) ); }; // error here
> |> }
> |> Sun CC version 2.1 gives me an error on the line
> |> marked above, saying
> |> "./ChildrenObjs.H", line 30: error: `new' of abstract class DependentLine
>
> For one thing, a class that inherits a pure virtual function must define it
> or redefine it to be pure virtual.

	G++ 1.39 compiles this with no trouble at all, which is
correct. For each pure virtual function, there is a path from
DependentLine to Object which defines that function. The ARM section
10.1.1 addresses this issue under the topic of name dominance. Page 205:
  
  	  A name B::f dominates a name A::f if its class B has A
  as a base. If a name dominates another no ambiguity exists
  between the two; the dominant name is used when there is a
  choice.

  (Followed by example)

Here Line::Draw dominates the Object::Draw that is also inherited
by DependentLine. Also Dependent::Update dominates Object::Update.
--
et tu mgates?

ilanc@microsoft.UUCP (Ilan CARON) (05/22/91)

mgates@entiat.boeing.com (Michael Gates) considers the following
"diamond" hierarchy:

			Object
			  Draw()=0;
			  Update()=0;
			  Dup()=0;

	Line : virtual Object		Dependent : virtual Object
	  Draw();			  Update();

			DependentLine : Line, Dependent
			  Dup() { new DependentLine; }
and says that:

>	G++ 1.39 compiles this with no trouble at all, which is
>correct. For each pure virtual function, there is a path from
>DependentLine to Object which defines that function. The ARM section
>10.1.1 addresses this issue under the topic of name dominance.
>
> [ARM def of dominance omitted]
>
>Here Line::Draw dominates the Object::Draw that is also inherited
>by DependentLine. Also Dependent::Update dominates Object::Update.

While the above is true, it is irrelevant.

Dominance is an ambiguity resolution rule (IMHO of only marginal value),
that is used in resolving client code references to names.  It says
nothing about how to *construct* an object -- in particular, it says
nothing wrt virtual function table contents.

As I pointed out in a previous posting (I believe):

> You might think that it should prefer the implementation as opposed to
> the pure virtual definition -- but on reflection that doesn't make    
> much sense.  Consider this fragment:                                  
>
>	Line *pLine = new DependentLine;                                      
>	pLine->Update();                    // ??? use Dependent::Update() ???
	^^^^^^^^^^^^^^^^
What does G++ do with this -- never-never-land? (I suppose Michael
might be thinking that this is a programmer bug).

>
> So, in conclusion, DependentLine is an abstract class because it   
> doesn't provide an explicit implementation for Draw() and Update().

Note that the issue is not how to resolve references to names in the
scope of DependentLine, rather whether DependentLine is an abstract
class or not (since we're trying to construct an instance of it).

--ilan caron (uunet!microsoft!ilanc)

mgates@entiat.boeing.com (Michael Gates) (05/25/91)

In article <72495@microsoft.UUCP> ilanc@microsoft.UUCP (Ilan CARON) writes:
>   mgates@entiat.boeing.com (Michael Gates) considers the following
>   "diamond" hierarchy:
>
>			   Object
>			     Draw()=0;
>			     Update()=0;
>			     Dup()=0;
>
>	   Line : virtual Object		Dependent : virtual Object
>	     Draw();			  Update();
>
>			   DependentLine : Line, Dependent
>			     Dup() { new DependentLine; }
>   and says that:
>
>   >	G++ 1.39 compiles this with no trouble at all, which is
>   >correct. For each pure virtual function, there is a path from
>   >DependentLine to Object which defines that function. The ARM section
>   >10.1.1 addresses this issue under the topic of name dominance.
>   >
>   > [ARM def of dominance omitted]
>   >
>   >Here Line::Draw dominates the Object::Draw that is also inherited
>   >by DependentLine. Also Dependent::Update dominates Object::Update.
>
>   While the above is true, it is irrelevant.
>
>   Dominance is an ambiguity resolution rule (IMHO of only marginal value),
>   that is used in resolving client code references to names.  It says
>   nothing about how to *construct* an object -- in particular, it says
>   nothing wrt virtual function table contents.

	It is relevant to the extent that it is the only part of the
ARM that *comes close* to addressing the problem. Since it is not
explicitly laid to rest anywhere in the book, I certainly have no
trouble changing my "which is correct" to "which is reasonable
(IMHO)". The reasonableness and probable correctness is supported by
section 10.10c, pages 233-235. I will explain after this quote.

>   As I pointed out in a previous posting (I believe):
>
>   > You might think that it should prefer the implementation as opposed to
>   > the pure virtual definition -- but on reflection that doesn't make    
>   > much sense.  Consider this fragment:                                  
>   >
>   >	Line *pLine = new DependentLine;                                      
>   >	pLine->Update();                    // ??? use Dependent::Update() ???
>	   ^^^^^^^^^^^^^^^^
>   What does G++ do with this -- never-never-land? (I suppose Michael
>   might be thinking that this is a programmer bug).

	No, G++ *does* use Dependent::Update(). This is what it is
supposed to do, per the second half of page 234:

   example (showing above behaviour) followed by

   "A call to a virtual function through one path in an inheritance
   structure may result in the invocation of a function redefined on
   another path. This is an elegant way for a base class to act as a
   means of communication between sibling classes, such as..."

Clearly the dominance rule (or something similar) is being used to
determine the contents of the final (DependentLine) vtable. In the
example in the ARM, the final class (DependentLine here, BMW in the
book) can get to the virtual base's "Update" (g() in the book)
directly, can get to the virtual base's "Update" through "Line" (BW in
the book), and can get to the redefined "Update" of "Dependent" (MW in
the book). The latter is chosen to place in the final vtable.

	The reason I say "dominance rule (or something similar)" above
is on page 235, 2nd paragraph:

   "To avoid ambiguous function definitions, all redefinitions of a
   virtual function from a virtual base must occur on a single path
   through the inheritence structure."

Before anyone jumps up with 

>   Note that the issue is not how to resolve references to names in the
>   scope of DependentLine, rather whether DependentLine is an abstract
>   class or not (since we're trying to construct an instance of it).

let me point out that the ARM section in question is discussing how to
build vtables. Unfortunately the case of pure virtual functions is not
dealt with. In absence of direction, why should they be treated
differently?
--
et tu mgates?

Michael.Gates@sunbrk.FidoNet.Org (Michael Gates) (05/25/91)

In article <72495@microsoft.UUCP> ilanc@microsoft.UUCP (Ilan CARON) writes:
>   mgates@entiat.boeing.com (Michael Gates) considers the following
>   "diamond" hierarchy:
>
>			   Object
>			     Draw()=0;
>			     Update()=0;
>			     Dup()=0;
>
>	   Line : virtual Object		Dependent : virtual Object
>	     Draw();			  Update();
>
>			   DependentLine : Line, Dependent
>			     Dup() { new DependentLine; }
>   and says that:
>
>   >	G++ 1.39 compiles this with no trouble at all, which is
>   >correct. For each pure virtual function, there is a path from
>   >DependentLine to Object which defines that function. The ARM section
>   >10.1.1 addresses this issue under the topic of name dominance.
>   >
>   > [ARM def of dominance omitted]
>   >
>   >Here Line::Draw dominates the Object::Draw that is also inherited
>   >by DependentLine. Also Dependent::Update dominates Object::Update.
>
>   While the above is true, it is irrelevant.
>
>   Dominance is an ambiguity resolution rule (IMHO of only marginal value),
>   that is used in resolving client code references to names.  It says
>   nothing about how to *construct* an object -- in particular, it says
>   nothing wrt virtual function table contents.

	It is relevant to the extent that it is the only part of the
ARM that *comes close* to addressing the problem. Since it is not
explicitly laid to rest anywhere in the book, I certainly have no
trouble changing my "which is correct" to "which is reasonable
(IMHO)". The reasonableness and probable correctness is supported by
section 10.10c, pages 233-235. I will explain after this quote.

>   As I pointed out in a previous posting (I believe):
>
>   > You might think that it should prefer the implementation as opposed to
>   > the pure virtual definition -- but on reflection that doesn't make    
>   > much sense.  Consider this fragment:                                  
>   >
>   >	Line *pLine = new DependentLine;                                      
>   >	pLine->Update();                    // ??? use Dependent::Update() ???
>	   ^^^^^^^^^^^^^^^^
>   What does G++ do with this -- never-never-land? (I suppose Michael
>   might be thinking that this is a programmer bug).

	No, G++ *does* use Dependent::Update(). This is what it is
supposed to do, per the second half of page 234:

   example (showing above behaviour) followed by

   "A call to a virtual function through one path in an inheritance
   structure may result in the invocation of a function redefined on
   another path. This is an elegant way for a base class to act as a
   means of communication between sibling classes, such as..."

Clearly the dominance rule (or something similar) is being used to
determine the contents of the final (DependentLine) vtable. In the
example in the ARM, the final class (DependentLine here, BMW in the
book) can get to the virtual base's "Update" (g() in the book)
directly, can get to the virtual base's "Update" through "Line" (BW in
the book), and can get to the redefined "Update" of "Dependent" (MW in
the book). The latter is chosen to place in the final vtable.

	The reason I say "dominance rule (or something similar)" above
is on page 235, 2nd paragraph:

   "To avoid ambiguous function definitions, all redefinitions of a
   virtual function from a virtual base must occur on a single path
   through the inheritence structure."

Before anyone jumps up with 

>   Note that the issue is not how to resolve references to names in the
>   scope of DependentLine, rather whether DependentLine is an abstract
>   class or not (since we're trying to construct an instance of it).

let me point out that the ARM section in question is discussing how to
build vtables. Unfortunately the case of pure virtual functions is not
dealt with. In absence of direction, why should they be treated
differently?
--
et tu mgates?

 * Origin: Seaeast - Fidonet<->Usenet Gateway - sunbrk (1:343/15.0)

kulkarni@cs.umn.edu ( Srinivas R. Kulkarni (06/27/91)

piero@bnrmtl.bnr.ca (Piero Colagrosso) writes:
>
> 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
>...
> The result: It seems that the calculation of the "this" pointer
> which is passed to m1 (i.e.e C::m1 is done incorrectly, resulting in
> defective code (an unexpected crash was the original symptom
>...( scrumpy@.1991Jun19.104037.13020 )

We have observed a similar syndrome. Unfortunately, the resultant code
isn't always kind enough to crash--in our case it quietly returns the
wrong answer! What's curious is that it seems to handle the simple cases
but fails on a slightly more complex one.

Here is the (simplified) situation in which we have observed the problem.
The classes are A, B, C and AA, BB, and CC. Methods m1, m2, and m3 with
implementations on classes as indicated by '...'.  All inheritance links
are 'virtual', as are all method declarations.

                  A...m1
                 /|
                / |
         m1...AA  B...m2
               | /|
               |/ |
         m2...BB  C...m3
               | /
               |/
         m3...CC

An instance *cc of CC is created and we invoke m1, m2, and m3 on it with
the expected results as indicated by the trace and returned state information.
The pointer cc is cast to a pointer *c of C, and the same methods are invoked.
The correct _functions_ are entered in each case, but the "this" pointer is
wrong in the case of m1, while it is correct in the cases of m2 and m3.
Help!

---------------------------------------------------------------------------
#include <stream.h>

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


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


class C : public virtual B {
  public:
   virtual int m3(void );
};



A::m1( void )
{
  cout << "\ninvoking virtual A::m1 for object " << this << ": ";
  return 0;
}

B::m2( void )
{
  cout << "\ninvoking virtual B::m2 for object " << this << ": ";
  return 0;
}

C::m3( void )
{
  cout << "\ninvoking virtual C::m3 for object " << this << ": ";
  return 0;
}



class AA : public virtual A {
  public:
   int Avar;

   virtual int m1( void );
};

class BB : public virtual AA, public virtual B {
   public:
   int Bvar;
   virtual int m2( void );
};

class CC : public virtual BB, public virtual C {
   public:
   int Cvar;
   virtual int m3( void );
};


AA::m1(void)
{
   cout << " Invoke AA.m1 on object " << this << ": ";
   return Avar;
}

BB::m2(void)
{
   cout << " Invoke BB.m2 on object " << this << ": ";
   return Bvar;
}

CC::m3(void)
{
   cout << " Invoke CC.m3 on object " << this << ": ";
   return Cvar;
}


main()
{
        C *c;
        CC* cc = new CC;

        cc->Avar = 1;
        cc->Bvar = 2;
        cc->Cvar = 3;


        cout << "\n cc->m1(): " << cc->m1();
        cout << "\n cc->m2(): " << cc->m2();
        cout << "\n cc->m3(): " << cc->m3();

        c = cc;            // cast it
        cout <<  "\n\nCast CC to C\n";

        cout << "\n  c->m1(): " << c->m1();
        cout << "\n  c->m2(): " << c->m2();

        cout << "\n  c->m3(): " << c->m3() << "\n";
}

---------------------------------------------------------------------------
Sample output:


 cc->m1():  Invoke AA.m1 on object 0x6614: 1         <<<-- desired result
 cc->m2():  Invoke BB.m2 on object 0x65f4: 2
 cc->m3():  Invoke CC.m3 on object 0x65d8: 3

Cast CC to C

  c->m1():  Invoke AA.m1 on object 0x65f4: 2         <<<-- should be as above
  c->m2():  Invoke BB.m2 on object 0x65f4: 2
  c->m3():  Invoke CC.m3 on object 0x65d8: 3
---------------------------------------------------------------------------

We are using a Sparc 1+ with the Sun (AT&T) Compiler 2.1 and SunOS 4.1.1.
The code seems (from my understanding of the language) to be legal;
certainly, the compiler does not issue even a peep of a warning message.
Nonetheless, the results are disasterously wrong.

Any explanations/workarounds/sympathy would be greatly appreciated.

Please e-mail responses to:

	 kulkarni@umn-cs.cs.umn.edu  or harp@ssdc.honeywell.com