Just been working on these for my ASM Runtime (http://www.masmforum.com/simple/index.php?topic=3148.15) (excuse the shameless self promotion) and I discovered something strange.
The current definition of the ACCEL structure, according to both MSDN and MASM32, is this:
ACCEL STRUCT
fVirt BYTE ?
key WORD ?
cmd WORD ?
ACCEL ENDS
This, while according to Microsoft, is correct, however, it doesn't work. After much angst (and a bit of debugging... :wink) I found that the following works:
ACCEL STRUCT
fVirt WORD ?
key WORD ?
cmd WORD ?
ACCEL ENDS
However, I have nothing (besides the fact that this actually works) to back this up.
zooba,
> However, I have nothing (besides the fact that this actually works) to back this up.
Thats probably a reasonable start. :bg
It is not uncommon to find stuff ups in the API and related reference.
AFAIK, default packing applies to this structure - the second field needs to be WORD aligned, so a padding byte is inserted after the first (BYTE) field by the C/C++ compiler...
Pelle
zooba,
AFAICT the compiler would align the structure on an 8-byte boundary, and the structure members at an address that is a multiple of the member's size. I don't know about the structure alignment requirements, but Windows apparently assumes that the structure members are aligned as they would be by the compiler. There have been discussions here of this sort of problem, with this and other structures/unions, but I don't recall which forum or topic.
MSDN: Data Alignment Examples (http://msdn.microsoft.com/library/default.asp?url=/library/en-us/randz/protocol/data_alignment_examples.asp)
MSDN: Structure Alignment (http://msdn.microsoft.com/library/default.asp?url=/library/en-us/vccore/html/_core_structure_alignment.asp)
With MASM, you can specify an alignment for structures/unions:
Quote
name{STRUCT|UNION} [alignment] [,NONUNIQUE]
field declarations
name ENDS
The actual alignment will be the minimum of the alignment and the field's size. For MASM 6.14 the alignment can be specified as 1, 2, 4, 8, or 16, with 1 the default. (I realize that for you a bare description would suffice, but in the interest of illustrating it for the beginners...):
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
include \masm32\include\masm32rt.inc
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
.data
accel ACCEL <>
_ACCEL STRUCT WORD
fVirt DB ?
key DW ?
cmd DW ?
_ACCEL ENDS
_accel _ACCEL <>
.code
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
start:
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
print uhex$(offset accel.fVirt),13,10
print uhex$(offset accel.key),13,10
print uhex$(offset accel.cmd),13,10
print uhex$(offset _accel.fVirt),13,10
print uhex$(offset _accel.key),13,10
print uhex$(offset _accel.cmd),13,10
mov eax, input(13,10,"Press enter to exit...")
exit
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
end start
00403000
00403001
00403003
00403005
00403007
00403009
And I verified that the modified structure works correctly. In my test app this worked as expected:
accel _ACCEL <FVIRTKEY or FCONTROL,VK_Z,10001>
But this did not:
accel ACCEL <FVIRTKEY or FCONTROL,VK_Z,10001>
Thanks Pelle and Michael.
I think Pelle's answer sums it up best - MASM takes things literally while C/C++ messes around with them and doesn't tell you :P
Michael's solution is probably 'best', but it makes it slightly harder to read. In any case, (and this should probably go into the WINDOWS.INC forum now), can it be fixed for the next release?
BTW. will the SIZEOF operator correctly interpret the WORD alignment?
Edit - I'll answer my own question - yes it does :wink
Cheers,
Zooba
Let me see, if I have it right, the structure should be something like this.
ACCEL STRUCT WORD
fVirt BYTE ?
key WORD ?
cmd WORD ?
ACCEL ENDS
Yep, that's right. :U
Done,
Its now in the reference version files that build WINDOWS.INC. :cheekygreen:
And after all that I see Donkey raised the exact same issue almost a year ago here (http://www.masmforum.com/simple/index.php?topic=332.0) ::)
The reason why I changed the method of building the file is so it could be edited. A year ago it was a melodrama to successfully edit the file as it was very large, I have a different system that is broken up into about 30 categories.
It is still no joy to edit and the sequential dependencies must be correct but at least its under control now.
Quote from: hutch-- on December 18, 2005, 08:48:55 AM...it was a melodrama...
Is that the one where someone holds up signs saying "Yay!" and "Boo!"?
Out of interest, how are you dividing it? By DLLs? And how come you havent written yourself a tool to deal with sequential dependencies :P :toothy
zooba,
> ...it was a melodrama...
No, like Sarah Bernhardt. The problem was it had been edited in bits by various people over a number of years and to fix it up and get it to work took a mountain of work and testing to make sure the dependencies were in the right order. Once I had the basic layout working I split the file into as many categories as was convenient, some by equate list, others as structures and the rest by DLL when it was a large collection.
Putting it all back together is as simple as a batch file to list the files which are named with a leading pair of numbers into a list then I use another simple tool to copy them sequentially into the WINDOWS.INC file. It is as simple as changing the correct file, saving it and running a batch file that does the rest.