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
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.
Hi
Vortex,
What are all functions to allocate system memmory ? What they return ?
Can we call GlobalAlloc with uFlags = GMEM_DISCARDABLE ?
Regards
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.
Hi Vortex
Whats the difference between GlobalAlloc and LocalAlloc if the heap is the same ?
Regards
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.
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.
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
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.
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
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
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 ...)
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.
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
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.
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.
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
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
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
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.
QuoteDon't we need AllocConsole or CreateProcess, first ?
No, you don't need
AllocConsole if you build your project as
console application.
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);
Although it is too late to change now; I find his point to be well stated.
Paul
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
buffedb I got a problem, in ANSI how do you make that symbol? []
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...
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
«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
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
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.
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.
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