The MASM Forum Archive 2004 to 2012

General Forums => The Workshop => Topic started by: RuiLoureiro on March 26, 2005, 02:51:14 PM

Title: Windows API32 documentation
Post by: RuiLoureiro on March 26, 2005, 02:51:14 PM
Hi all

   I have the help file Win32 Programmer's reference. I read some topics. I didn't read it deeply. However, I noticed, in some cases, it seems to be «sheets of poems». In this way, i can say «its not a good programming tool» because I think that a good programming tool is a set of good data\variable structures and a good set of procedures to lead with it, a good organization, good documentation and a good assembler with a good documentation, too.It is my particular opinion, of course. I respect other opinions.
   For example, i looked for memory management functions. I found GlobalAlloc, GlobalLock, GlobalHandle, GlobalFree, etc. To simplify, we can say that this functions can be seen in the format «output FUNCTIONNAME(input1, input2,...)» the output and inputs can have «the size of a train!»- it doesn't matter, to understand the essencial.

Thereafter [ ] means my comments and « » parts of documentation.

«A process can use the GlobalAlloc and LocalAlloc to allocate memory. In ... Win32 API, the local heap = the global heap. ... there is no difference between the memory objects allocated by these functions»

[I cut words without meaning. eg: its raining and it is wet. Rain imply wet ]

We can see(the essencial):
[In all cases: «If the function fails, the return value is NULL»]
--------------------------------------------------------------------------------
HGLOBAL GlobalHandle( pMem )

pMem = Points to the first byte of the global memory block. This pointer is returned by the GlobalLock function.

[This seems (should) to be: This pointer is returned by the GlobalLock [or GlobalAlloc if uFlags= GMEM_FIXED ] ... and  points to the first byte of the global memory block. Its is a conversion function.].

Return Values: ...is the handle of the specified global memory.

«GlobalLock: handle into a pointer [and ???]; GlobalHandle: the pointer back into a handle».
--------------------------------------------------------------------------------
LPVOID GlobalLock( hMem )
Return Values: ...is a pointer to the first byte of the memory block

«with the GMEM_FIXED flag ... the value of the returned pointer is equal to the value of the specified handle».[So, hMem = pMem if uFlags= GMEM_FIXED]

«...locks a global memory ... returns a pointer to the first byte ... The memory block ... cannot be moved or discarded»
--------------------------------------------------------------------------------
HGLOBAL GlobalAlloc( uFlags, dwBytes)

«dwBytes= number of bytes to allocate»
uFlags= GMEM_FIXED     fixed memory.
      = GMEM_MOVEABLE  moveable memory. The return value is a pointer to the
      memory block [sometimes is block, sometimes is object.]
        «This flag cannot be combined with the GMEM_FIXED» [ => obviously
         ! one thing is fixed or movable. It cannot be both, unless ... !].
      «return value is the handle».«32-bit private to the calling process»
      «Translate handle into pointer, use the GlobalLock»
      = GPTR  [ GMEM_FIXED and GMEM_ZEROINIT ]
      = GHND  [ GMEM_MOVEABLE and GMEM_ZEROINIT ]
               «GMEM_ZEROINIT = initializes memory contents to zero»

[uFlags=GMEM_FIXED =>] Allocates fixed memory. The return value is a pointer to the memory block. To access the memory, the calling process simply casts the return value to a pointer.

Return Values: ...is the handle of ... allocated memory.
--------------------------------------------------------------------------------
HGLOBAL GlobalFree( hMem )

[hMem = ...This handle is returned by either the GlobalAlloc or GlobalReAlloc]
--------------------------------------------------------------------------------
Conclusion: confused !

Has anyboby a good Win32 programmer´s reference about this issue ?
Best regards
Title: Re: Windows API32 documentation
Post by: Vortex on March 26, 2005, 05:04:54 PM
Quote
I have the help file Win32 Programmer's reference. I read some topics. I didn't read it deeply. However, I noticed, in some cases, it seems to be «sheets of poems». In this way, i can say «its not a good programming tool» because I think that a good programming tool is a set of good data\variable structures and a good set of procedures to lead with it, a good organization, good documentation and a good assembler with a good documentation

RuiLoureiro,

The architecture of windows is complicated,so you can't expect a win32 reference documenting all the functions in a simple way. There are a lot of topics related to win32 programming, you should study them step by step. Don't hesitate to post your questions, there will be peoples ready to help you.
Title: Re: Windows API32 documentation
Post by: RuiLoureiro on March 26, 2005, 08:09:17 PM
Hi
   Vortex,

   What are all functions to allocate system memmory ? What they return ?
   Can we call GlobalAlloc with uFlags = GMEM_DISCARDABLE ?

Regards
Title: Re: Windows API32 documentation
Post by: Vortex on March 26, 2005, 09:44:14 PM
Hi RuiLoureiro,

You shoudl be carefull with the flag GMEM_DISCARDABLE. Quote from win32.hlp
Quote
GMEM_DISCARDABLE   Allocates discardable memory. This flag cannot be combined with the GMEM_FIXED flag. Some Win32-based applications may ignore this flag.

As you mentioned, GlobalAlloc can be used to allocate memory. There are also the HeapAlloc and VirtualAlloc functions for memory allocation.
Title: Re: Windows API32 documentation
Post by: RuiLoureiro on March 26, 2005, 09:50:39 PM
Hi Vortex
    Whats the difference between GlobalAlloc and LocalAlloc if the heap is the same ?


Regards
Title: Re: Windows API32 documentation
Post by: Vortex on March 26, 2005, 10:06:40 PM
Hi RuiLoureiro,

Quote from Win32.hlp
Quote
The LocalAlloc function allocates the specified number of bytes from the heap. In the linear Win32 API environment, there is no difference between the local heap and the global heap.
Title: Re: Windows API32 documentation
Post by: hutch-- on March 27, 2005, 02:52:08 AM
RuiLoureiro,

You can go to the forum web site and get the reference material yourself. Just check the top right corner of the forum for the link. This means you will have the reference material on your local machine which is much faster to work with. This way when members help you out, you have the reference to use it properly.
Title: Re: Windows API32 documentation
Post by: roticv on March 27, 2005, 02:59:23 AM
f0dder's point of view would be that LocalAlloc/GlobalAlloc is bad and it is better to use HeapAlloc.  :toothy http://f0dder.schwump.net/memalloc.htm
Title: Re: Windows API32 documentation
Post by: hutch-- on March 27, 2005, 03:20:42 AM
For all of the "hoo hah" GlobalAlloc using the fixed memory flag allocates so fast you cannot get a timing on it. Any of the other flags are out of date as they represent technology that is a left over from 16 bit Windows.

GlobalAlloc comfortably handles memory allocation well over the 1 gigabyte mark and it does that just as fast so there is no general rule that excludes it in performance terms.

When you need large quantities of small allocations, it is very had to go past OLE string memory as it is designed to do exactly that from a preallocated string pool. It is designed and used by the operating system for UNICODE display so it needs to be both fast and flexible.

Windows has multiple strategies for allocating memory and this is exactly the reason why general reductionist theories are of no real use in code design, you effectively dial in what you need on the basis of the task you are going to perform, not on some second hand opinion that does not have the technical backing of the operating system.
Title: Re: Windows API32 documentation
Post by: raymond on March 27, 2005, 04:15:57 AM
RuiLoureiro

Don't despair. My first impressions when I started with Win32 was that this Reference material was written in hieroglyphs!!! :bdg After a while, you will learn to read only what is essential.

You must realize that it was written primarily for C/C++ programmers which is the base language for MS. In assembly, there is no need to "cast" a handle to a pointer. :tdown

Raymond
Title: Re: Windows API32 documentation
Post by: RuiLoureiro on March 27, 2005, 10:19:38 PM
Hi
   Thank you Vortex, Roticv and Raymond !

   A special thanks to Mr. Steeve Hutch ( --sson ) to your topic about the links on the top right corner and the information that otherwise i couldn't get easier. I am not only new in forums ( this is the first ), also in debates [and some day, i shall go to present my candidature to write something that looks like good in my english! ] It's not easy ! As my name indicate, i usually write in portuguese. When i write in the present tense something that is obvious it belongs to the past or so, you must use an appropriate translator.

   Hey, raymond, about Win32 Reference i found out! They demummify it elsewhere.
It´s the true reason! (Ah! Ah! Ah!)

   Seriously, the question about memory allocation is an intriguing one. They give information for a set of functions GlobalZZZZ and for a set LocalZZZZ. And they says: «A process can use the GlobalAlloc and LocalAlloc to allocate memory. In Win32 API, the local heap = the global heap. ... there is no difference between the memory objects allocated by these functions». So, if « there is no difference between the memory objects allocated by these functions» and «local heap = the global heap», we should conclude that the two sets of functions do the same thing ( but what is the best ?). Today, i read a lot of stuff about this issue and i didn't find out nothing relevant. So, I go to take down the functions GlobalZZZ and HeapXXX! And i go "to cook" all well.

Stay well
Regards
Title: Re: Windows API32 documentation
Post by: rea on March 28, 2005, 12:48:37 AM
By the way, that thing, about the global = local is because where not the same for 16 bit Windows, If Im not wrong.

They only provide the same functions in the win32 api for let the anterior aplications access the same  interface.. but yes, they will only have one set of names for the functions if they whant it.... by instance, would be nice look at the entry point of the global and local functions for see if the entry point is really the same or there is a repetition of code ;).... (because they have stated that is the same function ...)
Title: Re: Windows API32 documentation
Post by: sluggy on March 28, 2005, 03:06:16 AM
Rui,
while ultimately the memory does all come from the same heap, how the OS handles that memory allocation and the context is different between the various types of call. Check here (http://msdn.microsoft.com/library/default.asp?url=/library/en-us/memory/base/comparing_memory_allocation_methods.asp) and especially here (http://msdn.microsoft.com/library/default.asp?url=/library/en-us/memory/base/memory_management.asp) for a more indepth description.
Title: Re: Windows API32 documentation
Post by: RuiLoureiro on March 28, 2005, 09:12:26 PM
Hi all

Raymond,
   Yesterday, after i have putting my answer i went to the links where i discovered who is Mr. Raymond ! But i think, Mr. Raymond, you are a young because you (should)have a young spirit in spite of your age. This is very good. I have a boy in fourth year at university. He follows mathematics, may be because i was a teacher of mathematics in a high school for 6 years ( genetic problems?). Now, he gives me lessons in some matters but in programming questions he needs my helps. It's very interesting to find persons like you. I don't like to play golf and tennis. I was a football player, but not professional. But my kid likes tennis.
   ForTran is what we have in common. But my first program was about 1979 to project and analise a communication anttena (more than 1000 punched cards, rolls of output sheets). In windows programming i am sure you can teach me a lot of things i need to learn to make a good program, some day. For now i am studying what and how to do. I am in a planning phase.

«"cast" a handle to a pointer»: may be this «throw a handle into a pointer» ? It means the same as «cast fire over water» ? It is to ... laugh !
«search yourself, make yourself»

   about this little topic, i can say that all i know was constructed by me. But, as we know, in many cases, there are some undocumented details that makes the difference, and we don't know where to look for. We look for, we search and we don't find interesting or relevant things: the details we want.

Vortex,
   Your last answer don't "cast" light through the issue. I know you have many works about windows programming and i think you know more than what is written in documentation. As Mr. Hutch explained, GlobalAlloc should be used with fixed memory flag only. So, I had reason when i put my question.

Mark Larson page

   In code design, there are cases when we are interested in fast code;
   There are cases when we want the less code possible, etc.
   In my case, i am not concerned about both cases.
   I don't want to write a program which can be hard to understand, with too
   many unnecessary names. But I saw good information there and some of it
   is already my current practice.

Hi, Xor Stance, Rea, Sluggy

   Well-seen ! Can it depend on context ? Are The entry points the same ?

Stay well
Regards
Title: Re: Windows API32 documentation
Post by: tenkey on March 28, 2005, 10:30:54 PM
Quote from: RuiLoureiro on March 28, 2005, 09:12:26 PM
«"cast" a handle to a pointer»: may be this «throw a handle into a pointer» ? It means the same as «cast fire over water» ? It is to ... laugh !
«search yourself, make yourself»

The image I usually have of "casting" types is pouring plastic, plaster, or hot liquid metal into a cast, to mold it into the desired shape.
Title: Re: Windows API32 documentation
Post by: hutch-- on March 29, 2005, 02:15:30 AM
Something that is worth keeping in mind is that a "pointer" is nothing more than an address stored in a variable. If you have a procedure that has a local variable that you need the address for, you do it something like this,


    LOCAL MyVar :DWORD    ; allocate the local variable
    LOCAL MyPtr :DWORD    ; allocate a pointer for it

    lea eax, MyVar               ; load the address of the variable into the EAX register
    mov MyPtr, eax              ; store that address in another variable


With this done, you have the address available in a variable instead of having to obtain it every time you need to use it.
Title: Re: Windows API32 documentation
Post by: raymond on March 29, 2005, 04:25:35 PM
QuoteHe follows mathematics, may be because i was a teacher of mathematics in a high school for 6 years

With that background, you may be interested in a site maintained by another high school math teacher (England). Look at his "Project Euler". You may even enjoy a challenge with your son.

http://www.mathschallenge.net/

Raymond
Title: Re: Windows API32 documentation
Post by: RuiLoureiro on March 29, 2005, 08:44:23 PM
Hi

Xor Stance,
   I know DOS 16-bits asm. I have Icztutes (downloaded in Nov\2003);
I don't like C. I think, the thing i have to do can be done in console mode.
Ok, LP: stands for long pointer. There are many words\names\members\ ??? that begin with lp. But in Win32 it says nothing. All addresses are 32 bites. No ?
Thanks
-----------------------
tenkey,
   I invented those expressions above. Your image is funny. But when we use
GlobalAlloc to alloc fixed memory the handle=pointer. So, we don't need to use conversion functions and «"cast" a handle to a pointer» means ... nothing, I think. If the function as a handle as output and i need a pointer, i should prefer «convert» than «cast» ( input: handle, output: pointer). About «cast» i like the «weather forecast» ! We should play ... to go ahead. Thanks
-----------------------
Mr. Hutch—
   Good little topic.
   This is a particular question so importante. It because we are using the stack. In DOS 16-bites i never used lea instructions. All pointers were passed by registers. Before the call, i used «mov   ebx, offset XXX» ... then call YYY.
I took it down. Thanks

We could do this, too, no ?

.data
   MyVar   dd 100

.code   
   pMyVar    dd offset MyVar      ; pointer for MyVar in CODE

...
   invoke   ExecProc, pMyVar
;               ---------------------------
ExecProc   proc   pMyVar:DWORD   ; this pMyVar is local but is already pointer

      ret
ExecProc   endp
------------------------
Vortex,
        This code does nothing in my WinXP. It enter and exit. Why ?

.386
.model flat, stdcall
option casemap:none

include     \masm32\include\windows.inc
include     \masm32\include\kernel32.inc
include     \masm32\include\masm32.inc      ; StdOut, StdIn
includelib  \masm32\lib\kernel32.lib
includelib  \masm32\lib\masm32.lib

.data
msg1    db 'Please type your name',13,10,0
msg2    db 'Nice to see you ',0

.data?
buffer db 100 dup(?)

.code
start:
   invoke StdOut,ADDR msg1
   invoke StdIn,ADDR buffer,100       ; receive text input
   invoke StdOut,ADDR msg2
   invoke StdOut,ADDR buffer
   invoke ExitProcess,0
END start
--------------------------------------------
Don't  we need AllocConsole  or  CreateProcess, first ?

Does anyone know a good book about it ?

Thanks
Stay well
Title: Re: Windows API32 documentation
Post by: RuiLoureiro on April 05, 2005, 11:18:55 AM
Hi
reminder:   «The Workshop is where general purpose questions and answers are posted»

Point 1.   
               Where i before wrote «little topic» we should read «short topic».

Point 2.
                I am studying the files which came with masm32. In particular in 3
   directories: INCLUDE, LIB and M32LIB. That files have many names of
   procedures which i think it is hard to follow.

   What i am seeing is, for example, GetDC, GetCursor, GetKeyboard...
   GetMessage... GetWindow... etc. etc. So, we jump from DC to Cursor, to
   Messages, Windows, etc. etc. tons of Gets.
   (I didn't see GetPotatoes because I have problems with my glasses!)

   The names for the functions was constructed beginning with «ACTION»
   { Get, Set, Kill, etc. } and then «OBJECT» {DC, Cursor, Message,
     Window, etc. }. So, the model here is «ActionObject».

Point 3.   
               When i want something, the first idea i have is about one «thing»,
   not the action. Then, i think «what i need to do with that thing»?
   When i choose the thing, i begin to think about the necessary
   actions: I need «Get», i need «Set», i need «Verify», i need «Kill»,
   i need «Convert», etc. etc. So, in my model, the function names          would be «ActionObject»:


DC     object:   {DCSet, DCGet, DC???,...},
Cursor object:   {CursorGet, CursorSet, CursorHide, Cursor???,...}
Window object:    {WindowGet, WindowSet, Window???, ...}
File   object:   {FileRead, FileWrite, FilePosition, FileClose, FileOpen,
        FileExtract, FileInsert, FileAdd, File??? }

In this way, we had a collection of objects and for each one we had a set of actions. So, when we want to write a program, we begin thinking something like this: what objects i need ? what do i want to do with it ? Then we go to look for that object and see what actions are defined for it. If we need more actions we would write more procedures to those actions we need.

Is it hard to follow ?

Point 4.   
                For me, the model «ActionObject» is hard to follow. I go to see one
   thing and i find many things around that has nothing to see with it.
   My eyes are obliged to see not connected things.

Stay well
Regards
Title: Re: Windows API32 documentation
Post by: hutch-- on April 05, 2005, 11:35:24 AM
RuiLoureiro,

Until you get the documentation that various people have advised you to get, you are flying in the dark. The WIN32.HLP file and / or the MSDN help files from the PLATFORMSDK. Once you have the reference material you are in a position to use the assistance that various members have offered.

As far as the architecture of MASM32 and what files are in what directories, try reading the technical data that comes with the project as it will help you a lot there. The LIB and INCLUDE directories hold mainly import libraries for the functions that are in various Windows system DLLs but they also have the static library for the MASM32 library and the floating point library that Ray has written. The INCLUDE directory has the INCLUDE files that comtain the prototypes for the procedures in the matching libraries.
Title: Re: Windows API32 documentation
Post by: Vortex on April 05, 2005, 07:07:17 PM
QuoteDon't  we need AllocConsole  or  CreateProcess, first ?

No, you don't need AllocConsole if you build your project as console application.
Title: Re: Windows API32 documentation
Post by: QvasiModo on April 05, 2005, 10:23:58 PM
Quote from: RuiLoureiro on April 05, 2005, 11:18:55 AM
Point 3.   
               When i want something, the first idea i have is about one «thing»,
   not the action. Then, i think «what i need to do with that thing»?
   When i choose the thing, i begin to think about the necessary
   actions: I need «Get», i need «Set», i need «Verify», i need «Kill»,
   i need «Convert», etc. etc. So, in my model, the function names          would be «ActionObject»:


DC     object:   {DCSet, DCGet, DC???,...},
Cursor object:   {CursorGet, CursorSet, CursorHide, Cursor???,...}
Window object:    {WindowGet, WindowSet, Window???, ...}
File   object:   {FileRead, FileWrite, FilePosition, FileClose, FileOpen,
        FileExtract, FileInsert, FileAdd, File??? }

In this way, we had a collection of objects and for each one we had a set of actions. So, when we want to write a program, we begin thinking something like this: what objects i need ? what do i want to do with it ? Then we go to look for that object and see what actions are defined for it. If we need more actions we would write more procedures to those actions we need.

Is it hard to follow ?

I bet you love C++, then... ;)

    ReturnValue = Object.Action(Parameters);
Title: Re: Windows API32 documentation
Post by: pbrennick on April 06, 2005, 12:04:23 AM
Although it is too late to change now; I find his point to be well stated.

Paul
Title: Re: Windows API32 documentation
Post by: Vortex on April 06, 2005, 06:00:32 PM
RuiLoureiro,

If you build your executable as a GUI application, here is how to proceed with AllocConsole

.386
.model flat,stdcall
option casemap:none

include \masm32\include\windows.inc
include \masm32\include\kernel32.inc
include \masm32\include\masm32.inc

includelib \masm32\lib\kernel32.lib
includelib \masm32\lib\masm32.lib

.data
msg           db 'Press RETURN to exit',0
.data?
buffer db 128 dup(?)
.code
start:
invoke AllocConsole
invoke StdOut,ADDR msg
invoke StdIn,ADDR buffer,128
invoke FreeConsole
invoke ExitProcess,0

END start
Title: Re: Windows API32 documentation
Post by: Xor Stance on April 06, 2005, 09:23:15 PM
buffedb I got a problem, in ANSI how do you make that symbol? []
Title: Re: Windows API32 documentation
Post by: QvasiModo on April 06, 2005, 09:56:20 PM
Quote from: Xor Stance on April 06, 2005, 09:23:15 PM
buffedb I got a problem, in ANSI how do you make that symbol? []
I think it's a TAB character that got printed wrong...
Title: Re: Windows API32 documentation
Post by: RuiLoureiro on April 12, 2005, 05:03:29 PM
Hi all

First, I have a problem with my internet. I hope to solve it soon.
-----------------------------------------------------------------------------------------------------------------
I wrote an example that use ReadConsoleInput and WriteConsoleOutputCharacter.
I have problems to access INPUT_RECORD structure. Thank you
-------------------------------------------------------------------------------------------------------------------------------------------

.386
.model flat,stdcall
option casemap:none
;----------------------------------------------------------------------------------------------
include \masm32\include\windows.inc
;-------------------------------------------------
INPUT_RECORD STRUCT
EventType WORD  ?
   UNION              ; Start the UNION
     KeyEvent     KEY_EVENT_RECORD <>
     MouseEvent     MOUSE_EVENT_RECORD <>
     WindowBufferSizeEvent  WINDOW_BUFFER_SIZE_RECORD <>
     MenuEvent     MENU_EVENT_RECORD <>
     FocusEvent     FOCUS_EVENT_RECORD <>
   ENDS
INPUT_RECORD ENDS
lINPUT_RECORD     equ SIZEOF INPUT_RECORD
;--------------------------------------------------------------------------------------------------------------------------------------
;  THIS IS THE STRUCTURE OF EACH RECORD IN THE INPUT BUFFER when a key event happen
;
;KEY_EVENT_RECORD       STRUCT        ; this structure is in windows.inc
;         EventType WORD  ?              ; 0 This belongs to INPUT_RECORD ?
;  bKeyDown          DWORD ?              ; +2
;  wRepeatCount         WORD  ?              ; +6
;  wVirtualKeyCode    WORD  ?              ; +8
;  wVirtualScanCode  WORD  ?              ;+10
;  uChar              CHARTYPE <>       ;+12   ; Here is one case only UnicodeChar or AsciiChar ?
;  dwControlKeyState DWORD ?               ; ???
;KEY_EVENT_RECORD       ENDS
;
;CHARTYPE             UNION         ; this structure is in windows.inc
; UnicodeChar    WORD ?
; AsciiChar          db ?
;CHARTYPE             ENDS
;
; mov    al, _InputBuffer.uChar.AsciiChar    ; it gives an error
;
;******************************************************************************
include \masm32\include\kernel32.inc
include \masm32\include\masm32.inc
; ---------------------------------------------------------------------------------------------
includelib \masm32\lib\kernel32.lib
includelib \masm32\lib\masm32.lib
;-------------------------------------------------------------------------------------------
.data

  _CurCOL_LIN          dd 00000000              ; current LIN + COL
;-------------------------------------------------------------------------------------------
.data?
   _hInputBuffer        dd ?                     ; input handle
   _hScreenBuffer       dd ?                     ; screen handle
;******************************************************************************
; HOW CAN I CREATE **a buffer for 20 INPUT_RECORDS** using the structure INPUT_RECORD ?
;******************************************************************************
  _InputBuffer              db 20 * 18 dup(?)         ; input buffer 20 records
  _NumInpRecord        dd ?                             ; n chars read
;-----------------------------------------------------------------------------------   
  _PrintBuffer              db 100 dup (?)               ; output buffer
  _NumWriteChar        dd ?                              ; n chars written
;----------------------------------------------------------------------------------------
.code
start:
      invoke  GetStdHandle, STD_INPUT_HANDLE ; Get STANDARD INPUT BUFFER handle
   cmp     eax, INVALID_HANDLE_VALUE
   je      eStart
   mov     _hInputBuffer, eax                              ; input buffer handle

   ; ---------------------------------
   ; Get STANDARD SCREEN BUFFER handle
   ; ---------------------------------
      invoke  GetStdHandle, STD_OUTPUT_HANDLE ; Get STANDARD SCREEN BUFFER handle
   cmp     eax, INVALID_HANDLE_VALUE
   je      ibStart
   mov     _hScreenBuffer, eax                             ; screen buffer handle

      invoke    FlushConsoleInputBuffer, _hInputBuffer      ; Clear Input buffer
TryReadEvent:
         ; -----------------------------
         ; Read INPUT BUFFER
         ; -----------------------------
         invoke    ReadConsoleInput, _hInputBuffer,             ; handle
                                    ADDR _InputBuffer,            ; Record buffer
                                    1,                            ; Read 1 record
                                    ADDR _NumInpRecord            ; Records Read
         cmp      eax, 0
         je       sbStart         
         ; --------------------------------
         ; How many read records
         ; --------------------------------
         mov    eax, _NumInpRecord
         cmp    eax, 0
         je     TryReadEvent

         ; ------------------
         ; What Event ?
         ; ------------------
         movzx  edx, word ptr [_InputBuffer + 0]                    ; EventType
         cmp      edx, KEY_EVENT
         jne        TryReadEvent

         ; ---------------------------------
         ; Process Keyboard Event
         ; ---------------------------------
         mov    ebx, dword ptr [_InputBuffer + 2]                   ; pressed\released
         movzx  ecx, word ptr [_InputBuffer + 6]                    ; repeat count
         movzx  ecx, word ptr [_InputBuffer + 8]                    ; VirtualKeyCode
         movzx  edx, word ptr [_InputBuffer + 10]                  ; VirtualScanCode
         mov    al,  byte ptr [_InputBuffer + 12]                       ; AsciiChar ?
         mov    esi, dword ptr [_InputBuffer + 13]                   ; ControlKeyState  ?
         mov    edi, dword ptr [_InputBuffer + 14]                   ; ControlKeyState ?

         ; ----------------------
         ; If released, next
         ; ----------------------
         cmp    ebx, FALSE
         je     TryReadEvent                                        ; released

         cmp    dl, 13
         je     short sbStart                                      ; RETURN -> exit
         ; -------------------------
         ; Set Key into _PrintBuffer
         ; -------------------------
         mov    byte ptr _PrintBuffer, dl                           ; ASCII is IN DL !!!
TryWrite:         
         invoke WriteConsoleOutputCharacter, _hScreenBuffer,        ; handle
                                    ADDR _PrintBuffer,                          ; print buffer
                                    1,                                          ; write 1 char
                                    _CurCOL_LIN,                                ; COL + LIN (word)
                                    ADDR _NumWriteChar                          ; Number write
         cmp    eax, 0
         je     sbStart                                    ; error -> exit

         cmp    _NumWriteChar, 1
         jne    TryWrite                                          ; try write
         jmp    TryReadEvent                                ; begin
;---------------------------------------------------------------------
;                             E N D
;---------------------------------------------------------------------         
sbStart: invoke CloseHandle, _hScreenBuffer          ; Close screen buffer
ibStart: invoke CloseHandle, _hInputBuffer             ; Close input buffer
eStart:
   invoke ExitProcess, 0
END start

-----------------------------------------------------------------------------
QvasiModo,
   You lost the bet ! I don't like C.

Vortex,
    Ok, it is assumed and understood ! Thank you for your help, Vortex.
    I compiled your example (i called cons2.asm )in 4 ways:
1. 
\MASM32\BIN\Ml.exe /c /coff              CONS2.asm
\MASM32\BIN\Link.exe /SUBSYSTEM:CONSOLE  CONS2.obj
2.
\MASM32\BIN\Ml.exe /c /coff              CONS2.asm
\MASM32\BIN\Link.exe /SUBSYSTEM:WINDOWS  CONS2.obj

3. Using QEditor
   3.1   - Project -> Console assemble and link
   3.2   - Project -> Build all

In all 4 cases it works in the same way as i expected.

Stay well
RuiLoureiro
Title: Re: Windows API32 documentation
Post by: RuiLoureiro on April 20, 2005, 04:50:18 PM
                                                 «All exact science is dominated by the idea of approximation»
                                                                                                               «Bertrand Russell»
Hi
       All comments are welcome.

1.   I want to use the ReadConsoleInput function to read the keyboard in a console application. 
     I am interested only in the  KEY_EVENT_RECORDs.

2.  Documentation:
                           ( To clean the input buffer we use FlushConsoleInputBuffer )

  2.A) Format of the ReadConsoleInput function:
   ReadConsoleInput reads data from a console input buffer and removes it from the buffer.

  Format: ReadConsoleInput (  hConsoleInput,   ; handle of a console input buffer
                                             lpBuffer,                   ; address of the buffer for read data
                                             nLength,                   ; number of records to read
                                             lpNumberOfEventsRead  ; address of number of records read )

Details:    lpBuffer:  Points to an INPUT_RECORD buffer that receives the input buffer data;
                 nLength:  Specifies the size, in input records, of the buffer pointed to by the lpBuffer;
                 lpNumberOfEventsRead: Points to a 32-bit variable.

  2.B) Input Record structure:   
       INPUT_RECORD {
            WORD EventType; 
             union { KEY_EVENT_RECORD KeyEvent;                     ; [ Only 1 of the following cases ]
                          MOUSE_EVENT_RECORD MouseEvent;
                          WINDOW_BUFFER_SIZE_RECORD WindowBufferSizeEvent;
                          MENU_EVENT_RECORD MenuEvent;           «discard »
                          FOCUS_EVENT_RECORD FocusEvent;                     «discard »
    } Event; }

      Taking into account  KEY_EVENT_RECORD, in this case,  the InputRecord structure is:

INPUT_RECORD       STRUCT                                         CHARTYPE        UNION
  EventType             WORD  ?              ; +0                     UnicodeChar    WORD ?
  KeyDown               DWORD ?             ; +2                     AsciiChar         db ?
  RepeatCount          WORD  ?              ; +6                    CHARTYPE        ENDS
  VirtualKeyCode       WORD  ?              ; +8
  VirtualScanCode      WORD  ?              ;+10
  Char                      CHARTYPE <>      ;+12             ; Only UnicodeChar or AsciiChar
  ControlKeyState      DWORD ?             ;+14
INPUT_RECORD         ENDS                  ;+18 bytes    imply each InputRecord = 18 bytes

3.  I tested this structure in my computer and i saw this ( i dont know why !):
     Position:      + 10              +12                 +14                +16             [ values in decimal ]
Key      VirtualScanCode   Char               AsciiCode      ControlKeyState
CapsLock          0014               003A                0000            0080  ; CAPSLOCK_ON  = 0080
Pg Up              0021               0049                 0000            0180  ; ENHANCED_KEY=0100
A                    0041               001E                 0041            0080
a                     0041               001E                 0061            0000   [ values in hexa ]

As we are seeing, the character «a» has the Ascii Code 61h in the position +14, but the ascii code of the key PgUp ( 49h ) is in the position +12.

So, i concluded  that  the  INPUT RECORD structure is:
INPUT_RECORD                   STRUCT                         
  EventType                       WORD   ?          ; +0        = KEY_EVENT  => key is the case       
  KeyDown                         DWORD ?          ; +2        pressed=TRUE, released = FALSE
  RepeatCount                    WORD   ?          ; +6                   
  VirtualKeyCode                  WORD  ?          ; +8
  VirtualScanCode                 WORD  ?          ;+10
  CharCode                          WORD  ?          ;+12          ???
  AsciiCode                          WORD  ?          ;+14          AsciiCode = low byte
  ControlKeyState                 WORD  ?          ;+16          ControlState
INPUT_RECORD                     ENDS              ;+18 bytes    imply each InputRecord = 18 bytes

4.  What RepeatCount is ?

   It  «specifies a count indicating that a key is being held down. For example, when a key is held down, you might get five events with this member equal to 1, one event with this member equal to 5, or multiple events with this member greater than or equal to 1.» [ this means two cases or three ???. The first produces 6 input records ??? ]

[ i conclude that  when  EventType= KEY_EVENT:
            if      NumberOfEventsRead=1 and RepeatCount=1 and KeyDown=TRUE
            than a key was pressed and we dont know if there are more InputRecords with this state;
            When a key is released there is only one InputRecord so i expect RepeatCount = 1, but ?]

5.  With this set of information, after  returning from ReadConsoleInput,  i think the procedure to get each key should do this:

              1º - test the word in the field EventType ( the first word );
              2º - test 32-bit variable NumberOfEventsRead to see if the function read records;
              3º - read and save the  AsciiCode and ControlKeyState  fields;
              4º - wait for a input record with KeyDown=FALSE and discarding all other input records
                    ( each time i press a key until i release, i want 1 code only for that key ).

6.  What happened  if  we specify  nLength = 10  ( number of records to read  ) and the function
     returns  with NumberOfEventsRead = 1  and  KeyDown=TRUE ? What the next InputRecord
     are if we use  FlushConsoleInputBuffer after the previous case ? It seems that we cannot read
     records with KeyDown=FALSE. But is this true ?

Stay well



Title: Re: Windows API32 documentation
Post by: RuiLoureiro on April 21, 2005, 05:24:16 PM
Hi
              I think no one is interested in console applications, but... perhaps...
   In the previous part of this topic i left some questions opened. No one said something. Meanwhile i completed my example to test the function ReadConsoleInput  and the result is:  when i press a key i get:
                      Records read = 1; Key Event; state «released» ; Repetition is 0 .
when i move the mouse i get:
                      Records read = 1; not Key Event; state «released»;  Repetition is 15 / 9  ( it is variable) .

And  it  never returns from the procedure KeyWait to CharRead.
( i am trying to write my procs like ObjectAction )

Why  do i get  the state «released» ?  Why  do i get  Repetition = 0  ?
Whats wrong ?

The procedures are: RCLEditStr, CharRead and KeyWait
--------------------------------------------------------------------------------------------------
RCLEditStr              proc   pRCLStr:DWORD, nLin:DWORD, nCol:DWORD
                                push   ebx
                                push   esi

                             [    ...    ]
                             invoke  CharRead, nLin, nCol             ; Read to AL, AH, DX
                             jc          _rRCLEditStr
                              ; -----------
                              ; Filter AL
                              ; -----------
_rRCLEditStr:           stc                                                  ; error
_eRCLEditStr:           pop   esi
                              pop   ebx
                              ret
;
; exit
; ----
_oRCLEditStr:          clc                                                  ; OK
                             pop   esi
                             pop   ebx
                             ret
RCLEditStr              endp
; ---------------------------------------------------------------------------------------------
CharRead              proc      nLin:DWORD, nCol:DWORD
                             push     ebx

_iCharRead:             invoke    CursorSetPos, nLin, nCol                ; Potition cursor
                             call         KeyWait
                             jnc          short _CharRead1                         ; There is 1 InputRecord         
                             ;
                             pop     ebx
                             ret
                           
_CharRead1:          cmp     ebx, TRUE
                           je         short _CharRead2                        ; key was pressed
                           ;
                           ; key was released -> Clean InputBuffer
                            ; ---------------------------------------------------
                           call        PrintPresRel
                           invoke   FlushConsoleInputBuffer, _hInputBuffer
                            jmp      _iCharRead
                             ;
                             ; Get codes
                             ; ------------
_CharRead2:           mov      ecx, eax
                             invoke  RecordGet                               ; Get codes AL=ascii ...
                             ;
_eCharRead:          clc
                            ;
                            pop     ebx
                            ret
CharRead               endp
; ---------------------------------------------------------------------------------------------
KeyWait                proc
                           LOCAL   nRecRead:DWORD

_iKeyWait:             invoke  ReadConsoleInput, _hInputBuffer,         
                                         ADDR _InputBuffer,       
                                          1,
                                          ADDR nRecRead
                            cmp      eax, 0
                             jne      _KeyWait1                          ; not error         
                  stc
                  ret
                   
_KeyWait1:             mov    ecx, nRecRead                                   ; Number of records read
                             push   esi
                                        mov    esi, offset _InputBuffer
                                        movzx  edx,  word ptr [esi + 0]          ; EventType
                                        mov    ebx, dword ptr [esi + 2]          ; pressed\released
                                        movzx  eax, word ptr  [esi + 6]          ; repeat count
                              pop    esi
;
call      PrintEventTyp                   ; Print Event type
call      PrintRecRead                    ; Print Number of records read
call      PrintNumRep                    ; Print Repeat count
call      PrintPresRel                      ; Print Pressed or Released
                           ;
                           cmp    ecx, 0
                           je       short _iKeyWait                      ; read 0
                           ;
                           cmp    edx, KEY_EVENT
                           jne     short _iKeyWait                      ; not key
                            ;
                            clc
                            ret
KeyWait                endp

Thanks
Title: Re: Windows API32 documentation
Post by: RuiLoureiro on April 22, 2005, 11:34:39 AM
Hi
      I want to add two things: one is to say to MichaelW that i am sure he is a very good person
      and thanks for his help; the other is the following 6 points:
   
1.    I left an attached file cons18.zip in CAMPUS in my topic «Console Input-Output»;

2.    In the previous post i said that «KeyWait never returns to CharRead»;

3.    I modified CharRead in this way

            cmp     ebx, TRUE
            je      short _CharRead2                           ; key was pressed
            ;
            ; key was released -> Clean InputBuffer
            ; ----------------------------------------------------
            invoke   PASStrPrint, ADDR _MsgReleased1, 3, 40
            invoke   FlushConsoleInputBuffer, _hInputBuffer
            jmp      _iCharRead

where
                      dd 12               
_MsgReleased1  db "ret Released"   


4.    In this case, i see that when i hold down a key, KeyWait returns       
       but EBX = FALSE ( the result is «ret Released» ) !

5.    According to documentation, the DWORD at position +2 in the structure INPUT_RECORD is
       KeyDown DWORD ? ; meaning pressed= TRUE    or   released= FALSE.
      
6.    Then, we can conclude that SOMETHING IS WRONG: in my computer  or in the documentation.


Title: Re: Windows API32 documentation
Post by: hutch-- on April 22, 2005, 01:12:10 PM
RuiLoureiro,

Try and use the "code" tags when you post code as it is currently very hard to read as you paste it in.

Put the word "code" in square brackets BEFORE the code you paste in and put the negation of it "/code" in square brackets AFTER the code then it will format properly and be a lot easier to read.
Title: Re: Windows API32 documentation
Post by: RuiLoureiro on April 22, 2005, 03:10:34 PM
Hi MASTER Hutch

    How are you ? I hope you are fine.
    Good observation and help. I have some difficulties within this text processor. I need some more time ... (and tons of  skill, too ) !

Thank you
My regards