The MASM Forum Archive 2004 to 2012

General Forums => The Campus => Topic started by: ragdog on May 01, 2011, 03:34:24 PM

Title: SendInput
Post by: ragdog on May 01, 2011, 03:34:24 PM
Hi

I try to coding with SendInput now must i fill this struct

By compile get i this error


  INPUT struct
wVk    WORD    ?
wScan    WORD    ?
dwFlags    DWORD    ?
time    DWORD    ?
dwExtraInfo    DWORD    ?
    INPUT ends 
.data
inputx INPUT <>

.code
mov inputx.type , 1
mov inputx.wVk ,VK_CONTROL; // STRG                         <<<<<<<<<<error A2081: missing operand after unary operator
mov inputx.wScan , 0;
mov inputx.dwFlags , 0; // KEY_DOWN + VKCODE

mov inputx.type , 1
mov inputx.wVk , 0;                                                    <<<<<<<<<<<error A2081: missing operand after unary operator
mov inputx.wScan , 0x2E; // 'C'
mov inputx.dwFlags, KEYEVENTF_SCANCODE; // KEY_DOWN + SCANCODE <<<error A2206: missing operator in expression



Title: Re: SendInput
Post by: dedndave on May 01, 2011, 03:43:42 PM
not sure "INPUT" is a good name for the structure type definition  :P
INPUTSTRUC struct
  wVk         WORD  ?
  wScan       WORD  ?
  dwFlags     DWORD ?
  time        DWORD ?
  dwExtraInfo DWORD ?
INPUTSTRUC ends

.data
inputx INPUTSTRUC <>
Title: Re: SendInput
Post by: Tedd on May 01, 2011, 04:00:31 PM
It would help to define it correctly.. (you missed off 'type')


MOUSEINPUT struct
    _dx         LONG    ?
    _dy         LONG    ?
    mouseData   DWORD   ?
    dwFlags     DWORD   ?
    time        DWORD   ?
    dwExtraInfo PULONG  ?
MOUSEINPUT ends

KEYBDINPUT struct
    wVk         WORD    ?
    wScan       WORD    ?
    dwFlags     DWORD   ?
    time        DWORD   ?
    dwExtraInfo PULONG  ?
KEYBDINPUT ends

HARDWAREINPUT struct
    uMsg        DWORD   ?
    wParamL     WORD    ?
    wParamH     WORD    ?
HARDWAREINPUT ends

INPUT struct
    dwType      DWORD   ?
    UNION
        mi      MOUSEINPUT <>
        ki      KEYBDINPUT <>
        hi      HARDWAREINPUT <>
    ENDS
INPUT ends

;constants for dwType
INPUT_MOUSE     equ 0
INPUT_KEYBOARD  equ 1
INPUT_HARDWARE  equ 2

Title: Re: SendInput
Post by: dedndave on May 01, 2011, 04:12:14 PM
so - INPUT is a good name   :bg
surprising it isn't defined in windows.inc
still, it does not seem like that would cause the "unary operator" error he is getting
it shouldn't crash until the INVOKE is executed - lol
Title: Re: SendInput
Post by: qWord on May 01, 2011, 04:17:15 PM
Quotemov inputx.wScan , 0x2E;
this syntax is not supported by masm.
-> 2Eh
Title: Re: SendInput
Post by: Tedd on May 01, 2011, 04:23:20 PM
Quote from: dedndave on May 01, 2011, 04:12:14 PM
still, it does not seem like that would cause the "unary operator" error he is getting
It's a misleading parser error - the error actually comes from the line before, with an unknown symbol name ('type').

There are a few other problems, but that's the fun of blindly copy-and-pasting code :P
Title: Re: SendInput
Post by: ragdog on May 01, 2011, 04:43:07 PM
Thanks Tedd for this structur

Now can i comile this

But why can i not send a Key Combo (CTRL+G)?


invoke SetFocus,hMain
mov inputx.dwType ,INPUT_KEYBOARD
mov inputx.ki.wVk,VK_CONTROL
mov inputx.ki.wScan , 0;
mov inputx.ki.dwFlags , 0; // KEY_DOWN + VKCODE
;
mov inputx.dwType , INPUT_KEYBOARD;
mov inputx.ki.wVk , 0;
mov inputx.ki.wScan , 047h; // 'G'
mov inputx.ki.dwFlags , KEYEVENTF_SCANCODE; // KEY_DOWN + SCANCODE
invoke SendInput,2, addr inputx, sizeof INPUT; // Sende

;// KEY_UP
mov inputx.ki.dwFlags , KEYEVENTF_KEYUP; // KEYUP + VKCODE
mov inputx.ki.dwFlags , KEYEVENTF_SCANCODE + KEYEVENTF_KEYUP; // KEYUP + SCANCODE

invoke SendInput,2, addr inputx, sizeof INPUT; // Send
Title: Re: SendInput
Post by: MichaelW on May 01, 2011, 05:01:31 PM
You need an array of INPUT structures with one element for each input event.

This source uses a slightly different set of structures. It simply toggles the NumLock key each time it is run (build as a console app).

;=========================================================================
    include \masm32\include\masm32rt.inc
;=========================================================================

printf MACRO format:REQ, args:VARARG
    IFNB <args>
        invoke crt_printf, cfm$(format), args
    ELSE
        invoke crt_printf, cfm$(format)
    ENDIF
    EXITM <>
ENDM

;=========================================================================

    MOUSEINPUT STRUCT
      d_x         DWORD ?
      d_y         DWORD ?
      mouseData   DWORD ?
      dwFlags     DWORD ?
      time        DWORD ?
      dwExtraInfo DWORD ?
    MOUSEINPUT ENDS

    KEYBDINPUT STRUCT
      wVk         WORD ?
      wScan       WORD ?
      dwFlags     DWORD ?
      time        DWORD ?
      dwExtraInfo DWORD ?
    KEYBDINPUT ENDS

    INPUT STRUCT
      dwType      DWORD ?
      UNION
        mi        MOUSEINPUT <>
        ki        KEYBDINPUT <>
      ENDS
    INPUT ENDS

;=========================================================================
    .data

        ;----------------------------------------
        ; Define an array of 2 INPUT structures.
        ;----------------------------------------

        inp INPUT 2 dup(<>)

    .code
;=========================================================================
start:
;=========================================================================

    ;-------------------------------------------------------------
    ; We need the size of the INPUT structure to index the array.
    ;-------------------------------------------------------------

    N = SIZEOF INPUT
    printf( "%d\n", N )

    ;-----------------------------------------------------
    ; The array needs to specify 2 events, a press of the
    ; NumLock key and release of the NumLock key.
    ;-----------------------------------------------------

    mov inp[N*0].dwType, INPUT_KEYBOARD
    mov inp[N*0].ki.wVk, VK_NUMLOCK
    mov inp[N*0].ki.dwFlags, 0
    mov inp[N*1].dwType, INPUT_KEYBOARD
    mov inp[N*1].ki.wVk, VK_NUMLOCK
    mov inp[N*1].ki.dwFlags, KEYEVENTF_KEYUP

    invoke SendInput, 2, ADDR inp, SIZEOF INPUT
    printf( "%d\n\n", eax )

    inkey "Press any key to exit..."
    exit
;=========================================================================
end start

Title: Re: SendInput
Post by: dedndave on May 01, 2011, 05:19:02 PM
you mean it won't work by using the same structure 4 times ?
1) ctrl key down
2) g key down
3) g key up
4) ctrl key up
Title: Re: SendInput
Post by: MichaelW on May 01, 2011, 05:23:24 PM
I've never tried passing a single event per call, but reusing the same structure obviously will not work for a single call passing multiple events. And I'm virtually certain that shift-key events need to be passed in the same call as the key they modify.
Title: Re: SendInput
Post by: dedndave on May 01, 2011, 05:27:13 PM
Quote...reusing the same structure obviously will not work for a single call passing multiple events

ohhhhh - i was talking about 4 calls

...i guess it's just as well to do it with an array of 4 events
but - i would use initialized data for the array
the code required to fill the structure is bigger than the structure   :P
Title: Re: SendInput
Post by: MichaelW on May 01, 2011, 11:24:15 PM
Splitting my code into two calls breaks it, so it can toggle NumLock on but not off. And then the next time I try to toggle NumLock manually I have to press and release it twice.
Title: Re: SendInput
Post by: dedndave on May 02, 2011, 01:10:54 AM
ok - now, i have a question   :P

if we have a simple nested structure, we can initialize it like so
somedata somestruct <0,0,<0,0>,0,0>

but, now we have a nested structure with a union (various sizes)
;---------------------------------

MOUSEINPUT    STRUCT    ;24 bytes
  _dx         dd ?
  _dy         dd ?
  mouseData   dd ?
  dwFlags     dd ?
  time        dd ?
  dwExtraInfo dd ?
MOUSEINPUT    ENDS

KEYBDINPUT    STRUCT    ;16 bytes
  wVk         dw ?
  wScan       dw ?
  dwFlags     dd ?
  time        dd ?
  dwExtraInfo dd ?
KEYBDINPUT    ENDS

HARDWAREINPUT STRUCT    ;8 bytes
  uMsg        dd ?
  wParamL     dw ?
  wParamH     dw ?
HARDWAREINPUT ENDS

INPUT         STRUCT    ;28 bytes
  dwType      dd ?
  UNION
    mi MOUSEINPUT    <>
    ki KEYBDINPUT    <>
    hi HARDWAREINPUT <>
  ENDS
INPUT         ENDS

;---------------------------------

how do i tell the initializer i want a keyboard struct ?
like this ?
kia INPUT <INPUT_KEYBOARD,ki<VK_TAB, 0, KEYEVENTF_KEYDOWN, 0,0>>
Title: Re: SendInput
Post by: MichaelW on May 02, 2011, 01:28:24 AM
The Microsoft documentation does not show any way to specify a type when initializing a union, but it does state that the type of the initializer must be the type of the first field. The way my structure was laid out the MOUSEINPUT structure was first in the union, so wVk was being initialized as a DWORD. To make the initialization work I had to place the KEYBDINPUT structure first in the union.

;=========================================================================
    include \masm32\include\masm32rt.inc
;=========================================================================

printf MACRO format:REQ, args:VARARG
    IFNB <args>
        invoke crt_printf, cfm$(format), args
    ELSE
        invoke crt_printf, cfm$(format)
    ENDIF
    EXITM <>
ENDM

;=========================================================================

    MOUSEINPUT STRUCT
      d_x         DWORD ?
      d_y         DWORD ?
      mouseData   DWORD ?
      dwFlags     DWORD ?
      time        DWORD ?
      dwExtraInfo DWORD ?
    MOUSEINPUT ENDS

    KEYBDINPUT STRUCT
      wVk         WORD ?
      wScan       WORD ?
      dwFlags     DWORD ?
      time        DWORD ?
      dwExtraInfo DWORD ?
    KEYBDINPUT ENDS

;    INPUT STRUCT
;      dwType      DWORD ?
;      UNION
;        mi        MOUSEINPUT <>
;        ki        KEYBDINPUT <>
;      ENDS
;    INPUT ENDS

    INPUT STRUCT
      dwType      DWORD ?
      UNION
        ki        KEYBDINPUT <>
        mi        MOUSEINPUT <>
      ENDS
    INPUT ENDS

;=========================================================================
    .data

        ;-------------------------------------------------------
        ; Define and initialize an array of 2 INPUT structures.
        ;-------------------------------------------------------

        inp   INPUT <INPUT_KEYBOARD,{<VK_NUMLOCK,0,0>}>
              INPUT <INPUT_KEYBOARD,{<VK_NUMLOCK,0,KEYEVENTF_KEYUP>}>

    .code
;=========================================================================
start:
;=========================================================================

    ;-------------------------------------------------------------
    ; We need the size of the INPUT structure to index the array.
    ;-------------------------------------------------------------

    N = SIZEOF INPUT
    printf( "%d\n", N )

    ;-----------------------------------------------------
    ; The array needs to specify 2 events, a press of the
    ; NumLock key and release of the NumLock key.
    ;-----------------------------------------------------

    ;mov inp[N*0].dwType, INPUT_KEYBOARD
    ;mov inp[N*0].ki.wVk, VK_NUMLOCK
    ;mov inp[N*0].ki.dwFlags, 0
    ;mov inp[N*1].dwType, INPUT_KEYBOARD
    ;mov inp[N*1].ki.wVk, VK_NUMLOCK
    ;mov inp[N*1].ki.dwFlags, KEYEVENTF_KEYUP

    invoke SendInput, 2, ADDR inp, SIZEOF INPUT
    printf( "%d\n\n", eax )

    inkey "Press any key to exit..."
    exit
;=========================================================================
end start


This has been an interesting exercise, but I think I'll stick with the longer initialization :lol

Title: Re: SendInput
Post by: dedndave on May 02, 2011, 01:41:17 AM
thanks Michael
seems a little sloppy on ms' part - lol
but - i can see where this could get confusing very fast
i think you are on to a solution that will work in this case
only the keyboard input (nested) structure starts with a word type
so, i am guessing that something like "word ptr VK_TAB" might work
that doesn't really solve the issue in other cases, though

i think a "brute force" approach might be worth a try in more complex cases
kia LABEL INPUT
dd INPUT_KEYBOARD
dw xxx
dw xxx
dd xxx
dd xxx
dd xxx
dd xxx
dd xxx

the use of "LABEL" might do the job   :P
that may allow you to access individual members in normal structure fashion
Title: Re: SendInput
Post by: MichaelW on May 02, 2011, 01:51:49 AM
It was worth a try, but in my code "word ptr VK_NUMLOCK" was accepted but did not correct the problem with the order of the structures in the union.

Title: Re: SendInput
Post by: dedndave on May 02, 2011, 02:16:38 AM
this assembles without error
        .NOLIST
        .XCREF
        INCLUDE \masm32\include\masm32rt.inc
        .586
        .LIST

;---------------------------------

MOUSEINPUT    STRUCT    ;24 bytes
  _dx         dd ?
  _dy         dd ?
  mouseData   dd ?
  dwFlags     dd ?
  time        dd ?
  dwExtraInfo dd ?
MOUSEINPUT    ENDS

KEYBDINPUT    STRUCT    ;16 bytes
  wVk         dw ?
  wScan       dw ?
  dwFlags     dd ?
  time        dd ?
  dwExtraInfo dd ?
KEYBDINPUT    ENDS

HARDWAREINPUT STRUCT    ;8 bytes
  uMsg        dd ?
  wParamL     dw ?
  wParamH     dw ?
HARDWAREINPUT ENDS

INPUT         STRUCT    ;28 bytes
  dwType      dd ?
  UNION
    mi MOUSEINPUT    <>
    ki KEYBDINPUT    <>
    hi HARDWAREINPUT <>
  ENDS
INPUT         ENDS

;---------------------------------

        .DATA

kia LABEL INPUT
dd INPUT_KEYBOARD                         ;dwType, control key down
dw VK_CONTROL                             ;wVk = VK_CONTROL
dw 0                                      ;wScan
dd 0                                      ;dwFlags = key down
dd 0                                      ;time
dd 0                                      ;dwExtraInfo
dd 0,0                                    ;pad

dd INPUT_KEYBOARD                         ;dwType, "g" key down
dw 0                                      ;wVk
dw 47h                                    ;wScan = "g"
dd KEYEVENTF_SCANCODE                     ;dwFlags = key down
dd 0                                      ;time
dd 0                                      ;dwExtraInfo
dd 0,0                                    ;pad

dd INPUT_KEYBOARD                         ;dwType, "g" key up
dw 0                                      ;wVk
dw 47h                                    ;wScan = "g"
dd KEYEVENTF_SCANCODE or KEYEVENTF_KEYUP  ;dwFlags = key up
dd 0                                      ;time
dd 0                                      ;dwExtraInfo
dd 0,0                                    ;pad

dd INPUT_KEYBOARD                         ;dwType, control key up
dw VK_CONTROL                             ;wVk = VK_CONTROL
dw 0                                      ;wScan
dd KEYEVENTF_KEYUP                        ;dwFlags = key up
dd 0                                      ;time
dd 0                                      ;dwExtraInfo
dd 0,0                                    ;pad

        .CODE

_main   PROC

;        INVOKE  SetFocus,hWin
        INVOKE  SendInput,4,offset kia,sizeof INPUT
        INVOKE  ExitProcess,0

_main   ENDP

        END     _main


now, to put it in a test program   :P
Title: Re: SendInput
Post by: dedndave on May 02, 2011, 02:32:42 AM
 :bg
(http://img811.imageshack.us/img811/8154/kinput.png)

if you make another structure array for extended keys, i think you can use the same two arrays to synthesize any key entry

which brings up another point...
don't write code that assumes the user can only type 200 wpm   :lol
Title: Re: SendInput
Post by: ragdog on May 02, 2011, 10:04:29 AM
Thanks @all

@dedndave

What if this for a tool os info Dump?
Title: Re: SendInput
Post by: dedndave on May 02, 2011, 10:36:12 AM
it's a project i have been working on for some time
i am updating my OS Info Dump program - here is the old one.....
http://www.masm32.com/board/index.php?topic=11963.msg90835#msg90835
that was a console app

i wanted to learn GUI programming, so this is fancier than it needs to be for OS Info   :P
i am almost ready to post the OS Info Dump 3.0 - a few things to fix
i will post the source code when i do
along the way, i wanted to learn about windows messages, so i wrote some code that will display WM messages
i just happened to have it set up like that, so it was easy to add the SendInput code for testing

i have a future project in mind that i can use the GUI text window for   :bg
Title: Re: SendInput
Post by: ragdog on May 02, 2011, 11:46:07 AM
Nice tool

I have already your old os dump
But i see you can look with your new version on windows messages :U

Ok i wait if you upload your new version

Greate Job
Title: Re: SendInput
Post by: dedndave on May 02, 2011, 03:26:13 PM
that really isn't part of the OS Info program - lol
i added that in as an include file so that i could learn about windows messages
http://www.masm32.com/board/index.php?topic=16460.msg137899#msg137899

i have a few things to fix before i post the OS Info program
but, if you want a program to view messages, Winspector is much more robust
http://www.softpedia.com/get/Security/Security-Related/Winspector.shtml
Title: Re: SendInput
Post by: dedndave on May 02, 2011, 04:02:15 PM
well - it looks like the code still needs a little work
i added WM_CHAR to the message filter
then, ran the program again - no WM_CHAR message   :red

(http://img545.imageshack.us/img545/5069/sendinp.png)

lines 1-4 represent the programatic keystroke using SendInput
lines 5-9 represent the manual keystroke sequence
i don't understand why it generates a "G" when i pressed "g" - caps lock is off   :bg
Title: Re: SendInput
Post by: dedndave on May 02, 2011, 04:33:54 PM
if the "right" way doesn't work, cheat   :bg

(http://img90.imageshack.us/img90/4767/sndmsg.png)

        .DATA

KyTable dd WM_KEYDOWN,VK_CONTROL,1D0001h
        dd WM_KEYDOWN,'G',220001h
        dd WM_CHAR,7,220001h
        dd WM_KEYUP,'G',0C0220001h
        dd WM_KEYUP,VK_CONTROL,0C01D0001h

        .CODE

        push    ebx
        push    esi
        push    edi
        mov     ebx,hWin
        push    5
        INVOKE  SetFocus,ebx
        mov     edi,offset KyTable
        pop     esi

kLoop0: INVOKE  SendMessage,ebx,[edi],[edi+4],[edi+8]
        add     edi,12
        dec     esi
        jnz     kLoop0

        pop     edi
        pop     esi
        pop     ebx


of course, that does not go through the systems' key filter
it does not generate a Bell character
you might try sending chr$(7), or.....
        INVOKE  Beep,1000,100
:bg