The MASM Forum

Miscellaneous => Irvine Book Questions. => Topic started by: infoMASM on March 14, 2014, 07:08:11 PM

Title: Area of a Cirlce problem
Post by: infoMASM on March 14, 2014, 07:08:11 PM
Hello everyone! I am having trouble with my code to try to find the area of a circle based on what the user inputs as a radius. I need to use the ReadFloat and WriteFloat procedures and use the FLDPI instruction to load pi onto the register stack. I do not believe I am getting the correct values with my code. If anyone can help I would really appreciate it!! Thank you!

INCLUDE Irvine32.inc
.data
myMessage BYTE "Lets calculate the area of a circle!!!!",0dh,0ah,0
prompt BYTE "Please enter a radius: ",0
result BYTE "Your area of the circle is: ",0
temp DWORD ?

.code
main PROC
call Clrscr

mov edx,OFFSET myMessage
call WriteString

mov edx, OFFSET prompt
call WriteString
call ReadFloat ;get radius put on stack

FLDPI ;put pi on stack
fmul st(0), st(0) ;radius squared
fmul st(0), st(1) ;multiplied by pi

mov edx, OFFSET result ;see them off
call WriteString
call WriteFloat ;write from stack
call Crlf

exit
main ENDP

END main
Title: Re: Area of a Cirlce problem
Post by: GoneFishing on March 14, 2014, 07:26:23 PM
After executing FLDPI instruction PI is loaded into st(0) register.
So what you're doing is calculating PI^2 
Title: Re: Area of a Cirlce problem
Post by: jj2007 on March 14, 2014, 07:48:46 PM
You should use either Olly or the deb macro (http://www.webalice.it/jj2006/MasmBasicQuickReference.htm#Mb1019) to understand what your code does:

include \masm32\MasmBasic\MasmBasic.inc              ; download (http://www.masm32.com/board/index.php?topic=94.0)
include \masm32\MasmBasic\IrvineMb\Irvine32Mb.inc      ; needed to build Irvine's (many but not all) 32-bit examples

.data
myMessage BYTE "Lets calculate the area of a circle!!!!",0dh,0ah,0
prompt      BYTE "Please enter a radius: ",0
result      BYTE "Your area of the circle is: ",0
temp      DWORD ?

  Init
  call Clrscr

  mov edx,OFFSET myMessage
  call WriteString

  mov edx, OFFSET prompt
  call WriteString
  call ReadFloat      ;get radius put on stack
  deb 4, "ReadFloat", ST(0), ST(1), ST(2)
  FLDPI                  ;put pi on stack
  deb 4, "fldpi", ST(0), ST(1), ST(2)
  fmul st(0), st(0)      ;radius squared
  deb 4, "fmul_1", ST(0), ST(1), ST(2)
  fmul st(0), st(1)      ;multiplied by pi
  deb 4, "fmul_2", ST(0), ST(1), ST(2)

  mov edx, OFFSET result      ;see them off
  call WriteString
  call WriteFloat      ;write from stack
  call Crlf
  Exit
end start


Output:

Lets calculate the area of a circle!!!!
Please enter a radius: 100

ReadFloat
ST(0)           100.000000000000000
ST(1)           0.0
ST(2)           0.0

fldpi
ST(0)           3.14159265358979324
ST(1)           100.000000000000000
ST(2)           0.0

fmul_1
ST(0)           9.86960440108935862
ST(1)           100.000000000000000
ST(2)           0.0

fmul_2
ST(0)           986.960440108935862
ST(1)           100.000000000000000
ST(2)           0.0

Your area of the circle is: +9.8696044E+002


So you see, as vertograd already posted, you are calculating 100*PI*PI - which is not intended...
Title: Re: Area of a Cirlce problem
Post by: GoneFishing on March 14, 2014, 08:19:31 PM
Also you may use VKdebug macros ( assuming you have MASM32 pack installed on your system : run QEDITOR -> menu Help -> VKdebug Help ):
Quote
include  \masm32\include\masm32rt.inc
include   \masm32\include\debug.inc
includelib \masm32\lib\debug.lib

.data
       var REAL10 123.12345
.code
       start:
              call main
              inkey
              exit
main proc 
        FLD var
        FLDPI
        DumpFPU  ; VKdebug macro , displays the FPU content
        ret
main endp
end  start
Title: Re: Area of a Cirlce problem
Post by: jj2007 on March 14, 2014, 09:47:34 PM
Quote from: vertograd on March 14, 2014, 08:19:31 PM
Also you may use VKdebug macros

They throw a bunch of errors, apparently there are name conflicts with Irvine32.inc

debug.inc(235) : error A2005: symbol redefinition : DumpMem
etc
Title: Re: Area of a Cirlce problem
Post by: GoneFishing on March 14, 2014, 09:59:30 PM
I didn't test them for compatibility with Irvine's macros ( that's why I didn't include IRVINE32.INC in my code sample ) which seem to have their own debugging routines ...




Title: Re: Area of a Cirlce problem
Post by: dedndave on March 15, 2014, 12:39:50 AM
you're original code is pretty close
i moved the first FMUL up one line
and added 2 FFREE's
i didn't test it, but it looks good on paper   :P
call ReadFloat ;get radius put on stack
fmul st(0), st(0) ;radius squared

FLDPI ;put pi on stack
fmul st(0), st(1) ;multiplied by pi

mov edx, OFFSET result ;see them off
call WriteString
call WriteFloat ;write from stack
call Crlf

ffree st(0)
ffree st(1)
Title: Re: Area of a Cirlce problem
Post by: raymond on March 15, 2014, 06:37:08 AM
infoMASM
If you intend to continue using fpu instructions directly, you should have at least a basic understanding of what you are doing. You may thus want to spend a few minutes to start reading the following:

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

If you don't care about the learning, you could always use the accompanying fpu library which has a detailed help file.
Title: Re: Area of a Cirlce problem
Post by: jj2007 on March 15, 2014, 08:03:30 AM
Raymond's tutorial is the best you can get, infoMasm - go for it. Chapter 1 will help you to understand why your instructions above were done in the "wrong" order. Especially Figure 1 is an excellent description of how the FPU works.

@Raymond: One para in Chapter 1 that might be slightly misleading:
QuoteIf, however, an attempt is made to load a value when all the compartments have a value in them, the barrel would still turn by one notch but the attempted loading would fail (just like trying to insert a bullet into a compartment which already contains one). And, in addition, whatever valid value would have been in that compartment now at the TOP is also destroyed, leaving unusable trash in that register at the TOP.

Actually, the value destroyed is the one in ST(7); the value that was in ST(0) is still available, now in ST(1).

QuoteRule #1: An FPU 80-bit register compartment MUST be free (empty) in order to load a value into it.

All values get loaded into ST(0), and normally, ST(0) is not free. But ST(7) must be free. There are two options to guarantee that ST(7) is free:
- fstp st (which discards, however, the current value of ST(0))
- ffree ST(7) (in the frequent case that you still need ST(0))

Simple test:

include \masm32\MasmBasic\MasmBasic.inc
  Init
  FpuFill      ; fill the FPU with 1001 ... 1008
  int 3        ; breakpoint for Olly (http://www.ollydbg.de/version2.html)
  fldpi        ; see what happens with ST(0)
  Exit
end start
Title: Re: Area of a Cirlce problem
Post by: infoMASM on March 15, 2014, 08:53:37 AM
Thank you guys for the help! I wasn't understanding that when I used the FLDPI that pi was being loaded onto st(0) and I was squaring pi. I got the code to work by calling FLDPI first then using the ReadFloat call to get the radius from the user. Thank you everyone!
Title: Re: Area of a Cirlce problem
Post by: raymond on March 15, 2014, 11:30:12 AM
Quotewhatever valid value would have been in that compartment now at the TOP is also destroyed

What WAS in st(7) is now at the top as described earlier in the paragraph. And that value now in st(0) is then trashed. The value destroyed WAS (and not is) the one in st(7). The value now in st(7) was the one in st(6) and is not destroyed.

I fully agree that the value that was in ST(0) is still available, now in ST(1).

This may just seem as semantics (and might be slightly misleading to some if not studied carefully) but must be clearly understood.
Title: Re: Area of a Cirlce problem
Post by: dedndave on March 15, 2014, 12:02:33 PM
ReadFloat initializes the FPU
try the code i posted previously
Title: Re: Area of a Cirlce problem
Post by: jj2007 on March 15, 2014, 12:59:47 PM
Quote from: raymond on March 15, 2014, 11:30:12 AMmight be slightly misleading to some if not studied carefully

Obviously I should study your tutorial more carefully ;-)

So where did I misinterpret Rule #1?

@Dave:
> ReadFloat initializes the FPU
That would be bad news, but it's wrong news ;): Put fldpi, int 3 before ReadFloat, then hit F8 in Olly and see 3.14... in ST(1)
Title: Re: Area of a Cirlce problem
Post by: dedndave on March 15, 2014, 01:29:07 PM
my mistake, Jochen
SetFpu is called, which sets the control word
Title: Re: Area of a Cirlce problem
Post by: jj2007 on March 15, 2014, 01:46:38 PM
Quote from: dedndave on March 15, 2014, 01:29:07 PM
SetFpu is called, which sets the control word
ReadFloat does not modify the precision, though. And both ST6 & ST7 must be free.

This one doesn't trash ST6:
MovVal ST(0), Input$("Gimme a float please: \t",   "1000.001")

And this one trashes ST6 but produces a nice result:

  fldpi
  MovVal ST(0), Input$("Please enter a radius: \t", Str$("%If", 17.8412411615277111))
  fmul st(0), st(0)      ; radius squared
  fmul st(0), st(1)      ; multiplied by pi


I didn't know that fmul st(0), st(0) is possible. The fmul st(0), st(1) can be shortened as fmul without arguments.
Title: Re: Area of a Cirlce problem
Post by: raymond on March 15, 2014, 02:08:14 PM
QuoteSo where did I misinterpret Rule #1?

Rule #1: An FPU 80-bit register compartment MUST be free (empty) in order to load a value into it.

Whenever the FPU is instructed to load another value, it first brings st(7) to the top before attempting the load, without checking if it is free or not. If st(7) HAD been free, the load is successful; otherwise, st(0) now becomes trash.
Title: Re: Area of a Cirlce problem
Post by: jj2007 on March 15, 2014, 05:55:20 PM
 ;)
Title: Re: Area of a Cirlce problem
Post by: Gunther on March 15, 2014, 10:57:41 PM
Quote from: raymond on March 15, 2014, 02:08:14 PM
Rule #1: An FPU 80-bit register compartment MUST be free (empty) in order to load a value into it.

Whenever the FPU is instructed to load another value, it first brings st(7) to the top before attempting the load, without checking if it is free or not. If st(7) HAD been free, the load is successful; otherwise, st(0) now becomes trash.

Raymond is right. He describes a stack rollover.

Gunther
Title: Re: Area of a Cirlce problem
Post by: jj2007 on March 16, 2014, 01:17:01 AM
Quote from: Gunther on March 15, 2014, 10:57:41 PM
Raymond is right. He describes a stack rollover.

If your FPU is full, and you want to use fld MyReal8, what do you do before the fld?
a) ffree st(7)
b) ffree st(0)

Hint: only one answer is correct 8)
Title: Re: Area of a Cirlce problem
Post by: dedndave on March 16, 2014, 01:59:43 AM
a, ffree st(7)
Title: Re: Area of a Cirlce problem
Post by: Force on March 26, 2014, 11:05:38 PM
Hi Raymond

I am trying to make a Ftoa

isn't that  simpliest way for it?


Values :

.data
buf db 50 dup (?)
result db 50 dup (?)
n dd 0
mlt dd 100

val1  dd -125.56
val2  dd 68.9
val   dd 0


and code

fld val1
fld val2
fadd st(0),st(1)
fst val


fimul mlt
fistp val
invoke dwtoa,val,addr buf



buf is =5666
and i separated last 2 characters
result is -56.66 after separated


Title: Re: Area of a Cirlce problem
Post by: qWord on March 27, 2014, 02:39:57 AM
Quote from: Force on March 26, 2014, 11:05:38 PMI am trying to make a Ftoa

isn't that  simpliest way for it?
that method works when you know the range of your results. Otherwise you could extract the mantissa digits by converting the value to a n-digit integer as:
mantissa = x * 10-floor(log10(|x|)) * 10n-1   with 1 ≤ n ≤ 18
The actual digits could be get (e.g.) using the FPU instruction FBSTP, which saves a 18 digit BCD-integer. Remarks that this method is not exact due to rounding errors.
include \masm32\include\masm32rt.inc
.686
.const
    value REAL8 -123456.7890123456789E+10
    nm1 REAL10 1.0E17
    inf REAl10 07fff8000000000000000r ; 1.0*2^(emax+1)
.code
main proc
LOCAL exp10:SDWORD
LOCAL bcd:TBYTE
LOCAL sz[32]:CHAR
   
    finit
    fld value
    .repeat
       
        ; test special values
        fldz
        fcomip st,st(1)
        .if PARITY?
            ; value = NaN
            fstp st
            mov DWORD ptr sz,"NaN"
            .break
        .elseif ZERO?
            ; value = +-zero
            fstp st
            mov DWORD ptr sz,"0.0"
            .break
        .endif
        fld inf
        fcomi st,st(1)
        .if ZERO?
            ; value = +Infinite
            fstp st
            fstp st
            mov DWORD ptr sz,"fnI+"
            mov sz[4],0
            .break
        .endif
        fchs
        fcomip st,st(1)
        .if ZERO?
            ; value = -Infinite
            fstp st
            mov DWORD ptr sz,"fnI-"
            mov sz[4],0
            .break
        .endif

        ; log10(|x|)
        fld st
        fabs
        fldlg2
        fxch
        fyl2x
       
        ; floor(log10(|x|))
        fstcw WORD ptr [esp-4]
        mov ax,WORD ptr [esp-4]
        and ax,NOT 0110000000000y
        or ax,0110000000000y
        mov WORD ptr [esp-2],ax
        fldcw WORD ptr [esp-2]
        frndint
        fldcw WORD ptr [esp-4]
        fist exp10
   
        ; -exp10
        fchs
       
        ;/* 10^(-exp10) = 2^(st(0)*ld(10)) */
        fldl2t
        fmulp st(1),st
        fld st(0)
        frndint
        fsub st(1),st(0)
        fxch
        f2xm1
        fld1
        faddp st(1),st
        fscale
        fstp st(1)
       
        ; mantissa = x * 10^(-exp10)
        fmulp st(1),st
       
        ; mantissa -> integer
        fld nm1
        fmulp st(1),st
        ; set BCD integer
        fbstp bcd
       
        ; get sign
        xor edi,edi
        .if BYTE ptr bcd[9] & 80h
            mov sz[edi],'-'
            add edi,1
        .endif
       
        ; get leading digits and place dot
        movzx eax,BYTE ptr bcd[8]
        mov edx,eax
        shr edx,4
        and eax,0fh
        add edx,'0'
        add eax,'0'
        mov sz[edi+0],dl
        mov sz[edi+1],'.'
        mov sz[edi+2],al
        add edi,3
       
        ; get remaining digits
        mov esi,7
        .while SDWORD ptr esi >= 0
            movzx eax,BYTE ptr bcd[esi]
            mov edx,eax
            shr edx,4
            and eax,0fh
            add edx,'0'
            add eax,'0'
            mov sz[edi+0],dl
            mov sz[edi+1],al
            add edi,2
            add esi,-1
        .endw
       
        ; print exponent if needed
        .if exp10
            mov sz[edi],'E'
            add edi,1
            ; ... integer -> string
            fnc crt_sprintf,ADDR sz[edi],"%+d",exp10
        .else
            mov sz[edi],0
        .endif
    .until 1
   
    print ADDR sz,13,10
    fnc crt_printf,"%E\n",value
    inkey
   
   
    exit
main endp
end main
Title: Re: Area of a Cirlce problem
Post by: Force on March 27, 2014, 09:45:46 AM
Thanks qWord
sure your code will help me