[comp.lang.c++] const enums -- Lawyers opinions??

kc@rna.UUCP (Kaare Christian) (01/08/90)

Hey there language lawyer types...

I ran across the following while using Zortech C++ 2. The behavior
seems wrong to me, but I couldn't decide from the ugly reddish
manual what was the correct behavior.

In the following example, my intent is to declare a constant enum by or'ing 
together some enum constants. What I discovered with ZTC is that I can
only do this if I cast each of the enum values to const. Since enum values
are already constants (but not "const", it seems), this seems burdensome.
Is this a Zortechism, or is this standard C++ behavior?

enum bits { zero = 1, one = 2, two = 4 };

bits ZeroAndOne = zero | one;	// OK

const bits OneAndTwo = one | two;	// Fails with ztc2.01
	// complaint is type mistmatch, had <enum bits> and <const enum bits>

const bits OneAndTwoAgain = (const bits)one | (const bits)two;	// OK
	// but casts on the right look very redundant

comeau@utoday.UUCP (Greg Comeau) (01/09/90)

In article <963@rna.UUCP> kc@rna.UUCP (Kaare Christian) writes:
>Is this a Zortechism, or is this standard C++ behavior?
>enum bits { zero = 1, one = 2, two = 4 };
>bits ZeroAndOne = zero | one;	// OK
>const bits OneAndTwo = one | two;	// Fails with ztc2.01
>	// complaint is type mistmatch, had <enum bits> and <const enum bits>
>const bits OneAndTwoAgain = (const bits)one | (const bits)two;	// OK
>	// but casts on the right look very redundant

Actually what we have here for each case it a type mismatch since the only valid
'bits' are 'zero', 'one', and 'two'.  A common extension in C and C++ is to let
the assignment happen regardless (with a warning, not an error) because
there is an enum/int conversion often present even if the given int is not
a value enum "option" (value?).

As to "<enum bits> and <const enum bits>", I'd say that is even worse than
superfluous and is wrong since the | results in a respective promotion/
conversion producing an int which is similar to having said 'bits xyz = 5;'
whether const or not.  Actually, I'm a bit bewildered that the last case works
since the result cannot be a 'bits' even if the bit pattern was ok, no less
making each operand a const to make it work (although I think I see what it
was doing).
-- 
Greg, Comeau Computing, 91-34 120th Street, Richmond Hill, NY, 11418
Also, mag writer for UNIX Today! (SysAdm columnist), Microsoft Systems Journal
(C programming), + others. Also, BIX c.language & c.plus.plus conf. moderator.
Here:attmail!csanta!greg / BIX:comeau / CIS:72331, 3421 / voice:718-849-2355

jimad@microsoft.UUCP (JAMES ADCOCK) (01/10/90)

cfront 2.0 accepts this with the appropriate warning that you're assigning
an int to an enum -- when you or individual bits together you end up with
values not in the set original defined in the bits enum -- so how can the
receiver of the or results be a "bits" ?

The following slight change makes more sense to me, and compiles under cfront
2.0 without warning:

enum bit { .... };

typedef int bits;

...

bits someresult = one | two | whatever;

[not to imply bit or bits are good names to use for these things, nor
 zero, one, two, ...]

Another approach to setting individual bits is to use one or the other
Bit, bits, or Bitset classes floating around.  The typical trick is to
identify individual bits by their shift number, then automagically shift
1 << shiftnum inline using constructors, inline operators, or whatever,
to allow a small set of these values to be packed in an int.  Since most
compilers eval say 1 << 3 to be 8 at compile time, these classes are close
to the efficiency offered by the more historical enum approaches.

jss@jra.ardent.com (01/10/90)

In article <10197@microsoft.UUCP> jimad@microsoft.UUCP (JAMES ADCOCK) writes:
>
>cfront 2.0 accepts this with the appropriate warning that you're assigning
>an int to an enum -- when you or individual bits together you end up with
>values not in the set original defined in the bits enum -- so how can the
>receiver of the or results be a "bits" ?
>

The question of what an enum type "really is",  has bedeveled
the concept ever since it was introduced into C.  Over the
years a lot has been said, but I believe a consensus finally
emerged and both ANSI C and the most recent C++ reference manual agree 
that they are integral types.  

This means that its perfectly sensible to assign arbitrary integer
values to variables of enum type. 

Jerry Schwarz

williams@umaxc.weeg.uiowa.edu (Kent Williams) (01/10/90)

I just received what was recently referred to as the "Ugly Reddish
Book" from AT&T (i.e. AT&T C++ Product Reference) That sounds like a
nickname that should stick!

Incidentally, they sent me the wrong book (release notes instead of
selected readings), and I called them up.  The woman apologized
profusely and is sending the proper book federal express.

The funny thing was she didn't want me to mail the wrong book back
because it 'wasn't worth it.'  This is great!  If they send it to me,
it costs me 20$, but they don't want me to send it back because it's
not worth it.  I think I've discovered some sort of economic diode!


--
                               
Kent Williams                  "What's an Address Bus?  How do Icons work?" 
williams@umaxc.weeg.uiowa.edu  -- Advertisement for Time-Life Books 

kc@rna.UUCP (Kaare Christian) (01/10/90)

Regarding my const enums query, I originally ran into the problem while
programming an AM9513 counter timer chip. It has several registers, and
I wanted to make sure that I only stuffed the appropriate bit patterns
into each register. C++ 2.0's stronger enum typing (no automatic conversion
of int to enum) made it possible for me to create safe set of
subroutines for programming the chip. For example, I made a MasterMode
enum type containing all the bit fields of the am9513 master mode
register, and I made a WriteMasterMode function that only accepted a
MasterMode argument. That all worked great. The only hitch is that, to create
a value for, say, the master mode register, I had to OR together several
of the MasterMode enum constants. I was aware that ORing enums together
was likely to trigger an automatic conversion to int, and possibly a
warning when that was converted back to the original enum type. What I
found surprising was that I couldn't get this to work when const entered
the picture. My motivation for const was to continue to have the type
safety while avoiding the penalty of consuming a storage location for
each of my predefined bitpatterns. Isn't that one of the main reasons
for const global data types?

In article <10197@microsoft.UUCP>, jimad@microsoft.UUCP (JAMES ADCOCK) writes:
> 
> The following slight change makes more sense to me, and compiles under cfront
> 2.0 without warning:
> 
> enum bit { .... };
> 
> typedef int bits;
> 
> ...
> 
> bits someresult = one | two | whatever;
> 
I'm aware of the technique, but it doesn't let me use type checking
to avoid putting a countermode value into a mastermode register.

> [not to imply bit or bits are good names to use for these things, nor
>  zero, one, two, ...]
Yup, bad names in general, it was just the smallest example I could
devise.

I guess that a better solution than what I showed before is the
following:
   bits x = (bits)(one | two);
   const bits y = (const bits)(one | two);
This coerces the expr result back to what I want, and looks more
reasonable.

But in general, the 2.0 tightening of the enum type rules seems not tight
enough. Arithmetic operations on enums promote to int, thereby losing
the typing. Why not have arithmetic operations on enums *act* as int
operations, but retain typing? One could only do binop stuff on enums of
the same type (unless explicit casts were used), and enum op enum would
always produce an enum type.

In hindsight, I think I should have used a classier solution (pun
intended) or bitfields.

Kaare Christian

comeau@utoday.UUCP (Greg Comeau) (01/11/90)

In article <9925@ardent.UUCP> jss@jra.ardent.com () writes:
>In article <10197@microsoft.UUCP> jimad@microsoft.UUCP (JAMES ADCOCK) writes:
>>cfront 2.0 accepts this with the appropriate warning that you're assigning
>>an int to an enum -- when you or individual bits together you end up with
>>values not in the set original defined in the bits enum -- so how can the
>
>The question of what an enum type "really is",  ...a consensus finally
>emerged and both ANSI C and the most recent C++ reference manual agree 
>that they are integral types. This means that its perfectly sensible to
>assign arbitrary integer values to variables of enum type. 
>
>Jerry Schwarz

Although I agree that enum's are now integral types in both dpANSI C and
C++ one must take the semantics of enums into consideration before
saying that they (integrals) are completely convertable amongst each other.
First although enums contain integer constant values (int's actually), they
are not an integer types (a basic type accto ANSI) per se.  Also, although
ANSI C makes no mandation as I recall, in C++ enums are actually types and
therefore although the enumerators values are convertable to int values,
only enumerator values are supposed to be assigned to objects of the
respective enumeration (in other words these is no implicit integral
promotion or conversion when going into an enum lvalue).
-- 
Greg, Comeau Computing, 91-34 120th Street, Richmond Hill, NY, 11418
Also, mag writer for UNIX Today! (SysAdm columnist), Microsoft Systems Journal
(C programming), + others. Also, BIX c.language & c.plus.plus conf. moderator.
Here:attmail!csanta!greg / BIX:comeau / CIS:72331, 3421 / voice:718-849-2355

jimad@microsoft.UUCP (JAMES ADCOCK) (01/11/90)

>The question of what an enum type "really is",  has bedeveled
>the concept ever since it was introduced into C.  Over the
>years a lot has been said, but I believe a consensus finally
>emerged and both ANSI C and the most recent C++ reference manual agree 
>that they are integral types.  
>
>This means that its perfectly sensible to assign arbitrary integer
>values to variables of enum type. 
>
>Jerry Schwarz

I would claim on the contrary that the decision by both ANSI-C and C++
to consider enums integral types is a pragmatic decision stating that
strict type checking on enums is more of a pain than a help for 
programmers.  I would claim that if one creates an enum for the days
of the week and then store the integer 10000 in an instance of that
enum, then you've made a programming mistake -- even if it is permissable
according to the language definition.  There are a lot of areas in C++
like that -- where the language for historical or pragmatic reasons allows
programmers to do things that are generally very bad programming practice.

I believe it is important that programs "scan" right to human readers --
if a variable claims to be of a certain enum, it should only hold legal
values of that enum.  To do otherwise is a bug.  IMHO.

franka@mentor.com (Frank A. Adrian) (01/11/90)

In article <418@ns-mx.uiowa.edu> williams@umaxc.weeg.uiowa.edu.UUCP (Kent Williams) writes:
>I just received what was recently referred to as the "Ugly Reddish
>Book"...
>Incidentally, they sent me the wrong book...
>The funny thing was she didn't want me to mail the wrong book back
>because it 'wasn't worth it.'  I think I've discovered some sort
>of economic diode!

And now we know why ATT has to resort to suing MCI instead of providing service
to make money.

Just my own grumpy opinion...

Frank A. Adrian
Mentor Graphics, Inc.
franka@mntgfx.com

comeau@utoday.UUCP (Greg Comeau) (01/12/90)

In article <10215@microsoft.UUCP> jimad@microsoft.UUCP (JAMES ADCOCK) writes:
>I would claim that if one creates an enum for the days
>of the week and then store the integer 10000 in an instance of that
>enum, then you've made a programming mistake -- even if it is permissable
>according to the language definition.  There are a lot of areas in C++
>like that -- where the language for historical or pragmatic reasons allows
>programmers to do things that are generally very bad programming practice.

At least the 2.0 spec of C++ is in accordance with your train of though
and therefore such an assignment is prohibited.

ANSI C does not mention anything about this as far as I can detect
(I could be wrong though).
-- 
Greg, Comeau Computing, 91-34 120th Street, Richmond Hill, NY, 11418
Producers of CC C++, SysAdm columnist for UNIX Today!, Microsoft Systems Journal
(C programming), + others. Also, BIX c.language & c.plus.plus conf. moderator.
Here:attmail!csanta!greg / BIX:comeau / CIS:72331, 3421 / voice:718-849-2355

jimad@microsoft.UUCP (JAMES ADCOCK) (01/12/90)

>The question of what an enum type "really is",  has bedeveled
>the concept ever since it was introduced into C.  Over the
>years a lot has been said, but I believe a consensus finally
>emerged and both ANSI C and the most recent C++ reference manual agree 
>that they are integral types.  
>
>This means that its perfectly sensible to assign arbitrary integer
>values to variables of enum type. 

Hm, are you saying this has changed recently? In my [not very new] reference
manual appendix B pg 109 says:

enum e { A };

sizeof(A) equals sizeof(e), which need not equal sizeof(int)

[unlike ANSI-C]
For example, it would not be unreasonable for some compiler on some
machine to choose to represent small enums in a byte, in which case 
assigning an int to such an enum could lead to an unexpected truncation.
"Integral" is not the same as "integer."  

Or has this changed?