openscad/src/parser.y

578 lines
12 KiB
Plaintext

/*
* OpenSCAD (www.openscad.org)
* Copyright (C) 2009-2011 Clifford Wolf <clifford@clifford.at> and
* Marius Kintel <marius@kintel.net>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* As a special exception, you have permission to link this program
* with the CGAL library and distribute executables, as long as you
* follow the requirements of the GNU GPL in regard to all of the
* software in the executable aside from CGAL.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*/
%expect 1 /* Expect 1 shift/reduce conflict for ifelse_statement - "dangling else problem" */
%{
#include <sys/types.h>
#include <sys/stat.h>
#ifndef _MSC_VER
#include <unistd.h>
#endif
#include "module.h"
#include "expression.h"
#include "value.h"
#include "function.h"
#include "printutils.h"
#include <sstream>
#include <boost/foreach.hpp>
#include <boost/filesystem.hpp>
namespace fs = boost::filesystem;
#include "boosty.h"
int parser_error_pos = -1;
int parserlex(void);
void yyerror(char const *s);
int lexerget_lineno(void);
int lexerlex_destroy(void);
int lexerlex(void);
std::vector<Module*> module_stack;
Module *currmodule;
class ArgContainer {
public:
std::string argname;
Expression *argexpr;
};
class ArgsContainer {
public:
std::vector<std::string> argnames;
std::vector<Expression*> argexpr;
};
%}
%union {
char *text;
double number;
class Value *value;
class Expression *expr;
class ModuleInstantiation *inst;
class IfElseModuleInstantiation *ifelse;
class ArgContainer *arg;
class ArgsContainer *args;
}
%token TOK_MODULE
%token TOK_FUNCTION
%token TOK_IF
%token TOK_ELSE
%token <text> TOK_ID
%token <text> TOK_STRING
%token <text> TOK_USE
%token <number> TOK_NUMBER
%token TOK_TRUE
%token TOK_FALSE
%token TOK_UNDEF
%token LE GE EQ NE AND OR
%right '?' ':'
%left OR
%left AND
%left '<' LE GE '>'
%left EQ NE
%left '!' '+' '-'
%left '*' '/' '%'
%left '[' ']'
%left '.'
%type <expr> expr
%type <expr> vector_expr
%type <inst> module_instantiation
%type <ifelse> if_statement
%type <ifelse> ifelse_statement
%type <inst> children_instantiation
%type <inst> module_instantiation_list
%type <inst> single_module_instantiation
%type <args> arguments_call
%type <args> arguments_decl
%type <arg> argument_call
%type <arg> argument_decl
%debug
%%
input:
/* empty */ |
TOK_USE { currmodule->usedlibs[$1] = NULL; } input |
statement input ;
inner_input:
/* empty */ |
statement inner_input ;
statement:
';' |
'{' inner_input '}' |
module_instantiation {
if ($1) {
currmodule->addChild($1);
} else {
delete $1;
}
} |
TOK_ID '=' expr ';' {
bool add_new_assignment = true;
for (size_t i = 0; i < currmodule->assignments_var.size(); i++) {
if (currmodule->assignments_var[i] != $1)
continue;
delete currmodule->assignments_expr[i];
currmodule->assignments_expr[i] = $3;
add_new_assignment = false;
}
if (add_new_assignment) {
currmodule->assignments_var.push_back($1);
currmodule->assignments_expr.push_back($3);
free($1);
}
} |
TOK_MODULE TOK_ID '(' arguments_decl optional_commas ')' {
Module *p = currmodule;
module_stack.push_back(currmodule);
currmodule = new Module();
p->modules[$2] = currmodule;
currmodule->argnames = $4->argnames;
currmodule->argexpr = $4->argexpr;
free($2);
delete $4;
} statement {
currmodule = module_stack.back();
module_stack.pop_back();
} |
TOK_FUNCTION TOK_ID '(' arguments_decl optional_commas ')' '=' expr {
Function *func = new Function();
func->argnames = $4->argnames;
func->argexpr = $4->argexpr;
func->expr = $8;
currmodule->functions[$2] = func;
free($2);
delete $4;
} ';' ;
/* Will return a dummy parent node with zero or more children */
children_instantiation:
module_instantiation {
$$ = new ModuleInstantiation();
if ($1) {
$$->children.push_back($1);
}
} |
'{' module_instantiation_list '}' {
$$ = $2;
} ;
if_statement:
TOK_IF '(' expr ')' children_instantiation {
$$ = new IfElseModuleInstantiation();
$$->argnames.push_back("");
$$->argexpr.push_back($3);
if ($$) {
$$->children = $5->children;
} else {
for (size_t i = 0; i < $5->children.size(); i++)
delete $5->children[i];
}
$5->children.clear();
delete $5;
} ;
ifelse_statement:
if_statement {
$$ = $1;
} |
if_statement TOK_ELSE children_instantiation {
$$ = $1;
if ($$) {
$$->else_children = $3->children;
} else {
for (size_t i = 0; i < $3->children.size(); i++)
delete $3->children[i];
}
$3->children.clear();
delete $3;
} ;
module_instantiation:
single_module_instantiation ';' {
$$ = $1;
} |
single_module_instantiation children_instantiation {
$$ = $1;
if ($$) {
$$->children = $2->children;
} else {
for (size_t i = 0; i < $2->children.size(); i++)
delete $2->children[i];
}
$2->children.clear();
delete $2;
} |
ifelse_statement {
$$ = $1;
} ;
module_instantiation_list:
/* empty */ {
$$ = new ModuleInstantiation();
} |
module_instantiation_list module_instantiation {
$$ = $1;
if ($$) {
if ($2) $$->children.push_back($2);
} else {
delete $2;
}
} ;
single_module_instantiation:
TOK_ID '(' arguments_call ')' {
$$ = new ModuleInstantiation($1);
$$->argnames = $3->argnames;
$$->argexpr = $3->argexpr;
free($1);
delete $3;
} |
'!' single_module_instantiation {
$$ = $2;
if ($$)
$$->tag_root = true;
} |
'#' single_module_instantiation {
$$ = $2;
if ($$)
$$->tag_highlight = true;
} |
'%' single_module_instantiation {
$$ = $2;
if ($$)
$$->tag_background = true;
} |
'*' single_module_instantiation {
delete $2;
$$ = NULL;
};
expr:
TOK_TRUE {
$$ = new Expression(Value(true));
} |
TOK_FALSE {
$$ = new Expression(Value(false));
} |
TOK_UNDEF {
$$ = new Expression(Value::undefined);
} |
TOK_ID {
$$ = new Expression();
$$->type = "L";
$$->var_name = $1;
free($1);
} |
expr '.' TOK_ID {
$$ = new Expression();
$$->type = "N";
$$->children.push_back($1);
$$->var_name = $3;
free($3);
} |
TOK_STRING {
$$ = new Expression(Value(std::string($1)));
free($1);
} |
TOK_NUMBER {
$$ = new Expression(Value($1));
} |
'[' expr ':' expr ']' {
Expression *e_one = new Expression(Value(1.0));
$$ = new Expression();
$$->type = "R";
$$->children.push_back($2);
$$->children.push_back(e_one);
$$->children.push_back($4);
} |
'[' expr ':' expr ':' expr ']' {
$$ = new Expression();
$$->type = "R";
$$->children.push_back($2);
$$->children.push_back($4);
$$->children.push_back($6);
} |
'[' optional_commas ']' {
$$ = new Expression(Value(Value::VectorType()));
} |
'[' vector_expr optional_commas ']' {
$$ = $2;
} |
expr '*' expr {
$$ = new Expression();
$$->type = "*";
$$->children.push_back($1);
$$->children.push_back($3);
} |
expr '/' expr {
$$ = new Expression();
$$->type = "/";
$$->children.push_back($1);
$$->children.push_back($3);
} |
expr '%' expr {
$$ = new Expression();
$$->type = "%";
$$->children.push_back($1);
$$->children.push_back($3);
} |
expr '+' expr {
$$ = new Expression();
$$->type = "+";
$$->children.push_back($1);
$$->children.push_back($3);
} |
expr '-' expr {
$$ = new Expression();
$$->type = "-";
$$->children.push_back($1);
$$->children.push_back($3);
} |
expr '<' expr {
$$ = new Expression();
$$->type = "<";
$$->children.push_back($1);
$$->children.push_back($3);
} |
expr LE expr {
$$ = new Expression();
$$->type = "<=";
$$->children.push_back($1);
$$->children.push_back($3);
} |
expr EQ expr {
$$ = new Expression();
$$->type = "==";
$$->children.push_back($1);
$$->children.push_back($3);
} |
expr NE expr {
$$ = new Expression();
$$->type = "!=";
$$->children.push_back($1);
$$->children.push_back($3);
} |
expr GE expr {
$$ = new Expression();
$$->type = ">=";
$$->children.push_back($1);
$$->children.push_back($3);
} |
expr '>' expr {
$$ = new Expression();
$$->type = ">";
$$->children.push_back($1);
$$->children.push_back($3);
} |
expr AND expr {
$$ = new Expression();
$$->type = "&&";
$$->children.push_back($1);
$$->children.push_back($3);
} |
expr OR expr {
$$ = new Expression();
$$->type = "||";
$$->children.push_back($1);
$$->children.push_back($3);
} |
'+' expr {
$$ = $2;
} |
'-' expr {
$$ = new Expression();
$$->type = "I";
$$->children.push_back($2);
} |
'!' expr {
$$ = new Expression();
$$->type = "!";
$$->children.push_back($2);
} |
'(' expr ')' {
$$ = $2;
} |
expr '?' expr ':' expr {
$$ = new Expression();
$$->type = "?:";
$$->children.push_back($1);
$$->children.push_back($3);
$$->children.push_back($5);
} |
expr '[' expr ']' {
$$ = new Expression();
$$->type = "[]";
$$->children.push_back($1);
$$->children.push_back($3);
} |
TOK_ID '(' arguments_call ')' {
$$ = new Expression();
$$->type = "F";
$$->call_funcname = $1;
$$->call_argnames = $3->argnames;
$$->children = $3->argexpr;
free($1);
delete $3;
} ;
optional_commas:
',' optional_commas | ;
vector_expr:
expr {
$$ = new Expression();
$$->type = 'V';
$$->children.push_back($1);
} |
vector_expr ',' optional_commas expr {
$$ = $1;
$$->children.push_back($4);
} ;
arguments_decl:
/* empty */ {
$$ = new ArgsContainer();
} |
argument_decl {
$$ = new ArgsContainer();
$$->argnames.push_back($1->argname);
$$->argexpr.push_back($1->argexpr);
delete $1;
} |
arguments_decl ',' optional_commas argument_decl {
$$ = $1;
$$->argnames.push_back($4->argname);
$$->argexpr.push_back($4->argexpr);
delete $4;
} ;
argument_decl:
TOK_ID {
$$ = new ArgContainer();
$$->argname = $1;
$$->argexpr = NULL;
free($1);
} |
TOK_ID '=' expr {
$$ = new ArgContainer();
$$->argname = $1;
$$->argexpr = $3;
free($1);
} ;
arguments_call:
/* empty */ {
$$ = new ArgsContainer();
} |
argument_call {
$$ = new ArgsContainer();
$$->argnames.push_back($1->argname);
$$->argexpr.push_back($1->argexpr);
delete $1;
} |
arguments_call ',' optional_commas argument_call {
$$ = $1;
$$->argnames.push_back($4->argname);
$$->argexpr.push_back($4->argexpr);
delete $4;
} ;
argument_call:
expr {
$$ = new ArgContainer();
$$->argexpr = $1;
} |
TOK_ID '=' expr {
$$ = new ArgContainer();
$$->argname = $1;
$$->argexpr = $3;
free($1);
} ;
%%
int parserlex(void)
{
return lexerlex();
}
void yyerror (char const *s)
{
// FIXME: We leak memory on parser errors...
PRINTB("Parser error in line %d: %s\n", lexerget_lineno() % s);
currmodule = NULL;
}
extern void lexerdestroy();
extern FILE *lexerin;
extern const char *parser_input_buffer;
const char *parser_input_buffer;
std::string parser_source_path;
Module *parse(const char *text, const char *path, int debug)
{
lexerin = NULL;
parser_error_pos = -1;
parser_input_buffer = text;
parser_source_path = std::string(path);
module_stack.clear();
Module *rootmodule = currmodule = new Module();
// PRINTB_NOCACHE("New module: %s %p", "root" % rootmodule);
parserdebug = debug;
int parserretval = parserparse();
lexerdestroy();
lexerlex_destroy();
if (parserretval != 0) return NULL;
parser_error_pos = -1;
return rootmodule;
}