summaryrefslogtreecommitdiff
path: root/amount.cc
diff options
context:
space:
mode:
Diffstat (limited to 'amount.cc')
-rw-r--r--amount.cc90
1 files changed, 54 insertions, 36 deletions
diff --git a/amount.cc b/amount.cc
index cb1b3376..8915e91f 100644
--- a/amount.cc
+++ b/amount.cc
@@ -5,6 +5,7 @@
#include "debug.h"
#include <deque>
+#include <sstream>
#include "gmp.h"
@@ -43,21 +44,37 @@ static mpz_t temp;
static mpz_t divisor;
static mpz_t true_value;
-static struct init_amounts {
- init_amounts() {
- mpz_init(temp);
- mpz_init(divisor);
- mpz_init(true_value);
- mpz_set_ui(true_value, 1);
- }
-#ifdef DO_CLEANUP
- ~init_amounts() {
- mpz_clear(true_value);
- mpz_clear(divisor);
- mpz_clear(temp);
- }
-#endif
-} _init;
+commodity_t::updater_t * commodity_t::updater;
+commodities_map commodity_t::commodities;
+commodity_t * commodity_t::null_commodity;
+
+void initialize_amounts()
+{
+ mpz_init(temp);
+ mpz_init(divisor);
+ mpz_init(true_value);
+ mpz_set_ui(true_value, 1);
+
+ commodity_t::updater = NULL;
+ commodity_t::null_commodity = commodity_t::find_commodity("", true);
+}
+
+void shutdown_amounts()
+{
+ mpz_clear(true_value);
+ mpz_clear(divisor);
+ mpz_clear(temp);
+
+ if (commodity_t::updater)
+ delete commodity_t::updater;
+
+ for (commodities_map::iterator i = commodity_t::commodities.begin();
+ i != commodity_t::commodities.end();
+ i++)
+ delete (*i).second;
+
+ commodity_t::commodities.clear();
+}
static void mpz_round(mpz_t out, mpz_t value, int value_prec, int round_prec)
{
@@ -549,6 +566,12 @@ amount_t& amount_t::operator*=(const amount_t& amt)
mpz_mul(MPZ(quantity), MPZ(quantity), MPZ(amt.quantity));
quantity->prec += amt.quantity->prec;
+ if (quantity->prec > commodity->precision + 6U) {
+ mpz_round(MPZ(quantity), MPZ(quantity),
+ quantity->prec, commodity->precision + 6U);
+ quantity->prec = commodity->precision + 6U;
+ }
+
return *this;
}
@@ -568,6 +591,12 @@ amount_t& amount_t::operator/=(const amount_t& amt)
mpz_tdiv_q(MPZ(quantity), MPZ(quantity), MPZ(amt.quantity));
quantity->prec += 6;
+ if (quantity->prec > commodity->precision + 6U) {
+ mpz_round(MPZ(quantity), MPZ(quantity),
+ quantity->prec, commodity->precision + 6U);
+ quantity->prec = commodity->precision + 6U;
+ }
+
return *this;
}
@@ -712,6 +741,10 @@ std::ostream& operator<<(std::ostream& _out, const amount_t& amt)
mpz_clear(rquotient);
mpz_clear(remainder);
+ // Things are output to a string first, so that if anyone has
+ // specified a width or fill for _out, it will be applied to the
+ // entire amount string, and not just the first part.
+
_out << out.str();
return _out;
@@ -831,6 +864,12 @@ void amount_t::parse(std::istream& in)
delete[] buf;
}
+void amount_t::parse(const std::string& str)
+{
+ std::istringstream stream(str);
+ parse(stream);
+}
+
static char buf[4096];
static int index = 0;
@@ -912,27 +951,6 @@ void amount_t::read_quantity(std::istream& in)
}
-commodity_t::updater_t * commodity_t::updater = NULL;
-commodities_map commodity_t::commodities;
-commodity_t * commodity_t::null_commodity =
- commodity_t::find_commodity("", true);
-
-#ifdef DO_CLEANUP
-static struct cleanup_t
-{
- ~cleanup_t() {
- if (commodity_t::updater)
- delete commodity_t::updater;
-
- for (commodities_map::iterator i
- = commodity_t::commodities.begin();
- i != commodity_t::commodities.end();
- i++)
- delete (*i).second;
- }
-} _cleanup;
-#endif
-
commodity_t * commodity_t::find_commodity(const std::string& symbol,
bool auto_create)
{