The MASM Forum Archive 2004 to 2012

General Forums => The Workshop => Topic started by: lamer on January 09, 2006, 06:35:30 PM

Title: "CALL" with Windows API
Post by: lamer on January 09, 2006, 06:35:30 PM
Hi all!
It seems to be a bit stupid question, but anyway...
Can I stay quiet using call instead of invoke for Windows API calls?
I mean cleaning up the stack.
Will this code be executed without any nasty outcomes:

.data?
hLib        dd        ?
.data
szUserLib  db       "user32.dll",0
szSomeFunc   db  "SomeAPI",0

.code

LOCAL hFunc    :DWORD

invoke LoadLibrary,addr szUserLib
.if eax
   mov hLib eax
   invoke GetProcAddress,addr szSomeFunc
   .if eax
      mov hFunc,eax
      push 1
      push 2
      call hFunc
   .endif
   invoke FreeLibrary,hLib
.endif


Thank you
Title: Re: "CALL" with Windows API
Post by: Vortex on January 09, 2006, 07:10:13 PM
Yes, you can do it. Why not?
Title: Re: "CALL" with Windows API
Post by: lamer on January 09, 2006, 07:42:18 PM
Who knows what they do in their functions? :bg
Generally speaking I think too - why not?
Thanks.
Title: Re: "CALL" with Windows API
Post by: hutch-- on January 09, 2006, 08:47:29 PM
lamer,

The only difference is between calling conventions, STDCALL you just use push / call notation but with C calling you must balance the stack yourself after the procedure has returned.
Title: Re: "CALL" with Windows API
Post by: lamer on January 09, 2006, 09:40:55 PM
hutch,
So .model flat,stdcall
allows me not to care about stack balancing?
Title: Re: "CALL" with Windows API
Post by: QvasiModo on January 10, 2006, 12:43:54 AM
Hi :)

The .model flat,stdcall directive only sets the default calling convention as stdcall, so no, you can't stop worrying (unless you use invoke, that is).

An example of a C call is wsprintf:


invoke wsprintf, addr OutputBuffer, CTXT("%i"), SomeInteger


Becomes...


push SomeInteger
push CTXT("%i")
lea eax, OutputBuffer
push eax
call wsprintf
add esp, 12
Title: Re: "CALL" with Windows API
Post by: Ian_B on January 10, 2006, 12:53:40 AM
Quote from: QvasiModo on January 10, 2006, 12:43:54 AM

push SomeInteger
push CTXT("%i")
lea eax, OutputBuffer        <------- NOTE THESE LINES!
push eax                     <-------
call wsprintf
add esp, 12


This can cause some issues if you aren't careful. MASM apparently always uses the EAX register as noted above to perform calculations before pushing the results. If you had stored one of the parameters in EAX and were going to push that AFTER such a calculation, the value would be trashed. I believe the compiler throws an error for this, but it does pay to take care what order you push parameters if any values are stored in EAX and there are address calculations being made in the parameter list.

Personally, I choose to explicitly set registers to values where at all possible using LEA etc. before the invoke, then use the registers as parameters, so I don't inadvertently get caught out. It also means a tiny bit more control over when the calculations are made for better optimisation.

IanB
Title: Re: "CALL" with Windows API
Post by: lamer on January 10, 2006, 04:31:09 AM
But how can I know which convention the function uses?  :(
Is there any mention in documentation?
Title: Re: "CALL" with Windows API
Post by: MichaelW on January 10, 2006, 05:06:07 AM
It would be mentioned somewhere in the documentation, but probably not in the function-specific documentation, unless the function deviated from the norm. To my knowledge all of the Win32 API functions other than wsprintf use the STDCALL calling convention, and all of the CRT functions (MSVCRT.DLL, CRTDLL.DLL, etc) use the C calling convention.
Title: Re: "CALL" with Windows API
Post by: lamer on January 10, 2006, 06:36:50 AM
Thanks! :thumbu
Title: Re: "CALL" with Windows API
Post by: hutch-- on January 10, 2006, 08:58:42 AM
lamer,

The basics are that the stack must be corrected at the exit of the procedure that uses the stack for arguments passed to it. With STDCALL the procedure does the correction and that is why you regularly see "RET number" where number may for example be 12 bytes this "RET 12" The number must match the number of bytes that were pushed onto the stack before the procedure was called.

This technique is fine if you have a fixed number of arguments of known sizes but it does not work when you have a variable number of arguments. The alternative is the C calling convention where the calling code does the stack cleanup, usually by adding the byte count to ESP. The procedure that is called when its a C calling convention oly uses a RET without the trailing number.

A C call in code would looks something like this.


push arg2
push arg1
call proc
add esp, 8


With STDCALL where the procedure balances the stack, you have this.


push arg2
push arg1
call proc
Title: Re: "CALL" with Windows API
Post by: lamer on January 10, 2006, 01:18:01 PM
hutch,
Thank you very much!
My mind is clear now :U