summaryrefslogtreecommitdiff
path: root/amount.cc
diff options
context:
space:
mode:
Diffstat (limited to 'amount.cc')
-rw-r--r--amount.cc76
1 files changed, 57 insertions, 19 deletions
diff --git a/amount.cc b/amount.cc
index 1ba5da7f..d8e5a85d 100644
--- a/amount.cc
+++ b/amount.cc
@@ -1,4 +1,5 @@
#include "ledger.h"
+#include "error.h"
#include "util.h"
#include "gmp.h"
@@ -7,8 +8,6 @@
#define MPZ(x) ((MP_INT *)(x))
-#define INIT() if (! quantity) _init()
-
namespace ledger {
static mpz_t full_divisor;
@@ -229,8 +228,14 @@ amount_t& amount_t::operator=(const double value)
amount_t& amount_t::operator+=(const amount_t& amt)
{
if (amt.quantity) {
- assert(! commodity || commodity == amt.commodity);
- INIT();
+ if (! quantity) {
+ _init();
+ commodity = amt.commodity;
+ }
+
+ if (commodity != amt.commodity)
+ throw amount_error("Adding amounts with different commodities");
+
mpz_add(MPZ(quantity), MPZ(quantity), MPZ(amt.quantity));
}
return *this;
@@ -239,8 +244,14 @@ amount_t& amount_t::operator+=(const amount_t& amt)
amount_t& amount_t::operator-=(const amount_t& amt)
{
if (amt.quantity) {
- assert(commodity == amt.commodity);
- INIT();
+ if (! quantity) {
+ _init();
+ commodity = amt.commodity;
+ }
+
+ if (commodity != amt.commodity)
+ throw amount_error("Subtracting amounts with different commodities");
+
mpz_sub(MPZ(quantity), MPZ(quantity), MPZ(amt.quantity));
}
return *this;
@@ -372,7 +383,10 @@ bool amount_t::operator<(const amount_t& amt) const
return amt > 0;
if (! amt.quantity) // equivalent to zero
return *this < 0;
- assert(commodity == amt.commodity);
+
+ if (commodity != amt.commodity)
+ throw amount_error("Comparing amounts with different commodities");
+
return mpz_cmp(MPZ(quantity), MPZ(amt.quantity)) < 0;
}
@@ -382,7 +396,10 @@ bool amount_t::operator<=(const amount_t& amt) const
return amt >= 0;
if (! amt.quantity) // equivalent to zero
return *this <= 0;
- assert(commodity == amt.commodity);
+
+ if (commodity != amt.commodity)
+ throw amount_error("Comparing amounts with different commodities");
+
return mpz_cmp(MPZ(quantity), MPZ(amt.quantity)) <= 0;
}
@@ -392,7 +409,10 @@ bool amount_t::operator>(const amount_t& amt) const
return amt < 0;
if (! amt.quantity) // equivalent to zero
return *this > 0;
- assert(commodity == amt.commodity);
+
+ if (commodity != amt.commodity)
+ throw amount_error("Comparing amounts with different commodities");
+
return mpz_cmp(MPZ(quantity), MPZ(amt.quantity)) > 0;
}
@@ -402,16 +422,23 @@ bool amount_t::operator>=(const amount_t& amt) const
return amt <= 0;
if (! amt.quantity) // equivalent to zero
return *this >= 0;
- assert(commodity == amt.commodity);
+
+ if (commodity != amt.commodity)
+ throw amount_error("Comparing amounts with different commodities");
+
return mpz_cmp(MPZ(quantity), MPZ(amt.quantity)) >= 0;
}
bool amount_t::operator==(const amount_t& amt) const
{
- if (commodity != amt.commodity)
+ if (! quantity && ! amt.quantity)
+ return true;
+ else if (! quantity || ! amt.quantity)
return false;
- assert(amt.quantity);
- assert(quantity);
+
+ if (commodity != amt.commodity)
+ throw amount_error("Comparing amounts with different commodities");
+
return mpz_cmp(MPZ(quantity), MPZ(amt.quantity)) == 0;
}
@@ -448,7 +475,10 @@ amount_t& amount_t::operator*=(const amount_t& amt)
if (! amt.quantity)
return *this;
- INIT();
+ if (! quantity) {
+ _init();
+ commodity = amt.commodity;
+ }
mpz_mul(MPZ(quantity), MPZ(quantity), MPZ(amt.quantity));
mpz_tdiv_q(MPZ(quantity), MPZ(quantity), full_divisor);
@@ -461,7 +491,10 @@ amount_t& amount_t::operator/=(const amount_t& amt)
if (! amt.quantity)
return *this;
- INIT();
+ if (! quantity) {
+ _init();
+ commodity = amt.commodity;
+ }
mpz_mul(MPZ(quantity), MPZ(quantity), full_divisor);
mpz_tdiv_q(MPZ(quantity), MPZ(quantity), MPZ(amt.quantity));
@@ -474,7 +507,10 @@ amount_t& amount_t::operator%=(const amount_t& amt)
if (! amt.quantity)
return *this;
- INIT();
+ if (! quantity) {
+ _init();
+ commodity = amt.commodity;
+ }
mpz_mul(MPZ(quantity), MPZ(quantity), full_divisor);
mpz_tdiv_r(MPZ(quantity), MPZ(quantity), MPZ(amt.quantity));
@@ -650,7 +686,7 @@ void parse_commodity(std::istream& in, std::string& symbol)
if (c == '"')
in.get(c);
else
- assert(0);
+ throw amount_error("Quoted commodity symbol lacks closing quote");
} else {
READ_INTO(in, buf, 256, c, ! std::isspace(c) && ! std::isdigit(c) &&
c != '-' && c != '.');
@@ -670,7 +706,8 @@ void amount_t::parse(std::istream& in)
unsigned int flags = COMMODITY_STYLE_DEFAULTS;;
unsigned int precision = MAX_PRECISION;
- INIT();
+ if (! quantity)
+ _init();
char c = peek_next_nonws(in);
if (std::isdigit(c) || c == '.' || c == '-') {
@@ -794,7 +831,8 @@ void amount_t::read_quantity(std::istream& in)
in.read((char *)&len, sizeof(len));
if (len) {
in.read(buf, len);
- INIT();
+ if (! quantity)
+ _init();
#ifdef WRITE_AMOUNTS_TEXTUALLY
buf[len] = '\0';
mpz_set_str(MPZ(quantity), buf, 10);