The MASM Forum Archive 2004 to 2012

General Forums => The Workshop => Topic started by: MazeGen on September 22, 2005, 02:37:15 PM

Title: How to generate JA/JNA jump using .IF syntax?
Post by: MazeGen on September 22, 2005, 02:37:15 PM
Hi all,
I often use .IF-.ELSEIF-.ELSE-.ENDIF blocks, but it seems that this syntax can't generate JA/JNA jumps.
Let's consider JA, Jump if Above, as an example.
We can't use something like .IF !ABOVE?, we can use just flag names (ZERO?, CARRY?, OVERFLOW?, SIGN?, and PARITY?) as an operand. Jump if Above is the same like Jump if Not Below nor Equal (JNBE). If we try to use this condition - .IF !CARRY? || !ZERO? - MASM seems to be unable to recognize that only one jump is necessary and generates two jumps, JAE and JE, what is buggy:


.686
.MODEL FLAT, STDCALL
.LISTALL
.LISTMACROALL

.CODE
Start:

.IF !CARRY? || !ZERO?
   mov eax,ebx
.ENDIF

END Start



.IF !CARRY? || !ZERO?
00000000  73 02    *     jae    @C0002
00000002  74 02    *     je     @C0001
00000004    *@C0002:
00000004  8B C3    mov eax,ebx
.ENDIF
00000006    *@C0001:


BTW, condition !(CARRY? && ZERO?) generates the same code.

So, is there any trick how to force this syntax to generate JA/JNA jumps?
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: Tedd on September 22, 2005, 04:33:14 PM
I don't think you can do it in the same way as, say, ZERO?
But you could use ".IF (eax < 4)" for example - however, I have a feeling that might produce JGE rather than JAE (signed, rather than unsigned.) But I'm pretty sure there is a way to specifically request unsigned (just don't ask me how :P)
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: MazeGen on September 22, 2005, 04:41:28 PM
Quote from: Tedd on September 22, 2005, 04:33:14 PM
I don't think you can do it in the same way as, say, ZERO?

I hope some coder knows how to do that...

Quote from: Tedd on September 22, 2005, 04:33:14 PM
But you could use ".IF (eax < 4)" for example -

That's the point. I have a macro for catching internal errors in my program and the error is often signalized only by flag(s), so I can't use such operand in general.

Quote from: Tedd on September 22, 2005, 04:33:14 PM
however, I have a feeling that might produce JGE rather than JAE (signed, rather than unsigned.) But I'm pretty sure there is a way to specifically request unsigned (just don't ask me how :P)

AFAIK .IF uses unsigned compare by default. You can override it, for instance, by using SDWORD PTR eax < 4.
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: Tedd on September 23, 2005, 10:54:25 AM
The only supported flag tests are:
CARRY?        Carry bit set
OVERFLOW?     Overflow bit set
PARITY?       Parity bit set
SIGN?         Sign bit set
ZERO?         Zero bit set


And if you try to combine two conditions, it will produce two tests, rather than the single one you want.
So it looks like you're stuck with typing them out yourself.
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: Ratch on September 23, 2005, 02:03:35 PM
MazeGen,

Quote
I often use .IF-.ELSEIF-.ELSE-.ENDIF blocks, but it seems that this syntax can't generate JA/JNA jumps.
Let's consider JA, Jump if Above, as an example.

     Who says that it can't?


                                         .IF EAX <= 3
00000000  83 F8 03    *                     cmp    eax, 003h
00000003  77 04       *                     ja     @C0001
00000005  90                                NOP
00000006  90                                NOP


Quote
We can't use something like .IF !ABOVE?, we can use just flag names (ZERO?, CARRY?, OVERFLOW?, SIGN?, and PARITY?) as an operand. Jump if Above is the same like Jump if Not Below nor Equal (JNBE). If we try to use this condition - .IF !CARRY? || !ZERO? - MASM seems to be unable to recognize that only one jump is necessary and generates two jumps, JAE and JE, what is buggy:

     The code MASM generates for IF !CARRY? || !ZERO?  is not buggy.  It is necessary.  The CF and the ZF are two independent flags, and JAE does not include a test for zero.

Quote
BTW, condition !(CARRY? && ZERO?) generates the same code.

     It should, they are logically equivalent.  See DeMorgan's Rule.

Quote
So, is there any trick how to force this syntax to generate JA/JNA jumps?

     No tricks necessary.  Just code what you want and let MASM take care of the rest.

     If you want a signed comparision, do this.  Ratch


.ELSEIF SDWORD PTR EAX < 3
00000010  EB 23    *     jmp    @C0006
00000012    *@C0004:
00000012  83 F8 03    *     cmp    sdword ptr eax, 003h
00000015  7D 04    *     jge    @C0007
00000017  90    NOP
00000018  90    NOP

Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: Tedd on September 23, 2005, 02:30:22 PM
Ratch, MazeGen's point is that he wants to test for these conditions without necessarily using ".IF (eax <= 3)" but to simply test the flags (having been set previously in the program; possibly even in a separate function) and so would prefer something like ".IF (ABOVE?)"

Quote from: Ratch on September 23, 2005, 02:03:35 PM
The CF and the ZF are two independent flags, and JAE does not include a test for zero.
JA         CF=0 and ZF=0
JAE        CF=0

..obviously MG was refering to JA with the test including zf :wink



The high-level syntax was designed for (and by?) high-level programmers -- who have little reason to test crazy things like that ::)
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: MazeGen on September 23, 2005, 02:37:31 PM
Quote from: Tedd on September 23, 2005, 02:30:22 PM
Ratch, MazeGen's point is that he wants to test for these conditions without necessarily using ".IF (eax <= 3)" but to simply test the flags (having been set previously in the program; possibly even in a separate function) and so would prefer something like ".IF (ABOVE?)"

Quote from: Ratch on September 23, 2005, 02:03:35 PM
The CF and the ZF are two independent flags, and JAE does not include a test for zero.
JA         CF=0 and ZF=0
JAE        CF=0

..obviously MG was refering to JA with the test including zf :wink

Yeah, you got it.

Quote from: Tedd on September 23, 2005, 02:30:22 PM
The high-level syntax was designed for (and by?) high-level programmers

I've never used anything else than assembly language. I use these syntax because it seems to be more readable for me.
Please don't start a flamewar whether is it really more readable or not, it is just my opinion.

Quote from: Tedd on September 23, 2005, 02:30:22 PM
-- who have little reason to test crazy things like that ::)

:green2

Me, for example :green
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: MichaelW on September 23, 2005, 09:35:24 PM
If it's just for convenience and code readability you could use a macro, like this for example:

; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
    include \masm32\include\masm32rt.inc

    ABOVE macro
        LOCAL rval
        .data
          rval dd 0
        .code
        jna   @F
        mov   rval, 1
      @@:
        EXITM <rval>
    endm
    ABOVE? EQU ABOVE()

; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
    .data
    .code
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
start:
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
    mov   ebx, 1

    cmp   ebx, 1
    .IF ABOVE?
        print "above",13,10
    .ELSE
        print "not above",13,10
    .ENDIF

    cmp   ebx, 0
    .IF ABOVE?
        print "above",13,10
    .ELSE
        print "not above",13,10
    .ENDIF

    mov   eax, input(13,10,"Press enter to exit...")
    exit
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
end start


Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: Ratch on September 23, 2005, 09:41:25 PM
MazeGen,
    Stupid me.  Here I thought you were pointing out a bug in MASM.  My contention is that MASM generates correct code whether you do the comparisons directly, or use flags.  Notice in the snippet below that MASM generates correct code in both cases, but the JA instruction checks BOTH the CF & ZF at the same time, so is therefore shorter.  Ratch


.IF EAX <= 3
00000000  83 F8 03   *    cmp    eax, 003h
00000003  77 02   *    ja     @C0001
00000005  90   NOP
00000006  90   NOP
.ENDIF
00000007   *@C0001:

00000007  83 F8 03 CMP EAX,3
.IF !(!CARRY? && !ZERO?)  ;the logic function has to be inverted to fall through
0000000A  72 02   *    jb     @C0004
0000000C  75 02   *    jne    @C0003
0000000E   *@C0004:
0000000E  90   NOP
0000000F  90   NOP
.ENDIF
00000010   *@C0003:



     Question to moderator.  Why does the (code)(/code) sequence mess up the display of a MASM listing so badly?

     Here is the code without assembly.

.IF EAX <= 3
   NOP
   NOP
.ENDIF

CMP EAX,3
.IF !(!CARRY? && !ZERO?)  ;the logic function has to be inverted to fall through
   NOP
   NOP
.ENDIF
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: Ratch on September 25, 2005, 11:20:08 PM
MazeGen,
     You're welcome.  Ratch
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: sluggy on September 26, 2005, 01:22:30 PM
Quote from: Ratch on September 23, 2005, 09:41:25 PM
     Question to moderator.  Why does the (code)(/code) sequence mess up the display of a MASM listing so badly?


It is because you have a mixture of spaces and tabs in your source that you copied & pasted, and you are pasting it into a very simple edit window that doesn't always handle it nicely.
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: MazeGen on September 26, 2005, 01:32:06 PM
Michael,
your solution requires too much overhead from my point of view. Thanks anyway.
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: MazeGen on September 26, 2005, 01:40:01 PM
Ratch,
thanks for your elaboration. You have figured out how to simulate something like !ABOVE?, very interesting :8)

The condition !(!CARRY? && !ZERO?) is not much clear, I try to use some macro or something instead.

Quote from: Ratch on September 25, 2005, 11:20:08 PM
MazeGen,
     You're welcome.  Ratch

Sorry, I'm not native English speaker. What it means in this context here?
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: Ratch on September 26, 2005, 11:53:37 PM
MazeGen,
     It would be a wise and good thing to forget about using the flags to determine the relationship of two operands, except for simple relationships  That just obfuscates and often lengthens your code.  For instance, if you want to determine A > B, then you need to determine if ZF = 1 and SF != OF .  Since there is no single instruction that compares SF and OF except JL and JGE, why not make it easier on yourself and anyone else who reads your code by just using the relational jump in the first place.

Quote
The condition !(!CARRY? && !ZERO?) is not much clear, I try to use some macro or something instead.

     See what I mean?

Quote
Sorry, I'm not native English speaker. What it means in this context here?

     Forget it, it's not important.  Ratch
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: Farabi on September 29, 2005, 03:33:07 PM
Hai. Jz and je is all the same. So do jg or ja, so it cannot be happen.
(Am I wrong remember the opcode?)
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: Ratch on September 29, 2005, 05:16:54 PM
Farabi,
     ????? Ratch
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: QvasiModo on September 29, 2005, 06:37:04 PM
Quote from: Farabi on September 29, 2005, 03:33:07 PM
Hai. Jz and je is all the same. So do jg or ja, so it cannot be happen.
(Am I wrong remember the opcode?)

JZ and JE are the same, but JG and JA are not. JG is for signed comparisons and JA for unsigned. Of course the opcodes are also different.
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: Farabi on September 30, 2005, 07:46:28 AM
Quote from: QvasiModo on September 29, 2005, 06:37:04 PM
Quote from: Farabi on September 29, 2005, 03:33:07 PM
Hai. Jz and je is all the same. So do jg or ja, so it cannot be happen.
(Am I wrong remember the opcode?)

JZ and JE are the same, but JG and JA are not. JG is for signed comparisons and JA for unsigned. Of course the opcodes are also different.

Thanks. Yes, you are right. I have forget it.
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: sonthakit on August 06, 2006, 08:01:04 AM
I agree that Developer of MASM32 should improve the code include above? and other jump condition
Because I have same problem when I write

sub eax,ebx
jbe XXX

I can not use .if directive to generate that code. MASM cannot make that code.
And I did not agree to use

sub eax,ebx
.if eax <= 0
because it generate the code

sub eax,ebx
or eax,eax
jbe XXX

I concern about my program speed very much in the nano-second. I did not want any unnecessary code in my program.
Developer! if you read my post. Please include this comment in new version of MASM32

Thank
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: hutch-- on August 06, 2006, 08:27:59 AM
This code,


    sub eax,ebx
    .if eax <= 0                ; unsigned comparison
    nop
    .endif

    nop
    nop
    nop

    sub eax,ebx
    .if SDWORD PTR eax <= 0     ; signed comparison
    nop
    .endif

    ret


generates this disassembly.


00401025                    fn_00401025:                ; Xref 00401000
00401025 2BC3                   sub     eax,ebx
00401027 83F800                 cmp     eax,0
0040102A 7701                   ja      loc_0040102D
0040102C 90                     nop
0040102D                    loc_0040102D:               ; Xref 0040102A
0040102D 90                     nop
0040102E 90                     nop
0040102F 90                     nop
00401030 2BC3                   sub     eax,ebx
00401032 83F800                 cmp     eax,0
00401035 7F01                   jg      loc_00401038
00401037 90                     nop
00401038                    loc_00401038:               ; Xref 00401035
00401038 C3                     ret
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: sonthakit on August 08, 2006, 09:30:45 AM
Sorry. I write the code wrong.

It is not or eax,eax but cmp eax,0

But my point is := I is not necessary to code cmp eax,0
because sub eax,ebx always set the EFLAGS
and you can jbe immediately.
How I can use .if without code cmp eax,0
I think it is not possible this time.
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: hutch-- on August 08, 2006, 10:42:31 AM
hmmmm,

Quote
I concern about my program speed very much in the nano-second. I did not want any unnecessary code in my program.
Developer! if you read my post. Please include this comment in new version of MASM32

If you ae serious about speed, write your own code manually, the block .IF syntax is for less critical high level parts of code.
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: sonthakit on August 10, 2006, 09:33:26 AM
This MASM32 version has command .if zero?
So. I suggest add the command

.if above?
.if !above?

next version of MASM.
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: hutch-- on August 10, 2006, 10:14:27 AM
If you have suggestions about how to improve MASM, send them to Microsoft as they wrote MASM.
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: mnemonic on August 10, 2006, 10:15:02 AM
sonthakit,

AFAIK there are no developers of MASM (ml.exe) registered in this forum.
If you want any features to be added to MASM you will have to contact Microsoft.

The MASM32 project has different aims, e.g. providing you include files and helper macros.

Don't mix that up.  :wink

EDIT: Too slow...  :bg
Title: Re: How to generate JA/JNA jump using .IF syntax?
Post by: MazeGen on August 10, 2006, 10:19:59 AM
I thing this topic goes completely wrong way. I'm locking it.