The MASM Forum

General => The Campus => Topic started by: Zen on May 21, 2014, 05:12:26 AM

Title: The Ultimate Message Loop
Post by: Zen on May 21, 2014, 05:12:26 AM
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.)
Title: Re: The Ultimate Message Loop
Post by: dedndave on May 21, 2014, 05:29:39 AM
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
Title: Re: The Ultimate Message Loop
Post by: Will on May 21, 2014, 10:13:11 AM
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.)
Title: Re: The Ultimate Message Loop
Post by: hutch-- on May 21, 2014, 10:19:13 AM
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.
Title: Re: The Ultimate Message Loop
Post by: jj2007 on May 21, 2014, 12:08:32 PM
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
  .Endw


This works - I've never seen it fail. However, occasionally you see

.Break .if sdword ptr eax<=0

instead, 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
  .Endw


P.S.: Dave's shr eax, 1 thing is actually one byte shorter :(
Title: Re: The Ultimate Message Loop
Post by: dedndave on May 21, 2014, 12:58:15 PM
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)
Title: Re: The Ultimate Message Loop
Post by: sinsi on May 21, 2014, 03:01:57 PM
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 Get­Message will not fail with -1.
Title: Re: The Ultimate Message Loop
Post by: MichaelW on May 21, 2014, 03:08:03 PM
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.
Title: Re: The Ultimate Message Loop
Post by: hutch-- on May 21, 2014, 03:22:37 PM
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.
Title: Re: The Ultimate Message Loop
Post by: farrier on May 22, 2014, 01:52:00 AM
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

Title: Re: The Ultimate Message Loop
Post by: dedndave on May 22, 2014, 02:18:32 AM
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
Title: Re: The Ultimate Message Loop
Post by: Gunther on May 22, 2014, 02:24:22 AM
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
Title: Re: The Ultimate Message Loop
Post by: dedndave on May 22, 2014, 03:28:04 AM
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
Title: Re: The Ultimate Message Loop
Post by: Zen on May 22, 2014, 03:35:44 AM
...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:
Title: Re: The Ultimate Message Loop
Post by: jj2007 on May 22, 2014, 03:38:50 AM
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
Title: Re: The Ultimate Message Loop
Post by: Zen on May 22, 2014, 03:48:38 AM
...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:
Title: Re: The Ultimate Message Loop
Post by: dedndave on May 22, 2014, 03:49:19 AM
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
Title: Re: The Ultimate Message Loop
Post by: dedndave on May 22, 2014, 04:17:06 AM
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
Title: Re: The Ultimate Message Loop
Post by: Gunther on May 22, 2014, 04:22:53 AM
Dave,

thank you for providing the source.  :t Works fine and should be instructive for starters.

Gunther
Title: Re: The Ultimate Message Loop
Post by: Zen on May 22, 2014, 04:28:05 AM
DAVE !!!
I have previously downloaded a number of your "greatest hits", including several varieties of MDI projects,,,THANKS,...
Title: Re: The Ultimate Message Loop
Post by: 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 ;-)
Title: Re: The Ultimate Message Loop
Post by: dedndave on May 22, 2014, 04:40:25 AM
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
Title: Re: The Ultimate Message Loop
Post by: dedndave on May 22, 2014, 04:43:38 AM
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
Title: Re: The Ultimate Message Loop
Post by: dedndave on May 22, 2014, 04:51:06 AM
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
Title: Re: The Ultimate Message Loop
Post by: jj2007 on May 22, 2014, 09:50:08 AM
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.
Title: Re: The Ultimate Message Loop
Post by: hutch-- on May 22, 2014, 12:49:17 PM
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.
Title: Re: The Ultimate Message Loop
Post by: hutch-- on May 22, 2014, 07:44:46 PM
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
Title: Re: The Ultimate Message Loop
Post by: jj2007 on May 22, 2014, 08:15:45 PM
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 ;-)
Title: Re: The Ultimate Message Loop
Post by: hutch-- on May 22, 2014, 08:29:10 PM
 :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
Title: Re: The Ultimate Message Loop
Post by: dedndave on May 22, 2014, 11:23:12 PM
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
Title: Re: The Ultimate Message Loop
Post by: dedndave on May 22, 2014, 11:43:23 PM
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
Title: Re: The Ultimate Message Loop
Post by: dedndave on May 23, 2014, 12:13:31 AM
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
Title: Re: The Ultimate Message Loop
Post by: Tedd on May 23, 2014, 02:36:47 AM
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.
Title: Re: The Ultimate Message Loop
Post by: Zen on May 23, 2014, 03:51:17 AM
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:
Title: Re: The Ultimate Message Loop
Post by: hutch-- on May 23, 2014, 04:21:55 AM
Tedd is right of course but HAY, you would not want to miss out on a really serious optimisation.  :biggrin:
Title: Re: The Ultimate Message Loop
Post by: dedndave on May 23, 2014, 04:35:44 AM
it never hurts to be more efficient
you write enough efficient code - it becomes habit
Title: Re: The Ultimate Message Loop
Post by: Zen on May 23, 2014, 05:05:02 AM
...I'm the GODZILLA of code bloat,... 8)
Title: Re: The Ultimate Message Loop
Post by: Tedd on May 23, 2014, 11:44:32 PM
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.
Title: Re: The Ultimate Message Loop
Post by: hutch-- on May 24, 2014, 10:35:10 AM
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 ?