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
After executing FLDPI instruction PI is loaded into st(0) register.
So what you're doing is calculating PI^2
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...
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
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
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 ...
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)
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.
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
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!
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.
ReadFloat initializes the FPU
try the code i posted previously
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)
my mistake, Jochen
SetFpu is called, which sets the control word
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.
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.
;)
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
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)
a, ffree st(7)
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
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|)) * 10
n-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
Thanks qWord
sure your code will help me