I wrote this function to round singles to integers:
function Round(const Val: Single): Integer;
begin
asm
cvtss2si eax,Val
mov Result,eax
end;
end;
It works, but I need to change the rounding mode. Apparently, per this, I need to set the MXCSR register.
How do I do this in Delphi?
The reason I am doing this in the first place is I need "away-from-zero" rounding (like in C#), which is not possible even via SetRoundingMode.
On modern Delphi, to set MXCSR you can call
SetMXCSR
from theSystem
unit. To read the current value useGetMXCSR
.Do beware that
SetMXCSR
, just likeSet8087CW
is not thread-safe. Despite my efforts to persuade Embarcadero to change this, it seems that this particular design flaw will remain with us forever.On older versions of Delphi you use the
LDMXCSR
andSTMXCSR
opcodes. You might write your own versions like this:These versions are thread-safe and I hope will compile and work on older Delphi versions.
Do note that using the name
Round
for your function is likely to cause a lot of confusion. I would advise that you do not do that.Finally, I checked in the Intel documentation and both of the Intel floating point units (x87, SSE) offer just the rounding modes specified by the IEEE754 standard. They are:
So, your desired rounding mode is not available.