#ifndef _VALUE_H #define _VALUE_H #include "amount.h" #include "balance.h" #include "debug.h" namespace ledger { // The following type is a polymorphous value type used solely for // performance reasons. The alternative is to compute value // expressions (valexpr.cc) in terms of the largest data type, // balance_t. This was found to be prohibitively expensive, especially // when large logic chains were involved, since many temporary // allocations would occur for every operator. With value_t, and the // fact that logic chains only need boolean values to continue, no // memory allocations need to take place at all. class transaction_t; class value_t { public: char data[sizeof(balance_pair_t)]; enum type_t { BOOLEAN, INTEGER, AMOUNT, BALANCE, BALANCE_PAIR, ANY } type; value_t() { DEBUG_PRINT("ledger.memory.ctors", "ctor value_t"); *((unsigned int *) data) = 0; type = INTEGER; } value_t(const value_t& value) { DEBUG_PRINT("ledger.memory.ctors", "ctor value_t"); *this = value; } value_t(const bool value) { DEBUG_PRINT("ledger.memory.ctors", "ctor value_t"); *((bool *) data) = value; type = BOOLEAN; } value_t(const unsigned int value) { DEBUG_PRINT("ledger.memory.ctors", "ctor value_t"); *((unsigned int *) data) = value; type = INTEGER; } value_t(const amount_t& value) { DEBUG_PRINT("ledger.memory.ctors", "ctor value_t"); new((amount_t *)data) amount_t(value); type = AMOUNT; } value_t(const balance_t& value) { DEBUG_PRINT("ledger.memory.ctors", "ctor value_t"); new((balance_t *)data) balance_t(value); type = BALANCE; } value_t(const balance_pair_t& value) { DEBUG_PRINT("ledger.memory.ctors", "ctor value_t"); new((balance_pair_t *)data) balance_pair_t(value); type = BALANCE_PAIR; } ~value_t() { DEBUG_PRINT("ledger.memory.dtors", "dtor value_t"); destroy(); } void destroy(); value_t& operator=(const value_t& value); value_t& operator=(const bool value) { destroy(); *((bool *) data) = value; type = BOOLEAN; return *this; } value_t& operator=(const unsigned int value) { destroy(); *((unsigned int *) data) = value; type = INTEGER; return *this; } value_t& operator=(const amount_t& value) { destroy(); new((amount_t *)data) amount_t(value); type = AMOUNT; return *this; } value_t& operator=(const balance_t& value) { destroy(); new((balance_t *)data) balance_t(value); type = BALANCE; return *this; } value_t& operator=(const balance_pair_t& value) { destroy(); new((balance_pair_t *)data) balance_pair_t(value); type = BALANCE_PAIR; return *this; } value_t& operator+=(const value_t& value); value_t& operator-=(const value_t& value); value_t& operator*=(const value_t& value); value_t& operator/=(const value_t& value); value_t& operator+=(const transaction_t& xact); bool operator==(const value_t& value); bool operator!=(const value_t& value) { return ! (*this == value); } bool operator<(const value_t& value); bool operator<=(const value_t& value); bool operator>(const value_t& value); bool operator>=(const value_t& value); template operator T() const; void cast(type_t cast_type); void negate(); void abs(); value_t cost() const; }; template value_t::operator T() const { switch (type) { case BOOLEAN: return *((bool *) data); case INTEGER: return *((unsigned int *) data); case AMOUNT: return *((amount_t *) data); case BALANCE: return *((balance_t *) data); case BALANCE_PAIR: return *((balance_pair_t *) data); default: assert(0); break; } assert(0); return 0; } } // namespace ledger #endif // _VALUE_H