The following testing code does correctly in VS either with debug or release, and also in GCC. It also does correctly for ICC with debug, but not when optimization enabled (
Because the while (q.isEmpty()) ;
loop contains no statements that can ever cause an externally-visible side-effect, the entire loop is being optimised out of existence. It's the same reason that:
for (int i = 0; i < 10; i++)
;
could be optimised out of existence, as long as i
was not volatile
(stores to volatile
objects are part of the "externally visible" effects of a program).
In the C language, it is actually an outstanding bone of contention as to whether an infinite loop is allowed to be optimised away in this manner (I do not know what the situation is with C++). As far as I know, a consensus has never been reached on this issue - smart and knowledgeable people have taken both sides.
Any chance the actual code that you built and ran was missing the semicolon after while(q.isEmpty())
? That would certainly result in the next line being called infinitely.
The C++ standard allows loops without side-effects to be removed, even if they don't terminate:
It is generally felt that it is important to allow the transformation of potentially non-terminating loops (e.g. by merging two loops that iterate over the same potentially infinite set, or by eliminating a side-effect-free loop), even when that may not otherwise be justified in the case in which the first loop never terminates. http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2007/n2429.htm
See the discussion here: http://blog.regehr.org/archives/161
I think it may have been your version of gcc. I compiled your prog under 4.4.2 and it worked exactly as it should have.
It sure sounds like a bug. Here's a (pretty wild) guess about what reasoning might have lead to it...
After inlining, it sees:
while (q.m_first == q.m_last) /* do nothing */ ;
do_something();
and any sequence of do nothing repeatedly ; do something
can be translated to simply "do something". This falls down if the repeated part is endless (as in this case). But perhaps they don't test their compiling on examples that intentionally have endless looping ;-).
As a slight aside, this version of icc does what you want. That is, it never calls doSomething()
.
[9:41am][wlynch@computer /tmp] icc --version
icc (ICC) 11.0 20081105