**************************************************************************
MAPM
Version 3.85
April 2, 2001
Michael C. Ring
ringx004@tc.umn.edu
Latest release will be available at
http://tc.umn.edu/~ringx004
**************************************************************************
* *
* Copyright (C) 1999 - 2001 Michael C. Ring *
* *
* This software is Freeware. *
* *
* Permission to use, copy, and distribute this software and its *
* documentation for any purpose with or without fee is hereby granted, *
* provided that the above copyright notice appear in all copies and *
* that both that copyright notice and this permission notice appear *
* in supporting documentation. *
* *
* Permission to modify the software is granted, but not the right to *
* distribute the modified code. Modifications are to be distributed *
* as patches to released version. *
* *
* This software is provided "as is" without express or implied warranty. *
* *
**************************************************************************
---------------------------------------
Mike's Arbitrary Precision Math Library
---------------------------------------
Mike's Arbitrary Precision Math Library is a set of functions that
allow the user to perform math to any level of accuracy that is
desired. The inspiration for this library was Lloyd Zusman's similar
APM package that was released in ~1988. I borrowed some of his ideas
in my implementation, creating a new data type (MAPM) and how the data
type was used by the user. However, there were a few things I wanted
my library to do that the original library did not :
1) Round a value to any desired precision. This is very handy when
multiplying for many iterations. Since multiplication guarantees an
exact result, the number of digits will grow without bound. I wanted
a way to trim the number of significant digits that were retained.
2) A natural support for floating point values. From most of the other
libraries I looked at, they seem to have a preference for integer
only type math manipulations. This library will also do integer only
math if you desire.
And if a library can only do integers, it can't do ...
3) Trig functions and other common C math library functions. This library
will perform the following functions to any desired precision level :
SQRT, CBRT, SIN, COS, TAN, ARC-SIN, ARC-COS, ARC-TAN, ARC-TAN2, LOG,
LOG10, EXP, POW, SINH, COSH, TANH, ARC-SINH, ARC-COSH, ARC-TANH, and
also FACTORIAL. The full 'math.h' is not duplicated, though I think
these are most of the important ones. My definition of what's important
is what I've actually used in a real application.
NOTE: MAPM Library History can now be found in 'history.txt'
(I really wasn't sure what to call this library. 'Arbitrary Precision Math'
is a defacto standard for what this does, but that name was already taken,
so I just put an 'M' in front of it ...)
**************************************************************************
KNOWN BUGS : None
**************************************************************************
IF YOU ARE IN A HURRY ...
UNIX: (assumes gcc compiler)
run make (build library + 3 executables)
--- OR ---
run: mklib (this will create the library, lib_mapm.a)
run: mkdemo (this will create 3 executables, 'calc', 'validate',
and 'primenum')
DOS / Win NT/9x (in a DOS window for NT/9x):
see the file 'README.DOS' for instructions.
**************************************************************************
calc: This is a command line version of an RPN calculator. If you are
familiar with RPN calculators, the use of this program will be
quite obvious. The default is 30 decimal places but this can be
changed with the '-d' command line option. This is not an
interactive program, it just computes an expression from the command
line. Run 'calc' with no arguments to get a list of the operators.
compute : (345.2 * 87.33) - (11.88 / 3.21E-2)
calc 345.2 87.33 x 11.88 3.21E-2 / -
result: [2.977622254205607476635514018692E+4]
compute PI to 70 decimal places : (6 * arcsin(0.5))
calc -d70 6 .5 as x
result :
3.1415926535897932384626433832795028841971693993751058209749445923078164E+0
validate : This program will compare the MAPM math functions to the C
standard library functions, like sqrt, sin, exp, etc. This
should give the user a good idea that the library is operating
correctly. The program will also compute some known quantities
to 70 digits of precision, like PI, log(2), etc. There was nothing
special about '70' other than that number of digits fit nicely
on one line.
primenum: This program will generate the first 10 prime numbers starting
with the number entered as an argument.
Example: primenum 1234567890 will output (actually 1 per line):
this took ~5 sec on my PC, a 350MHz PII.
1234567891, 1234567907, 1234567913, 1234567927, 1234567949,
1234567967, 1234567981, 1234568021, 1234568029, 1234568047
**************************************************************************
To use the library, simply include 'm_apm.h' and link your program
with the library, lib_mapm.a (unix) or lib_mapm.lib (dos).
For unix builds, you also may need to specify the math library (-lm) when
linking. The reason is some of the MAPM functions use an iterative algorithm.
When you use an iterative solution, you have to supply an initial guess. I
use the standard math library to generate this initial guess. I debated
whether this library should be stand-alone, i.e. generate it's own initial
guesses with some algorithm. In the end, I decided using the standard math
library would not be a big inconvienence and also it was just too tempting
having an immediate 15 digits of precision. When you prime the iterative
routines with 15 accurate digits, the MAPM functions converge faster.
See the file 'algorithms.used' to see a description of the algorithms I
used in the library. Some I derived on my own, others I borrowed from people
smarter than me. Version 2 of the library supports a 'fast' multiplication
algorithm. The algorithm used is described in the algorithms.used file. A
considerable amount of time went into finding fast algorithms for the
library. However, some could possibly be even better. If anyone has a
more efficient algorithm for any of these functions, I would like to here
from you.
See the file 'function.ref' to see a description of all the functions in
the library and the calling conventions, prototypes, etc.
See the file 'struct.ref' which documents how I store the numbers internally
in the MAPM data structure. This will not be needed for normal use, but it
will be very useful if you need to change/add to the existing library.
**************************************************************************
QUICK TEMPLATE FOR NORMAL USE :
The MAPM math is done on a new data type called "M_APM". This is
actually a pointer to a structure, but the contents of the structure
should never be manipulated: all operations on MAPM entities are done
through a functional interface.
The MAPM routines will automatically allocate enough space in their
results to hold the proper number of digits.
The caller must initialize all MAPM values before the routines can
operate on them (including the values intended to contain results of
calculations). Once this initialization is done, the user never needs
to worry about resizing the MAPM values, as this is handled inside the
MAPM routines and is totally invisible to the user.
The result of a MAPM operation cannot be one of the other MAPM operands.
If you want this to be the case, you must put the result into a
temporary variable and then assign (m_apm_copy) it to the appropriate operand.
All of the MAPM math functions begin with m_apm_*.
There are some predefined constants for your use. In case it's not obvious,
these should never appear as a 'result' parameter in a function call.
The following constants are available : (declared in m_apm.h)
MM_Zero MM_One MM_Two MM_Three
MM_Four MM_Five MM_Ten
MM_PI MM_HALF_PI MM_2_PI MM_E
MM_LOG_E_BASE_10 MM_LOG_10_BASE_E MM_LOG_2_BASE_E MM_LOG_3_BASE_E
The non-integer constants above (PI, log(2), etc) are accurate to 128
decimal places. The file mapmcnst.c contains these constants. I've
included 512 digit constants in this file also that are commented out.
If you need more than 512 digits, you can simply use the 'calc' program
to generate more precise constants (or create more precise constants at
run-time in your app). The number of significant digits in the constants
should be 6-8 more than the value specified in the #define.
Basic plan of attack:
(1) get your 'numbers' into M_APM format.
(2) do your high precision math.
(3) get the M_APM numbers back into a format you can use.
-------- (1) --------
#include "m_apm.h" /* must be included before any M_APM 'declares' */
M_APM area_mapm; /* declare your variables */
M_APM tmp_mapm;
M_APM array_mapm[10]; /* can use normal array notation */
M_APM array2d_mapm[10][10];
area_mapm = m_apm_init() /* init your variables */
tmp_mapm = m_apm_init();
for (i=0; i < M; i++) /* must init every element of the array */
array_mapm[i] = m_apm_init();
for (i=0; i < M; i++)
for (j=0; j < N; j++)
array2d_mapm[i][j] = m_apm_init();
/*
* there are 3 ways to convert your number into an M_APM
* (see the file function.ref)
*
* a) literal string (exponential notation OK)
* b) long variable
* c) double variable
*/
m_apm_set_string(tmp_mapm, "5.3286E-7");
m_apm_set_long(array_mapm[6], -872253L);
m_apm_set_double(array2d_mapm[3][7], -529.4486711);
-------- (2) --------
do your math ...
m_apm_add(cc_mapm, aa_mapm, MM_PI);
m_apm_divide(bb_mapm, DECIMAL_PLACES, aa_mapm, MM_LOG_2_BASE_E);
m_apm_sin(bb_mapm, DECIMAL_PLACES, aa_mapm);
whatever ...
/*
* There are 2 functions for converting an M_APM number into
* something useful. (SEE THE FILE 'function.ref')
*
* For both functions, M_APM number -> string is the conversion
*
* ===================
* ==== METHOD 1 ==== : Meant for use with floating point values
* ===================
*
* the format will be in scientific (exponential) notation
*
* output string = [-]n.nnnnnE+x or ...E-x
*
* where 'n' are digits and the exponent will be always be present,
* including E+0
*
* it's easy to convert this to a double:
*
* double dtmp = atof(out_buffer);
*
*
* ===================
* ==== METHOD 2 ==== : Meant for use with integer values
* ===================
*
* the format will simply be digits with a possible leading '-' sign.
*
* output string = [-]nnnnnn
*
* where 'n' are digits.
*
* it's easy to convert this to a long :
* long mtmp = atol(out_buffer);
*
* ... or an int :
* int itmp = atoi(out_buffer);
*
* Note that if the M_APM number has a fractional portion, the fraction
* will be truncated and only the integer portion will be output.
*/
-------- (3) --------
char out_buffer[256];
m_apm_to_string(out_buffer, DECIMAL_PLACES, mapm_number);
m_apm_to_integer_string(out_buffer, mapm_number);
**************************************************************************
MAPM C++ WRAPPER CLASS:
Orion Sky Lawlor (olawlor@acm.org) has added a very nice C++ wrapper
class to m_apm.h. This C++ class will have no effect if you just use
a normal C compiler. The library will operate as before with no user
impacts.
For now, I recommend compiling the library as 'C'. In order to compile
the library as C++, all the function declarations will need to be updated.
This may or may not happen in the near future. Since the C++ wrapper
class works very nicely as is, there is no pressing need to update the
entire library yet.
See the file 'cpp_function.ref' to see a description of how to use
the MAPM class.
To compile and link the C++ demo program: (assuming the library is
already built)
UNIX:
g++ cpp_demo.cpp lib_mapm.a -s -o cpp_demo -lm
GCC for DOS: (gxx is the C++ compiler)
gxx cpp_demo.cpp lib_mapm.a -s -o cpp_demo.exe -lm
Using the C++ wrapper allows you to do things like:
// Compute the factorial of the integer n
MAPM factorial(MAPM n)
{
MAPM i;
MAPM product = 1;
for (i=2; i <= n; i++)
product *= i;
return product;
}
The syntax is the same as if you were just writing normal code, but all
the computations will be performed with the high precision math library,
using the new 'datatype' MAPM.
The default precision of the computations is as follows:
Addition, subtraction, and multiplication will maintain ALL significant
digits.
All other operations (divide, sin, etc) will use the following rules:
1) if the operation uses only one input value [y = sin(x)], the result 'y'
will be the same precision as 'x', with a minimum of 30 digits if 'x' is
less than 30 digits.
2) if the operation uses two input values [z = atan2(y,x)], the result 'z'
will be the max digits of 'x' or 'y' with a minimum of 30.
The default precision is 30 digits. You can change the precision at
any time with the function 'm_apm_cpp_precision'. (See function.ref)
----> m_apm_cpp_precision(80);
will result in all operations being accurate to a minimum of 80 significant
digits. If any operand contains more than the minimum number of digits, then
the result will contain the max number of digits of the operands.
NOTE!: Some real life use with the C++ wrapper has revealed a certain
tendency for a program to become quite slow after many iterations
(like in a for/while loop). After a little debug, the reason
became clear. Remember that multiplication will maintain ALL
significant digits :
20 digit number x 20 digit number = 40 digits
40 digit number x 40 digit number = 80 digits
80 digit number x 80 digit number = 160 digits
etc.
So after numerous iterations, the number of significant digits
was growing without bound. The easy way to fix the problem is
to simply *round* your result after a multiply or some other
complex operation. For example:
#define MAX_DIGITS 256
p1 = (p0 * sin(b1) * exp(1 + u1)) / sqrt(1 + b1);
p1 = p1.round(MAX_DIGITS);
If you 'round' as shown here, your program will likely be
nearly as fast as a straight 'C' version.
If you have any questions or problems with the C++ wrapper, please let
me know. I am not very C++ proficient, but I'd still like to know about any
problems. Orion Sky Lawlor (olawlor@acm.org) is the one who implemented
the MAPM class, so he'll have to resolve any real hardcore problems, if you
have any.
**************************************************************************
TESTING :
Testing the library was interesting. How do I know it's right? Since I
test the library against the standard C runtime math library (see below)
I have high confidence that the MAPM library is giving correct results.
The logic here is the basic algorithms are independent of the number of
digits calculated, more digits just takes longer.
The MAPM library has been tested in the following environments :
Linux i486 / gcc 2.7.2.3
Linux i686 / gcc 2.91.66
FreeBSD 4.1 / gcc 2.95.1
HP-UX 9.x /gcc 2.5.8
HP-UX 10.x / gcc 2.95.2
Sun 5.5.1 (output from uname)
Sun Solaris 2.6 / gcc 2.95.1
DOS 5.0 using Microsoft C 5.1 and 8.00c (16 bit EXEs)
DOS 5.0 using GCC 2.8.1 for DOS
DOS 5.0 using GCC 2.95.2 for DOS
DOS ??? using Borland Turbo C++ 3.0
WIN NT+SP5 using Borland C++ 5.02 IDE, 5.2 & 5.5 command line.
WIN98 using Borland C++ 5.5 command line.
WIN98 & NT using LCC-WIN32 Ver 3.2
WIN98 & NT using Microsoft Visual C++ 6.0
As a general rule, when calculating a quantity to a given number of decimal
places, I calculated 4-6 extra digits and then rounded the result to what was
asked for. I decided to be conservative and give a correct answer rather than
to be faster and possibly have the last 2-3 digits in error. Also, some of
the functions call other functions (calculating arc-cos will call cos, log
will call exp, etc.) so I had to calculate a few extra digits in each iteration
to guarantee the loops terminated correctly.
1) I debugged the 4 basic math operations. I threw numerous test cases at
each of the operations until I knew they were correct.
Also note that the math.h type functions all call the 4 basic operations
numerous times. So if all the math.h functions work, it is highly
probable the 4 basic math operations work also.
2) 'math.h' type functions.
SQRT: Not real hard to check. Single stepping through the iterative
loop showed it was always converging to the sqrt.
CBRT: Similar to sqrt, single stepping through the iterative loop
showed it was always converging to the cube root.
EXP: I wrote a separate algorithm which expanded the Taylor series
manually and compared the results against the library.
POW: Straightforward since this just calls 'EXP'.
LOG: I wrote a separate algorithm which expanded the Taylor series
manually and compared the results against the library. This
took a long time to execute since the normal series converges
VERY slowly for the log function. This is why the LOG function
uses an iterative algorithm.
LOG10: Straightforward since this just calls 'LOG'.
SIN/COS: I wrote a separate algorithm which expanded the Taylor series
manually and compared the results against the library.
TAN: Straightforward since this just calls 'SIN' and 'COS'.
ARC-x: Single stepping through the iterative loop showed the arc
family of functions were always converging. Also used these
to compute PI. The value of PI is now known to many, many
digits. I computed PI to 1000+ digits by computing:
6 * arcsin(0.5) and 4 * arctan(1) and 3 * arccos(0.5)
and compared the output to the published 'real' values of PI.
The arc family of functions exercise considerable portions
of the library.
HYPERBOLIC: The hyperbolic functions just use exp, log, and the 4 basic
math operations. All of these functions simply use other
existing functions in the library.
FINALLY: Run the program 'validate'. This program compares the first
13-14 significant digits of the standard C library against
the MAPM library. If this program passes, you can feel
confident that the MAPM library is giving correct results.
This is because the basic algorithms do not change when
calculating more digits, it just takes longer.
**************************************************************************
|