From 36bc0bc9b092f6dc476ce0f2be3637ff4961cbad Mon Sep 17 00:00:00 2001 From: Antonin Portelli Date: Thu, 30 Jan 2014 19:28:30 +0000 Subject: [PATCH] math compiler supports function calls + standard C math library wrapper --- examples/ex_test.cpp | 13 ++++-- latan/Function.cpp | 81 +++++++++++++++++++++++++++++++++++ latan/Function.hpp | 45 ++++++++++++++++++++ latan/Makefile.am | 4 ++ latan/Math.cpp | 95 ++++++++++++++++++++++++++++++++++++++++++ latan/Math.hpp | 73 ++++++++++++++++++++++++++++++++ latan/MathCompiler.cpp | 70 ++++++++++++++++++++++++++----- latan/MathCompiler.hpp | 45 ++++++++++++++++---- latan/MathLexer.lpp | 7 ++-- latan/MathParser.ypp | 23 ++++++++-- latan/includes.hpp | 1 + 11 files changed, 426 insertions(+), 31 deletions(-) create mode 100644 latan/Function.cpp create mode 100644 latan/Function.hpp create mode 100644 latan/Math.cpp create mode 100644 latan/Math.hpp diff --git a/examples/ex_test.cpp b/examples/ex_test.cpp index cb03976..944dd31 100644 --- a/examples/ex_test.cpp +++ b/examples/ex_test.cpp @@ -1,4 +1,5 @@ #include +#include #include using namespace std; @@ -8,15 +9,21 @@ int main(int argc, char* argv[]) { MathCompiler C(argv[1]); VarTable vtable; + FunctionTable ftable; stack dstack; const VirtualProgram& P = C(); + ftable["exp"] = &StdMath::exp; + ftable["atan2"] = &StdMath::atan2; cout << P << endl; - for (int i=0;i &dStack, VarTable &vTable __dumb, + FunctionTable &fTable) +{ + if (keyExists(name_, fTable)) + { + dStack.push((*fTable[name_])(dStack)); + } + else + { + LATAN_ERROR(Range, "unknown function '" + name_ + "'"); + } +} + +void Call::print(std::ostream &out) const +{ + out << CODE_MOD << "call" << CODE_MOD << name_; +} + #define DEF_OP(name, nArg, exp, insName)\ -void name::operator()(stack &dStack, VarTable &vTable __dumb)\ +void name::operator()(stack &dStack, VarTable &vTable __dumb,\ + FunctionTable &fTable __dumb)\ {\ double x[nArg];\ for (int i = 0; i < nArg; ++i)\ @@ -320,14 +354,21 @@ void MathCompiler::compile(const MathNode& n) case MathNode::Type::op: if (n.getName() == "=") { - compile(n[1]); - if (n.isRoot()) + if (n[0].getType() == MathNode::Type::var) { - out_.push_back(new Pop(n[0].getName())); + compile(n[1]); + if (n.isRoot()) + { + out_.push_back(new Pop(n[0].getName())); + } + else + { + out_.push_back(new Store(n[0].getName())); + } } else { - out_.push_back(new Store(n[0].getName())); + LATAN_ERROR(Compilation, "invalid LHS for '='"); } } else @@ -352,6 +393,13 @@ void MathCompiler::compile(const MathNode& n) compile(n[i]); } break; + case MathNode::Type::func: + for (unsigned int i = 0; i < n.getNArg(); ++i) + { + compile(n[i]); + } + out_.push_back(new Call(n.getName())); + break; default: LATAN_ERROR(Compilation, "unknown node type (node '" + n.getName() + "')"); diff --git a/latan/MathCompiler.hpp b/latan/MathCompiler.hpp index 61a8a0b..cd1e8ae 100644 --- a/latan/MathCompiler.hpp +++ b/latan/MathCompiler.hpp @@ -6,6 +6,7 @@ #include #include #include +#include #include #include @@ -27,7 +28,8 @@ public: cst = 0, op = 1, var = 2, - keyw = 3 + keyw = 3, + func = 4 }; }; public: @@ -41,6 +43,8 @@ public: const std::string& getName(void) const; unsigned int getType(void) const; unsigned int getNArg(void) const; + void setName(const std::string &name); + void pushArg(MathNode *node); // operator const MathNode &operator[](const unsigned int i) const; // test @@ -55,7 +59,8 @@ private: /****************************************************************************** * Virtual machine code classes * ******************************************************************************/ -typedef std::map VarTable; +typedef std::map VarTable; +typedef std::map FunctionTable; // Abstract base class Instruction @@ -63,13 +68,14 @@ class Instruction public: virtual ~Instruction(); // instruction execution - virtual void operator()(std::stack &dStack, VarTable &vTable) = 0; + virtual void operator()(std::stack &dStack, VarTable &vTable, + FunctionTable &fTable) = 0; friend std::ostream& operator<<(std::ostream &out, const Instruction &ins); private: virtual void print(std::ostream &out) const = 0; }; -// push, pop and store +// Push class Push: public Instruction { private: @@ -87,7 +93,8 @@ public: explicit Push(const double val); explicit Push(const std::string &name); // instruction execution - virtual void operator()(std::stack &dStack, VarTable &vTable); + virtual void operator()(std::stack &dStack, VarTable &vTable, + FunctionTable &fTable); private: virtual void print(std::ostream& out) const; private: @@ -96,38 +103,58 @@ private: std::string name_; }; +// Pop class Pop: public Instruction { public: //constructor explicit Pop(const std::string &name); // instruction execution - virtual void operator()(std::stack &dStack, VarTable &vTable); + virtual void operator()(std::stack &dStack, VarTable &vTable, + FunctionTable &fTable); private: virtual void print(std::ostream& out) const; private: std::string name_; }; +// Store class Store: public Instruction { public: //constructor explicit Store(const std::string &name); // instruction execution - virtual void operator()(std::stack &dStack, VarTable &vTable); + virtual void operator()(std::stack &dStack, VarTable &vTable, + FunctionTable &fTable); private: virtual void print(std::ostream& out) const; private: std::string name_; }; -// Float operations +// Call function +class Call: public Instruction +{ +public: + //constructor + explicit Call(const std::string &name); + // instruction execution + virtual void operator()(std::stack &dStack, VarTable &vTable, + FunctionTable &fTable); +private: + virtual void print(std::ostream& out) const; +private: + std::string name_; +}; + +// Floating point operations #define DECL_OP(name)\ class name: public Instruction\ {\ public:\ - virtual void operator()(std::stack &dStack, VarTable &vTable);\ +virtual void operator()(std::stack &dStack, VarTable &vTable,\ + FunctionTable &fTable);\ private:\ virtual void print(std::ostream &out) const;\ } diff --git a/latan/MathLexer.lpp b/latan/MathLexer.lpp index 4f6e53c..7bee910 100644 --- a/latan/MathLexer.lpp +++ b/latan/MathLexer.lpp @@ -42,9 +42,9 @@ DIGIT [0-9] ALPHA [a-zA-Z_] FLOAT (({DIGIT}+(\.{DIGIT}*)?)|({DIGIT}*\.{DIGIT}+))([eE][+-]?{DIGIT}+)? KEYWORD return -END ; +PUNC [;,()] OP [+\-*/^=] -PAR [()] +PAR [] BLANK [ \t] %% @@ -55,12 +55,11 @@ BLANK [ \t] } {OP} {RET(*yytext);} {KEYWORD} {RET(*yytext);} -{PAR} {RET(*yytext);} +{PUNC} {RET(*yytext);} {ALPHA}({ALPHA}|{DIGIT})* { strncpy(yylval->val_str,yytext,MAXIDLENGTH); RETTOK(ID); } -{END} {RETTOK(END);} <*>\n {yylloc->last_column = 0;} <*>{BLANK} <*>. {yylval->val_char = yytext[0]; RETTOK(ERR);} diff --git a/latan/MathParser.ypp b/latan/MathParser.ypp index be60b0d..41f48ab 100644 --- a/latan/MathParser.ypp +++ b/latan/MathParser.ypp @@ -26,7 +26,6 @@ Latan::MathNode *val_node; } -%token END %token ERR %token FLOAT %token ID @@ -39,6 +38,7 @@ %nonassoc UMINUS %type expr +%type funcargs %{ int _math_lex(YYSTYPE *lvalp, YYLTYPE *llocp, void *scanner); @@ -60,7 +60,7 @@ program: /* empty string */ - | program expr END {state->data->push_back($2);} + | program expr ';' {state->data->push_back($2);} ; expr: @@ -68,12 +68,15 @@ expr: {$$ = new MathNode($FLOAT, MathNode::Type::cst);} | ID {$$ = new MathNode($ID,MathNode::Type::var);} + | ID '=' expr + { + $$ = new MathNode("=", MathNode::Type::op, 2, \ + new MathNode($ID,MathNode::Type::var), $3); + } | 'r' expr {$$ = new MathNode("return", MathNode::Type::keyw, 1, $2);} | '-' expr %prec UMINUS {$$ = new MathNode("-", MathNode::Type::op, 1, $2);} - | expr '=' expr - {$$ = new MathNode("=", MathNode::Type::op, 2, $1, $3);} | expr '+' expr {$$ = new MathNode("+", MathNode::Type::op, 2, $1, $3);} | expr '-' expr @@ -86,4 +89,16 @@ expr: {$$ = new MathNode("^", MathNode::Type::op, 2, $1, $3);} | '(' expr ')' {$$ = $2;} + | ID '(' funcargs ')' + {$$ = $3; $$->setName($ID);} ; + +funcargs: + /* empty string */ + {$$ = new MathNode("", MathNode::Type::func);} + | expr + {$$ = new MathNode("", MathNode::Type::func, 1, $1);} + | funcargs ',' expr + {$$ = $1; $$->pushArg($3);} + ; + diff --git a/latan/includes.hpp b/latan/includes.hpp index fa9cdad..61c67d1 100644 --- a/latan/includes.hpp +++ b/latan/includes.hpp @@ -5,6 +5,7 @@ #include #include #include +#include #include #include "../config.h"