v4 preprocessor/macro: added handling of arrays

git-svn-id: https://www.dynare.org/svn/dynare/dynare_v4@1711 ac1d8469-bf42-47a9-8791-bf33cf982152
issue#70
sebastien 2008-02-19 14:24:40 +00:00
parent 8412f57c90
commit 2e8251ebfd
4 changed files with 215 additions and 25 deletions

View File

@ -55,7 +55,7 @@ class MacroDriver;
}; };
%{ %{
#include <stdlib.h> // Pour atoi() #include <cstdlib> // Pour atoi()
#include "MacroDriver.hh" #include "MacroDriver.hh"
/* this "connects" the bison parser in the driver to the flex scanner class /* this "connects" the bison parser in the driver to the flex scanner class
@ -71,6 +71,7 @@ class MacroDriver;
%token <int_val> INTEGER %token <int_val> INTEGER
%token <string_val> NAME STRING %token <string_val> NAME STRING
%left COMMA
%left LOGICAL_OR %left LOGICAL_OR
%left LOGICAL_AND %left LOGICAL_AND
%left LESS GREATER LESS_EQUAL GREATER_EQUAL EQUAL_EQUAL EXCLAMATION_EQUAL %left LESS GREATER LESS_EQUAL GREATER_EQUAL EQUAL_EQUAL EXCLAMATION_EQUAL
@ -79,7 +80,7 @@ class MacroDriver;
%left UMINUS UPLUS EXCLAMATION %left UMINUS UPLUS EXCLAMATION
%left LBRACKET %left LBRACKET
%type <mv> expr %type <mv> expr array_expr
%% %%
%start statement_list_or_nothing; %start statement_list_or_nothing;
@ -133,10 +134,18 @@ expr : INTEGER
{ $$ = $2; } { $$ = $2; }
| EXCLAMATION expr | EXCLAMATION expr
{ $$ = !*$2; delete $2; } { $$ = !*$2; delete $2; }
| expr LBRACKET expr RBRACKET | expr LBRACKET array_expr RBRACKET
{ $$ = (*$1)[*$3]; delete $1; delete $3; } { $$ = (*$1)[*$3]; delete $1; delete $3; }
| LBRACKET array_expr RBRACKET
{ $$ = $2; }
; ;
array_expr : expr
{ $$ = $1->toArray(); delete $1; }
| array_expr COMMA expr
{ $$ = $3->append(*$1); delete $1; delete $3; }
;
%% %%
void void

View File

@ -113,6 +113,7 @@ typedef Macro::parser::token token;
<MACRO>\[ { return token::LBRACKET; } <MACRO>\[ { return token::LBRACKET; }
<MACRO>\] { return token::RBRACKET; } <MACRO>\] { return token::RBRACKET; }
<MACRO>: { return token::COLON; } <MACRO>: { return token::COLON; }
<MACRO>, { return token::COMMA; }
<MACRO>= { return token::EQUAL; } <MACRO>= { return token::EQUAL; }
<MACRO>[!] { return token::EXCLAMATION; } <MACRO>[!] { return token::EXCLAMATION; }
<MACRO>"||" { return token::LOGICAL_OR; } <MACRO>"||" { return token::LOGICAL_OR; }
@ -162,6 +163,9 @@ typedef Macro::parser::token token;
BEGIN(END_INCLUDE); BEGIN(END_INCLUDE);
} }
/* Ignore \r, because under Cygwin, outputting \n automatically adds another \r */
<INITIAL>[\r]+ { yylloc->step(); }
/* Copy everything else to output */ /* Copy everything else to output */
<INITIAL>[\n]+ { yylloc->lines(yyleng); yylloc->step(); ECHO; } <INITIAL>[\n]+ { yylloc->lines(yyleng); yylloc->step(); ECHO; }
<INITIAL>. { yylloc->step(); ECHO; } <INITIAL>. { yylloc->step(); ECHO; }

View File

@ -17,10 +17,12 @@
* along with Dynare. If not, see <http://www.gnu.org/licenses/>. * along with Dynare. If not, see <http://www.gnu.org/licenses/>.
*/ */
#include <sstream>
#include "MacroValue.hh" #include "MacroValue.hh"
MacroValue::~MacroValue()
{
}
MacroValue * MacroValue *
MacroValue::operator-(const MacroValue &mv) const throw (TypeError) MacroValue::operator-(const MacroValue &mv) const throw (TypeError)
{ {
@ -97,6 +99,10 @@ IntMV::IntMV(int value_arg) : value(value_arg)
{ {
} }
IntMV::~IntMV()
{
}
MacroValue * MacroValue *
IntMV::operator+(const MacroValue &mv) const throw (TypeError) IntMV::operator+(const MacroValue &mv) const throw (TypeError)
{ {
@ -243,10 +249,36 @@ IntMV::clone() const
return new IntMV(value); return new IntMV(value);
} }
MacroValue *
IntMV::toArray() const
{
vector<int> v;
v.push_back(value);
return new ArrayMV<int>(v);
}
MacroValue *
IntMV::append(const MacroValue &array) const
{
const ArrayMV<int> *array2 = dynamic_cast<const ArrayMV<int> *>(&array);
if (array2 == NULL)
throw TypeError("Type mismatch for append operation");
else
{
vector<int> v(array2->values);
v.push_back(value);
return new ArrayMV<int>(v);
}
}
StringMV::StringMV(const string &value_arg) : value(value_arg) StringMV::StringMV(const string &value_arg) : value(value_arg)
{ {
} }
StringMV::~StringMV()
{
}
MacroValue * MacroValue *
StringMV::operator+(const MacroValue &mv) const throw (TypeError) StringMV::operator+(const MacroValue &mv) const throw (TypeError)
{ {
@ -277,6 +309,22 @@ StringMV::operator!=(const MacroValue &mv) const throw (TypeError)
return new IntMV(value != mv2->value); return new IntMV(value != mv2->value);
} }
MacroValue *
StringMV::operator[](const MacroValue &mv) const throw (TypeError)
{
const ArrayMV<int> *mv2 = dynamic_cast<const ArrayMV<int> *>(&mv);
if (mv2 == NULL)
throw TypeError("Expression inside [] must be an integer array");
string result;
for(vector<int>::const_iterator it = mv2->values.begin();
it != mv2->values.end(); it++)
{
char c = value.at(*it - 1);
result.append(&c);
}
return new StringMV(result);
}
string string
StringMV::toString() const StringMV::toString() const
{ {
@ -288,3 +336,25 @@ StringMV::clone() const
{ {
return new StringMV(value); return new StringMV(value);
} }
MacroValue *
StringMV::toArray() const
{
vector<string> v;
v.push_back(value);
return new ArrayMV<string>(v);
}
MacroValue *
StringMV::append(const MacroValue &array) const
{
const ArrayMV<string> *array2 = dynamic_cast<const ArrayMV<string> *>(&array);
if (array2 == NULL)
throw TypeError("Type mismatch for append operation");
else
{
vector<string> v(array2->values);
v.push_back(value);
return new ArrayMV<string>(v);
}
}

View File

@ -24,6 +24,7 @@ using namespace std;
#include <string> #include <string>
#include <vector> #include <vector>
#include <sstream>
class MacroValue class MacroValue
{ {
@ -34,6 +35,7 @@ public:
const string message; const string message;
TypeError(const string &message_arg) : message(message_arg) {}; TypeError(const string &message_arg) : message(message_arg) {};
}; };
virtual ~MacroValue();
virtual MacroValue *operator+(const MacroValue &mv) const throw (TypeError) = 0; virtual MacroValue *operator+(const MacroValue &mv) const throw (TypeError) = 0;
virtual MacroValue *operator-(const MacroValue &mv) const throw (TypeError); virtual MacroValue *operator-(const MacroValue &mv) const throw (TypeError);
virtual MacroValue *operator-() const throw (TypeError); virtual MacroValue *operator-() const throw (TypeError);
@ -51,14 +53,18 @@ public:
virtual MacroValue *operator[](const MacroValue &mv) const throw (TypeError); virtual MacroValue *operator[](const MacroValue &mv) const throw (TypeError);
virtual string toString() const = 0; virtual string toString() const = 0;
virtual MacroValue *clone() const = 0; virtual MacroValue *clone() const = 0;
virtual MacroValue *toArray() const = 0;
virtual MacroValue *append(const MacroValue &array) const = 0;
}; };
class IntMV : public MacroValue class IntMV : public MacroValue
{ {
friend class StringMV;
private: private:
int value; int value;
public: public:
IntMV(int value_arg); IntMV(int value_arg);
virtual ~IntMV();
virtual MacroValue *operator+(const MacroValue &mv) const throw (TypeError); virtual MacroValue *operator+(const MacroValue &mv) const throw (TypeError);
virtual MacroValue *operator-(const MacroValue &mv) const throw (TypeError); virtual MacroValue *operator-(const MacroValue &mv) const throw (TypeError);
virtual MacroValue *operator-() const throw (TypeError); virtual MacroValue *operator-() const throw (TypeError);
@ -75,6 +81,8 @@ public:
virtual MacroValue *operator!() const throw (TypeError); virtual MacroValue *operator!() const throw (TypeError);
virtual string toString() const; virtual string toString() const;
virtual MacroValue *clone() const; virtual MacroValue *clone() const;
virtual MacroValue *toArray() const;
virtual MacroValue *append(const MacroValue &array) const;
}; };
class StringMV : public MacroValue class StringMV : public MacroValue
@ -83,6 +91,7 @@ private:
string value; string value;
public: public:
StringMV(const string &value_arg); StringMV(const string &value_arg);
virtual ~StringMV();
virtual MacroValue *operator+(const MacroValue &mv) const throw (TypeError); virtual MacroValue *operator+(const MacroValue &mv) const throw (TypeError);
// virtual MacroValue *operator<(const MacroValue &mv) const throw (TypeError); // virtual MacroValue *operator<(const MacroValue &mv) const throw (TypeError);
// virtual MacroValue *operator>(const MacroValue &mv) const throw (TypeError); // virtual MacroValue *operator>(const MacroValue &mv) const throw (TypeError);
@ -90,39 +99,137 @@ public:
// virtual MacroValue *operator>=(const MacroValue &mv) const throw (TypeError); // virtual MacroValue *operator>=(const MacroValue &mv) const throw (TypeError);
virtual MacroValue *operator==(const MacroValue &mv) const throw (TypeError); virtual MacroValue *operator==(const MacroValue &mv) const throw (TypeError);
virtual MacroValue *operator!=(const MacroValue &mv) const throw (TypeError); virtual MacroValue *operator!=(const MacroValue &mv) const throw (TypeError);
// virtual MacroValue *operator[](const MacroValue &mv) const throw (TypeError); //! Subscripting operator
virtual string toString() const; /*!
virtual MacroValue *clone() const; Argument must be an ArrayMV<int>. Indexes begin at 1.
}; \todo Add bound error checking
*/
/*
class IntArrayMV : public MacroValue
{
private:
vector<int> values;
public:
IntArrayMV(const vector<int> &values_arg);
virtual MacroValue *operator+(const MacroValue &mv) const throw (TypeError);
virtual MacroValue *operator==(const MacroValue &mv) const throw (TypeError);
virtual MacroValue *operator!=(const MacroValue &mv) const throw (TypeError);
virtual MacroValue *operator[](const MacroValue &mv) const throw (TypeError); virtual MacroValue *operator[](const MacroValue &mv) const throw (TypeError);
virtual string toString() const; virtual string toString() const;
virtual MacroValue *clone() const; virtual MacroValue *clone() const;
virtual MacroValue *toArray() const;
virtual MacroValue *append(const MacroValue &array) const;
}; };
class StringArrayMV : public MacroValue template<typename T>
class ArrayMV : public MacroValue
{ {
friend class IntMV;
friend class StringMV;
friend class ArrayMV<string>; // Necessary for operator[] to access values of integer array when subscripting a string array
private: private:
vector<string> values; vector<T> values;
public: public:
StringArrayMV(const vector<string> &values_arg); ArrayMV(const vector<T> &values_arg);
virtual ~ArrayMV();
virtual MacroValue *operator+(const MacroValue &mv) const throw (TypeError); virtual MacroValue *operator+(const MacroValue &mv) const throw (TypeError);
virtual MacroValue *operator==(const MacroValue &mv) const throw (TypeError); virtual MacroValue *operator==(const MacroValue &mv) const throw (TypeError);
virtual MacroValue *operator!=(const MacroValue &mv) const throw (TypeError); virtual MacroValue *operator!=(const MacroValue &mv) const throw (TypeError);
//! Subscripting operator
/*!
Argument must be an ArrayMV<int>. Indexes begin at 1.
\todo Add bound error checking
*/
virtual MacroValue *operator[](const MacroValue &mv) const throw (TypeError); virtual MacroValue *operator[](const MacroValue &mv) const throw (TypeError);
virtual string toString() const; virtual string toString() const;
virtual MacroValue *clone() const; virtual MacroValue *clone() const;
virtual MacroValue *toArray() const;
virtual MacroValue *append(const MacroValue &array) const;
}; };
*/
template<typename T>
ArrayMV<T>::ArrayMV(const vector<T> &values_arg) : values(values_arg)
{
}
template<typename T>
ArrayMV<T>::~ArrayMV()
{
}
template<typename T>
MacroValue *
ArrayMV<T>::operator+(const MacroValue &mv) const throw (TypeError)
{
const ArrayMV<T> *mv2 = dynamic_cast<const ArrayMV<T> *>(&mv);
if (mv2 == NULL)
throw TypeError("Type mismatch for operands of + operator");
else
{
vector<T> values_copy(values);
values_copy.insert(values_copy.end(),
mv2->values.begin(),
mv2->values.end());
return new ArrayMV<T>(values_copy);
}
}
template<typename T>
MacroValue *
ArrayMV<T>::operator==(const MacroValue &mv) const throw (TypeError)
{
const ArrayMV<T> *mv2 = dynamic_cast<const ArrayMV<T> *>(&mv);
if (mv2 == NULL)
throw new IntMV(0);
else
return new IntMV(values == mv2->values);
}
template<typename T>
MacroValue *
ArrayMV<T>::operator!=(const MacroValue &mv) const throw (TypeError)
{
const ArrayMV<T> *mv2 = dynamic_cast<const ArrayMV<T> *>(&mv);
if (mv2 == NULL)
throw new IntMV(1);
else
return new IntMV(values != mv2->values);
}
template<typename T>
MacroValue *
ArrayMV<T>::operator[](const MacroValue &mv) const throw (TypeError)
{
const ArrayMV<int> *mv2 = dynamic_cast<const ArrayMV<int> *>(&mv);
if (mv2 == NULL)
throw TypeError("Expression inside [] must be an integer array");
vector<T> result;
for(vector<int>::const_iterator it = mv2->values.begin();
it != mv2->values.end(); it++)
result.push_back(values[*it - 1]);
return new ArrayMV<T>(result);
}
template<typename T>
string
ArrayMV<T>::toString() const
{
ostringstream ss;
for(typename vector<T>::const_iterator it = values.begin();
it != values.end(); it++)
ss << *it;
return ss.str();
}
template<typename T>
MacroValue *
ArrayMV<T>::clone() const
{
return new ArrayMV<T>(values);
}
template<typename T>
MacroValue *
ArrayMV<T>::toArray() const
{
return clone();
}
template<typename T>
MacroValue *
ArrayMV<T>::append(const MacroValue &mv) const
{
throw TypeError("Cannot append an array at the end of another one. Should use concatenation.");
}
#endif #endif