From d213b32ffc6adaffa8e9ea4081b47c7b9403112c Mon Sep 17 00:00:00 2001 From: John Wiegley Date: Fri, 1 Aug 2008 02:20:09 -0400 Subject: Value expressions now auto-compile themselves on first use if the expr_t object is not constant. This saves classes that use expr_t from having to track such a detail themselves. --- expr.cc | 60 +++++++++++++++++++++++++++++++++++++----------------------- expr.h | 2 ++ 2 files changed, 39 insertions(+), 23 deletions(-) diff --git a/expr.cc b/expr.cc index 086f2494..9e961ca0 100644 --- a/expr.cc +++ b/expr.cc @@ -37,17 +37,19 @@ namespace ledger { std::auto_ptr expr_t::parser; -expr_t::expr_t() +expr_t::expr_t() : compiled(false) { TRACE_CTOR(expr_t, ""); } -expr_t::expr_t(const expr_t& other) : ptr(other.ptr), str(other.str) +expr_t::expr_t(const expr_t& other) + : ptr(other.ptr), str(other.str), compiled(false) { TRACE_CTOR(expr_t, "copy"); } -expr_t::expr_t(const string& _str, const unsigned int flags) : str(_str) +expr_t::expr_t(const string& _str, const unsigned int flags) + : str(_str), compiled(false) { TRACE_CTOR(expr_t, "const string&"); @@ -55,7 +57,8 @@ expr_t::expr_t(const string& _str, const unsigned int flags) : str(_str) ptr = parser->parse(str, flags); } - expr_t::expr_t(std::istream& in, const unsigned int flags) +expr_t::expr_t(std::istream& in, const unsigned int flags) + : compiled(false) { TRACE_CTOR(expr_t, "std::istream&"); @@ -63,7 +66,7 @@ expr_t::expr_t(const string& _str, const unsigned int flags) : str(_str) } expr_t::expr_t(const ptr_op_t& _ptr, const string& _str) - : ptr(_ptr), str(_str) + : ptr(_ptr), str(_str), compiled(false) { TRACE_CTOR(expr_t, "const ptr_op_t&, const string&"); } @@ -75,8 +78,11 @@ expr_t::~expr_t() throw() expr_t& expr_t::operator=(const expr_t& _expr) { - str = _expr.str; - ptr = _expr.ptr; + if (this != &_expr) { + str = _expr.str; + ptr = _expr.ptr; + compiled = _expr.compiled; + } return *this; } @@ -85,8 +91,9 @@ void expr_t::parse(const string& _str, const unsigned int flags) if (! parser.get()) throw_(parse_error, "Value expression parser not initialized"); - str = _str; - ptr = parser->parse(str, flags); + str = _str; + ptr = parser->parse(str, flags); + compiled = false; } void expr_t::parse(std::istream& in, const unsigned int flags) @@ -94,23 +101,34 @@ void expr_t::parse(std::istream& in, const unsigned int flags) if (! parser.get()) throw_(parse_error, "Value expression parser not initialized"); - str = ""; - ptr = parser->parse(in, flags); + str = ""; + ptr = parser->parse(in, flags); + compiled = false; } void expr_t::compile(scope_t& scope) { - if (ptr.get()) - ptr = ptr->compile(scope); + if (ptr.get()) { + ptr = ptr->compile(scope); + compiled = true; + } } -value_t expr_t::calc(scope_t& scope) const +value_t expr_t::calc(scope_t& scope) { - if (ptr.get()) + if (ptr.get()) { + if (! compiled) + compile(scope); return ptr->calc(scope); + } return NULL_VALUE; } +value_t expr_t::calc(scope_t& scope) const +{ + return ptr.get() ? ptr->calc(scope) : NULL_VALUE; +} + bool expr_t::is_constant() const { return ptr.get() && ptr->is_value(); @@ -141,26 +159,22 @@ void expr_t::print(std::ostream& out, scope_t& scope) const void expr_t::dump(std::ostream& out) const { - if (ptr) - ptr->dump(out, 0); + if (ptr) ptr->dump(out, 0); } void expr_t::read(std::ostream& in) { - if (ptr) - ptr->read(in); + if (ptr) ptr->read(in); } void expr_t::read(const char *& data) { - if (ptr) - ptr->read(data); + if (ptr) ptr->read(data); } void expr_t::write(std::ostream& out) const { - if (ptr) - ptr->write(out); + if (ptr) ptr->write(out); } void expr_t::initialize() diff --git a/expr.h b/expr.h index ab8c07f2..bfb372b6 100644 --- a/expr.h +++ b/expr.h @@ -59,6 +59,7 @@ public: private: ptr_op_t ptr; string str; + bool compiled; static void initialize(); static void shutdown(); @@ -97,6 +98,7 @@ public: void parse(std::istream& in, const unsigned int flags = 0); void compile(scope_t& scope); + value_t calc(scope_t& scope); value_t calc(scope_t& scope) const; bool is_constant() const; -- cgit v1.2.3