mirror of https://github.com/KLayout/klayout.git
541 lines
14 KiB
C++
541 lines
14 KiB
C++
|
|
/*
|
|
|
|
KLayout Layout Viewer
|
|
Copyright (C) 2006-2017 Matthias Koefferlein
|
|
|
|
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.
|
|
|
|
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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef HDR_tlExpression
|
|
#define HDR_tlExpression
|
|
|
|
#include "tlCommon.h"
|
|
|
|
#include "tlException.h"
|
|
#include "tlVariant.h"
|
|
#include "tlString.h"
|
|
|
|
#include <map>
|
|
#include <vector>
|
|
#include <memory>
|
|
|
|
namespace tl
|
|
{
|
|
|
|
class Eval;
|
|
class EvalTarget;
|
|
class Expression;
|
|
class ExpressionNode;
|
|
class ExpressionParserContext;
|
|
|
|
/**
|
|
* @brief An interface handling the evaluation context
|
|
*
|
|
* This object serves to provide extended context for the expressions:
|
|
*
|
|
* First, this object is supposed to replace angle-bracket expressions
|
|
* of the kind <something> and <<something>> by a real value.
|
|
* The handler can be configured through Eval::set_ctx_handler.
|
|
*
|
|
* Second, this object provides the database unit value for physical unit conversions.
|
|
*/
|
|
class TL_PUBLIC ContextHandler
|
|
{
|
|
public:
|
|
/**
|
|
* @brief Constructor
|
|
*/
|
|
ContextHandler()
|
|
{
|
|
}
|
|
|
|
/**
|
|
* @brief Destructor
|
|
*/
|
|
virtual ~ContextHandler()
|
|
{
|
|
}
|
|
|
|
/**
|
|
* @brief Evaluates a single-bracket expression
|
|
*
|
|
* This method receives the content of a single bracket and is supposed to
|
|
* deliver an evaluated value.
|
|
*/
|
|
virtual tl::Variant eval_bracket (const std::string &content) const = 0;
|
|
|
|
/**
|
|
* @brief Evaluates a double-bracket expression
|
|
*
|
|
* This method receives the content of a double bracket and is supposed to
|
|
* deliver an evaluated value.
|
|
*/
|
|
virtual tl::Variant eval_double_bracket (const std::string &content) const = 0;
|
|
|
|
/**
|
|
* @brief Provide the database unit value
|
|
*/
|
|
virtual double dbu () const = 0;
|
|
};
|
|
|
|
/**
|
|
* @brief An exception thrown by the evaluation
|
|
*/
|
|
class TL_PUBLIC EvalError
|
|
: public tl::Exception
|
|
{
|
|
public:
|
|
EvalError (const std::string &what, const ExpressionParserContext &context);
|
|
};
|
|
|
|
/**
|
|
* @brief An exception indicating that no such method exists
|
|
*/
|
|
class TL_PUBLIC NoMethodError
|
|
: public EvalError
|
|
{
|
|
public:
|
|
NoMethodError (const std::string &cls_name, const std::string &method, const ExpressionParserContext &context);
|
|
};
|
|
|
|
/**
|
|
* @brief The expression parser context
|
|
*/
|
|
class TL_PUBLIC ExpressionParserContext
|
|
: public tl::Extractor
|
|
{
|
|
public:
|
|
/**
|
|
* @brief Default constructor
|
|
*/
|
|
ExpressionParserContext ();
|
|
|
|
/**
|
|
* @brief Constructor
|
|
*
|
|
* @param expr The expression to which this context refers to
|
|
* @param ex The initial location of the parser
|
|
*/
|
|
ExpressionParserContext (const Expression *expr, const tl::Extractor &ex);
|
|
|
|
/**
|
|
* @brief Reimplementation of tl::Extractor's error method
|
|
*/
|
|
virtual void error (const std::string &message);
|
|
|
|
/**
|
|
* @brief Gets a string indication where we are currently
|
|
*/
|
|
std::string where () const;
|
|
|
|
/**
|
|
* @brief Sets the expression parent
|
|
*/
|
|
void set_expr (const Expression *expr)
|
|
{
|
|
mp_expr = expr;
|
|
}
|
|
|
|
private:
|
|
const Expression *mp_expr;
|
|
tl::Extractor m_ex0;
|
|
};
|
|
|
|
/**
|
|
* @brief A node within an expression tree
|
|
*/
|
|
class TL_PUBLIC ExpressionNode
|
|
{
|
|
public:
|
|
/**
|
|
* @brief Constructor
|
|
*/
|
|
ExpressionNode (const ExpressionParserContext &context);
|
|
|
|
/**
|
|
* @brief Constructor with reservation of a certain number of child nodes
|
|
*/
|
|
ExpressionNode (const ExpressionParserContext &context, size_t children);
|
|
|
|
/**
|
|
* @brief Copy ctor
|
|
*/
|
|
ExpressionNode (const ExpressionNode &other, const tl::Expression *expr);
|
|
|
|
/**
|
|
* @brief Destructor
|
|
*/
|
|
virtual ~ExpressionNode ();
|
|
|
|
/**
|
|
* @brief Add a child node
|
|
*/
|
|
void add_child (ExpressionNode *node);
|
|
|
|
/**
|
|
* @brief Execute the node
|
|
*/
|
|
virtual void execute (EvalTarget &out) const = 0;
|
|
|
|
/**
|
|
* @brief Clone the node
|
|
*/
|
|
virtual ExpressionNode *clone (const tl::Expression *expr) const = 0;
|
|
|
|
protected:
|
|
std::vector <ExpressionNode *> m_c;
|
|
ExpressionParserContext m_context;
|
|
|
|
/**
|
|
* @brief Sets the expression parent
|
|
*/
|
|
void set_expr (const tl::Expression *expr)
|
|
{
|
|
m_context.set_expr (expr);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @brief A class handler for user objects within tl::Variant
|
|
*
|
|
* In order to enable objects for expressions, the user object in tl::Variant must be provided with
|
|
* a class derived from tl::VariantUserClassBase which implements eval_cls to return an EvalClass
|
|
* implementation which executes the method.
|
|
*/
|
|
class TL_PUBLIC EvalClass
|
|
{
|
|
public:
|
|
/**
|
|
* @brief Constructor
|
|
*
|
|
* @param test_function_name The name of the function which will be created and which tests if the variant is of the given type.
|
|
*/
|
|
EvalClass () { }
|
|
|
|
/**
|
|
* @brief Destructor
|
|
*/
|
|
virtual ~EvalClass () { }
|
|
|
|
/**
|
|
* @brief Execute the method with the given name on the object
|
|
*
|
|
* @param node The current location in the syntax tree
|
|
* @param out The return value
|
|
* @param object The object on which to execute the method
|
|
* @param method The name of the method
|
|
* @param args The arguments of the method
|
|
*
|
|
* If no method of this kind exists, the implementation may throw a NoMethodError.
|
|
*/
|
|
virtual void execute (const ExpressionParserContext &context, tl::Variant &out, tl::Variant &object, const std::string &method, const std::vector<tl::Variant> &args) const = 0;
|
|
};
|
|
|
|
/**
|
|
* @brief A base class for a function
|
|
*/
|
|
class TL_PUBLIC EvalFunction
|
|
{
|
|
public:
|
|
/**
|
|
* @brief Constructor
|
|
*/
|
|
EvalFunction () { }
|
|
|
|
/**
|
|
* @brief Destructor
|
|
*/
|
|
virtual ~EvalFunction () { }
|
|
|
|
/**
|
|
* @brief The actual execution method
|
|
*
|
|
* @param ex The position inside the current expression
|
|
* @param args The arguments of the method
|
|
* @return The return value
|
|
*/
|
|
virtual void execute (const ExpressionParserContext &context, tl::Variant &out, const std::vector<tl::Variant> &args) const = 0;
|
|
};
|
|
|
|
/**
|
|
* @brief Represents a expression to evaluate
|
|
*/
|
|
class TL_PUBLIC Expression
|
|
{
|
|
public:
|
|
/**
|
|
* @brief Default constructor
|
|
*/
|
|
Expression ();
|
|
|
|
/**
|
|
* @brief Copy constructor
|
|
*/
|
|
Expression (const Expression &d);
|
|
|
|
/**
|
|
* @brief Assignment
|
|
*/
|
|
Expression &operator= (const Expression &d);
|
|
|
|
/**
|
|
* @brief Execution of the expression
|
|
*/
|
|
tl::Variant execute () const;
|
|
|
|
/**
|
|
* @brief Execution of the expression (return by reference)
|
|
*/
|
|
void execute (EvalTarget &v) const;
|
|
|
|
/**
|
|
* @brief Gets the text of the expression
|
|
*/
|
|
const char *text () const
|
|
{
|
|
return mp_text != 0 ? mp_text : m_local_text.c_str ();
|
|
}
|
|
|
|
/**
|
|
* @brief Sets the local text of the expression
|
|
*/
|
|
void set_text (const std::string &s)
|
|
{
|
|
m_local_text = s;
|
|
}
|
|
|
|
/**
|
|
* @brief Sets the external text of the expression
|
|
*/
|
|
void set_text (const char *s)
|
|
{
|
|
mp_text = s;
|
|
}
|
|
|
|
private:
|
|
const char *mp_text;
|
|
std::string m_local_text;
|
|
std::auto_ptr<ExpressionNode> m_root;
|
|
Eval *mp_eval;
|
|
|
|
friend class Eval;
|
|
|
|
/**
|
|
* @brief Private constructor for Eval
|
|
*/
|
|
Expression (Eval *eval, const std::string &expr);
|
|
|
|
/**
|
|
* @brief Private constructor for Eval
|
|
*/
|
|
Expression (Eval *eval, const char *expr);
|
|
|
|
/**
|
|
* @brief Accessor to the root node
|
|
*/
|
|
std::auto_ptr<ExpressionNode> &root ()
|
|
{
|
|
return m_root;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @brief Provides the context for the expression parser and evaluation
|
|
*/
|
|
class TL_PUBLIC Eval
|
|
{
|
|
public:
|
|
/**
|
|
* @brief Create a new object for expression evaluation
|
|
*
|
|
* @param parent The parent evaluation context
|
|
* @param sloppy True to enable sloppy evaluation for pure parsing
|
|
*/
|
|
Eval (const Eval *parent = 0, bool sloppy = false);
|
|
|
|
/**
|
|
* @brief virtual dtor to enable dynamic_cast on derived classes.
|
|
*/
|
|
virtual ~Eval ();
|
|
|
|
/**
|
|
* @brief Sets an angle-bracket handler
|
|
*
|
|
* See \ContextHandler for details.
|
|
* This method will not take ownership over the object.
|
|
*/
|
|
void set_ctx_handler (const ContextHandler *ctx_handler)
|
|
{
|
|
mp_ctx_handler = ctx_handler;
|
|
}
|
|
|
|
/**
|
|
* @brief Gets the context handler
|
|
*
|
|
* If no handler is set locally, the parent context is looked up for one.
|
|
* If no context layout is present, 0 is returned.
|
|
*/
|
|
const ContextHandler *ctx_handler () const
|
|
{
|
|
if (mp_ctx_handler) {
|
|
return mp_ctx_handler;
|
|
} else if (mp_parent) {
|
|
return mp_parent->ctx_handler ();
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief Define a global function for use within an expression
|
|
*/
|
|
static void define_global_function (const std::string &name, EvalFunction *function)
|
|
{
|
|
m_global.define_function (name, function);
|
|
}
|
|
|
|
/**
|
|
* @brief Define a function for use within an expression
|
|
*/
|
|
void define_function (const std::string &name, EvalFunction *function);
|
|
|
|
/**
|
|
* @brief Define a global variable for use within an expression
|
|
*/
|
|
static void set_global_var (const std::string &name, const tl::Variant &var)
|
|
{
|
|
m_global.set_var (name, var);
|
|
}
|
|
|
|
/**
|
|
* @brief Define a variable for use within an expression
|
|
*/
|
|
void set_var (const std::string &name, const tl::Variant &var);
|
|
|
|
/**
|
|
* @brief Parse an expression from the extractor
|
|
*
|
|
* @param ex The extractor from which to parse the expression
|
|
* @param top If true, an expression is parsed at top level (as eval does). If false, the exression is parsed at 'atomic' level (as the string interpolation after the '$' does.
|
|
* @param expr An expression that can be evaluated (out)
|
|
*/
|
|
void parse (Expression &expr, tl::Extractor &ex, bool top = true);
|
|
|
|
/**
|
|
* @brief Convenience method that returns the expression object (caution: poor performance)
|
|
*
|
|
* @param ex The extractor from which to parse the expression
|
|
* @param top If true, an expression is parsed at top level (as eval does). If false, the exression is parsed at 'atomic' level (as the string interpolation after the '$' does.
|
|
*/
|
|
Expression parse (tl::Extractor &ex, bool top = true)
|
|
{
|
|
Expression expr;
|
|
parse (expr, ex, top);
|
|
return expr;
|
|
}
|
|
|
|
/**
|
|
* @brief Parse an expression from a string
|
|
*
|
|
* @param s The string from which to parse the expression
|
|
* @param top If true, an expression is parsed at top level (as eval does). If false, the exression is parsed at 'atomic' level (as the string interpolation after the '$' does.
|
|
* @return An expression string that can be passed to eval.
|
|
*/
|
|
void parse (Expression &expr, const std::string &s, bool top = true);
|
|
|
|
/**
|
|
* @brief Convenience method that returns the expression object (caution: poor performance)
|
|
*
|
|
* @param s The string from which to parse the expression
|
|
* @param top If true, an expression is parsed at top level (as eval does). If false, the exression is parsed at 'atomic' level (as the string interpolation after the '$' does.
|
|
*/
|
|
Expression parse (const std::string &s, bool top = true)
|
|
{
|
|
Expression expr;
|
|
parse (expr, s, top);
|
|
return expr;
|
|
}
|
|
|
|
/**
|
|
* @brief Parse an expression string from the extractor
|
|
*
|
|
* @param ex The extractor from which to parse the expression
|
|
* @param top If true, an expression is parsed at top level (as eval does). If false, the exression is parsed at 'atomic' level (as the string interpolation after the '$' does.
|
|
* @return An expression string that can be used later to construct an Expression from using the parse method
|
|
*/
|
|
static std::string parse_expr (tl::Extractor &ex, bool top = true);
|
|
|
|
/**
|
|
* @brief Interpolate the string and return the result
|
|
*
|
|
* Interpolation will replace all expressions of the form
|
|
* '$<atomic>' by their string value.
|
|
*/
|
|
std::string interpolate (const std::string &str);
|
|
|
|
/**
|
|
* @brief Provide access to the match substrings
|
|
*/
|
|
std::vector<std::string> &match_substrings ()
|
|
{
|
|
return m_match_substrings;
|
|
}
|
|
|
|
/**
|
|
* @brief Provide access to the match substrings (const version)
|
|
*/
|
|
const std::vector<std::string> &match_substrings () const
|
|
{
|
|
return m_match_substrings;
|
|
}
|
|
|
|
private:
|
|
friend class Expression;
|
|
|
|
const Eval *mp_parent;
|
|
std::map <std::string, tl::Variant> m_local_vars;
|
|
std::map <std::string, EvalFunction *> m_local_functions;
|
|
bool m_sloppy;
|
|
const ContextHandler *mp_ctx_handler;
|
|
std::vector<std::string> m_match_substrings;
|
|
|
|
tl::Variant eval (const std::string &expr);
|
|
void eval_top (ExpressionParserContext &context, std::auto_ptr<ExpressionNode> &v);
|
|
void eval_assign (ExpressionParserContext &context, std::auto_ptr<ExpressionNode> &v);
|
|
void eval_if (ExpressionParserContext &context, std::auto_ptr<ExpressionNode> &v);
|
|
void eval_boolean (ExpressionParserContext &context, std::auto_ptr<ExpressionNode> &v);
|
|
void eval_conditional (ExpressionParserContext &context, std::auto_ptr<ExpressionNode> &v);
|
|
void eval_shift (ExpressionParserContext &context, std::auto_ptr<ExpressionNode> &v);
|
|
void eval_addsub (ExpressionParserContext &context, std::auto_ptr<ExpressionNode> &v);
|
|
void eval_product (ExpressionParserContext &context, std::auto_ptr<ExpressionNode> &v);
|
|
void eval_bitwise (ExpressionParserContext &context, std::auto_ptr<ExpressionNode> &v);
|
|
void eval_unary (ExpressionParserContext &context, std::auto_ptr<ExpressionNode> &v);
|
|
void eval_atomic (ExpressionParserContext &context, std::auto_ptr<ExpressionNode> &v, int am);
|
|
void eval_suffix (ExpressionParserContext &context, std::auto_ptr<ExpressionNode> &v);
|
|
void resolve_name (const std::string &name, const EvalFunction *&function, const tl::Variant *&value) const;
|
|
void resolve_var_name (const std::string &name, tl::Variant *&value);
|
|
|
|
static Eval m_global;
|
|
};
|
|
|
|
}
|
|
|
|
#endif
|
|
|