Version: SMASH-3.2
tabulation.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2015-2019,2024
3  * SMASH Team
4  *
5  * GNU General Public License (GPLv3 or later)
6  */
7 
8 #include "smash/tabulation.h"
9 
10 #include "smash/numeric_cast.h"
11 
12 namespace smash {
13 
14 Tabulation::Tabulation(double x_min, double range, size_t num,
15  std::function<double(double)> f)
16  : x_min_(x_min), x_max_(x_min + range), inv_dx_(num / range) {
17  if (num < 2) {
18  throw std::runtime_error("Tabulation needs at least two values");
19  }
20  values_.resize(num + 1);
21  const double dx = range / num;
22  for (size_t i = 0; i <= num; i++) {
23  values_[i] = f(x_min_ + i * dx);
24  }
25 }
26 
27 double Tabulation::get_value_step(double x) const {
28  if (x < x_min_) {
29  return 0.;
30  }
31  const unsigned int n =
32  numeric_cast<unsigned int>(std::floor((x - x_min_) * inv_dx_ + 0.5));
33  if (n >= values_.size()) {
34  return values_.back();
35  } else {
36  return values_[n];
37  }
38 }
39 
40 double Tabulation::get_value_linear(double x, Extrapolation extrapol) const {
41  if (x < x_min_) {
42  return 0.;
43  }
44  if (extrapol == Extrapolation::Zero && x > x_max_) {
45  return 0.0;
46  }
47  if (extrapol == Extrapolation::Const && x > x_max_) {
48  return values_.back();
49  }
50  const double index_double = (x - x_min_) * inv_dx_;
51  // here n is the lower index
52  const size_t n =
53  std::min(static_cast<size_t>(index_double), values_.size() - 2);
54  const double r = index_double - n;
55  return values_[n] + (values_[n + 1] - values_[n]) * r;
56 }
57 
64 static void swrite(std::ofstream& stream, double x) {
65  stream.write(reinterpret_cast<const char*>(&x), sizeof(x));
66 }
67 
74 static double sread_double(std::ifstream& stream) {
75  double x;
76  stream.read(reinterpret_cast<char*>(&x), sizeof(x));
77  return x;
78 }
79 
86 static void swrite(std::ofstream& stream, size_t x) {
87  // We want to support 32-bit and 64-bit platforms, so we store a 64-bit
88  // integer on all platforms.
89  const auto const_size_x = static_cast<uint64_t>(x);
90  stream.write(reinterpret_cast<const char*>(&const_size_x),
91  sizeof(const_size_x));
92 }
93 
100 static size_t sread_size(std::ifstream& stream) {
101  uint64_t x;
102  stream.read(reinterpret_cast<char*>(&x), sizeof(x));
103  if (x > std::numeric_limits<size_t>::max()) {
104  throw std::runtime_error("trying to read vector larger than supported");
105  }
106  return x;
107 }
108 
115 static void swrite(std::ofstream& stream, const std::vector<double> x) {
116  swrite(stream, x.size());
117  if (x.size() > 0) {
118  stream.write(reinterpret_cast<const char*>(x.data()),
119  sizeof(x[0]) * x.size());
120  }
121 }
122 
129 static std::vector<double> sread_vector(std::ifstream& stream) {
130  const size_t n = sread_size(stream);
131  std::vector<double> x;
132  x.resize(n);
133  stream.read(reinterpret_cast<char*>(x.data()), sizeof(double) * n);
134  return x;
135 }
136 
143 static void swrite(std::ofstream& stream, sha256::Hash x) {
144  // The size is always the same, so there is no need to write it.
145  stream.write(reinterpret_cast<const char*>(x.data()),
146  sizeof(x[0]) * x.size());
147 }
148 
155 static sha256::Hash sread_hash(std::ifstream& stream) {
156  sha256::Hash x;
157  stream.read(reinterpret_cast<char*>(x.data()), x.size());
158  return x;
159 }
160 
161 void Tabulation::write(std::ofstream& stream, sha256::Hash hash) const {
162  swrite(stream, hash);
163  swrite(stream, x_min_);
164  swrite(stream, x_max_);
165  swrite(stream, inv_dx_);
166  swrite(stream, values_);
167 }
168 
169 Tabulation Tabulation::from_file(std::ifstream& stream, sha256::Hash hash) {
170  sha256::Hash hash_from_stream = sread_hash(stream);
171  Tabulation t;
172  if (hash != hash_from_stream) {
173  return t;
174  }
175  t.x_min_ = sread_double(stream);
176  t.x_max_ = sread_double(stream);
177  t.inv_dx_ = sread_double(stream);
178  t.values_ = sread_vector(stream);
179  return t;
180 }
181 
182 } // namespace smash
A class for storing a one-dimensional lookup table of floating-point values.
Definition: tabulation.h:35
double x_min_
lower bound for tabulation
Definition: tabulation.h:113
double inv_dx_
inverse step size 1/dx
Definition: tabulation.h:119
double x_max_
upper bound for tabulation
Definition: tabulation.h:116
std::vector< double > values_
vector for storing tabulated values
Definition: tabulation.h:110
static Tabulation from_file(std::ifstream &stream, sha256::Hash hash)
Construct a tabulation object by reading binary data from a stream.
Definition: tabulation.cc:169
Tabulation()
Construct an empty tabulation object.
Definition: tabulation.h:40
void write(std::ofstream &stream, sha256::Hash hash) const
Write a binary representation of the tabulation to a stream.
Definition: tabulation.cc:161
double get_value_step(double x) const
Look up a value from the tabulation (without any interpolation, simply using the closest tabulated va...
Definition: tabulation.cc:27
double get_value_linear(double x, Extrapolation extrapolation=Extrapolation::Linear) const
Look up a value from the tabulation using linear interpolation.
Definition: tabulation.cc:40
constexpr int n
Neutron.
std::array< uint8_t, HASH_SIZE > Hash
A SHA256 hash.
Definition: sha256.h:25
Definition: action.h:24
static std::vector< double > sread_vector(std::ifstream &stream)
Read binary representation of a vector of doubles.
Definition: tabulation.cc:129
static double sread_double(std::ifstream &stream)
Read binary representation of a double.
Definition: tabulation.cc:74
static sha256::Hash sread_hash(std::ifstream &stream)
Read binary representation of a SHA256 hash.
Definition: tabulation.cc:155
Extrapolation
The kind of extrapolation used by the tabulation.
Definition: tabulation.h:26
static void swrite(std::ofstream &stream, double x)
Write binary representation to stream.
Definition: tabulation.cc:64
static size_t sread_size(std::ifstream &stream)
Read binary representation of a size_t.
Definition: tabulation.cc:100