[fpc-pascal] Floating-point number representation on various platforms
Tom Verhoeff
T.Verhoeff at tue.nl
Sat Feb 18 09:43:31 CET 2006
On Sun, Feb 05, 2006 at 01:39:25PM +0100, Jonas Maebe wrote:
>
> On 05 Feb 2006, at 13:32, Tom Verhoeff wrote:
>
> >Question 2: Any suggestions as to how to make it more platform
> >independent,
> >or how to support multiple platforms in an easy way?
>
> The only differences are little/big endian. So if it works on PPC and
> I386, you have support for all platforms. Just use {$ifdef
> FPC_LITTLE_ENDIAN}
[This is about IEEE floating-point numbers]
Two further questions.
Which platforms use which endian? (Of course, I could just experiment,
or maybe even read the FPC source, but asking is simpler. :-)
When I looked at my code again, it seems to me that the change that I made
to support both PPC and I386 is not endian-related, but I may be wrong,
or it may be more complicated.
So, here are some pieces of the code, it uses a variant array to
access the bits of the IEEE floating-point numbers in Single and Double:
-----
interface
const
SignIndexSingle = 31; { position of sign bit in single format }
ExpSizeSingle = 8; { size of exponent field in single format }
MaxExpSingle = 255;
ExpBiasSingle = 127;
EminSingle = 1 - ExpBiasSingle;
EmaxSingle = MaxExpSingle - 1 - ExpBiasSingle;
FracSizeSingle = 23; { size of fraction field in single format }
type
BitIndexSingle = 0 .. SignIndexSingle;
ConvertSingle = record
case Boolean of
False: ( d: Single )
; True : ( s: set of BitIndexSingle ) { i in s == bit i = 1 }
end;
procedure WriteBitsSingle ( const x: Single );
procedure WriteBinarySingle ( const x: Single );
function PowerOf2Single ( const x: Single; n: Integer ): Single;
const
SignIndexDouble = 63; { position of sign bit in double format }
ExpSizeDouble = 11; { size of exponent field in double format }
MaxExpDouble = 2047;
ExpBiasDouble = 1023;
EminDouble = 1 - ExpBiasDouble;
EmaxDouble = MaxExpSingle - 1 - ExpBiasDouble;
FracSizeDouble = 52; { size of fraction field in double format }
type
BitIndexDouble = 0 .. SignIndexDouble;
ConvertDouble = record
case Boolean of
False: ( d: Double )
; True : ( s: set of BitIndexDouble ) { i in s == bit i = 1 }
end;
procedure WriteBitsDouble ( const x: Double );
procedure WriteBinaryDouble ( const x: Double );
procedure FlipBitDouble ( var x: Double; i: BitIndexDouble );
function PowerOf2Double ( const x: Double; n: Integer ): Double;
-----
It turns out that my code for the Single type works on both PPC
and I386 without change. However for Double, I had to write e.g.
-----
procedure WriteBitsDouble ( const x: Double );
var c: ConvertDouble; i, i_: BitIndexDouble;
begin
with c do begin
d := x
; for i := SignIndexDouble downto 0 do begin
{$IFDEF CPUPOWERPC}
i_ := (i + 32) mod 64
{$ENDIF}
{$IFDEF CPUI386}
i_ := i
{$ENDIF}
; if i_ in s then write ( '1' )
else write ( '0' )
; if i in [ SignIndexDouble, SignIndexDouble - ExpSizeDouble ] then
write ( ' ' )
end { for i }
end { with c }
end; { WriteBitsDouble }
-----
That is, the lower and upper 32 bits (4 bytes) got swapped.
Maybe this is the case because both the 64-bit set and the Double
get allocated differently depending on endianess?
Can someone enlighten me?
Thanks,
Tom
--
E-MAIL: T.Verhoeff @ TUE.NL | Fac. of Math. & Computing Science
PHONE: +31 40 247 41 25 | Eindhoven University of Technology
FAX: +31 40 247 54 04 | PO Box 513, NL-5600 MB Eindhoven
http://www.win.tue.nl/~wstomv/ | The Netherlands
More information about the fpc-pascal
mailing list