storage class != Storage Class (Was: Why are typedef names ...)

Gregory Smith greg at utcsri.UUCP
Mon Dec 8 03:25:32 AEST 1986


>In article <7374 at utzoo.UUCP>, henry at utzoo.UUCP writes:
>> Can you name a few?  Extern is the way it has always been in C; compilers
>> which default to static are broken.  (Please don't cite C++, that isn't C.)
>> -- 
In article <6430 at alice.uUCp> bs at alice.UUCP writes:
>
>Most (all?) C compilers and lint will accept the following program:
>
>file1:
>
>	typedef int I;
>	I i = 1;
>
>file2:
>	typedef char* I;
>	I p = "asdf";
>
>Is it legal? Personally, I think not because the name I which is external by
>default has two definitions (but the C++ compiler also accepts it).
>
But the storage class is given: 'typedef'. See pg 200, 8.8. In C, there
are really only three 'Storage Classes' per se: static, auto, and
register (the heap is supported entirely by run-time code).  Auto is on
the stack and static is at a constant address. All functions are static
in this sense.

However, storage class specifiers are used to specify more than just
Storage Class.  The concept of 'extern' has nothing to do with where an
object is stored, but rather refers to the scope of its name during linking.
'typedef' names such as 'I' above are not objects and do not exist
after compilation. So it makes no sense to apply a Storage Class of any
form to 'I'. You are suggesting that the 'link-time scope' rules should
be extended to typedefs (if only to make the example illegal), but
neither of these concepts apply to 'Storage Class'. They are two
different things which sc-specifiers have been made to do in addition
to specifying Storage Class.

It is worth noting at this point that the default storage class
(!=Storage Class) for top-level data definitions is neither static nor
exactly extern:

static	int	a;
	int	b;	/* all three mean different things */
extern	int	c;
--------------- cc -S ---------
	.lcomm	_a,4
	.comm	_b,4

The definition of 'b' produces a common-block definition while 'c' does
not. Under this implementation of C, you can write 'int b' in more than
one file, but under many others, you can write 'int a' in only one
file, and all others must say 'extern int b'.

So there are three 'storage classes' (really link-time scope classes)
for top-level objects:

(1) static: object is defined here in this file and is not externally visible.
(2) <none>: object is defined here in this file and *is* externally visible.
(3) extern: object is not defined here but is available externally.

...with a few exceptions for convenience:
char *getenv();
is treated as 'extern char *getenv()' since it is obviously not defined
there.  A definition of type (3) can exist in the same file with a (2)
and then the type (3) definitions have no effect, other than allowing
forward definition of type.
Again, all top-level objects have static Storage Class.

..and of course...
(4) typedef: not an object but a type, obviously not visible externally.
-- 
----------------------------------------------------------------------
Greg Smith     University of Toronto      UUCP: ..utzoo!utcsri!greg
Have vAX, will hack...



More information about the Comp.lang.c mailing list