Windows2003-3790/base/mvdm/wow16/kernel31/winexec.asm
2020-09-30 16:53:55 +02:00

332 lines
8.3 KiB
NASM

;-----------------------------------------------------------------------;
;
; WINEXEC.ASM -
;
; Windows Exec Function
;
;-----------------------------------------------------------------------;
include kernel.inc
ifdef WOW
include wowcmpat.inc
;The following defines come from winerror.inc, which is new in the
;win95 source tree, but hasn't been merged into wow
ERROR_FILE_NOT_FOUND equ 2
ERROR_BAD_LENGTH equ 24
endif
ifdef FE_SB
externFP FarMyIsDBCSLeadByte
endif
ifdef WOW
externFP MyGetAppWOWCompatFlagsEx
endif
externW WinFlags ; (kdata.asm)
externD WinAppHooks ; (kdata.asm)
sBegin NRESCODE
assumes cs,NRESCODE
;-----------------------------------------------------------------------;
;
; WinExec() -
;
;-----------------------------------------------------------------------;
; HANDLE PASCAL WinExec(lpszFile,wShow)
cProc IWinExec, <FAR,PUBLIC>,<si,di>
parmD lpszFile ; Pathname ptr
parmW wShow ; Mode flag
;
; szCmdLine buffer must be big enough for passed in path & parms, plus
; .EXE, null terminator, and newline terminator.
; So we add a little room for good measure here.
;
LocalV szCmdLine,260+256+8 ; Path and command line (to account for .exe)
LocalW pszParm ; Ptr to start of parameters
LocalB bDotFound ; Non-zero if a period is in the string
LocalB bDblQFound ; Non-zero if the string starts with "
LocalV loadparams, %(SIZE EXECBLOCK)
LocalD FCB1
cBegin
; Copy first part of line into szCmdLine.
lds si,lpszFile
smov es,ss
lea di,szCmdLine
mov cx,260 ; MAX_PATH
xor al,al
mov bDotFound,al
mov bDblQFound,al
; The Dbl-quote is used as a delimiter for cmdname as in
; winexec("\"c:\fldr with spaces\" cmd line", ..);
mov al, ds:[si] ; get Ist char
cmp al,'"' ; is it a "
jne WELoop1 ; N:
mov bDblQFound,al ; Y: remember that
lodsb ; skip the dblQ
; Loop until a blank or NULL is found.
WELoop1:
lodsb
cmp bDblQFound, 0 ; If str didn't start with dblQ
je WESpaceCheck ; then SPACE is a delimiter
; otherwise " is the delimiter
cmp al,'"' ; look for second DblQ
jne WESkipSpaceCheck
and bDblQFound,0 ; reset this, so space is the delimiter
lodsb ; skip past the dbl Q
WESpaceCheck:
cmp al,' ' ; Exit loop if blank or NULL
je WECont1
WESkipSpaceCheck:
cmp al,9
je WECont1
or al,al
je WECont1
cmp al,'.'
jne WELoopCont
mov bDotFound,al
WELoopCont:
;** We have to check to see if the dot we found was actually in
;** a directory name, not in the filename itself.
cmp al, '\' ; Separator?
je WE_Separator
cmp al, '/'
jne WE_Not_Separator
WE_Separator:
mov bDotFound,0 ; No dots count yet
WE_Not_Separator:
stosb
dec cx
jz WE_filename_too_long
ifdef FE_SB ;Apr.26,1990 by AkiraK
push cx
call FarMyIsDBCSLeadByte
pop cx
jc WELoop1
movsb
dec cx
jz WE_filename_too_long
endif
jmp short WELoop1
WE_filename_too_long:
krDebugOut DEB_TRACE, "WinExecEnv: filename too long, > 259"
mov ax,ERROR_FILE_NOT_FOUND
jmp WinExecEnvExit
WECont1:
mov dx,ax ; Store final char in DX
; Does the command have an extention?
cmp bDotFound,0
jne WEHasExt
mov ax,0452Eh ;'.E'
stosw
mov ax,04558h ;'XE'
stosw
WEHasExt:
xor ax,ax ; NULL terminate string
stosb
mov pszParm,di ; Store pointer to parm string
stosb ; length = 0
mov al,0dh ; line feed terminator
stosb
dec di ; back up
or dl,dl ; Exec if lpszFile was null terminated
jz WEExec
; Copy everything else into szParm.
mov cx,255 ; Max length of cmd tail +1
WELoop2:
lodsb
or al,al
ifdef WOW
jz WEDoneParm
else
jz WECont2
endif
stosb
dec cx
jnz WELoop2
ifdef WOW
jmps @F
;
; On NT we have a compatibility flag, WOWCFEX_LONGWINEXECTAIL which is
; set for applications such as TSSETUP, the setup program for Intergraph's
; NT-only Transcend app. This app uses a command tail on the order of
; 143 bytes for a Win32 worker app, and it worked on 3.5 and 3.51,
; so we need to continue to let at least this app cheat.
;
WEDoneParm:
sub cx, 128
ja WECont2
; tail was longer than 126 characters
call MyGetAppWOWCompatFlagsEx
test dx, word ptr cs:[WE_GACFEX_LONGWINEXECTAIL+2]
jz @F
jmps WECont2
WE_GACFEX_LONGWINEXECTAIL:
DD WOWCFEX_LONGWINEXECTAIL
@@:
endif
; Cmd tail too long to fit in PSP !!!
; Fail the call.
; We _could_ alloc some memory to hold the cmd tail and make its
; owner be the new hTask. However InitTask is documented to return
; es=PSP, es:bx=cmd_tail. So we're in trouble. Could truncate the cmd line.
; Cleaner to fail the exec.
; Could grow the PSP and tack the big cmd tail on at the end. Scary!
; Unfortunately, kernel32.ExecWin16Program maps ERROR_BAD_LENGTH to
; ERROR_GEN_FAILURE.
krDebugOut DEB_TRACE, "WinExecEnv: command tail too long, > 126"
mov ax,ERROR_BAD_LENGTH
jmps WinExecEnvExit
WECont2:
; Terminate it with a carriage return.
mov al,0Dh
stosb
; Prefix the parameter string with its length.
mov bx,pszParm ; ax = length + 2
mov ax,di
sub ax,bx
dec ax ; don't include line feed char or length
dec ax
mov ss:[bx],al
; Set up the FCBs.
WEExec:
mov word ptr FCB1[0],2 ; FCB1[0] = 2;
mov ax,wShow ; FCB1[1] = wShow;
mov word ptr FCB1[2],ax
xor ax,ax
mov loadparams.envseg,ax ; loadparms.segEnv = 0;
mov loadparams.lpfcb2.lo,ax ; loadparms.lpFCB2 = (LPSTR)NULL;
mov loadparams.lpfcb2.hi,ax
mov ax,pszParm ; loadparms.lpCmdLine = (LPSTR)pszParm;
mov loadparams.lpCmdLine.lo,ax
mov loadparams.lpCmdLine.hi,ss
lea ax,FCB1 ; loadparms.lpFCB1 = (LPSTR)fcb1buf;
mov loadparams.lpfcb1.lo,ax
mov loadparams.lpfcb1.hi,ss
; Exec the progam.
smov ds,ss
lea dx,szCmdLine ; ds:ax == ptr to file to exec
lea bx,loadparams ; es:bx == ptr to param block
mov ax,4B00h ; dos exec
int 21h
WinExecEnvExit:
cEnd
;**************************************************************************
; RegisterWinoldapHook(Addr,Opcode):
;
; Description: (Opcode == 1) => hook, (Opcode == 0) => unhook.
; Addr is a ptr to struct of the form WinoldapHookList
; struct WinoldapHookList { struct WinoldapHookList *ptr; DWORD Hook;};
;**************************************************************************
cProc RegisterWinoldapHook,<FAR,PUBLIC,PASCAL>,<ds,es,di,si,dx>
parmD Address
parmB Opcode
cBegin
SetKernelDSNRes ; set kernel's DS
mov ax, WinFlags
test ax, WF_STANDARD
mov ax,0
jz RWH_Call
lea si,WinAppHooks
les di,Address
cmp Opcode,0 ; unhook ?
jz RWH_Unhook
RWH_Exchange:
mov ax,ds:[si] ; old
mov es:[di],ax ; next of new
mov ax,ds:[si+2]
mov es:[di+2],ax ; next set
mov ds:[si],di
mov di,es
mov ds:[si+2],di ; new one becomes first
mov ax,1
jmp SHORT RWH_Call
RWH_Unhook:
mov dx,es
RWH_Compare:
cmp di,ds:[si]
jnz RWH_Nexthook
cmp dx,ds:[si+2]
jnz RWH_NextHook
mov ax,es:[di]
mov ds:[si],ax
mov ax,es:[di+2]
mov ds:[si+2],ax
mov ax,1
jmp SHORT RWH_Call
RWH_NextHook:
lds si,ds:[si]
mov ax,ds
or ax,si
jnz short RWH_Compare
RWH_Call:
UnSetKernelDS ; unset it
cEnd
;**********************************************************************
;
; GetWinoldapHooks:
; Description: Called exclusively by winoldap to get a pointer to a list
; of WinoldapHookList.
; Entry: None
; EXIT: DX:AX -> WinoldapHookList...
; USES: Flags.
;**********************************************************************
cProc GetWinoldapHooks,<FAR,PUBLIC,PASCAL>,<ds,si>
cBegin
SetKernelDSNRes ; set kernel DS
lea si,WinAppHooks
mov ax, word ptr ds:[si]
mov dx, word ptr ds:[si+2]
UnSetKernelDS ; unset it
cEnd
sEnd NRESCODE
end