pmoore@hemel.bull.co.uk (Paul Moore) (11/13/90)
Well a lot of conflicting opinions expressed - thanks. They range from "desctructor calling specs are deliberately vague" to "it must be a bug" via "you shouldnt do that anyway". Tnaks to Reid Ellis for the test prog that showed that it is probably a bug in the zortech compiler. His test prog was (paraphrased) func() { cin >> i; if (i==0) return 0; foo bar; return 0; } with foo class foo{ public: ~foo{cout << "foo destruct"}; }; This works fine - i tryed on my compiler. So I went back to my code and found that I had oversimplified to example - changing the above to func() { ..... if (i==0) goto err_exit; foo bar; err_exit:; return 0; } now fails - ie the constructor is not called but the destructor is. I'm sure this proves that it is a bug in the zortech compiler since the behaviuor must be consistent. I'm still interested in what other compilers do as the lack of concensus shows that this is not clearly defined.
dcurtis@crc.ac.uk (Dr. David Curtis) (11/14/90)
<>func() <>{ <> ..... <> if (i==0) <> goto err_exit; <> <> foo bar; <>err_exit:; <> return 0; <>} <>now fails - ie the constructor is not called but the destructor is. I'm sure <>this proves that it is a bug in the zortech compiler since the behaviuor <>must be consistent. <> <>I'm still interested in what other compilers do as the lack of concensus <>shows that this is not clearly defined. I don't think it's fair to call this a bug. The Zortech manual has just such an example (V2.0, p.203): "Re: Using goto to skip initialisation In C this may or may not cause problems, and is certainly bad style. In C++ it can be positively disastrous since the class definition might in fact call a constructor which allocates memory, and for which a destructor will be called later." Quite so. If anything, this is a language problem, not a compiler problem. Using goto to skip initialisation? Just say no. Newsgroups: comp.lang.c++ Subject: Re: conditional scope and destructors results Summary: Expires: References: <1990Nov13.114958.20853@hemel.bull.co.uk> Sender: Reply-To: dcurtis@crc.ac.uk (Dr. David Curtis) Followup-To: Distribution: Organization: MRC Human Genome Resource Centre Keywords:
kaiser@ananke.stgt.sub.org (Andreas Kaiser) (11/17/90)
In your msg to All, dated <16 Nov 90 19:01>, it said: PM> Tnaks to Reid Ellis for the test prog that showed that it is probably PM> a bug in the zortech compiler. His test prog was (paraphrased) PM> This works fine - i tryed on my compiler. So I went back to my code PM> and found that I had oversimplified to example - changing the above to PM> func() PM> { PM> ..... PM> if (i==0) PM> goto err_exit; PM> foo bar; PM> err_exit:; PM> return 0; PM> } PM> now fails - ie the constructor is not called but the destructor is. PM> I'm sure this proves that it is a bug in the zortech compiler since PM> the behaviuor must be consistent. ARM, page 91: "It is illegal to jump past a declaration with an explicit or implicit initializer unless the declaration is in an inner block that is not entered (...) or unless the jump is from a point where the variable has already been initialized." Andreas -- :::::::::::::::::::: uucp: kaiser@ananke.stgt.sub.org :: Andreas Kaiser :: fido: 2:509/5.2512 & 2:507/18.7206 ::::::::::::::::::::
dcurtis@crc.ac.uk (Dr. David Curtis) (11/21/90)
Jamshid Afshar has pointed out that I was wrong to say that it was not the compiler's problem if you try to skip past an initialiser with a goto. In fact it is illegal to do this, so the compiler ought to flag it as an error rather than just tell you not to do it in the manual.
jimad@microsoft.UUCP (Jim ADCOCK) (11/28/90)
In article <366@tin.crc.ac.uk> dcurtis@crc.ac.uk (Dr. David Curtis) writes: |Jamshid Afshar has pointed out that I was wrong to say that it was not the |compiler's problem if you try to skip past an initialiser with a goto. In fact |it is illegal to do this, so the compiler ought to flag it as an error |rather than just tell you not to do it in the manual. I disagree. First, Ellis & Stroustrup do not make it clear whether this restriction is intended to be imposed on the programmer alone, or whether this is something that it is mandatory for the compiler to diagnose. But, they do say it is illegal to jump forward past an initializer -- not that it is illegal to write code that might permit a jump past an initializer. It would seem that a compiler would at least require a flow analyser to determine whether a jump might occur past an initializer in simple deterministic cases. Such a level of sophistication is not typically assumed of compilers in the design of languages. However, in other than simple deterministic cases it would seem that even compilers with flow analysers would not be able to determine if the jump will actually be executed or not. I would suspect that smart compilers would cause a compiler time error if it can be assuredly determined that a constructor is being so by-passed, send a warning in situations where the constructor might be so by-passed, and neither warn nor send an error in more complicated situations. Dumb compilers might do none of the above. In any case, since this is not a deterministic problem for compilers, it seems to me the intent of the prohibition was on programmers against such writing programs, not a requirement for compilers to detect such situations. Hopefully this is another situation where the ANSI committee will clarify the rights and responsibilities of compilers verses programmers.
philip@pescadero.Stanford.EDU (Philip Machanick) (11/29/90)
In article <13@microsoft.UUCP>, jimad@microsoft.UUCP (Jim ADCOCK) writes: |> In article <366@tin.crc.ac.uk> dcurtis@crc.ac.uk (Dr. David Curtis) writes: |> |Jamshid Afshar has pointed out that I was wrong to say that it was not the |> |compiler's problem if you try to skip past an initialiser with a goto. In fact |> |it is illegal to do this, so the compiler ought to flag it as an error |> |rather than just tell you not to do it in the manual. > > I disagree. First, Ellis & Stroustrup do not make it clear whether this > restriction is intended to be imposed on the programmer alone, or > whether this is something that it is mandatory for the compiler to > diagnose. But, they do say it is illegal to jump forward past an > initializer -- not that it is illegal to write code that might permit > a jump past an initializer. Maybe there is a need for clearer wording. I can't find a specific definition in E&S of "illegal", and the term is used relatively rarely - on a quick glance, I found one other use (p.96), where it is clearly intended to be synonymous with "error" (which I interpret as "compile-time error"). > I would suspect that smart compilers would cause a compiler time error > if it can be assuredly determined that a constructor is being so > by-passed [...] Maybe this is impossible to do in general, but it should be possible to use a conservative strategy. For example, cfront 2.0 throws out the following int i=1; if (i!=1) goto L123; float x=0.0; L123: even though the goto will never be executed. There's a case for tightening up the wording, but I believe the responsibility for detecting potential errors of this kind should rest with the compiler, even if it occasionally results in minor inconvenience (through the adoption of a conservative strategy). -- Philip Machanick philip@pescadero.stanford.edu
pmoore@hemel.bull.co.uk (Paul Moore) (11/30/90)
jimad@microsoft.UUCP (Jim ADCOCK) writes: >I disagree. First, Ellis & Stroustrup do not make it clear whether this >restriction is intended to be imposed on the programmer alone, or >whether this is something that it is mandatory for the compiler to >of languages. However, in other than simple deterministic cases ....... >Hopefully this is another situation where the ANSI committee will >clarify the rights and responsibilities of compilers verses programmers. My personal thoughts are that the compiler must at least issue a warning. Non trivial cases of this problem are not easy to spot and occur in what seemed to me to be reasonbly shaped code - it took me some time to discover my problem. I think that the days where there was a book called "all the things that can go wrong with c but the compiler wont tell you" that you had to read have gone.
mat@mole-end.UUCP (Mark A Terribile) (12/01/90)
> > [Ellis and Stroustrup] say it is illegal to jump forward past an > > initializer -- not that it is illegal to write code that might permit > > a jump past an initializer. . . . > > I would suspect that smart compilers would cause a compiler time error > > if it can be assuredly determined that a constructor is being by-passed ... > Maybe this is impossible to do in general, but it should be possible to > use a conservative strategy. For example, cfront 2.0 throws out the following > int i=1; > if (i!=1) goto L123; > float x=0.0; > L123: > even though the goto will never be executed. There's a case for tightening > up the wording, but I believe the responsibility for detecting potential > errors of this kind should rest with the compiler, even if it occasionally > results in minor inconvenience (through the adoption of a conservative > strategy). There's a real problem with this real nice idea, to wit, what defines the algorithm that determines whether or not the compiler can `assuredly determine' that the declaration will never be elaborated? What strict, unambiguous, standard-suitable definition is to be found for the criteria against which a piece of code is to be tested? Will such a definition be USEFUL to the ordinary Sue, Ann, and Cheryl who are programming in C++? Would it be useful to the maintenance programmer who inadvertantly broke a flow-control assumption on which depended the correctness of some remote piece of code? Some of the answers might be acceptible if there were a well enough developed body of computing theory. In fact, little or none of our present computing theories can give us a good language with which to talk about this problem. -- (This man's opinions are his own.) From mole-end Mark Terribile
jimad@microsoft.UUCP (Jim ADCOCK) (12/11/90)
In article <1990Nov28.191327.8778@Neon.Stanford.EDU> philip@pescadero.stanford.edu writes: >In article <13@microsoft.UUCP>, jimad@microsoft.UUCP (Jim ADCOCK) writes: >|> In article <366@tin.crc.ac.uk> dcurtis@crc.ac.uk (Dr. David Curtis) writes: >Maybe this is impossible to do in general, but it should be possible to >use a conservative strategy. For example, cfront 2.0 throws out the following > >int i=1; >if (i!=1) goto L123; >float x=0.0; >L123: > >even though the goto will never be executed. There's a case for tightening >up the wording, but I believe the responsibility for detecting potential >errors of this kind should rest with the compiler, even if it occasionally >results in minor inconvenience (through the adoption of a conservative >strategy). I don't disagree. My argument is with what the ARM presently says or doesn't say. The ANSI-C spec clearly using the word "jump" to mean that which happens during the dynamic execution of a program, whereas a "jump statement" is what exists at compile time. The informality of a reference manual [like the ARM] allows such ambiguities to exist. Hopefully, when we have a formal ANSI-C++ spec, these kinds of ambiguities will not exist -- such being the difference between a reference manual and a complete spec. As you point out, either one prohibits a "jump statement" referencing a target label past a constructor, in which case compilers can conservatively prohibit all such constructs, or one prohibits a "jump" past a constructor, in which case compilers can only catch those "jumps" which can be predicted at compile time. I can't think of any good reason why it shouldn't be the "jump statement" that is prohibited.