The MASM Forum

General => The Workshop => Topic started by: jj2007 on August 04, 2014, 09:33:46 AM

Title: Half C, half stdcall
Post by: jj2007 on August 04, 2014, 09:33:46 AM
Playing with a hybrid of C and stdcall:

Switch$ MACRO src$
  ctCs$=0
  push reparg(src$)      ; three args are needed but ...
  push len([esp])        ; ... we push two of them here
  ifndef SwProc
      .if 0
            SwProc:
            push esi
            push edi
            push ecx
            mov esi, [esp+12+12]     ; switch arg
            mov ecx, [esp+12+8]      ; switch arg len
            mov edi, [esp+12+4]      ; case arg
            ; could be szCmp etc, but let's keep it simple:
            repe cmpsb      ; !Zero? means not equal
            .if Zero?
                  test byte ptr [edi], -1
            .endif
            pop ecx
            pop edi
            pop esi
            retn 4
      .endif
  endif
  xor ecx, ecx
ENDM

Case$ MACRO cmp$
  if ctCs$
      .else
  endif
  ctCs$=ctCs$+1
  push reparg(cmp$)      ; only one arg pushed here
  call SwProc
  .if Zero?
ENDM


Usage:

include \masm32\include\masm32rt.inc
include Switch$.inc
.code
start:
      mov esi, chr$("Test")
      mov edi, chr$("This is edi")

      Switch$ esi
      Case$ "Mist"
            print "found Mist"
      Case$ edi
            print "found edi"
      Case$ "Tes"
            print "found Tes"
      Case$ "Test"
            print "found Test"
      Case$ "Testx"
            print "found TestX"
      Else$
            print "found something else"
      Endsw$

      inkey chr$(13, 10, "OK")
      exit
end start
Title: Re: Half C, half stdcall
Post by: Gunther on August 05, 2014, 12:07:24 AM
Jochen,

runs fine under Windows 7-64:
Quote
found Test
OK

But the only time advantage would be the FASTCALL.

Gunther
Title: Re: Half C, half stdcall
Post by: Siekmanski on August 05, 2014, 04:56:25 AM
found Test
OK

Windows 8.1
Title: Re: Half C, half stdcall
Post by: jj2007 on August 05, 2014, 10:20:07 AM
Thanks for testing, Gunther and Marinus :icon14:

How would you do it with fastcall?
Title: Re: Half C, half stdcall
Post by: qWord on August 05, 2014, 10:44:02 AM
I  don't think that a register calling convention would speed this code, because of the limited number of available register under x86-32.
Even if it is a bit off topic, but what about creating a hash table while assembling? I've played with that in the past and found that you can get good results, but  it sometimes requires manual fine-tuning.
Title: Re: Half C, half stdcall
Post by: jj2007 on August 05, 2014, 04:13:26 PM
Quote from: qWord on August 05, 2014, 10:44:02 AMwhat about creating a hash table while assembling?

How would that work with e.g. Case$ edi ?

My basic point here was to demonstrate the "hybrid" calling convention:
- push two (or more) args in "C mode"
- repeatedly push one (or more) args for use with a stdcall proc
- which picks the two args from stack
- finally, correct the stack, as the C calling convention does

Not really revolutionary, not really faster, but it saves a bit of code because you have to push only one Case$ argument in the Case$ macro.
Title: Re: Half C, half stdcall
Post by: Gunther on August 05, 2014, 07:28:50 PM
Jochen,

Quote from: jj2007 on August 05, 2014, 10:20:07 AM
How would you do it with fastcall?

that's indeed a bit difficult in your case. FASTCALL isn't been standardized, and is implemented differently, depending on the compiler vendor. Here (http://agner.org/optimize/calling_conventions.pdf) is a good overview.

Gunther
Title: Re: Half C, half stdcall
Post by: qWord on August 06, 2014, 01:29:47 AM
Quote from: jj2007 on August 05, 2014, 04:13:26 PMHow would that work with e.g. Case$ edi ?
that won't work of course.
Title: Re: Half C, half stdcall
Post by: hutch-- on August 26, 2014, 02:37:11 AM
The problem with combining C and STDCALL is the near RET instruction (C3) will only take an immediate to balance the stack, this is why the C calling convention corrects ESP after the RET have returned.

The easiest way to pass a variable number of arguments is in a structure, one argument as the address of the structure, balance the stack with a RET 4 and access as many arguments as you like, I imagine this is why its so popular with C/C++ compilers. You can easily make a structure that has as many members as you like and you just fill in the member names you want before you make the call passing the structure address. Be cheeky and pass the structure address in EAX and you have no stack to balance at all.  :biggrin:
Title: Re: Half C, half stdcall
Post by: Gunther on August 26, 2014, 07:22:08 AM
Quote from: hutch-- on August 26, 2014, 02:37:11 AM
Be cheeky and pass the structure address in EAX and you have no stack to balance at all.  :biggrin:

That's indeed very cheeky and clever.  :t

Gunther
Title: Re: Half C, half stdcall
Post by: jj2007 on August 26, 2014, 09:20:08 AM
Quote from: hutch-- on August 26, 2014, 02:37:11 AMBe cheeky and pass the structure address in EAX and you have no stack to balance at all.  :biggrin:

I pass the address of the structure in esp:

Case$ MACRO cmp$
  ..
  call SwProc