LSST Applications  21.0.0-147-g0e635eb1+1acddb5be5,22.0.0+052faf71bd,22.0.0+1ea9a8b2b2,22.0.0+6312710a6c,22.0.0+729191ecac,22.0.0+7589c3a021,22.0.0+9f079a9461,22.0.1-1-g7d6de66+b8044ec9de,22.0.1-1-g87000a6+536b1ee016,22.0.1-1-g8e32f31+6312710a6c,22.0.1-10-gd060f87+016f7cdc03,22.0.1-12-g9c3108e+df145f6f68,22.0.1-16-g314fa6d+c825727ab8,22.0.1-19-g93a5c75+d23f2fb6d8,22.0.1-19-gb93eaa13+aab3ef7709,22.0.1-2-g8ef0a89+b8044ec9de,22.0.1-2-g92698f7+9f079a9461,22.0.1-2-ga9b0f51+052faf71bd,22.0.1-2-gac51dbf+052faf71bd,22.0.1-2-gb66926d+6312710a6c,22.0.1-2-gcb770ba+09e3807989,22.0.1-20-g32debb5+b8044ec9de,22.0.1-23-gc2439a9a+fb0756638e,22.0.1-3-g496fd5d+09117f784f,22.0.1-3-g59f966b+1e6ba2c031,22.0.1-3-g849a1b8+f8b568069f,22.0.1-3-gaaec9c0+c5c846a8b1,22.0.1-32-g5ddfab5d3+60ce4897b0,22.0.1-4-g037fbe1+64e601228d,22.0.1-4-g8623105+b8044ec9de,22.0.1-5-g096abc9+d18c45d440,22.0.1-5-g15c806e+57f5c03693,22.0.1-7-gba73697+57f5c03693,master-g6e05de7fdc+c1283a92b8,master-g72cdda8301+729191ecac,w.2021.39
LSST Data Management Base Package
TransmissionCurve.cc
Go to the documentation of this file.
1 /*
2  * LSST Data Management System
3  * Copyright 2017 LSST/AURA.
4  *
5  * This product includes software developed by the
6  * LSST Project (http://www.lsst.org/).
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 LSST License Statement and
19  * the GNU General Public License along with this program. If not,
20  * see <http://www.lsstcorp.org/LegalNotices/>.
21  */
22 
23 #include <algorithm>
24 #include <memory>
25 
26 #include "ndarray.h"
27 
28 #include "gsl/gsl_interp.h"
29 #include "gsl/gsl_interp2d.h"
30 #include "gsl/gsl_errno.h"
31 
36 
37 using namespace std::string_literals;
38 
39 #define LSST_CHECK_GSL(type, status) \
40  if (status) throw LSST_EXCEPT(type, "GSL error: "s + ::gsl_strerror(status))
41 
42 namespace lsst {
43 namespace afw {
44 
45 template std::shared_ptr<image::TransmissionCurve> table::io::PersistableFacade<
47 
48 namespace image {
49 
50 namespace {
51 
52 /*
53  * The TransmissionCurve implementation returned by TransmissionCurve::makeIdentity.
54  *
55  * This is zero-state singleton whose throughput is always exactly 1.
56  */
57 class IdentityTransmissionCurve : public TransmissionCurve {
58 public:
59  static constexpr char const* NAME = "IdentityTransmissionCurve";
60 
62  static std::shared_ptr<IdentityTransmissionCurve> instance(new IdentityTransmissionCurve());
63  return instance;
64  }
65 
66  std::pair<double, double> getWavelengthBounds() const override {
67  constexpr double inf = std::numeric_limits<double>::infinity();
68  return std::make_pair(-inf, inf);
69  }
70 
71  std::pair<double, double> getThroughputAtBounds() const override { return std::make_pair(1.0, 1.0); }
72 
73  void sampleAt(lsst::geom::Point2D const&, ndarray::Array<double const, 1, 1> const& wavelengths,
74  ndarray::Array<double, 1, 1> const& out) const override {
75  out.deep() = 1.0;
76  }
77 
78  bool isPersistable() const noexcept override { return true; }
79 
80 protected:
81  // transforming an IdentityTransmissionCurve is a no-op
84  return shared_from_this();
85  }
86 
87  // multiplying an IdentityTransmissionCurve always yields the other operand
89  std::shared_ptr<TransmissionCurve const> other) const override {
90  return other;
91  }
92 
93  std::string getPersistenceName() const override { return NAME; }
94 
95  void write(OutputArchiveHandle& handle) const override { handle.saveEmpty(); }
96 
97  class Factory : public table::io::PersistableFactory {
98  public:
99  std::shared_ptr<table::io::Persistable> read(InputArchive const& archive,
100  CatalogVector const& catalogs) const override {
101  LSST_ARCHIVE_ASSERT(catalogs.size() == 0u);
102  return get();
103  }
104 
105  Factory() : table::io::PersistableFactory(NAME) {}
106  };
107 
108  static Factory registration;
109 
110 private:
111  IdentityTransmissionCurve() = default;
112 };
113 
114 IdentityTransmissionCurve::Factory IdentityTransmissionCurve::registration;
115 
116 /*
117  * InterpolatedTransmissionCurve: implements makeSpatiallyConstant and makeRadial.
118  *
119  * InterpolatedTransmissionCurve is templated on an implementation class
120  * that does most of the work, letting it handle the boilerplate that is
121  * common to both spatially-constant and radially-varying curves.
122  *
123  * We use two tricks to avoid repetition that bear some explanation:
124  *
125  * - Even though the two implementations have different state, we can use
126  * the same PersistenceHelper template for both by using a std::vector
127  * to hold the keys for the arrays. Impl2d has one extra array to save,
128  * so its vector is one element longer.
129  *
130  * - The throughput array held by Impl2d is conceptually a 2-d array, but we
131  * actually store a flattened view into it. That's okay because both GSL
132  * and the persistence layer only care about the pointer and the size. That
133  * requires going through some hoops in makeRadial, but from then on it's
134  * clean sailing.
135  */
136 
137 template <typename T>
138 using GslPtr = std::unique_ptr<T, void (*)(T*)>;
139 
140 template <typename T>
141 GslPtr<T> makeGslPtr(T* p, void (*free)(T*)) {
142  if (p == nullptr) {
143  throw std::bad_alloc();
144  }
145  return GslPtr<T>(p, free);
146 }
147 
148 using ArrayKeyVector = std::vector<table::Key<table::Array<double>>>;
149 
150 /*
151  * Implementation object as a template parameter for the instantiation of
152  * InterpolatedTransmissionCurve used by makeSpatiallyConstant.
153  */
154 class Impl1d {
155 public:
156  static constexpr bool isSpatiallyConstant = true;
157  static constexpr char const* NAME = "SpatiallyConstantTransmissionCurve";
158 
159  // Initialize the GSL interpolator and take ownership of the given arrays.
160  // Array size consistency checks are done by makeSpatiallyConstant,
161  // so we don't need to worry about them here.
162  Impl1d(ndarray::Array<double, 1, 1> const& throughput, ndarray::Array<double, 1, 1> const& wavelengths)
163  : _throughput(throughput),
164  _wavelengths(wavelengths),
165  _interp(makeGslPtr(::gsl_interp_alloc(::gsl_interp_linear, _wavelengths.size()),
166  &::gsl_interp_free)) {
167  int status = ::gsl_interp_init(_interp.get(), _wavelengths.getData(), _throughput.getData(),
168  _wavelengths.size());
169  LSST_CHECK_GSL(pex::exceptions::LogicError, status);
170  }
171 
172  std::pair<double, double> getWavelengthBounds() const {
173  return std::make_pair(_wavelengths.front(), _wavelengths.back());
174  }
175 
176  static void setupPersistence(table::Schema& schema, ArrayKeyVector& keys) {
177  keys.push_back(schema.addField<table::Array<double>>("throughput", "array of known throughput values",
178  "", 0));
179  keys.push_back(schema.addField<table::Array<double>>(
180  "wavelengths", "array of known wavelength values", "angstrom", 0));
181  }
182 
183  void persist(table::BaseRecord& record, ArrayKeyVector const& keys) const {
184  record.set(keys[0], _throughput);
185  record.set(keys[1], _wavelengths);
186  }
187 
188  static Impl1d unpersist(table::BaseRecord& record, ArrayKeyVector const& keys) {
189  return Impl1d(record[keys[0]], record[keys[1]]);
190  }
191 
192  // A helper object constructed every time InterpolatedTransmissionCurve::sampleAt
193  // is called, and then invoked at every iteration of the loop therein.
194  struct Functor {
195  Functor(Impl1d const&, lsst::geom::Point2D const&)
196  : _accel(makeGslPtr(::gsl_interp_accel_alloc(), &gsl_interp_accel_free)) {}
197 
198  double operator()(Impl1d const& parent, double wavelength) {
199  double result = 0.0;
200  int status = ::gsl_interp_eval_e(parent._interp.get(), parent._wavelengths.getData(),
201  parent._throughput.getData(), wavelength, _accel.get(), &result);
202  LSST_CHECK_GSL(pex::exceptions::RuntimeError, status);
203  return result;
204  }
205 
206  private:
207  GslPtr<::gsl_interp_accel> _accel;
208  };
209 
210 private:
211  ndarray::Array<double, 1, 1> _throughput;
212  ndarray::Array<double, 1, 1> _wavelengths;
213  GslPtr<::gsl_interp> _interp;
214 };
215 
216 /*
217  * Implementation object as a template parameter for the instantiation of
218  * InterpolatedTransmissionCurve used by makeRadial.
219  */
220 class Impl2d {
221 public:
222  static constexpr bool isSpatiallyConstant = false;
223  static constexpr char const* NAME = "RadialTransmissionCurve";
224 
225  Impl2d(ndarray::Array<double, 1, 1> const& throughput, ndarray::Array<double, 1, 1> const& wavelengths,
226  ndarray::Array<double, 1, 1> const& radii)
227  : _throughput(throughput),
228  _wavelengths(wavelengths),
229  _radii(radii),
230  _interp(makeGslPtr(
231  ::gsl_interp2d_alloc(::gsl_interp2d_bilinear, _wavelengths.size(), _radii.size()),
232  &::gsl_interp2d_free)) {
233  int status = ::gsl_interp2d_init(_interp.get(), _wavelengths.getData(), _radii.getData(),
234  _throughput.getData(), _wavelengths.size(), _radii.size());
235  LSST_CHECK_GSL(pex::exceptions::LogicError, status);
236  }
237 
238  std::pair<double, double> getWavelengthBounds() const {
239  return std::make_pair(_wavelengths.front(), _wavelengths.back());
240  }
241 
242  static void setupPersistence(table::Schema& schema, ArrayKeyVector& keys) {
243  keys.push_back(schema.addField<table::Array<double>>(
244  "throughput",
245  "flattenned 2-d array of known throughput values, with radius dimension fastest", "", 0));
246  keys.push_back(schema.addField<table::Array<double>>(
247  "wavelengths", "array of known wavelength values", "angstrom", 0));
248  keys.push_back(schema.addField<table::Array<double>>("radii", "array of known radius values", "", 0));
249  }
250 
251  void persist(table::BaseRecord& record, ArrayKeyVector const& keys) const {
252  record.set(keys[0], _throughput);
253  record.set(keys[1], _wavelengths);
254  record.set(keys[2], _radii);
255  }
256 
257  static Impl2d unpersist(table::BaseRecord& record, ArrayKeyVector const& keys) {
258  return Impl2d(record[keys[0]], record[keys[1]], record[keys[2]]);
259  }
260 
261  // A helper object constructed every time InterpolatedTransmissionCurve::sampleAt
262  // is called, and then invoked at every iteration of the loop therein.
263  struct Functor {
264  Functor(Impl2d const& parent, lsst::geom::Point2D const& point)
265  : _radius(point.asEigen().norm()),
266  _radiusAccel(makeGslPtr(::gsl_interp_accel_alloc(), &gsl_interp_accel_free)),
267  _wavelengthAccel(makeGslPtr(::gsl_interp_accel_alloc(), &gsl_interp_accel_free)) {
268  _radius = std::max(_radius, parent._radii.front());
269  _radius = std::min(_radius, parent._radii.back());
270  }
271 
272  double operator()(Impl2d const& parent, double wavelength) {
273  double result = 0.0;
274  int status =
275  ::gsl_interp2d_eval_e(parent._interp.get(), parent._wavelengths.getData(),
276  parent._radii.getData(), parent._throughput.getData(), wavelength,
277  _radius, _radiusAccel.get(), _wavelengthAccel.get(), &result);
278  LSST_CHECK_GSL(pex::exceptions::RuntimeError, status);
279  return result;
280  }
281 
282  private:
283  double _radius;
284  GslPtr<::gsl_interp_accel> _radiusAccel;
285  GslPtr<::gsl_interp_accel> _wavelengthAccel;
286  };
287 
288 private:
289  ndarray::Array<double, 1, 1> _throughput;
290  ndarray::Array<double, 1, 1> _wavelengths;
291  ndarray::Array<double, 1, 1> _radii;
292  GslPtr<::gsl_interp2d> _interp;
293 };
294 
295 template <typename Impl>
296 class InterpolatedTransmissionCurve : public TransmissionCurve {
297 public:
298  InterpolatedTransmissionCurve(Impl impl, std::pair<double, double> throughputAtBounds)
299  : _atBounds(throughputAtBounds), _impl(std::move(impl)) {
300  if (!std::isfinite(_atBounds.first) || !std::isfinite(_atBounds.second)) {
301  throw LSST_EXCEPT(pex::exceptions::InvalidParameterError,
302  "Throughput values at bounds must be finite");
303  }
304  }
305 
306  std::pair<double, double> getWavelengthBounds() const override { return _impl.getWavelengthBounds(); }
307 
308  std::pair<double, double> getThroughputAtBounds() const override { return _atBounds; }
309 
310  void sampleAt(lsst::geom::Point2D const& point, ndarray::Array<double const, 1, 1> const& wavelengths,
311  ndarray::Array<double, 1, 1> const& out) const override {
312  LSST_THROW_IF_NE(wavelengths.getSize<0>(), out.getSize<0>(), pex::exceptions::LengthError,
313  "Length of wavelength array (%d) does not match size of output array (%d)");
314  typename Impl::Functor functor(_impl, point);
315  auto bounds = _impl.getWavelengthBounds();
316  auto wlIter = wavelengths.begin();
317  for (auto outIter = out.begin(); outIter != out.end(); ++outIter, ++wlIter) {
318  double& y = *outIter;
319  if (*wlIter < bounds.first) {
320  y = _atBounds.first;
321  } else if (*wlIter > bounds.second) {
322  y = _atBounds.second;
323  } else {
324  y = functor(_impl, *wlIter);
325  }
326  }
327  }
328 
329  bool isPersistable() const noexcept override { return true; }
330 
331 protected:
332  std::shared_ptr<TransmissionCurve const> _transformedByImpl(
334  if (_impl.isSpatiallyConstant) {
335  return shared_from_this();
336  } else {
337  return TransmissionCurve::_transformedByImpl(transform);
338  }
339  }
340 
341  std::string getPersistenceName() const override { return Impl::NAME; }
342 
343  struct PersistenceHelper {
344  table::Schema schema;
345  table::Key<double> throughputAtMin;
346  table::Key<double> throughputAtMax;
347  ArrayKeyVector arrays;
348 
349  static PersistenceHelper const& get() {
350  static PersistenceHelper const instance;
351  return instance;
352  }
353 
354  private:
355  PersistenceHelper()
356  : schema(),
358  schema.addField<double>("throughputAtMin", "throughput below minimum wavelength")),
360  schema.addField<double>("throughputAtMax", "throughput above minimum wavelength")) {
361  Impl::setupPersistence(schema, arrays);
362  }
363  };
364 
365  void write(OutputArchiveHandle& handle) const override {
366  auto const& keys = PersistenceHelper::get();
367  auto catalog = handle.makeCatalog(keys.schema);
368  auto record = catalog.addNew();
369  record->set(keys.throughputAtMin, _atBounds.first);
370  record->set(keys.throughputAtMax, _atBounds.second);
371  _impl.persist(*record, keys.arrays);
372  handle.saveCatalog(catalog);
373  }
374 
375  class Factory : public table::io::PersistableFactory {
376  public:
377  std::shared_ptr<table::io::Persistable> read(InputArchive const& archive,
378  CatalogVector const& catalogs) const override {
379  auto const& keys = PersistenceHelper::get();
380  LSST_ARCHIVE_ASSERT(catalogs.size() == 1u);
381  LSST_ARCHIVE_ASSERT(catalogs.front().getSchema() == keys.schema);
382  auto& record = catalogs.front().front();
383  return std::make_shared<InterpolatedTransmissionCurve>(
384  Impl::unpersist(record, keys.arrays),
385  std::make_pair(record.get(keys.throughputAtMin), record.get(keys.throughputAtMax)));
386  }
387 
388  Factory() : table::io::PersistableFactory(Impl::NAME) {}
389  };
390 
391  static Factory registration;
392 
393 private:
394  std::pair<double, double> _atBounds;
395  Impl _impl;
396 };
397 
398 template <typename Impl>
399 typename InterpolatedTransmissionCurve<Impl>::Factory InterpolatedTransmissionCurve<Impl>::registration;
400 
401 template class InterpolatedTransmissionCurve<Impl1d>;
402 template class InterpolatedTransmissionCurve<Impl2d>;
403 
404 /*
405  * ProductTransmissionCurve: default for TransmissionCurve::multipliedBy().
406  *
407  * This is a straightforward lazy-evaluation object. Its only state is the
408  * two operands it delegates to.
409  */
410 class ProductTransmissionCurve : public TransmissionCurve {
411 public:
412  static constexpr char const* NAME = "ProductTransmissionCurve";
413 
414  ProductTransmissionCurve(std::shared_ptr<TransmissionCurve const> a,
416  : _a(std::move(a)), _b(std::move(b)) {}
417 
418  std::pair<double, double> getWavelengthBounds() const override {
419  auto aWavelengthBounds = _a->getWavelengthBounds();
420  auto bWavelengthBounds = _b->getWavelengthBounds();
421  auto aThroughputAtBounds = _a->getThroughputAtBounds();
422  auto bThroughputAtBounds = _b->getThroughputAtBounds();
423 
424  auto determineWavelengthBound = [](double aWavelength, double bWavelength, double aThroughput,
425  double bThroughput, auto isFirstOuter) -> double {
426  // Use the outermost wavelength bound only if its throughput
427  // values are not being multiplied by zeros from the operand with
428  // the innermost wavelength bound.
429  if (isFirstOuter(aWavelength, bWavelength)) {
430  return (bThroughput == 0.0) ? bWavelength : aWavelength;
431  } else {
432  return (aThroughput == 0.0) ? aWavelength : bWavelength;
433  }
434  };
435 
436  return std::make_pair(determineWavelengthBound(aWavelengthBounds.first, bWavelengthBounds.first,
437  aThroughputAtBounds.first, bThroughputAtBounds.first,
439  determineWavelengthBound(aWavelengthBounds.second, bWavelengthBounds.second,
440  aThroughputAtBounds.second, bThroughputAtBounds.second,
442  }
443 
444  std::pair<double, double> getThroughputAtBounds() const override {
445  auto aAtBounds = _a->getThroughputAtBounds();
446  auto bAtBounds = _b->getThroughputAtBounds();
447  return std::make_pair(aAtBounds.first * bAtBounds.first, aAtBounds.second * bAtBounds.second);
448  }
449 
450  void sampleAt(lsst::geom::Point2D const& position, ndarray::Array<double const, 1, 1> const& wavelengths,
451  ndarray::Array<double, 1, 1> const& out) const override {
452  _a->sampleAt(position, wavelengths, out);
453  ndarray::Array<double, 1, 1> tmp = ndarray::allocate(wavelengths.getSize<0>());
454  _b->sampleAt(position, wavelengths, tmp);
455  out.deep() *= tmp;
456  }
457 
458  bool isPersistable() const noexcept override { return _a->isPersistable() && _b->isPersistable(); }
459 
460 protected:
461  std::string getPersistenceName() const override { return NAME; }
462 
463  struct PersistenceHelper {
464  table::Schema schema;
465  table::Key<int> a;
466  table::Key<int> b;
467 
468  static PersistenceHelper const& get() {
469  static PersistenceHelper const instance;
470  return instance;
471  }
472 
473  private:
474  PersistenceHelper()
475  : schema(),
476  a(schema.addField<int>("a", "archive ID of first operand")),
477  b(schema.addField<int>("b", "archive ID of second operand")) {}
478  };
479 
480  class Factory : public table::io::PersistableFactory {
481  public:
482  std::shared_ptr<table::io::Persistable> read(InputArchive const& archive,
483  CatalogVector const& catalogs) const override {
484  auto const& keys = PersistenceHelper::get();
485  LSST_ARCHIVE_ASSERT(catalogs.size() == 1u);
486  LSST_ARCHIVE_ASSERT(catalogs.front().getSchema() == keys.schema);
487  auto const& record = catalogs.front().front();
488  return std::make_shared<ProductTransmissionCurve>(
489  archive.get<TransmissionCurve>(record.get(keys.a)),
490  archive.get<TransmissionCurve>(record.get(keys.b)));
491  }
492 
493  Factory() : table::io::PersistableFactory(NAME) {}
494  };
495 
496  void write(OutputArchiveHandle& handle) const override {
497  auto const& keys = PersistenceHelper::get();
498  auto catalog = handle.makeCatalog(keys.schema);
499  auto record = catalog.addNew();
500  record->set(keys.a, handle.put(_a));
501  record->set(keys.b, handle.put(_b));
502  handle.saveCatalog(catalog);
503  }
504 
505  static Factory registration;
506 
507 private:
510 };
511 
512 ProductTransmissionCurve::Factory ProductTransmissionCurve::registration;
513 
514 /*
515  * TransformedTransmissionCurve: default for TransmissionCurve::transform.
516  *
517  * This is a another straightforward lazy-evaluation object. Its only state
518  * is the two operands it delegates to.
519  */
520 class TransformedTransmissionCurve : public TransmissionCurve {
521 public:
522  static constexpr char const* NAME = "TransformedTransmissionCurve";
523 
524  TransformedTransmissionCurve(std::shared_ptr<TransmissionCurve const> nested,
526  : _nested(std::move(nested)), _transform(std::move(transform)) {}
527 
528  std::pair<double, double> getWavelengthBounds() const override { return _nested->getWavelengthBounds(); }
529 
530  std::pair<double, double> getThroughputAtBounds() const override {
531  return _nested->getThroughputAtBounds();
532  }
533 
534  void sampleAt(lsst::geom::Point2D const& position, ndarray::Array<double const, 1, 1> const& wavelengths,
535  ndarray::Array<double, 1, 1> const& out) const override {
536  return _nested->sampleAt(_transform->applyInverse(position), wavelengths, out);
537  }
538 
539  bool isPersistable() const noexcept override {
540  return _nested->isPersistable() && _transform->isPersistable();
541  }
542 
543 protected:
544  // transforming a TransformedTransmissionCurve composes the transforms
545  std::shared_ptr<TransmissionCurve const> _transformedByImpl(
547  return std::make_shared<TransformedTransmissionCurve>(_nested, transform->then(*_transform));
548  }
549 
550  std::string getPersistenceName() const override { return NAME; }
551 
552  struct PersistenceHelper {
553  table::Schema schema;
554  table::Key<int> nested;
555  table::Key<int> transform;
556 
557  static PersistenceHelper const& get() {
558  static PersistenceHelper const instance;
559  return instance;
560  }
561 
562  private:
563  PersistenceHelper()
564  : schema(),
565  nested(schema.addField<int>("nested", "archive ID of the nested TransmissionCurve")),
566  transform(schema.addField<int>("transform", "archive ID of the coordinate transform")) {}
567  };
568 
569  void write(OutputArchiveHandle& handle) const override {
570  auto const& keys = PersistenceHelper::get();
571  auto catalog = handle.makeCatalog(keys.schema);
572  auto record = catalog.addNew();
573  record->set(keys.nested, handle.put(_nested));
574  record->set(keys.transform, handle.put(_transform));
575  handle.saveCatalog(catalog);
576  }
577 
578  class Factory : public table::io::PersistableFactory {
579  public:
580  std::shared_ptr<table::io::Persistable> read(InputArchive const& archive,
581  CatalogVector const& catalogs) const override {
582  auto const& keys = PersistenceHelper::get();
583  LSST_ARCHIVE_ASSERT(catalogs.size() == 1u);
584  LSST_ARCHIVE_ASSERT(catalogs.front().getSchema() == keys.schema);
585  auto const& record = catalogs.front().front();
586  return std::make_shared<TransformedTransmissionCurve>(
587  archive.get<TransmissionCurve>(record.get(keys.nested)),
588  archive.get<geom::TransformPoint2ToPoint2>(record.get(keys.transform)));
589  }
590 
591  Factory() : table::io::PersistableFactory(NAME) {}
592  };
593 
594  static Factory registration;
595 
596 private:
599 };
600 
601 TransformedTransmissionCurve::Factory TransformedTransmissionCurve::registration;
602 
603 } // namespace
604 
605 /*
606  * TransmissionCurve itself
607  */
608 
609 std::shared_ptr<TransmissionCurve const> TransmissionCurve::makeIdentity() {
610  return IdentityTransmissionCurve::get();
611 }
612 
613 std::shared_ptr<TransmissionCurve const> TransmissionCurve::makeSpatiallyConstant(
614  ndarray::Array<double const, 1> const& throughput, ndarray::Array<double const, 1> const& wavelengths,
615  double throughputAtMin, double throughputAtMax) {
616  ::gsl_set_error_handler_off();
617  LSST_THROW_IF_NE(wavelengths.getSize<0>(), throughput.getSize<0>(), pex::exceptions::LengthError,
618  "Length of wavelength array (%d) does not match size of throughput array (%d)");
619  return std::make_shared<InterpolatedTransmissionCurve<Impl1d>>(
620  Impl1d(ndarray::copy(throughput), ndarray::copy(wavelengths)),
622 }
623 
624 std::shared_ptr<TransmissionCurve const> TransmissionCurve::makeRadial(
625  ndarray::Array<double const, 2> const& throughput, ndarray::Array<double const, 1> const& wavelengths,
626  ndarray::Array<double const, 1> const& radii, double throughputAtMin, double throughputAtMax) {
627  ::gsl_set_error_handler_off();
629  wavelengths.getSize<0>(), throughput.getSize<0>(), pex::exceptions::LengthError,
630  "Length of wavelength array (%d) does not match first dimension of of throughput array (%d)");
632  radii.getSize<0>(), throughput.getSize<1>(), pex::exceptions::LengthError,
633  "Length of radii array (%d) does not match second dimension of of throughput array (%d)");
634  // GSL wants a column major array (Array<T,2,-2>). But ndarray can only flatten row-major arrays
635  // (Array<T,2,2>). So we allocate a row-major array, assign the caller's throughput array to a
636  // transposed view of it, and then flatten the row-major array.
637  ndarray::Array<double, 2, 2> throughputTransposed = ndarray::allocate(throughput.getShape().reverse());
638  throughputTransposed.transpose() = throughput;
639  ndarray::Array<double, 1, 1> throughputFlat = ndarray::flatten<1>(throughputTransposed);
640  return std::make_shared<InterpolatedTransmissionCurve<Impl2d>>(
641  Impl2d(throughputFlat, ndarray::copy(wavelengths), ndarray::copy(radii)),
643 }
644 
645 std::shared_ptr<TransmissionCurve const> TransmissionCurve::multipliedBy(
646  TransmissionCurve const& other) const {
647  auto a = shared_from_this();
648  auto b = other.shared_from_this();
649  auto result = a->_multipliedByImpl(b);
650  if (result == nullptr) {
651  result = b->_multipliedByImpl(a);
652  }
653  if (result == nullptr) {
654  result = std::make_shared<ProductTransmissionCurve>(std::move(a), std::move(b));
655  }
656  return result;
657 }
658 
659 std::shared_ptr<TransmissionCurve const> TransmissionCurve::transformedBy(
661  return _transformedByImpl(std::move(transform));
662 }
663 
664 ndarray::Array<double, 1, 1> TransmissionCurve::sampleAt(
665  lsst::geom::Point2D const& position, ndarray::Array<double const, 1, 1> const& wavelengths) const {
666  ndarray::Array<double, 1, 1> out = ndarray::allocate(wavelengths.getSize<0>());
667  sampleAt(position, wavelengths, out);
668  return out;
669 }
670 
671 std::shared_ptr<TransmissionCurve const> TransmissionCurve::_transformedByImpl(
673  return std::make_shared<TransformedTransmissionCurve>(shared_from_this(), std::move(transform));
674 }
675 
676 std::shared_ptr<TransmissionCurve const> TransmissionCurve::_multipliedByImpl(
678  return nullptr;
679 }
680 
681 std::string TransmissionCurve::getPythonModule() const { return "lsst.afw.image"; }
682 
683 } // namespace image
684 } // namespace afw
685 } // namespace lsst
py::object result
Definition: _schema.cc:429
#define LSST_EXCEPT(type,...)
Create an exception with a given type.
Definition: Exception.h:48
int y
Definition: SpanSet.cc:48
#define LSST_CHECK_GSL(type, status)
table::Key< int > nested
table::Schema schema
ArrayKeyVector arrays
table::Key< double > throughputAtMax
table::Key< double > throughputAtMin
table::Key< int > b
table::Key< int > a
#define LSST_ARCHIVE_ASSERT(EXPR)
An assertion macro used to validate the structure of an InputArchive.
Definition: Persistable.h:48
#define LSST_THROW_IF_NE(N1, N2, EXC_CLASS, MSG)
Check whether the given values are equal, and throw an LSST Exception if they are not.
Definition: asserts.h:38
A spatially-varying transmission curve as a function of wavelength.
EigenVector const & asEigen() const noexcept(IS_ELEMENT_NOTHROW_COPYABLE)
Return a fixed-size Eigen representation of the coordinate object.
Reports attempts to exceed implementation-defined length limits for some classes.
Definition: Runtime.h:76
T free(T... args)
T infinity(T... args)
T isfinite(T... args)
T make_pair(T... args)
T max(T... args)
T min(T... args)
T move(T... args)
Transform< Point2Endpoint, Point2Endpoint > TransformPoint2ToPoint2
Definition: Transform.h:300
Backwards-compatibility support for depersisting the old Calib (FluxMag0/FluxMag0Err) objects.
std::string getPythonModule() const override
std::string getPersistenceName() const override
void write(OutputArchiveHandle &handle) const override
A base class for image defects.
STL namespace.
T transform(T... args)
std::shared_ptr< table::io::Persistable > read(table::io::InputArchive const &archive, table::io::CatalogVector const &catalogs) const override
Definition: warpExposure.cc:0