Skip to content

Commit 9a6ae11

Browse files
committed
implement Cl3 using series expansion of Wu et.al.
1 parent bfbc7c2 commit 9a6ae11

File tree

4 files changed

+85
-1
lines changed

4 files changed

+85
-1
lines changed

test/alt/alt.h

+1
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@ void tsil_dilog_complex(long double re, long double im, long double* res_re, lon
4141
void tsil_trilog_complex(long double re, long double im, long double* res_re, long double* res_im);
4242

4343
double clausen_2_wu(double x);
44+
double clausen_3_wu(double x);
4445

4546
#ifdef __cplusplus
4647
}

test/alt/wu/wu.c

+77-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
#include <math.h>
22

33
/**
4-
* @brief Clausen function \f$\mathrm{Cl}_2(\theta) = \mathrm{Im}(\mathrm{Li}_2(e^{i\theta}))\f$
4+
* @brief Clausen function \f$\mathrm{Cl}_2(\theta)\f$
55
* @param x real angle
66
* @return \f$\mathrm{Cl}_2(\theta)\f$
77
* @author Alexander Voigt
@@ -75,3 +75,79 @@ double clausen_2_wu(double x)
7575

7676
return sgn*x*sum;
7777
}
78+
79+
/**
80+
* @brief Clausen function \f$\mathrm{Cl}_3(\theta)\f$
81+
* @param x real angle
82+
* @return \f$\mathrm{Cl}_3(\theta)\f$
83+
* @author Alexander Voigt
84+
*
85+
* Implementation as series expansion from Jiming Wu, Xiaoping Zhang,
86+
* Dongjie Liu: "An efficient calculation of the Clausen functions
87+
* Cl_n(0)(n >= 2)"
88+
*/
89+
double clausen_3_wu(double x)
90+
{
91+
const double PI = 3.14159265358979324;
92+
const double PI2 = 2*PI;
93+
const double C[] = {
94+
1.2020569031595943,
95+
-0.75,
96+
1.7361111111111111e-02,
97+
1.6203703703703704e-04,
98+
2.6573129251700680e-06,
99+
5.0521751910640800e-08,
100+
1.0280221244025958e-09,
101+
2.1775508813272637e-11,
102+
4.7396483546174904e-13,
103+
1.0523517259825012e-14,
104+
2.3724645155504571e-16,
105+
5.4136342063674700e-18,
106+
1.2475096984511389e-19,
107+
2.8982349732072164e-21,
108+
6.7795306977020209e-23,
109+
1.5951668979204825e-24,
110+
3.7722999459245736e-26,
111+
8.9602350004691215e-28,
112+
2.1365627618154762e-29,
113+
5.1121551453039508e-31,
114+
1.2269426427592847e-32,
115+
2.9528444795333068e-34,
116+
7.1242132481949272e-36,
117+
1.7227144823673827e-37,
118+
4.1742940635473232e-39
119+
};
120+
121+
if (x < 0) {
122+
x = -x;
123+
}
124+
125+
if (x >= PI2) {
126+
x = fmod(x, PI2);
127+
}
128+
129+
if (x > PI) {
130+
const double p0 = 6.28125;
131+
const double p1 = 0.0019353071795864769253;
132+
x = (p0 - x) + p1;
133+
}
134+
135+
if (x == 0) {
136+
return C[0];
137+
}
138+
139+
if (x == PI) {
140+
return -0.75*C[0];
141+
}
142+
143+
const double x2 = x*x;
144+
double sum = 0;
145+
146+
for (int i = sizeof(C)/sizeof(C[0]) - 1; i >= 0; --i) {
147+
sum = x2*sum + C[i];
148+
}
149+
150+
sum += x2/2*log(2*sin(x/2));
151+
152+
return sum;
153+
}

test/bench_Cl.cpp

+3
Original file line numberDiff line numberDiff line change
@@ -100,6 +100,9 @@ int main() {
100100
bench_fn([&](double x) { return polylogarithm::Cl3(x); }, values_d,
101101
"polylogarithm C++", "double");
102102

103+
bench_fn([&](double x) { return clausen_3_wu(x); }, values_d,
104+
"Wu", "double");
105+
103106
bench_fn([&](long double x) { return polylogarithm::Cl3(x); }, values_l,
104107
"polylogarithm C++", "long double");
105108

test/test_Cl3.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
#define DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN 1
22

33
#include "doctest.h"
4+
#include "alt.h"
45
#include "Cl3.hpp"
56
#include "read_data.hpp"
67
#include <cmath>
@@ -63,17 +64,20 @@ TEST_CASE("test_real_fixed_values")
6364
const auto cl64_expected = static_cast<double>(cl128_expected);
6465

6566
const auto cl64_poly = polylogarithm::Cl3(x64);
67+
const auto cl64_wu = clausen_3_wu(x64);
6668
const auto cl128_poly = polylogarithm::Cl3(x128);
6769

6870
INFO("x(64) = " << x64);
6971
INFO("Cl3(64) real = " << cl64_expected << " (expected)");
7072
INFO("Cl3(64) real = " << cl64_poly << " (polylogarithm C++)");
73+
INFO("Cl3(64) real = " << cl64_wu << " (Wu et.al.)");
7174
INFO("------------------------------------------------------------");
7275
INFO("x(128) = " << x128);
7376
INFO("Cl3(128) real = " << cl128_expected << " (expected)");
7477
INFO("Cl3(128) real = " << cl128_poly << " (polylogarithm C++)");
7578

7679
CHECK_CLOSE(cl64_poly , cl64_expected , 2*eps64 );
80+
CHECK_CLOSE(cl64_wu , cl64_expected , 2*eps64 );
7781
CHECK_CLOSE(cl128_poly, cl128_expected, 2*eps128);
7882
}
7983
}

0 commit comments

Comments
 (0)