836 lines
22 KiB
836 lines
22 KiB
5 years ago
|
/*
|
||
|
_____________________________________________________________________________
|
||
|
|
||
|
Process Functions Header v2.2
|
||
|
_____________________________________________________________________________
|
||
|
|
||
|
2008-2010 Erik Pilsits aka wraithdu
|
||
|
License: zlib/libpng
|
||
|
|
||
|
See documentation for more information about the following functions.
|
||
|
|
||
|
Usage in script:
|
||
|
1. !include "ProcFunc.nsh"
|
||
|
2. [Section|Function]
|
||
|
${ProcFunction} "Param1" "Param2" "..." $var
|
||
|
[SectionEnd|FunctionEnd]
|
||
|
|
||
|
|
||
|
ProcFunction=[GetProcessPID|GetProcessPath|GetProcessParent|GetProcessName|
|
||
|
EnumProcessPaths|ProcessWait|ProcessWait2|ProcessWaitClose|
|
||
|
CloseProcess|TerminateProcess|Execute]
|
||
|
|
||
|
There is also a LogicLib extension:
|
||
|
${If} ${ProcessExists} file.exe
|
||
|
...
|
||
|
${EndIf}
|
||
|
|
||
|
_____________________________________________________________________________
|
||
|
|
||
|
Thanks to:
|
||
|
_____________________________________________________________________________
|
||
|
|
||
|
Some functions based on work by Donald Miller and Phoenix1701@gmail.com
|
||
|
|
||
|
_____________________________________________________________________________
|
||
|
|
||
|
Individual documentation:
|
||
|
_____________________________________________________________________________
|
||
|
|
||
|
${ProcessExists} "[process]"
|
||
|
"[process]" ; Name or PID
|
||
|
|
||
|
Use with a LogicLib conditional command like If or Unless.
|
||
|
Evaluates to true if the process exists or false if it does not or
|
||
|
the CreateToolhelp32Snapshot fails.
|
||
|
|
||
|
${GetProcessPID} "[process]" $var
|
||
|
"[process]" ; Name or PID
|
||
|
|
||
|
$var(output) ; -2 - CreateToolhelp32Snapshot failed
|
||
|
; 0 - process does not exist
|
||
|
; >0 - PID
|
||
|
|
||
|
${GetProcessPath} "[process]" $var
|
||
|
"[process]" ; Name or PID
|
||
|
|
||
|
$var(output) ; -2 - CreateToolhelp32Snapshot failed
|
||
|
; -1 - OpenProcess failed
|
||
|
; 0 - process does not exist
|
||
|
; Or path to process
|
||
|
|
||
|
${GetProcessParent} "[process]" $var
|
||
|
"[process]" ; Name or PID
|
||
|
|
||
|
$var(output) ; -2 - CreateToolhelp32Snapshot failed
|
||
|
; 0 - process does not exist
|
||
|
; Or PPID
|
||
|
|
||
|
${GetProcessName} "[PID]" $var
|
||
|
"[PID]" ; PID
|
||
|
|
||
|
$var(output) ; -2 - CreateToolhelp32Snapshot failed
|
||
|
; 0 - process does not exist
|
||
|
; Or process name
|
||
|
|
||
|
${EnumProcessPaths} "Function" $var
|
||
|
"Function" ; Callback function
|
||
|
$var(output) ; -2 - EnumProcesses failed
|
||
|
; 1 - success
|
||
|
|
||
|
Function "Function"
|
||
|
Pop $var1 ; matching path string
|
||
|
Pop $var2 ; matching process PID
|
||
|
...user commands
|
||
|
Push [1/0] ; must return 1 on the stack to continue
|
||
|
; must return some value or corrupt the stack
|
||
|
; DO NOT save data in $0-$9
|
||
|
FunctionEnd
|
||
|
|
||
|
${ProcessWait} "[process]" "[timeout]" $var
|
||
|
"[process]" ; Name
|
||
|
"[timeout]" ; -1 - do not timeout
|
||
|
; >0 - timeout in milliseconds
|
||
|
|
||
|
$var(output) ; -2 - CreateToolhelp32Snapshot failed
|
||
|
; -1 - operation timed out
|
||
|
; Or PID
|
||
|
|
||
|
${ProcessWait2} "[process]" "[timeout]" $var
|
||
|
"[process]" ; Name
|
||
|
"[timeout]" ; -1 - do not timeout
|
||
|
; >0 - timeout in milliseconds
|
||
|
|
||
|
$var(output) ; -1 - operation timed out
|
||
|
; Or PID
|
||
|
|
||
|
${ProcessWaitClose} "[process]" "[timeout]" $var
|
||
|
"[process]" ; Name
|
||
|
"[timeout]" ; -1 - do not timeout
|
||
|
; >0 - timeout in milliseconds
|
||
|
|
||
|
$var(output) ; -1 - operation timed out
|
||
|
; 0 - process does not exist
|
||
|
; Or PID of ended process
|
||
|
|
||
|
${CloseProcess} "[process]" $var
|
||
|
"[process]" ; Name or PID
|
||
|
|
||
|
$var(output) ; 0 - process does not exist
|
||
|
; Or PID of ended process
|
||
|
|
||
|
${TerminateProcess} "[process]" $var
|
||
|
"[process]" ; Name or PID
|
||
|
|
||
|
$var(output) ; -1 - operation failed
|
||
|
; 0 - process does not exist
|
||
|
; Or PID of ended process
|
||
|
|
||
|
${Execute} "[command]" "[working_dir]" $var
|
||
|
"[command]" ; '"X:\path\to\prog.exe" arg1 arg2 "arg3 with space"'
|
||
|
"[working_dir]" ; Working directory ("X:\path\to\dir") or nothing ("")
|
||
|
|
||
|
$var(output) ; 0 - failed to create process
|
||
|
; Or PID
|
||
|
*/
|
||
|
|
||
|
|
||
|
;_____________________________________________________________________________
|
||
|
;
|
||
|
; Macros
|
||
|
;_____________________________________________________________________________
|
||
|
;
|
||
|
; Change log window verbosity (default: 3=no script)
|
||
|
;
|
||
|
; Example:
|
||
|
; !include "ProcFunc.nsh"
|
||
|
; ${PROCFUNC_VERBOSE} 4 # all verbosity
|
||
|
; ${PROCFUNC_VERBOSE} 3 # no script
|
||
|
|
||
|
!ifndef PROCFUNC_INCLUDED
|
||
|
!define PROCFUNC_INCLUDED
|
||
|
|
||
|
!include Util.nsh
|
||
|
!include LogicLib.nsh
|
||
|
|
||
|
!verbose push
|
||
|
!verbose 3
|
||
|
!ifndef _PROCFUNC_VERBOSE
|
||
|
!define _PROCFUNC_VERBOSE 3
|
||
|
!endif
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
!define PROCFUNC_VERBOSE `!insertmacro PROCFUNC_VERBOSE`
|
||
|
!verbose pop
|
||
|
|
||
|
!macro PROCFUNC_VERBOSE _VERBOSE
|
||
|
!verbose push
|
||
|
!verbose 3
|
||
|
!undef _PROCFUNC_VERBOSE
|
||
|
!define _PROCFUNC_VERBOSE ${_VERBOSE}
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
|
||
|
!define PROCESS_QUERY_INFORMATION 0x0400
|
||
|
!define PROCESS_TERMINATE 0x0001
|
||
|
!define PROCESS_VM_READ 0x0010
|
||
|
!define SYNCHRONIZE 0x00100000
|
||
|
|
||
|
!define WAIT_TIMEOUT 0x00000102
|
||
|
|
||
|
!ifdef NSIS_UNICODE
|
||
|
!define _PROCFUNC_WSTRING "&w260"
|
||
|
!else
|
||
|
!define _PROCFUNC_WSTRING "&w520"
|
||
|
!endif
|
||
|
|
||
|
!macro ProcessExists
|
||
|
!error "ProcessExists has been renamed to GetProcessPID"
|
||
|
!macroend
|
||
|
!macro _ProcessExists _a _b _t _f
|
||
|
!insertmacro _LOGICLIB_TEMP
|
||
|
!verbose push
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
Push `${_b}`
|
||
|
${CallArtificialFunction} LLProcessExists_
|
||
|
IntCmp $_LOGICLIB_TEMP 0 `${_f}`
|
||
|
Goto `${_t}`
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
!define ProcessExists `"" ProcessExists`
|
||
|
|
||
|
!macro GetProcessPID
|
||
|
!macroend
|
||
|
!define GetProcessPID "!insertmacro GetProcessPIDCall"
|
||
|
!macro GetProcessPIDCall process outVar
|
||
|
!verbose push
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
Push 0
|
||
|
Push `${process}`
|
||
|
!ifdef CallArtificialFunction_TYPE ; macro nesting disallowed, breaks otherwise if used from WaitClose
|
||
|
${CallArtificialFunction2} ProcFuncs_
|
||
|
!else
|
||
|
${CallArtificialFunction} ProcFuncs_
|
||
|
!endif
|
||
|
Pop ${outVar}
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
|
||
|
!macro GetProcessPath
|
||
|
!macroend
|
||
|
!define GetProcessPath "!insertmacro GetProcessPathCall"
|
||
|
!macro GetProcessPathCall process outVar
|
||
|
!verbose push
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
Push 1
|
||
|
Push `${process}`
|
||
|
${CallArtificialFunction} ProcFuncs_
|
||
|
Pop ${outVar}
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
|
||
|
!macro GetProcessParent
|
||
|
!macroend
|
||
|
!define GetProcessParent "!insertmacro GetProcessParentCall"
|
||
|
!macro GetProcessParentCall process outVar
|
||
|
!verbose push
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
Push 2
|
||
|
Push `${process}`
|
||
|
${CallArtificialFunction} ProcFuncs_
|
||
|
Pop ${outVar}
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
|
||
|
!macro GetProcessName
|
||
|
!macroend
|
||
|
!define GetProcessName "!insertmacro GetProcessNameCall"
|
||
|
!macro GetProcessNameCall process outVar
|
||
|
!verbose push
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
Push 6
|
||
|
Push `${process}`
|
||
|
${CallArtificialFunction} ProcFuncs_
|
||
|
Pop ${outVar}
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
|
||
|
!macro EnumProcessPaths
|
||
|
!macroend
|
||
|
!define EnumProcessPaths "!insertmacro EnumProcessPathsCall"
|
||
|
!macro EnumProcessPathsCall user_func outVar
|
||
|
!verbose push
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
Push $0
|
||
|
GetFunctionAddress $0 `${user_func}`
|
||
|
Push `$0`
|
||
|
${CallArtificialFunction} EnumProcessPaths_
|
||
|
Exch
|
||
|
Pop $0
|
||
|
Pop ${outVar}
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
|
||
|
!macro ProcessWait
|
||
|
!macroend
|
||
|
!define ProcessWait "!insertmacro ProcessWaitCall"
|
||
|
!macro ProcessWaitCall process timeout outVar
|
||
|
!verbose push
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
Push `${timeout}`
|
||
|
Push `${process}`
|
||
|
${CallArtificialFunction} ProcessWait_
|
||
|
Pop ${outVar}
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
|
||
|
!macro ProcessWait2
|
||
|
!macroend
|
||
|
!define ProcessWait2 "!insertmacro ProcessWait2Call"
|
||
|
!macro ProcessWait2Call process timeout outVar
|
||
|
!verbose push
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
Push `${timeout}`
|
||
|
Push `${process}`
|
||
|
${CallArtificialFunction} ProcessWait2_
|
||
|
Pop ${outVar}
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
|
||
|
!macro ProcessWaitClose
|
||
|
!macroend
|
||
|
!define ProcessWaitClose "!insertmacro ProcessWaitCloseCall"
|
||
|
!macro ProcessWaitCloseCall process timeout outVar
|
||
|
!verbose push
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
Push `${timeout}`
|
||
|
Push `${process}`
|
||
|
${CallArtificialFunction} ProcessWaitClose_
|
||
|
Pop ${outVar}
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
|
||
|
!macro CloseProcess
|
||
|
!macroend
|
||
|
!define CloseProcess "!insertmacro CloseProcessCall"
|
||
|
!macro CloseProcessCall process outVar
|
||
|
!verbose push
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
Push `${process}`
|
||
|
${CallArtificialFunction} CloseProcess_
|
||
|
Pop ${outVar}
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
|
||
|
!macro TerminateProcess
|
||
|
!macroend
|
||
|
!define TerminateProcess "!insertmacro TerminateProcessCall"
|
||
|
!macro TerminateProcessCall process outVar
|
||
|
!verbose push
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
Push `${process}`
|
||
|
${CallArtificialFunction} TerminateProcess_
|
||
|
Pop ${outVar}
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
|
||
|
!macro Execute
|
||
|
!macroend
|
||
|
!define Execute "!insertmacro ExecuteCall"
|
||
|
!macro ExecuteCall cmdline wrkdir outVar
|
||
|
!verbose push
|
||
|
!verbose ${_PROCFUNC_VERBOSE}
|
||
|
Push `${wrkdir}`
|
||
|
Push `${cmdline}`
|
||
|
${CallArtificialFunction} Execute_
|
||
|
Pop ${outVar}
|
||
|
!verbose pop
|
||
|
!macroend
|
||
|
|
||
|
!macro ProcFuncs_
|
||
|
System::Store "s" ; store registers in System's private stack
|
||
|
Pop $0 ; process / PID
|
||
|
Pop $1 ; mode
|
||
|
|
||
|
Push 0 ; set return value if not found
|
||
|
|
||
|
; set mode of operation in $1
|
||
|
${Select} $1 ; mode 0 = GetProcessPID, mode 1 = GetProcessPath, mode 2 = GetProcessParent
|
||
|
${Case} 0
|
||
|
StrCpy $2 $0 4 -4
|
||
|
${If} $2 == ".exe"
|
||
|
; exists from process name
|
||
|
StrCpy $1 0
|
||
|
${Else}
|
||
|
; exists from pid
|
||
|
StrCpy $1 1
|
||
|
${EndIf}
|
||
|
${Case} 1
|
||
|
StrCpy $2 $0 4 -4
|
||
|
${If} $2 == ".exe"
|
||
|
; get path from process name
|
||
|
StrCpy $1 2
|
||
|
${Else}
|
||
|
; get path from pid
|
||
|
StrCpy $1 3
|
||
|
${EndIf}
|
||
|
${Case} 2
|
||
|
StrCpy $2 $0 4 -4
|
||
|
${If} $2 == ".exe"
|
||
|
; get parent from process name
|
||
|
StrCpy $1 4
|
||
|
${Else}
|
||
|
; get parent from pid
|
||
|
StrCpy $1 5
|
||
|
${EndIf}
|
||
|
${EndSelect}
|
||
|
|
||
|
System::Call '*(&l4,i,i,i,i,i,i,i,i,${_PROCFUNC_WSTRING})i .r2' ; $2 = PROCESSENTRY32W structure
|
||
|
; take system process snapshot in $3
|
||
|
System::Call 'kernel32::CreateToolhelp32Snapshot(i 2, i 0)i .r3'
|
||
|
${Unless} $3 = -1
|
||
|
System::Call 'kernel32::Process32FirstW(i r3, i r2)i .r4'
|
||
|
${Unless} $4 = 0
|
||
|
${Do}
|
||
|
${Select} $1
|
||
|
${Case3} 0 2 4
|
||
|
; get process name in $5
|
||
|
System::Call '*$2(i,i,i,i,i,i,i,i,i,${_PROCFUNC_WSTRING} .r5)'
|
||
|
${Case4} 1 3 5 6
|
||
|
; get process PID in $5
|
||
|
System::Call '*$2(i,i,i .r5)'
|
||
|
${EndSelect}
|
||
|
; is this process the one we are looking for?
|
||
|
${If} $5 == $0 ; string test works ok for numeric PIDs as well
|
||
|
${Select} $1 ; mode 0/1 = GetProcessPID, mode 2/3 = GetProcessPath, mode 4/5 = GetProcessParent, mode 6 = GetProcessName
|
||
|
${Case2} 0 1
|
||
|
; return pid
|
||
|
Pop $5 ; old return value
|
||
|
System::Call '*$2(i,i,i .s)'; process pid to stack
|
||
|
${Case2} 2 3
|
||
|
; return full path
|
||
|
Pop $5
|
||
|
; open process
|
||
|
System::Call '*$2(i,i,i .s)'; process pid to stack
|
||
|
System::Call 'kernel32::OpenProcess(i ${PROCESS_QUERY_INFORMATION}|${PROCESS_VM_READ}, i 0, i s)i .r5' ; process handle to $5
|
||
|
${Unless} $5 = 0
|
||
|
; full path to stack
|
||
|
System::Call 'psapi::GetModuleFileNameExW(i r5, i 0, w .s, i ${NSIS_MAX_STRLEN})'
|
||
|
System::Call 'kernel32::CloseHandle(i r5)'
|
||
|
${Else}
|
||
|
Push -1 ; OpenProcess failure return value
|
||
|
${EndUnless}
|
||
|
${Case2} 4 5
|
||
|
; return parent PID
|
||
|
Pop $5
|
||
|
System::Call '*$2(i,i,i,i,i,i,i .s)'; parent pid to stack
|
||
|
${Case} 6
|
||
|
; return base name
|
||
|
Pop $5
|
||
|
System::Call '*$2(i,i,i,i,i,i,i,i,i,${_PROCFUNC_WSTRING} .s)'
|
||
|
${EndSelect}
|
||
|
${Break}
|
||
|
${EndIf}
|
||
|
System::Call 'kernel32::Process32NextW(i r3, i r2)i .r4'
|
||
|
${LoopUntil} $4 = 0
|
||
|
System::Call 'kernel32::CloseHandle(i r3)' ; close snapshot
|
||
|
${EndUnless}
|
||
|
${Else}
|
||
|
Pop $5
|
||
|
Push -2 ; function failure return value
|
||
|
${EndUnless}
|
||
|
System::Free $2 ; free buffer
|
||
|
|
||
|
System::Store "l" ; restore registers
|
||
|
!macroend
|
||
|
|
||
|
!macro EnumProcessPaths_
|
||
|
System::Store "s" ; store registers in System's private stack
|
||
|
Pop $0 ; user_func
|
||
|
|
||
|
StrCpy $1 1 ; OK to loop
|
||
|
|
||
|
System::Alloc 1024
|
||
|
Pop $2 ; process list buffer
|
||
|
; get an array of all process ids
|
||
|
System::Call 'psapi::EnumProcesses(i r2, i 1024, *i .r3)i .r4' ; $3 = sizeof buffer
|
||
|
${Unless} $4 = 0
|
||
|
IntOp $3 $3 / 4 ; Divide by sizeof(DWORD) to get $3 process count
|
||
|
IntOp $3 $3 - 1 ; decrement for 0 base loop
|
||
|
${For} $4 0 $3
|
||
|
${IfThen} $1 != 1 ${|} ${Break} ${|}
|
||
|
; get a PID from the array
|
||
|
IntOp $5 $4 * 4 ; calculate offset
|
||
|
IntOp $5 $5 + $2 ; add offset to original buffer address
|
||
|
System::Call '*$5(i .r5)' ; get next PID = $5
|
||
|
${Unless} $5 = 0
|
||
|
System::Call 'kernel32::OpenProcess(i ${PROCESS_QUERY_INFORMATION}|${PROCESS_VM_READ}, i 0, i r5)i .r6'
|
||
|
${Unless} $6 = 0 ; $6 is hProcess
|
||
|
; get full path
|
||
|
System::Call 'psapi::GetModuleFileNameExW(i r6, i 0, w .r7, i ${NSIS_MAX_STRLEN})i .r8' ; $7 = path
|
||
|
${Unless} $8 = 0 ; no path
|
||
|
System::Store "s" ; store registers in System's private stack
|
||
|
Push $5 ; PID to stack
|
||
|
Push $7 ; path to stack
|
||
|
Call $0 ; user func must return 1 on the stack to continue looping
|
||
|
System::Store "l" ; restore registers
|
||
|
Pop $1 ; continue?
|
||
|
${EndUnless}
|
||
|
System::Call 'kernel32::CloseHandle(i r6)'
|
||
|
${EndUnless}
|
||
|
${EndUnless}
|
||
|
${Next}
|
||
|
Push 1 ; return value
|
||
|
${Else}
|
||
|
Push -2 ; function failure return value
|
||
|
${EndUnless}
|
||
|
System::Free $2 ; free buffer
|
||
|
|
||
|
System::Store "l" ; restore registers
|
||
|
!macroend
|
||
|
|
||
|
!macro ProcessWait_
|
||
|
System::Store "s" ; store registers in System's private stack
|
||
|
Pop $0 ; process
|
||
|
Pop $1 ; timeout
|
||
|
|
||
|
StrCpy $6 1 ; initialize loop
|
||
|
StrCpy $7 0 ; initialize timeout counter
|
||
|
|
||
|
System::Call '*(&l4,i,i,i,i,i,i,i,i,${_PROCFUNC_WSTRING})i .r2' ; $2 = PROCESSENTRY32W structure
|
||
|
${DoWhile} $6 = 1 ; processwait loop
|
||
|
; take system process snapshot in $3
|
||
|
System::Call 'kernel32::CreateToolhelp32Snapshot(i 2, i 0)i .r3'
|
||
|
${Unless} $3 = -1
|
||
|
System::Call 'kernel32::Process32FirstW(i r3, i r2)i .r4'
|
||
|
${Unless} $4 = 0
|
||
|
${Do}
|
||
|
; get process name in $5
|
||
|
System::Call '*$2(i,i,i,i,i,i,i,i,i,${_PROCFUNC_WSTRING} .r5)'
|
||
|
${If} $5 == $0
|
||
|
; exists, return pid
|
||
|
System::Call '*$2(i,i,i .s)'; process pid to stack ; process pid
|
||
|
StrCpy $6 0 ; end loop
|
||
|
${Break}
|
||
|
${EndIf}
|
||
|
System::Call 'kernel32::Process32NextW(i r3, i r2)i .r4'
|
||
|
${LoopUntil} $4 = 0
|
||
|
System::Call 'kernel32::CloseHandle(i r3)' ; close snapshot
|
||
|
${EndUnless}
|
||
|
${Else}
|
||
|
Push -2
|
||
|
${Break}
|
||
|
${EndUnless}
|
||
|
; timeout loop
|
||
|
${If} $6 = 1
|
||
|
${If} $1 >= 0
|
||
|
IntOp $7 $7 + 500 ; increment timeout counter
|
||
|
${AndIf} $7 >= $1 ; timed out, break loop
|
||
|
Push -1 ; timeout return value
|
||
|
${Break} ; end loop if timeout
|
||
|
${EndIf}
|
||
|
Sleep 500 ; pause before looping
|
||
|
${EndIf}
|
||
|
${Loop} ; processwaitloop
|
||
|
System::Free $2 ; free buffer
|
||
|
|
||
|
System::Store "l" ; restore registers
|
||
|
!macroend
|
||
|
|
||
|
!macro ProcessWait2_
|
||
|
System::Store "s" ; store registers in System's private stack
|
||
|
System::Store "P0" ; FindProcDLL return value
|
||
|
Pop $0 ; process
|
||
|
Pop $1 ; timeout
|
||
|
|
||
|
StrCpy $2 0 ; initialize timeout counter
|
||
|
|
||
|
${Do}
|
||
|
FindProcDLL::FindProc $0
|
||
|
${IfThen} $R0 = 1 ${|} ${Break} ${|}
|
||
|
${If} $1 >= 0
|
||
|
IntOp $2 $2 + 250
|
||
|
${AndIf} $2 >= $1
|
||
|
Push -1 ; timeout return value
|
||
|
${Break}
|
||
|
${EndIf}
|
||
|
Sleep 250
|
||
|
${Loop}
|
||
|
|
||
|
${If} $R0 = 1 ; success, get pid
|
||
|
${GetProcessPID} $0 $0
|
||
|
Push $0 ; return pid
|
||
|
${EndIf}
|
||
|
|
||
|
System::Store "R0" ; restore registers
|
||
|
System::Store "l"
|
||
|
!macroend
|
||
|
|
||
|
!macro ProcessWaitClose_
|
||
|
System::Store "s" ; store registers in System's private stack
|
||
|
Pop $0 ; process / PID
|
||
|
Pop $1 ; timeout
|
||
|
|
||
|
; passed process name or pid
|
||
|
StrCpy $2 $0 4 -4
|
||
|
${If} $2 == ".exe"
|
||
|
${GetProcessPID} $0 $0
|
||
|
${EndIf}
|
||
|
|
||
|
; else passed pid directly
|
||
|
|
||
|
${Unless} $0 = 0
|
||
|
System::Call 'kernel32::OpenProcess(i ${SYNCHRONIZE}, i 0, i r0)i .r2'
|
||
|
${Unless} $2 = 0 ; $2 is hProcess
|
||
|
System::Call 'kernel32::WaitForSingleObject(i r2, i $1)i .r1'
|
||
|
${If} $1 = ${WAIT_TIMEOUT}
|
||
|
Push -1 ; timed out
|
||
|
${Else}
|
||
|
Push $0 ; return pid of ended process
|
||
|
${EndIf}
|
||
|
System::Call 'kernel32::CloseHandle(i r2)'
|
||
|
${Else}
|
||
|
Push 0 ; failure return value
|
||
|
${EndUnless}
|
||
|
${Else}
|
||
|
Push 0 ; failure return value
|
||
|
${EndUnless}
|
||
|
|
||
|
System::Store "l" ; restore registers
|
||
|
!macroend
|
||
|
|
||
|
!macro CloseProcess_
|
||
|
System::Store "s" ; store registers in System's private stack
|
||
|
Pop $0 ; process / PID
|
||
|
|
||
|
; passed process name or pid
|
||
|
StrCpy $1 $0 4 -4
|
||
|
${If} $1 == ".exe"
|
||
|
${GetProcessPID} $0 $0
|
||
|
${EndIf}
|
||
|
|
||
|
; else passed pid directly
|
||
|
|
||
|
${Unless} $0 = 0 ; $0 = target pid
|
||
|
Push $0 ; return pid of process
|
||
|
; use EnumWindows and a callback
|
||
|
System::Get '(i .r1, i)i sr4' ; $1 = hwnd, $4 = callback#, s (stack) = source for return value
|
||
|
Pop $3 ; $3 = callback address
|
||
|
System::Call 'user32::EnumWindows(k r3, i)i' ; enumerate top-level windows
|
||
|
${DoWhile} $4 == "callback1"
|
||
|
System::Call 'user32::GetWindowThreadProcessId(i r1, *i .r2)i' ; $2 = pid that created the window
|
||
|
${If} $2 = $0 ; match to target pid
|
||
|
SendMessage $1 16 0 0 /TIMEOUT=1 ; send WM_CLOSE to all top-level windows owned by process, timeout immediately
|
||
|
${EndIf}
|
||
|
Push 1 ; callback return value; keep enumerating windows (returning 0 stops)
|
||
|
StrCpy $4 "" ; clear callback#
|
||
|
System::Call '$3' ; return from callback
|
||
|
${Loop}
|
||
|
System::Free $3 ; free callback
|
||
|
${Else}
|
||
|
Push 0 ; failure return value
|
||
|
${EndUnless}
|
||
|
|
||
|
System::Store "l" ; restore registers
|
||
|
!macroend
|
||
|
|
||
|
!macro TerminateProcess_
|
||
|
System::Store "s" ; store registers in System's private stack
|
||
|
Pop $0 ; process / PID
|
||
|
|
||
|
; passed process name or pid
|
||
|
StrCpy $1 $0 4 -4
|
||
|
${If} $1 == ".exe"
|
||
|
${GetProcessPID} $0 $0
|
||
|
${EndIf}
|
||
|
|
||
|
; else passed pid directly
|
||
|
|
||
|
${Unless} $0 = 0
|
||
|
System::Call 'kernel32::OpenProcess(i ${PROCESS_TERMINATE}, i 0, i r0)i .r1'
|
||
|
${Unless} $1 = 0 ; $1 is hProcess
|
||
|
System::Call 'kernel32::TerminateProcess(i r1, i 0)i .r1'
|
||
|
${If} $1 = 0 ; fail
|
||
|
Push -1
|
||
|
${Else}
|
||
|
Push $0 ; return pid of ended process
|
||
|
${EndIf}
|
||
|
System::Call 'kernel32::CloseHandle(i r1)'
|
||
|
${Else}
|
||
|
Push 0 ; failure return value
|
||
|
${EndUnless}
|
||
|
${Else}
|
||
|
Push 0 ; failure return value
|
||
|
${EndUnless}
|
||
|
|
||
|
System::Store "l" ; restore registers
|
||
|
!macroend
|
||
|
|
||
|
!macro Execute_
|
||
|
System::Store "s" ; store registers in System's private stack
|
||
|
Pop $0 ; cmdline
|
||
|
Pop $1 ; wrkdir
|
||
|
|
||
|
System::Alloc 68 ; 4*16 + 2*2 / STARTUPINFO structure = $2
|
||
|
Pop $2
|
||
|
System::Call '*$2(i 68)' ; set cb = sizeof(STARTUPINFO)
|
||
|
System::Call '*(i,i,i,i)i .r3' ; PROCESS_INFORMATION structure = $3
|
||
|
|
||
|
${If} $1 == ""
|
||
|
StrCpy $1 "i"
|
||
|
${Else}
|
||
|
StrCpy $1 'w "$1"'
|
||
|
${EndIf}
|
||
|
|
||
|
System::Call `kernel32::CreateProcessW(i, w '$0', i, i, i 0, i 0, i, $1, i r2, i r3)i .r4` ; return 0 if fail
|
||
|
${Unless} $4 = 0 ; failed to create process
|
||
|
System::Call '*$3(i .r4, i .r5, i .r6)' ; read handles and PID
|
||
|
System::Call 'kernel32::CloseHandle(i $4)' ; close hProcess
|
||
|
System::Call 'kernel32::CloseHandle(i $5)' ; close hThread
|
||
|
Push $6 ; return PID
|
||
|
${Else}
|
||
|
Push 0 ; return val if failed
|
||
|
${EndUnless}
|
||
|
|
||
|
System::Free $2 ; free STARTUPINFO struct
|
||
|
System::Free $3 ; free PROCESS_INFORMATION struct
|
||
|
|
||
|
System::Store "l" ; restore registers
|
||
|
!macroend
|
||
|
|
||
|
!macro LLProcessExists_
|
||
|
System::Store "s" ; store registers in System's private stack
|
||
|
Pop $0 ; process name
|
||
|
|
||
|
StrCpy $_LOGICLIB_TEMP 0
|
||
|
|
||
|
System::Call '*(&l4,i,i,i,i,i,i,i,i,${_PROCFUNC_WSTRING})i .r2' ; $2 = PROCESSENTRY32W structure
|
||
|
; take system process snapshot in $3
|
||
|
System::Call 'kernel32::CreateToolhelp32Snapshot(i 2, i 0)i .r3'
|
||
|
IntCmp $3 -1 done
|
||
|
System::Call 'kernel32::Process32FirstW(i r3, i r2)i .r4'
|
||
|
IntCmp $4 0 endloop
|
||
|
loop:
|
||
|
System::Call '*$2(i,i,i,i,i,i,i,i,i,${_PROCFUNC_WSTRING} .r5)'
|
||
|
StrCmp $5 $0 0 next_process
|
||
|
StrCpy $_LOGICLIB_TEMP 1
|
||
|
Goto endloop
|
||
|
next_process:
|
||
|
System::Call 'kernel32::Process32NextW(i r3, i r2)i .r4'
|
||
|
IntCmp $4 0 endloop
|
||
|
Goto loop
|
||
|
endloop:
|
||
|
System::Call 'kernel32::CloseHandle(i r3)' ; close snapshot
|
||
|
done:
|
||
|
System::Free $2 ; free buffer
|
||
|
|
||
|
System::Store "l" ; restore registers
|
||
|
!macroend
|
||
|
|
||
|
!endif ; PROCFUNC_INCLUDED
|
||
|
|
||
|
/****************************************************************************
|
||
|
Functions
|
||
|
=========
|
||
|
|
||
|
HANDLE WINAPI OpenProcess(
|
||
|
__in DWORD dwDesiredAccess,
|
||
|
__in BOOL bInheritHandle,
|
||
|
__in DWORD dwProcessId
|
||
|
);
|
||
|
|
||
|
BOOL WINAPI CreateProcess(
|
||
|
__in_opt LPCTSTR lpApplicationName,
|
||
|
__inout_opt LPTSTR lpCommandLine,
|
||
|
__in_opt LPSECURITY_ATTRIBUTES lpProcessAttributes,
|
||
|
__in_opt LPSECURITY_ATTRIBUTES lpThreadAttributes,
|
||
|
__in BOOL bInheritHandles,
|
||
|
__in DWORD dwCreationFlags,
|
||
|
__in_opt LPVOID lpEnvironment,
|
||
|
__in_opt LPCTSTR lpCurrentDirectory,
|
||
|
__in LPSTARTUPINFO lpStartupInfo,
|
||
|
__out LPPROCESS_INFORMATION lpProcessInformation
|
||
|
);
|
||
|
|
||
|
typedef struct _STARTUPINFO {
|
||
|
DWORD cb;
|
||
|
LPTSTR lpReserved;
|
||
|
LPTSTR lpDesktop;
|
||
|
LPTSTR lpTitle;
|
||
|
DWORD dwX;
|
||
|
DWORD dwY;
|
||
|
DWORD dwXSize;
|
||
|
DWORD dwYSize;
|
||
|
DWORD dwXCountChars;
|
||
|
DWORD dwYCountChars;
|
||
|
DWORD dwFillAttribute;
|
||
|
DWORD dwFlags;
|
||
|
WORD wShowWindow;
|
||
|
WORD cbReserved2;
|
||
|
LPBYTE lpReserved2;
|
||
|
HANDLE hStdInput;
|
||
|
HANDLE hStdOutput;
|
||
|
HANDLE hStdError;
|
||
|
} STARTUPINFO,
|
||
|
*LPSTARTUPINFO;
|
||
|
|
||
|
typedef struct _PROCESS_INFORMATION {
|
||
|
HANDLE hProcess;
|
||
|
HANDLE hThread;
|
||
|
DWORD dwProcessId;
|
||
|
DWORD dwThreadId;
|
||
|
} PROCESS_INFORMATION,
|
||
|
*LPPROCESS_INFORMATION;
|
||
|
|
||
|
BOOL WINAPI EnumProcesses(
|
||
|
__out DWORD* pProcessIds,
|
||
|
__in DWORD cb,
|
||
|
__out DWORD* pBytesReturned
|
||
|
);
|
||
|
|
||
|
DWORD WINAPI GetModuleBaseName(
|
||
|
__in HANDLE hProcess,
|
||
|
__in_opt HMODULE hModule,
|
||
|
__out LPTSTR lpBaseName,
|
||
|
__in DWORD nSize
|
||
|
);
|
||
|
|
||
|
DWORD WINAPI GetModuleFileNameEx(
|
||
|
__in HANDLE hProcess,
|
||
|
__in_opt HMODULE hModule,
|
||
|
__out LPTSTR lpFilename,
|
||
|
__in DWORD nSize
|
||
|
);
|
||
|
|
||
|
BOOL WINAPI CloseHandle(
|
||
|
__in HANDLE hObject
|
||
|
);
|
||
|
|
||
|
DWORD WINAPI WaitForSingleObject(
|
||
|
__in HANDLE hHandle,
|
||
|
__in DWORD dwMilliseconds
|
||
|
);
|
||
|
|
||
|
BOOL WINAPI TerminateProcess(
|
||
|
__in HANDLE hProcess,
|
||
|
__in UINT uExitCode
|
||
|
);
|
||
|
|
||
|
BOOL EnumWindows(
|
||
|
__in WNDENUMPROC lpEnumFunc,
|
||
|
__in LPARAM lParam
|
||
|
);
|
||
|
|
||
|
DWORD GetWindowThreadProcessId(
|
||
|
__in HWND hWnd,
|
||
|
__out LPDWORD lpdwProcessId
|
||
|
);
|
||
|
|
||
|
BOOL PostMessage(
|
||
|
__in HWND hWnd,
|
||
|
__in UINT Msg,
|
||
|
__in WPARAM wParam,
|
||
|
__in LPARAM lParam
|
||
|
);
|
||
|
|
||
|
****************************************************************************/
|