1130 lines
45 KiB

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%%
%%% Version 12, 1999-09-07, Bruno
%%% -----------------------------
%%%
%%% * Habe neue Referenzen zu den Karatsuba-Arbeiten hinzugefügt, entnommen
%%% dem großartigen Paper CECM-98-118 (98_118-Borwein-Bradley-Crandall.dvi).
%%% * Habe in acm.bst, Funktion format.title, das change.case rausgenommen,
%%% das die Namen von Euler, Riemann u.a. in Kleinbuchstaben konvertiert
%%% hatte.
%%%
%%% Version 11, 1998-12-20, Bruno
%%% -----------------------------
%%%
%%% * Habe Referenzen zu den Karatsuba-Arbeiten hinzugefügt.
%%%
%%% Version 10, 1998-03-10, Thomas & Bruno
%%% --------------------------------------
%%%
%%% * Korrigiere die Formel für a(n) bei zeta(3).
%%% * Schönheitsfehler im Literaturverzeichnis.
%%%
%%% Version 9, 1998-03-06, Thomas & Bruno
%%% -------------------------------------
%%%
%%% * Schreibe \frac{1}{|x|} statt \frac{1}{x}.
%%%
%%% Version 8, 1998-01-16b, Thomas
%%% ------------------------------
%%%
%%% * Drei Literaturverweise für LiDIA statt nur einem.
%%%
%%% Version 7, 1998-01-16a, Bruno
%%% -----------------------------
%%%
%%% * Adresse: Praefix F fuer Frankreich
%%% * Abstract: Erwaehne zeta(3)
%%% * Kleinere Korrekturen der O()-Abschaetzungen
%%%
%%% Version 6, 1998-01-14, Thomas
%%% -----------------------------
%%%
%%% * habe meine Adresse ge"andert.
%%%
%%% * habe Resultat f"ur die Euler Konstante + Kettenbruch eingef"uhrt
%%%
%%% Version 5, 1997-12-11, Thomas
%%% -----------------------------
%%%
%%% * Habe die Anzahl der Scritte bei der Euler Konstante von
%%% x = ceiling(N log(2)/4)^2 auf x = ceiling((N+2) log(2)/4)^2
%%% hochgesetzt (Mail von Sofroniou)
%%%
%%% * Habe Kommentar eingef"ugt bzgl Checkpointing bei der Euler
%%% Konstante und Gamma(x) (Mail von Sofroniou)
%%%
%%% * Habe Section zu Geschwindigkeit von Maple gegen"uber Pari
%%% verbessert (mail von Laurent Bernardi).
%%%
%%% Version 4, 1997-01-09, Thomas
%%% -----------------------------
%%%
%%% * Habe die Komplexitätsaussage für sinh, cosh ergänzt.
%%% * Habe die Versionen der getesteten CAS ergänzt.
%%%
%%% Version 3, 1997-01-07, Bruno
%%% ----------------------------
%%%
%%% * Meine Firma schreibt sich mit vier Grossbuchstaben.
%%% * Apery schreibt sich m.W. mit einem Akzent.
%%% * Die Fehlermeldung meldet n2-n1>0, nicht n1-n2>0.
%%% * N -> \(N\) (zweimal)
%%% * Leerzeile entfernt nach Display-Formeln, bei denen der Absatz
%%% weitergeht. Hat den Effekt eines \noindent.
%%% * Im Abschnitt: arctan(x) for rational x: "another way" -> "the fastest way"
%%% * "[87]" -> "\cite{87}"
%%% * Das Cohen-Villegas-Zagier brauchen wir nun doch nicht zu zitieren.
%%% * Die "Note:" am Ende von Abschnitt ueber die Gamma-Funktion optisch
%%% verkleinert.
%%% * Die Formel fuer die hypergeometrische Funktionen optisch verschoenert.
%%% Andere Formeln ebenso.
%%% * Figure 2, erste Spalte rechtsbuendig.
%%% * "out performs" -> "outperforms"
%%% * "the streng" -> "the strength"
%%% * Hinweis auf die Parallelisierbarkeit im Abstract.
%%% * Bibtex-Style gehackt, damit nicht jeder zweite Autor auf seine
%%% Anfangsbuchstaben verkuerzt und alleinstehende Autoren ihres
%%% Vornamens beraubt werden.
%%%
%%% Version 2, 1997-01-06, Thomas
%%% -----------------------------
%%%
%%% * geänderte Abschnitte sind auskommentiert mit %%%. Alle
%%% Änderungen sind als Vorschlag zu verstehen. Der Grund
%%% wird im folgenden angegeben. Falls Du mit einer Änderung
%%% einverstanden bist, kannst Du alle %%%-Zeilen löschen.
%%%
%%% * Lyx defines wurden entfernt. Grund: das ISSAC-acmconf.sty
%%% erlaubt keine fremde macros. Übersetzung mit LaTeX geht.
%%% * habe Keyboardumlaute (ä,ü,ö,ß) in LaTeX umgeschrieben.
%%% Grund: damit die Submission in einem file geht.
%%% * Habe fontenc und psfig usepackage Befehle entfernt.
%%% Grund: fonts bestimmt acmconf.sty und keine Bilder vorhanden.
%%% * Habe bibliography mit BibTeX (binsplit.bib) und acm.bst
%%% erstellt. Grund: wird von ISSAC '97 verlangt.
%%% * Habe langen Formeln in einer eqnarray Umgebung gesteckt.
%%% Grund: acmconf.sty läuft im twocolumn-Modus. Lange Formeln
%%% haben die Ausgabe durcheinander gebracht.
%%% * Habe Reihenfolge bei der Beschreibung der elementare
%%% Funktionen geändert, sodaß zuerst die rationale und dann
%%% die reelle version beschrieben wird. Grund: Einheitlichkeit.
%%% * Habe sinh mit binary-splitting gegen sinh mit exp-Berechnung
%%% getestet. Sie sind ungefähr gleich gut auf meinen Pentium,
%%% mir machen "Wackler" beim cosh. cosh ist ab und zu, sowohl
%%% bei kleiner als auch bei große Präzision langsamer. Habe
%%% dies und dem Abschnitt sinh, cosh ausgefüllt. Grund: es hat
%%% gefehlt.
%%% * Habe artanh Abschnitt entfernt. Grund: ich habe in der
%%% Einleitung der elementaren Funktionen darauf verwiesen, daß
%%% man die Berechnung anderer Funktionen (wie artanh) auf die
%%% hier erwähnte zurückführen oder auf analoger Weise
%%% implementieren kann. Ich denke man braucht nicht alles explizit
%%% anzugeben.
%%%
%%% * Habe Dein Dankeschön an mich entfernt.
%%% * Habe Abschnitt über Konvergenzbeschleunigung entfernt.
%%% Grund: das geht in Dein MathComp paper.
%%%
%%% * Habe neue Formel für pi eingefügt. Grund: einfacher,
%%% effizienter und stimmt mit der angegebenen Referenz
%%% überein.
%%% * Habe die Berechnung der Apery Konstante angepasst.
%%% Grund: die hier angegebenen Formel wurde mit einer
%%% umgeformten Reihe berechnet. Wenn man dieses nicht
%%% kennt, wirkt es verwirrend. Keine Effizienz-steigerung.
%%% * Habe die Beschreibung für die erste version der Euler
%%% Konstante entfernt. Grund: wird von der zweiten version
%%% in jeder Hinsicht (Beweis, Effizienz) gedeckt.
%%% * Habe Abschnitte über Checkpointing und Parallelisierung
%%% eingefügt. Ein Beispiel über die Wirksamkeit habe ich
%%% bei der Apery Konstante angegeben. Grund: damit können wir
%%% das Paper auch bei PASCO '97 einreichen.
%%%
%%% * Habe Beispiel-C++-Implementierung für abpq Reihen eingefügt.
%%% Grund: zeigen wie einfach es ist wenn man die Formeln hat ;-)
%%% * Habe Beispiel-Implementierung für abpqcd Reihen eingefügt.
%%% Grund: dito
%%% * Habe Computational results und Conclusions Abschnitt eingefügt.
%%% * Habe die Namen der Konstanten (C, G, ...) and die entsprechenden
%%% Abschnitten eingefügt. Grund: diese Namen werden bei den
%%% Tabellen im Abschnitt Computational results benutzt.
%%% * Habe Verweis an LiDIA eingefügt. Grund: wird bei Computational
%%% results erw\"ahnt.
%%%
%%% Version 1, 1996-11-30, Bruno
%%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%% Bruno Haible, Thomas Papanikolaou. %%%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\documentstyle{acmconf}
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
% Plain TeX macros
%\catcode`@=11 % @ ist ab jetzt ein gewoehnlicher Buchstabe
%\def\@Re{\qopname@{Re}} \def\re#1{{\@Re #1}}
%\def\@Im{\qopname@{Im}} \def\im#1{{\@Im #1}}
%\catcode`@=12 % @ ist ab jetzt wieder ein Sonderzeichen
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
% LaTeX2e macros
\catcode`@=11 % @ ist ab jetzt ein gewoehnlicher Buchstabe
\def\re{\mathop{\operator@font Re}\nolimits}
\def\im{\mathop{\operator@font Im}\nolimits}
\def\artanh{\mathop{\operator@font artanh}\nolimits}
\catcode`@=12 % @ ist ab jetzt wieder ein Sonderzeichen
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\begin{document}
\title{Fast multiprecision evaluation of series of rational numbers}
\author{
\begin{tabular}{ccc}
{Bruno Haible} & \hspace*{2cm} & {Thomas Papanikolaou}\\
{\normalsize ILOG} && {\normalsize Laboratoire A2X}\\
{\normalsize 9, rue de Verdun} && {\normalsize 351, cours de la Lib\'eration}\\
{\normalsize F -- 94253 Gentilly Cedex} && {\normalsize F -- 33405 Talence Cedex}\\
{\normalsize {\tt haible@ilog.fr}} && {\normalsize {\tt papanik@math.u-bordeaux.fr}}\\
\end{tabular}
}
\maketitle
\begin{abstract}
We describe two techniques for fast multiple-precision evaluation of linearly
convergent series, including power series and Ramanujan series. The computation
time for \(N\) bits is \( O((\log N)^{2}M(N)) \), where \( M(N) \) is the time
needed to multiply two \(N\)-bit numbers. Applications include fast algorithms
for elementary functions, \(\pi\), hypergeometric functions at rational points,
$\zeta(3)$, Euler's, Catalan's and Ap{\'e}ry's constant. The algorithms are
suitable for parallel computation.
\end{abstract}
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\section{Introduction}
Multiple-precision evaluation of real numbers has become efficiently possible
since Sch\"onhage and Strassen \cite{71} have showed that the bit complexity of
the multiplication of two \(N\)-bit numbers is
\( M(N)=O(N\:\log N\:\log\log N) \).
This is not only a theoretical result; a C++ implementation \cite{96a} can
exploit this already for \( N=40000 \) bits. Algorithms for computing
elementary functions (exp, log, sin, cos, tan, asin, acos, atan, sinh, cosh,
tanh, arsinh, arcosh, artanh) have appeared in \cite{76b}, and a remarkable
algorithm for \( \pi \) was found by Brent and Salamin \cite{76c}.
However, all these algorithms suffer from the fact that calculated results
are not reusable, since the computation is done using real arithmetic (using
exact rational arithmetic would be extremely inefficient). Therefore functions
or constants have to be recomputed from the scratch every time higher precision
is required.
In this note, we present algorithms for fast computation of sums of the form
\[S=\sum _{n=0}^{\infty }R(n)F(0)\cdots F(n)\]
where \( R(n) \) and \( F(n) \) are rational functions in \( n \) with rational
coefficients, provided that this sum is linearly convergent, i.e. that the
\( n \)-th term is \( O(c^{-n}) \) with \( c>1 \). Examples include elementary
and hypergeometric functions at rational points in the {\em interior} of the
circle of convergence, as well as \( \pi \) and Euler's, Catalan's and
Ap{\'e}ry's constants.
The presented algorithms are {\em easy to implement} and {\em extremely
efficient}, since they take advantage of pure integer arithmetic. The
calculated results are {\em exact}, making {\em checkpointing} and
{\em reuse} of computations possible. Finally,
the computation of our algorithms {\em can be easily parallelised}.
After publishing the present paper, we were informed that the results of
section 2 were already published by E.~Karatsuba in \cite{91,91b,93,95c}.
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\section{Evaluation of linearly convergent series}
The technique presented here applies to all linearly convergent sums of the
form
\[ S=\sum ^{\infty }_{n=0}
\frac{a(n)}{b(n)}\frac{p(0)\cdots p(n)}{q(0)\cdots q(n)}\]
where \( a(n) \), \( b(n) \), \( p(n) \), \( q(n) \) are integers with
\( O(\log n) \) bits. The most often used case is that \( a(n) \), \( b(n) \),
\( p(n) \), \( q(n) \) are polynomials in \( n \) with integer coefficients.
\begin{description}
\item [Algorithm:]~
\end{description}
Given two index bounds \( n_{1} \) and \( n_{2} \), consider the partial sum
\[
S=\sum _{n_{1}\leq n<n_{2}}
\frac{a(n)}{b(n)}\frac{p(n_{1})\cdots p(n)}{q(n_{1})\cdots q(n)}\]
It is not computed directly. Instead, we compute the integers
\( P={p(n_{1})}\cdots {p(n_{2}-1)} \), \( Q={q(n_{1})}\cdots {q(n_{2}-1)} \),
\( B={b(n_{1})}\cdots {b(n_{2}-1)} \) and \( T=BQS \). If \( n_{2}-n_{1}<5 \),
these are computed directly. If \( n_{2}-n_{1}\geq 5 \), they are computed
using {\em binary splitting}: Choose an index \( n_{m} \) in the middle of
\( n_{1} \) and \( n_{2} \), compute the components \( P_{l} \), \( Q_{l} \),
\( B_{l} \), \( T_{l} \) belonging to the interval \( n_{1}\leq n<n_{m} \),
compute the components \( P_{r} \), \( Q_{r} \), \( B_{r} \), \( T_{r} \)
belonging to the interval \( n_{m}\leq n<n_{2} \), and set
\( P=P_{l}P_{r} \), \( Q=Q_{l}Q_{r} \), \( B=B_{l}B_{r} \) and
\( T=B_{r}Q_{r}T_{l}+B_{l}P_{l}T_{r} \).
Finally, this algorithm is applied to \( n_{1}=0 \) and
\( n_{2}=n_{\max }=O(N) \), and a final floating-point division
\( S=\frac{T}{BQ} \) is performed.
\begin{description}
\item [Complexity:]~
\end{description}
The bit complexity of computing \( S \) with \( N \) bits of precision is
\( O((\log N)^{2}M(N)) \).
\begin{description}
\item [Proof:]~
\end{description}
Since we have assumed the series to be linearly convergent, the \( n \)-th
term is \( O(c^{-n}) \) with \( c>1 \). Hence choosing
\( n_{\max }=N\frac{\log 2}{\log c}+O(1) \) will ensure that the round-off
error is \( <2^{-N} \). By our assumption that \( a(n) \), \( b(n) \),
\( p(n) \), \( q(n) \) are integers with \( O(\log n) \) bits, the integers
\( P \), \( Q \), \( B \), \( T \) belonging to the interval
\( n_{1}\leq n<n_{2} \) all have \( O((n_{2}-n_{1})\log n_{2}) \) bits.
The algorithm's recursion depth is \( d=\frac{\log n_{\max }}{\log 2}+O(1) \).
At recursion depth \( k \) (\( 1\leq k\leq d \)), integers having each
\( O(\frac{n_{\max }}{2^{k}}\log n_{\max }) \) bits are multiplied. Thus,
the entire computation time \( t \) is
\begin{eqnarray*}
t &=& \sum ^{d}_{k=1}
2^{k-1}O\left( M\left( \frac{n_{\max }}{2^{k}}\log n_{\max }\right)\right)\\
&=& \sum ^{d}_{k=1} O\left( M\left( n_{\max }\log n_{\max }\right) \right)\\
&=& O(\log n_{\max }M(n_{\max }\log n_{\max }))
\end{eqnarray*}
Because of \( n_{\max }=O( \frac{N}{\log c}) \) and
\begin{eqnarray*}
M\left(\frac{N}{\log c} \log \frac{N}{\log c}\right)
&=& O\left(\frac{1}{\log c} N\, (\log N)^{2}\, \log \log N\right)\\
&=& O\left(\frac{1}{\log c} \log N\, M(N)\right)
\end{eqnarray*}
we have
\[ t = O\left(\frac{1}{\log c} (\log N)^{2}M(N)\right) \]
Considering \(c\) as constant, this is the desired result.
\begin{description}
\item [Checkpointing/Parallelising:]~
\end{description}
A checkpoint can be easily done by storing the (integer) values of
\( n_1 \), \( n_2 \), \( P \), \( Q \), \( B \) and \( T \).
Similarly, if \( m \) processors are available, then the interval
\( [0,n_{max}] \) can be divided into \( m \) pieces of length
\( l = \lfloor n_{max}/m \rfloor \). After each processor \( i \) has
computed the sum of its interval \( [il,(i+1)l] \), the partial sums are
combined to the final result using the rules described above.
\begin{description}
\item [Note:]~
\end{description}
For the special case \( a(n)=b(n)=1 \), the binary splitting algorithm has
already been documented in \cite{76a}, section 6, and \cite{87}, section 10.2.3.
Explicit computation of \( P \), \( Q \), \( B \), \( T \) is only required
as a recursion base, for \( n_{2}-n_{1}<2 \), but avoiding recursions for
\( n_{2}-n_{1}<5 \) gains some percent of execution speed.
The binary splitting algorithm is asymptotically faster than step-by-step
evaluation of the sum -- which has binary complexity \( O(N^{2}) \) -- because
it pushes as much multiplication work as possible to the region where
multiplication becomes efficient. If the multiplication were implemented
as an \( M(N)=O(N^{2}) \) algorithm, the binary splitting algorithm would
provide no speedup over step-by-step evaluation.
\begin{description}
\item [Implementation:]~
\end{description}
In the following we present a simplified C++ implementation of
the above algorithm\footnote{A complete implementation can be found in
CLN \cite{96a}. The implementation of the binary-splitting method will
be also available in {\sf LiDIA-1.4}}. The initialisation is done by a
structure {\tt abpq\_series} containing arrays {\tt a}, {\tt b}, {\tt p}
and {\tt q} of multiprecision integers ({\tt bigint}s). The values of
the arrays at the index \( n \) correspond to the values of the functions
\( a \), \( b \), \( p \) and \( q \) at the integer point \( n \).
The (partial) results of the algorithm are stored in the
{\tt abpq\_series\_result} structure.
\begin{verbatim}
// abpq_series is initialised by user
struct { bigint *a, *b, *p, *q;
} abpq_series;
// abpq_series_result holds the partial results
struct { bigint P, Q, B, T;
} abpq_series_result;
// binary splitting summation for abpq_series
void sum_abpq(abpq_series_result & r,
int n1, int n2,
const abpq_series & arg)
{
// check the length of the summation interval
switch (n2 - n1)
{
case 0:
error_handler("summation device",
"sum_abpq:: n2-n1 should be > 0.");
break;
case 1: // the result at the point n1
r.P = arg.p[n1];
r.Q = arg.q[n1];
r.B = arg.b[n1];
r.T = arg.a[n1] * arg.p[n1];
break;
// cases 2, 3, 4 left out for simplicity
default: // the general case
// the left and the right partial sum
abpq_series_result L, R;
// find the middle of the interval
int nm = (n1 + n2) / 2;
// sum left side
sum_abpq(L, n1, nm, arg);
// sum right side
sum_abpq(R, nm, n2, arg);
// put together
r.P = L.P * R.P;
r.Q = L.Q * R.Q;
r.B = L.B * R.B;
r.T = R.B * R.Q * L.T + L.B * L.P * R.T;
break;
}
}
\end{verbatim}
Note that the multiprecision integers could be replaced here by integer
polynomials, or by any other ring providing the operators \( = \) (assignment),
\( + \) (addition) and \( * \) (multiplication). For example, one could regard
a bivariate polynomial over the integers as a series over the second variable,
with polynomials over the first variable as its coefficients. This would result
an accelerated algorithm for summing bivariate (and thus multivariate)
polynomials.
\subsection{Example: The factorial}
This is the most classical example of the binary splitting algorithm and was
probably known long before \cite{87}.
Computation of the factorial is best done using the binary splitting algorithm,
combined with a reduction of the even factors into odd factors and
multiplication with a power of 2, according to the formula
\[
n!=2^{n-\sigma _{2}(n)}\cdot \prod _{k\geq 1}
\left( \prod _{\frac{n}{2^{k}}<2m+1\leq \frac{n}{2^{k-1}}}(2m+1)\right) ^{k}\]
and where the products
\[
P(n_{1},n_{2})=\prod _{n_{1}<m\leq n_{2}}(2m+1)\]
are evaluated according to the binary splitting algorithm:
\( P(n_{1},n_{2})=P(n_{1},n_{m})P(n_{m},n_{2}) \) with
\( n_{m}=\left\lfloor \frac{n_{1}+n_{2}}{2}\right\rfloor \)
if \( n_{2}-n_{1}\geq 5 \).
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsection{Example: Elementary functions at rational points}
The binary splitting algorithm can be applied to the fast computation of the
elementary functions at rational points \( x=\frac{u}{v} \), simply
by using the power series. We present how this can be done for
\( \exp (x) \), \( \ln (x) \), \( \sin (x) \), \( \cos (x) \),
\( \arctan (x) \), \( \sinh (x) \) and \( \cosh (x) \). The calculation
of other elementary functions is similar (or it can be reduced to the
calculation of these functions).
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsubsection{\( \exp (x) \) for rational \( x \)}
This is a direct application of the above algorithm with \( a(n)=1 \),
\( b(n)=1 \), \( p(0)=q(0)=1 \), and \( p(n)=u \), \( q(n)=nv \) for \( n>0 \).
Because the series is not only linearly convergent -- \( \exp (x) \) is an
entire function --, \( n_{\max }=O(\frac{N}{\log N + \log \frac{1}{|x|}}) \),
hence the bit complexity is
\[ O\left(\frac{(\log N)^2}{\log N + \log \frac{1}{|x|}} M(N)\right) \]
Considering \(x\) as constant, this is \( O(\log N\: M(N)) \).
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsubsection{\( \exp (x) \) for real \( x \)}
This can be computed using the addition theorem for exp, by a trick due to
Brent \cite{76a} (see also \cite{87}, section 10.2, exercise 8). Write
\[
x=x_{0}+\sum _{k=0}^{\infty }\frac{u_{k}}{v_{k}}\]
with \( x_{0} \) integer, \( v_{k}=2^{2^{k}} \) and
\( |u_{k}|<2^{2^{k-1}} \), and compute
\[
\exp (x)=
\exp (x_{0})\cdot \prod _{k\geq 0}\exp \left( \frac{u_{k}}{v_{k}}\right) \]
This algorithm has bit complexity
\[ O\left(\sum\limits_{k=0}^{O(\log N)} \frac{(\log N)^2}{\log N + 2^k} M(N)\right)
= O((\log N)^{2}M(N)) \]
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsubsection{ \( \ln (x) \) for rational \( x \)}
For rational \( |x-1|<1 \), the binary splitting algorithm can also be applied
directly to the power series for \( \ln (x) \). Write \( x-1=\frac{u}{v} \)
and compute the series with \( a(n)=1 \), \( b(n)=n+1 \), \( q(n)=v \),
\( p(0)=u \), and \( p(n)=-u \) for \( n>0 \).
This algorithm has bit complexity \( O((\log N)^{2}M(N)) \).
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsubsection{\( \ln (x) \) for real \( x \)}
This can be computed using the ``inverse'' Brent trick:
Start with \( y:=0 \).
As long as \( x\neq 1 \) within the actual precision, choose \( k \)
maximal with \( |x-1|<2^{-k} \). Put \( z=2^{-2k}\left[ 2^{2k}(x-1)\right] \),
i.e. let \( z \) contain the first \( k \) significant bits of \( x-1 \).
\( z \) is a good approximation for \( \ln (x) \). Set \( y:=y+z \) and
\( x:=x\cdot \exp (-z) \).
Since \( x\cdot \exp (y) \) is an invariant of the algorithm, the final
\( y \) is the desired value \( \ln (x) \).
This algorithm has bit complexity
\[ O\left(\sum\limits_{k=0}^{O(\log N)} \frac{(\log N)^2}{\log N + 2^k} M(N)\right)
= O((\log N)^{2}M(N)) \]
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsubsection{ \( \sin (x) \), \( \cos (x) \) for rational \( x \)}
These are direct applications of the binary splitting algorithm: For
\( \sin (x) \), put \( a(n)=1 \), \( b(n)=1 \), \( p(0)=u \),
\( q(0)=v \), and \( p(n)=-u^{2} \), \( q(n)=(2n)(2n+1)v^{2} \) for
\( n>0 \). For \( \cos (x) \), put \( a(n)=1 \), \( b(n)=1 \),
\( p(0)=1 \), \( q(0)=1 \), and \( p(n)=-u^{2} \), \( q(n)=(2n-1)(2n)v^{2} \)
for \( n>0 \). Of course, when both \( \sin (x) \) and \( \cos (x) \) are
needed, one should only compute
\( \sin (x) \) this way, and then set
\( \cos (x)=\pm \sqrt{1-\sin (x)^{2}} \). This is a 20\% speedup at least.
The bit complexity of these algorithms is \( O(\log N\: M(N)) \).
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsubsection{ \( \sin (x) \), \( \cos (x) \) for real \( x \)}
To compute \( \cos (x)+i\sin (x)=\exp (ix) \) for real \( x \), again the
addition theorems and Brent's trick
can be used. The resulting algorithm has bit complexity
\( O((\log N)^{2}M(N)) \).
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsubsection{ \( \arctan (x) \) for rational \( x \)}
For rational \( |x|<1 \), the fastest way to compute \( \arctan (x) \) with
bit complexity \( O((\log N)^{2}M(N)) \) is
to apply the binary splitting algorithm directly to the power series
for \( \arctan (x) \). Put \( a(n)=1 \), \( b(n)=2n+1 \), \( q(n)=1 \),
\( p(0)=x \) and \( p(n)=-x^{2} \) for \( n>0 \).
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsubsection{ \( \arctan (x) \) for real \( x \)}
This again can be computed using the ``inverse'' Brent trick:
Start out with \( z:=\frac{1}{\sqrt{1+x^{2}}}+i\frac{x}{\sqrt{1+x^{2}}} \)
and \( \varphi :=0 \). During the algorithm \( z \) will be a complex number
with \( |z|=1 \) and \( \re (z)>0 \).
As long as \( \im (z)\neq 0 \) within the actual precision, choose \( k \)
maximal with \( |\im (z)|<2^{-k} \).
Put \( \alpha =2^{-2k}\left[ 2^{2k}\im (z)\right] \), i.e. let \( \alpha \)
contain the first \( k \) significant bits of \( \im (z) \). \( \alpha \)
is a good approximation for \( \arcsin (\im (z)) \). Set
\( \varphi :=\varphi +\alpha \) and \( z:=z\cdot \exp (-i\alpha ) \).
Since \( z\cdot \exp (i\varphi ) \) is an invariant of the algorithm, the
final \( \varphi \) is the desired
value \( \arcsin \frac{x}{\sqrt{1+x^{2}}} \).
This algorithm has bit complexity
\[ O\left(\sum\limits_{k=0}^{O(\log N)} \frac{(\log N)^2}{\log N + 2^k} M(N)\right)
= O((\log N)^{2}M(N)) \]
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsubsection{\( \sinh (x) \), \( \cosh (x) \) for rational and real \( x \)}
These can be computed by similar algorithms as \( \sin (x) \) and
\( \cos (x) \) above, with the same asymptotic bit complexity. The
standard computation, using \( \exp (x) \) and its reciprocal (calculated
by the Newton method) results also to the same complexity and works equally
well in practice.
The bit complexity of these algorithms is \( O(\log N\: M(N)) \) for rational
\( x \) and \( O((\log N)^{2}M(N)) \) for real \( x \).
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsection{Example: Hypergeometric functions at rational points}
The binary splitting algorithm is well suited for the evaluation of a
hypergeometric series
\[
F\left( \begin{array}{ccc}
a_{1}, & \ldots , & a_{r}\\
b_{1}, & \ldots , & b_{s}
\end{array}
\big| x\right) =\sum ^{\infty }_{n=0}
\frac{a_{1}^{\overline{n}}\cdots
a_{r}^{\overline{n}}}{b_{1}^{\overline{n}}\cdots b_{s}^{\overline{n}}}x^{n}\]
with rational coefficients \( a_{1} \), ..., \( a_{r} \), \( b_{1} \),
..., \( b_{s} \) at a rational point \( x \) in the interior of the circle of
convergence. Just put \( a(n)=1 \), \( b(n)=1 \), \( p(0)=q(0)=1 \), and
\( \frac{p(n)}{q(n)}=\frac{(a_{1}+n-1)\cdots
(a_{r}+n-1)x}{(b_{1}+n-1)\cdots (b_{s}+n-1)} \) for \( n>0 \). The evaluation
can thus be done with
bit complexity \( O((\log N)^{2}M(N)) \) for
\( r=s \) and \( O(\log N\: M(N)) \) for \( r<s \).
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsection{Example: \( \pi \)}
The Ramanujan series for \( \pi \)
\[
\frac{1}{\pi }=\frac{12}{C^{3/2}}\sum^{\infty }_{n=0}
\frac{(-1)^n(6n)!(A+nB)}{(3n)!n!^{3}C^{3n}}\]
with \( A=13591409 \), \( B=545140134 \), \( C=640320 \) found by the
Chudnovsky's \footnote{A special case of \cite{87}, formula (5.5.18),
with N=163.} and which is used by the {\sf LiDIA} \cite{95,97,97b} and the Pari
\cite{95b} system to compute \( \pi \), is usually written as an algorithm
of bit complexity \( O(N^{2}) \). It is, however, possible to apply
binary splitting to the sum. Put \( a(n)=A+nB \), \( b(n)=1 \),
\( p(0)=1 \), \( q(0)=1 \), and \( p(n)=-(6n-5)(2n-1)(6n-1) \),
\( q(n)=n^{3}C^3/24 \) for \( n>0 \). This reduces the complexity to
\( O((\log N)^{2}M(N)) \). Although this is theoretically slower than
Brent-Salamin's quadratically convergent iteration, which has a bit
complexity of \( O(\log N\: M(N)) \), in practice the binary splitted
Ramanujan sum is three times faster than Brent-Salamin, at least in the
range from \( N=1000 \) bits to \( N=1000000 \) bits.
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%% \subsection{Example: Catalan's constant}
\subsection{Example: Catalan's constant \( G \)}
A linearly convergent sum for Catalan's constant
\[
G:=\sum ^{\infty }_{n=0}\frac{(-1)^{n}}{(2n+1)^{2}}\]
is given in \cite{87}, p. 386:
\[
G = \frac{3}{8}\sum ^{\infty }_{n=0}\frac{1}{{2n \choose n} (2n+1)^{2}}
+\frac{\pi }{8}\log (2+\sqrt{3})
\]
The series is summed using binary splitting, putting \( a(n)=1 \),
\( b(n)=2n+1 \), \( p(0)=1 \), \( q(0)=1 \), and
\( p(n)=n \), \( q(n)=2(2n+1) \) for \( n>0 \). Thus
\( G \) can be computed with bit complexity \( O((\log N)^{2}M(N)) \).
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsection{Example: The Gamma function at rational points}
For evaluating \( \Gamma (s) \) for rational \( s \), we first reduce \( s \)
to the range \( 1\leq s\leq 2 \) by the formula \( \Gamma (s+1)=s\Gamma (s) \).
To compute \( \Gamma (s) \) with a precision of \( N \) bits, choose a
positive integer \( x \) with \( xe^{-x}<2^{-N} \). Partial integration lets
us write
\begin{eqnarray*}
\Gamma (s)&=& \int ^{\infty }_{0}e^{-t}t^{s-1}dt\\
&=& x^{s}e^{-x}\:\sum ^{\infty }_{n=0}
\frac{x^{n}}{s(s+1)\cdots (s+n)}
+\int^{\infty }_{x}e^{-t}t^{s-1}dt\\
\end{eqnarray*}
The last integral is \( <xe^{-x}<2^{-N} \). The series is evaluated as a
hypergeometric function (see above); the number of terms to be summed up is
\( O(N) \), since \( x=O(N) \). Thus the entire computation can be done with
bit complexity \( O((\log N)^{2}M(N)) \).
\begin{description}
\item [Note:]~
\end{description}
This result is already mentioned in \cite{76b}.
E.~Karatsuba \cite{91} extends this result to \( \Gamma (s) \) for algebraic
\( s \).
For \( \Gamma (s) \) there is no checkpointing possible because of the
dependency on \( x \) in the binary splitting.
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsection{Example: The Riemann Zeta value \( \zeta (3) \) \label{zeta}}
Recently, Doron Zeilberger's method of ``creative telescoping'' has been
applied to Riemann's zeta function at \( s=3 \) (see \cite{96c}), which is
also known as {\em Ap{\'e}ry's constant}:
\[
\zeta (3)=
\frac{1}{2}\sum ^{\infty }_{n=1}
\frac{(-1)^{n-1}(205n^{2}-160n+32)}{n^{5}{2n \choose n}^{5}}
\]
This sum consists of three hypergeometric series. Binary splitting can also be
applied directly, by putting \( a(n)=205n^{2}+250n+77 \), \( b(n)=1 \),
\( p(0)=1 \), \( p(n)=-n^{5} \) for \( n>0 \), and \( q(n)=32(2n+1)^{5} \).
Thus the bit complexity of computing \( \zeta (3) \) is
\( O((\log N)^{2}M(N)) \).
\begin{description}
\item [Note:]~
\end{description}
Using this the authors were able to establish a new record in the
calculation of \( \zeta (3) \) by computing 1,000,000 decimals \cite{96d}.
The computation took 8 hours on a Hewlett Packard 9000/712 machine. After
distributing on a cluster of 4 HP 9000/712 machines the same computation
required only 2.5 hours. The half hour was necessary for reading the partial
results from disk and for recombining them. Again, we have used binary-splitting
for recombining: the 4 partial result produced 2 results which were combined
to the final 1,000,000 decimals value of \( \zeta (3) \).
This example shows the importance of checkpointing. Even if a machine crashes
through the calculation, the results of the other machines are still usable.
Additionally, being able to parallelise the computation reduced the computing
time dramatically.
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\section{Evaluation of linearly convergent series of sums}
The technique presented in the previous section also applies to all linearly
convergent sums of the form
\[
U=\sum ^{\infty }_{n=0}\frac{a(n)}{b(n)}\left( \frac{c(0)}{d(0)}+\cdots
+\frac{c(n)}{d(n)}\right) \frac{p(0)\cdots p(n)}{q(0)\cdots q(n)}\]
where \( a(n) \), \( b(n) \), \( c(n) \), \( d(n) \), \( p(n) \),
\( q(n) \) are integers with \( O(\log n) \) bits. The most often
used case is again that \( a(n) \), \( b(n) \), \( c(n) \), \( d(n) \),
\( p(n) \), \( q(n) \) are polynomials in \( n \) with
integer coefficients.
\begin{description}
\item [Algorithm:]~
\end{description}
Given two index bounds \( n_{1} \)and \( n_{2} \), consider the partial sums
\[
S=\sum _{n_{1}\leq n<n_{2}}\frac{a(n)}{b(n)}
\frac{p(n_{1})\cdots p(n)}{q(n_{1})\cdots q(n)}\]
and
\[
U=\sum _{n_{1}\leq n<n_{2}}\frac{a(n)}{b(n)}
\left( \frac{c(n_{1})}{d(n_{1})}+\cdots +\frac{c(n)}{d(n)}\right)
\frac{p(n_{1})\cdots p(n)}{q(n_{1})\cdots q(n)}\]
As above, we compute the integers \( P={p(n_{1})}\cdots {p(n_{2}-1)} \),
\( Q={q(n_{1})}\cdots {q(n_{2}-1)} \), \( B={b(n_{1})}\cdots {b(n_{2}-1)} \),
\( T=BQS \), \( D={d(n_{1})}\cdots {d(n_{2}-1)} \),
\( C=D\left( \frac{c(n_{1})}{d(n_{1})}+\cdots +
\frac{c(n_{2}-1)}{d(n_{2}-1)}\right) \) and \( V=DBQU \).
If \( n_{2}-n_{1}<4 \), these
are computed directly. If \( n_{2}-n_{1}\geq 4 \), they are computed using
{\em binary splitting}: Choose an index \( n_{m} \) in the middle of
\( n_{1} \)and \( n_{2} \), compute the components \( P_{l} \), \( Q_{l} \),
\( B_{l} \), \( T_{l} \), \( D_{l} \), \( C_{l} \), \( V_{l} \) belonging
to the interval \( n_{1}\leq n<n_{m} \), compute the components \( P_{r} \),
\( Q_{r} \), \( B_{r} \), \( T_{r} \), \( D_{r} \), \( C_{r} \),
\( V_{r} \) belonging to the interval \( n_{m}\leq n<n_{2} \), and set
\( P=P_{l}P_{r} \), \( Q=Q_{l}Q_{r} \), \( B=B_{l}B_{r} \),
\( T=B_{r}Q_{r}T_{l}+B_{l}P_{l}T_{r} \), \( D=D_{l}D_{r} \),
\( C=C_{l}D_{r}+C_{r}D_{l} \) and
\( V=D_{r}B_{r}Q_{r}V_{l}+D_{r}C_{l}B_{l}P_{l}T_{r}+D_{l}B_{l}P_{l}V_{r} \).
Finally, this algorithm is applied to \( n_{1}=0 \) and
\( n_{2}=n_{\max}=O(N) \), and final floating-point divisions
\( S=\frac{T}{BQ} \) and \( U=\frac{V}{DBQ} \) are performed.
\begin{description}
\item [Complexity:]~
\end{description}
The bit complexity of computing \( S \) and \( U \) with \( N \) bits of
precision is \( O((\log N)^{2}M(N)) \).
\begin{description}
\item [Proof:]~
\end{description}
By our assumption that \( a(n) \), \( b(n) \), \( c(n) \), \( d(n) \),
\( p(n) \), \( q(n) \) are integers with \( O(\log n) \) bits,
the integers \( P \), \( Q \), \( B \), \( T \), \( D \), \( C \),
\( V \) belonging to the interval \( n_{1}\leq n<n_{2} \) all have
\( O((n_{2}-n_{1})\log n_{2}) \) bits. The rest of the proof is as in the
previous section.
\begin{description}
\item [Checkpointing/Parallelising:]~
\end{description}
A checkpoint can be easily done by storing the (integer) values of
\( n_1 \), \( n_2 \), \( P \), \( Q \), \( B \), \( T \) and additionally
\( D \), \( C \), \( V \). Similarly, if \( m \) processors are available,
then the interval \( [0,n_{max}] \) can be divided into \( m \) pieces of
length \( l = \lfloor n_{max}/m \rfloor \). After each processor \( i \) has
computed the sum of its interval \( [il,(i+1)l] \), the partial sums are
combined to the final result using the rules described above.
\begin{description}
\item [Implementation:]~
\end{description}
The C++ implementation of the above algorithm is very similar
to the previous one. The initialisation is done now by a structure
{\tt abpqcd\_series} containing arrays {\tt a}, {\tt b}, {\tt p},
{\tt q}, {\tt c} and {\tt d} of multiprecision integers. The values of
the arrays at the index \( n \) correspond to the values of the functions
\( a \), \( b \), \( p \), \( q \), \( c \) and \( d \) at the integer point
\( n \). The (partial) results of the algorithm are stored in the
{\tt abpqcd\_series\_result} structure, which now contains 3 new elements
({\tt C}, {\tt D} and {\tt V}).
\begin{verbatim}
// abpqcd_series is initialised by user
struct { bigint *a, *b, *p, *q, *c, *d;
} abpqcd_series;
// abpqcd_series_result holds the partial results
struct { bigint P, Q, B, T, C, D, V;
} abpqcd_series_result;
void sum_abpqcd(abpqcd_series_result & r,
int n1, int n2,
const abpqcd_series & arg)
{
switch (n2 - n1)
{
case 0:
error_handler("summation device",
"sum_abpqcd:: n2-n1 should be > 0.");
break;
case 1: // the result at the point n1
r.P = arg.p[n1];
r.Q = arg.q[n1];
r.B = arg.b[n1];
r.T = arg.a[n1] * arg.p[n1];
r.D = arg.d[n1];
r.C = arg.c[n1];
r.V = arg.a[n1] * arg.c[n1] * arg.p[n1];
break;
// cases 2, 3, 4 left out for simplicity
default: // general case
// the left and the right partial sum
abpqcd_series_result L, R;
// find the middle of the interval
int nm = (n1 + n2) / 2;
// sum left side
sum_abpqcd(L, n1, nm, arg);
// sum right side
sum_abpqcd(R, nm, n2, arg);
// put together
r.P = L.P * R.P;
r.Q = R.Q * L.Q;
r.B = L.B * R.B;
bigint tmp = L.B * L.P * R.T;
r.T = R.B * R.Q * L.T + tmp;
r.D = L.D * R.D;
r.C = L.C * R.D + R.C * L.D;
r.V = R.D * (R.B * R.Q * L.V + L.C * tmp)
+ L.D * L.B * L.P * R.V;
break;
}
}
\end{verbatim}
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\subsection{Example: Euler's constant \( C \) \label{eulergamma}}
\begin{description}
\item [Theorem:]~
\end{description}
Let \( f(x)=\sum ^{\infty }_{n=0}\frac{x^{n}}{n!^{2}} \) and
\( g(x)=\sum ^{\infty }_{n=0}H_{n}\frac{x^{n}}{n!^{2}} \). Then for
\( x\rightarrow \infty \),
\( \frac{g(x)}{f(x)}=\frac{1}{2}\log x+C+O\left( e^{-4\sqrt{x}}\right) \).
\begin{description}
\item [Proof:]~
\end{description}
The Laplace method for asymptotic evaluation of exponentially growing
sums and integrals yields
\[
f(x)=
e^{2\sqrt{x}}x^{-\frac{1}{4}}\frac{1}{2\sqrt{\pi }}(1+O(x^{-\frac{1}{4}}))\]
and
\[
g(x)=e^{2\sqrt{x}}x^{-\frac{1}{4}}\frac{1}{2\sqrt{\pi }}
\left(\frac{1}{2}\log x+C+O(\log x\cdot x^{-\frac{1}{4}})\right)\]
On the other hand, \( h(x):=\frac{g(x)}{f(x)} \) satisfies the
differential equation
\[
xf(x)\cdot h''(x)+(2xf'(x)+f(x))\cdot h'(x)=f'(x)\]
hence
\[
h(x)=\frac{1}{2}\log x+C+c_{2}
\int ^{\infty }_{x}\frac{1}{tf(t)^{2}}dt=\frac{1}{2}\log x+C+O(e^{-4\sqrt{x}})\]
\begin{description}
\item [Algorithm:]~
\end{description}
To compute \( C \) with a precision of \( N \) bits, set
\[ x=\left\lceil (N+2)\: \frac{\log 2}{4}\right\rceil ^{2} \]
and evaluate the series for \( g(x) \) and \( f(x) \) simultaneously,
using the binary-splitting algorithm,
with \( a(n)=1 \), \( b(n)=1 \), \( c(n)=1 \), \( d(n)=n+1 \),
\( p(n)=x \), \( q(n)=(n+1)^{2} \). Let \( \alpha =3.591121477\ldots \)
be the solution of the equation \( -\alpha \log \alpha +\alpha +1=0 \). Then
\( \alpha \sqrt{x}-\frac{1}{4\log \alpha }\log \sqrt{x}+O(1) \)
terms of the series suffice for the relative error to be bounded
by \( 2^{-N} \).
\begin{description}
\item [Complexity:]~
\end{description}
The bit complexity of this algorithm is \( O((\log N)^{2}M(N)) \).
\begin{description}
\item [Note:]~
\end{description}
This algorithm was first mentioned in \cite{80}. It is by far
the fastest known algorithm for computing Euler's constant.
For Euler's constant there is no checkpointing possible because
of the dependency on \( x \) in the binary splitting.
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\section{Computational results}
In this section we present some computational results of our CLN and
{\sf LiDIA} implementation of the algorithms presented in this note. We use
the official version (1.3) and an experimental version (1.4a) of {\sf LiDIA}.
We have taken advantage of {\sf LiDIA}'s ability to replace its kernel
(multiprecision arithmetic and memory management) \cite{95,97,97b}, so we were
able to use in both cases CLN's fast integer arithmetic routines.
\subsection{Timings}
The table in Figure \ref{Fig1} shows the running times for the calculation of
\( \exp(1) \), \( \log(2) \), \( \pi \), \( C \), \( G \) and \( \zeta(3) \)
to precision 100, 1000, 10000 and 100000 decimal digits. The timings are given
in seconds and they denote the {\em real} time needed, i.e. system and user
time. The computation was done on an Intel Pentium with 133Hz and 32MB of RAM.
\begin{figure}[htb]
\begin{center}
\begin{tabular}{|l|l|l|l|l|l|l|}
\hline
D &\( \exp(1) \)&\( \log(2) \)&\( \pi \)&\( C \) &\( G \)&\( \zeta(3) \)\\
\hline
\hline
\( 10^2 \) &0.0005 & 0.0020 &0.0014 & 0.0309 &0.0179 & 0.0027 \\
\hline
\( 10^3 \) &0.0069 & 0.0474 &0.0141 & 0.8110 &0.3580 & 0.0696 \\
\hline
\( 10^4 \) &0.2566 & 1.9100 &0.6750 & 33.190 &13.370 & 2.5600 \\
\hline
\( 10^5 \) &5.5549 & 45.640 &17.430 & 784.93 &340.33 & 72.970 \\
\hline
\end{tabular}
\caption{{\sf LiDIA-1.4a} timings of computation of constants using
binary-splitting}\label{Fig1}
\end{center}
\end{figure}
The second table (Figure \ref{Fig2}) summarizes the performance of
\( exp(x) \) in various Computer Algebra systems\footnote{We do not list
the timings of {\sf LiDIA-1.4a} since these are comparable to those of CLN.}.
For a fair comparison of the algorithms, both argument and precision are
chosen in such a way, that system--specific optimizations (BCD arithmetic
in Maple, FFT multiplication in CLN, special exact argument handling in
{\sf LiDIA}) do not work. We use \( x = -\sqrt{2} \) and precision
\( 10^{(i/3)} \), with \( i \) running from \( 4 \) to \( 15 \).
\begin{figure}[htb]
\begin{center}
\begin{tabular}{|r|l|l|l|l|}
\hline
D & Maple & Pari & {\sf LiDIA-1.3} & CLN \\
\hline
\hline
21 & 0.00090 & 0.00047 & 0.00191 & 0.00075 \\
\hline
46 & 0.00250 & 0.00065 & 0.00239 & 0.00109 \\
\hline
100 & 0.01000 & 0.00160 & 0.00389 & 0.00239 \\
\hline
215 & 0.03100 & 0.00530 & 0.00750 & 0.00690 \\
\hline
464 & 0.11000 & 0.02500 & 0.02050 & 0.02991 \\
\hline
1000 & 0.4000 & 0.2940 & 0.0704 & 0.0861 \\
\hline
2154 & 1.7190 & 0.8980 & 0.2990 & 0.2527 \\
\hline
4641 & 8.121 & 5.941 & 1.510 & 0.906 \\
\hline
10000 & 39.340 & 39.776 & 7.360 & 4.059 \\
\hline
21544 & 172.499 & 280.207 & 39.900 & 15.010 \\
\hline
46415 & 868.841 & 1972.184& 129.000 & 39.848 \\
\hline
100000 & 4873.829 & 21369.197& 437.000 & 106.990 \\
\hline
\end{tabular}
\caption{Timings of computation of \( \exp(-\sqrt{2}) \)}\label{Fig2}
\end{center}
\end{figure}
MapleV R3 is the slowest system in this comparison. This is probably due to
the BCD arithmetic it uses. However, Maple seems to have an asymptotically
better algorithm for \( exp (x) \) for numbers having more than 10000 decimals.
In this range it outperforms Pari-1.39.03, which is the fastest system in the
0--200 decimals range.
The comparison indicating the strength of binary-splitting is between
{\sf LiDIA-1.3} and CLN itself. Having the same kernel, the only
difference is here that {\sf LiDIA-1.3} uses Brent's \( O(\sqrt{n}M(n)) \)
for \( \exp(x) \), whereas CLN changes from Brent's method to a
binary-splitting version for large numbers.
As expected in the range of 1000--100000 decimals CLN outperforms
{\sf LiDIA-1.3} by far. The fact that {\sf LiDIA-1.2.1} is faster
in the range of 200--1000 decimals (also in some trig. functions)
is probably due to a better optimized \( O(\sqrt{n}M(n)) \) method
for \( \exp(x) \).
\subsection {Distributed computing of \( \zeta (3) \)}
Using the method described in \ref{zeta} the authors were the first to
compute 1,000,000 decimals of \( \zeta (3) \) \cite{96d}.
The computation took 8 hours on a Hewlett Packard 9000/712 machine. After
distributing on a cluster of 4 HP 9000/712 machines the same computation
required only 2.5 hours. The half hour was necessary for reading the partial
results from disk and for recombining them. Again, we have used binary-splitting
for recombining: the 4 partial result produced 2 results which were combined
to the final 1,000,000 decimals value of \( \zeta (3) \).
This example shows the importance of checkpointing. Even if a machine crashes
through the calculation, the results of the other machines are still usable.
Additionally, being able to parallelise the computation reduced the computing
time dramatically.
\subsection{Euler's constant \( C \)}
We have implemented a version of Brent's and McMillan's algorithm \cite{80} and
a version accelerated by binary-splitting as shown in \ref{eulergamma}.
The computation of \( C \) was done twice on a SPARC-Ultra machine
with 167 MHz and 256 MB of RAM. The first computation using the non-acellerated
version required 160 hours. The result of this computation was then verified
by the binary splitting version in (only) 14 hours.
The first 475006 partial quotients of the continued fraction of \( C \)
were computed on an Intel Pentium with 133 MHz and 32 MB of RAM in 3 hours
using a programm by H. te Riele based on \cite{96e}, which was translated to
{\sf LiDIA} for efficiency reasons. Computing the 475006th
convergent produced the following improved theorem:
\medskip
\centerline{If \( C \) is a rational number, \(C=p/q\), then \( |q| > 10^{244663} \)}
\medskip
Details of this computation (including statistics on the partial
quotients) can be found in \cite{98}.
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\section{Conclusions}
Although powerful, the binary splitting method has not been widely used.
Especially, no information existed on the applicability of this method.
In this note we presented a generic binary-splitting summation device for
evaluating two types of linearly convergent series. From this we derived simple
and computationally efficient algorithms for the evaluation of elementary
functions and constants. These algorithms work with {\em exact}
objects, making them suitable for use within Computer Algebra systems.
We have shown that the practical performance of our algorithms is
superior to current system implementations. In addition to existing methods,
our algorithms provide the possibility of checkpointing and parallelising.
These features can be useful for huge calculations, such as those done in
analytic number theory research.
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\section{Thanks}
The authors would like to thank J\"org Arndt, for pointing us to
chapter 10 in \cite{87}. We would also like to thank Richard P. Brent for
his comments and Hermann te Riele for providing us his program for the
continued fraction computation of Euler's constant.
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
\bibliography{binsplit}
\bibliographystyle{acm}
\end{document}