[comp.lang.c++] Linkage of static members

pcg@cs.aber.ac.uk (Piercarlo Grandi) (06/29/90)

In article <31208@cup.portal.com> wmmiller@cup.portal.com (William
Michael Miller) writes:

   If allowed, "static" in the definition would mean that the member
   had internal linkage, which contradicts the statement in the ARM,
   section 9.4, that "static members of a global class have external
   linkage."

Ahh, but consts have internal linkage unless _explicitly_ specified
extern. Which rule prevails?

   The correct rule for static members (data and functions) is to put
   the "static" modifier in the class declaration and omit it in the
   definition.

This would be wonderful... But where is it said that a storage class
specifier is illegal in the definition of a static class member?

The following snippet of code compiles without errors whatsoever under
g++ 1.36.x:

    struct s { static v; static const c; static f(); };

    static 		s::v = 1;		// external
    static const 	s::c = 2;		// external
    static 		s::f() { return; }	// internal

    struct b { static v; static const c; };

    static		b::v;			// external
    static const	b::c;			// external BSS

    struct e { static v; static const c; static f(); };

    extern 		e::v = 1;		// external
    extern const 	e::c = 2;		// external
    extern 		e::f() { return; }	// external

    struct t { static v; static const c; static f(); };

    typedef		t::v;			// external, typedef ignored
    typedef const 	t::c;			// external BSS, typedef ignored
    typedef		t::f();			// oh my... completely ignored

And the generated code is (slightly compactified):

    #NO_APP
    gcc_compiled.:
    .globl _s$v; .data; .even;_s$v: .long 1
    .text; .even; f__1s: link a6,#0; jra L1; L1: unlk a6; rts
    .globl _e$v; .data; .even;_e$v:	.long 1
    .text; .even; .globl _f__1e; f__1e: link a6,#0; jra L2; L2: unlk a6; rts
    .comm _t$c,4
    .comm _t$v,4
    .globl _e$c; .even;_e$c: .long 2
    .comm _b$c,4
    .comm _b$v,4
    .globl _s$c; .even;_s$c: .long 2

There is clearly some problem here with G++ 1.36.x, but would anyone
care to explain to Michael Tiemann what should be done here? Quoting
chapter and verse of some reliable (time stamped, given the torrid pace
of C++ evolution) source?

Note that I have my own ideas about all this mess; just like all the
confusion about member pointers is caused by the useless notion of
member function, and that around MI is caused by the erroneous notion of
prefixing, all the trouble here is caused by the lack of a notion of
class objec, or the unresolved tension between class-as-module and
class-as-object-template.

In other words, to clarify C++ in a consistent way, you have to jettison
or restructure some of its most critical areas...
--
Piercarlo "Peter" Grandi           | ARPA: pcg%cs.aber.ac.uk@nsfnet-relay.ac.uk
Dept of CS, UCW Aberystwyth        | UUCP: ...!mcsun!ukc!aber-cs!pcg
Penglais, Aberystwyth SY23 3BZ, UK | INET: pcg@cs.aber.ac.uk

wmmiller@cup.portal.com (William Michael Miller) (07/01/90)

pcg@cs.aber.ac.uk (Piercarlo Grandi) writes:
> Ahh, but consts have internal linkage unless _explicitly_ specified
> extern. Which rule prevails?

There's no conflict.  The rule to which you refer states: "A name of file
scope that is explicitly declared const and not explicitly declared extern
is local to its file" (ARM 3.3).  Const class members do not have file
scope.

> This would be wonderful... But where is it said that a storage class
> specifier is illegal in the definition of a static class member?

It's not as explicit as one might like, and I expect that X3J16 may want to
add some wording to the description here.  It is implicit, however, in a
comparison of sections 3.3, 7.1.1, and 9.4 of the ARM.  3.3 and 9.4 say that
a name declared in a class as static has external linkage.  3.3 and 7.1.1
say that a name at file scope specified static has internal linkage.  7.1.1
says that all linkage specifications for a name must agree and gives
examples where it is illegal to specify a name as static once it has already
received, even implicitly, external linkage.

(Rereading these two comments back-to-back make it evident that the
terminology in the ARM needs to be tightened up: const class members don't
have file scope because they're declared inside a class; however, static
class members *must* be defined at file scope.  Perhaps concepts like
"introducing declaration" and "defining declaration" should be used.)

> Note that I have my own ideas about all this mess; just like all the
> confusion about member pointers is caused by the useless notion of
> member function, and that around MI is caused by the erroneous notion of
> prefixing, all the trouble here is caused by the lack of a notion of
> class objec, or the unresolved tension between class-as-module and
> class-as-object-template.

You seem to find a lot of problems that don't particularly seem like
problems to me.  C++ is clearly not the most elegant, concise, and formally
pure language conceivable, but I don't think it's as fundamentally flawed as
you do.

------------------------------------------------------------------------------
William M. Miller, Glockenspiel, Inc.; P. O. Box 366, Sudbury, MA 01776-0003
wmmiller@cup.portal.com         BIX: wmiller            CI$: 72105,1744

pcg@cs.aber.ac.uk (Piercarlo Grandi) (07/03/90)

In article <31285@cup.portal.com> wmmiller@cup.portal.com (William
Michael Miller) writes:

   pcg@cs.aber.ac.uk (Piercarlo Grandi) writes:
   > Ahh, but consts have internal linkage unless _explicitly_ specified
   > extern. Which rule prevails?

   There's no conflict.  The rule to which you refer states: "A name of file
   scope that is explicitly declared const and not explicitly declared extern
   is local to its file" (ARM 3.3).  Const class members do not have file
   scope.

But static class members are *defined* at file scope, as you say:

   (Rereading these two comments back-to-back make it evident that the
   terminology in the ARM needs to be tightened up: const class members don't
   have file scope because they're declared inside a class; however, static
   class members *must* be defined at file scope.  Perhaps concepts like
   "introducing declaration" and "defining declaration" should be used.)

I am starting to think that the 3.0 ARM will be in several volumes, and
will be sold by door-to-door salesmen, and that soon Harvard will open a
Language Law school, and all that :-).

   You seem to find a lot of problems that don't particularly seem like
   problems to me.  C++ is clearly not the most elegant, concise, and formally
   pure language conceivable, but I don't think it's as fundamentally flawed as
   you do.

I do not think it is fundamentally flawed; I think it does mostly the
right things for all the wrong reasons, and in a terribly haphazard way.
I seem to understand that C++ is being developed with a "let's try this
feature" approach, which may very well be pragmatic, but has some
undesirable consequences. Eiffel (which I do not like) at least has a
_seemingly consistent_ language and software design philossophy behind
(and then some incredible slips like references!).

Maybe it is the problems of success; for an example of a detail it is
arguable that 'this' (which should not exist, however) should be a
reference and not a pointer, but it's too late to change that.

I was recently rereading Stroustrup's paper on MI in the Usenix journal.
He has an appendix rejecting one flavour of delegation, an extremely
elegant and C-ish technique, because users were confused by it. Instead
of thinking that maybe the proposed flavour of delegation was wrong, he
throws away delegation entirely, and bang! we get several dozen pages
and untold complications about the sordid consequences of Simula67 style
prefixing extended to multiple prefixing. I wonder how confused are
users now...
--
Piercarlo "Peter" Grandi           | ARPA: pcg%cs.aber.ac.uk@nsfnet-relay.ac.uk
Dept of CS, UCW Aberystwyth        | UUCP: ...!mcsun!ukc!aber-cs!pcg
Penglais, Aberystwyth SY23 3BZ, UK | INET: pcg@cs.aber.ac.uk

bs@alice.UUCP (Bjarne Stroustrup) (07/03/90)

Piercarlo "Peter" Grandi writes:

 > I seem to understand that C++ is being developed with a "let's try this
 > feature" approach

C++ wasn't developed that way. I just happen to hate pompous preachers that
screams their bogus ``principles'' from the rooftops as the one and only truth.
I guess I can be blamed for not formulating the ideas underlying the C++ features
in a few catchy phrases, but C++ is not and never was just a random bag of features.

I set out to create a language that would allow me to apply the notions primarily
borrrowed from Simula of a program as a model of some aspect of reality and
a class as the concrete representation of a concept of a concept in the program.
This is supported by notions of data hiding and encapsulation primarily
borrowed from the operating systems world.

To make that goal feasible/affordable I adopted the principle that what you
don't use you don't pay for and the notion that a program must be able to
cooperate effectively with ``the rest of a system.'' That is, zero overhead
use must be possible and that a C++ program is part of an open and extensible
system. These latter principles (constraints, rather) is what makes C++ usable
outside the academic ghetto and outside the personal workstation sub-culture.

 > I was recently rereading Stroustrup's paper on MI in the Usenix journal.
 > He has an appendix rejecting one flavour of delegation, an extremely
 > elegant and C-ish technique, because users were confused by it. Instead
 > of thinking that maybe the proposed flavour of delegation was wrong, he
 > throws away delegation entirely,

So I should dogmatically have kept it in despite it causing grief? Or maybe
I should have dropped everything else to figure out exactly why it was wrong
and fixed it? The latter was certainly tempting - because I liked the notion
of delegation and still do - and I tried unsuccessfully for a while. However,
I suspect you underestimate the time and effort needed to get everything done.
I simply never found the time to explore delegation further.

Also, please read that note again. I most certainly suspected that it was
the flavor of delegation tried  - rather than the general concept of delegation
- that was wrong and I said so. That design of delegation for C++ was an
experiment that failed. I documented that. Would you have preferred me NOT
to document the failure? That would not only have allowed people to keep
their preconceived and untried notions (kept them happy) but also allowed
me to wrap myself in a mantle of infallibility (there are no shortage of
people that likes to do that - it keeps them admired and happy). To my
mind that would have been fradulent, however, and would undoubtedly have
led someone to repeat my mistakes.