2020-09-30 16:53:49 +02:00

595 lines
18 KiB
C

/***
*eval.c - if expression evaluator
*
* Copyright (c) 1992-2001, Microsoft Corporation. All rights reserved.
*
*Purpose:
* produce truth values and simplified conditions from compound if
* statements.
*
*Revision History:
* 09-30-92 MAL Original version
* 10-13-93 SKS Recognize comments of the form /-*IFSTRIP=IGN*-/ to
* override ifstrip behavior.
* 09-01-94 SKS Add support for more operators: == != < > <= >=
* Add terseflag (-t) to suppress mesgs about directives
* 10-04-94 SKS Add support for more operators: EQ NE LT GT LE NE
* @ is an identifier character (e.g., MASM's @Version)
* 01-04-00 GB Add support for internal crt builds
*
*******************************************************************************/
#include <stdio.h>
#include <ctype.h>
#include <string.h>
#include <stdlib.h>
#include "eval.h" /* Header for this module */
#include "symtab.h" /* Symbol table access */
#include "constant.h" /* Constants for tokens etc */
#include "errormes.h" /* Errors and Warning messages */
/* Types */
typedef struct condrec
{
int truth;
char *condition;
} condrec, *cond;
/* Global variables */
extern int terseFlag; /* controls display of forced directives */
extern char **comments; /* Language dependent comment strings */
extern int *commlen; /* Lengths of above strings */
extern int nonumbers; /* allow numeric expressions */
extern enum {NON_CRT = 0, CRT = 1} progtype;
char *operators[] = {
"!", "(", ")", "||", "&&", "defined" ,
"==" , "!=" , "<" , ">" , "<=" , ">=" ,
"EQ" , "NE" , "LT" , "GT" , "LE" , "GE" };
int oplengths[] = {
1 , 1 , 1 , 2 , 2 , 7 ,
2 , 2 , 1 , 1 , 2 , 2 ,
2 , 2 , 2 , 2 , 2 , 2 };
/* # significant chars */
#define numoperators 18
/* These tokens must be in the same order as 'operators' */
#define NOT 0
#define OPENPARENTHESIS 1
#define CLOSEPARENTHESIS 2
#define OR 3
#define AND 4
#define DEFINEDFN 5
#define EQUALS 6
#define NOTEQUALS 7
#define LESSTHAN 8
#define LESSOREQ 9
#define GREATERTHAN 10
#define GREATEROREQ 11
#define EQUALS_ASM 12
#define NOTEQUALS_ASM 13
#define LESSTHAN_ASM 14
#define LESSOREQ_ASM 15
#define GREATERTHAN_ASM 16
#define GREATEROREQ_ASM 17
#define UINT 100
#define ID 101
#define ENDOFLINE 102
#define UNKNOWN 103
/* Global state */
/* String holding input, the current pointer into it, and the current token */
char conditions[MAXLINELEN], *tokenptr, currenttoken[MAXCONDLEN];
int token = -1;
/* Function prototypes */
cond evaluateexpression(void);
cond orexpression(void);
cond andexpression(void);
cond unaryexpression(void);
cond parenthesesexpression(void);
cond atomicexpression(void);
cond createcondition(void);
void destroycondition(cond);
char *createstring(int);
void destroystring(char *);
void gettoken(void);
int issymbolchar(char);
/* CFW - added complex expression warning */
void evalwarn()
{
warning("cannot parse expression - ignoring", conditions);
}
/*
* Comments may be in the input source which contain IFSTRIP Directives:
*
* For C source they should look like:
*
* #if foo>bar !*IFSTRIP=DEF*! '!' here stands for '/'
*
* and for Assembler source they should look like:
*
* if foo ;;IFSTRIP=UND;;
*
* Note that the directive and an optional preceding blank or tab
* are deleted from the input source. The directive can thus be
* followed by a backslash to continue the line, another comment, etc.
*/
static char IfStripCStr[] =
"/*IFSTRIP="; /* -*IFSTRIP=IGN*- */
static char IfStripAStr[] =
/*0123456789*- -* 0123456789012345 */
";;IFSTRIP="; /* ;;IFSTRIP=IGN;; */
/* IGN may also be DEF or UND */
#define IFSTRIPVALOFF 10
#define IFSTRIPVALEND 13
#define IFSTRIPVALLEN 15
void evaluate(char *outputstring, int *value, char *inputstring)
{
int forcevalue = IGNORE;
cond result;
strcpy(conditions, inputstring); /* Prepare the string for tokenising */
tokenptr = conditions;
gettoken(); /* Read in the first token of input */
result = evaluateexpression();
/* check for bad/complex expression */
if (token != ENDOFLINE)
{
char *adir = NULL;
char *cdir = NULL;
if(((cdir = strstr(inputstring, IfStripCStr)) && cdir[IFSTRIPVALEND] == '*' && cdir[IFSTRIPVALEND+1] == '/')
|| ((adir = strstr(inputstring, IfStripAStr)) && adir[IFSTRIPVALEND] == ';' && adir[IFSTRIPVALEND+1] == ';'))
{
char *pstr;
char *ifstr;
/* fprintf(stderr,"<< evaluate(): (%s)\n", inputstring); */
pstr = ifstr = ( adir ? adir : cdir ) ;
/*
* Have recognized the /-*-IFSTRIP= directive, interpret its argument
* and remove the directive comment from the input/output text.
* Back up exactly one white space character (blank or tab) if possible.
*/
if(pstr > inputstring && (pstr[-1] == '\t' || pstr[-1] == ' '))
-- pstr;
if(!memcmp(ifstr+IFSTRIPVALOFF, "DEF", 3)) /* DEFINED */
forcevalue = DEFINED;
else if(!memcmp(ifstr+IFSTRIPVALOFF, "UND", 3)) /* UNDEFINED */
forcevalue = UNDEFINED;
else if(memcmp(ifstr+IFSTRIPVALOFF, "IGN", 3)) /* IGNORE */
warning("cannot recognize IFSTRIP: directive - ignoring", conditions);
/* else "IGNORE" -- forcevalue is already set by default to IGNORE */
if(!terseFlag)
warning("ifstrip directive forced evaluation", conditions);
/* remove the directive comment (and preceding blank or tab) from the input line */
strcpy(pstr, ifstr + IFSTRIPVALLEN); /* "C" comments have closing -*-/- */
/* fprintf(stderr,">> evaluate(): (%s)\n", inputstring); */
}
else
evalwarn();
if (result)
{
destroycondition(result);
result = NULL;
}
}
/* bad/complex expression, return IGNORE and entire expression */
if (!result)
{
*value = forcevalue;
strcpy(outputstring, inputstring);
return;
}
*value = result -> truth;
if(!result -> condition)
* outputstring = '\0';
else
strcpy(outputstring, result -> condition);
/* Convert from internal to external representation */
destroycondition(result);
}
cond evaluateexpression()
{
return orexpression();
}
cond orexpression()
{
cond condition1, condition2;
char *output;
condition1 = andexpression();
if (!condition1)
return NULL;
while (token == OR)
{
gettoken();
condition2 = andexpression();
if (!condition2)
{
destroycondition(condition1);
return NULL;
}
switch (condition1 -> truth)
{
case DEFINED: /* DEFINED || x == DEFINED */
/* condition1 set up correctly for next pass */
destroycondition(condition2);
break;
case UNDEFINED:
switch (condition2 -> truth)
{
case DEFINED: /* UNDEFINED || DEFINED == DEFINED */
destroycondition(condition1);
condition1 = condition2;
break;
case UNDEFINED: /* UNDEFINED || UNDEFINED == UNDEFINED */
destroycondition(condition2);
/* condition1 set up correctly for next pass */
break;
case IGNORE: /* UNDEFINED || IGNORE == IGNORE */
destroycondition(condition1);
condition1 = condition2;
break;
}
break;
case IGNORE:
switch (condition2 -> truth)
{
case DEFINED: /* IGNORE || DEFINED == DEFINED */
destroycondition(condition1);
condition1 = condition2;
break;
case UNDEFINED: /* IGNORE || UNDEFINED == IGNORE */
/* condition1 set up correctly for next pass */
destroycondition(condition2);
break;
case IGNORE: /* IGNORE || IGNORE == IGNORE */
output = createstring(strlen(condition1 -> condition)
+ strlen (condition2 -> condition)
+ (sizeof(" || ") - 1));
strcpy(output, condition1 -> condition);
strcat(output, " || ");
strcat(output, condition2 -> condition);
/* Build up the condition string */
destroystring(condition1 -> condition);
condition1 -> condition = output;
/* Place the new string in condition1 */
destroycondition(condition2);
break;
}
break;
}
}
return condition1;
}
cond andexpression()
{
cond condition1, condition2;
char *output;
condition1 = unaryexpression();
if (!condition1)
return NULL;
while (token == AND)
{
gettoken();
condition2 = unaryexpression();
if (!condition2)
{
destroycondition(condition1);
return NULL;
}
switch (condition1 -> truth)
{
case DEFINED:
switch (condition2 -> truth)
{
case DEFINED: /* DEFINED && DEFINED == DEFINED */
destroycondition(condition2);
/* condition1 set up correctly for next pass */
break;
case UNDEFINED: /* DEFINED && UNDEFINED == UNDEFINED */
destroycondition(condition1);
condition1 = condition2;
break;
case IGNORE: /* DEFINED && IGNORE == IGNORE */
destroycondition(condition1);
condition1 = condition2;
break;
}
break;
case UNDEFINED: /* UNDEFINED && x == UNDEFINED */
/* condition1 set up correctly for next pass */
destroycondition(condition2);
break;
case IGNORE:
switch (condition2 -> truth)
{
case DEFINED: /* IGNORE && DEFINED == IGNORE */
/* condition1 set up correctly for next pass */
destroycondition(condition2);
break;
case UNDEFINED: /* IGNORE && UNDEFINED == UNDEFINED */
destroycondition(condition1);
condition1 = condition2;
break;
case IGNORE: /* IGNORE && IGNORE == IGNORE */
output = createstring(strlen(condition1 -> condition)
+ strlen (condition2 -> condition)
+ (sizeof(" && ") - 1));
strcpy(output, condition1 -> condition);
strcat(output, " && ");
strcat(output, condition2 -> condition);
/* Build up the condition string */
destroystring(condition1 -> condition);
condition1 -> condition = output;
/* Place the new string in condition1 */
destroycondition(condition2);
break;
}
break;
}
}
return condition1;
}
cond unaryexpression()
{
cond condition1;
char *output;
switch (token)
{
case NOT:
gettoken();
condition1 = unaryexpression();
if (!condition1)
return NULL;
if ((condition1 -> truth) == IGNORE)
{
output = createstring(strlen(condition1 -> condition) + 1);
*output = '!';
strcpy(output + 1, condition1 -> condition);
destroystring(condition1 -> condition);
condition1 -> condition = output;
}
else
condition1 -> truth = negatecondition(condition1 -> truth);
break;
case DEFINEDFN:
gettoken();
condition1 = parenthesesexpression();
if (!condition1)
return NULL;
if ((condition1 -> truth) == IGNORE)
{
output = createstring(strlen(condition1 -> condition)
+ (sizeof("defined ") - 1));
strcpy(output, "defined ");
strcat(output, condition1 -> condition);
destroystring(condition1 -> condition);
condition1 -> condition = output;
}
break;
default:
condition1 = parenthesesexpression();
if (!condition1)
return NULL;
break;
}
return condition1;
}
cond parenthesesexpression()
{
cond condition1;
char *output;
if (token == OPENPARENTHESIS)
{
gettoken();
condition1 = evaluateexpression();
if (!condition1)
return NULL;
if (token != CLOSEPARENTHESIS)
{
/* check for bad/complex expression */
evalwarn();
destroycondition(condition1);
return NULL;
}
gettoken();
if ((condition1 -> truth) == IGNORE)
{
output = createstring(strlen(condition1 -> condition) + 2);
*output = '(';
strcpy(output + 1, condition1 -> condition);
strcat(output, ")");
destroystring(condition1 -> condition);
condition1 -> condition = output;
}
}
else
condition1 = atomicexpression();
return condition1;
}
cond atomicexpression()
{
cond condition1 = createcondition();
switch (token)
{
case UINT:
if ( progtype == 1)
condition1 -> truth = DEFINED;
else
condition1 -> truth = (atoi(currenttoken) == 0) ? UNDEFINED : DEFINED;
break;
case ID:
condition1 -> truth = lookupsym(currenttoken);
if ((condition1 -> truth) == NOTPRESENT)
{
warning("Switch unlisted - ignoring", currenttoken);
condition1 -> truth = IGNORE;
}
if ((condition1 -> truth) == IGNORE) {
condition1 -> condition = createstring(strlen(currenttoken));
strcpy(condition1 -> condition, currenttoken);
}
break;
default:
/* bad/complex expression */
evalwarn();
destroycondition(condition1);
return NULL;
break;
}
gettoken();
return condition1;
}
/* Negate condition (MAL) */
__inline int negatecondition(int condvalue) /* inline for speed */
{
switch (condvalue)
{
case DEFINED:
return UNDEFINED;
case UNDEFINED:
return DEFINED;
default:
return condvalue;
};
}
/* Allocate the memory for an empty condition structure and return a pointer to it */
__inline cond createcondition()
{
cond retvalue;
retvalue = (cond) malloc(sizeof(condrec));
if (retvalue == NULL)
error("Memory overflow","");
retvalue -> condition = NULL;
return retvalue;
}
/* Destroy a condition structure */
__inline void destroycondition(cond condition1)
{
if (condition1 -> condition)
free(condition1 -> condition);
free(condition1);
}
/* Allocate the memory for a string of given length (not including terminator) and return the pointer */
__inline char *createstring(int length)
{
char *retvalue;
retvalue = (char *) malloc(length + 1);
if (retvalue == NULL)
error("Memory overflow","");
return retvalue;
}
/* Destroy a string */
__inline void destroystring(char *string)
{
free(string);
}
int iscomment(char *tokenptr)
{
int cindex;
for (cindex = 0; cindex < maxcomment; cindex++)
{
if (commlen[cindex] &&
!_strnicmp(tokenptr, comments[cindex], commlen[cindex]))
return TRUE;
}
return FALSE;
}
void gettoken()
{
int numofwhitespace, comparetoken = 0, found = FALSE, isnumber = TRUE;
char *digitcheck;
numofwhitespace = strspn(tokenptr, " \t");
/* CFW - skips comments, assumes comment is last thing on line */
if (numofwhitespace == (int) strlen(tokenptr))
token = ENDOFLINE;
else
{
tokenptr += numofwhitespace;
if (iscomment(tokenptr))
{
token = ENDOFLINE;
}
else
{
do
{
if (!_strnicmp(tokenptr, operators[comparetoken], oplengths[comparetoken]))
found = TRUE;
else
comparetoken++;
} while ( (!found) && (comparetoken < numoperators) );
if (found)
{
tokenptr += oplengths[comparetoken];
token = comparetoken;
/* currenttoken is left blank for all but UINTs and IDs */
}
else
{
digitcheck = tokenptr;
if (!nonumbers && isdigit(*digitcheck))
{
while (isdigit(*digitcheck))
digitcheck++;
strncpy(currenttoken, tokenptr, digitcheck - tokenptr);
tokenptr = digitcheck;
token = UINT;
}
else if (issymbolchar(*digitcheck))
{
while (issymbolchar(*digitcheck))
digitcheck++;
strncpy(currenttoken, tokenptr, digitcheck - tokenptr);
*(currenttoken + (digitcheck - tokenptr)) = '\0';
tokenptr = digitcheck;
token = ID;
}
else
token = UNKNOWN;
}
}
}
}
__inline int issymbolchar(char c)
{
return (iscsym(c) || (c == '$') || (c == '?') || (c == '@'));
}