The MASM Forum Archive 2004 to 2012

General Forums => The Workshop => Topic started by: 2-Bit Chip on September 03, 2009, 12:48:55 AM

Title: MASM: Why must you be so difficult?
Post by: 2-Bit Chip on September 03, 2009, 12:48:55 AM
How would this piece of code not work? What I am attempting to do is allocate 8 bytes for two DWORDS (the POINT structure) and place values into the structure.

The return value is the address for the new POINT structure.

POINT STRUCT
    X DWORD ?
    Y DWORD ?
POINT ENDS


...


NewPoint proc X:DWORD, Y:DWORD
    mov eax, alloc(SIZEOF POINT)
    mov [eax], DWORD PTR X
    mov [eax + 4], DWORD PTR Y
    ret
NewPoint endp






Would the below work? Wouldn't the local variable be destroyed right after the procedure ends?

NewPoint proc X:DWORD, Y:DWORD
    LOCAL lPoint: POINT
    mov [lPoint], DWORD PTR X
    mov [lPoint + 4], DWORD PTR Y
    ret
NewPoint endp
Title: Re: MASM: Why must you be so difficult?
Post by: hutch-- on September 03, 2009, 12:54:48 AM
Looks like you are trying to MOV memory to memory and there is no opcode to do that. Move it to a register first then to the memory operand you need it in.
Title: Re: MASM: Why must you be so difficult?
Post by: 2-Bit Chip on September 03, 2009, 12:57:46 AM
HOLY MOLY! I forgot ALL about that! Thank you very, very, very much Hutch!!  :cheekygreen:

You DA MAN!
Title: Re: MASM: Why must you be so difficult?
Post by: dedndave on September 03, 2009, 12:59:57 AM
it could be the use (or misuse) of the alloc macro
it returns a value, already in eax

        mov     hMem,alloc(nBytes)

moves the address (or handle, if you prefer) into hMem, but it moves it from eax
you might try this...

        alloc(nBytes)
        mov dword ptr [eax],X
        mov dword ptr [eax+4],Y
        ret
Title: Re: MASM: Why must you be so difficult?
Post by: 2-Bit Chip on September 03, 2009, 01:13:50 AM
Dave,

error A2070: invalid instruction operands
error A2070: invalid instruction operands


alloc(nBytes)

That single line would not work since the macro returns.
Title: Re: MASM: Why must you be so difficult?
Post by: dedndave on September 03, 2009, 01:29:26 AM
ok - then store it into memory, then grab it back out - lol
i wonder if this would work...

        push    alloc(nBytes)
        pop     eax

it is kind of similar to str$() or something - it passes the result back in eax
Title: Re: MASM: Why must you be so difficult?
Post by: MichaelW on September 03, 2009, 01:30:59 AM
2-bit,

Did you get what the first problem is? Stated another way, the problem is that [eax] and [eax+4] are memory operands (obviously), and (less obviously) X and Y are memory operands (effectively [ebp+8] and [ebp+0Ch]).
Title: Re: MASM: Why must you be so difficult?
Post by: dedndave on September 03, 2009, 01:32:59 AM
oh ! - lol - i didn't notice they were locals
that is why i use "[ebp+nnnn]" - i know what i am working with
the alloc may work as i suggested, then - nope
the push works
Title: Re: MASM: Why must you be so difficult?
Post by: dedndave on September 03, 2009, 01:40:45 AM

NewPoint proc X:DWORD, Y:DWORD
        push    alloc(SIZEOF POINT)
        pop     eax
        push    X
        pop     [eax]
        push    Y
        pop     [eax+4]
        ret
NewPoint endp
Title: Re: MASM: Why must you be so difficult?
Post by: 2-Bit Chip on September 03, 2009, 01:54:16 AM
Thank you Michael and Dave.

Michael,

I thought of the [eax] and [eax+4] as register operands. That is probably why I messed up. :(
Title: Re: MASM: Why must you be so difficult?
Post by: dedndave on September 03, 2009, 02:49:58 AM
i was thinking - this may be a better application for a macro, rather than a proc
you are introducing a lot of unnesseccary overhead with the proc

NewPoint MACRO X:Req,Y:Req
        push    alloc(SIZEOF POINT)
        pop     eax
        mov     [eax],X
        mov     [eax+4],Y
        ENDM

so long as X and Y aren't memory arguments
Title: Re: MASM: Why must you be so difficult?
Post by: Ghandi on September 03, 2009, 01:22:34 PM
Dedndave, 'alloc' is a macro in itself which returns the result in EAX, so you could eliminate the PUSH and POP in your macro:


NewPoint MACRO X:Req,Y:Req
        alloc(SIZEOF POINT)
        mov     [eax],X
        mov     [eax+4],Y
        ENDM


HR,
Ghandi
Title: Re: MASM: Why must you be so difficult?
Post by: dedndave on September 03, 2009, 01:32:58 PM
i tried that, Ghandi
the alloc macro is similar to the str$() (and a few others)
it only works if it is used in a context where it could be replaced with "eax"
if you try to assemble:
        alloc(nBytes)
the assembler sees:
        eax
and spits out an error
on the other hand, if you use push:
        push    alloc(nBytes)
the assembler sees:
        push    eax
and everything is ok
Title: Re: MASM: Why must you be so difficult?
Post by: hutch-- on September 03, 2009, 02:17:08 PM
The alloc, free macros are incredibly simple.


      alloc MACRO bytecount
        invoke GlobalAlloc,GMEM_FIXED or GMEM_ZEROINIT,bytecount
        EXITM <eax>
      ENDM

      free MACRO hmemory
        invoke GlobalFree,hmemory
      ENDM


It is purely a MASM notation issue that you MUST assign the return value to something, usually a memory operand that is used as a 32 bit pointer to the address of the allocated memory.


mov pMem, alloc(bytecount)

is seen as,
invoke GlobalAlloc,GMEM_FIXED or GMEM_ZEROINIT,bytecount
mov "your_target", eax


If you just wanted the return value in EAX you would not use the EXITM directive in the macro.

Thjis is the CHM help file content.

alloc

mov hMem, alloc(byte_count)

Description
Allocate a block of fixed memory.

Parameters
1. byte_count The number of bytes of fixed memory to allocate.

Return Value
The starting address of the block of memory.

Comments
Memory allocate with the alloc macro should be freed with the free macro.

Example
mov hMem, alloc(16384)
Title: Re: MASM: Why must you be so difficult?
Post by: dedndave on September 03, 2009, 02:29:04 PM
also - allocating memory in 8 byte chunks is kind of inefficient
it depends on the application, of course, but i might look for a better alternative altogether
you are storing 2 dwords in allocated memory - now, you have to store that dword address
lose-lose situation
you'd be better off to save the 2 dwords
all the code it takes to do it the other way is more than 2 bytes
Title: Re: MASM: Why must you be so difficult?
Post by: Ghandi on September 03, 2009, 04:25:39 PM
Quote
i tried that, Ghandi
the alloc macro is similar to the str$() (and a few others)
it only works if it is used in a context where it could be replaced with "eax"

Ah, my bad. Sorry Dedndave.

HR,
Ghandi
Title: Re: MASM: Why must you be so difficult?
Post by: dedndave on September 03, 2009, 04:44:14 PM
lol - no prob Ghandi
i tried it, too   :bg

a much better approach might be to:
1) keep an index into the allocated block for adding new points: NextNewPoint dd ?
2) allocate enough memory for all the points to be stored
3) store the base address (or "handle") for that allocation in: hNewPoints dd ?
also, initialize NextNewPoints with the same address as hNewPoints

then, use a macro (or a proc) like this:

NewPoint MACRO X:Req,Y:Req
        push    edx
        mov     edx,NextNewPoint
        mov     [edx],X
        mov     [edx+4],Y
        add     edx,8
        mov     NextNewPoint,edx
        pop     edx
        ENDM
Title: Re: MASM: Why must you be so difficult?
Post by: 2-Bit Chip on September 03, 2009, 10:23:26 PM
Dave,

The only method for saving that I have seen would be:

MakePoint proc X:DWORD, Y:DWORD
     LOCAL p: POINT


Or I would see:

    db 0,0,0,0,0

Assigning values without a name.
Title: Re: MASM: Why must you be so difficult?
Post by: dedndave on September 03, 2009, 11:24:16 PM
well, we know nothing about how these points are created to begin with, nor how they are used - lol
perhaps you could give us some insight as to the application
if you can reasonably predict how many points you will be creating
or, at least, a starting point, you can allocate the block of memory all at once, and enlarge it later if needed
the points do not appear to require names
but, you can find a specific point, according to the order they are created, by starting at the base of the allocated block
if they are arbitrary points of, say, a circle as it is generated, you may not need to specifically address them one-by-one
the method you are currently using must generate a lot of allocated "handles" - what are you doing with all of those ?