mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
cahoyAlbedos.hpp
Go to the documentation of this file.
1/** \file cahoyAlbedos.hpp
2 * \author Jared R. Males
3 * \brief Type definitions for the Cahoy et al 2010 albedo spectra
4 * \ingroup astrophot
5 *
6 */
7
8#ifndef mx_astro_cahoyAlbedos_hpp
9#define mx_astro_cahoyAlbedos_hpp
10
11#include "units.hpp"
12#include "../ioutils/readColumns.hpp"
13
14namespace mx
15{
16namespace astro
17{
18
19/// An albedo spectrum directly from the Cahoy et al (2010) \cite cahoy_2010 grid.
20/**
21 * \ingroup astrophot_spectra
22 */
23template <typename _units>
25{
26 typedef _units units;
27 typedef typename units::realT realT;
28
29 typedef struct
30 {
31 realT sep; ///< The separation, in AU. Can be 0.8, 2.0, 5.0, or 10.0.
32 int metal; ///< The metallicty. Can be 1, 3, 10, or 30.
33 realT phase; ///< The phase. Must be one of the values in the grid: 0.0, 0.1745, 0.3491, 0.5236, 0.6981,
34 ///< 0.8727, 1.0472, 1.2217, 1.3963,1.5708,1.7453,1.9199, 2.0944, 2.2689, 2.4435, 2.6180, 2.7925,2.9671,
35 ///< 3.139.
36 } paramsT;
37
38 static const bool freq = false;
39
40 /// Convert from um to SI m
41 static constexpr realT wavelengthUnits = static_cast<realT>( 1e6 );
42
43 /// Geometric albedos are dimensionless.
44 static constexpr realT fluxUnits = static_cast<realT>( 1 );
45
46 static constexpr const char *dataDirEnvVar = "CAHOYALBEDO_DATADIR";
47
48 /// The file name is constructed from the paramerters sep, metal, and phase.
49 static std::string fileName( const paramsT &params )
50 {
51 std::string fname;
52
53 if( params.sep == 0.8 )
54 {
55 fname = "0.8";
56 }
57 else if( params.sep == 2.0 )
58 {
59 fname = "2.0";
60 }
61 else if( params.sep == 5.0 )
62 {
63 fname = "5";
64 }
65 else if( params.sep == 10.0 )
66 {
67 fname = "10";
68 }
69 else
70 {
71 mxError( "cahoyAlbedos::fileName", MXE_INVALIDARG, "invalid separation (params.sep)" );
72 return "";
73 }
74
75 fname += "au_";
76 fname += ioutils::convertToString<int>( params.metal );
77 fname += "x_albedos/00_Spectral_Albedos ";
78
79 char pstr[9];
80 snprintf( pstr, 9, "%0.5f", params.phase );
81
82 fname += pstr;
83
84 fname += ".txt";
85
86 return fname;
87 }
88
89 /// Read the spectrum from the file specified by path. Extra columns are discarded.
90 static int readSpectrum( std::vector<realT> &rawLambda,
91 std::vector<realT> &rawSpectrum,
92 const std::string &path,
93 const paramsT &params )
94 {
96
97 return ioutils::readColumns( path, sk, rawLambda, sk, rawSpectrum );
98 }
99};
100
101/// A class to manage interpolation in separation and phase on the albedo spectrum grid from Cahoy et al (2010) \cite
102/// cahoy_2010.
103/** Usage:
104 \code
105 std::vector<double> lambda;
106 math::vectorScale(lambda, 100000, 4e-12, 600e-9); //Construct the wavelength scale
107
108 cahoyGrid<units::si<double>> grid;
109 grid.loadGrid(3, lambda); //Open the grid for 3x metallicity, pre-interpolating on the wavelength scale.
110
111 grid.setParameters({1.0, 0.1}); //Set the separation to 1.0 AU, and the phase angle to 0.1 radians.
112 grid.setSpectrum(lambda); //Perform the interpolation. Note that lambda is only used for a size check here.
113
114 \endcode
115 * After the above steps you now have the Ag spectrum at 1.0 AU and 0.1 radians phase on the lambda scale.
116 *
117 * \ingroup astrophot
118 */
119template <typename _units>
120struct cahoyGrid : public baseSpectrum<typename _units::realT>
121{
122 typedef _units units;
123 typedef typename units::realT realT;
124
125 typedef struct
126 {
127 realT sep;
128 realT phase;
129 } paramsT;
130
131 paramsT _params;
132
133 std::vector<realT> _sep;
134 std::vector<realT> _phase;
135
136 std::vector<realT> _lambda;
137 std::vector<std::vector<std::vector<realT>>> _ag;
138
139 /// Load the grid into memory for one of the metallicities and pre-interpolate onto the wavelength scale.
140 int loadGrid( int metal, ///< [in] the metalicity, can be 1, 3, 10 or 30.
141 std::vector<realT> &lambda ///< [in] the wavelength scale.
142 )
143 {
144 _sep = { 0.8, 2.0, 2.0, 5.0, 10.0 }; // 2 is in there twice to prepare for transition distance.
145 _phase = { 0.0,
146 0.1745,
147 0.3491,
148 0.5236,
149 0.6981,
150 0.8727,
151 1.0472,
152 1.2217,
153 1.3963,
154 1.5708,
155 1.7453,
156 1.9199,
157 2.0944,
158 2.2689,
159 2.4435,
160 2.6180,
161 2.7925,
162 2.9671,
163 3.139 };
164
165 _ag.resize( _sep.size() );
166 for( int i = 0; i < _ag.size(); ++i )
167 _ag[i].resize( _phase.size() );
168
170
171 for( int i = 0; i < _sep.size(); ++i )
172 {
173 for( int j = 0; j < _phase.size(); ++j )
174 {
175 rawSpect.setParameters( { _sep[i], metal, _phase[j] } );
176 if( rawSpect.setSpectrum( lambda ) < 0 )
177 {
178 mxError( "cahoGrid::loadGrid", MXE_FILERERR, "Error reading albedo spectrum." );
179 return -1;
180 }
181
182 _ag[i][j] = rawSpect._spectrum;
183 }
184 }
185
186 _sep[1] = 1.0; // The transition from cloudy to not-cloudy occurs here.
187
188 this->_spectrum.resize( lambda.size() );
189 return 0;
190 }
191
192 /// Set the separatio and phase of the spectrum.
193 int setParameters( const paramsT &params /**< [in] Struct containting the separation and phase of the spectrum */ )
194 {
195 _params = params;
196
197 return 0;
198 }
199
200 /// Get an interpolated spectrum at arbitrary non-grid point using bilinear interpolation.
201 /** If outside the endpoints of the grid, the grid is simply extended (i.e. constant albedo).
202 */
204 std::vector<realT>
205 &lambda /**< [in] the wavelength grid. Must be the same as used for construction and/or openGrid*/ )
206 {
207 int i_l, i_u, j_l, j_u;
208 realT phase, sep;
209
210 if( lambda.size() != this->_spectrum.size() )
211 {
212 mxError(
213 "cahoyGrid::setSpectrum",
214 MXE_INVALIDARG,
215 "wavelength grid (lambda) is not the same size as used for openGrid, or loadGrid not yet called." );
216 return -1;
217 }
218
219 // Normalize phase
220 phase = fmod( _params.phase, math::pi<realT>() );
221 if( phase < 0 )
222 phase += math::pi<realT>();
223
224 sep = _params.sep;
225
226 i_l = _sep.size() - 1;
227 while( _sep[i_l] > sep && i_l > 0 )
228 --i_l;
229
230 i_u = 0;
231 while( i_u < _sep.size() - 1 )
232 {
233 if( _sep[i_u] >= sep )
234 break;
235 ++i_u;
236 }
237
238 j_l = _phase.size() - 1;
239 while( j_l > 0 )
240 {
241 if( _phase[j_l] <= phase )
242 break;
243 --j_l;
244 }
245
246 j_u = 0;
247 while( j_u < _phase.size() - 1 )
248 {
249 if( _phase[j_u] >= phase )
250 break;
251 ++j_u;
252 }
253
254 realT x = sep - _sep[i_l];
255 if( x < 0 )
256 x = 0;
257
258 realT y = ( phase - _phase[j_l] );
259 if( y < 0 )
260 y = 0;
261
262 realT x0, x1;
263
264 for( int i = 0; i < this->_spectrum.size(); ++i )
265 {
266 x0 = _ag[i_l][j_l][i];
267 x1 = _ag[i_u][j_l][i];
268
269 if( y != 0 && j_u != j_l )
270 {
271 x0 += ( _ag[i_l][j_u][i] - _ag[i_l][j_l][i] ) * y / ( _phase[j_u] - _phase[j_l] );
272 x1 += ( _ag[i_u][j_u][i] - _ag[i_u][j_l][i] ) * y / ( _phase[j_u] - _phase[j_l] );
273 }
274 if( x == 0 || i_u == i_l )
275 this->_spectrum[i] = x0;
276 else
277 this->_spectrum[i] = x0 + ( x1 - x0 ) * x / ( _sep[i_u] - _sep[i_l] );
278 }
279
280 return 0;
281 }
282};
283
284} // namespace astro
285
286} // namespace mx
287
288#endif // mx_astro_cahoyAlbedos_hpp
int readColumns(const std::string &fname, arrTs &...arrays)
Read in columns from a text file.
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
The mxlib c++ namespace.
Definition mxError.hpp:106
Class to manage an astronomical spectrum.
int setSpectrum(gridT &lambda)
Load the spectrum and interpolate it onto a wavelength scale.
int setParameters(const paramsT &params)
Set the parameters of the spectrum, using the underlying spectrums parameter type.
Base spectrum class which provides manipulation and characterization functionality.
std::vector< realT > _spectrum
Contains the spectrum after it is set.
int loadGrid(int metal, std::vector< realT > &lambda)
Load the grid into memory for one of the metallicities and pre-interpolate onto the wavelength scale.
int setSpectrum(std::vector< realT > &lambda)
Get an interpolated spectrum at arbitrary non-grid point using bilinear interpolation.
int setParameters(const paramsT &params)
Set the separatio and phase of the spectrum.
An albedo spectrum directly from the Cahoy et al (2010) grid.
static constexpr realT fluxUnits
Geometric albedos are dimensionless.
static int readSpectrum(std::vector< realT > &rawLambda, std::vector< realT > &rawSpectrum, const std::string &path, const paramsT &params)
Read the spectrum from the file specified by path. Extra columns are discarded.
static std::string fileName(const paramsT &params)
The file name is constructed from the paramerters sep, metal, and phase.
static constexpr realT wavelengthUnits
Convert from um to SI m.
A dummy class to allow mx::readColumns to skip a column(s) in a file without requiring memory allocat...
Unit specifications and conversions.