mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
gslInterpolator.hpp
Go to the documentation of this file.
1/** \file gslInterpolator.hpp
2 * \brief Class for managing 1-D interpolation using the GNU Scientific Library
3 *
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 * \ingroup gen_math_files
7 *
8 */
9
10//***********************************************************************//
11// Copyright 2015-2022 Jared R. Males (jaredmales@gmail.com)
12//
13// This file is part of mxlib.
14//
15// mxlib is free software: you can redistribute it and/or modify
16// it under the terms of the GNU General Public License as published by
17// the Free Software Foundation, either version 3 of the License, or
18// (at your option) any later version.
19//
20// mxlib is distributed in the hope that it will be useful,
21// but WITHOUT ANY WARRANTY; without even the implied warranty of
22// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
23// GNU General Public License for more details.
24//
25// You should have received a copy of the GNU General Public License
26// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
27//***********************************************************************//
28
29#ifndef gslInterpolator_hpp
30#define gslInterpolator_hpp
31
32#include <vector>
33#include <gsl/gsl_interp.h>
34#include <gsl/gsl_errno.h>
35
36#include "../mxException.hpp"
37
38namespace mx
39{
40namespace math
41{
42
43/// GSL Linear Interpolation
44/** \ingroup interpolation
45 */
46template <typename _realT>
48{
49 typedef _realT realT;
50
51 static const gsl_interp_type *interpolator()
52 {
53 return ::gsl_interp_linear;
54 }
55};
56
57#ifndef MX_OLD_GSL
58/// GSL Steffen Interpolation
59/** \ingroup interpolation
60 */
61template <typename _realT>
63{
64 typedef _realT realT;
65
66 static const gsl_interp_type *interpolator()
67 {
68 return ::gsl_interp_steffen;
69 }
70};
71#endif
72
73/// Class to manage interpolation using the GSL interpolation library.
74/**
75 *
76 * \tparam interpT is the interpolation type, which also specifies the floating point precision.
77 *
78 * \ingroup interpolation
79 */
80template <typename interpT>
82{
83 public:
84 typedef typename interpT::realT realT;
85
86 static_assert( std::is_same<double, typename std::remove_cv<realT>::type>::value,
87 "GSL Interpolation only works with double" );
88
89 protected:
90 gsl_interp *m_interp{ nullptr }; ///< the gsl interpolator structure
91 gsl_interp_accel *m_acc{ nullptr }; ///< the gsl interpolation accelerator structure
92
93 realT *m_xin; ///< the input x data
94 realT *m_yin; ///< the input y data
95
96 public:
97 /// Default constructor
99
100 /// Raw pointer constructor
101 gslInterpolator( realT *xin, ///< [in] the input x data, this pointer is stored and must remain valid
102 realT *yin, ///< [in] the input y data, this pointer is stored and must remain valid
103 size_t Nin ///< [in] the size of data vectors
104 );
105
106 /// Vector constructor
107 gslInterpolator( std::vector<realT> &xin, ///< [in] the input x data, the pointer to xin.data() is stored and xin
108 ///< must remain unchanged
109 std::vector<realT> &yin ///< [in] the input y data, the pointer to yin.data() is stored and yin
110 ///< must remain unchanged
111 );
112
113 /// Destructor
114 /** Deallocates working memory.
115 */
117
118 /// Setup the interpolator for the supplied data pointers
119 /**
120 * \throws mx::err::allocerr if allocation fails
121 * \throws mx::err::liberr if GSL initialization fails
122 */
123 void setup( realT *xin, ///< [in] the input x data, this pointer is stored and must remain valid
124 realT *yin, ///< [in] the input y data, this pointer is stored and must remain valid
125 size_t Nin ///< [in] the size of data vectors
126 );
127
128 /// Setup the interpolator for the supplied data vectors
129 /**
130 * \throws mx::err::sizeerr if the vectors are not the same size
131 * \throws mx::err::allocerr if allocation fails
132 * \throws mx::err::liberr if GSL initialization fails
133 */
134 void setup( std::vector<realT>
135 &xin, ///< [in] the input x data, the pointer to xin.data() is stored and xin must remain unchanged
136 std::vector<realT>
137 &yin ///< [in] the input y data, the pointer to yin.data() is stored and yin must remain unchanged
138 );
139
140 /// Calculate the interpolated function value at a point
141 /**
142 * \returns the interpolated value at \p x
143 */
144 realT operator()( const realT &x /**< [in] the point at which to interpolate */ );
145};
146
147template <typename interpT>
151
152template <typename interpT>
154{
155 setup( xin, yin, Nin );
156}
157
158template <typename interpT>
159gslInterpolator<interpT>::gslInterpolator( std::vector<realT> &xin, std::vector<realT> &yin )
160{
161 setup( xin.data(), yin.data(), xin.size() );
162}
163
164template <typename interpT>
166{
167 if( m_interp != nullptr )
168 gsl_interp_free( m_interp );
169 if( m_acc != nullptr )
170 gsl_interp_accel_free( m_acc );
171}
172
173template <typename interpT>
174void gslInterpolator<interpT>::setup( realT *xin, realT *yin, size_t Nin )
175{
176 if( m_interp != nullptr )
177 gsl_interp_free( m_interp );
178 if( m_acc != nullptr )
179 gsl_interp_accel_free( m_acc );
180
181 m_interp = gsl_interp_alloc( interpT::interpolator(), Nin );
182 if( !m_interp )
183 {
184 mxThrowException( err::allocerr, "gslInterpolation::setup", "gsl_interp_alloc failed" );
185 }
186
187 m_acc = gsl_interp_accel_alloc();
188 if( !m_acc )
189 {
190 mxThrowException( err::allocerr, "gslInterpolation::setup", "gsl_interp_accel_alloc failed" );
191 }
192
193 int errv = gsl_interp_init( m_interp, xin, yin, Nin );
194 if( errv != 0 )
195 {
197 err::liberr, "gslInterpolation::setup", std::string( "gsl_interp_init failed: " ) + gsl_strerror( errv ) );
198 }
199
200 errv = gsl_interp_accel_reset( m_acc );
201 if( errv != 0 )
202 {
204 "gslInterpolation::setup",
205 std::string( "gsl_interp_accel_reset failed: " ) + gsl_strerror( errv ) );
206 }
207 m_xin = xin;
208 m_yin = yin;
209}
210
211template <typename interpT>
212void gslInterpolator<interpT>::setup( std::vector<realT> &xin, std::vector<realT> &yin )
213{
214 if( xin.size() != yin.size() )
215 {
216 mxThrowException( err::sizeerr, "gslInterpolator<interpT>::setup", "input vectors must be the same size" );
217 }
218 setup( xin.data(), yin.data(), xin.size() );
219}
220
221template <typename interpT>
222typename interpT::realT gslInterpolator<interpT>::operator()( const realT &x )
223{
224 realT y;
225 int errv = gsl_interp_eval_e( m_interp, m_xin, m_yin, x, m_acc, &y );
226 if( errv != 0 && errv != GSL_EDOM )
227 {
228 mxThrowException( err::liberr, "gslInterpolation::setup", "gsl_interp_eval_e failed" );
229 }
230 return y;
231}
232
233} // namespace math
234} // namespace mx
235
236#endif // gslInterpolator_hpp
mxException for an allocation error
mxException for errors returned by a library call
mxException for a size error
Class to manage interpolation using the GSL interpolation library.
gslInterpolator(std::vector< realT > &xin, std::vector< realT > &yin)
Vector constructor.
gsl_interp_accel * m_acc
the gsl interpolation accelerator structure
void setup(std::vector< realT > &xin, std::vector< realT > &yin)
Setup the interpolator for the supplied data vectors.
realT * m_xin
the input x data
realT operator()(const realT &x)
Calculate the interpolated function value at a point.
gslInterpolator()
Default constructor.
gslInterpolator(realT *xin, realT *yin, size_t Nin)
Raw pointer constructor.
gsl_interp * m_interp
the gsl interpolator structure
void setup(realT *xin, realT *yin, size_t Nin)
Setup the interpolator for the supplied data pointers.
realT * m_yin
the input y data
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
The mxlib c++ namespace.
Definition mxError.hpp:106
GSL Linear Interpolation.
GSL Steffen Interpolation.