I can't find any consistency in Delphi's handling of unsigned operations. Fixing this bug will break test tw3490. See the small program below with the outputs of FPC and Kylix.
varA,A2:byte;W:word;B,B2:cardinal;I:smallint;beginB:=$ffffffed;B2:=$fffffffd;Writeln(B-B2);W:=65535;A:=20;// FPC converts signed operations to DWord, a*w -> Dword type// Not sure what delphi does here....Writeln(a*w-256000000);end.
Kylix:
4294967280
-254689300
FPC unpatched:
4294967280
4040277996
FPC with patch to use int64 calculations for cardinal-cardinal:
-16
-254689300
The problem is that we cannot make it Delphi compatible if we don' know what the rules are.
And even if we can, it's quite possible it will break again when we/you move to 64 bit. E.g.:
{$q+}varb:byte;w:word;c:cardinal;beginb:=100;w:=100;c:=100;inc(b,-1);// generates overflow error at run time in Delphiinc(w,-1);// generates overflow error at run time in Delphiinc(c,-1);// does not generate any error at all in Delphiend.
A big problem with this is: what do we do in FPC on 64 bit platforms? Currently, give an error for inc(c,-1) there, but we do not give one for inc(qword_var,-1). This is Delphi-incompatible, but what can you do when you have nonsensical behaviour like this?
I think the best solution for FPC is to add the fix to make cardinal-cardinal return as int64. That gives the most predictive results and is consistent with the behavior of word-word and byte-byte.
From the linked news posts it is only written that it is not seen as a delphi bug. But the rules are not written clearly either.
And as last the remark of "optimization off" makes it clearly a delphi issue, because the optimizer should change behavior of the program.
But your remark about "optimization off" is not fair.
The sample code from Jonas does not produce any result or anything.
So the Kylix compiler does not generate ANY code for this in {$O+}.
You can see it in the CPU Window.
As soon as these variables are finally "used" for something, for example
writeln(b,w,c); it does generate code.
This optimization comment is quite interesting. The current behaviour partially stems from bug #6081 (closed). But the first inc() statement in that program indeed doesn't generate an overflow under Delphi only because it's optimized away. If you add a writeln(v), it indeed does generate an overflow under Delphi.
Being holier than the pope one could even say the delphi optimizer is buggy. Statements with side effects shouldn't be optimized away and those inc definitively have a side effect (throwing an exception).
I really wouldn't like to argue with you, because I have almost no idea about the great work you're doing.
But doesn't this get a bit philosophic now?
I see it like this:
Jonas Code "does" nothing in the end.
So it can safely be optimized away without side effect.
It's as good as no code at all, but no code doesn't use CPU time.
If you want to know what it "would do, if it did something" ;-) for debugging purpose you can always switch to {$O-}
The point is simply that "throwing an exception" is also doing something :) So Delphi is not only optimizing away an inc-statement, but also a raise-statement.
Well, it's a border and I think the decision to remove the statement is correct. It's like FPC deciding to change the order how parameters are evaluted if it generates better code then.