Howdy ,..
Mikaa has created this nice prog for solving binary expression it is by CC licence.
to compile :

cc -c -lm expressions.cpp -o bexpr
g++ bexpr


/****************************************************************
* Expressions.
*
* This programm calculates boolean expressions.
* Any englsih character stands for variable. Variables are
* case-insensetive.
* '+' stands for 'OR'
* '*' stands for 'AND'
* '!' (postfix) stands for unary 'NOT'
* To get results type '='.
* Max. variable count for now is 26.
*
* Programm reads expression and parses it into a binary tree
* of operations. Recursive descendant method is used for it.
* If syntax is Ok, it prints boolean operation table for
* expression, if not, then error message is printed.
*
* When a tree is ready, programm tries all combinations
* of variables values, evaluates expression for
* every combination and prints results.
*
* Well, the code is structed very bad for now, but it works. If
* you have suggestions, or if you found a bug --  please, let me
* know by mail ([EMAIL PROTECTED]) or by ICQ (231914268).
*
* Hope, you will find the programm usefull.
* WBR, Mikae.
*****************************************************************
*/
#include <stdio.h>
#include <stdlib.h>
#include <ctype.h>
#include <math.h>

#define MAX_VARIABLE_COUNT 26
#define EOE '=' //End Of Expression

enum SYNTAX_STATUS
{
        SYNTAX_STATUS_ERR_NO_PARENTH,
        SYNTAX_STATUS_ERR_PARENTH,
        SYNTAX_STATUS_ERR_OP,
        SYNTAX_STATUS_ERR_VAR,
        SYNTAX_STATUS_ERR_MEM,
        SYNTAX_STATUS_ERR_VAR_EXCEED,
        SYNTAX_STATUS_OK,
        SYNTAX_STATUS_DONE
};

enum NODE_TYPE
{
        NODE_TYPE_OPERAND,
        NODE_TYPE_OR,
        NODE_TYPE_AND,
        NODE_TYPE_NOT
};

struct NODE
{
        NODE *right;
        NODE *left;
        char name;
        NODE_TYPE type;
};

struct VARIABLE
{
        char name;
        int value;
};


/*********************
* Variables.
**********************
*/
int vars_count;
VARIABLE vars[MAX_VARIABLE_COUNT];

void init_vars()
{
        vars_count = 0;
}

//Looks for name in array of variables. If name not found
// checks, whether it is possible to add new variable or not.
// If variable was successfully added returns true, else returns false.
int insert_variable(char name)
{
        int found = 0, res = 1;

        for (int i = 0; i < vars_count; i++)
                if (vars[i].name == name)
                {
                        found = 1;
                        break;
                }

        if (!found)
        {
                if (vars_count < MAX_VARIABLE_COUNT)
                {
                        vars[vars_count].name = name;
                        vars[vars_count].value = 0;
                        vars_count++;
                }
                else
                        res = 0;
        }

        return res;
}

//Looks for name in array of variables. If found returns its value,
// else prints information about exception and returns 0.
int get_var_value(char name)
{
        int res = 0;
        int found = 0;

        for (int i = 0; i < vars_count; i++)
                if (vars[i].name == name)
                {
                        res = vars[i].value;
                        found = 1;
                        break;
                }

        if (!found)
                printf("DBG: Exception: name not found.\n");

        return res;
}

int comparator(const void *p1, const void *p2)
{
        return ((VARIABLE *)p1) ->name > ((VARIABLE *)p2) ->name;
}

//Sorts variables in the array by name. Not sure that I need it.
void sort_variables()
{
        qsort(vars, vars_count, sizeof(VARIABLE), comparator);
}

void print_vars_names()
{
        for (int i = 0; i < vars_count; i++)
                printf("%c ", vars[i].name);
}

//Just prints names of variables using space as delimeter.
void print_vars_values()
{
        for (int i = 0; i < vars_count; i++)
                printf("%i ", (vars[i].value != 0) ? 1 : 0);
}

/******************
* Work with trees.
*******************
*/
NODE *create_leaf(char name, NODE_TYPE type)
{
        NODE *res = (NODE *)malloc(sizeof(NODE));

        if (res != NULL)
        {
                res ->name = name;
                res ->type = type;
                res ->left = NULL;
                res ->right = NULL;
        }

        return res;
}

NODE *create_node(char name, NODE_TYPE type, NODE *left, NODE *right)
{
        NODE *node;

        node = create_leaf(name, type);
        if (node)
        {
                node ->left = left;
                node ->right = right;
        }

        return node;
}

//Just prints tree considering node's priorites. Returns length of
// printed chars.
int print_tree(NODE *root)
{
        int ct = 0;

        if (root ->left)
        {
                if ((root ->left ->type != NODE_TYPE_OPERAND) &&
                        (root ->type > root ->left ->type)
                        )
                        ct += printf("(");

                ct += print_tree(root ->left);

                if ((root ->left ->type != NODE_TYPE_OPERAND) &&
                        (root ->type > root ->left ->type)
                        )
                        ct += printf("\b) ") - 2;
        }

        ct += printf("%c ", root ->name);

        if (root ->right)
        {
                if ((root ->right ->type != NODE_TYPE_OPERAND) &&
                        (root ->type > root ->right ->type)
                        )
                        ct += printf("(");

                ct += print_tree(root ->right);

                if ((root ->right ->type != NODE_TYPE_OPERAND) &&
                        (root ->type > root ->right ->type)
                        )
                        ct += printf("\b) ") - 2;
        }

        return ct;
}

//Evaluates tree using current set of variables values.
int evaluate_tree(NODE *root)
{
        int right, left, res;

        if (root ->right)
                right = evaluate_tree(root ->right);

        if (root ->left)
                left = evaluate_tree(root ->left);

        switch (root ->type)
        {
        case NODE_TYPE_OPERAND:
                res = get_var_value(root ->name);
                break;
        case NODE_TYPE_AND:
                res = left && right;
                break;
        case NODE_TYPE_OR:
                res = left || right;
                break;
        case NODE_TYPE_NOT:
                res = !left;
                break;
        default:
                printf("DBG: Exception: unsupported node type.\n");
                break;
        }

        return res;
}

//Frees memory allocated for nodes.
void destroy_tree(NODE *root)
{
        if (root)
        {
                if (root ->right)
                        destroy_tree(root ->right);
                if (root ->left)
                        destroy_tree(root ->left);

                free(root);
        }
}

/************************
* Kinda lexem analyzator.
*************************
*/
char get_lexem()
{
        char c;

        while (isspace(c = getchar()))
                ;

        return c;
}

/**********************************
* Functions for expression parsing.
***********************************
*/
//The following four functions build expression tree recursively.
// 'parse_or' function starts parsing.
//
// char *lexem            [in]: next lexem for parsing.
// NODE *root            [out]: root of tree of parsed expression
// SYNTAX_STATUS *status [out]: status of syntax analyzing.
//                              Is equal to SYNTAX_STATUS_DONE if
//                              everything Ok.

NODE *parse_or(char *lexem, SYNTAX_STATUS *status);

NODE *parse_var(char *lexem, SYNTAX_STATUS *status)
{
        NODE *node = NULL;

        if (*lexem == '\0')
                *lexem = get_lexem();

        if (!isalpha(*lexem))
        {
                switch (*lexem)
                {
                case '(':
                        *lexem = get_lexem();
                        node = parse_or(lexem, status);
                        if (*status == SYNTAX_STATUS_ERR_PARENTH)
                        {
                                *status = SYNTAX_STATUS_OK;
                                *lexem = get_lexem();
                        }
                        else if ((*status == SYNTAX_STATUS_DONE) || (*status == 
SYNTAX_STATUS_OK))
                                *status = SYNTAX_STATUS_ERR_NO_PARENTH;
                        break;
                case EOE:
                        *status = SYNTAX_STATUS_DONE;
                        break;
                default:
                        *status = SYNTAX_STATUS_ERR_VAR;
                        node = NULL;
                        break;
                }
        }
        else
        {
                char name = toupper(*lexem);
                node = create_leaf(name, NODE_TYPE_OPERAND);
                if (!node)
                        *status = SYNTAX_STATUS_ERR_MEM;
                else if (!insert_variable(name))
                        *status = SYNTAX_STATUS_ERR_VAR_EXCEED;

                *lexem = get_lexem();
        }

        return node;
}

NODE *parse_not(char *lexem, SYNTAX_STATUS *status)
{
        NODE *left = NULL, *node;

        left = parse_var(lexem, status);

        if (*status == SYNTAX_STATUS_OK)
        {
                while ((*lexem == '!') && (*status == SYNTAX_STATUS_OK))
                {
                        node = create_node(*lexem, NODE_TYPE_NOT, left, NULL);
                        if (node)
                                left = node;
                        else
                                *status = SYNTAX_STATUS_ERR_MEM;

                        *lexem = get_lexem();
                }
        }

        return left;
}

NODE *parse_and(char *lexem, SYNTAX_STATUS *status)
{
        NODE *left = NULL;
        NODE *right;
        NODE *node;
        char operation;
        int run = 1;

        left = parse_not(lexem, status);
        if (*status != SYNTAX_STATUS_OK)
                run = 0;

        operation = *lexem;

        while (run)
                switch (operation)
                {
                case '*':
                        *lexem = '\0';
                        right = parse_not(lexem, status);
                        if (*status == SYNTAX_STATUS_OK)
                        {
                                node = create_node(operation, NODE_TYPE_AND, 
left, right);
                                if (node)
                                        left = node;
                                else
                                {
                                        *status = SYNTAX_STATUS_ERR_MEM;
                                        run = 0;
                                }
                        }
                        else
                                run = 0;
                        operation = *lexem;
                        break;
                default:
                        run = 0;
                        break;
                }

        return left;
}

NODE *parse_or(char *lexem, SYNTAX_STATUS *status)
{
        NODE *left = NULL;
        NODE *right;
        NODE *node;
        char operation;
        int run = 1;

        left = parse_and(lexem, status);
        if (*status != SYNTAX_STATUS_OK)
                run = 0;
        operation = *lexem;

        while (run)
                switch (operation)
                {
                case '+':
                        *lexem = '\0';
                        right = parse_and(lexem, status);
                        if (*status == SYNTAX_STATUS_OK)
                        {
                                node = create_node(operation, NODE_TYPE_OR, 
left, right);
                                if (node)
                                        left = node;
                                else
                                {
                                        *status = SYNTAX_STATUS_ERR_MEM;
                                        run = 0;
                                }
                        }
                        else
                                run = 0;
                        operation = *lexem;
                        break;
                case ')':
                        *status = SYNTAX_STATUS_ERR_PARENTH;
                        run = 0;
                        break;
                default:
                        *status = SYNTAX_STATUS_ERR_OP;
                        run = 0;
                        break;
                case EOE:
                        *status = SYNTAX_STATUS_DONE;
                        run = 0;
                        break;
                }

        return left;
}

/**********************
* Kinda error handling.
***********************
*/
//Just prints error description by error number.
void print_error(SYNTAX_STATUS status)
{
        char *err_str;

        switch (status)
        {
        case SYNTAX_STATUS_ERR_PARENTH:
                err_str = "Unexpected ')'";
                break;
        case SYNTAX_STATUS_ERR_NO_PARENTH:
                err_str = "Unmatched '('";
                break;
        case SYNTAX_STATUS_ERR_OP:
                err_str = "Expected operator";
                break;
        case SYNTAX_STATUS_ERR_VAR:
                err_str = "Expected variable";
                break;
        case SYNTAX_STATUS_ERR_MEM:
                err_str = "Not enough memory";
                break;
        case SYNTAX_STATUS_ERR_VAR_EXCEED:
                err_str = "Sorry, count of variables (26 for this version) 
exceeded";
                break;
        default:
                err_str = "DBG: Exception: Unknown error type";
                break;
        }

        printf("Error: %s.\n\n", err_str);
}

//Runs over all combinations of variables values,
// evaluating every set.
void evaluate(NODE *root)
{
        int v, p;

        p = (int)pow(2, vars_count);
        for (int i = p - 1; i >= 0; i--)
        {
                for (int j = 0; j < vars_count; j++)
                {
                        v = p >> 1;
                        v >>= j;
                        vars[j].value = i & v;

                }

                print_vars_values();
                printf(" %i", (evaluate_tree(root) != 0) ? 1 : 0);
                printf("\n");
        }
}

void print_line(int len)
{
        for (int i = 0; i < len; i++)
                printf("-");
}

void usage()
{
        printf("Any englsih character stands for variable.\n"
           "'+' stands for 'OR'\n"
           "'*' stands for 'AND'\n"
           "'!' (postfixed) stands for unary 'NOT'\n"
                   "For quit press CTRL+Z (Windows) or CTRL+D (Unix).\n\n"
                   "If you found a bug -- please, let me know ([EMAIL 
PROTECTED]).\n\n\n"
                  );
}

int main(int argc, char **argv)
{
        NODE          *root;
        SYNTAX_STATUS status;
        char          lexem;
        int           len;

        usage();

        printf("Enter expression. Type '=' to get results.\n\n");
        while ((lexem = get_lexem()) != EOF)
        {
                init_vars();
                status = SYNTAX_STATUS_OK;
                root = parse_or(&lexem, &status);
                if (status != SYNTAX_STATUS_DONE)
                {
                        print_error(status);
                        destroy_tree(root);
                        //remove trailing characters
                        while (getchar() != '\n')
                                ;
                }
                else
                {
                        if (root)
                        {
                                len = 0;
                                len += printf("Parsed expression: ");
                                len += print_tree(root);
                                printf("\n");
                                print_line(len);
                                printf("\n");
                                print_vars_names(); printf(" Result:");
                                printf("\n");
                                evaluate(root);
                                printf("\n");
                                destroy_tree(root);
                        }
                }

                printf("Enter expression. Type '=' to get results.\n\n");
        }

        return 0;
}

Reply via email to