mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
3244 lines
80 KiB
3244 lines
80 KiB
;; NEWHEADER structure is used in LookupIconIdFromDir
|
|
NEWHEADER struc
|
|
Reserved dw ?
|
|
ResType dw ?
|
|
ResCount dw ?
|
|
NEWHEADER ends
|
|
|
|
;; RESDIR structure is used in LookupIconIdFromDir
|
|
RESDIR struc
|
|
IconOrCursor dd ? ; IconDir or CursorDir structure.
|
|
Planes dw ?
|
|
BitCount dw ?
|
|
BytesInRes dd ?
|
|
idIcon dw ?
|
|
RESDIR ends
|
|
|
|
if 0
|
|
CHECKHINULL macro x
|
|
local chn_ex
|
|
ifdef DEBUG
|
|
push eax
|
|
mov eax, x
|
|
shr eax, 16
|
|
jz chn_ex
|
|
int 3
|
|
chn_ex:
|
|
pop eax
|
|
endif
|
|
endm
|
|
|
|
CHECKHISIGN macro x
|
|
local chs_ex
|
|
ifdef DEBUG
|
|
push eax
|
|
mov eax, x
|
|
shr eax, 16
|
|
inc ax
|
|
cmp ax, 1
|
|
jbe chs_ex
|
|
int 3
|
|
chs_ex:
|
|
pop eax
|
|
endif
|
|
endm
|
|
endif
|
|
|
|
|
|
CodeData equ <THUNK16CodeData>
|
|
|
|
ifdef IS_16
|
|
include thkrp.inc
|
|
include k32share.inc
|
|
include public.inc
|
|
include cbcheck.inc
|
|
include thkmacro.inc
|
|
include thkframe.inc
|
|
include struc.inc
|
|
include cbcid.inc
|
|
include usrcbid.inc
|
|
|
|
fLocalFlag equ fUsrLog16
|
|
WINABLE equ 1
|
|
|
|
TIMERINFO_TYPE struct
|
|
resolution DWORD ?
|
|
TIMERINFO_TYPE ends
|
|
|
|
|
|
MULTIKEYHELP32 STRUC
|
|
mk32_Size dd ?
|
|
mk32_Keylist db ?
|
|
mk32_szKeyPhrase db ?
|
|
MULTIKEYHELP32 ENDS
|
|
|
|
MULTIKEYHELP16 STRUC
|
|
mk16_Size dw ?
|
|
mk16_Keylist db ?
|
|
mk16_szKeyPhrase db ?
|
|
MULTIKEYHELP16 ENDS
|
|
|
|
HELPWININFO32 STRUC
|
|
hwi32_wStructSize dd ?
|
|
hwi32_x dd ?
|
|
hwi32_y dd ?
|
|
hwi32_dx dd ?
|
|
hwi32_dy dd ?
|
|
hwi32_wMax dd ?
|
|
hwi32_rgchMember db ?
|
|
HELPWININFO32 ENDS
|
|
|
|
HELPWININFO16 STRUC
|
|
hwi16_wStructSize dw ?
|
|
hwi16_x dw ?
|
|
hwi16_y dw ?
|
|
hwi16_dx dw ?
|
|
hwi16_dy dw ?
|
|
hwi16_wMax dw ?
|
|
hwi16_rgchMember db ?
|
|
HELPWININFO16 ENDS
|
|
|
|
FLASHWINFO16 STRUC
|
|
fw16_cbSize dw ?
|
|
fw16_hwnd dw ?
|
|
fw16_dwFlags dd ?
|
|
fw16_uCount dw ?
|
|
fw16_dwTimeout dd ?
|
|
FLASHWINFO16 ENDS
|
|
|
|
FLASHWINFO32 STRUC
|
|
fw32_cbSize dd ?
|
|
fw32_hwnd dd ?
|
|
fw32_dwFlags dd ?
|
|
fw32_uCount dd ?
|
|
fw32_dwTimeout dd ?
|
|
FLASHWINFO32 ENDS
|
|
|
|
externDef GetExePtr:far16
|
|
;;;externDef InsertPropAtom:far16
|
|
;;;externDef DeletePropAtom:far16
|
|
externDef GlobalAddAtom:far16
|
|
externDef GlobalDeleteAtom:far16
|
|
externDef GlobalFindAtom:far16
|
|
externDef ConvertDialogA:far16
|
|
externDef ConvertDialog:far16
|
|
externDef ConvertDialog32:far16
|
|
externDef CountDialogA:far16
|
|
externDef CountDialogU:far16
|
|
externDef ConvertMenuA:far16
|
|
externDef CountMenuA:far16
|
|
externDef CountMenuU:far16
|
|
externDef ConvertMenu32:far16
|
|
externDef GetNullhInst:far16
|
|
|
|
externDef GlobalAlloc:far16
|
|
externDef GlobalFree:far16
|
|
externDef GlobalLock:far16
|
|
externDef GlobalUnlock:far16
|
|
externDef GlobalFix:far16
|
|
externDef GlobalUnfix:far16
|
|
|
|
externDef MaphinstLS:far16
|
|
externDef MaphinstSL:far16
|
|
|
|
externDef IsClassNameMDICLIENT:far16
|
|
externDef ConvertDDEHandleLS:far16
|
|
externDef ConvertDDEHandleSL:far16
|
|
externDef QueryCallbackAddress:far16
|
|
externDef GetStdCBSL:far16
|
|
externDef GetStdCBLS:far16
|
|
externDef GetCurrentTask:far16 ;!!! temporary?
|
|
externDef GetCurrentThreadID:far16
|
|
externDef timerInfo:TIMERINFO_TYPE
|
|
externDef CALLBACK_BODY_16:far16
|
|
externDef GetCurrentHeap:far16
|
|
externDef ConvertMenuItemInfoLS:far16
|
|
|
|
externDef IsDialog:far16
|
|
externDef IsWindowClassType:far16
|
|
|
|
;;;externDef aUserData:dword
|
|
externDef Map32To16:far16
|
|
externDef DelMap:far16
|
|
externDef LoadLibrary32:far16
|
|
externDef ConvertCFDataLS:far16
|
|
externDef ConvertCFDataSL:far16
|
|
externDef IsThisADDEExecuteHandle:far16
|
|
externDef ConvertOleClipData:far16
|
|
|
|
externDef GetHCONVWindows:far16
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; DXAX2EAX
|
|
; mov dx to eax high word
|
|
;
|
|
;-----------------------------------------------------------------------;
|
|
DXAX2EAX macro
|
|
ror eax,16
|
|
xchg ax,dx
|
|
ror eax,16
|
|
endm
|
|
|
|
|
|
CF_LCID equ 16
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_SETCLIPBOARDDATA
|
|
;-----------------------------------------------------------------------;
|
|
body_SETCLIPBOARDDATA macro
|
|
local exit
|
|
local SetData
|
|
|
|
bp_nFormat equ <bp_top>
|
|
bp_dwData equ <bp_top+4>
|
|
|
|
mov eax, dword ptr bp_dwData
|
|
test eax, eax
|
|
jz SetData
|
|
|
|
push word ptr bp_nFormat
|
|
push eax
|
|
push word ptr 1 ;Nuke old copies of hData
|
|
call ConvertCFDataLS
|
|
movzx eax, ax
|
|
test ax, ax
|
|
jz exit
|
|
|
|
SetData:
|
|
push word ptr bp_nFormat
|
|
push ax
|
|
push 1
|
|
call SetClipboardData32
|
|
movzx eax, ax
|
|
|
|
test ax, ax
|
|
jz exit
|
|
|
|
mov eax, bp_dwData ;Return original hData
|
|
exit:
|
|
endm
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_GETCLIPBOARDDATA
|
|
;-----------------------------------------------------------------------;
|
|
body_GETCLIPBOARDDATA macro
|
|
local exit
|
|
bp_nFormat equ <bp_top>
|
|
|
|
push word ptr bp_nFormat
|
|
push 1
|
|
call GetClipboardData32
|
|
movzx eax, ax
|
|
test ax, ax
|
|
jz exit
|
|
|
|
push word ptr bp_nFormat
|
|
push ax ;hnd16
|
|
push word ptr 0 ;Don't nuke old copies of hData
|
|
call ConvertCFDataSL
|
|
|
|
; Return now in eax.
|
|
exit:
|
|
|
|
endm
|
|
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
;
|
|
; body_SETWINDOWSHOOKEXA
|
|
;
|
|
;-----------------------------------------------------------------------;
|
|
body_SETWINDOWSHOOKEXA macro
|
|
local type_ok
|
|
local get_out
|
|
local inserted_ok
|
|
local push_task
|
|
bp_nType equ <bp_top>
|
|
bp_pfn equ <bp_top+4>
|
|
bp_hmod equ <bp_top+8>
|
|
bp_hThread equ <bp_top+12>
|
|
|
|
mov di,word ptr bp_nType ;will need more than once
|
|
inc di ;make nType 0-based
|
|
cmp di,CBID_WH_END-CBID_WH_START
|
|
jb type_ok
|
|
;!!! SetExtendedError?
|
|
sub eax,eax ;error, invalid hook type
|
|
jmp get_out
|
|
|
|
type_ok:
|
|
dec di ;restore correct nType
|
|
push di ;push for SetWindowsHookEx
|
|
|
|
;map hook type onto callback type
|
|
movsx edi,di
|
|
add edi,CBID_WH_START+1 ;hook type --> callback type
|
|
push dword ptr bp_pfn ;32-bit callback address
|
|
push edi ;callback type
|
|
call GetStdCBSL ;create 16-bit callback stub
|
|
push eax
|
|
|
|
;
|
|
; Use module's or task's _real_ instance/module
|
|
; We need to do this in case bp_hmod is 0L, so we get current.
|
|
;
|
|
mov eax, dword ptr bp_hmod
|
|
call MaphInstLS
|
|
mov word ptr bp_hmod, ax
|
|
call MaphInstSL
|
|
push eax
|
|
|
|
push dword ptr bp_hThread
|
|
call SetWindowsHookEx32 ;make api call
|
|
DXAX2EAX
|
|
get_out:
|
|
endm
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_UNHOOKWINDOWSHOOK
|
|
;-----------------------------------------------------------------------;
|
|
body_UNHOOKWINDOWSHOOK macro
|
|
bp_nCode equ <bp_top>
|
|
bp_pfn equ <bp_top+4>
|
|
|
|
; Save nCode in eax for CBID_ math later
|
|
movsx eax, word ptr bp_nCode
|
|
push ax
|
|
|
|
push dword ptr bp_pfn
|
|
add eax, CBID_WH_START+1
|
|
push eax
|
|
call QueryCallbackAddress
|
|
push eax
|
|
|
|
call UnhookWindowsHook
|
|
cwde
|
|
endm
|
|
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_GETSETWINDOWDATA
|
|
;-----------------------------------------------------------------------;
|
|
body_GETSETWINDOWDATA macro
|
|
local exit
|
|
local Call_GSWD
|
|
local GSWD_return
|
|
local SWD_wndproc
|
|
local SWD_hinst
|
|
local GWD_wndproc
|
|
local GWD_hinst
|
|
local error
|
|
local apfnGet
|
|
local apfnSet
|
|
|
|
bp_hwnd equ <bp_top>
|
|
bp_nIndex equ <bp_top+4>
|
|
bp_dwValue equ <bp_top+8>
|
|
bp_nFlags equ <bp_top+12>
|
|
|
|
bp_isdialog equ <[bp-2]>
|
|
|
|
; Save space for bp_isdialog
|
|
xor ax, ax
|
|
push ax
|
|
|
|
mov bx, word ptr bp_nIndex
|
|
cmp bx, ax
|
|
jl @F
|
|
|
|
cmp bx, DWL_DLGPROC
|
|
jne Call_GSWD
|
|
|
|
; Is window really a dialog? If not, skip this!
|
|
push word ptr bp_hwnd
|
|
call IsDialog
|
|
mov word ptr bp_isdialog, ax
|
|
test ax, ax
|
|
jz Call_GSWD
|
|
|
|
test word ptr bp_nFlags, DATA_SET
|
|
jz Call_GSWD
|
|
jmp SWD_wndproc
|
|
|
|
@@:
|
|
cmp bx,GWL_USERDATA
|
|
je Call_GSWD
|
|
|
|
neg bx
|
|
add bx,GWL_WNDPROC
|
|
.errnz GWL_WNDPROC + 4
|
|
jl error
|
|
cmp bx,MAX_INDEX
|
|
ja error
|
|
test bl, 1
|
|
jnz error ;Don't allow odd indices.
|
|
|
|
test word ptr bp_nFlags, DATA_SET
|
|
jz Call_GSWD
|
|
jmp cs:apfnSet[bx]
|
|
|
|
apfnSet label word
|
|
dw offset SWD_wndproc ;0 ; -4 GWL_WNDPROC
|
|
dw offset SWD_hinst ;2 ; -6 GWL_HINSTANCE
|
|
dw offset Call_GSWD ;4 ; -8 GWL_HWNDPARENT
|
|
dw offset error ;6 ;-10
|
|
dw offset Call_GSWD ;8 ;-12 GWL_ID
|
|
dw offset error ;10 ;-14
|
|
dw offset Call_GSWD ;12 ;-16 GWL_STYLE
|
|
dw offset error ;14 ;-18
|
|
dw offset Call_GSWD ;16 ;-20 GWL_EXSTYLE
|
|
;; No need to add one more entry for -21 (GWL_USERDATA) because this is
|
|
;; the only Odd index we handle it as a special case eariler.
|
|
MAX_INDEX equ ($-apfnSet)
|
|
|
|
.errnz GWL_WNDPROC + 4
|
|
.errnz GWL_HINSTANCE + 6
|
|
.errnz GWL_HWNDPARENT + 8
|
|
.errnz GWL_ID + 12
|
|
.errnz GWL_STYLE + 16
|
|
.errnz GWL_EXSTYLE + 20
|
|
.errnz GWL_USERDATA + 21
|
|
|
|
.errnz GWW_HINSTANCE + 6
|
|
.errnz GWW_HWNDPARENT + 8
|
|
|
|
SWD_wndproc:
|
|
push dword ptr bp_dwValue
|
|
mov eax, CBID_WNDPROC
|
|
cmp word ptr bp_nIndex, DWL_DLGPROC
|
|
jne @F
|
|
|
|
; Is window really a dialog! If not, skip this.
|
|
mov eax, CBID_DLGPROC
|
|
@@:
|
|
push eax
|
|
call GetStdCBSL
|
|
mov dword ptr bp_dwValue, eax
|
|
jmp Call_GSWD
|
|
|
|
SWD_hinst:
|
|
mov eax, dword ptr bp_dwValue
|
|
call MaphinstLS
|
|
movzx eax, ax
|
|
mov dword ptr bp_dwValue, eax
|
|
; FALL THRU
|
|
|
|
Call_GSWD:
|
|
push word ptr bp_hwnd
|
|
push word ptr bp_nIndex
|
|
push dword ptr bp_dwValue
|
|
push word ptr bp_nFlags
|
|
call GetSetWindowData
|
|
DXAX2EAX
|
|
|
|
;
|
|
; Now thunk return value. Note that up above, we already did the
|
|
; sanity checking on indeces < 0, so we don't have to do it again.
|
|
;
|
|
mov bx, word ptr bp_nIndex
|
|
cmp bx, 0
|
|
jl @F
|
|
|
|
cmp bx, DWL_DLGPROC
|
|
jne GSWD_return
|
|
|
|
cmp word ptr bp_isdialog, 0
|
|
jz GSWD_return
|
|
|
|
jmp GWD_wndproc
|
|
|
|
@@:
|
|
cmp bx, GWL_USERDATA
|
|
je GSWD_return
|
|
|
|
neg bx
|
|
add bx,GWL_WNDPROC
|
|
jmp cs:apfnGet[bx]
|
|
|
|
apfnGet label word
|
|
dw offset GWD_wndproc ;0 ; -4 GWL_WNDPROC
|
|
dw offset GWD_hinst ;2 ; -6 GWL_HINSTANCE
|
|
dw offset GSWD_return ;4 ; -8 GWL_HWNDPARENT
|
|
dw offset error ;6 ;-10
|
|
dw offset GSWD_return ;8 ;-12 GWL_ID
|
|
dw offset error ;10 ;-14
|
|
dw offset GSWD_return ;12 ;-16 GWL_STYLE
|
|
dw offset error ;14 ;-18
|
|
dw offset GSWD_return ;16 ;-20 GWL_EXSTYLE
|
|
|
|
GWD_wndproc:
|
|
push eax
|
|
mov eax, CBID_WNDPROC
|
|
cmp word ptr bp_nIndex, DWL_DLGPROC
|
|
jne @F
|
|
mov eax, CBID_DLGPROC
|
|
|
|
@@:
|
|
push eax
|
|
call GetStdCBLS
|
|
jmp GSWD_return
|
|
|
|
GWD_hinst:
|
|
call MaphinstSL
|
|
jmp GSWD_return
|
|
|
|
error:
|
|
xor eax, eax
|
|
|
|
GSWD_return:
|
|
; Get rid of local word storage
|
|
add sp, 2
|
|
endm
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_GETSETCLASSDATA
|
|
;-----------------------------------------------------------------------;
|
|
body_GETSETCLASSDATA macro
|
|
local exit
|
|
local Call_GCSD
|
|
local SCD_ptr
|
|
local SCD_hinst
|
|
local SCD_wndproc
|
|
local GCD_ptr
|
|
local GCD_hinst
|
|
local GCD_wndproc
|
|
local error
|
|
local apfnSet
|
|
local apfnGet
|
|
local have_stub
|
|
local MAX_INDEX
|
|
|
|
bp_hwnd equ <bp_top>
|
|
bp_nIndex equ <bp_top+4>
|
|
bp_dwData equ <bp_top+8>
|
|
bp_nFlags equ <bp_top+12>
|
|
|
|
; We do this up front for index validation.
|
|
; Is index private (>= 0) or not in system range? If so, no thunking.
|
|
mov bx, word ptr bp_nIndex
|
|
cmp bx, 0
|
|
jge Call_GSCD
|
|
|
|
neg bx
|
|
add bx,GCL_MENUNAME
|
|
.errnz GCL_MENUNAME + 8
|
|
jl error
|
|
cmp bx,MAX_INDEX
|
|
jge error
|
|
test bl, 1
|
|
jnz error ;don't allow odd indices
|
|
|
|
test word ptr bp_nFlags, DATA_SET
|
|
jz Call_GSCD
|
|
jmp cs:apfnSet[bx]
|
|
|
|
apfnSet label word
|
|
dw offset SCD_ptr ;0 ; - 8 GCL_MENUNAME
|
|
dw offset Call_GSCD ;2 ; -10 GCL_HBRBACKGROUND
|
|
dw offset Call_GSCD ;4 ; -12 GCL_HCURSOR
|
|
dw offset Call_GSCD ;6 ; -14 GCL_HICON
|
|
dw offset SCD_hinst ;8 ; -16 GCL_HMODULE
|
|
dw offset Call_GSCD ;10 ; -18 GCL_CBWNDEXTRA
|
|
dw offset Call_GSCD ;12 ; -20 GCL_CBCLSEXTRA
|
|
dw offset error ;14
|
|
dw offset SCD_wndproc ;16 ; -24 GCL_WNDPROC
|
|
dw offset Call_GSCD ;18 ; -26 GCL_STYLE
|
|
dw offset error ;20
|
|
dw offset error ;22
|
|
dw offset Call_GSCD ;24 ; -32 GCW_ATOM
|
|
dw offset Call_GSCD ;26 ; -34 GCW_HICONSM
|
|
MAX_INDEX equ ($ - apfnSet)
|
|
|
|
.errnz 8 + GCL_MENUNAME
|
|
.errnz 10 + GCW_HBRBACKGROUND
|
|
.errnz 12 + GCW_HCURSOR
|
|
.errnz 14 + GCW_HICON
|
|
.errnz 16 + GCW_HMODULE
|
|
.errnz 18 + GCW_CBWNDEXTRA
|
|
.errnz 20 + GCW_CBCLSEXTRA
|
|
.errnz 24 + GCL_WNDPROC
|
|
.errnz 26 + GCL_STYLE
|
|
.errnz 32 + GCW_ATOM
|
|
.errnz 34 + GCW_HICONSM
|
|
|
|
|
|
SCD_ptr:
|
|
push dword ptr bp_dwData
|
|
call MapLS
|
|
mov dword ptr bp_dwData, eax
|
|
jmp Call_GSCD
|
|
|
|
SCD_wndproc:
|
|
push dword ptr bp_dwData
|
|
push dword ptr CBID_WNDPROC
|
|
call GetStdCBSL
|
|
mov dword ptr bp_dwData, eax
|
|
jmp Call_GSCD
|
|
|
|
SCD_hinst:
|
|
mov eax, dword ptr bp_dwData
|
|
call MaphinstLS
|
|
movzx eax, ax
|
|
mov dword ptr bp_dwData, eax
|
|
; FALL THRU
|
|
|
|
Call_GSCD:
|
|
push word ptr bp_hwnd
|
|
push word ptr bp_nIndex
|
|
push dword ptr bp_dwData
|
|
push word ptr bp_nFlags
|
|
call GetSetClassData
|
|
DXAX2EAX
|
|
|
|
;
|
|
; Now thunk return value. Note that up above, we already did the
|
|
; sanity checking on indeces < 0, so we don't have to do it again.
|
|
;
|
|
mov bx, word ptr bp_nIndex
|
|
cmp bx, 0
|
|
jge GSCD_return
|
|
|
|
neg bx
|
|
add bx,GCL_MENUNAME
|
|
jmp cs:apfnGet[bx]
|
|
|
|
apfnGet label word
|
|
dw offset GCD_ptr ;0 ; - 8 GCL_MENUNAME
|
|
dw offset GSCD_return ;2 ; -10 GCW_HBRBACKGROUND
|
|
dw offset GSCD_return ;4 ; -12 GCW_HCURSOR
|
|
dw offset GSCD_return ;6 ; -14 GCW_HICON
|
|
dw offset GCD_hinst ;8 ; -16 GCW_HMODULE
|
|
dw offset GSCD_return ;10 ; -18 GCW_CBWNDEXTRA
|
|
dw offset GSCD_return ;12 ; -20 GCW_CBCLSEXTRA
|
|
dw offset error ;14
|
|
dw offset GCD_wndproc ;16 ; -24 GCL_WNDPROC
|
|
dw offset GSCD_return ;18 ; -26 GCL_STYLE
|
|
dw offset error ;20
|
|
dw offset error ;22
|
|
dw offset GSCD_return ;24 ; -32 GCW_ATOM
|
|
dw offset GSCD_return ;26 ; -34 GCW_HICONSM
|
|
|
|
GCD_ptr:
|
|
; This is OK.
|
|
; User's DS doesn't move in linear memory. And menu strings are
|
|
; LocalAlloced() out of User's DS.
|
|
push eax
|
|
call MapSL
|
|
jmp GSCD_return
|
|
|
|
GCD_hinst:
|
|
call MaphinstSL
|
|
jmp GSCD_return
|
|
|
|
GCD_wndproc:
|
|
push eax
|
|
push dword ptr CBID_WNDPROC
|
|
call GetStdCBLS
|
|
jmp GSCD_return
|
|
|
|
error:
|
|
xor eax, eax
|
|
|
|
GSCD_return:
|
|
endm
|
|
|
|
|
|
|
|
|
|
;-------------------------------------------------------------------------
|
|
; DdeAccessData(), DdeUnAccessData()
|
|
;
|
|
; HACK! DDEML stores the data in movable Win16 global handles, so
|
|
; *we* have to fix the handle whenever the app holds a linear address to it.
|
|
; This means groveling around in ddeml's data structures to retrieve
|
|
; the global handle from the DDE data handle.
|
|
;
|
|
; No, we can't just copy the data either. DdeAccessData() returns a
|
|
; read/write pointer.
|
|
;
|
|
; BUGBUG! Doesn't repack metafiles yet. For M6, write a support routine
|
|
; inside ddeml to handle all this stuff, and get rid of this hack.
|
|
;
|
|
;-------------------------------------------------------------------------
|
|
|
|
HDDEDATA_GHND equ 4
|
|
|
|
PUSH__DDEACCESSDATA_hData macro iOffset, iJunk
|
|
|
|
push es
|
|
les bx, [bp + iOffset]
|
|
mov ax, es:[bx + HDDEDATA_GHND]
|
|
pop es
|
|
push ax
|
|
call GlobalFix
|
|
|
|
push dword ptr [bp + iOffset]
|
|
|
|
|
|
endm ;PUSH__DDEACCESSDATA_hData
|
|
|
|
|
|
PUSH__DDEUNACCESSDATA_hData macro iOffset, iJunk
|
|
|
|
push es
|
|
les bx, [bp + iOffset]
|
|
mov ax, es:[bx + HDDEDATA_GHND]
|
|
pop es
|
|
push ax
|
|
call GlobalUnFix
|
|
|
|
push dword ptr [bp + iOffset]
|
|
|
|
|
|
endm ;PUSH__DDEUNACCESSDATA_hData
|
|
|
|
|
|
|
|
|
|
|
|
;***************************************************************************
|
|
; PackDDELParam and friends.
|
|
;
|
|
; WARNING: A lot of thunking depends on the fact that PackDDELParam requires
|
|
; no explicit cleanup whatsoever. If you're thinking of changing that,
|
|
; think again.
|
|
;
|
|
; The value returned by PackDDELParam is a ready-to-be-used Win16
|
|
; DDE lparam. The message thunks pass this through with no further
|
|
; translation (*). Through the magic of handle grouping, FreeDDELParam
|
|
; has to do absolutely nothing.
|
|
;
|
|
; (*) Except for WM_DDE_EXECUTE. It will translate the handle in the
|
|
; message thunk instead of in PackDDELParam. This allows Win32 apps
|
|
; to neglect PackDDELParam for WM_DDE_EXECUTE: which works on both
|
|
; NT and Win32s. (They can't ignore it on the WM_DDE_ACK, however.
|
|
; NT & Win32s don't allow this either.)
|
|
;
|
|
;***************************************************************************
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_PACKDDELPARAM
|
|
;-----------------------------------------------------------------------;
|
|
body_PACKDDELPARAM macro
|
|
|
|
local wMsg, dwLo, dwHi
|
|
|
|
local exit
|
|
local pack_execute
|
|
local pack_default
|
|
local pack_ddeack, pack_maplo
|
|
|
|
wMsg equ [bp_top]
|
|
dwLo equ [bp_top+4]
|
|
dwHi equ [bp_top+8]
|
|
|
|
|
|
|
|
mov ax,word ptr wMsg
|
|
cmp ax, WM_DDE_EXECUTE
|
|
je pack_execute
|
|
|
|
cmp ax, WM_DDE_ACK
|
|
je pack_mapddeack
|
|
cmp ax, WM_DDE_ADVISE
|
|
je pack_maplo
|
|
cmp ax, WM_DDE_DATA
|
|
je pack_maplo
|
|
cmp ax, WM_DDE_POKE
|
|
je pack_maplo
|
|
|
|
; WM_DDE_REQUEST or WM_DDE_UNADVISE
|
|
;
|
|
; Both dwLo & dwHi are 16-bit significant only, so just mush them
|
|
; together. Note that a Win32 app could get away without calling
|
|
; PackDDELParam for these messages. But NT and Win32s support this,
|
|
; so we will too.
|
|
pack_default:
|
|
mov ax, word ptr dwHi
|
|
shl eax,16
|
|
mov ax, word ptr dwLo
|
|
jmp exit
|
|
|
|
|
|
; WM_DDE_EXECUTE: Just pass it along. Message thunk must do handle
|
|
; conversion. We'd prefer to do the conversion here but then, an app
|
|
; is forced to use PackDDElParam. But NT and Win32s allow them to
|
|
; neglect that -- so we must too.
|
|
pack_execute:
|
|
mov eax, dword ptr dwHi
|
|
jmp exit
|
|
|
|
; WM_DDE_ACK:
|
|
;
|
|
; Hi dword is either a Win32 global handle or an atom. Fortunately,
|
|
; we can distinguish the two by examining the high word.
|
|
;
|
|
;
|
|
pack_mapddeack:
|
|
cmp word ptr dwHi+2, 0 ;Handle or atom?
|
|
je pack_default
|
|
|
|
push word ptr wMsg
|
|
push dword ptr dwHi
|
|
call ConvertDDEHandleLS ;This routine requires no cleanup
|
|
movzx eax,ax
|
|
or ax,ax
|
|
jz exit
|
|
shl eax,16
|
|
mov ax, word ptr dwLo ;ax == status word
|
|
jmp exit
|
|
|
|
; WM_DDE_ADVISE, WM_DDE_DATA, WM_DDE_POKE
|
|
pack_maplo:
|
|
push word ptr wMsg
|
|
push dword ptr dwLo
|
|
call ConvertDDEHandleLS ;This routine requires no cleanup
|
|
movzx eax,ax
|
|
or ax,ax
|
|
jz exit
|
|
mov cx,ax
|
|
mov ax, word ptr dwHi ;ax == status word
|
|
shl eax,16
|
|
mov ax,cx
|
|
;Fall thru to exit
|
|
|
|
exit:
|
|
endm ;body_PACKDDELPARAM
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_UNPACKDDELPARAM
|
|
;-----------------------------------------------------------------------;
|
|
body_UNPACKDDELPARAM macro
|
|
local exit
|
|
local unpack_execute
|
|
local unpack_default
|
|
local unpack_ddeack, unpack_maplo
|
|
local storeit
|
|
local didhi,didlo
|
|
|
|
local wMsg,dwLParam,puLo,puHi
|
|
wMsg equ [bp_top]
|
|
dwLParam equ [bp_top+4]
|
|
puLo equ [bp_top+8]
|
|
puHi equ [bp_top+12]
|
|
|
|
|
|
mov ax,word ptr wMsg
|
|
cmp ax, WM_DDE_EXECUTE
|
|
je unpack_execute
|
|
|
|
cmp ax, WM_DDE_ACK
|
|
je unpack_ddeack
|
|
cmp ax, WM_DDE_ADVISE
|
|
je unpack_maplo
|
|
cmp ax, WM_DDE_DATA
|
|
je unpack_maplo
|
|
cmp ax, WM_DDE_POKE
|
|
je unpack_maplo
|
|
|
|
; WM_DDE_REQUEST or WM_DDE_UNADVISE
|
|
;
|
|
; Both dwLo & dwHi are 16-bit significant only, so just mush them
|
|
; together. Note that a Win32 app could get away without calling
|
|
; PackDDELParam for these messages. But NT and Win32s support this,
|
|
; so we will too.
|
|
unpack_default:
|
|
mov eax,dwLParam
|
|
mov edx,eax
|
|
and eax,0ffffh ;EAX = lo word
|
|
shr edx,16 ;EDX = hi word
|
|
jmp storeit
|
|
|
|
|
|
; WM_DDE_EXECUTE: Message thunk already converted handle. Put it back
|
|
; in puHi. We'd prefer to the conversion here but then, an app
|
|
; is forced to use PackDDElParam. But NT and Win32s allow them to
|
|
; neglect that -- so we must too.
|
|
unpack_execute:
|
|
mov edx, dwLParam
|
|
xor eax,eax
|
|
jmp storeit
|
|
|
|
; WM_DDE_ACK:
|
|
; Hi-word is either a global Win16 handle or atom.
|
|
;
|
|
; No easy way tell which. Pass the buck to kernel.
|
|
unpack_ddeack:
|
|
push dword ptr dwLParam
|
|
call IsThisADDEExecuteHandle
|
|
or ax,ax
|
|
jz unpack_default
|
|
|
|
; Yes, this code is unused for M5 but it's tested and we *will* be using
|
|
; it for M6 so don't you dare delete it!
|
|
push word ptr wMsg
|
|
push word ptr dwLParam+2
|
|
call ConvertDDEHandleSL ;This routine requires no cleanup
|
|
test eax,eax
|
|
jz exit
|
|
mov edx,eax
|
|
movzx eax, word ptr dwLParam
|
|
jmp storeit
|
|
|
|
|
|
; WM_DDE_ADVISE, WM_DDE_DATA, WM_DDE_POKE
|
|
unpack_maplo:
|
|
push word ptr wMsg
|
|
push word ptr dwLParam
|
|
call ConvertDDEHandleSL ;This routine requires no cleanup
|
|
test eax,eax
|
|
jz exit
|
|
movzx edx, word ptr dwLParam+2
|
|
; Fall through to storeit
|
|
|
|
storeit:
|
|
mov ds,FlatData ; ds:esi == source address
|
|
mov edi, puLo
|
|
or edi, edi
|
|
jz didlo
|
|
mov [edi],eax
|
|
didlo:
|
|
mov edi, puHi
|
|
or edi, edi
|
|
jz didhi
|
|
mov [edi],edx
|
|
didhi:
|
|
mov eax,1
|
|
jmp exit
|
|
|
|
unpack_badparam:
|
|
xor eax,eax
|
|
;Fall thru to exit
|
|
|
|
exit:
|
|
endm
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_FREEDDELPARAM
|
|
;
|
|
; Through the magic of handle grouping, no action is required here.
|
|
; If you ever change this, you'll also have to make the message thunks
|
|
; thunks clean up temporary lparams.
|
|
;-----------------------------------------------------------------------;
|
|
body_FREEDDELPARAM macro
|
|
xor eax,eax
|
|
inc eax
|
|
endm
|
|
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_DDEQUERYCONVINFO
|
|
;-----------------------------------------------------------------------;
|
|
body_DDEQUERYCONVINFO macro
|
|
local next, L1, L2
|
|
local bp_hConv, bp_idTransaction, bp_lpConvInfo
|
|
local bp_hwnd, bp_hwndPartner;
|
|
|
|
bp_hConv equ <bp_top>
|
|
bp_idTransaction equ <bp_top+4>
|
|
bp_lpConvInfo equ <bp_top+8>
|
|
|
|
bp_hwnd equ word ptr [bp-6]
|
|
bp_hwndPartner equ word ptr [bp-8]
|
|
|
|
|
|
|
|
xor eax,eax
|
|
push eax ; ptr param #1 pConvInfo
|
|
push eax ; store hwnd field
|
|
push eax ; store hwndPartner field
|
|
|
|
; pConvInfo
|
|
; pointer struct CONVINFO32 --> struct CONVINFO16
|
|
cld ; esi, edi will increment
|
|
|
|
sub sp,CONVINFO16_SIZE ; pConvInfo alloc space on stack
|
|
|
|
; different pointer types
|
|
mov eax,DWORD ptr bp_lpConvInfo ; base address
|
|
test eax,eax
|
|
jz L1 ; skip if null
|
|
|
|
|
|
; structures are not identical
|
|
; structures don't have pointers
|
|
|
|
mov [bp-4],sp ; save offset to buffer
|
|
mov [bp-2],ss ; save selector to buffer
|
|
|
|
mov di,ss
|
|
mov es,di
|
|
movzx edi,sp ; es:edi == destination address
|
|
|
|
mov esi,eax
|
|
mov ds,FlatData
|
|
|
|
ncopyd 7
|
|
ncopyt 5
|
|
copyd
|
|
|
|
lodsd ds:[esi]
|
|
mov ax, CONVCONTEXT16_SIZE
|
|
stosw
|
|
|
|
ncopyt 3
|
|
ncopyd 2
|
|
L1:
|
|
|
|
; *** END parameter packing
|
|
;-------------------------------------
|
|
; create new call frame and make the call
|
|
|
|
; hConv from: unsigned long
|
|
push dword ptr bp_hConv ; to unsigned long
|
|
|
|
; idTransaction from: unsigned long
|
|
push dword ptr bp_idTransaction ; to unsigned long
|
|
|
|
; pConvInfo from: struct CONVINFO32
|
|
mov eax,[bp-4]
|
|
push eax
|
|
test eax,eax
|
|
jz next
|
|
|
|
movzx eax,ax
|
|
mov word ptr ss:[eax].ci16_cb,CONVINFO16_SIZE
|
|
mov word ptr ss:[eax].ci16_ConvCtxt.cc16_cb,CONVCONTEXT16_SIZE
|
|
next:
|
|
call DdeQueryConvInfo ; call 16-bit version
|
|
movzx eax,ax
|
|
test eax,eax
|
|
jz L2
|
|
|
|
;-------------------------------------
|
|
; *** BEGIN parameter unpacking
|
|
|
|
cld ; esi, edi will increment
|
|
|
|
; pConvInfo
|
|
; pointer struct CONVINFO16 --> struct CONVINFO32
|
|
mov eax,DWORD ptr bp_lpConvInfo ; base address
|
|
test eax,eax
|
|
jz L2 ; skip if null
|
|
|
|
|
|
; structures are not identical
|
|
; structures don't have pointers
|
|
|
|
mov si, seg FlatData
|
|
mov es, si
|
|
mov es, es:[FlatData]
|
|
mov edi,eax ; es:edi == destination address
|
|
mov si,ss
|
|
mov ds,si
|
|
movzx esi,word ptr [bp-4] ; ds:esi == source address
|
|
|
|
; cb
|
|
; unsigned long --> unsigned long
|
|
mov dword ptr ds:[esi], CONVINFO32_SIZE
|
|
|
|
ncopyd 7
|
|
ncopyzx 5
|
|
copyd
|
|
|
|
mov word ptr ds:[esi], CONVCONTEXT32_SIZE
|
|
ncopyzx 3
|
|
|
|
copysx
|
|
|
|
ncopyd 2
|
|
|
|
|
|
xor eax,eax
|
|
|
|
; Put dummy bytes into SECURITY_QUALITY_OF_SERVICE
|
|
mov eax, 0ch ;SIZEOF(SECURITY_QUALITY_OF_SERVICE)
|
|
stosd es:[edi]
|
|
xor eax,eax
|
|
stosd es:[edi]
|
|
stosd es:[edi]
|
|
|
|
push es
|
|
push edi
|
|
|
|
push dword ptr bp_hConv
|
|
mov ax, ss
|
|
push ax
|
|
lea cx, bp_hwnd
|
|
push cx
|
|
mov ax, ss
|
|
push ax
|
|
lea cx, bp_hwndPartner
|
|
push cx
|
|
call GetHCONVWindows
|
|
|
|
pop edi
|
|
pop es
|
|
|
|
movzx eax,ax
|
|
or ax,ax
|
|
jz L2
|
|
|
|
movzx eax, bp_hwnd
|
|
stosd es:[edi]
|
|
movzx eax, bp_hwndPartner
|
|
stosd es:[edi]
|
|
|
|
|
|
|
|
mov eax, CONVINFO32_SIZE
|
|
|
|
L2:
|
|
endm
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------------
|
|
;
|
|
; GetClassInfoExA()
|
|
;
|
|
;----------------------------------------------------------------------------
|
|
body_GETCLASSINFOEXA macro
|
|
local L238
|
|
local PushInstance
|
|
local PushClassName
|
|
local StoreWndProc
|
|
local StoreMenuName
|
|
local Cleanup
|
|
|
|
bp_hInstance equ <bp_top + 0>
|
|
bp_lpszClassName equ <bp_top + 4>
|
|
bp_lpwc32 equ <bp_top + 8>
|
|
|
|
bp_lpszClassNameTemp equ <[bp - 4]>
|
|
|
|
; Skip this if lpsz32 or lpwc32 is NULL
|
|
mov eax, bp_lpszClassName
|
|
test eax,eax
|
|
jz L238 ; skip if null
|
|
mov eax, bp_lpwc32
|
|
test eax, eax
|
|
jz L238
|
|
|
|
; Save space for lpszClassNameTemp
|
|
xor eax, eax
|
|
push eax
|
|
|
|
; Get lpWndClassEx16
|
|
sub sp,WNDCLASSEX16_SIZE ; lpWndClassEx alloc space on stack
|
|
movzx esi,sp ; ds:esi == destination address
|
|
mov dword ptr ss:[esi], WNDCLASSEX16_SIZE
|
|
|
|
;
|
|
; Push 16-bit parameters
|
|
;
|
|
|
|
; hInstance: can be NULL (for system class)
|
|
mov eax, bp_hInstance
|
|
test eax, eax
|
|
jz PushInstance
|
|
call MaphInstLS
|
|
PushInstance:
|
|
push ax
|
|
|
|
; lpszClassName: can be atom or string
|
|
push dword ptr bp_lpszClassName
|
|
call MapLS
|
|
mov bp_lpszClassNameTemp, eax
|
|
push eax
|
|
|
|
; lpwc16
|
|
push ss
|
|
push si
|
|
|
|
;
|
|
; Make 16-bit call
|
|
; ZERO-EXTEND return value. Some apps know it's not really a BOOL,
|
|
; it's an ATOM and save the entire DWORD away to compare later.
|
|
;
|
|
call GetClassInfoEx
|
|
movzx eax, ax
|
|
|
|
; Save return value
|
|
push eax
|
|
test eax, eax
|
|
jz Cleanup
|
|
|
|
;
|
|
; Convert WNDCLASSEX from 16 to 32
|
|
;
|
|
mov es, FlatData
|
|
mov edi, bp_lpwc32
|
|
|
|
push ds
|
|
mov ax, ss
|
|
mov ds, ax
|
|
cld
|
|
|
|
;;Skip size
|
|
add esi, 4
|
|
add edi, 4
|
|
|
|
; style
|
|
copyd
|
|
|
|
; lpfnWndProc
|
|
lodsd ds:[esi]
|
|
push es
|
|
push eax
|
|
push dword ptr CBID_WNDPROC
|
|
call GetStdCBLS
|
|
pop es
|
|
stosd es:[edi]
|
|
|
|
; cbClsExtra, cbWndExtra
|
|
ncopyzx 2
|
|
|
|
; hInstance
|
|
lodsw ds:[esi] ; hInstance
|
|
mov eax, bp_hInstance
|
|
stosd es:[edi]
|
|
|
|
; hIcon, hCursor, hbrBackground
|
|
ncopyzx 3
|
|
|
|
; lpszMenuName
|
|
lodsd ds:[esi]
|
|
push eax
|
|
call MapSL ;BUGBUG! GetClassInfo
|
|
stosd es:[edi]
|
|
|
|
; lpszClassName: Just use what was passed in, just like User16 does!
|
|
lodsd ds:[esi]
|
|
mov eax, dword ptr bp_lpszClassName
|
|
stosd es:[edi]
|
|
|
|
; hIconSm
|
|
copyzx
|
|
|
|
pop ds
|
|
|
|
Cleanup:
|
|
pushd bp_lpszClassNameTemp
|
|
call UnmapLS
|
|
pop eax
|
|
L238:
|
|
endm
|
|
|
|
|
|
|
|
;===========================================
|
|
; BOGUS
|
|
; Same code for mapping instances everywhere
|
|
;===========================================
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; PUSH__DDECONNECT_pCC
|
|
;-----------------------------------------------------------------------;
|
|
PUSH__DDECONNECT_pCC macro iOffset, iTempOffset
|
|
local done
|
|
|
|
mov eax,[bp-iTempOffset]
|
|
push eax
|
|
test eax,eax
|
|
jz done
|
|
|
|
movzx eax,ax
|
|
mov word ptr ss:[eax].cc16_cb,CONVCONTEXT16_SIZE
|
|
done:
|
|
endm
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; PUSH__DDECONNECTLIST_pCC
|
|
;-----------------------------------------------------------------------;
|
|
PUSH__DDECONNECTLIST_pCC macro iOffset, iTempOffset
|
|
local done
|
|
|
|
mov eax,[bp-iTempOffset]
|
|
push eax
|
|
test eax,eax
|
|
jz done
|
|
|
|
movzx eax,ax
|
|
mov word ptr ss:[eax].cc16_cb,CONVCONTEXT16_SIZE
|
|
done:
|
|
endm
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_CREATEWINDOWEXA
|
|
;-----------------------------------------------------------------------;
|
|
body_CREATEWINDOWEXA macro
|
|
local null_ptr1
|
|
local store_ptr1
|
|
local null_ptr2
|
|
local have_x
|
|
local have_width
|
|
local ismdi
|
|
local isnotmdi
|
|
local done_with_lpParam
|
|
|
|
BODY_PARAM_16 bp_dwExStyle, 0
|
|
BODY_PARAM_16 bp_lpClassName, 4
|
|
BODY_PARAM_16 bp_lpWindowName, 8
|
|
BODY_PARAM_16 bp_dwStyle, 12
|
|
BODY_PARAM_16 bp_X, 16
|
|
BODY_PARAM_16 bp_Y, 20
|
|
BODY_PARAM_16 bp_nWidth, 24
|
|
BODY_PARAM_16 bp_nHeight, 28
|
|
BODY_PARAM_16 bp_hWndParent, 32
|
|
BODY_PARAM_16 bp_hMenu, 36
|
|
BODY_PARAM_16 bp_hInstance, 40
|
|
BODY_PARAM_16 bp_lpParam, 44
|
|
|
|
bp_lpClassNameTemp equ <[bp-4]>
|
|
bp_lpWindowNameTemp equ <[bp-8]>
|
|
bp_lpParamTemp equ <[bp-12]>
|
|
bp_couldbemdi equ <[bp-14]>
|
|
; temp storage
|
|
|
|
xor eax,eax
|
|
push eax ; ptr param #1 lpClassNameTemp
|
|
push eax ; ptr param #2 lpWindowNameTemp
|
|
push eax ; ptr param #3 lpParamTemp
|
|
inc ax
|
|
push ax ; wrd param #4 couldbemdi (set to 1)
|
|
|
|
mov cx, word ptr bp_lpClassName+2
|
|
jcxz notmdi ; Passed in an atom?
|
|
|
|
mov eax, dword ptr bp_lpClassName
|
|
mov es, FlatData
|
|
mov eax, es:[eax]
|
|
cmp eax, 'CIDM' ; Check for MDIC
|
|
je ismdi
|
|
cmp eax, 'cidm' ; and in lower case...
|
|
je ismdi
|
|
notmdi:
|
|
dec word ptr bp_couldbemdi ; dec to 0
|
|
ismdi:
|
|
|
|
push dword ptr bp_lpClassName
|
|
call MapLS ;PLUGGED
|
|
mov bp_lpClassNameTemp,eax
|
|
|
|
push dword ptr bp_lpWindowName
|
|
call MapLS ;PLUGGED
|
|
mov bp_lpWindowNameTemp,eax
|
|
|
|
; lpParam is pointer to CLIENTCREATESTRUCT if class is mdiclient, otherwise
|
|
; it's just a dword.
|
|
|
|
mov cx, word ptr bp_couldbemdi
|
|
jcxz isnotmdi
|
|
push dword ptr bp_lpClassNameTemp
|
|
call IsClassNameMDICLIENT
|
|
test ax,ax
|
|
jnz CreateMdi
|
|
|
|
isnotmdi:
|
|
push dword ptr bp_lpParam
|
|
call MapLS
|
|
mov bp_lpParamTemp, eax
|
|
mov bp_lpParam, eax
|
|
jmp done_with_lpParam
|
|
|
|
; lpParam points to CLIENTCREATESTRUCT, so repack it and update
|
|
; bp_lpParamTemp
|
|
|
|
CreateMdi:
|
|
;CLIENTCREATESTRUCT:
|
|
; HWND
|
|
; UINT
|
|
|
|
mov es, FlatData
|
|
mov ecx,dword ptr bp_lpParam
|
|
push word ptr es:[ecx+4] ;truncate UINT
|
|
push word ptr es:[ecx] ;truncate HWND
|
|
mov bp_lpParam[0],sp ;store new pointer to struct
|
|
mov bp_lpParam[2],ss
|
|
|
|
done_with_lpParam:
|
|
|
|
push dword ptr bp_dwExStyle
|
|
push dword ptr bp_lpClassNameTemp
|
|
push dword ptr bp_lpWindowNameTemp
|
|
push dword ptr bp_dwStyle
|
|
|
|
; x, y, cx, cy
|
|
mov eax, dword ptr bp_X
|
|
cmp eax, 80000000h ; CW_USEDEFAULT
|
|
jne have_x
|
|
ror eax,16 ; map to 8000h
|
|
have_x:
|
|
push ax
|
|
|
|
mov eax, dword ptr bp_Y
|
|
cmp eax, 80000000h
|
|
jne have_y
|
|
ror eax,16
|
|
have_y:
|
|
push ax
|
|
|
|
mov eax,dword ptr bp_nWidth
|
|
cmp eax,80000000h ; CW_USEDEFAULT
|
|
jne have_cx
|
|
ror eax,16 ; map to 8000h
|
|
have_cx:
|
|
push ax
|
|
|
|
mov eax,dword ptr bp_nHeight
|
|
cmp eax,80000000h
|
|
jne have_cy
|
|
ror eax,16
|
|
have_cy:
|
|
push ax
|
|
|
|
push word ptr bp_hWndParent
|
|
|
|
; push LOWORD of hMenu
|
|
push word ptr bp_hMenu.lo
|
|
|
|
; hInstance
|
|
MAP_NULL_HINST bp_hInstance
|
|
push ax
|
|
|
|
; lpParams
|
|
push dword ptr bp_lpParam
|
|
|
|
; push HIWORD of hMenu
|
|
push word ptr bp_hMenu.hi
|
|
|
|
call CreateWindowEx32 ; call 16-bit version
|
|
movzx eax,ax
|
|
|
|
push eax
|
|
pushd bp_lpParamTemp
|
|
call UnmapLS
|
|
pushd bp_lpClassNameTemp
|
|
call UnmapLS
|
|
pushd bp_lpWindowNameTemp
|
|
call UnmapLS
|
|
pop eax
|
|
endm
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_DDECLIENTTRANSACTION
|
|
;-----------------------------------------------------------------------;
|
|
body_DDECLIENTTRANSACTION macro
|
|
local dometafile
|
|
local null_ptr1
|
|
local null_ptr2
|
|
local dct_leave
|
|
|
|
BODY_PARAM_16 bp_pData, 0
|
|
BODY_PARAM_16 bp_cbData, 4
|
|
BODY_PARAM_16 bp_hConv, 8
|
|
BODY_PARAM_16 bp_hszItem, 12
|
|
BODY_PARAM_16 bp_wFmt, 16
|
|
BODY_PARAM_16 bp_wType, 20
|
|
BODY_PARAM_16 bp_dwTimeout, 24
|
|
BODY_PARAM_16 bp_pdwResult, 28
|
|
|
|
bp_pDataTemp equ <[bp-4]>
|
|
bp_pdwResultTemp equ <[bp-8]>
|
|
bp_pDataTempSel equ <[bp-12]>
|
|
|
|
xor eax,eax
|
|
push dword ptr bp_pData ; bp_pDataTemp: ptr param #1 pData
|
|
push eax ; bp_pdwResultTemp: ptr param #2 pdwResult
|
|
push eax ; bp_pDataTempSel: Sel for pData
|
|
|
|
;If cbData == -1, then pData is a handle, not a pointer.
|
|
cmp dword ptr bp_cbData, -1
|
|
jz null_ptr1
|
|
cmp dword ptr bp_pData, 0
|
|
jz null_ptr1
|
|
|
|
|
|
cmp word ptr bp_wFmt, CF_METAFILEPICT
|
|
je dometafile
|
|
|
|
push dword ptr bp_pData
|
|
call MapLS ;PLUGGED
|
|
mov bp_pDataTemp,eax
|
|
mov bp_pDataTempSel,eax
|
|
jmp null_ptr1
|
|
|
|
dometafile:
|
|
mov ds, FlatData
|
|
mov esi, bp_pData
|
|
push word ptr CF_METAFILEPICT
|
|
push dword ptr [esi]
|
|
push word ptr 0
|
|
call ConvertCFDataLS
|
|
movzx eax,ax
|
|
or ax,ax
|
|
jz dct_leave
|
|
|
|
; We'll use bp_pData as our (2-byte) substitute data buffer.
|
|
mov bp_pData,ax
|
|
mov word ptr bp_cbData,2
|
|
mov ax,ss
|
|
mov bp_pDataTemp,ax
|
|
lea di,bp_pData
|
|
mov bp_pDataTemp+2,di
|
|
|
|
null_ptr1:
|
|
|
|
push dword ptr bp_pdwResult
|
|
call MapLS ;PLUGGED
|
|
mov bp_pdwResultTemp,eax
|
|
null_ptr2:
|
|
|
|
push dword ptr bp_pDataTemp
|
|
push dword ptr bp_cbData
|
|
push dword ptr bp_hConv
|
|
push dword ptr bp_hszItem
|
|
push word ptr bp_wFmt
|
|
push word ptr bp_wType
|
|
push dword ptr bp_dwTimeout
|
|
push dword ptr bp_pdwResultTemp
|
|
|
|
call DDECLIENTTRANSACTION
|
|
DXAX2EAX
|
|
|
|
dct_leave:
|
|
; We can safely unmap these. The api does not hold either pointer
|
|
; after it returns.
|
|
push eax
|
|
push dword ptr bp_pDataTempSel
|
|
call UnMapLS
|
|
push dword ptr bp_pdwResultTemp
|
|
call UnMapLS
|
|
|
|
|
|
IF 0
|
|
;!!! BUGBUG: The following is a gross M5 workaround for B#9814.
|
|
;!!! DdeClientTransaction takes our ss and stuffs it in a long-term data
|
|
;!!! structure. To prevent a subsequent GP fault, we'll replace ss
|
|
;!!! with a new one so that the thunk dispatcher won't unmap the one
|
|
;!!! that DdeClientTransaction so rudely took.
|
|
mov ax,ss
|
|
push ax
|
|
push word ptr 0
|
|
call MapSL ;This is ok: part of M5 hack anyway
|
|
push eax
|
|
call MapLS
|
|
mov ss,dx
|
|
;!!! End of B#9814 workaround.
|
|
ENDIF
|
|
|
|
pop eax
|
|
endm
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_SYSTEMPARAMETERSINFOA
|
|
;-----------------------------------------------------------------------;
|
|
body_SYSTEMPARAMETERSINFOA macro
|
|
local exit, error_exit
|
|
local anpfnSPI, MAX_ACTION
|
|
local push_zero_0, push_fuwinini_0
|
|
local push_zero_1, push_fuwinini_1
|
|
local push_callframe_2, push_ax_2
|
|
local push_callframe_3, done_3
|
|
|
|
bp_uAction equ <bp_top>
|
|
bp_uParam equ <bp_top+4>
|
|
bp_lParam equ <bp_top+8>
|
|
bp_fuWinIni equ <bp_top+12>
|
|
|
|
;BP=SP upon entry
|
|
|
|
bp_pTmp equ <dword ptr [bp-4]>
|
|
|
|
mov ebx,dword ptr bp_uAction
|
|
cmp ebx,MAX_ACTION
|
|
ja check_new_actions
|
|
|
|
add bx,bx
|
|
jmp word ptr cs:anpfnSPI[bx]
|
|
|
|
check_new_actions:
|
|
mov dx, bx
|
|
shr bx, 12
|
|
and bx, 2
|
|
and dx, 1
|
|
add bx, dx
|
|
|
|
cmp bx,MAX_NEWACTION
|
|
ja error_exit
|
|
|
|
add bx,bx
|
|
jmp word ptr cs:anpfnNewSPI[bx]
|
|
|
|
anpfnNewSPI label word
|
|
dw offset thk_SPI_GETBOOLUSERPREFERENCE
|
|
dw offset thk_SPI_SETBOOLUSERPREFERENCE
|
|
dw offset thk_SPI_GETDWORDUSERPREFERENCE
|
|
dw offset thk_SPI_SETDWORDUSERPREFERENCE
|
|
MAX_NEWACTION equ ($ - anpfnNewSPI)/2
|
|
|
|
anpfnSPI label word
|
|
dw offset error_exit
|
|
dw offset thk_SPI_GETBEEP
|
|
dw offset thk_SPI_SETBEEP
|
|
dw offset thk_SPI_GETMOUSE
|
|
dw offset thk_SPI_SETMOUSE
|
|
dw offset thk_SPI_GETBORDER
|
|
dw offset thk_SPI_SETBORDER
|
|
dw offset error_exit
|
|
dw offset error_exit
|
|
dw offset error_exit
|
|
dw offset thk_SPI_GETKEYBOARDSPEED
|
|
dw offset thk_SPI_SETKEYBOARDSPEED
|
|
dw offset thk_SPI_LANGDRIVER
|
|
dw offset thk_SPI_ICONHORIZONTALSPACING
|
|
dw offset thk_SPI_GETSCREENSAVETIMEOUT
|
|
dw offset thk_SPI_SETSCREENSAVETIMEOUT
|
|
dw offset thk_SPI_GETSCREENSAVEACTIVE
|
|
dw offset thk_SPI_SETSCREENSAVEACTIVE
|
|
dw offset thk_SPI_GETGRIDGRANULARITY
|
|
dw offset thk_SPI_SETGRIDGRANULARITY
|
|
dw offset thk_SPI_SETDESKWALLPAPER
|
|
dw offset thk_SPI_SETDESKPATTERN
|
|
dw offset thk_SPI_GETKEYBOARDDELAY
|
|
dw offset thk_SPI_SETKEYBOARDDELAY
|
|
dw offset thk_SPI_ICONVERTICALSPACING
|
|
dw offset thk_SPI_GETICONTITLEWRAP
|
|
dw offset thk_SPI_SETICONTITLEWRAP
|
|
dw offset thk_SPI_GETMENUDROPALIGNMENT
|
|
dw offset thk_SPI_SETMENUDROPALIGNMENT
|
|
dw offset thk_SPI_SETDOUBLECLKWIDTH
|
|
dw offset thk_SPI_SETDOUBLECLKHEIGHT
|
|
dw offset thk_SPI_GETICONTITLELOGFONT
|
|
dw offset thk_SPI_SETDOUBLECLICKTIME
|
|
dw offset thk_SPI_SETMOUSEBUTTONSWAP
|
|
dw offset thk_SPI_SETICONTITLELOGFONT
|
|
dw offset thk_SPI_GETFASTTASKSWITCH
|
|
dw offset thk_SPI_SETFASTTASKSWITCH
|
|
dw offset thk_SPI_SETDRAGFULLWINDOWS
|
|
dw offset thk_SPI_GETDRAGFULLWINDOWS
|
|
dw offset thk_SPI_GETKEYBOARDLAYOUT
|
|
dw offset thk_SPI_SETKEYBOARDLAYOUT
|
|
dw offset thk_SPI_GETNONCLIENTMETRICS
|
|
dw offset thk_SPI_SETNONCLIENTMETRICS
|
|
dw offset thk_SPI_GETMINIMIZEDMETRICS
|
|
dw offset thk_SPI_SETMINIMIZEDMETRICS
|
|
dw offset thk_SPI_GETICONMETRICS
|
|
dw offset thk_SPI_SETICONMETRICS
|
|
dw offset thk_SPI_SETWORKAREA
|
|
dw offset thk_SPI_GETWORKAREA
|
|
dw offset thk_SPI_SETPENWINDOWS
|
|
dw offset thk_SPI_GETFILTERKEYS
|
|
dw offset thk_SPI_SETFILTERKEYS
|
|
dw offset thk_SPI_GETTOGGLEKEYS
|
|
dw offset thk_SPI_SETTOGGLEKEYS
|
|
dw offset thk_SPI_GETMOUSEKEYS
|
|
dw offset thk_SPI_SETMOUSEKEYS
|
|
dw offset thk_SPI_GETSHOWSOUNDS
|
|
dw offset thk_SPI_SETSHOWSOUNDS
|
|
dw offset thk_SPI_GETSTICKYKEYS
|
|
dw offset thk_SPI_SETSTICKYKEYS
|
|
dw offset thk_SPI_GETACCESSTIMEOUT
|
|
dw offset thk_SPI_SETACCESSTIMEOUT
|
|
dw offset thk_SPI_GETSERIALKEYS
|
|
dw offset thk_SPI_SETSERIALKEYS
|
|
dw offset thk_SPI_GETSOUNDSENTRY
|
|
dw offset thk_SPI_SETSOUNDSENTRY
|
|
dw offset thk_SPI_GETHIGHCONTRAST
|
|
dw offset thk_SPI_SETHIGHCONTRAST
|
|
dw offset thk_SPI_GETKEYBOARDPREF
|
|
dw offset thk_SPI_SETKEYBOARDPREF
|
|
dw offset thk_SPI_GETSCREENREADER
|
|
dw offset thk_SPI_SETSCREENREADER
|
|
dw offset thk_SPI_GETANIMATION
|
|
dw offset thk_SPI_SETANIMATION
|
|
dw offset thk_SPI_GETFONTSMOOTHING
|
|
dw offset thk_SPI_SETFONTSMOOTHING
|
|
dw offset thk_SPI_SETDRAGWIDTH
|
|
dw offset thk_SPI_SETDRAGHEIGHT
|
|
dw offset thk_SPI_SETHANDHELD
|
|
dw offset thk_SPI_GETLOWPOWERTIMEOUT
|
|
dw offset thk_SPI_GETPOWEROFFTIMEOUT
|
|
dw offset thk_SPI_SETLOWPOWERTIMEOUT
|
|
dw offset thk_SPI_SETPOWEROFFTIMEOUT
|
|
dw offset thk_SPI_GETLOWPOWERACTIVE
|
|
dw offset thk_SPI_GETPOWEROFFACTIVE
|
|
dw offset thk_SPI_SETLOWPOWERACTIVE
|
|
dw offset thk_SPI_SETPOWEROFFACTIVE
|
|
dw offset thk_SPI_SETCURSORS
|
|
dw offset thk_SPI_SETICONS
|
|
dw offset thk_SPI_GETDEFAULTINPUTLANG
|
|
dw offset thk_SPI_SETDEFAULTINPUTLANG
|
|
dw offset thk_SPI_SETLANGTOGGLE
|
|
dw offset thk_SPI_GETWINDOWSEXTENSION
|
|
dw offset thk_SPI_SETMOUSETRAILS
|
|
dw offset thk_SPI_GETMOUSETRAILS
|
|
dw offset thk_SPI_GETSNAPTODEFBUTTON
|
|
dw offset thk_SPI_SETSNAPTODEFBUTTON
|
|
dw offset thk_SPI_SETSCREENSAVERRUNNING ; 97
|
|
dw offset thk_SPI_GETMOUSEHOVERWIDTH ; 98
|
|
dw offset thk_SPI_SETMOUSEHOVERWIDTH ; 99
|
|
dw offset thk_SPI_GETMOUSEHOVERHEIGHT ; 100
|
|
dw offset thk_SPI_SETMOUSEHOVERHEIGHT ; 101
|
|
dw offset thk_SPI_GETMOUSEHOVERTIME ; 102
|
|
dw offset thk_SPI_SETMOUSEHOVERTIME ; 103
|
|
dw offset thk_SPI_GETWHEELSCROLLLINES ; 104
|
|
dw offset thk_SPI_SETWHEELSCROLLLINES ; 105
|
|
dw offset thk_SPI_GETMENUSHOWDELAY ; 106
|
|
dw offset thk_SPI_SETMENUSHOWDELAY ; 107
|
|
dw offset thk_SPI_GETUSERPREFERENCE ; 108
|
|
dw offset thk_SPI_SETUSERPREFERENCE ; 109
|
|
|
|
ifdef FE_IME
|
|
dw offset thk_SPI_GETSHOWIMEUI ; 110
|
|
dw offset thk_SPI_SETSHOWIMEUI ; 111
|
|
else
|
|
dw offset error_exit ; 110
|
|
dw offset error_exit ; 111
|
|
endif
|
|
dw offset thk_SPI_GETMOUSESPEED ; 112
|
|
dw offset thk_SPI_SETMOUSESPEED ; 113
|
|
dw offset thk_SPI_GETSCREENSAVERRUNNING ; 114
|
|
|
|
MAX_ACTION equ ($ - anpfnSPI)/2
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; UNIMPLEMENTED
|
|
;-----------------------------------------------------------------------;
|
|
thk_SPI_GETMOUSEHOVERWIDTH:
|
|
thk_SPI_SETMOUSEHOVERWIDTH:
|
|
thk_SPI_GETMOUSEHOVERHEIGHT:
|
|
thk_SPI_SETMOUSEHOVERHEIGHT:
|
|
thk_SPI_GETMOUSEHOVERTIME:
|
|
thk_SPI_SETMOUSEHOVERTIME:
|
|
thk_SPI_GETUSERPREFERENCE:
|
|
thk_SPI_SETUSERPREFERENCE:
|
|
error_exit:
|
|
sub eax,eax
|
|
jmp exit
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; uParam = UINT
|
|
; lParam = LPINT
|
|
;-----------------------------------------------------------------------;
|
|
thk_SPI_GETBEEP:
|
|
thk_SPI_GETBORDER:
|
|
thk_SPI_GETFASTTASKSWITCH:
|
|
thk_SPI_GETGRIDGRANULARITY:
|
|
thk_SPI_GETICONTITLEWRAP:
|
|
thk_SPI_GETKEYBOARDDELAY:
|
|
thk_SPI_GETKEYBOARDSPEED:
|
|
thk_SPI_GETMENUDROPALIGNMENT:
|
|
thk_SPI_GETSCREENSAVEACTIVE:
|
|
thk_SPI_GETSCREENSAVETIMEOUT:
|
|
thk_SPI_ICONHORIZONTALSPACING:
|
|
thk_SPI_ICONVERTICALSPACING:
|
|
thk_SPI_GETDRAGFULLWINDOWS:
|
|
thk_SPI_GETFONTSMOOTHING:
|
|
thk_SPI_GETMOUSETRAILS:
|
|
thk_SPI_GETSNAPTODEFBUTTON:
|
|
thk_SPI_SETSCREENSAVERRUNNING:
|
|
thk_SPI_GETBOOLUSERPREFERENCE:
|
|
thk_SPI_GETSCREENSAVERRUNNING:
|
|
ifdef FE_IME
|
|
thk_SPI_GETSHOWIMEUI:
|
|
endif
|
|
|
|
push byte ptr 0 ;reserve and init local var
|
|
mov ax,sp ;save addr of local var
|
|
|
|
push dword ptr bp_uAction
|
|
push word ptr bp_uParam
|
|
mov esi,bp_lParam
|
|
test esi,esi
|
|
jz push_zero_0
|
|
|
|
push ss
|
|
push ax
|
|
jmp short push_fuwinini_0
|
|
|
|
push_zero_0:
|
|
push esi
|
|
|
|
push_fuwinini_0:
|
|
push word ptr bp_fuWinIni
|
|
call SystemParametersInfo32
|
|
cwde
|
|
|
|
pop cx ;recover local var value, clean stack
|
|
|
|
mov esi,bp_lParam
|
|
test esi,esi
|
|
jz exit ;if 0, all done
|
|
|
|
mov es,FlatData
|
|
movsx ecx,cx
|
|
mov es:[esi],ecx
|
|
jmp exit
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; uParam = UINT
|
|
; lParam = DWORD
|
|
;-----------------------------------------------------------------------;
|
|
thk_SPI_SETBEEP:
|
|
thk_SPI_SETBORDER:
|
|
thk_SPI_SETDOUBLECLKHEIGHT:
|
|
thk_SPI_SETDOUBLECLICKTIME:
|
|
thk_SPI_SETDOUBLECLKWIDTH:
|
|
thk_SPI_SETFASTTASKSWITCH:
|
|
thk_SPI_SETGRIDGRANULARITY:
|
|
thk_SPI_SETICONTITLEWRAP:
|
|
thk_SPI_SETKEYBOARDDELAY:
|
|
thk_SPI_SETKEYBOARDSPEED:
|
|
thk_SPI_SETMENUDROPALIGNMENT:
|
|
thk_SPI_SETMOUSEBUTTONSWAP:
|
|
thk_SPI_SETSCREENSAVEACTIVE:
|
|
thk_SPI_SETSCREENSAVETIMEOUT:
|
|
thk_SPI_SETLOWPOWERTIMEOUT:
|
|
thk_SPI_SETPOWEROFFTIMEOUT:
|
|
thk_SPI_SETLOWPOWERACTIVE:
|
|
thk_SPI_SETPOWEROFFACTIVE:
|
|
thk_SPI_SETDRAGFULLWINDOWS:
|
|
thk_SPI_SETFONTSMOOTHING:
|
|
thk_SPI_SETPENWINDOWS:
|
|
thk_SPI_SETSHOWSOUNDS:
|
|
thk_SPI_SETKEYBOARDPREF:
|
|
thk_SPI_SETSCREENREADER:
|
|
thk_SPI_SETDRAGWIDTH:
|
|
thk_SPI_SETDRAGHEIGHT:
|
|
thk_SPI_SETCURSORS:
|
|
thk_SPI_SETICONS:
|
|
thk_SPI_SETLANGTOGGLE:
|
|
thk_SPI_GETWINDOWSEXTENSION:
|
|
thk_SPI_SETMOUSETRAILS:
|
|
thk_SPI_SETSNAPTODEFBUTTON:
|
|
thk_SPI_SETWHEELSCROLLLINES:
|
|
thk_SPI_SETMENUSHOWDELAY:
|
|
ifdef FE_IME
|
|
thk_SPI_SETSHOWIMEUI:
|
|
endif
|
|
thk_SPI_SETBOOLUSERPREFERENCE:
|
|
thk_SPI_SETDWORDUSERPREFERENCE:
|
|
thk_SPI_SETMOUSESPEED:
|
|
|
|
push dword ptr bp_uAction
|
|
push word ptr bp_uParam
|
|
push dword ptr bp_lParam
|
|
push word ptr bp_fuWinIni
|
|
call SystemParametersInfo32
|
|
cwde
|
|
jmp exit
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; uParam = WORD
|
|
; lParam = LPRECT
|
|
;-----------------------------------------------------------------------;
|
|
thk_SPI_SETWORKAREA:
|
|
thk_SPI_GETWORKAREA:
|
|
|
|
; We need scratch space for RECT16
|
|
sub sp, RECT16_SIZE
|
|
mov di,sp
|
|
|
|
push dword ptr bp_uAction
|
|
push word ptr bp_uParam
|
|
|
|
mov esi,bp_lParam
|
|
test esi,esi
|
|
jnz pack_lprect
|
|
push esi
|
|
jmp after_lprect
|
|
|
|
pack_lprect:
|
|
; Stick lprect16 address on stack first
|
|
push ss
|
|
push di
|
|
|
|
; Convert lprect32 (ds:esi) to lprect16 (es:di) before
|
|
push ds
|
|
mov ds, FlatData
|
|
mov ax, ss
|
|
mov es, ax
|
|
cld
|
|
|
|
PACK_RECT_32_16
|
|
pop ds
|
|
|
|
after_lprect:
|
|
push word ptr bp_fuWinIni
|
|
call SystemParametersInfo32
|
|
cwde
|
|
|
|
test esi, esi
|
|
jz stack_clean
|
|
|
|
; Save return value and DS
|
|
push eax
|
|
push ds
|
|
|
|
; Setup lprc16 in DS:SI (remember--PACK_RECT_32_16 changed DI)
|
|
; Setup lprc32 in ES:EDI (remember--PACK_RECT_32_16 changed ESI)
|
|
push ss
|
|
sub di, RECT16_SIZE
|
|
push di
|
|
|
|
mov edi, esi
|
|
sub edi, RECT32_SIZE
|
|
mov es, FlatData
|
|
|
|
pop si
|
|
pop ds
|
|
cld
|
|
|
|
PACK_RECT_16_32
|
|
pop ds
|
|
pop eax
|
|
|
|
stack_clean:
|
|
; Clean the stack
|
|
add sp, RECT16_SIZE
|
|
jmp exit
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; uParam = 0
|
|
; lParam = INT[3]
|
|
;-----------------------------------------------------------------------;
|
|
thk_SPI_GETMOUSE:
|
|
thk_SPI_SETMOUSE:
|
|
|
|
sub sp,6 ;make space for local array
|
|
mov bx,sp ;save addr of local var
|
|
|
|
push dword ptr bp_uAction
|
|
push word ptr bp_uParam
|
|
mov esi,bp_lParam
|
|
test esi,esi
|
|
jz push_zero_1
|
|
|
|
mov es,FlatData
|
|
mov ax,es:[esi+0]
|
|
mov ss:[bx+0],ax
|
|
mov ax,es:[esi+4]
|
|
mov ss:[bx+2],ax
|
|
mov ax,es:[esi+8]
|
|
mov ss:[bx+4],ax
|
|
|
|
push ss
|
|
push bx
|
|
jmp short push_fuwinini_1
|
|
|
|
push_zero_1:
|
|
push esi
|
|
|
|
push_fuwinini_1:
|
|
push word ptr bp_fuWinIni
|
|
call SystemParametersInfo32
|
|
cwde
|
|
|
|
pop bx ;recover local vars value, clean stack
|
|
pop cx
|
|
pop dx
|
|
|
|
mov esi,bp_lParam
|
|
test esi,esi
|
|
jz exit ;if 0, all done
|
|
|
|
mov es,FlatData
|
|
movsx ebx,bx
|
|
mov es:[esi+0],ebx
|
|
movsx ecx,cx
|
|
mov es:[esi+4],ecx
|
|
movsx edx,dx
|
|
mov es:[esi+8],edx
|
|
jmp exit
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; uParam = 0
|
|
; lParam = LPFN
|
|
;-----------------------------------------------------------------------;
|
|
thk_SPI_SETHANDHELD:
|
|
push dword ptr bp_uAction ;pass uAction
|
|
push word ptr bp_uParam ;pass uParam
|
|
push dword ptr bp_lParam ;convert lParam to 16:16
|
|
push dword ptr CBID_SENDMSGCALLBACK ;sendmsg-type callback stub
|
|
call GetStdCBSL
|
|
push eax ;pass pfn16 as lParam
|
|
push word ptr bp_fuWinIni
|
|
call SystemParametersInfo32
|
|
cwde
|
|
jmp exit
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; uParam = INT (if cbSize, 16-bit & 32-bit structs have to be ==)
|
|
; lParam = LPVOID
|
|
;-----------------------------------------------------------------------;
|
|
thk_SPI_LANGDRIVER:
|
|
thk_SPI_SETKEYBOARDLAYOUT:
|
|
thk_SPI_GETKEYBOARDLAYOUT:
|
|
thk_SPI_SETDESKPATTERN:
|
|
thk_SPI_SETDESKWALLPAPER:
|
|
thk_SPI_GETNONCLIENTMETRICS:
|
|
thk_SPI_SETNONCLIENTMETRICS:
|
|
thk_SPI_GETMINIMIZEDMETRICS:
|
|
thk_SPI_SETMINIMIZEDMETRICS:
|
|
thk_SPI_GETANIMATION:
|
|
thk_SPI_SETANIMATION:
|
|
thk_SPI_GETICONMETRICS:
|
|
thk_SPI_SETICONMETRICS:
|
|
thk_SPI_GETFILTERKEYS:
|
|
thk_SPI_SETFILTERKEYS:
|
|
thk_SPI_GETTOGGLEKEYS:
|
|
thk_SPI_SETTOGGLEKEYS:
|
|
thk_SPI_GETMOUSEKEYS:
|
|
thk_SPI_SETMOUSEKEYS:
|
|
thk_SPI_GETSHOWSOUNDS:
|
|
thk_SPI_GETSTICKYKEYS:
|
|
thk_SPI_SETSTICKYKEYS:
|
|
thk_SPI_GETACCESSTIMEOUT:
|
|
thk_SPI_SETACCESSTIMEOUT:
|
|
thk_SPI_GETSERIALKEYS:
|
|
thk_SPI_SETSERIALKEYS:
|
|
thk_SPI_GETSOUNDSENTRY:
|
|
thk_SPI_SETSOUNDSENTRY:
|
|
thk_SPI_GETHIGHCONTRAST:
|
|
thk_SPI_SETHIGHCONTRAST:
|
|
thk_SPI_GETKEYBOARDPREF:
|
|
thk_SPI_GETSCREENREADER:
|
|
thk_SPI_GETDEFAULTINPUTLANG:
|
|
thk_SPI_SETDEFAULTINPUTLANG:
|
|
thk_SPI_GETWHEELSCROLLLINES:
|
|
thk_SPI_GETMENUSHOWDELAY:
|
|
thk_SPI_GETLOWPOWERACTIVE:
|
|
thk_SPI_GETPOWEROFFACTIVE:
|
|
thk_SPI_GETLOWPOWERTIMEOUT:
|
|
thk_SPI_GETPOWEROFFTIMEOUT:
|
|
thk_SPI_GETDWORDUSERPREFERENCE:
|
|
thk_SPI_GETMOUSESPEED:
|
|
|
|
push dword ptr bp_uAction
|
|
push word ptr bp_uParam
|
|
push dword ptr bp_lParam
|
|
call MapLS ;PLUGGED
|
|
mov dword ptr bp_lParam,eax
|
|
push eax
|
|
push word ptr bp_fuWinIni
|
|
call SystemParametersInfo32
|
|
cwde
|
|
|
|
push eax
|
|
push dword ptr bp_lParam
|
|
call UnmapLS
|
|
pop eax
|
|
|
|
jmp exit
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; uParam = sizeof LOGFONT or 0
|
|
; lParam = input LPLOGFONT or 0
|
|
;-----------------------------------------------------------------------;
|
|
thk_SPI_SETICONTITLELOGFONT:
|
|
|
|
sub eax,eax
|
|
push eax ;reserve and init bp_pTmp
|
|
|
|
mov esi,bp_lParam
|
|
test esi,esi
|
|
jz push_callframe_2
|
|
|
|
sub sp,LOGFONT16_SIZE
|
|
mov ax,sp
|
|
push ds
|
|
mov ds,FlatData ;DS:ESI --> 32-bit source
|
|
mov di,ss
|
|
mov es,di
|
|
movzx edi,ax ;ES:EDI --> 16-bit dest
|
|
mov word ptr bp_pTmp[0],di
|
|
mov word ptr bp_pTmp[2],es
|
|
cld
|
|
PACK_MLOGFONT_32_16
|
|
pop ds
|
|
|
|
push_callframe_2:
|
|
push dword ptr bp_uAction
|
|
|
|
mov ax,word ptr bp_uParam
|
|
test ax,ax
|
|
jz push_ax_2
|
|
mov ax,LOGFONT16_SIZE
|
|
push_ax_2:
|
|
push ax
|
|
|
|
push dword ptr bp_pTmp
|
|
push word ptr bp_fuWinIni
|
|
call SystemParametersInfo32
|
|
cwde
|
|
jmp short exit
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; uParam = sizeof LOGFONT
|
|
; lParam = output LPLOGFONT
|
|
;-----------------------------------------------------------------------;
|
|
thk_SPI_GETICONTITLELOGFONT:
|
|
|
|
sub eax,eax
|
|
push eax ;reserve and init bp_pTmp
|
|
|
|
mov edi,bp_lParam
|
|
test edi,edi
|
|
jz push_callframe_3
|
|
|
|
sub sp,LOGFONT16_SIZE
|
|
mov word ptr bp_pTmp[0],sp
|
|
mov word ptr bp_pTmp[2],ss
|
|
|
|
push_callframe_3:
|
|
push dword ptr bp_uAction
|
|
push byte ptr LOGFONT16_SIZE
|
|
push dword ptr bp_pTmp
|
|
push word ptr bp_fuWinIni
|
|
call SystemParametersInfo32
|
|
cwde
|
|
|
|
mov edi,bp_lParam
|
|
test edi,edi
|
|
jz exit
|
|
|
|
push ds
|
|
mov es,FlatData ;ES:EDI --> 32-bit source
|
|
lds si,bp_pTmp ;DS:SI --> 16-bit dest
|
|
movzx esi,si ;DS:ESI --> 16-bit dest
|
|
cld
|
|
PACK_MLOGFONT_16_32
|
|
pop ds
|
|
|
|
exit:
|
|
mov sp,bp
|
|
endm
|
|
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_LOADMENUINDIRECTA
|
|
;-----------------------------------------------------------------------;
|
|
body_LOADMENUINDIRECTA macro
|
|
local done
|
|
|
|
bp_lpmt equ <bp_top>
|
|
|
|
bp_lpmt32Tmp equ <dword ptr [bp-4]>
|
|
bp_hmem16Tmp equ < word ptr [bp-6]>
|
|
bp_cbmt equ < word ptr [bp-8]>
|
|
bp_mem32Menu equ <dword ptr [bp-12]>
|
|
|
|
xor eax,eax
|
|
push eax
|
|
push eax
|
|
push eax
|
|
|
|
push dword ptr bp_lpmt
|
|
call MapLS ;PLUGGED
|
|
mov bp_lpmt32Tmp,eax
|
|
|
|
push eax
|
|
call CountMenuU
|
|
mov bp_cbmt,ax
|
|
|
|
push byte ptr GHND
|
|
movzx eax,ax
|
|
push eax
|
|
call GlobalAlloc ;if no scratch space, return no hmenu
|
|
movzx eax,ax ;prepare for error
|
|
test ax,ax
|
|
jz done
|
|
|
|
mov bp_hmem16Tmp,ax
|
|
|
|
push ax
|
|
push ax
|
|
call GlobalFix
|
|
call GlobalLock
|
|
push dx
|
|
push ax
|
|
call MapSL ;SAFE
|
|
mov bp_mem32Menu, eax
|
|
|
|
mov eax, bp_lpmt
|
|
push eax
|
|
mov ax, bp_cbmt
|
|
movzx eax, ax
|
|
push eax
|
|
mov eax, bp_mem32Menu
|
|
push eax
|
|
call ConvertMenu32
|
|
|
|
|
|
mov ax, bp_hmem16Tmp
|
|
GMH2Sel ax ;push this ahead of time for LoadMenuIndirect
|
|
push ax
|
|
push byte ptr 0
|
|
|
|
;push bp_cbmt ;cbIn
|
|
;push ax ;pOut
|
|
;push byte ptr 0 ;
|
|
;push bp_lpmt32Tmp ;pIn
|
|
;call ConvertMenuA ;convert menu to win3.1 format
|
|
|
|
call LoadMenuIndirect ;param already pushed
|
|
movzx eax,ax
|
|
|
|
push eax
|
|
push bp_hmem16Tmp ;cannot be null
|
|
push bp_hmem16Tmp
|
|
push bp_hmem16Tmp
|
|
call GlobalUnlock
|
|
call GlobalUnfix
|
|
call GlobalFree
|
|
pop eax
|
|
|
|
done:
|
|
push eax
|
|
pushd bp_lpmt32Tmp
|
|
call UnmapLS
|
|
pop eax
|
|
endm
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_CREATEDIALOGINDIRECTPARAMA
|
|
;-----------------------------------------------------------------------;
|
|
body_CREATEDIALOGINDIRECTPARAMA macro
|
|
local done
|
|
local done_with_dlgproc
|
|
|
|
bp_hInstance equ <bp_top>
|
|
bp_hDialogTemplate equ <bp_top+4>
|
|
bp_hWndParent equ <bp_top+8>
|
|
bp_lpDialogFunc equ <bp_top+12>
|
|
bp_dwInitParam equ <bp_top+16>
|
|
|
|
bp_lpdt32Tmp equ <dword ptr [bp-4]>
|
|
bp_hmem16Tmp equ < word ptr [bp-6]>
|
|
bp_cbdt equ < word ptr [bp-8]>
|
|
bp_mem32Dlg equ <dword ptr [bp-12]>
|
|
|
|
xor eax,eax
|
|
push eax
|
|
push eax
|
|
push eax
|
|
|
|
push dword ptr bp_lpDialogTemplate
|
|
call MapLS ;PLUGGED
|
|
mov bp_lpdt32Tmp,eax
|
|
|
|
push eax
|
|
call CountDialogU
|
|
mov bp_cbdt,ax
|
|
|
|
push byte ptr GPTR
|
|
movzx eax,ax
|
|
push eax
|
|
call GlobalAlloc ;if no scratch space, return no hmenu
|
|
movzx eax,ax ;prepare for error
|
|
test ax,ax
|
|
jz done
|
|
|
|
mov bp_hmem16Tmp,ax
|
|
|
|
; GPTR is GMEM_FIXED, so MapSL is safe
|
|
push ax
|
|
push 0
|
|
call MapSL
|
|
mov bp_mem32Dlg, eax
|
|
|
|
;push bp_cbdt ;cbIn
|
|
;push ax ;pOut
|
|
;push byte ptr 0 ;
|
|
;push bp_lpdt32Tmp ;pIn
|
|
;call ConvertDialogA ;convert menu to win3.1 format
|
|
|
|
pushd bp_hDialogTemplate
|
|
push 0
|
|
push bp_cbdt
|
|
pushd bp_mem32Dlg
|
|
call ConvertDialog32
|
|
|
|
|
|
; create new call frame and make the call
|
|
|
|
; hInstance
|
|
MAP_NULL_HINST bp_hInstance
|
|
push ax
|
|
|
|
; lpDlgTemplate
|
|
push word ptr bp_hmem16Tmp
|
|
push 0
|
|
|
|
; hwndOwner
|
|
push word ptr bp_hWndParent
|
|
|
|
; lpfnDialog
|
|
pushd bp_lpDialogFunc
|
|
pushd CBID_DLGPROC
|
|
call GetStdCBSL
|
|
push eax
|
|
|
|
push dword ptr bp_dwInitParam
|
|
|
|
call GetCurrentHeap
|
|
; mov ds,ax ; set by func
|
|
call CreateDialogIndirectParam ; call 16-bit version
|
|
|
|
; Save zero-extended return value
|
|
movzx eax,ax
|
|
push eax
|
|
|
|
push bp_hmem16Tmp
|
|
call GlobalFree
|
|
|
|
; Restore return value
|
|
pop eax
|
|
|
|
done:
|
|
; Save return value
|
|
push eax
|
|
|
|
pushd bp_lpdt32Tmp
|
|
call UnmapLS
|
|
|
|
; Restore return value
|
|
pop eax
|
|
|
|
;locals discarded by next instruction
|
|
endm
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; body_DIALOGBOXINDIRECTPARAMA
|
|
;-----------------------------------------------------------------------;
|
|
body_DIALOGBOXINDIRECTPARAMA macro
|
|
local done
|
|
local ret_ok
|
|
local cont
|
|
local done_with_dlgproc
|
|
local dlgproc_freed
|
|
|
|
bp_hInstance equ <bp_top>
|
|
bp_lpDialogTemplate equ <bp_top+4>
|
|
bp_hWndParent equ <bp_top+8>
|
|
bp_lpDialogFunc equ <bp_top+12>
|
|
bp_dwInitParam equ <bp_top+16>
|
|
|
|
bp_lpdt32Tmp equ <dword ptr [bp-4]>
|
|
bp_hmem16Tmp equ < word ptr [bp-6]>
|
|
bp_cbdt equ < word ptr [bp-8]>
|
|
bp_mem32Dlg equ <dword ptr [bp-12]>
|
|
|
|
xor eax,eax
|
|
push eax
|
|
push eax
|
|
push eax
|
|
|
|
push dword ptr bp_lpDialogTemplate
|
|
call MapLS ;PLUGGED
|
|
mov bp_lpdt32Tmp,eax
|
|
|
|
push eax
|
|
call CountDialogU
|
|
mov bp_cbdt,ax
|
|
|
|
push byte ptr GPTR
|
|
movzx eax,ax
|
|
push eax
|
|
call GlobalAlloc ;if no scratch space, return no hmenu
|
|
movzx eax,ax ;prepare for error
|
|
test ax,ax
|
|
jz done
|
|
|
|
mov bp_hmem16Tmp,ax
|
|
|
|
; GPTR is GMEM_FIXED, so MapSL is safe
|
|
push ax
|
|
push 0
|
|
call MapSL
|
|
mov bp_mem32Dlg, eax
|
|
|
|
pushd bp_lpDialogTemplate
|
|
push 0
|
|
push bp_cbdt
|
|
pushd bp_mem32Dlg
|
|
call ConvertDialog32
|
|
|
|
|
|
; create new call frame and make the call
|
|
|
|
;hInst
|
|
MAP_NULL_HINST bp_hInstance
|
|
push ax
|
|
|
|
; hTemplate
|
|
push word ptr bp_hmem16Tmp
|
|
|
|
; hwndOwner
|
|
push word ptr bp_hWndParent
|
|
|
|
; lpfnDialog
|
|
push dword ptr bp_lpDialogFunc
|
|
push dword ptr CBID_DLGPROC
|
|
call GetStdCBSL
|
|
push eax
|
|
|
|
push dword ptr bp_dwInitParam
|
|
|
|
call GetCurrentHeap
|
|
; mov ds,ax ; set by func
|
|
call DialogBoxIndirectParam ; call 16-bit version
|
|
|
|
DXAX2EAX ;; 16-bit version returns in DX:AX.
|
|
;; So, move it to EAX.
|
|
|
|
cont:
|
|
; Save return value
|
|
push eax
|
|
|
|
|
|
push bp_hmem16Tmp ;cannot be null
|
|
call GlobalFree
|
|
|
|
; Restore return value
|
|
pop eax
|
|
|
|
done:
|
|
; Save return value
|
|
push eax
|
|
|
|
pushd bp_lpdt32Tmp
|
|
call UnmapLS
|
|
|
|
; Restore return value
|
|
pop eax
|
|
|
|
;locals discarded by next instruction
|
|
endm
|
|
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------;
|
|
;----------------------------------------------------------------------;
|
|
|
|
; Inputs: eax = 32-bit lpNewItem from Win32 app.
|
|
; wFlags = MF flags passed to API (in cx for ChangeMenu!!)
|
|
; SegVarAddr = address of thunk-created local
|
|
;
|
|
; Output: eax = mapped version:
|
|
; -- input eax if MF_OWNERDRAW
|
|
; -- hi-word zeroed if MF_BITMAP
|
|
; -- MapLS's if MF_STRING
|
|
;
|
|
; SegVarAddr set to seg:offset if MF_STRING.
|
|
;
|
|
MAP_MENU_LPNEWITEM macro wFlags,SegVarAddr
|
|
|
|
local exit,isstring
|
|
|
|
; Assumes SegVarAddr initialized to 0.
|
|
mov dx,wFlags
|
|
test dx,MF_OWNERDRAW or MF_SEPARATOR
|
|
jnz exit
|
|
test dx,MF_BITMAP
|
|
jz isstring
|
|
movzx eax,ax
|
|
jmp exit
|
|
|
|
isstring:
|
|
push eax
|
|
call MapLS ;PLUGGED
|
|
mov dword ptr SegVarAddr,eax
|
|
exit:
|
|
|
|
endm ;MAP_MENU_LPNEWITEM
|
|
|
|
|
|
;----------------------------------------------------------------------;
|
|
;----------------------------------------------------------------------;
|
|
RAWPACK__MODIFYMENUA_lpNewItem macro iOffset,iOffsetTemp
|
|
|
|
mov eax,[bp+iOffset]
|
|
MAP_MENU_LPNEWITEM [bp_top+8],LOCAL__MODIFYMENUA_lpNewItemSeg
|
|
mov [bp-iOffsetTemp],eax
|
|
|
|
endm; RAWPACK__MODIFYMENUA_lpNewItem
|
|
|
|
|
|
RAWUNPACK__MODIFYMENUA_lpNewItem macro iOffset,iOffsetTemp
|
|
|
|
push dword ptr LOCAL__MODIFYMENUA_lpNewItemSeg
|
|
call UnmapLS
|
|
|
|
endm; RAWUNPACK__MODIFYMENUA_lpNewItem
|
|
|
|
|
|
;----------------------------------------------------------------------;
|
|
;----------------------------------------------------------------------;
|
|
RAWPACK__INSERTMENUA_lpNewItem macro iOffset,iOffsetTemp
|
|
|
|
mov eax,[bp+iOffset]
|
|
MAP_MENU_LPNEWITEM [bp_top+8],LOCAL__INSERTMENUA_lpNewItemSeg
|
|
mov [bp-iOffsetTemp],eax
|
|
|
|
endm; RAWPACK__INSERTMENUA_lpNewItem
|
|
|
|
|
|
RAWUNPACK__INSERTMENUA_lpNewItem macro iOffset,iOffsetTemp
|
|
|
|
push dword ptr LOCAL__INSERTMENUA_lpNewItemSeg
|
|
call UnmapLS
|
|
|
|
endm; RAWUNPACK__INSERTMENUA_lpNewItem
|
|
|
|
|
|
;----------------------------------------------------------------------;
|
|
;----------------------------------------------------------------------;
|
|
RAWPACK__CHANGEMENUA_lpNewItem macro iOffset,iOffsetTemp
|
|
|
|
mov eax,[bp+iOffset]
|
|
|
|
; HACK: The same flag bit that means MF_OWNERDRAW for
|
|
; every other *Menu() api means MF_APPEND for ChangeMenu(). To prevent
|
|
; MAP_MENU_LPNEWITEM from getting confused, turn off the bit in the
|
|
; flag we give to it.
|
|
mov cx,[bp_top+16]
|
|
and cx,not MF_OWNERDRAW
|
|
MAP_MENU_LPNEWITEM cx,LOCAL__CHANGEMENUA_lpNewItemSeg
|
|
mov [bp-iOffsetTemp],eax
|
|
|
|
endm; RAWPACK__CHANGEMENUA_lpNewItem
|
|
|
|
|
|
RAWUNPACK__CHANGEMENUA_lpNewItem macro iOffset,iOffsetTemp
|
|
|
|
push dword ptr LOCAL__CHANGEMENUA_lpNewItemSeg
|
|
call UnmapLS
|
|
|
|
endm; RAWUNPACK__CHANGEMENUA_lpNewItem
|
|
|
|
|
|
|
|
;------------------------------------------------------------------------;
|
|
;body_WINHELPA
|
|
;------------------------------------------------------------------------;
|
|
body_WINHELPA macro
|
|
local its_a_pointer
|
|
local its_lpmultikeyhelp
|
|
local its_lphelpwininfo
|
|
local leave_alone
|
|
local call_winhelp
|
|
local push_with_ssdi
|
|
local loop_top
|
|
local loop_top_2
|
|
|
|
bp_hwnd equ <bp_top>
|
|
bp_lpHelpFile equ <bp_top+4>
|
|
bp_wCommand equ <bp_top+8>
|
|
bp_dwData equ <bp_top+12>
|
|
|
|
bp_lpHelpFileTmp equ <[bp-4]>
|
|
bp_dwDataTmp equ <[bp-8]>
|
|
|
|
xor eax,eax
|
|
push eax
|
|
push eax
|
|
|
|
;;The various HELP_* constants are not arranged nicely for a jump
|
|
;;table, so just check for each of the known types needing special
|
|
;;thunking. If wCommand is not recognized,
|
|
|
|
mov bx,word ptr bp_wCommand
|
|
and bx, not HELP_TCARD
|
|
|
|
cmp bx, HELP_CONTEXTMENU
|
|
jb leave_alone
|
|
je its_a_pointer
|
|
|
|
cmp bx, HELP_WM_HELP
|
|
je its_a_pointer
|
|
|
|
cmp bx,HELP_KEY
|
|
je its_a_pointer
|
|
|
|
cmp bx,HELP_COMMAND
|
|
je its_a_pointer
|
|
|
|
cmp bx,HELP_PARTIALKEY
|
|
je its_a_pointer
|
|
|
|
cmp bx,HELP_MULTIKEY
|
|
je its_lpmultikeyhelp
|
|
|
|
cmp bx,HELP_SETWINPOS
|
|
je its_lphelpwininfo
|
|
|
|
|
|
;; Push dwData as a dword.
|
|
leave_alone:
|
|
push word ptr bp_hwnd
|
|
push dword ptr bp_lpHelpFile
|
|
call MapLS ;PLUGGED
|
|
mov dword ptr bp_lpHelpFileTmp,eax
|
|
push eax
|
|
push word ptr bp_wCommand
|
|
push dword ptr bp_dwData
|
|
jmp call_winhelp
|
|
|
|
;; Push dwData as LPSTR, so call MapLS
|
|
its_a_pointer:
|
|
push word ptr bp_hwnd
|
|
push dword ptr bp_lpHelpFile
|
|
call MapLS ;PLUGGED
|
|
mov dword ptr bp_lpHelpFileTmp,eax
|
|
push eax
|
|
push word ptr bp_wCommand
|
|
push dword ptr bp_dwData
|
|
call MapLS ;PLUGGED
|
|
mov dword ptr bp_dwDataTmp,eax
|
|
push eax
|
|
jmp call_winhelp
|
|
|
|
;;Push dwData as LPMULTIKEYHELP, which needs to be repacked.
|
|
its_lpmultikeyhelp:
|
|
mov esi,bp_dwData
|
|
test esi,esi ;;if zero, just push it
|
|
jz leave_alone
|
|
|
|
mov es,FlatData
|
|
mov ax,word ptr es:[esi].mk32_Size
|
|
mov cx,ax ;;save for loop count
|
|
add cx,(size MULTIKEYHELP16 - size MULTIKEYHELP32)
|
|
sub sp,cx ;;subtract adjusted size for 16-bit struct
|
|
mov di,sp ;;will access 16-bit struct thru SS:DI
|
|
mov ss:[di].mk16_Size,cx
|
|
mov cl,es:[esi].mk32_Keylist
|
|
mov ss:[di].mk16_Keylist,cl
|
|
|
|
mov cx,ax ;;32-bit struct size
|
|
sub cx,mk32_szKeyphrase ;;skip header fields
|
|
add esi,mk32_szKeyphrase
|
|
add di,mk16_szKeyphrase
|
|
|
|
loop_top:
|
|
mov al,es:[esi]
|
|
mov ss:[di],al
|
|
inc esi
|
|
inc di
|
|
loop loop_top
|
|
|
|
mov di,sp ;;SP --> base of structure
|
|
jmp short push_with_ssdi
|
|
|
|
;;Push dwData as LPHELPWININFO, which needs to be repacked.
|
|
its_lphelpwininfo:
|
|
mov esi,bp_dwData
|
|
test esi,esi ;;if zero, just push it
|
|
jz leave_alone
|
|
|
|
mov es,FlatData
|
|
mov ax,word ptr es:[esi].hwi32_wStructSize
|
|
mov cx,ax ;;save for loop count
|
|
add cx,(size HELPWININFO16 - size HELPWININFO32)
|
|
sub sp,cx ;;subtract adjusted size for 16-bit struct
|
|
mov di,sp ;;will access 16-bit struct thru SS:DI
|
|
mov ss:[di].hwi16_wStructSize,cx
|
|
|
|
;;Truncate all fields except for rgchMember to 16-bits.
|
|
|
|
mov cx,word ptr es:[esi].hwi32_x
|
|
mov ss:[di].hwi16_x,cx
|
|
mov cx,word ptr es:[esi].hwi32_y
|
|
mov ss:[di].hwi16_y,cx
|
|
mov cx,word ptr es:[esi].hwi32_dx
|
|
mov ss:[di].hwi16_dx,cx
|
|
mov cx,word ptr es:[esi].hwi32_dy
|
|
mov ss:[di].hwi16_dy,cx
|
|
mov cx,word ptr es:[esi].hwi32_wMax
|
|
mov ss:[di].hwi16_wMax,cx
|
|
|
|
mov cx,ax ;;32-bit struct size
|
|
sub cx,hwi32_rgchMember ;;skip header fields
|
|
add esi,hwi32_rgchMember
|
|
add di,hwi16_rgchMember
|
|
|
|
loop_top_2:
|
|
mov al,es:[esi]
|
|
mov ss:[di],al
|
|
inc esi
|
|
inc di
|
|
loop loop_top_2
|
|
|
|
mov di,sp ;;SP --> base of structure
|
|
|
|
push_with_ssdi:
|
|
push word ptr bp_hwnd
|
|
push dword ptr bp_lpHelpFile
|
|
call MapLS ;PLUGGED
|
|
mov dword ptr bp_lpHelpFileTmp,eax
|
|
push eax
|
|
push word ptr bp_wCommand
|
|
push ss ;;SS:DI --> thunked structure
|
|
push di
|
|
|
|
call_winhelp:
|
|
call WinHelp
|
|
cwde
|
|
|
|
|
|
push eax
|
|
push dword ptr bp_lpHelpFileTmp
|
|
call UnmapLS
|
|
push dword ptr bp_dwDataTmp
|
|
call UnmapLS
|
|
pop eax
|
|
;;The first instruction at Exit_16 clears temporary variables off stack.
|
|
endm
|
|
|
|
|
|
|
|
|
|
;==============================================================================
|
|
;
|
|
; InsertMenuItemA()
|
|
;
|
|
;==============================================================================
|
|
body_INSERTMENUITEMA macro
|
|
local ConvertMiim
|
|
local MakeCall
|
|
bp_hMenu equ <bp_top+ 0>
|
|
bp_nIndex equ <bp_top+ 4>
|
|
bp_fByPosition equ <bp_top+ 8>
|
|
bp_lpMiim32 equ <bp_top+12>
|
|
|
|
bp_SavePtr equ <[bp-4]>
|
|
|
|
xor eax,eax
|
|
push eax ;bp_SavePtr
|
|
|
|
; We need scratch space for MENUITEMINFO16
|
|
sub sp, MENUITEMINFO16_SIZE
|
|
mov di, sp
|
|
|
|
; Push 16bit parms
|
|
push word ptr bp_hMenu ;hMenu 16
|
|
push word ptr bp_nIndex ;nIndex 16
|
|
push word ptr bp_fByPosition ;fByPosition 16
|
|
|
|
; Is lpMenuItemInfo NULL?
|
|
mov eax, dword ptr bp_lpMiim32
|
|
test eax, eax
|
|
jnz ConvertMiim
|
|
|
|
push eax
|
|
jmp MakeCall
|
|
|
|
ConvertMiim:
|
|
push ss ;lpMenuItemInfo 16
|
|
push di
|
|
|
|
; Convert MENUITEMINFO LS
|
|
push eax
|
|
push di
|
|
lea ax, bp_SavePtr
|
|
push ax
|
|
call ConvertMenuItemInfoLS
|
|
|
|
MakeCall:
|
|
call InsertMenuItem32
|
|
cwde
|
|
|
|
;Save return value
|
|
push eax
|
|
|
|
;Cleanup
|
|
push dword ptr bp_SavePtr
|
|
call UnMapLS
|
|
|
|
;Restore return value
|
|
pop eax
|
|
endm
|
|
|
|
|
|
;==============================================================================
|
|
;
|
|
; SetMenuItemInfoA()
|
|
;
|
|
;==============================================================================
|
|
body_SETMENUITEMINFOA macro
|
|
local ConvertMiim
|
|
local MakeCall
|
|
bp_hMenu equ <bp_top+ 0>
|
|
bp_nIndex equ <bp_top+ 4>
|
|
bp_fByPosition equ <bp_top+ 8>
|
|
bp_lpMiim32 equ <bp_top+12>
|
|
|
|
bp_SavePtr equ <[bp-4]>
|
|
|
|
xor eax,eax
|
|
push eax ;bp_SavePtr
|
|
|
|
; We need scratch space for MENUITEMINFO16
|
|
sub sp, MENUITEMINFO16_SIZE
|
|
mov di, sp
|
|
|
|
;Push 16bit parms
|
|
push word ptr bp_hMenu
|
|
push word ptr bp_nIndex
|
|
push word ptr bp_fByPosition
|
|
|
|
; Is lpMenuItemInfo NULL?
|
|
mov eax, dword ptr bp_lpMiim32
|
|
test eax, eax
|
|
jnz ConvertMiim
|
|
|
|
push eax
|
|
jmp MakeCall
|
|
|
|
ConvertMiim:
|
|
push ss
|
|
push di
|
|
|
|
;Convert MENUITEMINFO LS
|
|
push eax
|
|
push di
|
|
lea ax, bp_SavePtr
|
|
push ax
|
|
call ConvertMenuItemInfoLS
|
|
|
|
MakeCall:
|
|
call SetMenuItemInfo32
|
|
cwde
|
|
|
|
;Save return value
|
|
push eax
|
|
|
|
;Cleanup
|
|
push dword ptr bp_SavePtr
|
|
call UnMapLS
|
|
|
|
;Restore return value
|
|
pop eax
|
|
endm
|
|
|
|
|
|
|
|
;==============================================================================
|
|
;
|
|
; GetMenuItemInfoA()
|
|
;
|
|
;==============================================================================
|
|
body_GETMENUITEMINFOA macro
|
|
local ConvertMiim
|
|
local MakeCall
|
|
local GetDone
|
|
local MenuTypeData
|
|
local MenuItemCch
|
|
local MenuItemEnd
|
|
|
|
bp_hMenu equ <bp_top+ 0>
|
|
bp_nIndex equ <bp_top+ 4>
|
|
bp_fByPosition equ <bp_top+ 8>
|
|
bp_lpMiim32 equ <bp_top+12>
|
|
|
|
bp_SavePtr equ <[bp-4]>
|
|
|
|
xor eax,eax
|
|
push eax
|
|
|
|
; We need scratch space for MENUITEMINFO16
|
|
sub sp, MENUITEMINFO16_SIZE
|
|
movzx edi, sp
|
|
|
|
; Push 16bit parms
|
|
push word ptr bp_hMenu
|
|
push word ptr bp_nIndex
|
|
push word ptr bp_fByPosition
|
|
|
|
; Is lpMiim32 NULL?
|
|
mov eax, dword ptr bp_lpMiim32
|
|
test eax, eax
|
|
jnz ConvertMiim
|
|
|
|
push eax
|
|
jmp MakeCall
|
|
|
|
ConvertMiim:
|
|
push ss
|
|
push di
|
|
|
|
; HACK
|
|
; If MIIM_TYPE is specified in the mask for a 95-sized structure, we
|
|
; need to add MIIM_STRING and MIIM_BITMAP to the mask so that we fill
|
|
; those fields
|
|
mov es, FlatData
|
|
mov esi, bp_lpMiim32
|
|
.errnz mii_32_fType - mii_32_cbSize - 8
|
|
mov ecx, dword ptr es:[esi+4]
|
|
test cx, 00010h ; if (fMask & MIIM_TYPE)
|
|
jz @F
|
|
or cx, 001C0h ; fMask |= MIIM_FTYPE |
|
|
mov dword ptr es:[esi+4], ecx ; MIIM_STRING | MIIM_BITMAP
|
|
@@:
|
|
|
|
; Convert MENUITEMINFO LS
|
|
push esi
|
|
push di
|
|
lea ax, bp_SavePtr
|
|
push ax
|
|
call ConvertMenuItemInfoLS
|
|
|
|
MakeCall:
|
|
call GetMenuItemInfo32
|
|
cwde
|
|
|
|
; Save return value
|
|
push eax
|
|
|
|
; Is bp_lpMiim32 NULL?
|
|
mov eax, bp_lpMiim32
|
|
test eax, eax
|
|
jz GetDone
|
|
|
|
mov es, FlatData ; DO THIS FIRST: FlatData is DS-relative
|
|
|
|
push ds
|
|
mov cx, ss
|
|
mov ds, cx
|
|
|
|
mov esi, edi
|
|
mov edi, eax ; DO THIS LAST: We need to move DI to SI first
|
|
cld
|
|
|
|
; cbSize - skip it
|
|
add esi, 4
|
|
add edi, 4
|
|
|
|
;fMask - save it
|
|
lodsd ds:[esi]
|
|
mov ecx, eax
|
|
add edi, 4
|
|
|
|
;fType
|
|
xor eax, eax ;Zero out HIWORD
|
|
lodsw ds:[esi]
|
|
mov edx, eax ; save fType
|
|
stosd es:[edi]
|
|
|
|
;fState,wID,hSubMenu,hbmpChecked,hbmpUnchecked
|
|
push ecx
|
|
ncopyzx 5
|
|
pop ecx
|
|
|
|
;dwItemData
|
|
copyd
|
|
|
|
test cx, 00010h ; if (!(fMask & MIIM_TYPE))
|
|
jz SkipTypeData ; goto SkipTypeData
|
|
mov eax,es:[edi-020h]
|
|
and ax,0FE3Fh ; fMask &= ~(MIIM_FTYPE | MIIM_STRING |
|
|
mov es:[edi-020h],eax ; MIIM_BITMAP);
|
|
xor eax,eax
|
|
mov ax,ds:[esi+006h]
|
|
or ax,ax ; if (!lpmii16->hbmpItem)
|
|
jz CheckString ; goto CheckString
|
|
stosd es:[edi] ; lpmii32->dwTypeData = lpmii16->hbmpItem
|
|
|
|
or dx, 00004h ; lpmii32->fType |= MFT_BITMAP
|
|
mov es:[edi-020h], edx
|
|
|
|
jmp MenuItemCch
|
|
|
|
CheckString:
|
|
mov ax,ds:[esi+004h]
|
|
or ax,ax ; if (!lpmii16->cch)
|
|
jz ZeroTypeData ; goto ZeroTypeData
|
|
|
|
SkipTypeData:
|
|
add esi, 4 ; it's a string -- so leave it as is
|
|
add edi, 4
|
|
jmp MenuItemCch
|
|
|
|
ZeroTypeData:
|
|
add esi, 4
|
|
xor eax,eax
|
|
stosd es:[edi]
|
|
|
|
MenuItemCch:
|
|
copyzx
|
|
|
|
; Check hbmpItem
|
|
test cx, 00010h ; if ((fMask & MIIM_TYPE))
|
|
jnz MenuItemEnd ; skip hbmpItem
|
|
|
|
mov eax,es:[edi-02ch]
|
|
cmp eax, 030h ; if old MENUITEMINFO
|
|
jc MenuItemEnd ; skip hbmpItem
|
|
|
|
mov eax,es:[edi-028h]
|
|
test ax,080h ; if not MIIM_BITMAP
|
|
jz MenuItemEnd ; skip hbmpItem
|
|
|
|
copyzx ; copy hbmpItem
|
|
|
|
MenuItemEnd:
|
|
pop ds
|
|
|
|
; Unmap pointer since we mapped dwTypeData no matter what!
|
|
push dword ptr bp_SavePtr
|
|
call UnmapLS
|
|
|
|
GetDone:
|
|
; Restore return value
|
|
pop eax
|
|
endm
|
|
|
|
|
|
;------------------------------------------------------------------------------
|
|
; RET__GETDLGITEMINT
|
|
;
|
|
; Either zero-extend or sign-extend ax into eax, depending on
|
|
; the value of the "bSigned" argument.
|
|
;------------------------------------------------------------------------------
|
|
RET__GETDLGITEMINT macro
|
|
|
|
local exit
|
|
local unsigned
|
|
|
|
cmp word ptr [bp_top + 12],0 ; bSigned
|
|
je unsigned
|
|
|
|
cwde
|
|
jmp exit
|
|
|
|
unsigned:
|
|
movzx eax,ax
|
|
|
|
exit:
|
|
|
|
endm ;RET__GETDLGITEMINT
|
|
|
|
|
|
;==============================================================================
|
|
;
|
|
; mouse_event
|
|
;
|
|
;==============================================================================
|
|
body_MOUSE_EVENT macro
|
|
|
|
bp_dwFlags equ <bp_top+ 0>
|
|
bp_dx equ <bp_top+ 4>
|
|
bp_dy equ <bp_top+ 8>
|
|
bp_cButtons equ <bp_top+12>
|
|
bp_dwExtraInfo equ <bp_top+16>
|
|
|
|
mov ax,word ptr bp_dwFlags
|
|
mov bx,word ptr bp_dx
|
|
mov cx,word ptr bp_dy
|
|
mov dx,word ptr bp_cButtons
|
|
mov si,word ptr bp_dwExtraInfo
|
|
mov di,word ptr bp_dwExtraInfo+2
|
|
|
|
call mouse_event
|
|
|
|
endm
|
|
|
|
|
|
;==============================================================================
|
|
;
|
|
; keybd_event
|
|
;
|
|
;==============================================================================
|
|
body_KEYBD_EVENT macro
|
|
|
|
bp_dwVirtualKey equ <bp_top+ 0>
|
|
bp_dwScanCode equ <bp_top+ 4>
|
|
bp_dwFlags equ <bp_top+ 8>
|
|
bp_dwExtraInfo equ <bp_top+12>
|
|
|
|
xor ax,ax
|
|
xor bx,bx
|
|
|
|
mov al,byte ptr bp_dwVirtualKey
|
|
test word ptr bp_dwFlags,KEYEVENTF_KEYUP
|
|
jz keybd_keydown
|
|
mov ah,80
|
|
keybd_keydown:
|
|
mov bl,byte ptr bp_dwScanCode
|
|
test word ptr bp_dwFlags,KEYEVENTF_EXTENDEDKEY
|
|
jz keybd_notextendedkey
|
|
mov bh,1
|
|
keybd_notextendedkey:
|
|
mov si,word ptr bp_dwExtraInfo
|
|
mov di,word ptr bp_dwExtraInfo+2
|
|
|
|
call keybd_event
|
|
|
|
endm
|
|
|
|
;==============================================================================
|
|
;
|
|
; AlignRects()
|
|
;
|
|
;==============================================================================
|
|
body_ALIGNRECTS macro
|
|
local done
|
|
local ar_loop1
|
|
local ar_loop2
|
|
|
|
bp_lprect equ <bp_top+ 0>
|
|
bp_count equ <bp_top+ 4>
|
|
bp_iPrimary equ <bp_top+ 8>
|
|
bp_flags equ <bp_top+12>
|
|
|
|
bp_hmem16Tmp equ < word ptr [bp-2]>
|
|
bp_Ret equ < word ptr [bp-4]>
|
|
bp_lprectTmp equ < dword ptr [bp-8]>
|
|
|
|
|
|
xor eax,eax
|
|
push eax
|
|
push eax
|
|
|
|
mov eax,bp_count ;prepare for error
|
|
test eax,eax
|
|
jz done
|
|
|
|
push byte ptr GPTR
|
|
shl eax, 3
|
|
push eax
|
|
call GlobalAlloc ;if no scratch space, return no hmenu
|
|
|
|
movzx eax,ax ;prepare for error
|
|
test ax,ax
|
|
jz done
|
|
mov bp_hmem16Tmp,ax
|
|
|
|
push dword ptr bp_lprect
|
|
call MapLS ;PLUGGED
|
|
mov dword ptr bp_lprectTmp,eax
|
|
movzx edi, ax
|
|
mov ax,dx
|
|
|
|
mov es, bp_hmem16Tmp
|
|
xor esi,esi
|
|
|
|
mov ecx,bp_count
|
|
shl ecx, 2
|
|
|
|
push ecx
|
|
push edi
|
|
push esi
|
|
push ax
|
|
push es
|
|
|
|
|
|
push ds
|
|
mov ds, ax
|
|
|
|
cld
|
|
ar_loop1:
|
|
lodsd ds:[esi]
|
|
stosw es:[di]
|
|
loop ar_loop1
|
|
pop ds
|
|
|
|
push bp_hmem16Tmp ;cannot be null
|
|
push 0
|
|
push word ptr bp_count
|
|
push word ptr bp_iPrimary
|
|
push word ptr bp_flags
|
|
call CleanUpDesktopRectangles
|
|
mov bp_Ret,ax
|
|
|
|
pop ax
|
|
pop es
|
|
pop edi
|
|
pop esi
|
|
pop ecx
|
|
|
|
push ds
|
|
mov ds, ax
|
|
|
|
cld
|
|
ar_loop2:
|
|
lodsw ds:[si]
|
|
cwde
|
|
stosd es:[edi]
|
|
loop ar_loop2
|
|
pop ds
|
|
|
|
push bp_hmem16Tmp ;cannot be null
|
|
call GlobalFree
|
|
|
|
push dword ptr bp_lprectTmp
|
|
call UnMapLS
|
|
|
|
movzx eax, bp_Ret
|
|
|
|
done:
|
|
|
|
endm
|
|
|
|
;==============================================================================
|
|
;
|
|
; FlashWindowEx()
|
|
;
|
|
;==============================================================================
|
|
body_FLASHWINDOWEX macro
|
|
local done
|
|
|
|
bp_lpfw equ <bp_top+ 0>
|
|
bp_fwSave equ < word ptr [bp-2]>
|
|
|
|
xor eax,eax
|
|
push eax ; lpfw->cbSize, lpfw->hwnd
|
|
push eax ; lpfw->dwFlags
|
|
push eax ; lpfw->uCount
|
|
|
|
mov esi,bp_lpfw
|
|
test esi,esi ;;if zero, just push it
|
|
jz done
|
|
|
|
mov es,FlatData
|
|
mov ecx, es:[esi] ; get fw32_cbSize to ecx
|
|
cmp cx, size FLASHWINFO32
|
|
jnz done
|
|
|
|
add esi, 4
|
|
|
|
mov di, sp
|
|
mov ss:[di].fw16_cbSize, (size FLASHWINFO16)
|
|
lodsd es:[esi] ; get fw32_hwnd to eax
|
|
mov ss:[di].fw16_hwnd, ax
|
|
lodsd es:[esi] ; get fw32_dwFlags to eax
|
|
mov ss:[di].fw16_dwFlags, eax
|
|
lodsd es:[esi] ; get fw32_uCount to eax
|
|
mov ss:[di].fw16_uCount, ax
|
|
|
|
push ss
|
|
push di
|
|
call FlashWindowEx
|
|
|
|
done:
|
|
endm
|
|
|
|
endif ;IS_16
|