The MASM Forum Archive 2004 to 2012

Project Support Forums => MASM32 => Topic started by: hutch-- on November 28, 2010, 08:42:27 AM

Title: Current version of FPULIB.
Post by: hutch-- on November 28, 2010, 08:42:27 AM
While I would always recommend that you check Ray's site for the latest version, this version is stored locally on the UK server for members who may not be able to download files from a server in the US.

http://www.ray.masmcode.com
Title: Re: Current version of FPULIB.
Post by: frktons on November 28, 2010, 08:56:26 AM
Thanks Steve.  :U
Title: Re: Current version of FPULIB.
Post by: ratu on May 06, 2011, 03:13:19 PM
Just an info:

My first test of this lib (latest version from author site) revealed a serious bug:

FpuFLtoA doesn't work with negative exponents !

ratu
Title: Re: Current version of FPULIB.
Post by: raymond on May 07, 2011, 01:44:22 AM
Would you be kind enough to provide an example of the data you are trying to convert and the instruction (line of code) you used.
Title: Re: Current version of FPULIB.
Post by: MichaelW on May 07, 2011, 02:41:50 AM
Hi Raymond,

I suspect that ratu's post might have something to do with encouraging people to click your download link for FPULIB v2_33. It comes up with a Page Not Found window, and when you close the window you get a dialog that asks if you want to set your home page to...
Title: Re: Current version of FPULIB.
Post by: dedndave on May 07, 2011, 02:57:19 AM
that doesn't sound like "negative exponents" - lol
but, he should give us the value he tried to convert and what he expected for output   :P


http://www.ray.masmcode.com/downloads/Fpulib2_33.zip
Title: Re: Current version of FPULIB.
Post by: raymond on May 07, 2011, 07:12:41 PM
QuoteIt comes up with a Page Not Found window

I just tried it directly from the site and got it without any problem. Have you experienced yourself what you reported?
Title: Re: Current version of FPULIB.
Post by: juozas on May 07, 2011, 07:26:59 PM
Quoteand when you close the window you get a dialog that asks if you want to set your home page to...
This seemz to be a very suspicious. Did you check your pc for viruses, etc? Normally it shouldn't happen. What the browser btw?
Title: Re: Current version of FPULIB.
Post by: MichaelW on May 07, 2011, 07:52:35 PM
Quote from: raymond on May 07, 2011, 07:12:41 PM
Have you experienced yourself what you reported?

Yes, I would not have posted this if I could not get it to repeat. Now that I test further, if I paste the url in the address bar I can get the zip:

http://www.ray.masmcode.com/downloads/Fpulib2_33.zip

But if I just click the link on your page I get what I reported, and in recent times I have not observed this sort of behavior anywhere else.

Title: Re: Current version of FPULIB.
Post by: oex on May 07, 2011, 08:07:30 PM
I get the download either way.... What webpage link pointing to the zip is at fault?

http://www.masm32.com/board/index.php?topic=15462.0
http://www.ray.masmcode.com/fpu.html#fpulib

other?
Title: Re: Current version of FPULIB.
Post by: MichaelW on May 07, 2011, 08:40:10 PM
The link I have the problem with is here:

http://www.ray.masmcode.com/fpu.html#fpulib

Title: Re: Current version of FPULIB.
Post by: oex on May 07, 2011, 08:47:51 PM
I'm sorry Michael.... I cant seem to replicate your problem.... I know how frustrating website problems can be with only one tester but maybe Raymond will have more luck....

With regard to the negative exponent bug I have not tested....
Title: Re: Current version of FPULIB.
Post by: dedndave on May 07, 2011, 09:37:33 PM
hmmmm - check you hosts file, Michael (C:\Windows\System32\Drivers\etc)
the url may have been hijacked
otherwise, use HiJackThis to see if it has somehow been redirected
Title: Re: Current version of FPULIB.
Post by: MichaelW on May 07, 2011, 10:31:59 PM
The only problems with negative exponents that I find are at the extreme small end of the REAL10 range.

3.400000E+0038
1.180000E-0038

1.790000000000000E+0308
2.230000000000000E-0308

1.180000000000000E+4932
ERROR
ERROR
3.359999999999999E-4917


Code and EXE in attachment.

And BTW I also tested with the 19 digits reduced to 15, and this did not change the results.
Title: Re: Current version of FPULIB.
Post by: MichaelW on May 07, 2011, 10:43:00 PM
Dave,

My hosts file contains only the localhost entry. I started out trying to download the library so I could test, but when I ran into the problem and verified that it was not just a temporary malfunction, I thought I should report what was happening. Apparently it's only a problem on my end.
Title: Re: Current version of FPULIB.
Post by: jj2007 on May 07, 2011, 11:08:35 PM
Quote from: MichaelW on May 07, 2011, 08:40:10 PM
The link I have the problem with is here:

http://www.ray.masmcode.com/fpu.html#fpulib


No such problem here, it works. And files are identical to what I downloaded in August 2010, except for FPUlibtester.exe
Title: Re: Current version of FPULIB.
Post by: raymond on May 08, 2011, 01:57:31 AM
Quote from: MichaelW on May 07, 2011, 10:31:59 PM
The only problems with negative exponents that I find are at the extreme small end of the REAL10 range.

3.400000E+0038
1.180000E-0038

1.790000000000000E+0308
2.230000000000000E-0308

1.180000000000000E+4932
ERROR
ERROR
3.359999999999999E-4917


Code and EXE in attachment.

And BTW I also tested with the 19 digits reduced to 15, and this did not change the results.


Thanks for that test Michael. And testing with 15 of 19 digits would not make any difference because the function, as written, immediately reduces anything higher than 15 to that limit.

I've found the reason why an error occurs in that range and E-4917 would be the smallest without getting an error. This will be corrected within the next few days. The possibility of converting denormalized REAL10 numbers will also be included.

I wonder if ratu's post had anything to do with this flaw. :red
Title: Re: Current version of FPULIB.
Post by: dedndave on May 08, 2011, 02:57:47 AM
QuoteI wonder if ratu's post had anything to do with this flaw.  :red

:bg
we knew you'd find the bug, Ray   :U
Title: Re: Current version of FPULIB.
Post by: raymond on May 14, 2011, 08:51:05 PM
The reason why the FpuFLtoA function was returning an error for values smaller than 3.36e-4917 was due to the following factors in the conversion algo:
i) The value needs to be converted to what would be an 18-digit integer before it is dumped to memory as a packed BCD with the "fbstp" instruction.
ii) The log10 of the value is computed to get the power of 10 required to express it in scientific notation.
iii) But using the reverse of that power of 10 would only provide a single digit for the integer portion. The number of decimal digits specified must also be added to obtain all the required digits for proper display.

For example, if the value to be converted was 5.123456789e-65 and the converted value was to be displayed with 6 decimal digits, it would have to be multiplied by 10(6+65) to get an integer with 7 digits, i.e. 5123457. However, if the value to be converted was 5.123456789e-4930, it would have to be multiplied by 10(6+4930) which exceeds the upper range of the FPU. This resulted in an invalid operation which was detected by the algo causing the ERROR message to be displayed.

Remedy:
If the required power of 10 to convert the value to an 18-digit integer exceeds 4931, the excess is kept in a memory variable X and the value is first multiplied by 104931. If X!=0, that is then further multiplied by 10X before storing the packed BCD with the required number of digits.

The Fpulibtester in the new zip file was also updated to correct the broken scientific notation. Any resulting value lower than 10-15 also gets displayed in scientific notation as default regardless of the choice with the radio buttons. The FpuSinh function also needed a minor correction to display the required value.

This latest version of the Fpulib v2-34 is available as usual from my site
http://www.ray.masmcode.com/fpu.html#fpulib
and the zipped file is also attached.
Title: Re: Current version of FPULIB.
Post by: jj2007 on May 15, 2011, 12:23:59 AM
Raymond,

Your FpuLib is a fantastic resource. Thanks for the tremendous efforts that have gone into this library.

:U
Title: Re: Current version of FPULIB.
Post by: raymond on May 15, 2011, 01:23:56 AM
Thanks :wink
Title: Re: Current version of FPULIB.
Post by: MichaelW on May 15, 2011, 01:59:37 AM
Well, I hope I'm doing something wrong, but while the changes fixed the problems at the small end of the REAL10 range, positive exponents at the large end of all three ranges, values that previously worked, don't work now. For a REAL8, the largest workable value, determined by trial, appears to be just below 10.0e15.

;================================================================================
    include \masm32\include\masm32rt.inc
    include fpu.inc
    includelib fpu.lib
;================================================================================
; Approximate limits per Simply Fpu:
; REAL4: 3.4e38, 1.17e-38
; REAL8: 1.79e308, 2.22e-308
; REAL10: 1.19e4932, 3.36e-4932
; Approximate limits per MASM 6.0 Programmers Guide:
; REAL4: 3.4e38, 1.18e-38
; REAL8: 1.79e308, 2.23e-308
; REAL10: 1.18e4932, 3.37e-4932
;================================================================================
    .data

        r4_0      REAL4 3.40e38
        r4_1      REAL4 1.18e-38

        r8_00     REAL8 9.99999999999999e15
        r8_01     REAL8 10.0e15
        r8_0      REAL8 1.79e308
        r8_1      REAL8 2.23e-308

        r10_0     REAL10 1.18e4932
        r10_1     REAL10 3.37e-4932

        buff      db     40 dup(0)

    .code
;================================================================================
start:
;================================================================================

    invoke FpuFLtoA, ADDR r4_0, 6, ADDR buff, SRC1_REAL4 or SRC2_DIMM or STR_SCI
    print ADDR buff,13,10

    invoke FpuFLtoA, ADDR r4_1, 6, ADDR buff, SRC1_REAL4 or SRC2_DIMM or STR_SCI
    print ADDR buff,13,10,13,10

    invoke FpuFLtoA, ADDR r8_00,15, ADDR buff, SRC1_REAL8 or SRC2_DIMM or STR_SCI
    print ADDR buff,13,10

    invoke FpuFLtoA, ADDR r8_01,15, ADDR buff, SRC1_REAL8 or SRC2_DIMM or STR_SCI
    print ADDR buff,13,10

    invoke FpuFLtoA, ADDR r8_0, 15, ADDR buff, SRC1_REAL8 or SRC2_DIMM or STR_SCI
    print ADDR buff,13,10

    invoke FpuFLtoA, ADDR r8_1, 15, ADDR buff, SRC1_REAL8 or SRC2_DIMM or STR_SCI
    print ADDR buff,13,10,13,10

    invoke FpuFLtoA, ADDR r10_0, 15, ADDR buff, SRC1_REAL or SRC2_DIMM or STR_SCI
    print ADDR buff,13,10

    invoke FpuFLtoA, ADDR r10_1, 15, ADDR buff, SRC1_REAL or SRC2_DIMM or STR_SCI
    print ADDR buff,13,10,13,10



    inkey "Press any key to exit..."
    exit
;================================================================================
end start


ERROR
1.180000E-0038

9.999999999999990E+0015
ERROR
ERROR
2.230000000000000E-0308

ERROR
3.370000000000000E-4932

Title: Re: Current version of FPULIB.
Post by: raymond on May 15, 2011, 03:53:16 AM
 :red :red A real newbie error. :red :red

Unless the .if statement is qualified, the comparison is unsigned. That's what happened in the short added code. :snooty: If the value to be converted would exceed a 16-digit integer, it gets displlayed by default in the scientific notation. It thus has to be multiplied by a negative power of 10 to reduce its size before converting it to a packed BCD. The unsigned comparison of that negative power with 4931 was obviously the wrong thing to do. :eek

The attachement in my previous post has been edited with the revised zip file containing the corrected function. The included Fpulibtester has also been recompiled with the corrected code.

The corrected file was also uploaded to my site. My apology for any inconvenience. And, many thanks again Michael for picking that up so rapidly.
Title: Re: Current version of FPULIB.
Post by: MichaelW on May 15, 2011, 09:17:11 PM
This turned out to be more complex, and slower, than what I originally envisioned. It should be possible to do this same basic thing with a REAL4, or with an appropriate CRT a REAL10.

;==============================================================================
    include \masm32\include\masm32rt.inc
    include fpu.inc
    includelib fpu.lib
;==============================================================================

printf MACRO format:REQ, args:VARARG
    IFNB <args>
        invoke crt_printf, cfm$(format), args
    ELSE
        invoke crt_printf, cfm$(format)
    ENDIF
    EXITM <>
ENDM

;==============================================================================
    .data
        r8      REAL8 ?
        coef1   dq 0
        coef2   dq 0
        exp1    dd 0
        exp2    dd 0
        buff1   db 40 dup(0)
        szcoef1 db 40 dup(0)
        szexp1  db 10 dup(0)
        buff2   db 40 dup(0)
        szcoef2 db 40 dup(0)
        szexp2  db 10 dup(0)
    .code
;==============================================================================

rand_r8 proc
  @@:
    invoke nrandom, -1
    push eax
    invoke nrandom, -1
    push eax
    fld REAL8 PTR [esp]           ; load 64-bit random integer as REAL8
    add esp, 8
    fstp r8
    invoke FpuExam, ADDR r8, SRC1_REAL8
    test eax, XAM_VALID
    jz  @B                        ; try again if invalid
    fld r8
    ret
rand_r8 endp

;==============================================================================
start:
;==============================================================================
    xor ebx, ebx
    .WHILE ebx < 5000000

      invoke rand_r8
      fstp r8

      invoke FpuFLtoA,ADDR r8,15,ADDR buff1,SRC1_REAL8 or SRC2_DIMM or STR_SCI
      ;printf( "%s\t", ADDR buff1 )

      ;----------------------------------------------------
      ; Extract coefficient and convert to 64-bit integer.
      ;----------------------------------------------------

      invoke crt_strtok, ADDR buff1, chr$(".Ee")
      invoke crt_strcpy, ADDR szcoef1, eax
      ;printf( "%s\t", ADDR szcoef1 )
      invoke crt_strtok, 0, chr$(".Ee")
      invoke crt_strcat, ADDR szcoef1, eax
      ;printf( "%s\t", ADDR szcoef1 )
      invoke crt__atoi64, ADDR szcoef1
      ;printf( "%I64d\t", edx::eax )
      mov DWORD PTR coef1+4, edx
      mov DWORD PTR coef1, eax
      ;printf( "%I64d\t", coef1 )

      ;-------------------------------------------------
      ; Extract exponent and convert to 32-bit integer.
      ;-------------------------------------------------

      invoke crt_strtok, 0, chr$(".Ee")
      invoke crt_strcpy, ADDR szexp1, eax
      ;printf( "%s\n", ADDR szexp1 )
      invoke crt_atoi, ADDR szexp1
      mov exp1, eax

      invoke crt_sprintf, ADDR buff2, cfm$("%.15e"), r8
      ;printf( "%s\t", ADDR buff2 )

      invoke crt_strtok, ADDR buff2, chr$(".Ee")
      invoke crt_strcpy, ADDR szcoef2, eax
      ;printf( "%s\t", ADDR szcoef2 )
      invoke crt_strtok, 0, chr$(".Ee")
      invoke crt_strcat, ADDR szcoef2, eax
      ;printf( "%s\t", ADDR szcoef2 )
      invoke crt__atoi64, ADDR szcoef2
      ;printf( "%I64d\t", edx::eax )
      mov DWORD PTR coef2+4, edx
      mov DWORD PTR coef2, eax
      ;printf( "%I64d\t", coef2 )

      invoke crt_strtok, 0, chr$("Ee")
      invoke crt_strcpy, ADDR szexp2, eax
      ;printf( "%s\n", ADDR szexp2 )
      invoke crt_atoi, ADDR szexp2
      mov exp2, eax

      ;--------------------------------------------------
      ; Compare the coefficients and display an error if
      ; the absolute value of the difference exceeds 1.
      ;--------------------------------------------------

      mov eax, DWORD PTR coef1+4
      cmp eax, DWORD PTR coef2+4
      jne coef_error
      push DWORD PTR coef1
      mov eax, DWORD PTR coef1
      sub eax, DWORD PTR coef2
      pop DWORD PTR coef1
      invoke crt_abs, eax
      cmp eax, 1
      ja  coef_error
      ;printf( "\nabs %d\n", eax )
      jmp @F
    coef_error:
      printf( "ERROR %I64d\t%I64d\n", coef1, coef2 )
    @@:

      ;--------------------------------------
      ; Compare the exponents and diaplay an
      ; error if they are not identical.
      ;--------------------------------------

      mov eax, exp1
      cmp eax, exp2
      je  @F
      printf( "ERROR %d\t%d\n", exp1, exp2 )
    @@:

      inc ebx
      test ebx, 0ffffh
      jnz @F
      printf( "." )
    @@:
    .ENDW
    printf( "\n\n" )

    inkey "Press any key to exit..."
    exit
;==============================================================================
end start