[comp.lang.c++] Pointer conversion macro brainteaser

peierls@svax.cs.cornell.edu (Tim Peierls) (07/23/90)

The overparenthesized and implementation dependent nightmare below
"converts" a base pointer to a derived pointer, in defiance of the rules.
If you don't see any use for such a trick or if you are appalled at
the whole idea, then read no further.

The question is, for all you spec hackers, under what conditions is
this macro guaranteed by E&S to work?  By "conditions" I mean nasty
things like implementation defined behavior, and by "work" I mean conform
to the description below in some useful if not rigorous way.

#define bp2dp(B,D,p)  ((p)?((D*)(((char*)(p))+1-((size_t)(B*)(D*)(void*)1))):0)

Usage:
    B* bp;
    D* dp;
    dp = bp2dp(B,D,bp);

    Assuming that class D is (non-virtually) derived from class B, the
    macro takes a B* and returns a D*, such that (using the example names)
    "bp == (B*) dp", for all bp for which "(B*) dp" is defined (including 0).

It works for cfront 2.0 and probably everywhere else.  A reasonable
translator or compiler should have no problem turning the entire
expression into a compare and a subtract.  For example, cfront turns
((B*)(D*)(void*)1) into a conditional expression with a constant
condition that cc can turn into a constant expression.

A related question:  When does the offsetof(T,mem) macro work?
(It's defined in my stddef.h.)

Name    : Tim Peierls
Mail	: peierls@svax.cs.cornell.edu
Scotch  : Laphroaig

jimad@microsoft.UUCP (Jim ADCOCK) (07/25/90)

In article <43645@cornell.UUCP> peierls@cs.cornell.edu (Tim Peierls) writes:
|The question is, for all you spec hackers, under what conditions is
|this macro guaranteed by E&S to work?  By "conditions" I mean nasty
|things like implementation defined behavior, and by "work" I mean conform
|to the description below in some useful if not rigorous way.
|
|#define bp2dp(B,D,p)  ((p)?((D*)(((char*)(p))+1-((size_t)(B*)(D*)(void*)1))):0)

If a coding trick uses any constructs considered by E&S to be "implementation
dependent," then any "guarantees" from E&S are off.  I don't have an E&S in 
front of me at the moment, but here's a short list of a few of the
implementation dependencies I believe I see in this macro:

* cast of small integer to void*
* cast back of void* to something different than originally cast to void*
* cast of B* to size_t
* cast of B* to char*
* use of odd address for structure pointer
* cast of char* to D*

|It works for cfront 2.0 and probably everywhere else.  

I disagree with both the first part and the second part of this statement.

When I tried this macro on a cfront 2.0 derivative with B a virtual base
class of D, the code compiled silently, then crashed at runtime.  The 
reason it crashed was that in the case of a virtual base class, a D instance
contains a pointer to the virtual base class B instance.  Casting from D* to
B* requires an actual evaluation of the pointer -- which resides in the 
hypothetical structure residing at address "1"

Since the cfront implementation of virtual base classes seems reasonable
[assuming one feels virtual base classes are reasonable :-] it follows
that this macro will probably fail everywhere else too.

|A reasonable translator or compiler should have no problem turning the entire
|expression into a compare and a subtract.  

Again, not true if a class uses a virtual base.

peierls@svax.cs.cornell.edu (Tim Peierls) (07/31/90)

>In article <43645@cornell.UUCP> peierls@cs.cornell.edu (Tim Peierls) writes:
>|The question is, for all you spec hackers, under what conditions is
>|this macro guaranteed by E&S to work? ...
>|
>|#define bp2dp(B,D,p)  ((p)?((D*)(((char*)(p))+1-((size_t)(B*)(D*)(void*)1))):0)
>|It works for cfront 2.0 and probably everywhere else.  
>

In article <56042@microsoft.UUCP> jimad@microsoft.UUCP (Jim ADCOCK) writes:
>I disagree with both the first part and the second part of this statement.
>
>When I tried this macro on a cfront 2.0 derivative with B a virtual base
>class of D ...
>

I did say in my original posting that D had to be derived non-virtually
from B.

I was looking for a characterization of implementations where this macro
fails rather than an enumeration of implementation dependent behavior,
but perhaps what you are saying is that no useful characterization exists
beyond such an enumeration.  I'll buy that.
Name    : Tim Peierls
Mail	: peierls@svax.cs.cornell.edu
Scotch  : Laphroaig

jimad@microsoft.UUCP (Jim ADCOCK) (08/01/90)

In article <43906@cornell.UUCP> peierls@svax.cs.cornell.edu (Tim Peierls) writes:
>>In article <43645@cornell.UUCP> peierls@cs.cornell.edu (Tim Peierls) writes:
>>|The question is, for all you spec hackers, under what conditions is
>>|this macro guaranteed by E&S to work? ...
>>|
>>|#define bp2dp(B,D,p)  ((p)?((D*)(((char*)(p))+1-((size_t)(B*)(D*)(void*)1))):0)
>>|It works for cfront 2.0 and probably everywhere else.  
>>
>I was looking for a characterization of implementations where this macro
>fails rather than an enumeration of implementation dependent behavior,
>but perhaps what you are saying is that no useful characterization exists
>beyond such an enumeration.  I'll buy that.

Sorry, I forgot your restriction while I was trying to figure out what 
the macro was really trying to do.  I take it the intent is to try to 
bypass restrictions on private derivation?  What if an optimizing compiler
uses the restrictions placed on private derivation as part of its optimizing
scheme?  Should optimizing compilers have to be pessimistic about their
optimizations, assuming users will perform various pointer hacks?

I might simplify your original question [slightly :-] as follows:

When is pointer *arithmetic* *guaranteed* to work?

Answer [I claim] :

1) When you're adding an integer offset to a foo ptr that points into an
array of foos, or one off the far end, such that after the integer addition
the foo ptr still points into the foo array, or one off the far end.

2) When you're finding the difference between two foo ptrs, both of which point
into one array of foos, or one off the far end.

3) thats it, folks.