A library for working with phylogenetic and population genetic data.
v0.32.0
compensated_sum.hpp
Go to the documentation of this file.
1 #ifndef GENESIS_UTILS_MATH_COMPENSATED_SUM_H_
2 #define GENESIS_UTILS_MATH_COMPENSATED_SUM_H_
3 
4 /*
5  Genesis - A toolkit for working with phylogenetic data.
6  Copyright (C) 2014-2023 Lucas Czech
7 
8  This program is free software: you can redistribute it and/or modify
9  it under the terms of the GNU General Public License as published by
10  the Free Software Foundation, either version 3 of the License, or
11  (at your option) any later version.
12 
13  This program is distributed in the hope that it will be useful,
14  but WITHOUT ANY WARRANTY; without even the implied warranty of
15  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16  GNU General Public License for more details.
17 
18  You should have received a copy of the GNU General Public License
19  along with this program. If not, see <http://www.gnu.org/licenses/>.
20 
21  Contact:
22  Lucas Czech <lczech@carnegiescience.edu>
23  Department of Plant Biology, Carnegie Institution For Science
24  260 Panama Street, Stanford, CA 94305, USA
25 */
26 
34 #include <cassert>
35 #include <cmath>
36 #include <cstdint>
37 #include <limits>
38 #include <stdexcept>
39 
40 namespace genesis {
41 namespace utils {
42 
43 // =================================================================================================
44 // Kahan Summation Algorithms
45 // =================================================================================================
46 
47 // Forward declaration
48 template<typename SummationAlgorithm>
50 
54 struct KahanSummation{};
55 
60 
64 struct KleinSummation{};
65 
70 
75 
80 
81 // =================================================================================================
82 // Kahan Sum
83 // =================================================================================================
84 
106 template<typename SummationAlgorithm = NeumaierSum>
107 class CompensatedSum
108 {
109 public:
110 
111  // ---------------------------------------------------------
112  // Constructor and Rule of Five
113  // ---------------------------------------------------------
114 
115  CompensatedSum() = default;
116 
120  CompensatedSum( double value )
121  : sum_(value)
122  {}
123 
130  template<class It>
131  CompensatedSum( It first, It last )
132  {
133  while( first != last ) {
134  add( *first );
135  ++first;
136  }
137  }
138 
142  CompensatedSum( std::initializer_list<double> list )
143  {
144  for( auto const e : list ) {
145  add( e );
146  }
147  }
148 
149  ~CompensatedSum() = default;
150 
151  CompensatedSum(CompensatedSum const&) = default;
152  CompensatedSum(CompensatedSum&&) = default;
153 
154  CompensatedSum& operator= (CompensatedSum const&) = default;
156 
157  // ---------------------------------------------------------
158  // Operators
159  // ---------------------------------------------------------
160 
164  inline void operator += ( double value )
165  {
166  add( value );
167  }
168 
174  inline void operator -= ( double value )
175  {
176  add( -value );
177  }
178 
185  inline CompensatedSum& operator= ( double value )
186  {
187  sum_ = value;
188  cor_0_ = 0.0;
189  cor_1_ = 0.0;
190  cor_2_ = 0.0;
191  return *this;
192  }
193 
197  inline operator double() const
198  {
199  // We forward to the get() function,
200  // as some algorithms do not simply return the sum here.
201  return get();
202  }
203 
204  // ---------------------------------------------------------
205  // Summation Functions
206  // ---------------------------------------------------------
207 
208  inline void add( double value )
209  {
210  add_( algorithm_, value );
211  }
212 
213  inline void reset()
214  {
215  sum_ = 0.0;
216  cor_0_ = 0.0;
217  cor_1_ = 0.0;
218  cor_2_ = 0.0;
219  }
220 
221  inline double get() const
222  {
223  // Corrections of the two advanced algorithms are only applied once in the very end.
224  // We always add them here for simplicity, as the unused ones are 0, which is safe to add.
225  return sum_ + cor_0_ + cor_1_ + cor_2_;
226  }
227 
228 private:
229 
230  // ---------------------------------------------------------
231  // Summation Implementation
232  // ---------------------------------------------------------
233 
234  inline void add_( KahanSummation, double value )
235  {
236  // Standard Kahan Summation
237  // Use volatile registers to avoid aggressive compiler optimization
238  auto const y = value - cor_0_;
239  volatile auto const t = sum_ + y;
240  volatile auto const z = t - sum_;
241  cor_0_ = z - y;
242  sum_ = t;
243 
244  // Base calculation for reference, without volatile
245  // auto const y = value - cor_0_;
246  // auto const t = sum_ + y;
247  // cor_0_ = ( t - sum_ ) - y;
248  // sum_ = t;
249  }
250 
251  inline void add_( NeumaierSummation, double value )
252  {
253  // Kahan Babushka Neumaier Summation
254  auto const t = sum_ + value;
255  if( std::abs( sum_ ) >= std::abs( value )) {
256  // If sum is bigger, low-order digits of value are lost.
257  volatile auto const v = sum_ - t;
258  cor_0_ += v + value;
259  } else {
260  // Else low-order digits of sum are lost.
261  volatile auto const v = value - t;
262  cor_0_ += v + sum_;
263  }
264  sum_ = t;
265  }
266 
267  inline void add_( KleinSummation, double value )
268  {
269  // Kahan Babushka Klein Summation
270  double c1 = 0.0;
271  double c2 = 0.0;
272  auto t = sum_ + value;
273  if( std::abs( sum_ ) >= std::abs( value )) {
274  volatile auto const v = sum_ - t;
275  c1 = v + value;
276  } else {
277  volatile auto const v = value - t;
278  c1 = v + sum_;
279  }
280  sum_ = t;
281  t = cor_1_ + c1;
282  if( std::abs( cor_1_ ) >= std::abs( c1 )) {
283  volatile auto const v = cor_1_ - t;
284  c2 = v + c1;
285  } else {
286  volatile auto const v = c1 - t;
287  c2 = v + cor_1_;
288  }
289  cor_1_ = t;
290  cor_2_ = cor_2_ + c2;
291  }
292 
293  // ---------------------------------------------------------
294  // Data Members
295  // ---------------------------------------------------------
296 
297 private:
298 
299  // Algorithm, selected at compile time.
300  SummationAlgorithm algorithm_{};
301 
302  // Sum and correction term.
303  double sum_ = 0.0;
304  double cor_0_ = 0.0;
305  double cor_1_ = 0.0;
306  double cor_2_ = 0.0;
307 
308 };
309 
310 } // namespace utils
311 } // namespace genesis
312 
313 #endif // include guard
genesis::utils::CompensatedSum
Compensated summation algorithmm, such as Kahan, Neumaier, and Klein summation.
Definition: compensated_sum.hpp:49
genesis::utils::CompensatedSum::add
void add(double value)
Definition: compensated_sum.hpp:208
genesis::utils::CompensatedSum::CompensatedSum
CompensatedSum(std::initializer_list< double > list)
Construct a CompensatedSum, summing over an initializer list of values.
Definition: compensated_sum.hpp:142
genesis::utils::CompensatedSum::~CompensatedSum
~CompensatedSum()=default
genesis::utils::CompensatedSum::CompensatedSum
CompensatedSum(It first, It last)
Construct a CompensatedSum, summing over a range of double.
Definition: compensated_sum.hpp:131
genesis::utils::KleinSummation
Tag for tag dispatching the algorithm in CompensatedSum. See there for details.
Definition: compensated_sum.hpp:64
genesis::utils::NeumaierSummation
Tag for tag dispatching the algorithm in CompensatedSum. See there for details.
Definition: compensated_sum.hpp:59
genesis::utils::CompensatedSum::reset
void reset()
Definition: compensated_sum.hpp:213
genesis::utils::CompensatedSum::operator+=
void operator+=(double value)
Add a value to the sum.
Definition: compensated_sum.hpp:164
genesis
Container namespace for all symbols of genesis in order to keep them separate when used as a library.
Definition: placement/formats/edge_color.cpp:42
genesis::utils::CompensatedSum::get
double get() const
Definition: compensated_sum.hpp:221
genesis::utils::CompensatedSum::CompensatedSum
CompensatedSum()=default
genesis::utils::CompensatedSum::CompensatedSum
CompensatedSum(double value)
Constructor that initializes the sum to a given value.
Definition: compensated_sum.hpp:120
genesis::utils::KahanSummation
Tag for tag dispatching the algorithm in CompensatedSum. See there for details.
Definition: compensated_sum.hpp:54
genesis::utils::CompensatedSum::operator=
CompensatedSum & operator=(CompensatedSum const &)=default
genesis::utils::CompensatedSum::operator-=
void operator-=(double value)
Subtract a value from the sum.
Definition: compensated_sum.hpp:174