The MASM Forum Archive 2004 to 2012

General Forums => The Workshop => Topic started by: jdoe on September 07, 2007, 10:49:37 PM

Title: Azimut Project
Post by: jdoe on September 07, 2007, 10:49:37 PM

Instead of opening multiple topics for the same goal, I thought that one should be enough.

I'm working on a Windows SDK/DDK translation for MASM (Prototypes/Constants/Structures) and I need from time to time few answers that some of them were already gotten elsewhere on this board.

I have now all the set of SDK/DKK (2000/XP/2003/Vista/2008) and what I'll have to offer when completed is all of them merged together.

At this day, I have all the prototypes for the DLLs (*.lib and *.inc). The include files, as an extra feature that I can't live without, is taking care of the UNICODE definition to make it easier to switch from ANSI to UNICODE.

I'll be thankful to those taking time to give me answers and in case some of them may seem obvious, believe me, they are not for someone that is not a C programmer. I can't expect my deduction to be flawless.

Regards

------------------------------------------------------------------------

Version 0.3
Import libraries and prototypes
Resource Header File
Unicode support (TCHAR and String Macro)

http://pages.videotron.com/jdoe/index.html

Title: Re: Azimut Project
Post by: jdoe on September 07, 2007, 11:13:11 PM
There are structures that needs to be aligned. The program build correctly but fails at run-time if the alignment is not corrected.

I know one of these cases that I came across when coding a COM shell extension. It is FORMATETC that need to be DWORD aligned.



FORMATETC STRUCT DWORD
   cfFormat                   WORD ?
   ptd                        DWORD ?
   dwAspect                   DWORD ?
   lindex                     DWORD ?
   tymed                      DWORD ?
FORMATETC ENDS




Do you know more of these structures and the alignment required ?

Title: Re: Azimut Project
Post by: GregL on September 08, 2007, 02:08:02 AM
jdoe,

You see where the problem is right? If DWORD alignment is not specified the cfFormat field will throw off the alignment for the rest of the fields. If DWORD alignment is specified MASM will insert (2) padding bytes to restore the alignment of the rest of the fields. A C compiler does this for you, so C programmers  usually don't have to deal with it. Intel x86 processors will read misaligned data but will have substantial performance losses. Some CPUs will not read misaligned data.

Read - The MASM Programmer's Guide - Chapter 5 - Defining and Using Complex Data Types - Alignment Value and Offsets for Structures.

Here's another article (http://msdn2.microsoft.com/en-us/library/Aa290049(VS.71).aspx)  from MSDN.

The rule of thumb is:

  REAL10 should be 16 byte aligned
  REAL8 and QWORD should be 8 byte aligned
  REAL4 and DWORD should be 4 byte aligned
  WORD should be 2 byte aligned
  BYTE doesn't need any alignment

Title: Re: Azimut Project
Post by: jdoe on September 08, 2007, 02:28:01 AM
Quote from: Greg on September 08, 2007, 02:08:02 AM

You see where the problem is right? 



Yes, but what I don't understand is why in most case misalignment don't fails and in particular case it does ?

I'll look at the link you post for some clues. Thanks.


Title: Re: Azimut Project
Post by: GregL on September 08, 2007, 02:40:50 AM
jdoe,

Oh ..., sorry, I thought you were asking how to determine which structures needed alignment adjustments. Well, I bet it's a pointer problem.

Title: Re: Azimut Project
Post by: jdoe on September 08, 2007, 02:48:26 AM
Quote from: Greg on September 08, 2007, 02:40:50 AM

Oh ... , I thought you were asking how to determine which structures needed alignment. Well, I bet it's a pointer problem.



You were not mistaken but you'll see my point with this example.




URL_COMPONENTSA STRUCT
    dwStructSize          DWORD ?
    lpszScheme            DWORD ?
    dwSchemeLength        DWORD ?
    nScheme               DWORD ?
    lpszHostName          DWORD ?
    dwHostNameLength      DWORD ?
    nPort                 WORD ?
    lpszUserName          DWORD ?
    dwUserNameLength      DWORD ?
    lpszPassword          DWORD ?
    dwPasswordLength      DWORD ?
    lpszUrlPath           DWORD ?
    dwUrlPathLength       DWORD ?
    lpszExtraInfo         DWORD ?
    dwExtraInfoLength     DWORD ?
URL_COMPONENTSA ENDS




nPort make the rest to not be aligned but it will not fails. Why ?

Title: Re: Azimut Project
Post by: GregL on September 08, 2007, 03:13:19 AM
jdoe,

Without seeing the code, my only guess is it that has something to do with a pointer to the FORMATETC structure, as the structure data is actually changing memory location with the alignment.

Wait, FORMATETC is failing when it's not DWORD aligned, ..., I don't know.

Title: Re: Azimut Project
Post by: MichaelW on September 08, 2007, 05:11:03 AM
It fails in my tests.

; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
    include \masm32\include\masm32rt.inc
    include \masm32\macros\ucmacros.asm
    includelib winhttp.lib

    WinHttpCreateUrl PROTO :DWORD,:DWORD,:DWORD,:DWORD

    WinHttpCrackUrl  PROTO :DWORD,:DWORD,:DWORD,:DWORD

    _URL_COMPONENTSA STRUCT DWORD
      dwStructSize       dd ?
      lpszScheme         dd ?
      dwSchemeLength     dd ?
      nScheme            dd ?
      lpszHostName       dd ?
      dwHostNameLength   dd ?
      nPort              dw ?
      lpszUserName       dd ?
      dwUserNameLength   dd ?
      lpszPassword       dd ?
      dwPasswordLength   dd ?
      lpszUrlPath        dd ?
      dwUrlPathLength    dd ?
      lpszExtraInfo      dd ?
      dwExtraInfoLength  dd ?
    _URL_COMPONENTSA ENDS
   
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
    .data
   
      WSTR url1,"http://www.masm32.com/board/index.php?topic=7866.msg57747#msg57747"

      urlComp   _URL_COMPONENTSA <>
      url2      dw 100 dup(0)
      urlLen    dd 0     
    .code
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
start:
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
    invoke RtlZeroMemory, ADDR urlComp, SIZEOF urlComp
    mov urlComp.dwStructSize,     SIZEOF urlComp
    mov urlComp.dwSchemeLength,   -1
    mov urlComp.dwHostNameLength, -1
    mov urlComp.dwUrlPathLength,  -1
    mov urlComp.dwExtraInfoLength,-1

    invoke crt_wcslen, ADDR url1
    mov urlLen, eax
    print ustr$(eax),13,10
    invoke crt_printf, chr$("%S%c%c"), ADDR url1, "<", 10

    inc urlLen          ; must be set to acceptable length

    invoke WinHttpCrackUrl, ADDR url1, urlLen, 0, ADDR urlComp
    print ustr$(eax),13,10

    invoke WinHttpCreateUrl, ADDR urlComp, 0, ADDR url2, ADDR urlLen
    print ustr$(eax),13,10
   
    print ustr$(urlLen),13,10
    invoke crt_printf, chr$("%S%c%c%c"), ADDR url2, "<", 10, 10

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

Results as posted:

66
http://www.masm32.com/board/index.php?topic=7866.msg57747#msg57747<
1
1
66
http://www.masm32.com/board/index.php?topic=7866.msg57747#msg57747<

Results with DWORD following STRUCT commented:

66
http://www.masm32.com/board/index.php?topic=7866.msg57747#msg57747<
0
0
67
<
Title: Re: Azimut Project
Post by: jdoe on September 08, 2007, 06:09:19 AM

Quote from: Greg on September 08, 2007, 03:13:19 AM
Without seeing the code, my only guess is it that has something to do with a pointer to the FORMATETC structure, as the structure data is actually changing memory location with the alignment.
Wait, FORMATETC is failing when it's not DWORD aligned, ..., I don't know.


Greg,

This the part of the code I was talking about that needs the FORMATETC structure.




IShellExtInit_Initialize PROC p_this:DWORD, p_pidlFolder:DWORD, p_pdtobj:DWORD, p_hkeyProgID:DWORD

    LOCAL l_fetc:FORMATETC
    LOCAL l_stgm:STGMEDIUM
    LOCAL l_dwMemSize:DWORD
    LOCAL l_hResult:DWORD

    mov l_hResult, E_FAIL

    ;
    ; We must have a IDataObject pointer to get paths
    ;

    .if (p_pdtobj != NULL)

        mov l_fetc.cfFormat, CF_HDROP
        mov l_fetc.ptd, NULL
        mov l_fetc.dwAspect, DVASPECT_CONTENT
        mov l_fetc.lindex, -1
        mov l_fetc.tymed, TYMED_HGLOBAL

        ;
        ; Get selected paths
        ; hGlobal of STGMEDIUM points to a DROPFILES structure
        ;

        CallMethod GetData, IDataObject, p_pdtobj, addr l_fetc, addr l_stgm
        .if (eax == S_OK)

        ...

        .endif




If "STRUCT DWORD" is not used then the GetData method of the IDataObject interface fails. Must be aligned. No workaround.

------------------

MichaelW,

Sorry, maybe I made you work for nothing on this one. Shame on me because I took a structure randomly that had a word in between just to try to point you where I don't understand. What adding "STRUCT DWORD" is doing to the URL_COMPONENTSA structure when the first member is already a DWORD. The nPort member can't be patch with 2 bytes at this location because it will add 2 bytes to the structure size.


Look...



SHFILEOPSTRUCT STRUCT
   hwnd                                DWORD ?
   wFunc                               DWORD ?
   pFrom                               DWORD ?
   pTo                                 DWORD ?
   fFlags                              WORD ?
   fAnyOperationsAborted               DWORD ?
   hNameMappings                       DWORD ?
   lpszProgressTitle                   DWORD ?
SHFILEOPSTRUCT ENDS





I'm using this one since very long time and never add problems with it, without alignment and never fails. What is the difference ?

Wow, my question was having a list of structure that really needs to be aligned and Now I'm confused about structures alignment.

:'(

Title: Re: Azimut Project
Post by: MichaelW on September 08, 2007, 07:02:24 AM
Windows expects the structure member alignments to match the alignments implemented by the C/C++ compilers. Depending on the structure and what you are doing with it, having some of the members misaligned may have no effect. For the URL_COMPONENTS structure, when used with the WinHttpCrackUrl and/or WinHttpCreateUrl function, it does have an effect, and these two apps demonstrate why.

typedef unsigned long       DWORD;
typedef unsigned short WCHAR;
typedef WCHAR *LPWSTR, *PWSTR;
typedef int INTERNET_SCHEME, * LPINTERNET_SCHEME;
typedef unsigned short      WORD;
typedef WORD INTERNET_PORT;
typedef struct
{
    DWORD   dwStructSize;       // size of this structure. Used in version check
    LPWSTR  lpszScheme;         // pointer to scheme name
    DWORD   dwSchemeLength;     // length of scheme name
    INTERNET_SCHEME nScheme;    // enumerated scheme type (if known)
    LPWSTR  lpszHostName;       // pointer to host name
    DWORD   dwHostNameLength;   // length of host name
    INTERNET_PORT nPort;        // converted port number
    LPWSTR  lpszUserName;       // pointer to user name
    DWORD   dwUserNameLength;   // length of user name
    LPWSTR  lpszPassword;       // pointer to password
    DWORD   dwPasswordLength;   // length of password
    LPWSTR  lpszUrlPath;        // pointer to URL-path
    DWORD   dwUrlPathLength;    // length of URL-path
    LPWSTR  lpszExtraInfo;      // pointer to extra information (e.g. ?foo or #foo)
    DWORD   dwExtraInfoLength;  // length of extra information
}
URL_COMPONENTS, * LPURL_COMPONENTS;

int main()
{
  URL_COMPONENTS urlComp;
  printf("sizeof(urlComp) %d%c",sizeof(urlComp),10);
  printf("&urlComp.dwHostNameLength %x%c",&urlComp.dwHostNameLength,10);
  printf("&urlComp.nPort %x%c",&urlComp.nPort,10);
  printf("&urlComp.lpszUserName %x%c",&urlComp.lpszUserName,10);
  while( !kbhit() );
  return 0;
}


sizeof(urlComp) 60
&urlComp.dwHostNameLength 12febc
&urlComp.nPort 12fec0
&urlComp.lpszUserName 12fec4


; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
    include \masm32\include\masm32rt.inc

    _URL_COMPONENTSA STRUCT
      dwStructSize       dd ?
      lpszScheme         dd ?
      dwSchemeLength     dd ?
      nScheme            dd ?
      lpszHostName       dd ?
      dwHostNameLength   dd ?
      nPort              dw ?
      lpszUserName       dd ?
      dwUserNameLength   dd ?
      lpszPassword       dd ?
      dwPasswordLength   dd ?
      lpszUrlPath        dd ?
      dwUrlPathLength    dd ?
      lpszExtraInfo      dd ?
      dwExtraInfoLength  dd ?
    _URL_COMPONENTSA ENDS

; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
    .data
      urlComp   _URL_COMPONENTSA <>
    .code
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
start:
; «««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««««
    print chr$("SIZEOF(_URL_COMPONENTSA) ")
    print ustr$(SIZEOF(_URL_COMPONENTSA)),13,10
    print chr$("OFFSET urlComp.dwHostNameLength ")
    print uhex$(OFFSET urlComp.dwHostNameLength),13,10
    print chr$("OFFSET urlComp.nPort ")
    print uhex$(OFFSET urlComp.nPort),13,10
    print chr$("OFFSET urlComp.lpszUserName ")
    print uhex$(OFFSET urlComp.lpszUserName),13,10

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


SIZEOF(_URL_COMPONENTSA) 58
OFFSET urlComp.dwHostNameLength 00403014
OFFSET urlComp.nPort 00403018
OFFSET urlComp.lpszUserName 0040301A


Title: Re: Azimut Project
Post by: jdoe on September 08, 2007, 07:28:57 AM

MichaelW,

So the alignment is done on each members of a structure.

Now it means that to be safe, all structures must be at least DWORD aligned if one of the member is less than a DWORD. But if one member is a REAL8. Hmmm...

I'll sleep on that... it's 3h30 here.


Thanks for your time Michael. Really appreciated.



:U


Title: Re: Azimut Project
Post by: japheth on September 08, 2007, 12:40:44 PM
Hello,

Quote from: jdoe on September 07, 2007, 10:49:37 PM

Instead of opening multiple topics for the same goal, I thought that one should be enough.

I'm working on a Windows SDK/DDK translation for MASM (Prototypes/Constants/Structures) and I need from time to time few answers that some of them were already gotten elsewhere on this board.

I have now all the set of SDK/DKK (2000/XP/2003/Vista) and what I'll have to offer when completed is all of them merged together.

are you aware of http://www.japheth.de/Win32Inc.html?

QuoteDo you know more of these structures and the alignment required ?

Yes. Which come into my mind are some in OAIDL (OLE automation), and for console, the INPUT_RECORD also needs dword alignment.

Thanks for the hint about the WORD member in URL_COMPONENTS! It's missing in Win32Inc.

Regards

Japheth
Title: Re: Azimut Project
Post by: MichaelW on September 08, 2007, 01:59:13 PM
The URL_COMPONENTS in the current MASM32 windows.inc is correct.

URL_COMPONENTSA STRUCT dword
dwStructSize       dd ?
lpszScheme         dd ?
dwSchemeLength     dd ?
nScheme            dd ?
lpszHostName       dd ?
dwHostNameLength   dd ?
nPort              dw ?
lpszUserName       dd ?
dwUserNameLength   dd ?
lpszPassword       dd ?
dwPasswordLength   dd ?
lpszUrlPath        dd ?
dwUrlPathLength    dd ?
lpszExtraInfo      dd ?
dwExtraInfoLength  dd ?
URL_COMPONENTSA ENDS
URL_COMPONENTS equ <URL_COMPONENTSA>
Title: Re: Azimut Project
Post by: GregL on September 08, 2007, 07:04:05 PM
jdoe,

Quote from: MichaelWWindows expects the structure member alignments to match the alignments implemented by the C/C++ compilers.

That's what it boils down to.





Title: Re: Azimut Project
Post by: jdoe on September 09, 2007, 12:48:32 AM

Finally, I found what in the C header files, tells which structures needs to be aligned and how.

So the statement I made before...

Quote
Now it means that to be safe, all structures must be at least DWORD aligned if one of the member is less than a DWORD

...is wrong.

----------

I need to check for "pshpack1.h, pshpack2.h, pshpack4.h, pshpack8.h", to know which structures (and how) needs to be aligned.



/*
* Set up Structure Packing to be 4 bytes
* for all wininet structures
*/
#if defined(_WIN64)
#include <pshpack8.h>
#else
#include <pshpack4.h>
#endif




Thanks for your clues.


Title: Re: Azimut Project
Post by: MichaelW on September 09, 2007, 06:44:13 AM
In his first reply in this thread Greg provided a link to an MSDN article that details the alignment rules used by the Microsoft C/C++ compilers, see the Structure and Union Layout section.

Title: Re: Azimut Project
Post by: jdoe on September 09, 2007, 07:46:51 AM
Quote from: MichaelW on September 09, 2007, 06:44:13 AM

In his first reply in this thread Greg provided a link to an MSDN article that details the alignment rules used by the Microsoft C/C++ compilers, see the Structure and Union Layout section.


Michael,

My last post end with "Thanks for your clues" and it was not a question. I don't really understand what you mean.

I'm working on a H2INC program so what I need, has to be in the header files as much as possible and knowing the alignment rules may have been of some interests if I had not found the utility of pshpack?.h and poppack.h.

In fact, the "Structure and Union Layout" section and the key words "#pragma pack" is exactly what put to me on the track at second reading. I was a little desperate to have to do the alignment stuff myself. Dealing with the constants in the header files is much more easier but I just realized that I need to be carefull when it comes to structures. I thought it was much simpler.

Thanks again guys and I hope you will be there for my next question. I know what it will be but each thing in its time.

See you later

:U

Title: Re: Azimut Project
Post by: jdoe on September 09, 2007, 08:14:06 AM
Quote from: japheth on September 08, 2007, 12:40:44 PM

are you aware of http://www.japheth.de/Win32Inc.html?


japheth,

Like I said, I am merging all SDK/DDK (Windows 2000 and higher). The result for now is 357 .lib files and around 850 .inc files. I will put the .inc files in a single one (most of them commented though for faster compilation time).

AZIMUT.INC ...
Quote

; Azimut Project
; Copyright (c) 2007 Philippe Coulombe
; All rights reserved

;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; Definition
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

IFNDEF _WIN32_IE
_WIN32_IE EQU 0500h
ENDIF

IFNDEF _WIN32_WINNT
_WIN32_WINNT EQU 05000000h
ENDIF

IFNDEF UNICODE
DBCHAR TYPEDEF BYTE
TCHAR TYPEDEF BYTE
ELSE
DBCHAR TYPEDEF WORD
TCHAR TYPEDEF WORD
ENDIF

;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; Macro
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

CTL_CODE MACRO DeviceType:REQ, Function:REQ, Method:REQ, Access:REQ
    exitm <((DeviceType shl 16) or (Access shl 14) or (Function shl 2) or Method)>
ENDM

;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

HRESULT_FROM_WIN32 MACRO x:REQ
    if ((x eq 0) or (x shr 31))
        exitm <(x)>
    else
        exitm <((x and 0000FFFFh) or (FACILITY_WIN32 shl 16) or 80000000h)>
    endif
ENDM

;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

MAKE_HRESULT MACRO sev:REQ, fac:REQ, code:REQ
    exitm <((sev shl 31) or (fac shl 16) or code)>
ENDM

;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

MAKE_SCODE MACRO sev:REQ, fac:REQ, code:REQ
    exitm <((sev shl 31) or (fac shl 16) or code)>
ENDM

;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; Include
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

;INCLUDE accctrl.inc
;INCLUDE aclapi.inc
;INCLUDE aclui.inc
;INCLUDE activecf.inc
;INCLUDE activeds.inc
;INCLUDE activscp.inc
;INCLUDE adhoc.inc
;INCLUDE admex.inc
;INCLUDE adoint.inc



But your Win32Inc is a good work.

:thumbu

Title: Re: Azimut Project
Post by: drizz on September 09, 2007, 06:18:12 PM
just be aware of struct alignment bugs!
asm file 1) Internal Assembler Error
asm file 2) Output incorrect
attached.

[attachment deleted by admin]
Title: Re: Azimut Project
Post by: jdoe on September 09, 2007, 07:43:34 PM
Quote from: drizz on September 09, 2007, 06:18:12 PM
just be aware of struct alignment bugs!
asm file 1) Internal Assembler Error
asm file 2) Output incorrect
attached.

drizz,

For asm file 1 bug, I checked in C header (wincon.h) and it is said nowhere that it must be DWORD aligned. When not, it assemble without "Internal error".

For asm file 2 bug, yes the data layout is weird and it seems to come from the use of DUP "2 dup(<>)".

Honestly, I have no choice to rely on the C header files. If there is a bug for few structures with ML, what can I do. It's Microsoft Business, not mine.

Thanks for the information.

:wink

Title: Re: Azimut Project
Post by: MichaelW on September 09, 2007, 09:15:30 PM
QuoteFor asm file 1 bug, I checked in C header (wincon.h) and it is said nowhere that it must be DWORD aligned. When not, it assemble without "Internal error".
The header file depends on the compiler to align the structure members correctly. If you attempt to use the structure without taking care to align it then the code that uses the structure will not work correctly. There is a thread somewhere here that discusses this exact problem.
The structure as aligned by the C/C++ compiler:

sizeof(ir) 20
&ir.EventType 12fed0
&ir.Event.KeyEvent 12fed4

And the unaligned structure in MASM:

SIZEOF ir 18
OFFSET ir.EventType 00403000
OFFSET ir.Event.KeyEvent00403002

You can avoid the bug and achieve the necessary alignment by padding the structure:

INPUT_RECORD struct
  EventType WORD ?
  WORD ?
  union Event
    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

Title: Re: Azimut Project
Post by: jdoe on September 09, 2007, 09:49:27 PM

Michael,

So, it is not the opposite... the compiler needs the header files to align structures ?
What are "#pragma pack" and "#pragma pop" for then ?
I thought I understood something. Thanks for the disillusion  :P

So the INPUT_RECORD structure is known to be a buggy one. Is there some more like that, you can remember ?
I will take care of them manually.

Title: Re: Azimut Project
Post by: MichaelW on September 10, 2007, 08:28:35 PM
AFAIK the #PRAGMA PACK directive is used to control packing alignment for special cases.

http://msdn2.microsoft.com/en-us/library/d9x1s805(VS.71).aspx
http://msdn2.microsoft.com/en-us/library/ms856729.aspx

Perhaps my statement should have been:

The header file depends on the default behavior of the compiler to align the structure members correctly.

Obviously you can use #PRAGMA PACK or /Zp to force alignments that will not work with the Windows API.

In addition to INPUT_RECORD, japheth mentioned OAIDL (OLE automation).

Another that I recall is PDH_FMT_COUNTERVALUE. It's not in Windows.inc, but IIRC to implement it in MASM you must add padding to properly align the union within the structure.
Title: Re: Azimut Project
Post by: jdoe on September 11, 2007, 12:22:04 AM
Quote from: MichaelW on September 10, 2007, 08:28:35 PM

Obviously you can use #PRAGMA PACK...



It is my best option for now and I am confident that it will be sufficient.

Title: Re: Azimut Project
Post by: jdoe on October 11, 2007, 02:42:44 AM
Hi,

I would like to know what is the difference between these (in header files)...

#if _AMD64
#if _M_IA64
#if _WIN64

Does they're all condition for Win64 or (_AMD64 and _M_IA64) makes a specific difference between AMD and Itanium processors ?

Thanks

Title: Re: Azimut Project
Post by: jdoe on October 13, 2007, 01:57:42 AM
Quote from: jdoe on October 11, 2007, 02:42:44 AM

#if _AMD64
#if _M_IA64
#if _WIN64

Does they're all condition for Win64 or (_AMD64 and _M_IA64) makes a specific difference between AMD and Itanium processors ?



I found that if I want to make my include files compatible with Win64, I must exclude all the _M_IA64 stuff from the header files because it is not really Win64 or I should say yes it is but not for ML64.EXE.

But I still don't know what to do with _AMD64. I thought that AMD64 was the same as the 64bits processor of Intel but it seems not because specific structure member use it.


KWAIT_BLOCK STRUCT
    WaitListEntry LIST_ENTRY <?>
    Thread DWORD ?
    Object DWORD ?
    NextWaitBlock DWORD ?
    WaitKey WORD ?
    WaitType BYTE ?
    SpareByte BYTE ?
IFDEF _AMD64_
    SpareLong DWORD ?
ENDIF
KWAIT_BLOCK ENDS


From that example, do I have to include SpareLong member only for AMD processor or for all WIN64 stuff.

:'(

Title: Re: Azimut Project
Post by: GregL on October 13, 2007, 02:38:23 AM
jdoe,

Looks like you figured it out, _M_IA64 is for Itanium.

Title: Re: Azimut Project
Post by: Vortex on October 13, 2007, 10:07:24 AM
Hi jdoe,

How is going the work with your project?
Title: Re: Azimut Project
Post by: jdoe on October 13, 2007, 08:00:25 PM
Quote from: Vortex on October 13, 2007, 10:07:24 AM
Hi jdoe,

How is going the work with your project?

Vortex,

Lately, after reading many header files, I decided that it would be more judicious to make my include files compatible with WIN64 stuff by adding the _WIN64 definition. The Windows 64 programming will get more and more popular so instead of doing another package later, I doing it once for 32 and 64 bits Windows.

I know I'm not very fast to come up with something but my living job takes a lot of my time right now because of many projects at the same time. My free time suffer of that.

I put all my other personal projects on the side, so what is sure, Azimut is gonna see the day before the end of 2007... I hope.

Title: Re: Azimut Project
Post by: jdoe on November 27, 2007, 12:25:22 AM
Hi,

For now, the includes *.lib and *.inc are completed for WIN32 (and CONSTANTS almost completed) but I had a look at WIN64 *.lib and they all looks like C import libraries. I read on the net that WIN64 use FASTCALL calling convention (first 4 parameters on registers and the rest on the stack) but the function name decoration don't fit the usual @function@4 name seen for WIN32. It seem to be no way to know on much parameter a function use like it is for C (:VARARG).

My questions are...

1) Is there any needs for PROTOs or only something like EXTERNDEF ExitProcess:PROC (or should I wrote EXTRN) ?
2) I didn't found examples using INVOKE so does this internal macro still exist or only CALL is used.
3) Does WIN64 APIs restore the stack before returning like it was for FASTCALL on WIN32 (I don't think so) ?
4) Does the WIN64 APIs have all the same parameters count as for WIN32 ?

Thanks for your enlightment.

Title: Re: Azimut Project
Post by: Vortex on November 27, 2007, 06:36:26 PM
jdoe,

If it's possible, could you post here some Win64 import libraries? I would like to have a look at them.
Title: Re: Azimut Project
Post by: jdoe on November 28, 2007, 12:03:41 AM
Quote from: Vortex on November 27, 2007, 06:36:26 PM
If it's possible, could you post here some Win64 import libraries? I would like to have a look at them.

Hi Vortex,

I finally found how I'm gonna do the protos for win64. Because of the nature of the calling convention on win64 and how the stack is used, the more adequate way IMHO is to write "externdef function:proc".
In the download sample below, I included dnsapi.inc for win32/win64 like it will appear in Azimut Project. Also included for you Vortex, dnsapi.lib from Vista SDK, in case you still want to give a look at it.



[attachment deleted by admin]
Title: Re: Azimut Project
Post by: jdoe on July 09, 2008, 07:59:51 AM

Hi,

This project is not dead. I've been busy at work with pl/sql projects and I didn't had the spare time I would like.
I've been able lately to find time to write an installation program and add Windows 2008 to the package. The next step is to complete my h2inc program for adding constants and structures. The first version of Azimut include, for now, the import libraries and prototypes.


See first post for download link.

http://www.masm32.com/board/index.php?topic=7866.0


Title: Re: Azimut Project
Post by: jdoe on July 17, 2008, 01:41:14 AM

Sorry the download link didn't worked few days because I had problem with my web hosting lately. I've reset all the settings so now everything is fine. I had in mind to change hosting but quit the idea for now. Maybe later.

New version of Azimut including the resource header file that is needed by the resource compiler when you use style constants for example. I don't think you will find something missing   :bdg

See first post for download link.

http://www.masm32.com/board/index.php?topic=7866.0

Title: Re: Azimut Project
Post by: jdoe on July 21, 2008, 02:13:32 AM

Version 0.3 (Unicode support - TCHAR and String Macro)

With the TCHAR data type and the AzmtStr macro, you can build program like the example below and just comment or uncomment the UNICODE definition and the program is Unicode or not. Read the UNICODE.TXT from the README folder to be aware of some Unicode limitations.



.486
.MODEL FLAT,STDCALL
OPTION CASEMAP:NONE


UNICODE EQU 1


INCLUDE \AZIMUT\INCLUDE\AZIMUT.INC

INCLUDE \AZIMUT\INC32\kernel32.inc
INCLUDE \AZIMUT\INC32\user32.inc

INCLUDELIB \AZIMUT\LIB32\kernel32.lib
INCLUDELIB \AZIMUT\LIB32\user32.lib


.DATA

ALIGN 4
AzmtStr szTest1, TCHAR, <AzmtStr/0>
ALIGN 4
AzmtStr szTest2, TCHAR, <This is a >
AzmtStr        , TCHAR, <very >
AzmtStr        , TCHAR, <very >
AzmtStr        , TCHAR, <very >
AzmtStr        , TCHAR, <very >
AzmtStr        , TCHAR, <very >
AzmtStr        , TCHAR, <very >
AzmtStr        , TCHAR, <very >
AzmtStr        , TCHAR, <very >
AzmtStr        , TCHAR, <very >
AzmtStr        , TCHAR, <very >
AzmtStr        , TCHAR, <very >
AzmtStr        , TCHAR, <very long string/0>


.CODE

Main:

    invoke MessageBox, 0, addr szTest2, addr szTest1, 0

    invoke ExitProcess, 0

END Main




This will be the last release for a while. I'm working right now on the most important (constants and structures).

See first post for download link.

http://www.masm32.com/board/index.php?topic=7866.0

Title: Re: Azimut Project
Post by: jdoe on November 20, 2008, 02:17:50 AM

I need opinions about a new way to do include files. To solve a little problem with UNICODE, I thought about doing them like below.
Appending a 'A' to the functions that are present with a 'W' for Unicode and no 'A' for ANSI version will make the UNICODE more reliable in the way that it will not be necessary to know those exceptions. When using a function with both ANSI and WIDE version it will be assumed that it needs a trailing A or W even though in facts, in the import library, there is not (like Process32First in kernel32).

Is there inconvenients I did not saw ?




EXTERNDEF _imp__Process32First@8:PTR PROTO_STDCALL_2
Process32FirstA EQU <_imp__Process32First@8>
EXTERNDEF _imp__Process32FirstW@8:PTR PROTO_STDCALL_2
Process32FirstW EQU <_imp__Process32FirstW@8>
EXTERNDEF _imp__Process32Next@8:PTR PROTO_STDCALL_2
Process32NextA EQU <_imp__Process32Next@8>
EXTERNDEF _imp__Process32NextW@8:PTR PROTO_STDCALL_2
Process32NextW EQU <_imp__Process32NextW@8>
EXTERNDEF _imp__ProcessIdToSessionId@8:PTR PROTO_STDCALL_2
ProcessIdToSessionId EQU <_imp__ProcessIdToSessionId@8>
EXTERNDEF _imp__PulseEvent@4:PTR PROTO_STDCALL_1
PulseEvent EQU <_imp__PulseEvent@4>

IFNDEF UNICODE
Process32First EQU <_imp__Process32First@8>
Process32Next EQU <_imp__Process32Next@8>
ELSE
Process32First EQU <_imp__Process32FirstW@8>
Process32Next EQU <_imp__Process32NextW@8>
ENDIF



Title: Re: Azimut Project
Post by: gwapo on November 20, 2008, 06:12:53 AM
Hi jdoe,

Have you tried the H2INC utility of MASMv8 and up?
http://msdn.microsoft.com/en-us/library/aa903540(VS.71).aspx (http://msdn.microsoft.com/en-us/library/aa903540(VS.71).aspx)

I'm not sure if it's part of Visual Studio or not, but I found it in my VC++ installation.

Cheers,

-chris
Title: Re: Azimut Project
Post by: jdoe on November 20, 2008, 06:41:35 AM
Quote from: gwapo on November 20, 2008, 06:12:53 AM
Have you tried the H2INC utility of MASMv8 and up?
http://msdn.microsoft.com/en-us/library/aa903540(VS.71).aspx (http://msdn.microsoft.com/en-us/library/aa903540(VS.71).aspx)

I'm not sure if it's part of Visual Studio or not, but I found it in my VC++ installation.


AFAIK, H2INC is an old tool that is not working anymore with recent Windows SDK/DDK header files.

Title: Re: Azimut Project
Post by: Vortex on November 20, 2008, 06:55:17 PM
Hi jdoe,

You can try Japheth's h2incX tool :

QuoteAbout h2incx

This tool's purpose is to convert C header files to Masm-style include files. It is much more powerful than Microsoft's h2inc tool. The main purpose is to convert the Win32 include files, but it should work with any C header files as well. It is a simple Win32 console application, but a 32bit DOS extended binary version is included as well to be used on Non-Win32 platforms.

Features

    * a private profile file is used which allows fine-tuning of the include files to generate.
    * huge C header sets can be converted in one run (for example the Win32 headers contained in the PSDK).
    * prototypes may be written so the include file fits for both dynamic linking to a dll (using the IAT entries) and static linking to a library.
    * optionally a .DEF file is written, which can then be converted to an import library with POLIB.

http://japheth.de/h2incX.html

Title: Re: Azimut Project
Post by: jdoe on December 15, 2008, 12:46:55 PM

My question has been misunderstood. I would like to know the advantage of both these methods of writing function protypes because they are encoded differently. Size speaking, the difference is obvious because it seems that one is not using a jump table, but how about speed or what else ?  I've put two different ouput from 2 disassembler.


INVOKE ExitProcess,  0



PROTO_STDCALL_1 TYPEDEF PROTO STDCALL :DWORD
EXTERNDEF _imp__ExitProcess@4:PTR PROTO_STDCALL_1
ExitProcess EQU <_imp__ExitProcess@4>



:00401000 6A00             PUSH 0
:00401002 FF1500204000     CALL DWORD PTR [00402000]
--------------------------------------------------------------------------
:00401000 6A00             PUSH 0
:00401002 FF1500204000     CALL DWORD PTR DS:[<&kernel32.ExitProcess>]




VERSUS


ExitProcess PROTO STDCALL a1:DWORD



:00401000 6A00             PUSH 00000000
:00401002 E801000000       CALL 00401008
:00401008 FF2500204000     JMP  DWORD PTR [00402000]
--------------------------------------------------------------------------
:00401000 6A00             PUSH 0
:00401002 E801000000       CALL <JMP.&kernel32.ExitProcess>
:00401008 FF2500204000     JMP  DWORD PTR DS:[<&kernel32.ExitProcess>