|
|
/*++
Copyright (c) 1988-1999 Microsoft Corporation
Module Name:
cenv.c
Abstract:
Environment variable support
--*/
#include "cmd.h"
struct envdata { LPTSTR Strings; } ;
struct envdata CmdEnv ; // Holds info to manipulate Cmd's environment
struct envdata * OriginalEnvironment; // original environment setup used with eStart
extern TCHAR PathStr[], PromptStr[] ; extern TCHAR AppendStr[]; /* @@ */
extern CHAR InternalError[] ; extern TCHAR Fmt16[], Fmt17[], EnvErr[] ; extern TCHAR SetArithStr[] ; extern TCHAR SetPromptStr[] ; extern unsigned flgwd ; extern TCHAR CdStr[] ; extern TCHAR DatStr[] ; extern TCHAR TimStr[] ; extern TCHAR ErrStr[] ; extern TCHAR CmdExtVerStr[] ;
extern unsigned LastRetCode; extern BOOL CtrlCSeen; extern UINT CurrentCP; extern BOOLEAN PromptValid;
extern int glBatType; // to distinguish OS/2 vs DOS errorlevel behavior depending on a script file name
int SetArithWork(TCHAR *tas);
unsigned SetLastRetCodeIfError( unsigned RetCode ) { if (RetCode != 0) { LastRetCode = RetCode; }
return RetCode; }
/*** ePath - Begin the execution of a Path Command
* * Purpose: * If the command has no argument display the current value of the PATH * environment variable. Otherwise, change the value of the Path * environment variable to the argument. * * int ePath(struct cmdnode *n) * * Args: * n - the parse tree node containing the path command * * Returns: * If changing the PATH variable, whatever SetEnvVar() returns. * SUCCESS, otherwise. * */
int ePath(n) struct cmdnode *n ; { if (glBatType != CMD_TYPE) { // if set command is executed from .bat file OR entered at command prompt
return( SetLastRetCodeIfError(PathWork( n, 1 ))); } else { return( LastRetCode = PathWork( n, 1 ) ); }
}
/*** eAppend - Entry point for Append routine
* * Purpose: * to call Append and pass it a pointer to the command line * arguments * * Args: * a pointer to the command node structure * */
int eAppend(n) struct cmdnode *n ; {
if (glBatType != CMD_TYPE) { // if set command is executed from .bat file OR entered at command prompt
return( SetLastRetCodeIfError(PathWork( n, 0 ))); } else { return( LastRetCode = PathWork( n, 0 ) ); }
}
int PathWork(n, flag) struct cmdnode *n ; int flag; /* 0 = AppendStr, 1 = PathStr */ { TCHAR *tas ; /* Tokenized argument string */ TCHAR c ;
/* M014 - If the only argument is a single ";", then we have to set
* a NULL path. */ if ( n->argptr ) { c = *(EatWS(n->argptr, NULL)) ; } else { c = NULLC; }
if ((!c || c == NLN) && /* If args are all whitespace */ mystrchr(n->argptr, TEXT(';'))) {
return(SetEnvVar(flag ? PathStr : AppendStr, TEXT(""))) ;
} else {
tas = TokStr(n->argptr, TEXT(";"), TS_WSPACE | TS_NWSPACE) ;
if (*tas) { return(SetEnvVar(flag ? PathStr : AppendStr, tas)) ; }
cmd_printf(Fmt16, flag ? PathStr : AppendStr, GetEnvVar(flag ? PathStr : AppendStr)) ; } return(SUCCESS) ; }
/*** ePrompt - begin the execution of the Prompt command
* * Purpose: * To modifiy the Prompt environment variable. * * int ePrompt(struct cmdnode *n) * * Args: * n - the parse tree node containing the prompt command * * Returns: * Whatever SetEnvVar() returns. * */
int ePrompt(n) struct cmdnode *n ; { if (glBatType != CMD_TYPE) { // if set command is executed from .bat file OR entered at command prompt
return(SetLastRetCodeIfError(SetEnvVar(PromptStr, TokStr(n->argptr, NULL, TS_WSPACE)))) ; } else { return(LastRetCode = SetEnvVar(PromptStr, TokStr(n->argptr, NULL, TS_WSPACE)) ) ; } }
/*** eSet - execute a Set command
* * Purpose: * To set/modify an environment or to display the current environment * contents. * * int eSet(struct cmdnode *n) * * Args: * n - the parse tree node containing the set command * * Returns: * If setting and the command is syntactically correct, whatever SetEnvVar() * returns. Otherwise, FAILURE. * * If displaying, SUCCESS is always returned. * */
int eSet(n) struct cmdnode *n ; { if (glBatType != CMD_TYPE) { // if set command is executed from .bat file OR entered at command prompt
return( SetLastRetCodeIfError(SetWork( n ))); } else { return( LastRetCode = SetWork( n ) ); } }
/*** SetPromptUser - set environment variable to value entered by user.
* * Purpose: * Set environment variable to value entered by user. * * int SetPromptUser(TCHAR *tas) * * Args: * tas - pointer to null terminated string of the form: * * VARNAME=promptString * * Returns: * If valid expression, return SUCCESS otherwise FAILURE. * */
int SetPromptUser(TCHAR *tas) { TCHAR *wptr; TCHAR *tptr; ULONG dwOutputModeOld; ULONG dwOutputModeCur; ULONG dwInputModeOld; ULONG dwInputModeCur; BOOLEAN fOutputModeSet = FALSE; BOOLEAN fInputModeSet = FALSE; HANDLE hndStdOut = NULL; HANDLE hndStdIn = NULL; DWORD cch; TCHAR szValueBuffer[ 1024 ];
//
// Find first non-blank argument.
//
if (tas != NULL) while (*tas && *tas <= SPACE) tas += 1;
// If no input, declare an error
//
if (!tas || !*tas) { PutStdErr(MSG_BAD_SYNTAX, NOARGS); return(FAILURE) ; }
//
// See if first argument is quoted. If so, strip off
// leading quote, spaces and trailing quote.
//
if (*tas == QUOTE) { tas += 1; while (*tas && *tas <= SPACE) tas += 1; tptr = _tcsrchr(tas, QUOTE); if (tptr) *tptr = NULLC; }
//
// Find the equal sign in the argument.
//
wptr = _tcschr(tas, EQ);
//
// If no equal sign, error.
//
if (!wptr) { PutStdErr(MSG_BAD_SYNTAX, NOARGS); return(FAILURE) ; }
//
// Found the equal sign, so left of equal sign is variable name
// and right of equal sign is prompt string. Dont allow user to set
// a variable name that begins with an equal sign, since those
// are reserved for drive current directories.
//
*wptr++ = NULLC;
//
// See if second argument is quoted. If so, strip off
// leading quote, spaces and trailing quote.
//
if (*wptr == QUOTE) { wptr += 1; while (*wptr && *wptr <= SPACE) wptr += 1; tptr = _tcsrchr(wptr, QUOTE); if (tptr) *tptr = NULLC; }
if (*wptr == EQ) { PutStdErr(MSG_BAD_SYNTAX, NOARGS); return(FAILURE) ; }
hndStdOut = GetStdHandle(STD_OUTPUT_HANDLE); if (GetConsoleMode( hndStdOut, &dwOutputModeOld) ) {
// make sure CRLF is processed correctly
dwOutputModeCur = dwOutputModeOld | ENABLE_PROCESSED_OUTPUT; fOutputModeSet = TRUE; SetConsoleMode(hndStdOut,dwOutputModeCur); GetLastError(); }
hndStdIn = GetStdHandle(STD_INPUT_HANDLE); if (GetConsoleMode( hndStdIn, &dwInputModeOld) ) {
// make sure input is processed correctly
dwInputModeCur = dwInputModeOld | ENABLE_LINE_INPUT | ENABLE_ECHO_INPUT | ENABLE_PROCESSED_INPUT; fInputModeSet = TRUE; SetConsoleMode(hndStdIn,dwInputModeCur); GetLastError(); }
//
// Loop till the user enters a value for the variable.
//
while (TRUE) { PutStdOut(MSG_LITERAL_TEXT, ONEARG, wptr ); szValueBuffer[0] = NULLC; if (ReadBufFromInput( GetStdHandle(STD_INPUT_HANDLE), szValueBuffer, sizeof(szValueBuffer)/sizeof(TCHAR), &cch ) != 0 && cch != 0 ) { //
// Strip off any trailing CRLF
//
while (cch > 0 && szValueBuffer[cch-1] < SPACE) cch -= 1;
break; } else { cch = 0; break; }
if (!FileIsDevice(STDIN) || !(flgwd & 1)) cmd_printf(CrLf) ; }
if (fOutputModeSet) { SetConsoleMode( hndStdOut, dwOutputModeOld ); } if (fInputModeSet) { SetConsoleMode( hndStdIn, dwInputModeOld ); }
if (cch) { szValueBuffer[cch] = NULLC; return(SetEnvVar(tas, szValueBuffer)) ; } else { return(FAILURE); } }
int SetWork(n) struct cmdnode *n ; { TCHAR *tas ; /* Tokenized argument string */ TCHAR *wptr ; /* Work pointer */ int i ; /* Work variable */
//
// If extensions are enabled, things are different
//
if (fEnableExtensions) { tas = n->argptr; //
// Find first non-blank argument.
//
if (tas != NULL) while (*tas && *tas <= SPACE) tas += 1;
//
// No arguments, same as old behavior. Display current
// set of environment variables.
//
if (!tas || !*tas) return(DisplayEnv( GetCapturedEnvironmentStrings( &CmdEnv ))) ;
//
// See if /A switch given. If so, let arithmetic
// expression evaluator do the work.
//
if (!_tcsnicmp(tas, SetArithStr, 2)) return SetArithWork(tas+2);
//
// See if /P switch given. If so, prompt user for value
//
if (!_tcsnicmp(tas, SetPromptStr, 2)) return SetPromptUser(tas+2);
//
// See if first argument is quoted. If so, strip off
// leading quote, spaces and trailing quote.
//
if (*tas == QUOTE) { tas += 1; while (*tas && *tas <= SPACE) tas += 1; wptr = _tcsrchr(tas, QUOTE); if (wptr) *wptr = NULLC; }
//
// Dont allow user to set a variable name that begins with
// an equal sign, since those are reserved for drive current
// directories. This check will also detect missing variable
// name, e.g.
//
// set %LOG%=c:\tmp\log.txt
//
// if LOG is not defined is an invalid statement.
//
if (*tas == EQ) { PutStdErr(MSG_BAD_SYNTAX, NOARGS); return(FAILURE) ; }
//
// Find the equal sign in the argument.
//
wptr = _tcschr(tas, EQ);
//
// If no equal sign, then assume argument is variable name
// and user wants to see its value. Display it.
//
if (!wptr) return DisplayEnvVariable(tas);
//
// Found the equal sign, so left of equal sign is variable name
// and right of equal sign is value.
//
*wptr++ = NULLC; return(SetEnvVar(tas, wptr)) ; }
tas = TokStr(n->argptr, ONEQSTR, TS_WSPACE|TS_SDTOKENS) ; if (!*tas) return(DisplayEnv( GetCapturedEnvironmentStrings( &CmdEnv ))) ;
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))) ) { /* M013 */ PutStdErr(MSG_BAD_SYNTAX, NOARGS); return(FAILURE) ;
} else { return(SetEnvVar(tas, wptr+2)) ; } } }
/*** DisplayEnvVariable - display a specific variable from the environment
* * Purpose: * To display a specific variable from the current environment. * * int DisplayEnvVariable( tas ) * * Returns: * SUCCESS if all goes well * FAILURE if it runs out of memory or cannot lock the env. segment */
int DisplayEnvVariable(tas) TCHAR *tas; { TCHAR *envptr ; TCHAR *vstr ; unsigned size ; UINT PrefixLength; int rc;
//
// Get environment. If there's none, we're done.
//
envptr = GetCapturedEnvironmentStrings( &CmdEnv ); if (envptr == (TCHAR *)NULL) { fprintf ( stderr, InternalError , "Null environment" ) ; return( FAILURE ) ; }
//
// Isolate the prefix to match against.
//
tas = EatWS(tas, NULL); if ((vstr = mystrrchr(tas, SPACE)) != NULL) { *vstr = NULLC; }
PrefixLength = mystrlen(tas);
//
// Walk through the environment looking for prefixes that match.
//
rc = FAILURE; while ((size = mystrlen(envptr)) > 0) {
//
// Stop soon if we see ^C
//
if (CtrlCSeen) { break; }
//
// If the prefix is long enough, then terminate the string and
// look for a prefix match. If we match, restore the string
// and display it
//
if (size >= PrefixLength) { TCHAR SavedChar = envptr[PrefixLength]; envptr[PrefixLength] = NULLC; if (!lstrcmpi( envptr, tas )) { envptr[PrefixLength] = SavedChar; cmd_printf(Fmt17, envptr ); rc = SUCCESS; } else { envptr[PrefixLength] = SavedChar; }
}
//
// Advance to the next string
//
envptr += size+1 ; }
if (rc != SUCCESS) { PutStdErr(MSG_ENV_VAR_NOT_FOUND, ONEARG, tas); }
return(rc) ; }
/*** MyGetEnvVar - get a pointer to the value of an environment variable
* * Purpose: * Return a pointer to the value of the specified environment variable. * * If the variable is not found, return NULL. * * TCHAR *MyGetEnvVar(TCHAR *varname) * * Args: * varname - the name of the variable to search for * * Returns: * See above. * * Side Effects: * Returned value points to within the environment block itself, so is * not valid after a set environment variable operations is perform. */
const TCHAR * MyGetEnvVarPtr(TCHAR *varname) { TCHAR *envptr ; /* Ptr to environment */ TCHAR *vstr ; unsigned size ; /* Length of current env string */ unsigned n ;
if (varname == NULL) { return( NULL ) ; }
envptr = GetCapturedEnvironmentStrings( &CmdEnv ); if (envptr == (TCHAR *)NULL) { return( NULL ) ; }
varname = EatWS(varname, NULL); if ((vstr = mystrrchr(varname, SPACE)) != NULL) *vstr = NULLC;
n = mystrlen(varname); while ((size = mystrlen(envptr)) > 0) { /* M015 */ if (CtrlCSeen) { break; } if (!_tcsnicmp(varname, envptr, n) && envptr[n] == TEXT( '=' )) {
return envptr+n+1; }
envptr += size+1 ; }
return(NULL); }
/*** DisplayEnv - display the environment
* * Purpose: * To display the current contents of the environment. * * int DisplayEnv() * * Returns: * SUCCESS if all goes well * FAILURE if it runs out of memory or cannot lock the env. segment */
int DisplayEnv(TCHAR *envptr) { unsigned size ; /* Length of current env string */
if (envptr == (TCHAR *)NULL) { fprintf ( stderr, InternalError , "Null environment" ) ; return( FAILURE ) ; }
while ((size = mystrlen(envptr)) > 0) { /* M015 */ if (CtrlCSeen) { return(FAILURE); } #if !DBG
// Dont show current directory variables in retail product
if (*envptr != EQ) #endif // DBG
cmd_printf(Fmt17, envptr) ; /* M005 */ envptr += size+1 ; }
return(SUCCESS) ; }
/*** GetEnvVar - get the value of an environment variable
* * Purpose: * Return a string containing the value of the specified environment * variable. The string value has been placed into a static buffer * that is valid until the next GetEnvVar call. * * If the variable is not found, return NULL. * * TCHAR *GetEnvVar(TCHAR *varname) * * Args: * varname - the name of the variable to search for * * Returns: * See above. * */
TCHAR GetEnvVarBuffer[LBUFLEN];
TCHAR * GetEnvVar(varname) PTCHAR varname ; { GetEnvVarBuffer[0] = TEXT( '\0' );
if (GetEnvironmentVariable(varname, GetEnvVarBuffer, sizeof(GetEnvVarBuffer) / sizeof(TCHAR))) { return(GetEnvVarBuffer); } else if (fEnableExtensions) { if (!_tcsicmp(varname, CdStr)) { GetDir(GetEnvVarBuffer, GD_DEFAULT) ; return GetEnvVarBuffer; } else if (!_tcsicmp(varname, ErrStr)) { _stprintf( GetEnvVarBuffer, TEXT("%d"), LastRetCode ); return GetEnvVarBuffer; } else if (!_tcsicmp(varname, CmdExtVerStr)) { _stprintf( GetEnvVarBuffer, TEXT("%d"), CMDEXTVERSION ); return GetEnvVarBuffer; } else if (!_tcsicmp(varname, TEXT("CMDCMDLINE"))) { return GetCommandLine(); } else if (!_tcsicmp(varname, DatStr)) { GetEnvVarBuffer[ PrintDate(NULL, PD_DATE, GetEnvVarBuffer, LBUFLEN) ] = NULLC; return GetEnvVarBuffer; } if ( !_tcsicmp(varname, TimStr)) { GetEnvVarBuffer[ PrintTime(NULL, PT_TIME, GetEnvVarBuffer, LBUFLEN) ] = NULLC; return GetEnvVarBuffer; } if ( !_tcsicmp(varname, TEXT("RANDOM"))) { _stprintf( GetEnvVarBuffer, TEXT("%d"), rand() ); return GetEnvVarBuffer; } } return(NULL); }
/*** CaptureEnvironmentStrings - make writeable copy of environment
* * Purpose: * Allocate memory and create copy of environment strings * * Args: * None * * Returns: * Allocated copy of environment strings or NULL */ LPTSTR CaptureEnvironmentStrings( VOID ) { LPTSTR EnvStrings = GetEnvironmentStrings( ); LPTSTR Copy = NULL; if (EnvStrings != NULL) { ULONG Size = GetEnvCb( EnvStrings ); //
// Allocate and copy strings
//
Copy = HeapAlloc( GetProcessHeap( ), HEAP_ZERO_MEMORY, Size ); if (Copy != NULL) { memcpy( Copy, EnvStrings, Size ); }
FreeEnvironmentStrings( EnvStrings ); }
return Copy; }
/*** InitEnv - Set up CMD's copy of the environment
* * Purpose: * Creates the copy of CMD's environment. * * Args: * None * * Returns: * None */ void InitEnv( void ) { CmdEnv.Strings = CaptureEnvironmentStrings( ); OriginalEnvironment = CopyEnv(); }
LPWSTR GetCapturedEnvironmentStrings( struct envdata *Environment ) { return Environment->Strings; }
/*** SetEnvVar - controls adding/changing an environment variable
* * Purpose: * Add/replace an environment variable. Grow it if necessary. * * int SetEnvVar(TCHAR *varname, TCHAR *varvalue, struct envdata *env) * * Args: * varname - name of the variable being added/replaced * varvalue - value of the variable being added/replaced * env - environment info structure being used * * Returns: * SUCCESS if the variable could be added/replaced. * FAILURE otherwise. * */
int SetEnvVar(varname, varvalue) TCHAR *varname ; TCHAR *varvalue ; { int retvalue; MEMORY_BASIC_INFORMATION MemoryInfo;
PromptValid = FALSE; // Force it to be recalculated
if (!_tcslen(varvalue)) { varvalue = NULL; // null to remove from env
} retvalue = SetEnvironmentVariable(varname, varvalue); HeapFree( GetProcessHeap( ), 0, CmdEnv.Strings ); CmdEnv.Strings = CaptureEnvironmentStrings();
return !retvalue; }
/*** MoveEnv - Move the contents of the environment (M008 - New function)
* * Purpose: * Used by CopyEnv, this function moves the existing * environment contents to the new location. * * MoveEnv(unsigned thndl, unsigned shndl, unsigned cnt) * * Args: * thndl - Handle of target environment * shndl - Handle of source environment * cnt - byte count to move * * Returns: * TRUE if no errors * FALSE otherwise * */
BOOL MoveEnv(tenvptr, senvptr, cnt) TCHAR *senvptr ; /* Ptr into source env seg */ TCHAR *tenvptr ; /* Ptr into target env seg */ ULONG cnt ; { if ((tenvptr == NULL) || (senvptr == NULL)) { fprintf(stderr, InternalError, "Null environment") ; return(FALSE) ; } memcpy(tenvptr, senvptr, cnt) ; /* M015 */ return(TRUE) ; }
/*** FreeEnv - free an environment created by CopyEnv
* * Purpose: * Free all memory associated with a copied environment * * Returns: * nothing * */
void FreeEnv( struct envdata *Environment ) { HeapFree( GetProcessHeap( ), 0, Environment->Strings ); HeapFree( GetProcessHeap( ), 0, Environment ); }
/*** CopyEnv - make a copy of the current environment
* * Purpose: * Make a copy of CmdEnv and put the new handle into the newly * created envdata structure. This routine is only called by * eSetlocal and init. * * struct envdata *CopyEnv() * * Returns: * A pointer to the environment information structure. * Returns NULL if unable to allocate enough memory * * Notes: * - M001 - This function was disabled, now reenabled. * - The current environment is copied as a snapshot of how it looked * before SETLOCAL was executed. * */
struct envdata *CopyEnv() { struct envdata *cce ; /* New env info structure */
cce = (struct envdata *) HeapAlloc( GetProcessHeap( ), HEAP_ZERO_MEMORY, sizeof( *cce )); if (cce == NULL) { return NULL; }
cce->Strings = CaptureEnvironmentStrings( ); if (cce->Strings == NULL) { HeapFree( GetProcessHeap( ), 0, cce ); PutStdErr( MSG_OUT_OF_ENVIRON_SPACE, NOARGS ); return NULL; }
return cce; }
/*** ResetEnv - restore the environment
* * Purpose: * Restore the environment to the way it was before the execution of * the SETLOCAL command. This function only called by eEndlocal. * * ResetEnv(struct envdata *env) * * Args: * env - structure containing handle, size and max dimensions of an * environment. * * Notes: * - M001 - This function was disabled, but has been reenabled. * - M001 - This function used to test for OLD/NEW style batch files * and delete the copy or the original environment as * appropriate. It now always deletes the original. * - M014 - Note that the modified local environment will never be * shrunk, so we can assume it will hold the old one. * */
void ResetEnv( struct envdata *env) { SetEnvironmentStrings( env->Strings ); HeapFree( GetProcessHeap( ), 0, CmdEnv.Strings ); CmdEnv.Strings = CaptureEnvironmentStrings();
#if 0
PTCHAR EnvString; PTCHAR Name; PTCHAR Value; //
// Delete everything in current environment
//
EnvString = GetCapturedEnvironmentStrings( &CmdEnv ); Name = EnvString; while (*Name != TEXT( '\0' )) { //
// Find equal sign
//
Value = Name + 1; while (*Value != TEXT( '\0' ) && *Value != TEXT( '=' )) { Value++; }
if (*Value == TEXT( '\0' )) { //
// SetEnvironmentVariable will succeed in deleting this
//
SetEnvironmentVariable( Name, NULL ); } else { *Value = TEXT( '\0' ); SetEnvironmentVariable( Name, NULL ); *Value = TEXT( '=' ); }
Name += _tcslen( Name ) + 1;
} //
// Add everything in env back into the environment
//
Name = env->Strings; while (*Name != TEXT( '\0' )) { //
// Find equal sign
//
Value = Name + 1; while (*Value != TEXT( '\0' ) && *Value != TEXT( '=' )) { Value++; }
if (*Value == TEXT( '\0' )) { //
// I have no clue how to add this in
//
SetEnvironmentVariable( Name, NULL ); } else { *Value = TEXT( '\0' ); SetEnvironmentVariable( Name, Value + 1 ); *Value = TEXT( '=' ); }
Name += _tcslen( Name ) + 1;
} HeapFree( GetProcessHeap( ), 0, CmdEnv.Strings ); CmdEnv.Strings = CaptureEnvironmentStrings(); #endif
}
ULONG GetEnvCb( TCHAR *penv ) { TCHAR *Scan = penv;
if (penv == NULL) { return(0); }
//
// NUL string terminates environment
//
while (*Scan) { //
// Skip over string and NUL
//
while (*Scan++) { } } Scan++;
return (Scan - penv) * sizeof( TCHAR ); }
//
// expr -> assign [, assign]* ,
//
// assign -> orlogexpr |
// VAR ASSIGNOP assign <op>=
//
// orlogexpr -> xorlogexpr [| xorlogexpr]* |
//
// xorlogexpr -> andlogexpr [^ andlogexpr]* ^
//
// andlogexpr -> shiftexpr [& shiftexpr]* &
//
// shiftexpr -> addexpr [SHIFTOP addexpr]* <<, >>
//
// addexpr -> multexpr [ADDOP multexpr]* +, -
//
// multexpr -> unaryexpr [MULOP unaryexpr]* *, /, %
//
// unaryexpr -> ( expr ) | ()
// UNARYOP unaryexpr +, -, !, ~
//
TCHAR szOps[] = TEXT("<>+-*/%()|^&=,"); TCHAR szUnaryOps[] = TEXT("+-~!");
typedef struct { PTCHAR Token; LONG Value; DWORD Error; } PARSESTATE, *PPARSESTATE;
VOID APerformUnaryOperation( PPARSESTATE State, TCHAR Op, LONG Value ) { switch (Op) { case TEXT( '+' ): State->Value = Value; break; case TEXT( '-' ): State->Value = -Value; break; case TEXT( '~' ): State->Value = ~Value; break; case TEXT( '!' ): State->Value = !Value; break; default: printf( "APerformUnaryOperation: '%c'\n", Op); break; } }
VOID APerformArithmeticOperation( PPARSESTATE State, TCHAR Op, LONG Left, LONG Right ) { switch (Op) { case TEXT( '<' ): State->Value = (Right >= 8 * sizeof( Left << Right)) ? 0 : (Left << Right); break; case TEXT( '>' ): State->Value = (Right >= 8 * sizeof( Left >> Right )) ? (Left < 0 ? -1 : 0) : (Left >> Right); break; case TEXT( '+' ): State->Value = Left + Right; break; case TEXT( '-' ): State->Value = Left - Right; break; case TEXT( '*' ): State->Value = Left * Right; break; case TEXT( '/' ): if (Right == 0) { State->Error = MSG_SET_A_DIVIDE_BY_ZERO; } else { State->Value = Left / Right; } break; case TEXT( '%' ): if (Right == 0) { State->Error = MSG_SET_A_DIVIDE_BY_ZERO; } else { State->Value = Left % Right; } break; case TEXT( '|' ): State->Value = Left | Right; break; case TEXT( '^' ): State->Value = Left ^ Right; break; case TEXT( '&' ): State->Value = Left & Right; break; case TEXT( '=' ): State->Value = Right; break; default: printf( "APerformArithmeticOperation: '%c'\n", Op); } }
//
// Return the numeric value of an environment variable (or 0)
//
LONG AGetValue( PTCHAR Start, PTCHAR End ) { TCHAR c = *End; const TCHAR *Value; PTCHAR Dummy;
*End = NULLC;
Value = MyGetEnvVarPtr( Start ); *End = c;
if (Value == NULL) { return 0; }
return _tcstol( Value, &Dummy, 0); }
DWORD ASetValue( PTCHAR Start, PTCHAR End, LONG Value ) { TCHAR Result[32]; TCHAR c = *End; DWORD Return = SUCCESS;
*End = NULLC;
_sntprintf( Result, 32, TEXT("%d"), Value ) ; Result[31] = TEXT( '\0' );
if (SetEnvVar( Start, Result) != SUCCESS) { Return = GetLastError(); }
*End = c; return Return; }
//
// Forward decls
//
PARSESTATE AParseAddExpr( PARSESTATE State ); PARSESTATE AParseAndLogExpr( PARSESTATE State ); PARSESTATE AParseAssign( PARSESTATE State ); PARSESTATE AParseExpr( PARSESTATE State ); PARSESTATE AParseMultExpr( PARSESTATE State ); PARSESTATE AParseOrLogExpr( PARSESTATE State ); PARSESTATE AParseShiftExpr( PARSESTATE State ); PARSESTATE AParseUnaryExpr( PARSESTATE State ); PARSESTATE AParseXorLogExpr( PARSESTATE State );
//
// Skip whitespace and return next character
//
BOOL ASkipWhiteSpace( PPARSESTATE State ) { while (*State->Token != NULLC && *State->Token <= SPACE) { State->Token++; }
return *State->Token != NULLC; }
TCHAR ANextChar( PPARSESTATE State ) { ASkipWhiteSpace( State ); return *State->Token; }
BOOL AParseVariable( PPARSESTATE State, PTCHAR *FirstChar, PTCHAR *EndOfName ) { TCHAR c = ANextChar( State );
//
// Next char is a digit or operator, can't be a variable
//
if (c == NULLC || _istdigit( c ) || _tcschr( szOps, c ) != NULL || _tcschr( szUnaryOps, c ) != NULL) {
return FALSE;
}
*FirstChar = State->Token;
//
// find end of variable
//
while (*State->Token && *State->Token > SPACE && !_tcschr( szUnaryOps, *State->Token ) && !_tcschr( szOps, *State->Token ) ) { State->Token += 1; }
*EndOfName = State->Token; return TRUE; }
// expr -> assign [, assign]*
PARSESTATE AParseExpr( PARSESTATE State ) { State = AParseAssign( State );
while (State.Error == SUCCESS) {
if (ANextChar( &State ) != TEXT( ',' )) { break; } State.Token++;
State = AParseAssign( State );
}
return State; }
// assign -> VAR ASSIGNOP assign |
// orlogexpr
PARSESTATE AParseAssign( PARSESTATE State ) { TCHAR c = ANextChar( &State ); PARSESTATE SavedState;
SavedState = State;
if (c == NULLC) { State.Error = MSG_SET_A_MISSING_OPERAND; return State; }
//
// See if we have VAR ASSIGNOP
//
do { PTCHAR FirstChar; PTCHAR EndOfName; TCHAR OpChar; LONG OldValue;
//
// Parse off variable
//
if (!AParseVariable( &State, &FirstChar, &EndOfName )) { break; }
//
// Look for <op>=
//
OpChar = ANextChar( &State );
if (OpChar == NULLC) { break; }
if (OpChar != TEXT( '=' )) { if (_tcschr( szOps, OpChar ) == NULL) { break; } State.Token++;
if (OpChar == TEXT( '<' ) || OpChar == TEXT( '>')) { if (ANextChar( &State ) != OpChar) { break; } State.Token++; }
}
if (ANextChar( &State ) != TEXT( '=' )) { break; } State.Token++;
//
// OpChar is the sort of operation to apply before assignment
// State has been advance to, hopefully, another assign. Parse it
// and see where we get
//
State = AParseAssign( State ); if (State.Error != SUCCESS) { return State; }
OldValue = AGetValue( FirstChar, EndOfName );
//
// Perform the operation and the assignment
//
APerformArithmeticOperation( &State, OpChar, OldValue, State.Value ); if (State.Error != SUCCESS) { return State; }
State.Error = ASetValue( FirstChar, EndOfName, State.Value );
return State; } while ( FALSE );
//
// Must be orlogexpr. Go back and parse over
//
return AParseOrLogExpr( SavedState ); }
// orlogexpr -> xorlogexpr [| xorlogexpr]* |
PARSESTATE AParseOrLogExpr( PARSESTATE State ) { State = AParseXorLogExpr( State ); while (State.Error == SUCCESS) { TCHAR Op = ANextChar( &State ); LONG Value = State.Value;
if (Op != TEXT( '|' )) { break; } State.Token++;
State = AParseXorLogExpr( State ); APerformArithmeticOperation( &State, Op, Value, State.Value ); } return State; }
// xorlogexpr -> andlogexpr [^ andlogexpr]* ^
PARSESTATE AParseXorLogExpr( PARSESTATE State ) { State = AParseAndLogExpr( State ); while (State.Error == SUCCESS) { TCHAR Op = ANextChar( &State ); LONG Value = State.Value;
if (Op != TEXT( '^' )) { break; } State.Token++;
State = AParseAndLogExpr( State ); APerformArithmeticOperation( &State, Op, Value, State.Value ); } return State; }
// andlogexpr -> shiftexpr [& shiftexpr]* &
PARSESTATE AParseAndLogExpr( PARSESTATE State ) { State = AParseShiftExpr( State ); while (State.Error == SUCCESS) { TCHAR Op = ANextChar( &State ); LONG Value = State.Value;
if (Op != TEXT( '&' )) { break; } State.Token++;
State = AParseShiftExpr( State ); APerformArithmeticOperation( &State, Op, Value, State.Value ); } return State; }
// shiftexpr -> addexpr [SHIFTOP addexpr]* <<, >>
PARSESTATE AParseShiftExpr( PARSESTATE State ) { State = AParseAddExpr( State ); while (State.Error == SUCCESS) { TCHAR Op = ANextChar( &State ); LONG Value = State.Value;
if (Op != TEXT( '<' ) && Op != TEXT( '>' )) { break; } State.Token++;
if (Op != ANextChar( &State )) { State.Error = MSG_SET_A_MISSING_OPERATOR; return State; } State.Token++;
State = AParseAddExpr( State ); APerformArithmeticOperation( &State, Op, Value, State.Value ); } return State; }
// addexpr -> multexpr [ADDOP multexpr]* +, -
PARSESTATE AParseAddExpr( PARSESTATE State ) { State = AParseMultExpr( State ); while (State.Error == SUCCESS) { TCHAR Op = ANextChar( &State ); LONG Value = State.Value;
if (Op != TEXT( '+' ) && Op != TEXT( '-' )) { break; } State.Token++;
State = AParseMultExpr( State ); APerformArithmeticOperation( &State, Op, Value, State.Value ); } return State; }
// multexpr -> unaryexpr [MULOP unaryexpr]* *, /, %
PARSESTATE AParseMultExpr( PARSESTATE State ) { State = AParseUnaryExpr( State ); while (State.Error == SUCCESS) { TCHAR Op = ANextChar( &State ); LONG Value = State.Value;
if (Op != TEXT( '*' ) && Op != TEXT( '/' ) && Op != TEXT( '%' )) { break; } State.Token++;
State = AParseUnaryExpr( State ); APerformArithmeticOperation( &State, Op, Value, State.Value ); } return State; }
// unaryexpr -> UNARYOP unaryexpr +, -, !, ~
// ( expr ) | ()
// NUMBER
// LITERAL
PARSESTATE AParseUnaryExpr( PARSESTATE State ) { TCHAR c = ANextChar( &State ); PTCHAR FirstChar; PTCHAR EndOfName;
if (c == NULLC) { State.Error = MSG_SET_A_MISSING_OPERAND; return State; }
// ( expr )
if (c == TEXT( '(' )) { State.Token++; State = AParseExpr( State ); if (State.Error != SUCCESS) { return State; } c = ANextChar( &State ); if (c != TEXT( ')' )) { State.Error = MSG_SET_A_MISMATCHED_PARENS; } else { State.Token++; } return State; }
// UNARYOP unaryexpr
if (_tcschr( szUnaryOps, c ) != NULL) { State.Token++; State = AParseUnaryExpr( State ); if (State.Error != SUCCESS) { return State; } APerformUnaryOperation( &State, c, State.Value ); return State; }
// NUMBER
if (_istdigit(c)) { errno = 0; State.Value = _tcstol( State.Token, &State.Token, 0 ); if (State.Value == LONG_MAX && errno == ERANGE) { State.Error = MSG_SET_NUMBER_TOO_LARGE; } else if (_istdigit( *State.Token ) || _istalpha( *State.Token )) { State.Error = MSG_SET_A_INVALID_NUMBER; } return State; }
// Must be literal
if (!AParseVariable( &State, &FirstChar, &EndOfName )) { State.Error = MSG_SET_A_MISSING_OPERAND; return State; }
State.Value = AGetValue( FirstChar, EndOfName ); return State; }
/*** SetArithWork - set environment variable to value of arithmetic expression
* * Purpose: * Set environment variable to value of arithmetic expression * * int SetArithWork(TCHAR *tas) * * Args: * tas - pointer to null terminated string of the form: * * VARNAME=expression * * Returns: * If valid expression, return SUCCESS otherwise FAILURE. * */
int SetArithWork(TCHAR *tas) { PARSESTATE State;
//
// If no input, declare an error
//
if (!tas || !*tas) { PutStdErr(MSG_BAD_SYNTAX, NOARGS); return(FAILURE) ; }
//
// Set up for parsing
//
State.Token = StripQuotes( tas ); State.Value = 0; State.Error = SUCCESS;
State = AParseExpr( State ); if (State.Error == SUCCESS && ANextChar( &State ) != NULLC) { State.Error = MSG_SET_A_MISSING_OPERATOR; }
if (State.Error != SUCCESS) { PutStdErr( State.Error, NOARGS ); //printf( "%ws\n", tas );
//printf( "%*s\n", State.Token - tas + 1, "^" );
} else if (!CurrentBatchFile) { cmd_printf( TEXT("%d"), State.Value ) ; }
return State.Error; }
|