Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Interpolation functions #1814

Closed
wants to merge 79 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
79 commits
Select commit Hold shift + click to select a range
f340d6f
added interpolation functions
pgree Mar 31, 2020
63172f1
moved prim tests to new files
pgree Mar 31, 2020
389f42b
Merge branch 'develop' into feature/58-interp
pgree Apr 1, 2020
98fdf8b
[Jenkins] auto-formatting by clang-format version 6.0.0 (tags/google/…
stan-buildbot Apr 1, 2020
efde025
fixing messed up branch
pgree Apr 1, 2020
2169b76
fixing messed up branch again
pgree Apr 1, 2020
b36451b
trying merge
pgree Apr 1, 2020
6c8726d
[Jenkins] auto-formatting by clang-format version 5.0.0-3~16.04.1 (ta…
stan-buildbot Apr 1, 2020
ee28a65
changes to reflect design doc discussion
pgree Aug 4, 2020
7582c50
fixing tests
pgree Aug 4, 2020
e91a3fc
reflects steve's changes
pgree Aug 4, 2020
eb0a0d9
adding docs
pgree Aug 5, 2020
fc4abb5
Merge branch 'feature/58-interp' of https://github.com/pgree/math int…
pgree Aug 5, 2020
774dccb
Merge commit '5b30e403bb1b2276dee2dd3d6736e52e67960651' into HEAD
yashikno Aug 5, 2020
c4582be
[Jenkins] auto-formatting by clang-format version 6.0.0-1ubuntu2~16.0…
stan-buildbot Aug 5, 2020
74b8419
fixing formatting
pgree Aug 5, 2020
0ce6663
doc typo fix
pgree Aug 5, 2020
c4dd75d
another doc typo fix
pgree Aug 5, 2020
be423da
another doc typo fix
pgree Aug 5, 2020
05737f4
another doc typo fix
pgree Aug 5, 2020
149f0d7
adding checks and tests
pgree Aug 5, 2020
8594a47
[Jenkins] auto-formatting by clang-format version 6.0.0-1ubuntu2~16.0…
stan-buildbot Aug 6, 2020
ee5b696
fix namespace
pgree Aug 6, 2020
adbfc60
fix namespace
pgree Aug 6, 2020
e4589c0
Merge branch 'feature/58-interp' of https://github.com/pgree/math int…
pgree Aug 6, 2020
3fde18a
added test for lin_interp
pgree Aug 6, 2020
c624bd0
added test for lin_interp
pgree Aug 6, 2020
77c0d25
adding includes
pgree Aug 6, 2020
f4dde7b
adding includes
pgree Aug 6, 2020
d37c844
adding inline statements
pgree Aug 7, 2020
d052649
Merge commit '988a44ac0960f73b05c1b91baa847c29e3822f43' into HEAD
yashikno Aug 8, 2020
dea4306
[Jenkins] auto-formatting by clang-format version 6.0.0-1ubuntu2~16.0…
stan-buildbot Aug 8, 2020
2a3020b
adding inline statements
pgree Aug 8, 2020
cdf5bbb
Merge branch 'feature/58-interp' of https://github.com/pgree/math int…
pgree Aug 8, 2020
cce952b
[Jenkins] auto-formatting by clang-format version 6.0.0-1ubuntu2~16.0…
stan-buildbot Aug 8, 2020
e9e9786
redo order of initializations
pgree Aug 12, 2020
2055a03
Merge branch 'feature/58-interp' of https://github.com/pgree/math int…
pgree Aug 12, 2020
9ee99cc
Merge commit 'f94d6f1397612bf7c84d0aaef1875d062a2398e2' into HEAD
yashikno Aug 12, 2020
4544f5e
[Jenkins] auto-formatting by clang-format version 6.0.0-1ubuntu2~16.0…
stan-buildbot Aug 12, 2020
5b9d65e
update docs
pgree Aug 12, 2020
06d55bd
Merge branch 'feature/58-interp' of https://github.com/pgree/math int…
pgree Aug 12, 2020
a5d00b6
[Jenkins] auto-formatting by clang-format version 6.0.0-1ubuntu2~16.0…
stan-buildbot Aug 12, 2020
b401faf
adding include
pgree Aug 12, 2020
15e820f
Merge branch 'feature/58-interp' of https://github.com/pgree/math int…
pgree Aug 12, 2020
2401a63
rand_r fix
pgree Aug 12, 2020
26b4529
rand fix
pgree Aug 12, 2020
561c909
rand fix
pgree Aug 13, 2020
2b98654
change rand
pgree Aug 14, 2020
0cc8ae9
modified mix test
pgree Aug 18, 2020
a3f988b
remove extra initializations
pgree Aug 20, 2020
85da4f6
remove using std
pgree Aug 21, 2020
e1e423e
Merge commit '7ba6890d5e64859c21fd09d69a465bfb06dab9d3' into HEAD
yashikno Aug 21, 2020
490cc03
[Jenkins] auto-formatting by clang-format version 6.0.0-1ubuntu2~16.0…
stan-buildbot Aug 21, 2020
22280c9
namespace issues
pgree Aug 21, 2020
71d97f3
Merge branch 'feature/58-interp' of https://github.com/pgree/math int…
pgree Aug 21, 2020
f5bbc96
namespace issues
pgree Aug 22, 2020
0e32b0c
params changed to vector
pgree Aug 31, 2020
dc49bc1
added new conv_gaus_line for sum of convolutions
pgree Sep 1, 2020
c1cc956
various fixes
pgree Sep 2, 2020
09829be
change distance to pointer arithmetic
pgree Sep 2, 2020
c25fc04
Merge commit '4144cdc2446da9029da5c3595d86a73f3dac0092' into HEAD
yashikno Sep 2, 2020
91199f8
[Jenkins] auto-formatting by clang-format version 6.0.1-14 (tags/RELE…
stan-buildbot Sep 2, 2020
07060e7
revert conv_gaus_line
pgree Sep 3, 2020
b65bb3a
fix merge conflicts
pgree Sep 3, 2020
443d6ff
Merge commit '0f3ab571e56c9210d31bbb9e9f62126a3b3fb257' into HEAD
yashikno Sep 3, 2020
57c7b0d
[Jenkins] auto-formatting by clang-format version 6.0.0-1ubuntu2~16.0…
stan-buildbot Sep 3, 2020
fa5067f
fix issues
pgree Oct 27, 2020
68654cd
fix merge conflicts
pgree Oct 27, 2020
64853bc
Merge commit '43295fbda6d6ab5b8cb2323ba2a001faedaa37a5' into HEAD
yashikno Oct 27, 2020
62ea083
[Jenkins] auto-formatting by clang-format version 6.0.0-1ubuntu2~16.0…
stan-buildbot Oct 27, 2020
b5f69eb
merge conflicts
pgree Oct 28, 2020
d0d75a2
merge conf
pgree Oct 28, 2020
a992cad
rerun tests
pgree Nov 11, 2020
226ee16
Merge commit '5291fc8cf03c901201243014957843ffb61c4705' into HEAD
yashikno Nov 11, 2020
57c2f8b
[Jenkins] auto-formatting by clang-format version 6.0.0-1ubuntu2~16.0…
stan-buildbot Nov 11, 2020
2cbbad1
renaming
pgree Sep 21, 2021
a9f71a2
Merge branch 'feature/58-interp' of https://github.com/pgree/math int…
pgree Sep 28, 2021
7c8ab8b
Merge commit '57f05b7354af652031e10f7e278f82bdbedb1a40' into HEAD
yashikno Sep 28, 2021
c19300c
[Jenkins] auto-formatting by clang-format version 6.0.0-1ubuntu2~16.0…
stan-buildbot Sep 28, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions stan/math/prim/fun.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@
#include <stan/math/prim/fun/columns_dot_self.hpp>
#include <stan/math/prim/fun/conj.hpp>
#include <stan/math/prim/fun/constants.hpp>
#include <stan/math/prim/fun/conv_gaus_line.hpp>
#include <stan/math/prim/fun/copysign.hpp>
#include <stan/math/prim/fun/corr_constrain.hpp>
#include <stan/math/prim/fun/corr_free.hpp>
Expand Down Expand Up @@ -137,6 +138,8 @@
#include <stan/math/prim/fun/initialize.hpp>
#include <stan/math/prim/fun/initialize_fill.hpp>
#include <stan/math/prim/fun/int_step.hpp>
#include <stan/math/prim/fun/interp_gauss.hpp>
#include <stan/math/prim/fun/interp_lin.hpp>
#include <stan/math/prim/fun/inv.hpp>
#include <stan/math/prim/fun/inv_Phi.hpp>
#include <stan/math/prim/fun/inv_cloglog.hpp>
Expand Down
47 changes: 47 additions & 0 deletions stan/math/prim/fun/conv_gaus_line.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
#ifndef STAN_MATH_PRIM_FUN_CONV_GAUS_LINE
#define STAN_MATH_PRIM_FUN_CONV_GAUS_LINE

#include <stan/math/prim/meta.hpp>
#include <stan/math/prim/prob/normal_cdf.hpp>
#include <cmath>
#include <vector>

namespace stan {
namespace math {

/**
* Evaluate the convolution of a line with a Gaussian kernel on an interval.
*
* \f$\int_{t_0}^{t_1} (at + b) e^{\frac{-(t-x)^2}{2\sigma^2}} dt \f$
*
* @tparam Tx type of x
* @param t0 lower integration bound
* @param t1 upper integration bound
* @param a coefficient of t in line
* @param b constant in line
* @param x point at which convolution is evaluated
* @param sig2 variance of the Gaussian kernel
* @return The value of the derivative
*/
template <typename Tx>
inline return_type_t<Tx> conv_gaus_line(double t0, double t1, double a,
double b, const Tx& x, double sig2) {
using stan::math::normal_cdf;
using std::exp;
using std::pow;
using std::sqrt;
const double sig = sqrt(sig2);

return_type_t<Tx> y
= (a * x + b) * (normal_cdf(t1, x, sig) - normal_cdf(t0, x, sig));
y += -a * sig2 / sqrt(2 * stan::math::pi() * sig2)
* (exp(-pow(t1 - x, 2) / (2 * sig2))
- exp(-pow(t0 - x, 2) / (2 * sig2)));

return y;
}

} // namespace math
} // namespace stan

#endif
197 changes: 197 additions & 0 deletions stan/math/prim/fun/interp_gauss.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,197 @@
#ifndef STAN_MATH_PRIM_FUN_INTERP_GAUSS
#define STAN_MATH_PRIM_FUN_INTERP_GAUSS

#include <stan/math/prim/fun/conv_gaus_line.hpp>
#include <stan/math/prim/fun/square.hpp>
#include <stan/math/prim/err.hpp>
#include <cmath>
#include <vector>
#include <algorithm>

namespace stan {
namespace math {

namespace internal {
/*
* find the smallest difference between successive elements in a sorted vector
*/
template <typename Tx>
inline double min_diff(const std::vector<Tx>& xs) {
double dmin = value_of(xs[1]) - value_of(xs[0]);
int N = xs.size();
for (int i = 1; i < N - 1; i++) {
if (value_of(xs[i + 1]) - value_of(xs[i]) < dmin) {
dmin = value_of(xs[i + 1]) - value_of(xs[i]);
}
}
return dmin;
}

} // namespace internal

/**
* Given a set of reference points \f$(xs_i, ys_i)\f$, create a mollifier
* that intersects the reference points. This function requires as input
* a vector, params, created by the function interp_gauss_precomp. The
* algorithm used to create the mollifier is an iterative algorithm that
* works as follows. First a linear interpolation is created through the
* reference points. Then, the
* linear interpolation is convolved with a Gaussian whose width is
* proportional the smallest distance between successive points
* \f$xs_i\f$ and \f$xs_{i+1}\f$. Since the convolution is unlikely to
* intersect the reference points, the y-values of the reference points
* are shifted and the process repeats itself until the interpolation
* intersects all reference points.
*
* Note: This interpolation scheme should be used when the function
* to be interpolated is well-resolved by the reference points.
*
* @tparam Tx type of x
* @param xs vector of independent variable of reference points
* @param ys vector of dependent variable of reference points
* @param params a vector created by interp_gauss_precomp
* @param x the point at which to evaluate the interpolation
* @return value of the interpolation at x
*/
template <typename Tx>
inline return_type_t<Tx> interp_gauss(const std::vector<double>& xs,
const std::vector<double>& ys,
const std::vector<double>& params,
const Tx& x) {
// enforce that interpolation point is between smallest and largest
// reference point
static char const* function = "interp_gauss";
check_less_or_equal(function, "Interpolation point", x, xs.back());
check_greater_or_equal(function, "Interpolation point", x, xs.front());
check_ordered(function, "xs", xs);
check_not_nan(function, "xs", xs);
check_not_nan(function, "ys", ys);
check_not_nan(function, "x", x);
check_greater(function, "xs", xs.size(), 1);

// number of standard deviations to extend endpoints for convolution
const double NSTDS = 10;
int N = xs.size();

// params is vector of the form (as, bs, sig2)

// create copy of xs so that endpoints can be extended
std::vector<double> xs2 = xs;

// extend out first and last lines for convolution
double sig = std::sqrt(params[2 * N - 2]);
xs2[0] = xs[0] - NSTDS * sig;
xs2[N - 1] = xs[N - 1] + NSTDS * sig;

// no need to convolve far from center of gaussian, so
// get lower and upper indexes for integration bounds
auto lb = std::lower_bound(xs.begin(), xs.end(), x - NSTDS * sig);
int ind_start = std::distance(xs.begin(), lb) - 1;
ind_start = std::max(0, ind_start);

auto ub = std::upper_bound(xs.begin(), xs.end(), x + NSTDS * sig);
int ind_end = std::distance(xs.begin(), ub);
ind_end = std::min(N - 1, ind_end);

// sum convolutions over intervals
return_type_t<Tx> y = 0;
for (int i = ind_start; i < ind_end; i++) {
y += conv_gaus_line(xs2[i], xs2[i + 1], params[i], params[(N - 1) + i], x,
params[2 * N - 2]);
}
return y;
}

/**
* This function was written to be used with interp_gauss. This function
* computes the shifted y-values of the reference points of an interpolation
* in such a way that when that piecewise linear function is convolved
* with a Gaussian kernel, the resulting function coincides with the
* points \f$(xs_i, ys_i)\f$ inputted into this function. The output of this
* function depends heavily on the choice of width of the Gaussian
* kernel, which at the time of writing, is set to one tenth the
* minimum distance between successive elements of the vector xs.
* A tolerance for the maximum distance between the interpolation and
* all reference points is also set manually and is not an input.
*
* @param xs vector of independent variable of reference points
* @param ys vector of dependent variable of reference points
* @return vector containing slopes, intercepts, and width of kernel
*/
inline std::vector<double> interp_gauss_precomp(const std::vector<double>& xs,
const std::vector<double>& ys) {
static char const* function = "interp_gauss_precomp";
check_not_nan(function, "xs", xs);
check_not_nan(function, "ys", ys);
check_ordered(function, "xs", xs);
check_greater(function, "xs", xs.size(), 1);

using internal::min_diff;
static const double max_diff = 1e-8;
// set Gaussian kernel to sig2_scale times smallest difference between
// successive points
static const double sig2_scale = 0.1;
int N = xs.size();

// create the vector to be returned that consists of as, bs, sig2
std::vector<double> params(2 * N - 1, 0.0);
params[2 * N - 2] = square(min_diff(xs) * sig2_scale);

// copy ys into a new vector that will be changed
std::vector<double> y2s = ys;

// interatively find interpolation that coincides with ys at xs
int max_iters = 100;
double dd;
for (int j = 0; j < max_iters; j++) {
// find slope and intercept of line between each point
for (size_t i = 0; i < N - 1; i++) {
params[i] = (y2s[i + 1] - y2s[i]) / (xs[i + 1] - xs[i]);
params[(N - 1) + i] = -xs[i] * params[i] + y2s[i];
}

double dmax = 0;
for (size_t i = 0; i < N; i++) {
dd = ys[i] - interp_gauss(xs, y2s, params, xs[i]);
y2s[i] += dd;
dmax = std::max(std::abs(dd), dmax);
}
if (dmax < max_diff) {
break;
}
}
return params;
}

/**
* This function combines interp_gauss_precomp and interp_gauss.
* It takes as input two vectors of reference points (xs and ys)
* in addition to a vector, xs_new, of points at which this
* function will evaluate the interpolation at those reference
* points.
*
* @tparam Tx type of xs_new
* @param xs vector of independent variable of reference points
* @param ys vector of dependent variable of reference points
* @param xs_new vector of point at which to evaluate interpolation
* @return vector of interpolation values
*/
template <typename Tx>
inline std::vector<Tx> interp_gauss(const std::vector<double>& xs,
const std::vector<double>& ys,
const std::vector<Tx>& xs_new) {
int n_interp = xs_new.size();
std::vector<Tx> ys_new(n_interp);

// create interpolation
std::vector<double> params = interp_gauss_precomp(xs, ys);
for (int i = 0; i < n_interp; i++) {
ys_new[i] = interp_gauss(xs, ys, params, xs_new[i]);
}
return ys_new;
}

} // namespace math
} // namespace stan

#endif
91 changes: 91 additions & 0 deletions stan/math/prim/fun/interp_lin.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,91 @@
#ifndef STAN_MATH_PRIM_FUN_INTERP_LIN
#define STAN_MATH_PRIM_FUN_INTERP_LIN

#include <stan/math/prim/err.hpp>
#include <cmath>
#include <vector>
#include <algorithm>

namespace stan {
namespace math {

/**
* This function performs linear interpolation. The function takes as
* input two vectors of reference points \f$(xs_i, ys_i)\f$ in addition
* to one value, \f$x\f$, at which the value of the linear interpolation
* is to be returned. The vector xs is required to be in increasing order
* For values of \f$x\f$ less than xs[0], the function returns ys[0].
* For values of \f$x\f$ greater than xs[n-1], the function returns
* ys[n-1], where xs is of length n.
*
* @param xs vector of independent variable of reference points
* @param ys vector of dependent variable of reference points
* @param x the point at which to evaluate the interpolation
* @return value of linear interpolation at x
*/
inline double interp_lin(const std::vector<double>& xs,
const std::vector<double>& ys, double x) {
static char const* function = "interp_lin";
check_ordered(function, "xs", xs);
check_not_nan(function, "xs", xs);
check_not_nan(function, "ys", ys);
check_not_nan(function, "x", x);
check_greater(function, "xs", xs.size(), 1);
int n = xs.size();

// if x is less than left endpoint or greater than right, return endpoint
if (x <= xs[0]) {
return ys[0];
}
if (x >= xs[n - 1]) {
return ys[n - 1];
}

// find in between which points the input, x, lives
auto ub = std::upper_bound(xs.begin(), xs.end(), x);
int ind = std::distance(xs.begin(), ub);

// check if the interpolation point falls on a reference point
if (x == xs[ind - 1]) {
return ys[ind - 1];
}

// do linear interpolation
double x1 = xs[ind - 1];
double x2 = xs[ind];
double y1 = ys[ind - 1];
double y2 = ys[ind];

return y1 + (x - x1) * (y2 - y1) / (x2 - x1);
}

/**
* This function takes as input two vectors of reference points (xs and ys)
* in addition to a vector, xs_new, of points at which this
* function will evaluate a linear interpolation through those reference
* points.
*
* @param xs vector of independent variable of reference points
* @param ys vector of dependent variable of reference points
* @param xs_new vector of point at which to evaluate interpolation
* @return vector of interpolation values
*/

template <typename Tx>
inline std::vector<Tx> interp_lin(const std::vector<double>& xs,
const std::vector<double>& ys,
const std::vector<Tx>& xs_new) {
int n_interp = xs_new.size();
std::vector<Tx> ys_new(n_interp);

// create interpolation
for (int i = 0; i < n_interp; i++) {
ys_new[i] = interp_lin(xs, ys, xs_new[i]);
}
return ys_new;
}

} // namespace math
} // namespace stan

#endif
1 change: 1 addition & 0 deletions stan/math/rev/fun.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@
#include <stan/math/rev/fun/cholesky_factor_constrain.hpp>
#include <stan/math/rev/fun/columns_dot_product.hpp>
#include <stan/math/rev/fun/columns_dot_self.hpp>
#include <stan/math/rev/fun/conv_gaus_line.hpp>
#include <stan/math/rev/fun/conj.hpp>
#include <stan/math/rev/fun/cos.hpp>
#include <stan/math/rev/fun/cosh.hpp>
Expand Down
Loading