|
|
/*++
Copyright (c) 1988-1999 Microsoft Corporation
Module Name:
cext.c
Abstract:
External command support
--*/
#include "cmd.h"
#define DBENV 0x0080
#define DBENVSCAN 0x0010
unsigned start_type ; /* D64 */
extern UINT CurrentCP; extern TCHAR Fmt16[] ; /* @@5h */
extern unsigned DosErr ; extern BOOL CtrlCSeen;
extern TCHAR CurDrvDir[] ;
extern TCHAR CmdExt[], BatExt[], PathStr[] ; extern TCHAR PathExtStr[], PathExtDefaultStr[]; extern TCHAR ComSpec[] ; /* M033 - Use ComSpec for SM memory */ extern TCHAR ComSpecStr[] ; /* M033 - Use ComSpec for SM memory */ extern void tokshrink(TCHAR*);
extern TCHAR PathChar ; extern TCHAR SwitChar ;
extern PTCHAR pszTitleCur; extern BOOLEAN fTitleChanged;
extern int LastRetCode ; extern HANDLE PipePid ; /* M024 - Store PID from piped cmd */
extern struct envdata CmdEnv ; // Holds info to manipulate Cmd's environment
extern int glBatType; // to distinguish OS/2 vs DOS errorlevel behavior depending on a script file name
TCHAR szNameEqExitCodeEnvVar[] = TEXT ("=ExitCode"); TCHAR szNameEqExitCodeAsciiEnvVar[] = TEXT ("=ExitCodeAscii"); TCHAR ShellOpenCommandString[] = TEXT( "\\Shell\\Open\\Command" );
WORD GetProcessSubsystemType( HANDLE hProcess );
/*** ExtCom - controls the execution of external programs
* * Purpose: * Synchronously execute an external command. Call ECWork with the * appropriate values to have this done. * * ExtCom(struct cmdnode *n) * * Args: * Parse tree node containing the command to be executed. * * Returns: * Whatever ECWork returns. * * Notes: * During batch processing, labels are ignored. Empty commands are * also ignored. * */
int ExtCom(n) struct cmdnode *n ; { if (CurrentBatchFile && *n->cmdline == COLON) return(SUCCESS) ;
if (n && n->cmdline && mystrlen(n->cmdline)) { return(ECWork(n, AI_SYNC, CW_W_YES)) ; /* M024 */ } ;
return(SUCCESS) ; }
/********************* START OF SPECIFICATION **************************/ /* */ /* SUBROUTINE NAME: ECWork */ /* */ /* DESCRIPTIVE NAME: Execute External Commands Worker */ /* */ /* FUNCTION: Execute External Commands */ /* This routine calls SearchForExecutable routine to search */ /* for the executable command. If the command ( .EXE, .COM, */ /* or .CMD file ) is found, the command is executed. */ /* */ /* ENTRY POINT: ECWork */ /* LINKAGE: NEAR */ /* */ /* INPUT: n - the parse tree node containing the command to be executed*/ /* */ /* ai - the asynchronous indicator */ /* - 0 = Exec synchronous with parent */ /* - 1 = Exec asynchronous and discard child return code */ /* - 2 = Exec asynchronous and save child return code */ /* */ /* wf - the wait flag */ /* - 0 = Wait for process completion */ /* - 1 = Return immediately (Pipes) */ /* */ /* OUTPUT: None. */ /* */ /* EXIT-NORMAL: */ /* If synchronous execution, the return code of the command is */ /* returned. */ /* */ /* If asynchronous execution, the return code of the exec call */ /* is returned. */ /* */ /* EXIT-ERROR: */ /* Return FAILURE to the caller. */ /* */ /* */ /********************** END OF SPECIFICATION **************************/ /*** ECWork - begins the execution of external commands
* * Purpose: * To search for and execute an external command. Update LastRetCode * if an external program was executed. * * int ECWork(struct cmdnode *n, unsigned ai, unsigned wf) * * Args: * n - the parse tree node containing the command to be executed * ai - the asynchronous indicator * - 0 = Exec synchronous with parent * - 1 = Exec asynchronous and discard child return code * - 2 = Exec asynchronous and save child return code * wf - the wait flag * - 0 = Wait for process completion * - 1 = Return immediately (Pipes) * * Returns: * If syncronous execution, the return code of the command is returned. * If asyncronous execution, the return code of the exec call is returned. * * Notes: * The pid of a program that will be waited on is placed in the global * variable Retcds.ptcod so that WaitProc can use it and so SigHand() * can kill it if necessary (only during SYNC exec's). * M024 - Added wait flag parm so pipes can get immediate return while * still doing an AI_KEEP async exec. * - Considerable revisions to structure. */
int ECWork(n, ai, wf) struct cmdnode *n ; unsigned ai ; unsigned wf ; { TCHAR *fnptr, /* Ptr to filename */ *argptr, /* Command Line String */ *tptr; /* M034 - Temps */ int i ; /* Work variable */ TCHAR * onb ; /* M035 - Obj name buffer */ ULONG rc;
if (!(fnptr = mkstr(MAX_PATH*sizeof(TCHAR)+sizeof(TCHAR)))) /* Loc/nam of file to exec */ return(FAILURE) ;
argptr = GetTitle( n ); tptr = argptr; if (argptr == NULL) {
return( FAILURE );
}
i = SearchForExecutable(n, fnptr) ; if (i == SFE_ISEXECOM) { /* Found .com or .exe file */
if (!(onb = (TCHAR *)mkstr(MAX_PATH*sizeof(TCHAR)+sizeof(TCHAR)))) /* M035 */ { return(FAILURE) ;
}
SetConTitle( tptr ); rc = ExecPgm( n, onb, ai, wf, argptr, fnptr, tptr); ResetConTitle( pszTitleCur );
return( rc );
} ;
if (i == SFE_ISBAT) { /* Found .cmd file, call BatProc */
SetConTitle(tptr); rc = BatProc(n, fnptr, BT_CHN) ; ResetConTitle( pszTitleCur ); return(rc) ; } ;
if (i == SFE_ISDIR) { return ChangeDir2(fnptr, TRUE); }
LastRetCode = DosErr; if (i == SFE_FAIL) { /* Out of Memory Error */ return(FAILURE) ; } ;
if (DosErr == MSG_DIR_BAD_COMMAND_OR_FILE) { PutStdErr(DosErr, ONEARG, n->cmdline); } else { PutStdErr(DosErr, NOARGS); } return(FAILURE) ; }
#ifndef WIN95_CMD
VOID RestoreCurrentDirectories( VOID )
/* this routine sets the current process's current directories to
those of the child if the child was the VDM to fix DOS batch files. */
{ ULONG cchCurDirs; UINT PreviousErrorMode; PTCHAR pCurDirs,pCurCurDir; BOOL CurDrive=TRUE; #ifdef UNICODE
PCHAR pT; #endif
cchCurDirs = GetVDMCurrentDirectories( 0, NULL ); if (cchCurDirs == 0) return;
pCurDirs = gmkstr(cchCurDirs*sizeof(TCHAR)); #ifdef UNICODE
pT = gmkstr(cchCurDirs); #endif
GetVDMCurrentDirectories( cchCurDirs, #ifdef UNICODE
pT #else
pCurDirs #endif
); #ifdef UNICODE
MultiByteToWideChar(CurrentCP, MB_PRECOMPOSED, pT, -1, pCurDirs, cchCurDirs); #endif
// set error mode so we don't get popup if trying to set curdir
// on empty floppy drive
PreviousErrorMode = SetErrorMode(SEM_FAILCRITICALERRORS); for (pCurCurDir=pCurDirs;*pCurCurDir!=NULLC;pCurCurDir+=(_tcslen(pCurCurDir)+1)) { ChangeDir2(pCurCurDir,CurDrive); CurDrive=FALSE; } SetErrorMode(PreviousErrorMode); //free(pCurDirs);
#ifdef UNICODE
FreeStr((PTCHAR)pT); #endif
} #endif // WIN95_CMD
/********************* START OF SPECIFICATION **************************/ /* */ /* SUBROUTINE NAME: ExecPgm */ /* */ /* DESCRIPTIVE NAME: Call DosExecPgm to execute External Command */ /* */ /* FUNCTION: Execute External Commands with DosExecPgm */ /* This routine calls DosExecPgm to execute the command. */ /* */ /* */ /* NOTES: This is a new routine added for OS/2 1.1 release. */ /* */ /* */ /* ENTRY POINT: ExecPgm */ /* LINKAGE: NEAR */ /* */ /* INPUT: n - the parse tree node containing the command to be executed*/ /* */ /* ai - the asynchronous indicator */ /* - 0 = Exec synchronous with parent */ /* - 1 = Exec asynchronous and discard child return code */ /* - 2 = Exec asynchronous and save child return code */ /* */ /* wf - the wait flag */ /* - 0 = Wait for process completion */ /* - 1 = Return immediately (Pipes) */ /* */ /* OUTPUT: None. */ /* */ /* EXIT-NORMAL: */ /* If synchronous execution, the return code of the command is */ /* returned. */ /* */ /* If asynchronous execution, the return code of the exec call */ /* is returned. */ /* */ /* EXIT-ERROR: */ /* Return FAILURE to the caller. */ /* */ /* EFFECTS: */ /* */ /* INTERNAL REFERENCES: */ /* ROUTINES: */ /* ExecError - Handles execution error */ /* PutStdErr - Print an error message */ /* WaitProc - wait for the termination of the specified process, */ /* its child process, and related pipelined */ /* processes. */ /* */ /* */ /* EXTERNAL REFERENCES: */ /* ROUTINES: */ /* DOSEXECPGM - Execute the specified program */ /* DOSSMSETTITLE - Set title for the presentation manager */ /* */ /********************** END OF SPECIFICATION **************************/
int ExecPgm( IN struct cmdnode *n, IN TCHAR *onb, IN unsigned int ai, IN unsigned int wf, IN TCHAR * argptr, IN TCHAR * fnptr, IN TCHAR * tptr ) { int i ; /* Work variable */ BOOL b; BOOL VdmProcess = FALSE; BOOL WowProcess = FALSE; LPTSTR CopyCmdValue; HANDLE hChildProcess;
HDESK hdesk; LPTSTR lpDesktop; DWORD cbDesktop = 0; HWINSTA hwinsta; DWORD cbWinsta = 0;
TCHAR szValEqExitCodeEnvVar [20]; TCHAR szValEqExitCodeAsciiEnvVar [12];
STARTUPINFO StartupInfo; PROCESS_INFORMATION ChildProcessInfo;
memset( &StartupInfo, 0, sizeof( StartupInfo ) ); StartupInfo.cb = sizeof( StartupInfo ); StartupInfo.lpTitle = tptr; StartupInfo.dwX = 0; StartupInfo.dwY = 1; StartupInfo.dwXSize = 100; StartupInfo.dwYSize = 100; StartupInfo.dwFlags = 0;//STARTF_SHELLOVERRIDE;
StartupInfo.wShowWindow = SW_SHOWNORMAL;
// Pass current Desktop to a new process.
hwinsta = GetProcessWindowStation(); GetUserObjectInformation( hwinsta, UOI_NAME, NULL, 0, &cbWinsta );
hdesk = GetThreadDesktop ( GetCurrentThreadId() ); GetUserObjectInformation (hdesk, UOI_NAME, NULL, 0, &cbDesktop);
if ((lpDesktop = HeapAlloc (GetProcessHeap(), HEAP_ZERO_MEMORY, cbDesktop + cbWinsta + 32) ) != NULL ) { LPTSTR p = lpDesktop; if ( GetUserObjectInformation (hwinsta, UOI_NAME, p, cbWinsta, &cbWinsta) ) { if (cbWinsta > 0) { p += ((cbWinsta/sizeof(TCHAR))-1); *p++ = L'\\'; } if ( GetUserObjectInformation (hdesk, UOI_NAME, p, cbDesktop, &cbDesktop) ) { StartupInfo.lpDesktop = lpDesktop; } } }
//
// Incredibly ugly hack for Win95 compatibility.
//
// XCOPY in Win95 reaches up into it's parent process to see if it was
// invoked in a batch file. If so, then XCOPY pretends COPYCMD=/Y
//
// There is no way we can do this for NT. The best that can be done is
// to detect if we're in a batch file starting XCOPY and then temporarily
// change COPYCMD=/Y.
//
{ const TCHAR *p = MyGetEnvVarPtr( TEXT( "COPYCMD" ));
if (p == NULL) { p = TEXT( "" ); }
CopyCmdValue = malloc( (mystrlen( p ) + 1) * sizeof( TCHAR )); if (CopyCmdValue == NULL) { PutStdErr( MSG_NO_MEMORY, NOARGS ); return FAILURE; } mystrcpy( CopyCmdValue, p );
if ((SingleBatchInvocation || SingleCommandInvocation || CurrentBatchFile ) && (p = mystrrchr( fnptr, TEXT( '\\' ))) != NULL && !lstrcmp( p, TEXT( "\\XCOPY.EXE" )) ) {
SetEnvVar( TEXT( "COPYCMD" ), TEXT( "/Y" ) );
} }
//
// If the restricted token exists then create the process with the
// restricted token.
// Else create the process without any restrictions.
//
if ((CurrentBatchFile != NULL) && (CurrentBatchFile->hRestrictedToken != NULL)) {
b = CreateProcessAsUser( CurrentBatchFile->hRestrictedToken, fnptr, argptr, (LPSECURITY_ATTRIBUTES) NULL, (LPSECURITY_ATTRIBUTES) NULL, TRUE, 0, NULL, CurDrvDir, &StartupInfo, &ChildProcessInfo );
} else { b = CreateProcess( fnptr, argptr, (LPSECURITY_ATTRIBUTES) NULL, (LPSECURITY_ATTRIBUTES) NULL, TRUE, 0, NULL, CurDrvDir, &StartupInfo, &ChildProcessInfo ); }
if (!b) { DosErr = i = GetLastError(); } else { hChildProcess = ChildProcessInfo.hProcess; CloseHandle(ChildProcessInfo.hThread); }
//
// Undo ugly hack
//
SetEnvVar( TEXT( "COPYCMD" ), CopyCmdValue ); free( CopyCmdValue );
HeapFree (GetProcessHeap(), 0, lpDesktop); if (!b) {
if (fEnableExtensions && DosErr == ERROR_BAD_EXE_FORMAT) { SHELLEXECUTEINFO sei;
memset(&sei, 0, sizeof(sei));
//
// Use the DDEWAIT flag so apps can finish their DDE conversation
// before ShellExecuteEx returns. Otherwise, apps like Word will
// complain when they try to exit, confusing the user.
//
sei.cbSize = sizeof(sei); sei.fMask = SEE_MASK_HASTITLE | SEE_MASK_NO_CONSOLE | SEE_MASK_FLAG_DDEWAIT | SEE_MASK_NOCLOSEPROCESS; sei.lpFile = fnptr; sei.lpParameters = n->argptr; sei.lpDirectory = CurDrvDir; sei.nShow = StartupInfo.wShowWindow;
try { b = ShellExecuteEx( &sei );
if (b) { hChildProcess = sei.hProcess; leave; }
if (sei.hInstApp == NULL) { DosErr = ERROR_NOT_ENOUGH_MEMORY; } else if ((DWORD_PTR)sei.hInstApp == HINSTANCE_ERROR) { DosErr = ERROR_FILE_NOT_FOUND; } else { DosErr = HandleToUlong(sei.hInstApp); }
} except( DosErr = GetExceptionCode( ), EXCEPTION_EXECUTE_HANDLER ) { b = FALSE; }
}
if (!b) { mystrcpy( onb, fnptr ); ExecError( onb ) ;
return (FAILURE) ; } }
#ifndef WIN95_CMD
VdmProcess = ((UINT_PTR)(hChildProcess) & 1) ? TRUE : FALSE; WowProcess = ((UINT_PTR)(hChildProcess) & 2) ? TRUE : FALSE; #endif // WIN95_CMD
if (hChildProcess == NULL || (fEnableExtensions && CurrentBatchFile == 0 && !SingleBatchInvocation && !SingleCommandInvocation && ai == AI_SYNC && (WowProcess || GetProcessSubsystemType(hChildProcess) == IMAGE_SUBSYSTEM_WINDOWS_GUI ) ) ) { //
// If extensions enabled and doing a synchronous exec of a GUI
// application, then change it into an asynchronous exec, with the
// return code discarded, ala Win 3.x and Win 95 COMMAND.COM
//
ai = AI_DSCD; }
i = SUCCESS; start_type = EXECPGM;
//
// Now that the process has been started, process the various
// termination conditions
//
if (ai == AI_SYNC) { //
// Synchronous exec: we wait for the child, establish status code
// environment variable, and update current directories if we're
// running a DOS app
//
LastRetCode = WaitProc( hChildProcess ); hChildProcess = NULL; i = LastRetCode ;
//
// Set up exit code environment variable both numeric and
// ascii displayable
//
_stprintf (szValEqExitCodeEnvVar, TEXT("%08X"), i); SetEnvVar(szNameEqExitCodeEnvVar, szValEqExitCodeEnvVar );
if ( (i >= 0x20) && (i <= 0x7e) ) { _stprintf (szValEqExitCodeAsciiEnvVar, TEXT("%01C"), i); SetEnvVar(szNameEqExitCodeAsciiEnvVar, szValEqExitCodeAsciiEnvVar ); } else SetEnvVar(szNameEqExitCodeAsciiEnvVar, TEXT("\0") );
//
// Update current directories if the process was handled by NTVDM. This
// is done since DOS .BAT scripts allow inheritance of current directory
// from child apps
//
#ifndef WIN95_CMD
if (VdmProcess) { RestoreCurrentDirectories(); } #endif // WIN95_CMD
} else if (ai == AI_DSCD) {
//
// Disconnected exec; we simply close the process handle and
// let it run to completion.
//
if (hChildProcess != NULL) { CloseHandle( hChildProcess ); hChildProcess = NULL; } } else if (ai == AI_KEEP) {
//
// Keep-the-handle-open exec: this is called only within the
// piping process. We've started the process, now we set a global
// so that the piping code can pick it up.
//
PipePid = hChildProcess; } else { #if !defined( WIN95_CMD ) && DBG
DbgBreakPoint( ); #endif
}
return (i) ; /* i == return from DOSEXECPGM */ }
/********************* START OF SPECIFICATION **************************/ /* */ /* SUBROUTINE NAME: SearchForExecutable */ /* */ /* DESCRIPTIVE NAME: Search for Executable File */ /* */ /* FUNCTION: This routine searches the specified executable file. */ /* If the file extension is specified, */ /* CMD.EXE searches for the file with the file extension */ /* to execute. If the specified file with the extension */ /* is not found, CMD.EXE will display an error message to */ /* indicate that the file is not found. */ /* If the file extension is not specified, */ /* CMD.EXE searches for the file with the order of these */ /* file extensions, .COM, .EXE, .CMD, and .BAT. */ /* The file which is found first will be executed. */ /* */ /* NOTES: 1) If a path is given, only the specified directory is */ /* searched. */ /* 2) If no path is given, the current directory of the */ /* drive specified is searched followed by the */ /* directories in the PATH environment variable. */ /* 3) If no executable file is found, an error message is */ /* printed. */ /* */ /* ENTRY POINT: SearchForExecutable */ /* LINKAGE: NEAR */ /* */ /* INPUT: */ /* n - parse tree node containing the command to be searched for */ /* loc - the string in which the location of the command is to be */ /* placed */ /* */ /* OUTPUT: None. */ /* */ /* EXIT-NORMAL: */ /* Returns: */ /* SFE_EXECOM, if a .EXE or .COM file is found. */ /* SFE_ISBAT, if a .CMD file is found. */ /* SFE_ISDIR, if a directory is found. */ /* SFE_NOTFND, if no executable file is found. */ /* */ /* EXIT-ERROR: */ /* Return FAILURE or */ /* SFE_FAIL, if out of memory. */ /* */ /* EFFECTS: None. */ /* */ /* INTERNAL REFERENCES: */ /* ROUTINES: */ /* DoFind - Find the specified file. */ /* GetEnvVar - Get full path. */ /* FullPath - build a full path name. */ /* TokStr - tokenize argument strings. */ /* mkstr - allocate space for a string. */ /* */ /* EXTERNAL REFERENCES: */ /* ROUTINES: */ /* None */ /* */ /********************** END OF SPECIFICATION **************************/
SearchForExecutable(n, loc) struct cmdnode *n ; TCHAR *loc ; { TCHAR *tokpath ; TCHAR *extPath ; TCHAR *extPathWrk ; TCHAR *fname; TCHAR *p1; TCHAR *tmps01; TCHAR *tmps02 = NULL; TCHAR pcstr[3]; LONG BinaryType;
size_t cName; // number of characters in file name.
int tplen; // Length of the current tokpath token
int dotloc; // loc offset where extension is appended
int pcloc; // M014 - Flag. True=user had partial path
int addpchar; // True - append PathChar to string @@5g
TCHAR *j ;
TCHAR wrkcmdline[MAX_PATH] ; unsigned tokpathlen; BOOL DoDot;
//
// Test for name too long first. If it is, we avoid wasting time
//
p1 = StripQuotes( n->cmdline ); if ((cName = mystrlen(p1)) >= MAX_PATH) { DosErr = MSG_LINES_TOO_LONG; return(SFE_NOTFND) ; }
//
// If cmd extensions enable, then treat CMD without an extension
// or path as a reference to COMSPEC variable. Guarantees we dont
// get a random copy of CMD.EXE
//
if (fEnableExtensions && (p1 == NULL || !_tcsnicmp( p1, TEXT( "cmd " ), 4))) { p1 = GetEnvVar( ComSpecStr ); if (p1 == NULL) { DosErr = MSG_INVALID_COMSPEC; return SFE_NOTFND; } }
mytcsnset(wrkcmdline, NULLC, MAX_PATH); mystrcpy(wrkcmdline, p1); FixPChar( wrkcmdline, SwitChar );
//
// Create the path character string, this will be search string
//
pcstr[0] = PathChar; pcstr[1] = COLON; pcstr[2] = NULLC;
//
// The variable pcloc is used as a flag to indicate whether the user
// did or didn't specify a drive or a partial path in his original
// input. It will be NZ if drive or path was specified.
//
pcloc = ((mystrcspn(wrkcmdline,pcstr)) < cName) ; pcstr[1] = NULLC ; // Fixup pathchar string
//
// handle the case of the user typing in a file name of
// ".", "..", or ending in "\"
// pcloc true say string has to have either a pathchar or colon
//
if ( pcloc ) { if (!(p1 = mystrrchr( wrkcmdline, PathChar ))) { p1 = mystrchr( wrkcmdline, COLON ); } p1++; // move to terminator if hanging ":" or "\"
} else { p1 = wrkcmdline; }
//
// p1 is guaranteed to be non-zero
//
if ( !(*p1) || !_tcscmp( p1, TEXT(".") ) || !_tcscmp( p1, TEXT("..") ) ) { //
// If CMD.EXE extensions enable, see if name matches
// subdirectory name.
//
if (fEnableExtensions) { DWORD dwFileAttributes;
dwFileAttributes = GetFileAttributes( loc ); if (dwFileAttributes != 0xFFFFFFFF && (dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) != 0 ) { return(SFE_ISDIR); }
if (dwFileAttributes == 0xFFFFFFFF && GetLastError() == ERROR_NOT_READY) { DosErr = ERROR_NOT_READY; return SFE_FAIL; } }
DosErr = MSG_DIR_BAD_COMMAND_OR_FILE; return(SFE_NOTFND) ; }
if (!(tmps01 = mkstr(2*sizeof(TCHAR)*MAX_PATH))) { DosErr = ERROR_NOT_ENOUGH_MEMORY; return(SFE_FAIL) ; }
//
// Handle the case of file..ext on a fat drive
//
mystrcpy( loc, wrkcmdline ); loc[ &p1[0] - &wrkcmdline[0] ] = 0; mystrcat( loc, TEXT(".") );
//
// Check for a malformed name
//
if (FullPath(tmps01, loc,MAX_PATH*2)) { //
// If CMD.EXE extensions enable, see if name matches
// subdirectory name.
//
if (fEnableExtensions) { DWORD dwFileAttributes;
dwFileAttributes = GetFileAttributes( loc ); if (dwFileAttributes != 0xFFFFFFFF && (dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) != 0 ) { return(SFE_ISDIR); } } DosErr = MSG_DIR_BAD_COMMAND_OR_FILE; return(SFE_NOTFND); }
if ( *lastc(tmps01) != PathChar ) { mystrcat( tmps01, TEXT("\\") ); }
//
// tmps01 contains full path + file name
//
mystrcat( tmps01, p1 );
tmps01 = resize(tmps01, (mystrlen(tmps01)+1)*sizeof(TCHAR)) ; if (tmps01 == NULL) { DosErr = ERROR_NOT_ENOUGH_MEMORY; return( SFE_FAIL ) ; }
//
// fname will point to last '\'
// tmps01 is to be path and fname is to be name
//
fname = mystrrchr(tmps01,PathChar) ; *fname++ = NULLC ;
DEBUG((DBENV, DBENVSCAN, "SearchForExecutable: Command:%ws",fname));
//
// If only fname type in get path string
//
if (!pcloc) { tmps02 = GetEnvVar(PathStr) ; }
DEBUG((DBENV, DBENVSCAN, "SearchForExecutable: PATH:%ws",tmps02));
DoDot = NeedCurrentDirectoryForExePath(loc);
//
// tmps02 is PATH environment variable
// compute enough for PATH environment plus file default path
//
tokpath = mkstr( ((DoDot ? 2 + mystrlen(tmps01) : 0) + mystrlen(tmps02) + 2)*sizeof(TCHAR)) ; if ( ! tokpath ) { DosErr = ERROR_NOT_ENOUGH_MEMORY; return( SFE_FAIL ) ; }
if (DoDot) { //
// Copy default path
//
mystrcat(tokpath,TEXT("\"")); mystrcat(tokpath,tmps01) ; mystrcat(tokpath,TEXT("\"")); } //
// If only name type in get also delim and path string
//
if (!pcloc) { if (DoDot) { mystrcat(tokpath, TEXT(";")) ; } mystrcat(tokpath,tmps02) ; }
//
// Shift left string at ';;'
//
tokshrink(tokpath); tokpath = TokStr(tokpath, TEXT(";"), TS_WSPACE) ; cName = mystrlen(fname) + 1 ; // file spec. length
//
// Build up the list of extensions we are going to search
// for. If extensions are enabled, get the list from the PATHEXT
// variable, otherwise use the hard coded default.
//
extPath = NULL; if (fEnableExtensions) extPath = GetEnvVar(PathExtStr);
if (extPath == NULL) extPath = PathExtDefaultStr;
tokshrink(extPath); extPath = TokStr(extPath, TEXT(";"), TS_WSPACE) ;
//
// Everything is now set up. Var tokpath contains a sequential series
// of asciz strings terminated by an extra null. If the user specified
// a drive or partial path, it contains only that one converted to full
// root-based form. If the user typed only a filename (pcloc = 0) it
// begins with the current directory and contains each directory that
// was contained in the PATH variable. This loop will search each of
// the tokpath elements once for each possible executable extention.
// Note that 'i' is used as a constant to test for excessive string
// length prior to performing the string copies.
//
for ( ; ; ) {
//
// Length of current path
//
tplen = mystrlen(tokpath) ; mystrcpy( tokpath, StripQuotes( tokpath ) ); tokpathlen = mystrlen(tokpath);
if (*lastc(tokpath) != PathChar) { addpchar = TRUE; tplen++; tokpathlen++; } else { addpchar = FALSE; } /* path + name too long */ //
// Check if path + name is too long
//
if (*tokpath && (tokpathlen + cName) > MAX_PATH) { tokpath += addpchar ? tplen : tplen+1; // get next path
continue; }
//
// If no more paths to search return descriptive error
//
if (*(tokpath) == NULLC) { if (pcloc) { if (DosErr == 0 || DosErr == ERROR_FILE_NOT_FOUND) DosErr = MSG_DIR_BAD_COMMAND_OR_FILE; } else { /* return generic message */ DosErr = MSG_DIR_BAD_COMMAND_OR_FILE; } return(SFE_NOTFND) ; }
//
// Install this path and setup for next one
//
mystrcpy(loc, tokpath) ; tokpath += addpchar ? tplen : tplen+1;
if (addpchar) { mystrcat(loc, pcstr) ; }
mystrcat(loc, fname) ; mystrcpy(loc, StripQuotes(loc) ); dotloc = mystrlen(loc) ;
DEBUG((DBENV, DBENVSCAN, "SearchForExecutable: PATH:%ws",loc));
//
// Check drive in each path to insure it is valid before searching
//
if (*(loc+1) == COLON) { if (!IsValidDrv(*loc)) continue ; };
//
// If fname has ext & ext > "." look for given filename
// this says that all executable files must have an extension
//
j = mystrrchr( fname, DOT ); if ( j && j[1] ) { //
// If access was denied and the user included a path,
// then say we found it. This handles the case where
// we don't have permission to do the findfirst and so we
// can't see the binary, but it actually exists -- if we
// have execute permission, CreateProcess will work
// just fine.
//
if (exists_ex(loc,TRUE) || (pcloc && (DosErr == ERROR_ACCESS_DENIED))) { //
// Recompute j as exists_ex trims trailing spaces
//
j = mystrrchr( loc, DOT ); if (j != NULL) { if ( !_tcsicmp(j,CmdExt) ) { return(SFE_ISBAT) ; } else if ( !_tcsicmp(j,BatExt) ) { return(SFE_ISBAT) ; } else { return(SFE_ISEXECOM) ; } } }
if ((DosErr != ERROR_FILE_NOT_FOUND) && DosErr) continue; // Try next path
} if (mystrchr( fname, STAR ) || mystrchr( fname, QMARK ) ) { DosErr = MSG_DIR_BAD_COMMAND_OR_FILE; return(SFE_NOTFND); }
//
// Search for each type of extension
//
extPathWrk = extPath; if (DoFind(loc, dotloc, TEXT(".*"), FALSE)) // Found anything?
while (*extPathWrk) { //
// if name + path + ext is less then max path length
//
if ( (cName + tokpathlen + mystrlen(extPathWrk)) <= MAX_PATH) { //
// See if this extension is a match.
//
if (DoFind(loc, dotloc, extPathWrk, TRUE)) { if (!_tcsicmp(extPathWrk, BatExt) || !_tcsicmp(extPathWrk, CmdExt)) return(SFE_ISBAT) ; // found .bat or .cmd
else return(SFE_ISEXECOM) ; // found executable
} else { //
// Any kind of error other than file not found, bail from
// search and try next element in path
//
if ((DosErr != ERROR_FILE_NOT_FOUND) && DosErr) break; } }
//
// Not this extension, try next.
while (*extPathWrk++) ; }
//
// If we get here, then no match with list of extensions.
// If no wierd errors, deal with NUll extension case.
//
if (DosErr == NO_ERROR || DosErr == ERROR_FILE_NOT_FOUND) { if (DoFind(loc, dotloc, TEXT("\0"), TRUE)) { if (GetBinaryType(loc,&BinaryType) && BinaryType == SCS_POSIX_BINARY) { // Found .
return(SFE_ISEXECOM) ; } } } } // end for
return(SFE_NOTFND); }
/*** DoFind - does indiviual findfirsts during searching
* * Purpose: * Add the extension to loc and do the find first for * SearchForExecutable(). * * DoFind(TCHAR *loc, int dotloc, TCHAR *ext) * * Args: * loc - the string in which the location of the command is to be placed * dotloc - the location loc at which the extension is to be appended * ext - the extension to append to loc * * Returns: * 1 if the file is found. * 0 if the file isn't found. * */
int DoFind(loc, dotloc, ext, metas) TCHAR *loc ; int dotloc ; TCHAR *ext ; BOOL metas; { *(loc+dotloc) = NULLC ; mystrcat(loc, ext) ;
DEBUG((DBENV, DBENVSCAN, "DoFind: exists_ex(%ws)",loc));
return(exists_ex(loc,metas)) ; /*@@4*/ }
/*** ExecError - handles exec errors
* * Purpose: * Print the exec error message corresponding to the error number in the * global variable DosErr. * @@ lots of error codes added. * ExecError() * */
void ExecError( onb ) TCHAR *onb; { unsigned int errmsg; unsigned int count;
count = ONEARG;
switch (DosErr) {
case ERROR_BAD_DEVICE: errmsg = MSG_DIR_BAD_COMMAND_OR_FILE; count = NOARGS; break;
case ERROR_LOCK_VIOLATION: errmsg = ERROR_SHARING_VIOLATION; break ;
case ERROR_NO_PROC_SLOTS: errmsg = ERROR_NO_PROC_SLOTS; count = NOARGS; break ;
case ERROR_NOT_DOS_DISK: errmsg = ERROR_NOT_DOS_DISK; break ;
case ERROR_NOT_ENOUGH_MEMORY: errmsg = ERROR_NOT_ENOUGH_MEMORY; count = NOARGS; break ;
case ERROR_PATH_NOT_FOUND: errmsg = MSG_CMD_FILE_NOT_FOUND; break ;
case ERROR_FILE_NOT_FOUND: errmsg = MSG_CMD_FILE_NOT_FOUND; break ;
case ERROR_ACCESS_DENIED: errmsg = ERROR_ACCESS_DENIED; break ;
case ERROR_EXE_MACHINE_TYPE_MISMATCH: errmsg = ERROR_EXE_MACHINE_TYPE_MISMATCH; break;
case ERROR_DRIVE_LOCKED: errmsg = ERROR_DRIVE_LOCKED; break ;
case ERROR_INVALID_STARTING_CODESEG: errmsg = ERROR_INVALID_STARTING_CODESEG; break ;
case ERROR_INVALID_STACKSEG: errmsg = ERROR_INVALID_STACKSEG; break ;
case ERROR_INVALID_MODULETYPE: errmsg = ERROR_INVALID_MODULETYPE; break ;
case ERROR_INVALID_EXE_SIGNATURE: errmsg = ERROR_INVALID_EXE_SIGNATURE; break ;
case ERROR_EXE_MARKED_INVALID: errmsg = ERROR_EXE_MARKED_INVALID; break ;
case ERROR_BAD_EXE_FORMAT: errmsg = ERROR_BAD_EXE_FORMAT; break ;
case ERROR_INVALID_MINALLOCSIZE: errmsg = ERROR_INVALID_MINALLOCSIZE; break ;
case ERROR_SHARING_VIOLATION: errmsg = ERROR_SHARING_VIOLATION; break ;
case ERROR_BAD_ENVIRONMENT: errmsg = ERROR_INFLOOP_IN_RELOC_CHAIN; count = NOARGS; break ;
case ERROR_INVALID_ORDINAL: errmsg = ERROR_INVALID_ORDINAL; break ;
case ERROR_CHILD_NOT_COMPLETE: errmsg = ERROR_CHILD_NOT_COMPLETE; break ;
case ERROR_DIRECTORY: errmsg = MSG_BAD_CURDIR; count = NOARGS; break;
case ERROR_NOT_ENOUGH_QUOTA: errmsg = ERROR_NOT_ENOUGH_QUOTA; count = NOARGS; break;
case MSG_REAL_MODE_ONLY: errmsg = MSG_REAL_MODE_ONLY; count = NOARGS; break ;
default: // printf( "Exec failed code %x\n", DosErr );
count = NOARGS; errmsg = MSG_EXEC_FAILURE ; /* M031 */
}
LastRetCode = errmsg; PutStdErr(errmsg, count, onb ); }
/*
* tokshrink @@4 * * remove duplicate ';' in a path */
void tokshrink( tokpath ) TCHAR *tokpath; { int i, j;
i = 0; do { if ( tokpath[i] == QUOTE ) { do { i++; } while ( tokpath[i] && tokpath[i] != QUOTE ); } if ( tokpath[i] && tokpath[i] != TEXT(';') ) { i++; } if ( tokpath[i] == TEXT(';') ) { j = i; while ( tokpath[j+1] == TEXT(';') ) { j++; } if ( j > i ) { mystrcpy( &tokpath[i], &tokpath[j] ); } i++; } } while ( tokpath[i] ); }
/*** eAssoc - execute an Assoc command
* * Purpose: * To set/modify the file associations stored in the registry under the * HKEY_LOCAL_MACHINE\Software\Classes key * * int eAssoc(struct cmdnode *n) * * Args: * n - the parse tree node containing the set command * * Returns: * If setting and the command is syntactically correct, whatever SetAssoc() * returns. Otherwise, FAILURE. * * If displaying, SUCCESS is always returned. * */
int eAssoc(n) struct cmdnode *n ; { if (glBatType != CMD_TYPE) { // if set command is executed from .bat file OR entered at command prompt
return( SetLastRetCodeIfError(AssocWork( n ))); } else { return( LastRetCode = AssocWork( n ) ); } }
int AssocWork(n) struct cmdnode *n ; { HKEY hKeyClasses; TCHAR *tas ; /* Tokenized argument string */ TCHAR *wptr ; /* Work pointer */ int i ; /* Work variable */ int rc ;
rc = RegOpenKey(HKEY_LOCAL_MACHINE, TEXT("Software\\Classes"), &hKeyClasses); if (rc) { return rc; }
tas = TokStr(n->argptr, ONEQSTR, TS_WSPACE|TS_SDTOKENS) ; if (!*tas) rc = DisplayAssoc(hKeyClasses, NULL) ;
else { for (wptr = tas, i = 0 ; *wptr ; wptr += mystrlen(wptr)+1, i++) ; /* If too many parameters were given, the second parameter */ /* wasn't an equal sign, or they didn't specify a string */ /* return an error message. */ if ( i > 3 || *(wptr = tas+mystrlen(tas)+1) != EQ || !mystrlen(mystrcpy(tas, StripQuotes(tas))) ) { if (i==1) { rc =DisplayAssoc(hKeyClasses, tas); } else { PutStdErr(MSG_BAD_SYNTAX, NOARGS); rc = FAILURE ; } } else { rc = SetAssoc(hKeyClasses, tas, wptr+2) ; } } ;
RegCloseKey(hKeyClasses) ;
return rc; }
/*** DisplayAssoc - display a specific file association or all
* * Purpose: * To display a specific file association or all * * int DisplayAssoc( hKeyClasses, tas ) * * Returns: * SUCCESS if all goes well * FAILURE if it runs out of memory or cannot lock the env. segment */
int DisplayAssoc(hKeyClasses, tas) HKEY hKeyClasses; TCHAR *tas; { int i; int rc = SUCCESS; TCHAR NameBuffer[MAX_PATH]; TCHAR ValueBuffer[MAX_PATH]; TCHAR *vstr ; DWORD cb;
if (tas == NULL) { for (i=0 ; rc == SUCCESS ; i++) { rc = RegEnumKey( hKeyClasses, i, NameBuffer, sizeof( NameBuffer ) / sizeof( TCHAR )); if (rc != SUCCESS) { if (rc==ERROR_NO_MORE_ITEMS) rc = SUCCESS; break; } else if (NameBuffer[0] == DOT) { cb = sizeof(ValueBuffer); rc = RegQueryValue(hKeyClasses, NameBuffer, ValueBuffer, &cb); if (rc != 0) { break; }
#if !DBG
if (_tcslen( ValueBuffer ) != 0) #endif
rc = cmd_printf(Fmt16, NameBuffer, ValueBuffer); }
if (CtrlCSeen) { return(FAILURE); } } } else { tas = EatWS(tas, NULL); if ((vstr = mystrrchr(tas, ' ')) != NULL) *vstr = NULLC;
cb = sizeof(ValueBuffer); rc = RegQueryValue(hKeyClasses, tas, ValueBuffer, &cb); if (rc == 0) rc = cmd_printf(Fmt16, tas, ValueBuffer); else PutStdErr(MSG_ASSOC_NOT_FOUND, ONEARG, tas); }
return(rc); }
/*** SetAssoc - controls adding/changing a file association
* * Purpose: * Add/replace a file association * * int SetAssoc(HKEY hKeyClasses, TCHAR *fileext, TCHAR *filetype) * * Args: * hKeyClasses - handle to HKEY_LOCAL_MACHINE\Software\Classes key * fileext - file extension string to associate * filetype - file type associate * * Returns: * SUCCESS if the association could be added/replaced. * FAILURE otherwise. * */
int SetAssoc(hKeyClasses, fileext, filetype) HKEY hKeyClasses; TCHAR *fileext ; TCHAR *filetype ; { int rc; int i; DWORD cb;
//
// Nothing was specified. We are to delete the key
//
if (filetype==NULL || *filetype==NULLC) { rc = RegDeleteKey(hKeyClasses, fileext); //
// The key may have subkeys. We only delete the default
// value, by opening the key and deleting the value
// with a zero-length name
//
if (rc != 0) { HKEY hKeyValue;
rc = RegOpenKey( hKeyClasses, fileext, &hKeyValue ); if (rc != 0) { //
// If there was no key to begin with, suppress the
// error
//
if (rc != ERROR_FILE_NOT_FOUND) { PutStdErr( rc, NOARGS ); } } else { rc = RegDeleteValue( hKeyValue, TEXT( "" ) ); if (rc != 0) { PutStdErr( rc, NOARGS ); } RegCloseKey( hKeyValue ); } } } else { //
// Set the value for the key
//
rc = RegSetValue(hKeyClasses, fileext, REG_SZ, filetype, _tcslen(filetype)); if (rc == 0) { cmd_printf( Fmt16, fileext, filetype ); } else PutStdErr(MSG_ERR_PROC_ARG, ONEARG, fileext); } //
// If the value was changed/deleted successfully, notify all
// apps about this
//
if (rc == 0) { try { SHChangeNotify( SHCNE_ASSOCCHANGED, SHCNF_IDLIST, NULL, NULL ); } except (DosErr = GetExceptionCode( ), EXCEPTION_EXECUTE_HANDLER) { } }
return rc; }
/*** eFType - execute an FType command
* * Purpose: * To set/modify the file types stored in the registry under the * HKEY_LOCAL_MACHINE\Software\Classes key * * int eFType(struct cmdnode *n) * * Args: * n - the parse tree node containing the set command * * Returns: * If setting and the command is syntactically correct, whatever SetFType() * returns. Otherwise, FAILURE. * * If displaying, SUCCESS is always returned. * */
int eFType(n) struct cmdnode *n ; { if (glBatType != CMD_TYPE) { // if set command is executed from .bat file OR entered at command prompt
return( SetLastRetCodeIfError(FTypeWork( n ))); } else { return( LastRetCode = FTypeWork( n ) ); } }
int FTypeWork(n) struct cmdnode *n ; { HKEY hKeyClasses; TCHAR *tas ; /* Tokenized argument string */ TCHAR *wptr ; /* Work pointer */ int i ; /* Work variable */ int rc ;
rc = RegOpenKey(HKEY_LOCAL_MACHINE, TEXT("Software\\Classes"), &hKeyClasses); if (rc) { return rc; }
tas = TokStr(n->argptr, ONEQSTR, TS_WSPACE|TS_SDTOKENS) ; if (!*tas) rc = DisplayFType(hKeyClasses, NULL) ;
else { for (wptr = tas, i = 0 ; *wptr ; wptr += mystrlen(wptr)+1, i++) ; /* If too many parameters were given, the second parameter */ /* wasn't an equal sign, or they didn't specify a string */ /* return an error message. */ if ( i > 3 || *(wptr = tas+mystrlen(tas)+1) != EQ || !mystrlen(mystrcpy(tas, StripQuotes(tas))) ) { if (i==1) { rc =DisplayFType(hKeyClasses, tas); } else { PutStdErr(MSG_BAD_SYNTAX, NOARGS); rc = FAILURE ; } } else { rc = SetFType(hKeyClasses, tas, wptr+2) ; } } ;
RegCloseKey(hKeyClasses) ;
return rc; }
/*** DisplayFType - display a specific file type or all
* * Purpose: * To display a specific file type or all * * int DisplayFType( hKeyClasses, tas ) * * Returns: * SUCCESS if all goes well * FAILURE if it runs out of memory or cannot lock the env. segment */
int DisplayFType(hKeyClasses, tas) HKEY hKeyClasses; TCHAR *tas; { int i; int rc; HKEY hKeyOpenCmd; TCHAR NameBuffer[MAX_PATH]; TCHAR ValueBuffer[MAX_PATH]; TCHAR *vstr ; DWORD cb, j, Type;
if (tas == NULL) { for (i = 0;;i++) { rc = RegEnumKey( hKeyClasses, i, NameBuffer, sizeof( NameBuffer ) / sizeof( TCHAR )); if (rc != 0) { if (rc==ERROR_NO_MORE_ITEMS) rc = SUCCESS; break; } else if (NameBuffer[0] != DOT) { j = _tcslen( NameBuffer ); if (j + _tcslen( ShellOpenCommandString ) + 1 <= MAX_PATH) { _tcscat(NameBuffer, ShellOpenCommandString ); _tcscpy(ValueBuffer,TEXT("*** no open command defined ***")); rc = RegOpenKey(hKeyClasses, NameBuffer, &hKeyOpenCmd); if (!rc) { NameBuffer[j] = TEXT('\0'); cb = sizeof(ValueBuffer); rc = RegQueryValueEx(hKeyOpenCmd, TEXT(""), NULL, &Type, (LPBYTE)ValueBuffer, &cb); RegCloseKey(hKeyOpenCmd); }
if (!rc) { cmd_printf(Fmt16, NameBuffer, ValueBuffer); } } } if (CtrlCSeen) { return(FAILURE); } } } else { if (*tas == DOT) { PutStdErr(MSG_FTYPE_NOT_FOUND, ONEARG, tas); return ERROR_INVALID_NAME; }
tas = EatWS(tas, NULL); if ((vstr = mystrrchr(tas, ' ')) != NULL) *vstr = NULLC;
if (_tcslen( tas ) + _tcslen( ShellOpenCommandString ) + 1 > MAX_PATH) { PutStdErr(MSG_FTYPE_TOO_LONG, ONEARG, tas); return ERROR_INVALID_NAME; } _tcscpy( NameBuffer, tas ); _tcscat( NameBuffer, ShellOpenCommandString ); rc = RegOpenKey(hKeyClasses, NameBuffer, &hKeyOpenCmd); if (rc) { PutStdErr(MSG_FTYPE_NOT_FOUND, ONEARG, tas); return rc; }
cb = sizeof(ValueBuffer); rc = RegQueryValueEx(hKeyOpenCmd, TEXT(""), NULL, &Type, (LPBYTE)ValueBuffer, &cb); if (rc == 0) { ValueBuffer[ (cb / sizeof( TCHAR )) - 1 ]; cmd_printf(Fmt16, tas, ValueBuffer); } else PutStdErr(MSG_FTYPE_NOT_FOUND, ONEARG, tas); RegCloseKey( hKeyOpenCmd ); }
return(rc); }
/*** SetFType - controls adding/changing the open command associated with a file type
* * Purpose: * Add/replace an open command string associated with a file type * * int SetFType(HKEY hKeyOpenCmd, TCHAR *filetype TCHAR *opencmd) * * Args: * hKeyClasses - handle to HKEY_LOCAL_MACHINE\Software\Classes * filetype - file type name * opencmd - open command string * * Returns: * SUCCESS if the file type could be added/replaced. * FAILURE otherwise. * */
int SetFType(hKeyClasses, filetype, opencmd) HKEY hKeyClasses; TCHAR *filetype ; TCHAR *opencmd ; { HKEY hKeyOpenCmd; TCHAR NameBuffer[MAX_PATH]; TCHAR c, *s; DWORD Disposition; int rc; int i; DWORD cb;
if (_tcslen( filetype ) + _tcslen( ShellOpenCommandString ) + 1 > MAX_PATH) { PutStdErr( MSG_FTYPE_TOO_LONG, NOARGS ); return FAILURE; } _tcscpy( NameBuffer, filetype ); _tcscat( NameBuffer, ShellOpenCommandString );
rc = RegOpenKey(hKeyClasses, NameBuffer, &hKeyOpenCmd); if (rc) { if (opencmd==NULL || *opencmd==NULLC) { PutStdErr(MSG_FTYPE_NOT_FOUND, ONEARG, filetype); return rc; }
s = NameBuffer; while (TRUE) { while (*s && *s != TEXT('\\')) { s += 1; } c = *s; *s = TEXT('\0'); rc = RegCreateKeyEx(hKeyClasses, NameBuffer, 0, NULL, 0, (REGSAM)MAXIMUM_ALLOWED, NULL, &hKeyOpenCmd, &Disposition ); if (rc) { PutStdErr(MSG_FTYPE_NOT_FOUND, ONEARG, filetype); return rc; }
if (c == TEXT('\0')) { break; }
*s++ = c; RegCloseKey(hKeyOpenCmd); } }
if (opencmd==NULL || *opencmd==NULLC) { rc = RegDeleteKey(hKeyOpenCmd, NULL); if (rc != 0) PutStdErr(MSG_FTYPE_NOT_FOUND, ONEARG, filetype); } else { rc = RegSetValueEx(hKeyOpenCmd, TEXT(""), 0, REG_EXPAND_SZ, (LPBYTE)opencmd, (_tcslen(opencmd)+1)*sizeof(TCHAR)); if (rc == 0) cmd_printf(Fmt16, filetype, opencmd); else PutStdErr(MSG_ERR_PROC_ARG, ONEARG, filetype); }
RegCloseKey(hKeyOpenCmd); return rc; }
typedef NTSTATUS (NTAPI *PNTQUERYINFORMATIONPROCESS)( IN HANDLE ProcessHandle, IN PROCESSINFOCLASS ProcessInformationClass, OUT PVOID ProcessInformation, IN ULONG ProcessInformationLength, OUT PULONG ReturnLength OPTIONAL );
HMODULE hNtDllModule; PNTQUERYINFORMATIONPROCESS lpNtQueryInformationProcess;
WORD GetProcessSubsystemType( HANDLE hProcess ) { PIMAGE_NT_HEADERS NtHeader; PPEB PebAddress; PEB Peb; SIZE_T SizeOfPeb; NTSTATUS Status; PROCESS_BASIC_INFORMATION ProcessInfo; BOOL b; PVOID ImageBaseAddress; LONG e_lfanew; WORD Subsystem;
Subsystem = IMAGE_SUBSYSTEM_UNKNOWN; if (hNtDllModule == NULL) { hNtDllModule = LoadLibrary( TEXT("NTDLL.DLL") ); if (hNtDllModule != NULL) { lpNtQueryInformationProcess = (PNTQUERYINFORMATIONPROCESS) GetProcAddress( hNtDllModule, "NtQueryInformationProcess" ); } else { hNtDllModule = INVALID_HANDLE_VALUE; } }
if (lpNtQueryInformationProcess != NULL) { //
// Get the Peb address
//
Status = (*lpNtQueryInformationProcess)( hProcess, ProcessBasicInformation, &ProcessInfo, sizeof( ProcessInfo ), NULL ); if (NT_SUCCESS( Status )) { PebAddress = ProcessInfo.PebBaseAddress;
//
// Read the subsystem type from the Peb
//
if (ReadProcessMemory( hProcess, PebAddress, &Peb, sizeof( Peb ), &SizeOfPeb ) ) { //
// See if we are running on a system that has the image subsystem
// type captured in the PEB. If so use it. Otherwise go the slow
// way and try to get it from the image header.
//
if (SizeOfPeb >= FIELD_OFFSET( PEB, ImageSubsystem ) && ((UINT_PTR)Peb.ProcessHeaps - (UINT_PTR)PebAddress) > FIELD_OFFSET( PEB, ImageSubsystem ) ) { Subsystem = (WORD)Peb.ImageSubsystem; } else { //
// read e_lfanew from imageheader
//
if (ReadProcessMemory( hProcess, &((PIMAGE_DOS_HEADER)Peb.ImageBaseAddress)->e_lfanew, &e_lfanew, sizeof( e_lfanew ), NULL ) ) { //
// Read subsystem version info
//
NtHeader = (PIMAGE_NT_HEADERS)((PUCHAR)Peb.ImageBaseAddress + e_lfanew); if (ReadProcessMemory( hProcess, &NtHeader->OptionalHeader.Subsystem, &Subsystem, sizeof( Subsystem ), NULL ) ) { } } } } } }
return Subsystem; }
|