You guys are geniuses,...:dazzled:
What is the most reliable assembly language version of the standard windows application Message Loop ???
(I've been reading alot of source code lately from old examples, and, I've noticed an IMMENSE amount of variability.)
in my "WinMain" routine, EDI is zero throughout much of the code
EDI = 0
;------------------------------
;message loop
mov esi,offset msg ;ESI = msg structure address
jmp short mLoop1
mLoop0: INVOKE TranslateMessage,esi
INVOKE DispatchMessage,esi
mLoop1: INVOKE GetMessage,esi,edi,edi,edi
inc eax ;exit only
shr eax,1 ;if 0 or -1
jnz mLoop0 ;otherwise, we loop
;------------------------------
;exit program
INVOKE ExitProcess,[esi].MSG.wParam
Quote from: Zen on May 21, 2014, 05:12:26 AM
You guys are geniuses,...:dazzled:
The problem with people who consider themselves geniuses is that some of them have too big an ego sometimes. Take hutch for example. He thinks he is so smart that anything ever written in DOS isn't worth even looking at. He thinks that anything written in DOS should be trash canned. He hasn't looked at ADAM and thinks it should be trash canned because it was written back in the early 1980s when DOS and the 8088 processor were state of the art. he moved my project of converting ADAM to MASM32 to the DOS section of this forum so few of you will look at it. I guess, because DOS programs aren't worth upgrading. I'm attaching Pro Football Prophet in the version that will run on Window 7. I have a better version but, rather than fix bugs that were in earlier versions of DOS emulation, Microsoft did away with "Full Screen Mode" so that version won't run on Windows 7. I defy hutch to write a system that will do as much as Pro Football Prophet will do. It's attached here. (It still has problems running on laptops but I intend to fix those before this NFL football season starts.)
Will, this is your first and last warning, try and start an argument in here and you will be given the opportunity to explore Win32 assembler programming in another venue.
Quote from: Will on May 21, 2014, 10:13:11 AMThe problem with people who consider themselves geniuses is that some of them have too big an ego sometimes.
Yes, we have all seen the bloated ego over the last weeks. Another problem is that people who consider themselves geniuses are sometimes real idiots.
On topic:
.While 1
invoke GetMessage, ADDR msg, NULL, 0, 0
.Break .if !eax
invoke TranslateMessage, ADDR msg
invoke DispatchMessage, ADDR msg
.EndwThis works - I've never seen it fail. However, occasionally you see
.Break .if sdword ptr eax<=0instead, claiming that eax will be negative in case of an error. The official MSDN (http://msdn.microsoft.com/en-us/library/windows/desktop/ms644928%28v=vs.85%29.aspx) version is this:
while( (bRet = GetMessage( &msg, NULL, 0, 0 )) != 0)
{
if (bRet == -1)
{
// handle the error and possibly exit
}
else
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
// Return the exit code to the system.
return msg.wParam;If you want to be on the safe side, check for zero and -1:
.While 1
lea edi, msg ; -6 bytes
xor eax, eax ; -1 - tribute to Dave ;-)
invoke GetMessage, edi, eax, eax, eax
inc eax
.Break .if Zero? ; an error
dec eax
.Break .if Zero? ; normal exit
invoke TranslateMessage, edi
invoke DispatchMessage, edi
.EndwP.S.: Dave's
shr eax, 1 thing is actually one byte shorter :(
actually, we had that discussion about 3 years ago :biggrin:
http://www.masmforum.com/board/index.php?topic=17205.0 (http://www.masmforum.com/board/index.php?topic=17205.0)
When will GetMessage return -1? (http://blogs.msdn.com/b/oldnewthing/archive/2013/03/22/10404367.aspx)
Quote from: Raymond Chen
But don't worry, the standard message pump is safe. If your parameters are exactly
•a valid pointer to a valid MSG structure,
•a null window handle,
•no starting message range filter,
•no ending message range filter,
then GetMessage will not fail with -1.
According to my PSDK:
Quote
If there is an error, the return value is -1. For example, the function fails if hWnd is an invalid window handle or lpMsg is an invalid pointer.
This is why I have never taken any notice of the potential -1 return value, if your code is correct, it never happens. If your code is not correct, fix it.
Well, if we're going for "Ultimate":
.while TRUE
invoke GetMessage, msg, NULL, 0, 0
.if eax, e, 0
jmp @f
.endif
invoke TranslateAccelerator, [gWnd], [hAccel], msg
.if eax, e, 0
invoke IsDialogMessage, [gWnd], msg
.if eax, e, 0
invoke TranslateMessage, msg
invoke DispatchMessage, msg
.endif
.endif
.endw
@@:
mov eax, [msg.wParam]
TranslateAccelerator handles keyboard shortcuts, I almost always use them!
IsDialogMessage allows special handling of keystrokes--Enter, Tab, ...--in Dialogs, and special handling of keystrokes for buttons, lists, etc which are part of "Regular" windows.
The code is fasm syntax! That's why things look a little different. My only assembler.
hth,
farrier
well - he wanted "ultimate standard" :lol:
so, i didn't put the accelerator stuff in there
but, here's the one i use for MDI windows with accelerators
;EBX = hwndFrame
;ESI = hwndClient
;EDI = 0
;----------------------------------------
;message loop
mov ebp,offset msg ;EBP = address of MSG structure
jmp short mLoop1
mLoop0: INVOKE TranslateMDISysAccel,esi,ebp
or eax,eax
jnz mLoop1
INVOKE TranslateAccelerator,ebx,haccelMain,ebp
or eax,eax
jnz mLoop1
INVOKE TranslateMessage,ebp
INVOKE DispatchMessage,ebp
mLoop1: INVOKE GetMessage,ebp,edi,edi,edi
inc eax ;exit only
shr eax,1 ;if 0 or -1
jnz mLoop0
;----------------------------------------
;terminate
INVOKE ExitProcess,[ebp].MSG.wParam
the message loop is not a great place for .if or .while
it's one of the places i still prefer the old label method
Dave,
Quote from: dedndave on May 22, 2014, 02:18:32 AM
the message loop is not a great place for .if or .while
it's one of the places i still prefer the old label method
good solution. :t
Gunther
some guys (Hutch is one of them) like to place the MSG structure on the WinMain stack frame
that doesn't make any sense, to me - lol
the stack frame is a great place for data that is both temporary and private
WinMain is not what i consider temporary
it has the same longevity as the instance of the process
i.e., you may as well put the MSG structure in the .DATA? section
it doesn't use or save any memory over the stack frame method
it doesn't make the EXE any larger (slightly smaller, perhaps - no stack frame code)
memory consumption is memory consumption, whether it's on the stack or in the uninitialized data section
that having been said, by not using a stack frame in WinMain, you can use EBP as a preserved general register :t
...Definitely geniuses,...Thanks alot to everyone :biggrin:
This is the Message Loop that I use in every assembly program,...it's the super-deluxe version (that I undoubtedly highjacked from an Australian assembly geek),...the best feature is that it IS IMPERVIOUS TO SYSTEM ERRORS,...
.WHILE TRUE
invoke GetMessage, ADDR msg, NULL, 0, 0
.BREAK .IF (!eax)
invoke TranslateMessage, ADDR msg
invoke DispatchMessage, ADDR msg
.ENDW
mov eax, msg.wParam
RET
...The only thing that really mystified me was the possible return of -1 from GetMessage, which was explained in Raymond Chen's blog post (thanks, SINSI),...I probably shouldn't have titled the original post "ultimate message loop",...what I was REALLY looking for was a message loop that is both mindlessly simple and completely foolproof,...(to, say,..."zen-optimization"),... :icon_cool:
Quote from: dedndave on May 22, 2014, 03:28:04 AMby not using a stack frame in WinMain, you can use EBP as a preserved general register :t
What exactly is the problem?
WinMain proc
LOCAL msg:MSG
..
lea edi, msg ; -6 bytes
.While 1
xor ebp, ebp ; another tribute to Dave ;-)
xor eax, eax
invoke GetMessage, edi, eax, eax, eax
inc eax
shr eax,1 ; DednDave, -1
.Break .if Zero? ; an error
invoke TranslateMessage, edi
invoke DispatchMessage, edi
.Endw
...Clearly DAVE has given this alot of thought,...
We REALLY should have a DAVE tribute thread (where we can savagely attack his character and humiliate him,...)
...All kidding aside,...in reality, I was writing a message loop for an MDI application, and, had (no surprise) totally trashed it,...
Dave is (I think) psychic, because he provided the perfect solution,...:shock:
it will work fine, Jochen - no problem
but, in contrast...
.DATA?
msg MSG <>
.CODE
WinMain PROC
mov esi,offset msg
;message loop
INVOKE ExitProcess,[esi].MSG.wParam
WinMain ENDP
EBP is now free to use
which brings up another issue.....
Start: INVOKE GetModuleHandle,NULL
mov hInstance,eax
INVOKE GetCommandLine
INVOKE WinMain,hInstance,hPrevInstance,eax,nCmdShow
INVOKE ExitProcess,eax
WinMain PROC hInst:HINSTANCE,hPrevInstance:HINSTANCE,lpCmdLine:LPSTR,nCmdShow:DWORD
mov eax,msg.wParam
ret
WinMain ENDP
END Start
more compiler twaddle - lol
we are assembly programmers - that stuff is all meaningless
Zen aka Baltoro.....
if you are interested in MDI, here's a small example
it includes the arrange menu items, a working MDI child menu, and accelerators
as you create and destroy MDI children, they are added and removed from the File menu
it maintains a count of children, initialized to 1 because the first child is created for you
Dave,
thank you for providing the source. :t Works fine and should be instructive for starters.
Gunther
DAVE !!!
I have previously downloaded a number of your "greatest hits", including several varieties of MDI projects,,,THANKS,...
Quote from: dedndave on May 22, 2014, 03:49:19 AMEBP is now free to use
That's correct, but what for? That's just a boring message loop, not an ultrahighspeedfancyinnermostloopthatneedsallmyregisters ;-)
MDI windows are a little tricky - lol
first, there is the window hierarchy
Frame
MDIClient
MDIChild
MDIChild
MDIChild
then, there are the create structures
after you get all that right, there is the matter of arranging, creating, destroying windows
and, the kicker is.....
how you exit WndProc makes a big difference
you have to allow many WM_COMMAND messages to exit via DefFrameProc so it can maintain the menu
Quote from: jj2007 on May 22, 2014, 04:35:40 AM
Quote from: dedndave on May 22, 2014, 03:49:19 AMEBP is now free to use
That's correct, but what for? That's just a boring message loop, not an ultrahighspeedfancyinnermostloopthatneedsallmyregisters ;-)
it's not just a free register
it's also the absence of a bunch of unnecessary stack frame code
stuff that doesn't glare at you when you examine the source code
when you look at a disassembled program, it looks idiotic, in my opinion
one of the things that makes me laugh is hInstance
how many copies of that thing you gonna need ? :lol:
you've got hInstance, a global that's initialized in the startup code
you've got hInst, the argument for WndProc
you've got wc.hInstance, the WNDCLASSEX member (often, on the stack frame)
i even see some guys create another local and
push hInst
pop hInstance
wtf ?????
i get it one time, store it in an uninitialized global WNDCLASSEX structure
and, when i want it, it's always there, in wc.hInstance
Quote from: dedndave on May 22, 2014, 04:51:06 AM
i even see some guys create another local and
push hInst
pop hInstance
wtf ?????
i get it one time, store it in an uninitialized global WNDCLASSEX structure
and, when i want it, it's always there, in wc.hInstance
If you have many uses of hInstance, the local version saves some bytes.
I tend to use a global in the uninitialised data section and for local code this works fine but there are enough instances where you are writing library modules where its easier to get it locally than require another parameter for the procedure. This is more like "object" creation than a simple library module but it has its place where you want a working "black box" without having to make too much effort to get it going.
Whats wrong wth a signed compare ? 0 or lower.
MsgLoop proc
LOCAL msg:MSG
push ebx
lea ebx, msg
jmp getmsg
msgloop:
invoke TranslateMessage, ebx
invoke DispatchMessage, ebx
getmsg:
invoke GetMessage,ebx,0,0,0
cmp eax, 0
jg msgloop
pop ebx
ret
MsgLoop endp
Quote from: hutch-- on May 22, 2014, 07:44:46 PM
Whats wrong wth a signed compare ? 0 or lower.
Theoretically, -2 could be a valid message. Documentation is scarce, so for my part, I'll stick with Dave's wasteful
inc eax/shr eax, 1/.Break .if Zero? solution - which happens to need exactly the same number of bytes, since cmp eax, 0 is a looong instruction ;-)
:biggrin:
MsgLoop proc
.data?
msg MSG <>
.code
push ebx
push esi
mov ebx, OFFSET msg
xor esi, esi
jmp getmsg
msgloop:
invoke TranslateMessage, ebx
invoke DispatchMessage, ebx
getmsg:
invoke GetMessage,ebx,esi,esi,esi
cmp eax, esi
jg msgloop
pop esi
pop ebx
ret
MsgLoop endp
no need to preserve EBX and ESI in most WinMain situations
but, the idea is sound
according to Alex, the "success" condition is always positive, so that might be a good method
on second thought....
the documentation says
exit = 0
error = -1
while it may be true (now) that success is always positive,
that doesn't mean they may not change it in future OS versions
mLoop1: INVOKE GetMessage,esi,edi,edi,edi
inc eax
jnz mLoop2
INVOKE MessageBox,0,offset szText,offset szTitle,MB_OK
;IDOK = 1
;but, if they close the MB...
mov al,1 ;could just as well JMP SHORT to exit
mLoop2: dec eax
jnz mLoop0
;------------------------------
;exit program
INVOKE ExitProcess,[esi].MSG.wParam
it could be offered as an exit or continue option
although, as Hutch said - if it's bad, fix it - lol
i guess our pointer and handle are always valid values
You guys worry me sometimes.
The message-loop has at least 3 API calls (more if you're handling accelerators, dialogs, etc.), none of which are super-fast; the overhead from condition testing is comparatively tiny. No matter how slick you make your tests, you're still going to be calling those big old messaging functions.
An additional free register in the message-loop isn't a concern because it's not the place for additional processing.
Just write it correctly and then forget about it.
On the subject of local Vs global variables: globals are for things that will be accessed from multiple places, locals are only for the procedure they're in. So the MSG structure is local to the message-loop because that's the only place it should ever be used. "hInstance" tends to be global because it's used from multiple places.
I think we have some SERIOUS OVERKILL now on this issue,...
Ha, Ha, Ha,...you guys are hilarious !!! :icon_eek:
...As I mentioned along time ago,...in a MASM Forum thread far, far away,...
This is what happens when you ask a bunch of geniuses a simple question,...:biggrin:
Tedd is right of course but HAY, you would not want to miss out on a really serious optimisation. :biggrin:
it never hurts to be more efficient
you write enough efficient code - it becomes habit
...I'm the GODZILLA of code bloat,... 8)
Increasing efficiency is fine, as long as it still works correctly. The problem is when it comes at the cost of functionality because "that's not important" or "that most likely probably maybe won't happen, hopefully."
Software is used for its functionality, so that should be the top priority; if you can also make it faster, that's great too.
I work on a barbarian view that building in error handling only produces more errors. One of the great joys of MASM is when you get it wrong it explodes in your face and usually the OS says less than nice things about your software. It is in fact no big deal to test a window handle or WNDCLASSEX structure to see if you get a valid return value and by doing so you don't have to waste your time and code handling errors that will never happen.
test eax, eax
jnz msgloop
Is all you need for correctly written code so why add garbage that will never be used ?