G
I agree that the standard is ambiguous about this (which is where weBarry said:Barry Schwarz wrote:
<snip and summarize thus>
((*f(0))++, (*f(1))++) + ((*f(2))++, (*f(3))++)
(we're assuming here that f(0) is done before f(1) is done before f(2)
is done before f(3) - other orderings are allowed but are not relevant
to this discussion. f(3) returns a pointer to the same address as f(1))
(I'm using some lazy shorthand here - f(0)++ actually means (*f(0))++I don't see how. The only way d[1] can be updated is if f() returns
its address. The expression f(1) will always return &d[1]. The
expression f(3) will return &d1 if twice is not 0. No other call to
f() will return &d[1].
etc)
because f(0) is done before f(1) and there is a comma sequence point
after f(0) and before f(1) the side effects of the f(0)++ must be
completed before f(1).
While the comma sequence point does in fact guarantee this, the
sequence point before the call to f(1) also guarantees it. It doesn't
matter why there is a sequence point, only that there is one.
because f(2) is done before f(3) and there is a comma sequence point
after f(2)++ and before f(3) the side effects of the f(2)++ must be
completed before f(3).
As I said in the portion you chose to snip, neither f(0) nor f(2)
matter since they cannot have any effect of d[1].
But f(1) can be called before f(2) but the increment deferred until the
end of the full expression. What this does mean is that part of f(1)++
No it cannot. There is a sequence point prior to the call to f(2) and
any side effect from f(1) must be completed before this sequence
point.
came in on this discussion anyway) but I believe that the _intent_ of
the standard is that
(*f())++ + (*f())++ has undefined behaviour because a _valid_
optimization for a compiler that can see that f() always returns the
same pointer is:
g=f(); (*g)++ + (*g)++;
i.e. the sequence points before and after the calls to f() only
constrain the side effects that occur in f() itself.
So
S, f(), S, f(), S, inc, inc, S
is allowed.
This is subtly different from the subject line. In that case I think
the standard requires the assignment of 5 to x before the 11 is
evaluated (and so assigned to x) but I don't think the standard
_requires_ that the assignment of 11 to x starts _after_ the assignment
of 5 to x.
Yes. But there is confusion when it comes to sub-expressions thatEvaluations never cross sequence points. The whole purpose of
sequence points is to insure that the evaluation is complete.
include sequence points. How much to they constrain things?
This is one valid interpretation. A second is that the sequence pointsThe standard guarantees that all side effects are complete prior to
the sequence point. Since there is a sequence point prior to the call
to f(2), the increment must be complete.
on the RHS have no effect on the LHS whatsoever and vice versa. A third
interpretation is that the sequence points on the RHS require that no
side effects are _still_ in progress on the LHS but do not otherwise
constrain when the side effects on the LHS can occur.
I would expect that it's actually the third interpretation that the
standard authors intended but it doesn't actually sound as reasonable
as either the first or second option when phrased as above.
Since evaluating g in its many forms does not involve a sequence
point, this is nothing but a red herring.
huh? there's a sequence point both before and after the call to f() in
each of these therefore there is a sequence point between the
modifications of g.
I think these are interesting because one example I gave where the
subject line expression could give undefined results is where the
hardware only has the ability to flip bits so a write requires a read
first. For (g=g+1) + f(); clearly we can have:
read g
call f
add 1 to the g we read
write new g
Now if my hardware requires a read before write, does the standard
require a second read before the write in the above or is it legitimate
to reuse the first read? I think the standard should require that
second read. I'm not convinced it does.
Tim.