On May 31, 2014, at 3:20 PM, Mike Alexander <m...@umich.edu> wrote: > --On May 31, 2014 11:19:23 PM +0200 Christian Stimming > <christ...@cstimming.de> wrote: > >> But back to your initial question: You said we occasionally >> "encounter overflow errors". I don't understand (yet) what the >> actual problem is. With our current rational numbers and int64_t >> numerator we have approx. 19 decimal digits of precision (see [2] >> for the digits of a 64 bit signed integer), if I consider the >> numerator as fully used. >> >> Are 19 significant decimal digits not enough? Are there thinkable >> cases when they are not enough? I tend to think the problem is >> rather found in our rational number's rounding, which is not the >> suitable rounding method for our financial application domain. If >> this is the problem, a different data type that does the rounding >> always according to decimal numbers, and not according to (in normal >> float/double calculations) binary floating point numbers, or (in >> gnc_numeric) according to rational numbers with some potentially >> unknown denominator. >> > > Rounding along with division is at least part of the cause of overflows. I > dealt with a number of overflow problems in the advanced portfolio report > that were the result of the interaction of rounding and division. I solved > it by essentially converting things to a decimal notation. I picked a > (hopefully) sufficiently large decimal denominator and rounded relevant > calculations to that denominator. I think this is more or less what you're > suggesting in the context of the current implementation. > > Division of rational numbers, which is what GnuCash does, involves the > calculation of GCDs (or something similar) which can produce very large > integers. Years ago I was involved with the Reduce algebraic system which > implements arbitrary precision rational numbers mainly for this reason. It > needs to exactly divide rational numbers with no rounding so it can, for > example, factor polynomials. All but the most trivial calculations quickly > exceeded the range of hardware integers. We don't need that, of course. Our > problem seems to be that we're using rational numbers where it isn't really > necessary.
Yes, this is indeed it, and my original proposal was to use a multi-precision library wrapped in a rational number library, and to convert the results to fixed-point decimal with a lot of digits to the right of the decimal point, hoping that that would cover the rounding issue. So I’ll do a trial implementation and put it on my GitHub repo for everyone to test out. It looks like there are four FOSS libraries available to choose from: The afore-mentioned bytereef library, ICU’s decNumber [1], GCC’s libdecnumber, and GCC’s std::decimal in libstdc++. The last is an implementation of a TR for C++ [2], [3]. libdecnumber apparently requires enabling at compile time and neither it nor std::decimal work with Clang. I’ve already downloaded and built the bytereef library so I’ll go with that rather than ICU decNumber. There are a few more bugs on my list to take care of, so it’ll be next week some time. Regards, John Ralls [1] http://download.icu-project.org/files/decNumber/ [2] A Draft is freely available at http://www.open-std.org/JTC1/SC22/WG21/docs/papers/2009/n2849.pdf [3] A proposal to incorporate an improved version into C++17 was submitted earlier this year: https://isocpp.org/blog/2014/01/n3871 _______________________________________________ gnucash-devel mailing list gnucash-devel@gnucash.org https://lists.gnucash.org/mailman/listinfo/gnucash-devel